用于ASP.NET MVC3的Azure和Rotativa PDF打印

时间:2012-06-07 21:55:45

标签: asp.net-mvc-3 azure pdf-generation

我正在使用Rotativa PDF print作为ASP.NET MVC应用程序从html内容生成pdf。当在标准IIS上自行运行MVC应用程序时,这非常有用; pdf几乎立即生成。

但是当MVC应用程序在Azure中部署为Web角色时(在开发环境和cloudapp.net上本地),生成pdf打印最多需要45秒,并且似乎也无法显示内容资源(链接)坏了)。有些事情似乎错了,不应该花那么长时间。

pdf生成本身是使用wkhtmltopdf工具完成的,该工具将html内容转换为PDF。 wkhtmltopdf是一个可执行文件,使用Process.Start执行,它再次由MVC应用程序调用。

    /// <summary>
    /// Converts given URL or HTML string to PDF.
    /// </summary>
    /// <param name="wkhtmltopdfPath">Path to wkthmltopdf.</param>
    /// <param name="switches">Switches that will be passed to wkhtmltopdf binary.</param>
    /// <param name="html">String containing HTML code that should be converted to PDF.</param>
    /// <returns>PDF as byte array.</returns>
    private static byte[] Convert(string wkhtmltopdfPath, string switches, string html)
    {
        // switches:
        //     "-q"  - silent output, only errors - no progress messages
        //     " -"  - switch output to stdout
        //     "- -" - switch input to stdin and output to stdout
        switches = "-q " + switches + " -";

        // generate PDF from given HTML string, not from URL
        if (!string.IsNullOrEmpty(html))
            switches += " -";

        var proc = new Process
                       {
                           StartInfo = new ProcessStartInfo
                                           {
                                               FileName = Path.Combine(wkhtmltopdfPath, "wkhtmltopdf.exe"),
                                               Arguments = switches,
                                               UseShellExecute = false,
                                               RedirectStandardOutput = true,
                                               RedirectStandardError = true,
                                               RedirectStandardInput = true,
                                               WorkingDirectory = wkhtmltopdfPath,
                                               CreateNoWindow = true
                                           }
                       };
        proc.Start();

        // generate PDF from given HTML string, not from URL
        if (!string.IsNullOrEmpty(html))
        {
            using (var sIn = proc.StandardInput)
            {
                sIn.WriteLine(html);
            }
        }

        var ms = new MemoryStream();
        using (var sOut = proc.StandardOutput.BaseStream)
        {
            byte[] buffer = new byte[4096];
            int read;

            while ((read = sOut.Read(buffer, 0, buffer.Length)) > 0)
            {
                ms.Write(buffer, 0, read);
            }
        }

        string error = proc.StandardError.ReadToEnd();

        if (ms.Length == 0)
        {
            throw new Exception(error);
        }

        proc.WaitForExit();

        return ms.ToArray();
    }

是否有人对可能导致问题和降低性能的因素有任何想法?

中号

1 个答案:

答案 0 :(得分:2)

我找到了根本原因。生成的pdf中的URL损坏,导致性能下降。由于负载平衡,Azure在URL中包含端口号,因此我需要将URL转换为没有portnumber的“公共”URL。