- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
项目详情:
我有一个全栈项目, View 使用 ReactJS,后端使用 NodeJS,数据库使用 MySQL。在我的网络服务器上,我使用 NGINX 作为网络服务器,并使用 PM2 作为反向代理来运行我的 Node 实例。
目标:
我想做的基本上是让用户从输入中选择 StartDate 和 EndDate。这将从 MYSQL 数据库中提取数据,然后在浏览器中将 PDF 下载给用户,用户将在单独的 PDF 中显示它。因此,例如,如果记录包含 30 个结果,它将是一个包含 30 个单独页面的 PDF。
问题:
我让这一切在我的本地主机上正常工作,它在浏览器中成功地将 PDF 下载给用户,但是当我将它推送到暂存或生产服务器时,它根本不会在浏览器中下载 PDF,我我不确定为什么。
在 staging 和 live 在 console.log 中抛出相同的错误:
这里是网络请求的屏幕截图。在响应中根本没有响应,我得到它只是空白但在本地主机上我确实得到了响应:
但是,在本地主机上,当用户单击“生成 PDF”按钮时,PDF 成功生成并在浏览器中下载没有问题。这是网络请求和它返回的响应的屏幕截图:
我有一大堆代码,所以我会尽量只展示相关的问题:
View (ReactJS):
import React, { useEffect, useState } from "react";
import TextField from "@material-ui/core/TextField";
import Button from "@material-ui/core/Button";
import Moment from "moment";
import AppController from "../../controllers/appController";
import Detailedservice from "../../services/Detailedservice";
const DetailedReport = () => {
const [dateValues, handleDateChange] = useState({
startDate: "",
endDate: "",
});
const [emptyText, setEmpty] = useState(false);
const [sucessText, setSuccess] = useState(false);
const [loadingState, isLoading] = useState(false);
const [scheduleData, setSchedule] = useState({
startDate: "",
endDate: "",
bathData: [],
});
const handleChange = (e) => {
handleDateChange({ ...dateValues, [e.target.name]: e.target.value });
};
const data = {
selectedDates: {
startDate: dateValues.startDate,
endDate: dateValues.endDate,
},
requestedData: scheduleData,
};
// This is what generates the PDF
const viewPdf = await Detailedservice.pdfTemplate(data);
// These lines download the PDF in the browser for the user
var link = document.createElement("a");
// Location of where the download file is located in
link.href = AppController.downloadPDF() + "MM_Report.pdf";
link.target = "_blank";
// Name of download file
link.download = "MM_Report.pdf";
link.dispatchEvent(new MouseEvent("click"));
};
return (
<div className="detailedreportContainer">
<div className="container">
<div className="row d-flex justify-content-center">
<div className="col-lg-2">
<div className="daterange" style={{ marginTop: "150px" }}>
<div className="daterangeCenter">
<TextField
id="startDate"
label="Start Date"
name="startDate"
type="date"
InputLabelProps={{
shrink: true,
}}
onChange={handleChange}
/>
</div>
</div>
</div>
<div className="col-lg-2">
<div className="daterange" style={{ marginTop: "150px" }}>
<div className="daterangeCenter d-flex justify-content-center">
<TextField
id="endDate"
label="End Date"
name="endDate"
type="date"
InputLabelProps={{
shrink: true,
}}
onChange={handleChange}
/>
</div>
</div>
</div>
</div>
<div
className="row d-flex justify-content-center"
style={{ marginTop: "20px" }}
>
<Button
variant="contained"
color="primary"
disabled={
!dateValues.startDate || !dateValues.endDate || emptyText
? true
: false
}
onClick={fetchSchedule}
>
{loadingState ? (
<i
className={"fas fa-circle-notch fa-spin"}
style={{ fontSize: "24px" }}
/>
) : (
"Generate Report"
)}
</Button>
</div>
</div>
</div>
</div>
);
};
export default DetailedReport;
模型:(这就是创建 PDF 的原因)
var db = require("../dbconnection");
var htmltopdf = require("html-pdf");
const path = require("path");
var options = {
width: "8.5in",
height: "11in",
format: "Letter",
};
var detailedReport = {
getSchedule: function (data, callback) {
console.log(data);
db.query(
"SELECT Subject, StartTime, EndTime, FullName, Resident, CommunicationMethod, Email, Phone, Reason from schedule where DateCreated between ? AND ? order by StartTime, EndTime asc",
[data.startDate, data.endDate],
callback
);
},
pdfMessage: function (data) {
let scheduleData = data.requestedData;
let dateonDocument = data.selectedDates;
var message = "";
// Looping over all schedules chosen and displaying them in multiple batch pdfs. This is possible using the following class "<div style='page-break-before: always'></div>" which should be used to display data on different pages
message += "<html>";
message += "<head>";
message += "<meta charset='utf-8' />";
message += "<style>";
message += "* {";
message += "border: 0;";
message += "margin: 0;";
message += "padding: 0;";
message += "}";
message += "body {";
message += "font-family: cursive, Helvetica, sans-serif;";
message += "width: 100%;";
message += "margin: 30px;";
message += "color: #000;";
message += "}";
message += "#uniqueTable,";
message += "#uniqueTable td,";
message += "#uniqueTable th {";
message += "border: 1px solid #000;";
message += "}";
message += "table {";
message += "border-collapse: collapse;";
message += "width: 90%;";
message += "margin: 10px 0;";
message += "text-align: left;";
message += "}";
message += "td {";
message += "font-size: 20px;";
message += "padding: 10px;";
message += "}";
message += "p {";
message += "padding: 3px 0;";
message += "font-size: 14px;";
message += "}";
message += "img {";
message += "padding-left: 60px;";
message += "padding-bottom: 10px;";
message += "display: block;";
message += "}";
message += "h1 {";
message += "font-size: 18px;";
message += "padding: 10px 0;";
message += "color: #35a768;";
message += "}";
message += "h2 {";
message += "font-size: 18px;";
message += "font-weight: bold;";
message += "color: #000;";
message += "}";
message += "hr {";
message += "border: 1px solid #000;";
message += "width: 90%;";
message += "}";
message += "table th {";
message += "font-size: 16px;";
message += "color: #0094e9;";
message += "font-weight: bold;";
message += "}";
message += "</style>";
message += "</head>";
message += "<body>";
message += "<div style='text-align:center'>";
message += "<h1>Report</h1>";
message +=
"<p>" +
dateonDocument.startDate +
" - " +
dateonDocument.endDate +
"</p>";
message += "</div>";
message += "<div style='page-break-before: always'></div>";
scheduleData.forEach((data) => {
message += "<div id='pageContent'>";
message += "<table>";
message += "<tr>";
message += "<td>";
message += "<img";
message +=
"src='https://amazonaws.com/mmlogomain.jpg'";
message += "style='width: 200px; margin: auto'";
message += "/>";
message += "</td>";
message += "</tr>";
message += "</table>";
message +=
"<div id='header' style='background-color: #e0f2ff; padding: 20px'>";
message += "<h1 style='color: #000; font-size: 24px'>";
message += "Detailed Report";
message += "</h1>";
message += "</div>";
message +=
"<div id='subheader' style='background-color: #0094e9; height: 30px'></div>";
message += "<table id='uniqueTable'>";
message += "<tbody>";
message += "<tr>";
message += "<td>";
message += "<h1>Date/Time Scheduled:</h1>";
message += "<p>" + data.StartTime + " - " + data.EndTime + "</p>";
message += "</td>";
message += "</tr>";
message += "<tr>";
message += "<td>";
message += "<h1>Communication Method:</h1>";
message += "<p>" + data.CommunicationMethod + "</p>";
message += "</td>";
message += "</tr>";
message += "</tbody>";
message += "</table>";
message += "<table id='uniqueTable'>";
message += "<tbody>";
message += "<tr>";
message += "<td>";
message += "<h1>Visitor's Full Name:</h1>";
message += "<p>" + data.FullName + "</p>";
message += "</td>";
message += "<td>";
message += "<h1>Resident Requesting to Speak to:</h1>";
message += "<p>" + data.Resident + "</p>";
message += "</td>";
message += "</tr>";
message += "<tr>";
message += "<td>";
message += "<h1>Phone Number:</h1>";
message += "<p>" + data.Phone + "</p>";
message += "</td>";
message += "<td rowspan='2'>";
message += "<h1>Reason for Visit:</h1>";
message += "<p>" + data.Reason + "</p>";
message += "</td>";
message += "</tr>";
message += "<tr>";
message += "<td>";
message += "<h1>Email Address:</h1>";
message += "<p>" + data.Email + "</p>";
message += "</td>";
message += "</tr>";
message += "</tbody>";
message += "</table>";
message += "</div>";
message += "<div style='page-break-before: always'></div>";
});
message += "</body>";
message += "</html>";
return message;
},
pdfTemplate: function (data, callback) {
let pdfPath = "MM_Report.pdf";
htmltopdf
.create(detailedReport.pdfMessage(data), options)
.toFile(
path.join(__dirname, "../public/reports/" + pdfPath),
function (err, res) {
callback(res);
}
);
},
};
module.exports = detailedReport;
路线
var express = require("express");
var router = express.Router();
var detailedReport = require("../models/detailedReport");
router.post("/getSchedule", function (req, res) {
detailedReport.getSchedule(req.body, function (err, rows) {
if (err) {
res.json(err);
console.log(err);
} else {
res.send(rows);
}
});
});
router.post("/pdfTemplate", function (req, res) {
detailedReport.pdfTemplate(req.body, function (err, rows) {
if (err) {
res.json(err);
} else {
res.json(rows);
}
});
});
module.exports = router;
服务:
import serviceBase from "./serviceBase";
const scheduleService = {
getSchedule: (data) => serviceBase.post("/api/getSchedule", data),
pdfTemplate: (data) => serviceBase.post("/api/pdfTemplate", data),
};
export default scheduleService;
最佳答案
据我正确理解/pdfTemplate?t=xyz 返回 pdf 的路径。
您需要确保 pdf 在服务器上 (../web/uploads/file.pdf) 而不是在您的本地目录 (C://docs/file.pdf)
另一种解决方案
例子:
//content - encoded base64 string
generateFile(content) {
const byteCharacters = atob(content);
const byteArrays = [];
for (let offset = 0; offset < byteCharacters.length; offset += 512) {
const slice = byteCharacters.slice(offset, offset + 512);
const byteNumbers = new Array(slice.length);
for (let i = 0; i < slice.length; i++) {
byteNumbers[i] = slice.charCodeAt(i);
}
const byteArray = new Uint8Array(byteNumbers);
byteArrays.push(byteArray);
}
const blob = new Blob(byteArrays, {type: "application/pdf"});
const link = document.createElement("a");
link.href = window.URL.createObjectURL(blob);
link.target = '_black';
link.click();
}
关于node.js - HTML-PDF npm 在本地主机上工作但不在 IP 登台服务器上工作?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/66703870/
我用运行 Node node --debug app OR node --debug-brk app 它有反应 debugger listening on port 5858 Express serv
这个问题在这里已经有了答案: What is the difference between (int *i) and (int* i) in context of both C and C++? [
我有一个应用程序,它通过消息队列将数据库写入命令分派(dispatch)给工作人员(数量非常大),因此无法保证它们的接收顺序。 我有两个 Node ,例如“Account”和“Media”。在此假设的
有没有办法在调用 ts-node 时将选项传递给 Node ?我正在尝试在 Node 中使用一个实验性功能,如果它能与 ts-node 一起使用,那就太好了。 这就是我目前正在做的事情: ts-nod
我有一个容器化的Node应用程序,它在DigitalOcean服务器上运行。当我更新服务器上的应用程序时,该应用程序必须关闭一小段时间。为了能够更新应用程序并避免停机,我目前正在阅读零停机时间部署/蓝
我正在编写一个 Node.js 应用程序。我正在使用 request 和 Cheerio 加载一组 URL 并获取该网站的大量信息,现在假设我想要获取的只是标题: var urls = {"url_1
如果不弹出以下错误,我无法安装任何 Node.js 模块。错误代码引用package.json文件。如果知道为什么会发生这种情况,我们将不胜感激。 最佳答案 这些不是错误,它们只是警告。一切都应该如此
如果我运行(从我的项目目录中): supervisor javascripts/index.js 我得到:/usr/bin/env: Node :没有这样的文件或目录 如果我运行: node java
我已遵循使用 Node-Inspector 的所有步骤 但是当我打开应用程序时,我在控制台上看不到任何脚本或日志。 我的应用程序在端口 4000 上运行。我认为唯一可能发生冲突的是端口 8080 上的
我在android中使用rxjava2,有时会遇到这样的问题: Observable.fromArray( // maybe a list about photo url in SD
我目前正在使用 Node 光纤来编写同步服务器端代码。我主要通过 try-catch block 进行错误处理,但外部库或其他小部分异步代码中总是有可能发生错误。我正在考虑使用新的域功能来尝试将这些错
看起来node-debug是node-inspector周围的一个shell?分别什么时候应该使用? 最佳答案 如果您安装node-debug,您只能访问node-debug命令。 如果您安装node
我目前正在代理后面工作,该代理不允许我执行此命令的 HTTP GET 请求阶段: Node node-sass/scripts/build.js 请求阶段: gyp http GET https://
听说node js可以用在服务端。我以前用过jsp。 jsp页面内部的java代码对客户端是不可见的。如果 Node js 只是 javascript,那么它如何对客户端不可见? 最佳答案 首先,No
我正在为 Node native 插件从 node-waf 构建迁移到 node-gyp 构建系统。 node-gyp 说它支持多个目标版本,但我在使用 node-gyp 时找不到如何指定目标 Nod
给定一个 $node ,我正在尝试在以下两种输出该 $node 的方式之间做出决定。 要么 $output = theme('node', $node); 或 node_build_content($
如果package.json中的窗口A打开一个新窗口B,node-main如何访问它?这是我的代码: package.json { "main": "index.html",
我试图在我的 xml 中的特定节点 ( ) 之前插入一个注释节点。这是它的方法: function test(xmlResponse) { var parser = new DOMParse
我正在尝试做npm install wrtc使用 Node 版本 16.14.0 但这还没有完成。它在给npm error code 1所以我试图将 Node 版本更改为以前的 lts 14.19.0
当我在 Visual Studio 中运行 Node.js 应用程序时,我收到以下消息:DeprecationWarning: 'node --debug' 和 'node --debug-brk'
我是一名优秀的程序员,十分优秀!