我试图从jira服务器下载一个文件使用URL,但我得到一个错误。
如何在代码中包含证书进行验证?
错误:
Error: unable to verify the first certificate in nodejs
at Error (native)
at TLSSocket.<anonymous> (_tls_wrap.js:929:36)
at TLSSocket.emit (events.js:104:17)
at TLSSocket._finishInit (_tls_wrap.js:460:8)
我的Nodejs代码:
var https = require("https");
var fs = require('fs');
var options = {
host: 'jira.example.com',
path: '/secure/attachment/206906/update.xlsx'
};
https.get(options, function (http_res) {
var data = "";
http_res.on("data", function (chunk) {
data += chunk;
});
http_res.on("end", function () {
var file = fs.createWriteStream("file.xlsx");
data.pipe(file);
});
});
@sch提供的答案对我帮助很大。我还想补充一些东西。
在开发环境中工作时,您的SSL证书是由您自己的一个自签名证书颁发的(因此不存在中间证书),NODE_EXTRA_CA_CERTS环境变量需要引用这个自签名证书。自签名证书需保存为PEM格式。导出证书后,我做了以下操作:
set NODE_EXTRA_CA_CERTS=C:\rootCert.pem
(值得注意的是,我在Windows上运行node,到PEM的路径没有引用)
使用命令行中的{{node}},我能够确认是否已经解决了这个问题,通过调用:
https.get("https://my.dev-domain.local")
这实际上为我解决了这个问题,从https://www.npmjs.com/package/ssl-root-cas
// INCORRECT (but might still work)
var server = https.createServer({
key: fs.readFileSync('privkey.pem', 'ascii'),
cert: fs.readFileSync('cert.pem', 'ascii') // a PEM containing ONLY the SERVER certificate
});
// CORRECT (should always work)
var server = https.createServer({
key: fs.readFileSync('privkey.pem', 'ascii'),
cert: fs.readFileSync('fullchain.pem', 'ascii') // a PEM containing the SERVER and ALL INTERMEDIATES
});
解决这个问题的另一种方法是使用下面的模块。
node_extra_ca_certs_mozilla_bundle
该模块无需任何代码修改,即可生成包含Mozilla信任的所有根证书和中间证书的PEM文件。您可以使用以下环境变量(适用于Nodejs v7.3+),
NODE_EXTRA_CA_CERTS
生成与上述环境变量一起使用的PEM文件。你可以使用以下方法安装模块:
npm install --save node_extra_ca_certs_mozilla_bundle
然后使用环境变量启动节点脚本。
NODE_EXTRA_CA_CERTS=node_modules/node_extra_ca_certs_mozilla_bundle/ca_bundle/ca_intermediate_root_bundle.pem node your_script.js
使用生成的PEM文件的其他方法如下:
https://github.com/arvind-agarwal/node_extra_ca_certs_mozilla_bundle
注:我是上述模块的作者。
我可以通过mozilla或chrome等浏览器获得证书链。
打开网站,进入网页的证书设置,下载证书链作为文件名(first-chain. txt)。Pem, second-chain.pem),应该是Pem格式
——开始证书
MIIF3jCCA8agAwIBAgIQAf1tMPyjylGoG7xkDjUDLTANBgkqhkiG9w0BAQwFADCB
......
——结束证书
——开始证书
MIIF3jCCA8agAwIBAgIQAf1tMPyjylGoG7xkDjUDLTANBgkqhkiG9w0BAQwFADCB
......
——结束证书
然后在你的nodejs代码中,我在typescript上做了,我添加了2个cas,因为我有2个webserver请求
从“HTTPS”导入HTTPS
从“ssl-root-cas”导入cas
......
扩展缓冲区
addFile(file: string): Buffer[]
}
const caList = cas.create()作为caList
caList.addFile (process.env。` ` certs/first-chain.pem ` `
caList.addFile (process.env。'certs/second-chain.pem')
然后,当我需要使websocket wss连接,我添加代理与新cas列表请求
this.client。connect(KtUrl, undefined, undefined, undefined, {
代理:新的https。代理({
ca: caList
})
})
还必须添加ssl-root-cas文件名ssl-root-cas.d的定义文件。这样打字稿就不会报错
声明模块'ssl-root-cas' {
function create(): string | Buffer | (string | Buffer)[] | undefined
}