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