statikbe / laravel-puppeteer-pdf-converter
This is a convenience wrapper for the Statik Puppeteer PDF conversion API on AWS Lambda.
Installs: 1 545
Dependents: 0
Suggesters: 0
Security: 0
Stars: 7
Watchers: 3
Forks: 0
Open Issues: 0
Requires
- php: ^8.1|^8.2|^8.3
- guzzlehttp/guzzle: ^7.0
- illuminate/contracts: ^9.0|^10.0|^11.0|^12.0
- spatie/laravel-package-tools: ^1.9.2
Requires (Dev)
- larastan/larastan: ^3.0.0
- laravel/pint: ^1.21
- nunomaduro/collision: ^8.0
- orchestra/testbench: ^10.0
- pestphp/pest: ^3.7
- pestphp/pest-plugin-laravel: ^3.1
- phpstan/extension-installer: ^1.4
- phpstan/phpstan-deprecation-rules: ^2.0
- phpstan/phpstan-phpunit: ^2.0
- phpunit/phpunit: ^11.0
README
Laravel Puppeteer PDF Converter & Merger
This is a convenience wrapper for a service that converts an HTML page to PDF. The service uses Puppeteer.js and is developed in-house at Statik. The code for the service is not yet open-sourced, but can be shared upon request.
Installation
You can install the package via composer:
composer require statikbe/laravel-puppeteer-pdf-converter
You can publish the config file with:
php artisan vendor:publish --tag="puppeteer-pdf-converter-config"
This is the contents of the published config file: puppeteer-pdf-converter.php
Configuration
Required: You need to set the url to the API in the .env
with key PDF_CONVERSION_API
.
Optional: The other configuration is optional. You can set default Puppeteer configuration, such as paper margins, scaling and paper width and height. The paper width and height defaults to A4. More information on possible Puppeteer configuration is available in the Puppeteer docs.
You can also manually set the PDF configuration when you create the PDF, see below. However, it is important to note that the
default configuration set in the config file, is always taken as the basis and is overridden by the manual options in code.
If a value is set to null
in the config file or manual options, the option will not be send to the service
(i.e. the option is disabled).
Merge API
Required: You need to set the url to the API in the .env
with key PDF_MERGER_API
and an API key in PDF_MERGER_API_KEY
.
Usage
Conversion API:
The library allows to convert an HTML page to a PDF for a route name (use convertRoute()
) or for a URL (use convertUrl()
).
If you want to override the PDF options that are set in the configuration file, you can create a PdfOptions
object.
//set options and get pdf from conversion API try { $pdfOptions = new PdfOptions(); $pdfOptions->setScale(0.6) ->setPageMargins(new PdfPageMargin(40, PdfPageMargin::MARGIN_IN_PIXELS)); $pdfUrl = PuppeteerPdfConverter::convertRoute('report_index', ['organisation' => $organisation], 'report.pdf', $pdfOptions); return redirect($pdfUrl); } catch(PdfApiException $exception){ Log::error(sprintf('PDF report could not be created: %s (for organisation: %s)', $exception->getMessage(), $organisation)); return view('pdf_error', ['error' => $exception->getMessage()]); }
The following exceptions can occur:
TimeoutException
: Since the PDF conversion service runs on AWS Lambda, it can take timeout due to cold-start issues. If 20 seconds pass, this exception will be thrown. You can use this exception to implement a retry mechanism.UnsuccessfulHttpResponseException
: When the given URL returns an HTTP status that is not successful (within the range of 200), this exception is thrown to avoid that error pages are rendered in the PDF.ConversionException
: If an internal error occurs on the service or in the Puppeteer library, this exception is returned.PdfApiException
: This exception is the super class of all above exceptions, and can thus be used to catch all of the above exceptions in one catch clause (as shown in the example).InvalidMarginUnitException
: This exception is thrown if the unit of the margins is something else than px or cm.
Temporary URLs
The library can generate temporary signed URLs. This is handy if you need to generate PDFs from routes that require authorisation. The PDF service cannot login to your application. But by using temporarily signed URLs, we can validate that the application has generated the URL and no one has tempered with it.
You need to check in the controller of the HTML page whether the signature of the URL is valid. See section Local development for an example.
Merger API
With the PdfMerger
class you can merge multiple PDFs into one while maintaining the PDF layers (including links and annotations).
This class provides two public functions to merge either URLs pointing to PDFs, or binary base64 strings representing PDFs, into one single PDF.
The mergePdfUrls
method requires an array of links to existing PDFs that you would like to merge, and optionally
the output format of the final document along with its filename. You can output a url to the PDF file, stored in an S3 bucket,
or receive the binary stream of the file.
Here is a simple example of how to use mergePdfUrls method:
$merger = new \Statikbe\PuppeteerPdfConverter\PdfMerger(); $urls = ['http://example.com/first.pdf', 'http://example.com/second.pdf']; $mergedPdfUrl = $merger->mergePdfUrls($urls, \Statikbe\PuppeteerPdfConverter\Enum\MergerOutput::URL, 'merged_file.pdf');
In this case, $mergedPdfUrl
will hold the URL to the resulting PDF called 'merged_file.pdf'.
mergePdfFiles
method works similarly, but it takes an array of local file paths to the PDF documents you want to merge:
$merger = new \Statikbe\PuppeteerPdfConverter\PdfMerger(); $files = ['/path/to/first.pdf', '/path/to/second.pdf']; $response = $merger->mergePdfFiles($files, \Statikbe\PuppeteerPdfConverter\Enum\MergerOutput::BASE64);
In this case, $response
will contain a binary stream of the merged PDFs ready to be saved or sent over the network.
This functionality can be used with an optional MergerOutput
parameter to determine the format of the result.
By default, this parameter is set to output a URL. Alternatively, you can set MergerOutput::BASE64
to retrieve
the merged result as a Base64-encoded string.
Remember to always include appropriate exception handling when using these methods as they may throw
a PdfApiException
if the AWS Lambda function encounters an error.
Local development
The PDF conversion service runs in the cloud. For local development, you can run it locally or use a tunneling service to expose your local web server. Examples of such tunneling software is Ngrok or Expose. Or if you use Valet, there is a sharing command
valet share
If you use Ngrok, you can start the service (replace APP_URL
with the value of the APP_URL
key in your .env
file):
ngrok http -host-header=APP_URL 80
You can then configure the tunneled URL in the .env
file with key NGROK_APP_URL
. You can also set URLs of Valet or Expose
in this variable. For example:
NGROK_APP_URL=http://dba0-94-224-113-240.ngrok.io
In case you use temporary signed URLs, there is a convenience function to check if a local tunnel is configured
(i.e. whether the NGROK_APP_URL
is set and the environment is set to local
).
PuppeteerPdfConverter::isLocalTunnelConfigured();
This can be used to bypass the valid signature check in your controller. Because the NGROK_APP_URL
will be replaced
in the generated URL, the signature will not be valid, hence we need to bypass the $request->hasValidSignature()
check.
Changelog
Please see CHANGELOG for more information on what has changed recently.
Contributing
You can use the Github issue tracker and pull requests are welcome!
Credits
License
The MIT License (MIT). Please see License File for more information.