Parse and stream tabular data form PDF documents using Node.js with Mozilla's PDF.js library.
This document explains how to use pdf-data-parser in your code or as a stand-alone program.
Only supports PDF files containing grid/table like content. Does not support reading PDF forms (XFA).
Related projects: html-data-parser, pdf-data-parser, xlsx-data-parser
For use as command line utility. Requires Node.js 18+.
npm -g install pdf-data-parser
For use as module in a Node.js project. See Developers Guide below.
npm install pdf-data-parser
Parse tabular data from a PDF file or URL.
pdp [--options=filename.json] [--cells=#] [--heading=title], [--repeating] [--headers=name1,name2,...] [--format=json|csv|rows] <filename|URL> [<output-file>]
`filename|URL` - path name or URL of PDF file to process, required.
`output-file` - local path name for output of parsed data, default stdout.
`--options` - JSON or JSONC file containing pdp options, optional.
`--format` - output data format JSON, CSV or rows (JSON arrays), default JSON.
`--cells` - number of cells for a data row, minimum or "min-max", default = "1-256".
`--heading` - text of heading to find in document that precedes desired data table, default none.
`--headers` - comma separated list of column names for data, default none, first table row contains names.
`--repeating` - table headers repeat on each PDF page, default = false.
Note: If the pdp
command conflicts with another program on your system use pdfdataparser
instead.
The options file supports options for all pdf-data-parser modules. Parser will read plain JSON files or JSONC files with Javascript style comments.
{
/* PdfDataParser options */
// url - local path name or URL of PDF file to process, required.
"url": "",
// output - local path name for output of parsed data, default stdout.
"output": "",
// format - output data format CSV, JSON or rows, default JSON, rows is JSON array of arrays (rows).
"format": "json",
// pages - string or array of page numbers to process, if undefined defaults to all pages. Examples: [ 1,3,4,5,7 ], [ 1, "3-5", 7 ], "1,3-5,7".
"pages": null,
// heading - text of heading to find in document that precedes desired data table, default none.
"heading": null,
// stopHeading - text of heading that follows desired data table, default none.
"stopHeading": null,
// cells - number of cells for a data row, minimum or "min-max", default = "1-256".
"cells": "1-256",
// repeating - table header row repeats on each PDF page, default = false.
"repeatingHeaders": false,
// pageHeader - height of page header area in points, default: 0. Content in this area will be excluded. When true any row matching the first row encountered will be excluded from output.
"pageHeader": 0,
// pageFooter - height of page footer area in points, default: 0. Content in this area will be excluded.
"pageFooter": 0,
// artifacts - parse PDF Artifacts content, default: false. Sometimes used in PDF documents for page header/footer, footnotes, annotations, etc.
"artifacts": false,
// lineHeight - approximate line height ratio based on font size; default: 1.67.
"lineHeight": 1.67,
// newlines - preserve new lines in cell data, default: false.
"newlines": false,
// orderXY - order cells by XY coordinates on page; default true. When false cells will be order as found in the PDF.js page content array.
"orderXY": true,
// trim whitespace from output values, false (0) = none, true (1) = both, 2 = starting only, 3 = trailing only, default: true.
"trim": true,
/* RowAsObjectTransform options */
// hasHeaders - data has a header row, if true and headers set then headers overrides header row.
"RowAsObject.hasHeader": true
// headers - comma separated list of column names for data, default none. When not defined the first table row encountered will be treated as column names.
"RowAsObject.headers": []
/* RepeatCellTransform options */
// column - column index of cell to repeat, default 0.
"RepeatCell.column": 0
/* RepeatHeadingTransform options */
// hasHeaders - data has a header row, if true and headers set then headers overrides header row.
"RepeatHeading.hasHeader": true
// header - column name for the repeating heading field. Can optionally contain suffix :m:n with index for inserting into header and data rows.
"RepeatHeading.header": "subheading:0:0"
}
Note: Transform property names can be shortened to hasHeader
, headers
, column
and header
.
pdp ./test/data/pdf/helloworld.pdf --headers=Greeting --format=csv
pdp https://sos.iowa.gov/elections/pdf/VRStatsArchive/2024/CoJan24.pdf --cells=8 --repeating
pdp --options=.\\test\\optionsRepeatCell.json
optionsRepeatCell.json:
{
"url": "./test/data/pdf/state_voter_registration_jan2024.pdf",
"output": "./test/output/pdp/repeat_cell.json",
"format": "json",
"pages": [ 1 ],
"pageHeader": 64,
"cells": 7,
"RepeatCell.column": 0
}
PdfDataParser given a PDF document will output an array of arrays (rows). For most projects use the streaming classes PdfDataReader and RowAsObjectTransform transform to convert the arrays to Javascript objects. With default settings PdfDataParser will output all rows in the document including headings and paragraphs. Using PdfDataParser Options the parser can filter content to retrieve the desired data table in the document.
PdfDataParser only works on a certain subset of PDF documents specifically those that contain some type of tabular data in a grid/table format. The parser uses marked content items and x,y position information returned by the Mozilla pdf.js API to transform PDF content items into rows of cells.
Rows and Cells terminology is used instead of Rows and Columns because the marked content in a PDF document flows more like an HTML page than database query results. Some rows may have more cells than other rows. For example a heading or description paragraph will be a row (array) with one cell (string). See Notes below.
Warning: PDF document does not contain Marked Content
If the PDF document does not contain marked content the parser will display a console warning. In this case PdfDataParser may not be able to reliably parse data in the document based solely on x,y positioning.
const { PdfDataParser } = require("pdf-data-parser");
let parser = new PdfDataParser({url: "filename.pdf"});
async function myFunc() {
var rows = await parser.parse();
// process the rows
}
PdfDataParser constructor takes an options object with the following fields. One of url
or data
arguments is required.
{String|URL} url
- The local path or URL of the PDF document.
{String|ArrayBuffer} data
- pdf file data in a TypedArray, e.g. options.data = new Uint8Array(buffer)
.
Common Options:
{Array} pages
- array of page numbers to process, if undefined defaults to all pages. Examples: [ 1 ], [ 3, 5, 7 ]
{String|RegExp} heading
- Section heading or text in the document after which the parser will look for tabular data; optional, default: none. The parser does a string comparison or regexp match looking for first occurrence of heading
value in the first cell of rows. If not specified then data output starts with first row of the document that contains enough cells.
{String|RegExp} stopHeading
- Section heading or text in the document after the tabular data; optional, default: none. The parser does a string comparison or regexp match looking for occurrence of stopHeading
value in the first cell of rows. If not specified then data output stops on value of cells
or the end of document.
{Number} cells
- Minimum number of cells in tabular data; optional, default: 1. If heading
is not specified then all rows in document with at least cells
length will be output. If heading
string is found parser will look for the first row that contains at least cells
count of cells after the heading. The parser will output rows until it encounters a row with less than cells
count of cells.
{Boolean} repeatingHeaders
- Indicates if table headers are repeated on each page, default: false. The table headers will be compare to the first row on each subsequent page. If found they will be removed from the output.
{Number} pageHeader
- Height of page header area in points, default: 0. Content within this area of the page will not be included in output. Use about 16 points per line including blank lines.
{Number} pageFooter
- Height of page footer area in points, default: 0. Content within this area of the page will not be included in output. Use about 16 points per line including blank lines.
Other Options:
{Boolean} artifacts
- Parse artifacts content, default: false. Artifacts content specifies objects on the page such as table/grid lines and table headers/footers. Grid lines do not have text content, but table headers and footers might. If page headers and footers show up in output try the pageHeader and pageFooter options.
{Number} lineHeight
- Approximate line height ratio based on font size; default 1.67. The parser extracts font size from the pdf content. The line height ratio maybe used when comparing the position of content items on the page.
{Boolean} newlines
- Preserve new lines in cell data; optional, default: false. When false newlines will be replaced by spaces. Preserving newlines characters will keep the formatting of multiline text such as descriptions. Though, newlines are problematic for cells containing multi-word identifiers and keywords that might be wrapped in the PDF text.
{Boolean} orderXY
- order cells by XY coordinates on page; default true. When false cells will be order as found in the PDF.js page content array. Some documents may have items ordered top-to-bottom, left-to-right in the PDF document. Most PDF documents will have items that are out of order and need to be placed on the page by X,Y coordinates.
{Boolean|Number} trim
- trim whitespace from output values, false (0) = no trimming, true (1) = both, 2 = starting only, 3 = trailing only, default: true.
PdfDataReader is a Node.js stream reader implemented with the Object mode option. It uses PdfDataParser to stream one data row (array) per chunk.
const { PdfDataReader } = require("pdf-data-parser");
let reader = new PdfDataReader({url: "filename.pdf"});
var rows = [];
reader.on('data', (row) => {
rows.push(row)
});
reader.on('end', () => {
// do something with the rows
});
reader.on('error', (err) => {
// log error
})
PdfDataReader constructor options are the same as PdfDataParser Options.
PdfDataReader operates in Object Mode. The reader outputs arrays (rows). To convert rows into Javascript objects use the RowAsObjectTransform transform. PdfDataReader operates in Object mode where a chunk is a Javascript Object of <name,value> pairs.
const { PdfDataReader, RowAsObjectTransform } = require("pdf-data-parser");
const { pipeline } = require('node:stream/promises');
let reader = new PdfDataReader(options);
let transform1 = new RowAsObjectTransform(options);
let writable = <some writable that can handle Object Mode data>
await pipeline(reader, transform1, writable);
RowAsObjectTransform constructor takes an options object with the following fields.
{array} headers
- array of cell property names; optional, default: none. If a headers array is not specified then parser will assume the first row found contains cell property names.
{Boolean} hasHeaders
- data has a header row, if true and headers options is set then provided headers override header row. Default is true.
If a row is encountered with more cells than in the headers array then extra cell property names will be the ordinal position. For example if the data contains five cells, but only three headers where specified. Specifying options = { headers: [ 'name', 'type', 'info' ] }
then the Javascript objects in the stream will contain { "name": "value1", "type": "value2", "info": "value3", "4": "value4", "5": "value5" }
.
The RepeatCellTransform will normalize data the was probably generated by a report writer. The specified cell will be repeated in following rows that contain one less cell. In the following example "Dewitt" will be repeated in rows 2 and 3.
PDF Document
County Precincts Date/Period Total
Dewitt 44 JUL 2023 52,297
44 OCT 2023 52,017
44 JAN 2024 51,712
Output
[ "County", "Precincts", "Date/Period", "Total" ]
[ "Dewitt", "44", "JUL 2023", "52,297" ]
[ "Dewitt", "44", "OCT 2023", "52,017" ]
[ "Dewitt", "44", "JAN 2024", "51,712" ]
const { PdfDataReader, RepeatCellTransform } = require("pdf-data-parser");
const { pipeline } = require('node:stream/promises');
let reader = new PdfDataReader(options);
let transform1 = new RepeatCellTransform({ column: 0 });
let writable = <some writable that can handle Object Mode data>
await pipeline(reader, transform1, writable);
RepeatCellTransform constructor takes an options object with the following fields.
{Number} column
- column index of cell to repeat, default 0.
The RepeatHeadingTransform will normalize data the was probably generated by a report writer. Subheadings are rows containing a single cell interspersed in data rows. The header name is inserted in to the header row. The subheading value will be repeated in rows that follow until another subheading is encountered. In the following example options = {header: "County:1:0"}
.
PDF Document
District Precincts Total
Congressional District 5
Maricopa 120 403,741
Pinal 30 102,512
Total: 150 506,253
Output
[ "District", "County", "Precincts", "Total" ]
[ "Congressional District 5", "Maricopa", "120", "403,741" ]
[ "Congressional District 5", "Pinal", "30", "102,512" ]
[ "Congressional District 5", "Total:", "150", "506,253" ]
const { PdfDataReader, RepeatHeadingTransform } = require("pdf-data-parser");
const { pipeline } = require('node:stream/promises');
let reader = new PdfDataReader(options);
let transform1 = new RepeatHeadingTransform({header: "County:1:0"});
let writable = <some writable that can handle Object Mode data>
await pipeline(reader, transform1, writable);
RepeatHeadingTransform constructor takes an options object with the following fields.
{String} header
- column name for the repeating heading field. Can optionally contain an index of where to insert the header in the header row. Default "heading:0".
{Boolean} hasHeaders
- data has a header row, if true and headers options is set then provided headers override header row. Default is true.
The pdfdataparser
CLI program uses the FormatCSV and FormatJSON transforms to covert Javascript Objects into strings that can be saved to a file.
const { PdfDataReader, RowAsObjectTransform, FormatCSV } = require("pdf-data-parser");
const { pipeline } = require('node:stream/promises');
let reader = new PdfDataReader(options);
let transform1 = new RowAsObjectTransform(options);
let transform2 = new FormatCSV();
await pipeline(reader, transform1, transform2, process.stdout);
In the source code the pdf-data-parser.js program and the Javascript files in the /test folder are good examples of using the library modules.
HelloWorld.pdf is a single page PDF document with the string "Hello, world!" positioned on the page. The PdfDataParser output is one row with one cell.
[
["Hello, world!"]
]
To transform the row array into an object specify the headers option to RowAsObjectTransform transform.
let transform = new RowAsObjectTransform({
headers: [ "Greeting" ]
})
Output as JSON objects:
[
{ "Greeting": "Hello, world!" }
]
ClassCodes.pdf contains one simple table spanning multiple pages. It is a straight forward parsing of all rows in the document.
let parser = new PdfDataParser({ url: "https://www2.census.gov/geo/pdfs/reference/ClassCodes.pdf" })
Parser output:
[
["Class Code","Class Code Description","Associated Geographic Entity"],
["A1","Airport with scheduled commercial flights that also serves as a military installation","Locality Point, Military Installation"],
...
["Z9","County subdivision not defined","County Subdivision"]
]
Nat_State_Topic_File_formats.pdf contains USGS file formats for various downloadable reference files. It is a rather complicated example containing multiple tables of data interspersed with headings, descriptive paragraphs, vertical column spans, cells split across pages and embedded hyperlinks. See Notes below.
For this example the parser will look for tabular data following the heading "Government Units File Format" found on pages 6 and 7 in Nat_State_Topic_File_formats.pdf.
let parser = new PdfDataParser({
url: "https://geonames.usgs.gov/docs/pubs/Nat_State_Topic_File_formats.pdf",
heading: "Government Units File Format",
cells: 3
})
Note that cells: 3
was specified even though the output has four cells. The fourth cell of this table sometimes contains a vertical spanning cell. Specifying cells: 4
would cause the parser to short circuit on the row after the vertical span, because it would only contain three cells.
Parser output:
[
["Name","Type","Length/Decimals","Description"],
["Feature ID","Number","10","ID number for the governmental unit."],
["Unit Type","Character","50","Type of government unit."],
...,
["Country Name","Character","100"],
["Feature Name","Character","120","Official feature name"]
]
CoJul22.pdf contains one simple table spanning multiple pages. This document contains page headers and footers with repeating table headers on each page. Use the repeatingHeaders option to remove the extra table headers from output data.
let parser = new PdfDataParser({ url: "./test/data/pdf/CoJul22.pdf", repeatingHeaders: true })
The page headers/footers in this document are in PDF.js Artifacts marked content. They will be ignored by default. To output the page headers and footers use the artifacts option.
let parser = new PdfDataParser({ url: "./test/data/pdf/CoJul22.pdf", artifacts: true })
If your document has page headers/footers contained in regular content items then the headers/footers can be ignored by using the pageHeader and pageFooter options. The settings of 50 and 35 ignore 3 and 2 lines respectively.
let parser = new PdfDataParser({ url: "./test/data/pdf/CoJul22.pdf", pageHeader: 50, pageFooter: 35 })
CongJul22.pdf contains four tables. This document contains page headers and footers.
An oddity of this document is there is an additional table header row that identifies each table. This content item, e.g. "US Representative District 1", is actually in the document content after the table. The PdfDataParser has specialized logic to insert the cell data in the appropriate flow order before output of data rows.
PdfDataParser does not support the splitting of output so the file would need to be read four times with separate heading
options or read all together and then hand edit the output. Alternatively, a custom Node.js stream transform or writer derived class could be used to split the data into multiple outputs.
parser1 = new PdfDataParser({ url: "./test/data/pdf/CongJul22.pdf", heading: "US Representative District 1", cells: 12 })
house1 = await parser.parse();
parser2 = new PdfDataParser({ url: "./test/data/pdf/CongJul22.pdf", heading: "US Representative District 2", cells: 12 })
house2 = await parser.parse();
parser3 = new PdfDataParser({ url: "./test/data/pdf/CongJul22.pdf", heading: "US Representative District 3", cells: 12 })
house3 = await parser.parse();
parser3 = new PdfDataParser({ url: "./test/data/pdf/CongJul22.pdf", heading: "US Representative District 4", cells: 12 })
house3 = await parser.parse();
The /test/data/pdf fold contains a PDF that was created by a report generator, state_voter_registration_jan2024.pdf. This PDF content does not contain Marked content and cell positioning is totally by X,Y coordinates. Also, it contains subheadings in tables and some missing cell values. The test files optionsRepeatCells.json | testRepeatCells.js and optionsRepeatHeading | testRepeatHeading.js contain examples of parsing this type of PDF document.
- Only supports PDF files containing grid/table like layouts. Does not support reading PDF forms (XFA).
- Tables that span multiple pages are supported as long as all cell text for an individual row is on the same page.
- Cells crossing page boundaries is not supported, currently. The cell will be split into multiple rows. The second row may not contain the proper number of cells, i.e. missing values are not supported, currently.
- Embedded hyperlinks are not supported. The link information is not provided by pdf.js API.
- Does not support identification of titles, headings, column headers, etc. by using style information for a cell. This style information is not provided by pdf.js API.
- Vertical spanning cells are parsed with first row where the cell is encountered. Subsequent rows will not contain the cell and have one less cell. Currently, vertical spanning cells must be at the end of the row otherwise the ordinal position of cells in the following rows will be incorrect, i.e. missing values are not supported.