生成带有页眉/页脚的页面的pdf时出现wkhtmltopdf错误

时间:2014-08-18 02:46:23

标签: ruby-on-rails ruby qt wkhtmltopdf

我正在使用pdfkit(在引擎盖下使用wkhtmltopdf)在我的rails应用程序中生成PDF。在指南here之后我得到了它主要用于PDF的基本情况。在尝试生成包含大量页面的PDF时,我现在遇到了一个问题,这些页面也有页眉/页脚。尝试生成PDF时,我在控制台中从wkhtmltopdf看到的错误是:

QEventDispatcherUNIXPrivate(): Unable to create thread pipe: Too many open files
QEventDispatcherUNIXPrivate(): Can not continue without a thread pipe

可用于重新创建错误的html的最小示例:

<!-- content of pdf_header_url is the character "h" -->
<meta content="<%= pdf_header_url %>" name="pdfkit-header-html"/>
<!-- content of pdf_footer_url is the character "f" -->
<meta content="<%= pdf_footer_url %>" name="pdfkit-footer_html"/>
<% [*1..3].each do |j|%>
  <h1><%= j %></h1>
  <ul>
    <% [*1..1000].each do |i|%>
      <li><%= i %></li>
    <% end %>
  </ul>
<% end %>

请注意,删除页眉/页脚标记可以使pdf呈现正常。

生成PDF的实际ruby代码是:

def view_report
  html = render_to_string(:template => 'pdf/pdf_body.html', :layout => false)
  kit = PDFKit.new(html)
  pdf = kit.to_pdf
  send_data pdf, :type => 'application/pdf', :disposition => 'inline', :filename => 'foo.pdf'
end

访问此控制器路线将生成PDF。最后,我还有一个页眉/页脚的控制器,因为那些&#34; partials&#34;需要通过网址获取:

class PdfController < ApplicationController

  def header
    render :layout => false
  end

  def footer
    render :layout => false
  end

end

pdf_header_url和pdf_footer_url的值实际上只是&#34; h&#34;和&#34; f&#34;为了一个最小的可重复的例子。

是否有任何熟悉wkhtmltopdf的人都有关于为解决这个问题而采取更深入的调试步骤的建议?

3 个答案:

答案 0 :(得分:8)

我今天收到了同样的错误消息,我用一个非常简单的解决方案解决了这个问题。问题是我的页眉和页脚需要是带有html,head和body标签的完整html文档。另外,看看你是否可以获得生成的页眉和页脚的html输出并验证它们。

答案 1 :(得分:4)

打开文件限制。

我确保我的页眉和页脚文件是Tom Hirschfeld建议的完整HTML文档,但我仍然收到太多打开文件的错误。

在搜索互联网之后,我发现您需要提高允许单个进程打开的文件数量限制。就我而言,我正在生成包含数百到数千页的PDF。它没有页眉和页脚工作得很好但是当合并页眉和页脚时,它似乎达到了这个打开的文件上限。

根据您运行的系统有不同的方法来调整此设置,但这是在Ubuntu服务器上对我有用的方法:

将以下内容添加到 /etc/security/limits.conf

的末尾
# Sets the open file maximum here.
# Generating large PDFs hits the default ceiling (1024) quickly. 
*    hard nofile 65535
*    soft nofile 65535
root hard nofile 65535 # Need these two lines because the wildcards
root soft nofile 65535 # (the * above) are not applied to the root user.

可以找到ulimit命令的良好参考here

我希望这会让一些人走上正轨。

答案 2 :(得分:3)

wkhtmltopdf每页使用2个文件描述符(页眉和页脚各一个),这些是生成每页自定义变量所必需的。您必须编辑/etc/security/limits.conf以将nofile(即最多没有打开的文件)设置为适当的高数字 - 可能需要进行一些实验才能找到适合您的值。