Mar
4
Written by:
Michael Washington
3/4/2017 2:09 PM
data:image/s3,"s3://crabby-images/fb464/fb464cdf77214b3c1e29da2904107131e792c009" alt="image image"
Notice: A few times, In testing I have encountered intermittent errors. This Stack Overflow post details the issues I found.
You can use Azure Functions to convert PDF files to PNG files.
data:image/s3,"s3://crabby-images/42781/427815f68ebfe133bd3550428e2f1ef10f47ed5c" alt="image image"
You can put a .pdf file in Microsoft Azure Storage (this can be done programmatically or using the Microsoft Azure Storage Explorer)…
data:image/s3,"s3://crabby-images/94509/9450975055650246f2ebb70f5f10e438bb06d1db" alt="image image"
… then enter a message in Azure Queue Storage with the name of the file to process.
data:image/s3,"s3://crabby-images/afe9f/afe9f1f43cc1ea3786d87a28bd8635f4ef1cfbad" alt="image image"
The PDF file will be split into individual .png files.
Create The Azure Function
data:image/s3,"s3://crabby-images/512ec/512ecb0841a0c72da65da9346aba605e616352b5" alt="image image"
Go to:
https://azure.microsoft.com/en-us/services/functions/
Log into your Azure account (or create a new one).
data:image/s3,"s3://crabby-images/1938d/1938df139e1e132b3aee7e00eba35060bb457c53" alt="image image"
Select a Subscription, enter a Name for your function, select a Region close to you, and click the Create + get started button.
data:image/s3,"s3://crabby-images/e0e59/e0e596592e261f5a4db1c1ca59366de6eea3af43" alt="image image"
Click the New Function button.
data:image/s3,"s3://crabby-images/7c092/7c092cb4a83edd43ea12655094d64236353b02e0" alt="image image"
Select the C# Language, the Data Processing Scenario, then the QueueTrigger-CSharp template.
data:image/s3,"s3://crabby-images/4a796/4a79682b34410c8f66992d191c565f7985700c33" alt="image image"
Enter a Name for the function.
Set the queue name (leave the default).
Select an existing Storage account connection or create a new one.
Click the Create button.
data:image/s3,"s3://crabby-images/8eae5/8eae5e9a035d9730620a14bbb96fd1b07b1fca11" alt="image image"
The function edit screen will show.
Click the expand buttons on the Logs and File View sections to show them.
data:image/s3,"s3://crabby-images/06c42/06c427f1445a8e6ac4d014fb7740d36e47ac5a4d" alt="image image"
Click the Add button to add a new file.
data:image/s3,"s3://crabby-images/5badc/5badc44df27474f51c28ae5ba4d8ba8ab8f19175" alt="image image"
Name the file project.json and press the Enter key.
data:image/s3,"s3://crabby-images/f32ae/f32aeb58fa83827d03592468baad30d7f5c91924" alt="image image"
Enter the following code and press the Save button:
{
"frameworks": {
"net46": {
"dependencies": {
"Ghostscript.NET": "1.2.1"
}
}
}
}
This will instruct Azure Functions to load the GhostScript.Net Nuget package.
GhostScript is still required and we will add that in a later step.
data:image/s3,"s3://crabby-images/e8cfb/e8cfb36c6ba5a753fb59c6ad4ebedecd13658fac" alt="image image"
Click on the run.csx file, enter the following code, and click the Save button:
#r "Microsoft.WindowsAzure.Storage"
#r "System.Drawing"
#r "System.Web"
#r "System.Configuration"
using System.Drawing;
using System.Drawing.Imaging;
using System;
using System.Configuration;
using Microsoft.WindowsAzure.Storage;
using Microsoft.WindowsAzure.Storage.Auth;
using Microsoft.WindowsAzure.Storage.Blob;
using System.Net;
using Ghostscript.NET;
using Ghostscript.NET.Rasterizer;
public static void Run(string myQueueItem, TraceWriter log)
{
log.Info($"C# Queue trigger function processed. Value passed: {myQueueItem}");
string uploadFileName = myQueueItem;
log.Info($"Get AzureWebJobsStorage and cloudUploadedFilesContainerName Settings");
string _storageConnectionString = ConfigurationManager.AppSettings["AzureWebJobsStorage"];
string _cloudUploadedFilesContainerName = "pdf-conversion-input";
log.Info($"Get CloudStorageAccount");
var _cloudStorageAccount = CloudStorageAccount.Parse(_storageConnectionString);
log.Info($"Get CloudBlobClient");
var _cloudBlobClient = _cloudStorageAccount.CreateCloudBlobClient();
log.Info($"Get ContainerReference");
var _cloudUploadedFilesBlobContainer =
_cloudBlobClient.GetContainerReference(_cloudUploadedFilesContainerName);
log.Info($"Get sourceBlockBlob");
CloudBlockBlob sourceBlockBlob =
_cloudUploadedFilesBlobContainer.GetBlockBlobReference(uploadFileName);
if (!sourceBlockBlob.Exists())
{
log.Info($"sourceBlockBlob does not exist.");
return;
}
// _pdfConversionFilePath is the path the temp folder is in
string _pdfConversionFilePath = @"D:\home\data\Functions\sampledata";
log.Info($"_pdfConversionFilePath: {_pdfConversionFilePath}");
string filePath = string.Format(@"{0}\{1}", _pdfConversionFilePath, uploadFileName);
log.Info($"filePath: {filePath}");
sourceBlockBlob.DownloadToFile(filePath, FileMode.Create);
// The png file will have the same name as the PDF file but with a png extension
string fileName = Path.GetFileNameWithoutExtension(filePath);
log.Info($"fileName: {fileName}");
string pngFilePath = string.Format(@"{0}\{1}.png", _pdfConversionFilePath, fileName);
log.Info($"pngFilePath: {pngFilePath}");
// Delete png file if already exists
log.Info($"File.Exists(pngFilePath): {File.Exists(pngFilePath)}");
if (File.Exists(pngFilePath))
{
File.Delete(pngFilePath);
}
// Use Ghostscript to convert from pdf to png
log.Info($"Get cloudConvertedFilesContainerName Settings");
string _cloudConvertedFilesContainerName = "pdf-conversion-output";
log.Info($"GetContainerReference");
var _cloudConvertedFilesContainer =
_cloudBlobClient.GetContainerReference(_cloudConvertedFilesContainerName);
int desired_x_dpi = 96;
int desired_y_dpi = 96;
string inputPdfPath = filePath;
string outputPath = Path.GetDirectoryName(pngFilePath);
GhostscriptVersionInfo gvi =
new GhostscriptVersionInfo(@"D:\home\data\Functions\packages\nuget\ghostscript.net\1.2.1\lib\net40\gsdll32.dll");
log.Info($"using (GhostscriptRasterizer _rasterizer = new GhostscriptRasterizer())");
using (GhostscriptRasterizer _rasterizer = new GhostscriptRasterizer())
{
log.Info($"_rasterizer.Open: {inputPdfPath}");
_rasterizer.Open(inputPdfPath, gvi, true);
for (int pageNumber = 1; pageNumber <= _rasterizer.PageCount; pageNumber++)
{
string pageFilePath = Path.Combine(outputPath, fileName + "-Page-" + pageNumber.ToString() + ".png");
log.Info($"pageFilePath: {pageFilePath}");
log.Info($"_rasterizer.GetPage: {pageNumber}");
Image img = _rasterizer.GetPage(desired_x_dpi, desired_y_dpi, pageNumber);
log.Info($"img.Save: {pageFilePath}");
img.Save(pageFilePath, ImageFormat.Png);
string pngfileName = Path.GetFileName(pageFilePath);
log.Info($"pngfileName: {pngfileName}");
log.Info($"GetBlockBlobReference");
CloudBlockBlob targetBlockBlob = _cloudConvertedFilesContainer.GetBlockBlobReference(pngfileName);
log.Info($"UploadFromStream: {pageFilePath}");
using (FileStream fileStream = File.OpenRead(pageFilePath))
{
targetBlockBlob.UploadFromStream(fileStream);
}
log.Info($"File.Delete png file: {pageFilePath}");
File.Delete(pageFilePath);
}
}
log.Info($"File.Delete PDF file: {filePath}");
File.Delete(filePath);
log.Info($"Completed processing {uploadFileName}");
}
Adding GhostScript
data:image/s3,"s3://crabby-images/7506e/7506ea33094c921227512afb6131b6ea4a363437" alt="image image"
Download the GhostScript 32 bit assembly from this link.
After you run the installer, the assembly that you need will be at the following location:
- C:\Program Files (x86)\gs\gs9.20\bin\gsdll32.dll
data:image/s3,"s3://crabby-images/c6461/c6461dcbcd8f7bf4f6690167aa291d9171c8ed36" alt="image image"
Return to the Azure portal, select the Function app (AzureFunction-App), and click Function app settings.
data:image/s3,"s3://crabby-images/54367/54367888e6e9ec8373588a811f2c949e687c1ed1" alt="image image"
Click the Go to Kudu button.
data:image/s3,"s3://crabby-images/ea94b/ea94b65bbf080c3ed018f660dbbb20c175ba2042" alt="image image"
In Kudu, select Debug console then CMD.
data:image/s3,"s3://crabby-images/c0884/c08849c38f97ae7af3757a264b3564a8e18c4c27" alt="image image"
Click on the data folder to navigate to it.
data:image/s3,"s3://crabby-images/6106f/6106f71c6bd83920570db9e0b91f0815fcaecdf3" alt="image image"
Then click on Functions.
data:image/s3,"s3://crabby-images/c9386/c938689642f76560e6dcaebf17840b6c3c3cb5cf" alt="image image"
Next, click on packages…
data:image/s3,"s3://crabby-images/a0005/a0005163ce982a519df7842238c462a03049e5cf" alt="image image"
nuget…
data:image/s3,"s3://crabby-images/3e993/3e993b463ca47812d31b668b6b55ff956364680f" alt="image image"
ghostscript.net…
data:image/s3,"s3://crabby-images/ac303/ac303bcfc03f0cea8640636f86ba3d9cca8665bd" alt="image image"
1.2.1…
data:image/s3,"s3://crabby-images/72ba1/72ba11dbdd152a423c68b003c059fbf346da7af4" alt="image image"
lib…
data:image/s3,"s3://crabby-images/b3cce/b3cce3f159c33b27eb939a62d0a5134a15d6a5f1" alt="image image"
Finally, net40.
data:image/s3,"s3://crabby-images/50e7f/50e7f338bf41a34d8b78d3491956966da33f8709" alt=""
Now drag and drop the gsdll32.dll file onto the web page in your web browser.
The file will upload to the directory.
Connect To Azure Storage
data:image/s3,"s3://crabby-images/74d0f/74d0fb6f348239869ac0060296435e169c9c03e7" alt="image image"
Download and install the Microsoft Azure Storage Explorer.
data:image/s3,"s3://crabby-images/97490/97490bb0a7932d6f3beaadd3c3d62de109c38453" alt="image image"
After you install it, open the Microsoft Azure Storage Explorer.
data:image/s3,"s3://crabby-images/7f7b3/7f7b39d30c2fe6a831562181a15832ed485218ca" alt="image image"
Log in using your Azure account.
data:image/s3,"s3://crabby-images/3daf9/3daf9e0be512a8c529f055165c93bc0658aca21f" alt="image image"
Navigate to the Azure storage account you specified when you created the function (if you forgot which one it is, you can see the AzureWebJobsStorage connection string in Configure app settings under Function app settings).
Expand the tree to show all the nodes.
Right-click on the Blob Containers node and select Create Blob Container.
data:image/s3,"s3://crabby-images/5004c/5004ca9a89a7bb702708140d78f00718c0021816" alt="image image"
Create:
- pdf-conversion-input
- pdf-conversion-output
data:image/s3,"s3://crabby-images/87560/87560cbfc10b2145f2cb4aef39b99cfa81bed838" alt="image image"
Select the pdf-conversion-input folder and use the Upload button to upload a PDF file.
data:image/s3,"s3://crabby-images/bcdd0/bcdd0c0b88abf353e4e2a0b2a995c9c4b9acb91d" alt="image image"
Right-click on the Queues node and select Create Queue.
data:image/s3,"s3://crabby-images/8f666/8f666f8d367c8dbc5d08d07ecad6926f32062c68" alt="image image"
Create a queue called myqueue-items.
data:image/s3,"s3://crabby-images/a8854/a885442c935f30d5fca3b13dc00ed7b8c3d8c578" alt="image image"
Select the myqueue-items, click the Add button, and add a message to the queue with the name of the PDF file that was uploaded to the pdf-conversion-input folder.
Note, you must use the exact casing for the file name.
data:image/s3,"s3://crabby-images/15219/152191eb13ec037cd9554af5cce12c083ba73e6b" alt="image image"
The PDF will be read, and a PNG file will be created for each page of the PDF file.
The files will show up in the pdf-conversion-output folder.
data:image/s3,"s3://crabby-images/7450c/7450c20b0190fd3a85aa7744e41bf18894e7425f" alt="image image"
You can monitor the function and diagnose any errors using the Logs in the Azure portal.
data:image/s3,"s3://crabby-images/36196/361968ec4c53cbf2935e3b5581064a85b7e75a45" alt=""
You can watch the files being processed in Kudu by navigating to:
D:\home\data\Functions\sampledata
GhostScript Errors
data:image/s3,"s3://crabby-images/00cfd/00cfd858a2a70bbd91cb7fe7bcf7439b77c4061e" alt="image image"
If you get the following error:
Exception while executing function: Functions.PDFtoPNG. mscorlib: Exception has been thrown by the target of an invocation. Ghostscript.NET: Ghostscript native library could not be found.
You need to make sure you have added gsdll32.dll to the correct directory, also that you have added the 32 bit version not the 64 bit version of the file.
If you get other errors, see this Stack Overflow post.
Links
Converting PDFs to Multipage Tiff files Using Azure WebJobs
Azure Functions
Microsoft Azure Storage Explorer
4 comment(s) so far...
Very nice work, thank you. Quick question, i'm getting this error in the log: "sourceBlockBlob does not exist", but after reading through the code i can't see why as i used the same names. ... but perhaps i misspelled something.
Any thoughts as to what I should look at? No problem if not, thanks again
By Raj Finn on
6/30/2017 7:04 AM
|
@Raj Finn - Just a guess, perhaps your storage container name and sub folder and file name do not match? Upper and lower case does matter ("UsConstitution" is not the same as "USCONSTITUTION"). Also ensure that this is working: string _storageConnectionString = ConfigurationManager.AppSettings["AzureWebJobsStorage"]; By setting it in the steps outlined in the article
By Michael Washington on
6/30/2017 7:09 AM
|
you can use this "http://www.cnetsdk.com/net-pdf-to-image-converter-sdk" pdf to image convert sdk and this .net pdf to image sdk.
By Paul Jones on
5/15/2018 7:51 PM
|
Try online website https://miniimagesvideos.com/jpgtopdf to compress jpeg to pdf. Any other digital media can be also compressed using this site. Like compress jpeg or compress images using https://miniimagesvideos.com or compress pdf using https://miniimagesvideos.com/pdf or compress video using https://miniimagesvideos.com/compress_video
By Neelam Jain on
9/10/2019 4:06 AM
|