I'm trying to generate a pdf from webpage, invoking wkhtmltopdf through java ProcessBuilder on Centos OS. The problem is that when I run a simple class with a main method the process terminates with:
Command has terminated with status: 139
Output:
Error: Loading pages (1/6) ....
and creates an empty pdf file (size 0B)
I've included a method that prints the arguments with which I invoke the wkhtmltopdf in the class and when I copy the command and run it in bash it works and creates the pdf. Even more: when I run the exact same class in windows it works just fine. What can cause this error code 139? Could it be a bug in wkhtmltopdf or I am doing something wrong?
Here's some more information:
OS:
[root@host sandbox]# uname -a
Linux xxxxxx.com 2.6.32-279.22.1.el6.x86_64 #1 SMP Wed Feb 6 03:10:46 UTC 2013 x86_64 x86_64 x86_64 GNU/Linux
[root@host bin]# ./wkhtmltopdf --version
Name:
wkhtmltopdf 0.12.0 03c001de254b857f08eba80b62d4b6490ffed41d
Command I am trying to run with process builder:
/root/wk/wkhtmltox/bin/wkhtmltopdf --window-status export-ready
--encoding UTF-8
--custom-header username username
--custom-header password pass
--run-script "<some correctly escaped js>"
http://xx.xx.xx.xx/url?param1=1¶m2=2
/root/sandbox/test.pdf
Code for pdf generation:
public String exportToPdf(final String bookmarkableUrl) {
String uuid = UUID.randomUUID().toString();
final String fullUrl = "http://" + hostName + ":" + port + bookmarkableUrl;
// .html extension at the end is very important - wkhtmltopdf won't read
// the file if not there
String generatedPdfPath = tempDirPath + "/EMF/" + uuid;
try {
ProcessBuilder processBuilder = new ProcessBuilder();
processBuilder.command(prepareCommandArguments(fullUrl, generatedPdfPath + PDF_FILE_EXTENSION));
Process start = processBuilder.start();
// One has to handle the error stream
handleStream(start.getErrorStream());
handleStream(start.getInputStream());
// Wait until process is executed.
start.waitFor();
} catch (IOException | InterruptedException e) {
throw new RuntimeException("Error while generating PDF", e);
}
return generatedPdfPath;
}
EDIT: Adding the code that I use for creating the list with command arguments:
private List<String> prepareCommandArguments(String inputUrl, String outputUrl) {
List<String> arguments = new ArrayList<>(15);
// absolute path to the wkhtmltopdf executable
arguments.add(wkhtmltopdfLocation);
// Wait until window.status is equal to this string before rendering page
arguments.add("--window-status");
arguments.add("export-ready");
// Set the default text encoding, for input
arguments.add("--encoding");
arguments.add("UTF-8");
// Set an additional HTTP header for system username
arguments.add("--custom-header");
arguments.add("username");
arguments.add(exportUsername);
// Set an additional HTTP header for system user password
arguments.add("--custom-header");
arguments.add("password");
arguments.add(exportPassword);
// Run this additional javascript after the page is done loading
// Used to remove irrelevant divisions and spanning of
// the html page, leaving only the print preview of the document
arguments.add("--run-script");
arguments.add(getScriptFromFile(jsFilePath));
// Bookmarkable url of the document
arguments.add(inputUrl);
// Path to the generated pdf
arguments.add(outputUrl);
return arguments;
}
Output of System.out.println(processBuilder.command()) just before Process.start() :
/root/wk/wkhtmltox/bin/wkhtmltopdf, --window-status, export-ready, --encoding, UTF-8, --custom-header, username, admin, --custom-header, password, admin, --run-script, "\$('.idoc-comments-column').remove(); \$('.idoc-left-column').remove(); \$('.idoc-left-column').remove(); \$('#topHeader').remove(); \$('#header').remove(); \$('.tree-header.breadcrumb_header').remove(); \$('.idoc-middle-column.pull-left.idoc-first-row').remove(); \$('.idoc-middle-column.pull-left').remove(); \$('.pull-left.text-center').remove(); \$('html').addClass('print-override-overflow'); \$('.idoc-editor').css('width', '80%'); \$('.idoc-editor').css('font-size', '14px'); \$('.idoc-editor').css('max-width', 'none'); \$('.idoc-editor').css('margin-left', '10%'); \$('.idoc-editor').css('margin-right', '10%'); \$('.idoc-editor').css('margin-top', '5%'); \$('.idoc-editor').css('margin-bottom', '5%');", http://xx.xx.xx.xx:xxxx/url/page.jsf?param1=1¶m2=2, /root/sandbox/file.pdf
You don't need to do any shell-escaping with ProcessBuilder
. Try passing the JS code as is without the dollar sign escaped and double quotes.