詳解node如何讓一個(gè)端口同時(shí)支持https與http
眾所周知node是一個(gè)高性能的web服務(wù)器,使用它可以很簡(jiǎn)單的創(chuàng)建一個(gè)http或https的服務(wù)器。
比如一個(gè)很簡(jiǎn)單的http服務(wù)器:
var http = require('http');
var https = require('https');
var httpPort = 3345;
var server = http.createServer(function(req, res){
res.writeHead(200, {'Content-Type': 'text/plain'});
res.end('hello world!');
}).listen(httpPort);
https服務(wù)器需要生成證書,詳情請(qǐng)看這篇文章:HTTPS 的原理和 NodeJS 的實(shí)現(xiàn)。這里我們直接看最終成果,附件證書。
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生成的每個(gè)服務(wù)器必須分配一個(gè)端口。那么如果我們?cè)诠ぷ髦杏龅揭粋€(gè)需求:讓同一個(gè)端口或地址既支持http協(xié)議又支持https協(xié)議,這時(shí)候我們?cè)撛趺崔k,有的同學(xué)很可能想到用nginx做反向代理,這不失為一個(gè)解決方案,但這也同樣意味著增加了產(chǎn)品的復(fù)雜度,用戶并不想去折騰ngnix。
辦法是有的,原理就要搬出OSI的七層模型:

HTTP與HTTPS都屬于應(yīng)用層協(xié)議,所以只要我們在底層協(xié)議中進(jìn)行反向代理,就可以解決這個(gè)問題! 剛好node可以讓我們很方便的創(chuàng)建一個(gè)tcp服務(wù)器!
所以我們的核心代碼如下:
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數(shù)據(jù)流的第一位是十六進(jìn)制“16”,轉(zhuǎn)換成十進(jìn)制就是22
var address = buf[0] === 22 ? httpsPort : httpPort;
//創(chuàng)建一個(gè)指向https或http服務(wù)器的鏈接
var proxy = net.createConnection(address, function() {
proxy.write(buf);
//反向代理的過程,tcp接受的數(shù)據(jù)交給代理鏈接,代理鏈接服務(wù)器端返回?cái)?shù)據(jù)交由socket返回給客戶端
socket.pipe(proxy).pipe(socket);
});
proxy.on('error', function(err) {
console.log(err);
});
});
socket.on('error', function(err) {
console.log(err);
});
}).listen(3344);
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
使用Nodejs?實(shí)現(xiàn)一個(gè)簡(jiǎn)單的?Redis客戶端(推薦)
在nodejs中支持TCP連接的是net模塊,?其中使用createConnection(config)或者直接new?Socket(config)來初始化一個(gè)TCP連接,這篇文章主要介紹了用Nodejs?實(shí)現(xiàn)一個(gè)簡(jiǎn)單的?Redis客戶端,需要的朋友可以參考下2022-11-11
Node.js服務(wù)器環(huán)境下使用Mock.js攔截AJAX請(qǐng)求的教程
Mock.js這個(gè)JavaScript庫(kù)最常見的用法便是被用來攔截AJAX請(qǐng)求,well,這里我們就來看一下Node.js服務(wù)器環(huán)境下使用Mock.js攔截AJAX請(qǐng)求的教程:2016-05-05
Express中使用Swagger的實(shí)現(xiàn)示例
swagger-express是一個(gè)規(guī)范和完整的框架實(shí)現(xiàn),本文主要介紹了Express中使用Swagger的實(shí)現(xiàn)示例,具有一定的參考價(jià)值,感興趣的可以了解一下2023-12-12

