"详解 Node 如何让一个端口同时支持 HTTPS 与 HTTP"
Node.js 作为一个高性能的 Web 服务器,可以很简单地创建一个 HTTP 或 HTTPS 的服务器。这篇文章主要介绍了 Node 如何让一个端口同时支持 HTTPS 与 HTTP。
在介绍解决方案之前,让我们先了解 Node.js 中创建 HTTP 和 HTTPS 服务器的基本知识。创建一个简单的 HTTP 服务器:
```javascript
var http = require('http');
var httpPort = 3345;
var server = http.createServer(function(req, res){
res.writeHead(200, {'Content-Type': 'text/plain'});
res.end('hello world!');
}).listen(httpPort);
```
创建一个 HTTPS 服务器需要生成证书,例如:
```javascript
var https = require('https');
var fs = require('fs');
var httpsPort = 3346;
var options = {
key: fs.readFileSync('./cakey.pem'),
cert: fs.readFileSync('./cacert.pem')
};
var sserver = https.createServer(options, function(req, res){
res.writeHead(200, {'Content-Type': 'text/plain'});
res.end('secured hello world!');
}).listen(httpsPort);
```
从上文我们可以看出,Node 生成的每个服务器必须分配一个端口。那么如果我们在工作中遇到一个需求:让同一个端口或地址既支持 HTTP 协议又支持 HTTPS 协议,这时候我们该怎么办?有的同学很可能想到用 Nginx 做反向代理,这不失为一个解决方案,但这也同样意味着增加了产品的复杂度,用户并不想去折腾 Nginx。
解决方案是使用 Node 的 TCP 服务器来实现反向代理。根据 OSI 七层模型,HTTP 和 HTTPS 都属于应用层协议,只要我们在底层协议中进行反向代理,就可以解决这个问题!Node 可以让我们很方便地创建一个 TCP 服务器!我们的核心代码如下:
```javascript
var net = require('net');
var http = require('http');
var https = require('https');
var fs = require('fs');
var httpPort = 3345;
var httpsPort = 3346;
var server = http.createServer(function(req, res){
res.writeHead(200, {'Content-Type': 'text/plain'});
res.end('hello world!');
}).listen(httpPort);
var options = {
key: fs.readFileSync('./cakey.pem'),
cert: fs.readFileSync('./cacert.pem')
};
var sserver = https.createServer(options, function(req, res){
res.writeHead(200, {'Content-Type': 'text/plain'});
res.end('secured hello world!');
}).listen(httpsPort);
net.createServer(function(socket){
socket.once('data', function(buf){
console.log(buf[0]);
// https 数据流的第一位是十六进制“16”,转换成十进制就是 22
var address = buf[0] === 22 ? httpsPort : httpPort;
// 创建一个指向 https 或 http 服务器的链接
var proxy = net.createConnection(address, function() {
proxy.write(buf);
// 反向代理的过程,tcp 接受的数据交给代理链接,代理链接服务器端返回数据交由 socket 返回给客户端
socket.pipe(proxy).pipe(socket);
});
});
```
在上面的代码中,我们使用了 Node 的 TCP 服务器来监听客户端的请求,并根据请求的协议(HTTPS 或 HTTP)将其转发到对应的服务器上。这就实现了让同一个端口或地址既支持 HTTP 协议又支持 HTTPS 协议的需求。
Node.js 提供了强大的支持来实现让同一个端口或地址既支持 HTTP 协议又支持 HTTPS 协议,解决了开发中常见的问题。