詳解node如何讓一個(gè)端口同時(shí)支持https與http
眾所周知node是一個(gè)高性能的web服務(wù)器,使用它可以很簡單的創(chuàng)建一個(gè)http或https的服務(wù)器。
比如一個(gè)很簡單的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ù)器需要生成證書,詳情請看這篇文章: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è)端口。那么如果我們在工作中遇到一個(gè)需求:讓同一個(gè)端口或地址既支持http協(xié)議又支持https協(xié)議,這時(shí)候我們該怎么辦,有的同學(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);
以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
使用Nodejs?實(shí)現(xiàn)一個(gè)簡單的?Redis客戶端(推薦)
在nodejs中支持TCP連接的是net模塊,?其中使用createConnection(config)或者直接new?Socket(config)來初始化一個(gè)TCP連接,這篇文章主要介紹了用Nodejs?實(shí)現(xiàn)一個(gè)簡單的?Redis客戶端,需要的朋友可以參考下2022-11-11Node.js服務(wù)器環(huán)境下使用Mock.js攔截AJAX請求的教程
Mock.js這個(gè)JavaScript庫最常見的用法便是被用來攔截AJAX請求,well,這里我們就來看一下Node.js服務(wù)器環(huán)境下使用Mock.js攔截AJAX請求的教程:2016-05-05Express中使用Swagger的實(shí)現(xiàn)示例
swagger-express是一個(gè)規(guī)范和完整的框架實(shí)現(xiàn),本文主要介紹了Express中使用Swagger的實(shí)現(xiàn)示例,具有一定的參考價(jià)值,感興趣的可以了解一下2023-12-12