眾所周知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í)有所幫助,也希望大家多多支持武林網(wǎng)。
新聞熱點(diǎn)
疑難解答
圖片精選
網(wǎng)友關(guān)注