0

我正在使用带有以下代码的 wkhtmltopdf.exe 生成 PDF。

 string url = HttpContext.Current.Request.Url.AbsoluteUri;

        //string[] strarry = sPath.Split('/');
        //int lengh = strarry.Length;

  var pdfUrl = HtmlToPdf(pdfOutputLocation: "~/PDF/", outputFilenamePrefix: "DT", urls: new string[] { url });

        WebClient req = new WebClient();
        HttpResponse response = HttpContext.Current.Response;
        response.Clear();
        response.ClearContent();
        response.ClearHeaders();
        response.Buffer = true;
        Response.ContentType = "application/pdf";
        response.AddHeader("Content-Disposition", "attachment;filename=\"" + pdfUrl.ToString().Substring(6) + "\"");
        byte[] data = req.DownloadData(Server.MapPath(pdfUrl.ToString()));
        response.BinaryWrite(data);
        File.Delete(Server.MapPath(pdfUrl.ToString()));
        response.End();

  public static string HtmlToPdf(string pdfOutputLocation, string outputFilenamePrefix, string[] urls,
 string[] options = null,
 string pdfHtmlToPdfExePath = "C:\\Program Files\\wkhtmltopdf\\wkhtmltopdf.exe")
    {
        string urlsSeparatedBySpaces = string.Empty;
        try
        {
            //Determine inputs
            if ((urls == null) || (urls.Length == 0))
                throw new Exception("No input URLs provided for HtmlToPdf");
            else
                urlsSeparatedBySpaces = String.Join(" ", urls); //Concatenate URLs

            string outputFolder = pdfOutputLocation;
            string outputFilename = outputFilenamePrefix + "_" + DateTime.Now.ToString("yyyy-MM-dd-hh-mm-ss-fff") + ".PDF"; // assemble destination PDF file name

            var p = new System.Diagnostics.Process()
            {
                StartInfo =
                {
                    FileName = pdfHtmlToPdfExePath,
                    Arguments = ((options == null) ? "" : String.Join(" ", options)) + " " + urlsSeparatedBySpaces + " " + outputFilename,
                    UseShellExecute = false, // needs to be false in order to redirect output
                    RedirectStandardOutput = true,
                    RedirectStandardError = true,
                    RedirectStandardInput = true, // redirect all 3, as it should be all 3 or none
                    WorkingDirectory = HttpContext.Current.Server.MapPath(outputFolder)
                }
            };

            p.Start();

            // read the output here...
            var output = p.StandardOutput.ReadToEnd();
            var errorOutput = p.StandardError.ReadToEnd();

            // ...then wait n milliseconds for exit (as after exit, it can't read the output)
            p.WaitForExit(60000);

            // read the exit code, close process
            int returnCode = p.ExitCode;
            p.Close();

            // if 0 or 2, it worked so return path of pdf
            if ((returnCode == 0) || (returnCode == 2))
                return outputFolder + outputFilename;
            else
                throw new Exception(errorOutput);



            //Response.ContentType = "application/pdf";
            //Response.AddHeader("content-length", theData.Length.ToString());
            //if (Request.QueryString["attachment"] != null)
            //    Response.AddHeader("content-disposition", "attachment; filename=ExampleSite.pdf");
            //else
            //    Response.AddHeader("content-disposition", "inline; filename=ExampleSite.pdf");
            //Response.BinaryWrite(theData);
            //HttpContext.Current.ApplicationInstance.CompleteRequest();
        }
        catch (Exception exc)
        {
            throw new Exception("Problem generating PDF from HTML, URLs: " + urlsSeparatedBySpaces + ", outputFilename: " + outputFilenamePrefix, exc);
        }
    }

从上面的代码中,PDF 生成良好。但是我有两个页面,其登录和注销用户的 URL 相同。例如,可以说 www.xyz/pdf/brason。这个相同的 URL 用于登录和注销用户,但内容将是根据用户登录或注销而有所不同。

现在,当我登录并尝试使用上面的代码生成 PDF 时,它总是向我显示注销用户页面的内容。我不知道如何解决这个问题。

4

2 回答 2

1

我想如果我理解正确,这是因为调用页面的 wkhtmltopdf 没有登录。wkhtmltopdf 有点像创建一个没有任何登录 cookie/会话的新隐身浏览器窗口,所以页面正确地认为它没有登录。你可以通过调试服务器在 wkhtmltopdf 调用它时获得的请求来检查。

如果这是问题,可能很难解决。解决方案取决于您的登录系统以及您可以采取哪些措施来解决该问题。如果您可以使用 cookie 复制登录,您可以自己设置登录 cookie,请参阅http://madalgo.au.dk/~jakobt/wkhtmltoxdoc/wkhtmltopdf_0.10.0_rc2-doc.html#Page%20Options了解更多信息如何设置cookie。

另一种选择是首先从返回登录的 HTML 的系统创建一个请求,然后将其保存到文件/流并将该文件/流提供给 wkhtmltopdf(我猜你可以使用 HttpContext.Current.Request 来做到这一点什么的,我不知道)。

另一种解决方法是创建登录页面的重复页面,该页面看起来与登录页面完全相同,但实际上并非如此 - 此页面仅用于欺骗 wkhtmltopdf。类似的东西www.xyz/pdf/brason?foolwkhtmltopdf=true,然后通过调用类似的东西来使用它if(url.ToLower() == "www.xyz/pdf/brason") {url="www.xyz/pdf/brason?foolwkhtmltopdf=true"; }。这可能是一个安全风险,具体取决于显示的信息。

希望这可以帮助!

于 2013-01-13T12:13:27.567 回答
0

我认为您需要在转换为 html 之前保存页面的输出。因为这会直接调用 url 并且当它调用它时您没有登录 convert to pdf 对其请求的响应我在尝试将 webform 转换为 pdf 但填充了值时遇到了同样的问题所以我将响应保存为 html 和给 wkhtmltopdf 保存的路径作为参数

 Response.ContentType = "application/pdf";
        Response.AddHeader("content-disposition", "attachment;filename=TestPage.pdf");
        Response.Cache.SetCacheability(HttpCacheability.NoCache);
        StringWriter sw = new StringWriter();
        HtmlTextWriter hw = new HtmlTextWriter(sw);
        this.Page.RenderControl(hw);
        StringReader sr = new StringReader(sw.ToString());
        string htmlpath =Server.MapPath("~/htmloutput.html");
        if (File.Exists(htmlpath))
        {
            File.Delete(htmlpath);

        }  

        File.Create(htmlpath).Dispose();
        using (TextWriter tw = new StreamWriter(htmlpath))
        {
            tw.WriteLine(sw.ToString());
            tw.Close();
        }

        string path = Server.MapPath("~/wkhtmltopdf-page.pdf");
        PdfConvert.ConvertHtmlToPdf(new Codaxy.WkHtmlToPdf.PdfDocument
        {
            Url = htmlpath,
            HeaderLeft = "[title]",
            HeaderRight = "[date] [time]",
            FooterCenter = "Page [page] of [topage]"

        }, new PdfOutput
        {

            OutputFilePath = path

        });

您可以在按钮单击事件上调用它。仅在 asp.net 网络表单上对此进行了测试。在 asp.net mvc 上,您需要一些其他方式来获取视图 html 输出

于 2015-12-18T10:46:15.050 回答