怎么編寫網(wǎng)站的HTTP 反向代理服務(wù)器?在 Node.js 上實(shí)現(xiàn)一個(gè)簡單的 HTTP 代理程序還是非常簡單的,本文章的例子的核心代碼只有 60 多行,只要理解 內(nèi)置 http 模塊 的基本用法即可,具體請(qǐng)看下文。
怎么編寫網(wǎng)站的HTTP 反向代理服務(wù)器?在 Node.js 上實(shí)現(xiàn)一個(gè)簡單的 HTTP 代理程序還是非常簡單的,本文章的例子的核心代碼只有 60 多行,只要理解 內(nèi)置 http 模塊 的基本用法即可,具體請(qǐng)看下文。
接口設(shè)計(jì)與相關(guān)技術(shù)
使用 http.createServer() 創(chuàng)建的 HTTP 服務(wù)器,處理請(qǐng)求的函數(shù)格式一般為 function (req, res) {} (下文簡稱為 requestHandler ),其接收兩個(gè)參數(shù),分別為 http.IncomingMessage 和 http.ServerResponse 對(duì)象,我們可以通過這兩個(gè)對(duì)象來取得請(qǐng)求的所有信息并對(duì)它進(jìn)行響應(yīng)。
主流的 Node.js Web 框架的中間件(比如 connect )一般都有兩種形式:
中間件不需要任何初始化參數(shù),則其導(dǎo)出結(jié)果為一個(gè) requestHandler
中間件需要初始化參數(shù),則其導(dǎo)出結(jié)果為中間件的初始化函數(shù),執(zhí)行該初始化函數(shù)時(shí),傳入一個(gè) options 對(duì)象,執(zhí)行后返回一個(gè) requestHandler
為了使代碼更規(guī)范,在本文例子中,我們將反向代理程序設(shè)計(jì)成一個(gè)中間件的格式,并使用以上第二種接口形式:
// 生成中間件
const handler = reverseProxy({
// 初始化參數(shù),用于設(shè)置目標(biāo)服務(wù)器列表
servers: ["127.0.0.1:3001", "127.0.0.1:3002", "127.0.0.1:3003"]
});
// 可以直接在 http 模塊中使用
const server = http.createServer(handler);
// 作為中間件在 connect 模塊中使用
說明:
上面的代碼中, reverseProxy 是反向
代理服務(wù)器中間件的初始化函數(shù),它接受一個(gè)對(duì)象參數(shù), servers 是后端服務(wù)器地址列表,每個(gè)地址為 IP 地址:端口 這樣的格式
執(zhí)行 reverseProxy() 后返回一個(gè) function (req, res) {} 這樣的函數(shù),用于處理 HTTP 請(qǐng)求,可作為 http.createServer() 和 connect 中間件的 app.use() 的處理函數(shù)
當(dāng)接收到客戶端請(qǐng)求時(shí),按順序循環(huán)從 servers 數(shù)組中取出一個(gè)服務(wù)器地址,將請(qǐng)求代理到這個(gè)地址的服務(wù)器上
服務(wù)器在接收到 HTTP 請(qǐng)求后,首先需要發(fā)起一個(gè)新的 HTTP 請(qǐng)求到要代理的目標(biāo)服務(wù)器,可以使用 http.request() 來發(fā)送請(qǐng)求:
const req = http.request(
{
hostname: "目標(biāo)服務(wù)器地址",
port: "80",
path: "請(qǐng)求路徑",
headers: {
"x-y-z": "請(qǐng)求頭"
}
},
function(res) {
// res 為響應(yīng)對(duì)象
console.log(res.statusCode);
}
);
// 如果有請(qǐng)求體需要發(fā)送,使用 write() 和 end()
req.end();
要將客戶端的請(qǐng)求體( Body 部分,在 POST 、 PUT 這些請(qǐng)求時(shí)會(huì)有請(qǐng)求體)轉(zhuǎn)發(fā)到另一個(gè)服務(wù)器上,可以使用 Stream 對(duì)象的 pipe() 方法,比如:
// req 和 res 為客戶端的請(qǐng)求和響應(yīng)對(duì)象
// req2 和 res2 為服務(wù)器發(fā)起的代理請(qǐng)求和響應(yīng)對(duì)象
// 將 req 收到的數(shù)據(jù)轉(zhuǎn)發(fā)到 req2
req.pipe(req2);
// 將 res2 收到的數(shù)據(jù)轉(zhuǎn)發(fā)到 res
res2.pipe(res);
說明:
req 對(duì)象是一個(gè) Readable Stream (可讀流),通過 data 事件來接收數(shù)據(jù),當(dāng)收到 end事件時(shí)表示數(shù)據(jù)接收完畢
res 對(duì)象是一個(gè) Writable Stream (可寫流),通過 write() 方法來輸出數(shù)據(jù), end() 方法來結(jié)束輸出。
以上就是小編對(duì)于怎么編寫網(wǎng)站的HTTP 反向代理服務(wù)器的解答。