prinsfrank / pdfparser
A low-memory, fast and maintainable conforming pdfparser to read PDF files
Fund package maintenance!
PrinsFrank
Requires
- php: ~8.1.0 || ~8.2.0 || ~8.3.0 || ~8.4.0
- ext-zlib: *
Requires (Dev)
- friendsofphp/php-cs-fixer: ^3.65
- phpstan/phpstan: ^2.0
- phpstan/phpstan-strict-rules: ^2
- phpunit/phpunit: ^11.5
- prinsfrank/pdf-samples: ^0.0.4
This package is auto-updated.
Last update: 2025-01-17 21:05:04 UTC
README
PDF Parser
A low-memory, fast and maintainable conforming PDF Parser
📣 Call for testers
Why this library?
Previously, there wasn't a PDF library that allows parsing of PDFs that was open source, MIT licensed and under active development. The PDFParser by smalot, while having been very useful over the years isn't under active development anymore. The parser of Setasign is not MIT licensed and not open source. And several other packages rely on java/js/python dependencies being installed that are called by PHP behind the scenes, losing any type information and underlying structure.
Instead, this package allows for parsing of a wide variety of PDF files while not relying on external dependencies, all while being MIT licensed!
Setup
To start right away, run the following command in your composer project;
composer require prinsfrank/pdfparser
Opening a PDF
To open a PDF file, you'll first need to load it and retrieve a Document
object. That can be done by either parsing a file directly, or parsing a PDF from a string variable.
Parsing a PDF file
Parsing a PDF from a file directly is the easiest option and also uses the least amount of memory. To do so, simply call the parseFile
method on a PdfParser
instance:
use PrinsFrank\PdfParser; $document = (new PdfParser()) ->parseFile(dirname(__DIR__, 3) . '/path/to/file.pdf');
Parsing PDF from string
It is also possible to parse a PDF from a string in a variable. To do so, pass the string as an argument for the parseFile
method on a PdfParser
instance. This has a bigger memory footprint while loading the file into memory, but the file will be written to a temp file while processing.
use PrinsFrank\PdfParser; $pdfAsString = file_get_contents(dirname(__DIR__, 3) . '/path/to/file.pdf'); $document = (new PdfParser()) ->parseString($pdfAsString);
The Document
Once you have opened a file from the filesystem with parseFile
or from a string variable using parseString
, you'll get back an instance of a Document
.
While initially parsing the document, a small number of variables are populated in the Document
instance that allow for further accessing of that document. This includes:
- The public
$stream
property: a PHP stream handle to the file on the filesystem, or - if a string is supplied - a handle to a temporary file. - The public
$version
property: Information about the PDF version of the file. - the public
$crossReferenceSource
property: A parsed crossReference table or stream, containing several crossReference(Sub)Sections that contain information about objects stored in the document and where to find them. - The private
$pages
property to cache any pages that have already been retrieved. This property is only set when the pages are actually retrieved using thegetPages
method. (See below)
The document also contains several methods to retrieve specific objects from it. Those are discussed below.
If you want to quickly retrieve all text from a document, you can use the getText
method.
Objects in a Document
and their decorators
A PDF is organized in objects. Not all objects are created equally. Some objects might be a Page, while others a Font. Some objects might be Generic and without a specific type. There are currently 18 specific types, and a generic object type. Some of those will be specified below.
Code specific for certain object types lives in that object types' decorator. Retrieving text for a Page makes sense, retrieving the text from a Font not so much, so the Page decorator contains the getText
method. Below you'll find some documentation for specific object decorators.
If you want to retrieve an object by its number, you can call the $document->getObject($objectNumber)
method. If you know that the object with that number is supposed to be of a specific type, you can supply the second argument. For example, if you want to get object 42 which you know is of type Page, you can call the method like this:
$page = $document->getObject(42, Page::class);
If the object is not of the correct type, this will result in an exception. If you don't care about the object type, pass null as the second argument or don't supply the second argument at all.
Decorated InformationDictionary
objects
If a PDF has a title, producer, author, creator, creationDate or modificationDate, it is stored in an InformationDictionary.
If a PDF has an InformationDictionary, it can be retrieved using the $document->getInformationDictionary()
method. Not All PDFs have this available, so this method might return null.
To access information from the InformationDictionary, there are several methods available:
/** @var \PrinsFrank\PdfParser\Document\Document $document */ $title = $document->getInformationDictionary()?->getTitle(); $producer = $document->getInformationDictionary()?->getProducer(); $author = $document->getInformationDictionary()?->getAuthor(); $creator = $document->getInformationDictionary()?->getCreator(); $creationDate = $document->getInformationDictionary()?->getCreationDate(); $modificationDate = $document->getInformationDictionary()?->getModificationDate();
If you want to access non-standard data from the information dictionary, you can also retrieve the entire dictionary from the object:
/** @var \PrinsFrank\PdfParser\Document\Document $document */ $dictionary = $document->getInformationDictionary()?->getDictionary();
Decorated Page
objects
Page objects can be retrieved from a document by calling the $document->getPage($pageNumber)
method for a single page, or $document->getPages()
for all pages. Note that $pageNumber
is zero-indexed, so even if different format page numbers are displayed at the bottom of a page, the first page in a document is still page 0, etc.
Once you have a Page
object, there are several methods available to retrieve information from that page. The main method of interest here is the $page->getText()
method. To retrieve all text from all pages, you could do something like this:
use PrinsFrank\PdfParser\PdfParser; $document = (new PdfParser())->parseFile('/path/to/file.pdf'); foreach ($document->getPages() as $index => $page) { echo 'Text on page ' . $index . ' : ' . $page->getText(); }
There is also a getText
method available on the Document to retrieve all text at once without even having to retrieve pages.
There are also methods available to get the underlying textObjectCollection using $page->getTextObjectCollection()
, the resource dictionary for a page using $page->getResourceDictionary()
and the font dictionary using $page->getFontDictionary()
.