首页 文章

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

提问于
浏览
6

我正在使用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 . 最后,我还有一个页眉/页脚控制器,因为那些“部分”需要通过url获取:

class PdfController < ApplicationController

  def header
    render :layout => false
  end

  def footer
    render :layout => false
  end

end

pdf_header_url和pdf_footer_url的值实际上只是“h”和“f”,以获得最小的可再现示例 .

有没有熟悉wkhtmltopdf的人有任何关于更深入调试步骤的建议来解决这个问题?

3 回答

  • 3

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

  • 7

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

  • 3

    打开文件限制 .

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

    在搜索互联网之后,我发现您需要提高允许打开的单个进程的文件数限制 . 就我而言,我正在生成包含数百到数千页的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 .

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

相关问题