源代码:
const express = require('express'),
httpProxy = require('http-proxy'),
fs = require('fs'),
proxy = new httpProxy.createProxyServer();
const appRoute = {
target: 'http://localhost:5000'
};
const routing = JSON.parse(fs.readFileSync('./odata.json'));
var allowCrossDomain = function(req, res) {
res.header('Access-Control-Allow-Origin', '*');
res.header('Access-Control-Allow-Credentials', 'true');
res.header('Access-Control-Allow-Methods', 'GET,PUT,POST,DELETE');
res.header('Access-Control-Allow-Headers', 'X-Requested-With, Accept, Origin, Referer, User-Agent, Content-Type, Authorization, X-Mindflash-SessionID');
// intercept OPTIONS method
if ('OPTIONS' === req.method) {
res.header(200);
} else {
var dirname = req.url.replace(/^\/([^\/]*).*$/, '$1');
var route = routing[dirname] || appRoute;
console.log(req.method + ': ' + route.target + req.url);
proxy.web(req, res, route);
}
};
var app = express();
app.use(allowCrossDomain);
app.listen(8085);
console.log("Proxy started on http://localhost:8085");
routing JSON 对象解析出来的结果:
data:image/s3,"s3://crabby-images/3a6b8/3a6b82e71ab022eceb8ec309c52e729abf293041" alt=""
app.use(allowCrossDomain);
设置路由器,对路径 /
,使用传入的 allowCrossDomain 函数:
data:image/s3,"s3://crabby-images/a48e4/a48e4180a5ae1b5795644b069459b15fb1964c33" alt=""
data:image/s3,"s3://crabby-images/a78f8/a78f8aca901f0866bdb645e2b5d6d889ca07f627" alt=""
然后监听在 8085 端口上。
我在另一个浏览器窗口,访问 localhost:8085
data:image/s3,"s3://crabby-images/967ab/967abc6c50d47fabca33cd46926119928014fd2a" alt=""
在 routing 配置对象里找不到对应的值:
data:image/s3,"s3://crabby-images/d5aaf/d5aaf9663f211e50bcd1a8aefcf5198f696b8e2d" alt=""
因此使用默认的 appRoute.
被重定向到 5000 端口:
data:image/s3,"s3://crabby-images/6fed2/6fed23142b52d80e12820cd40b3a3b45963c4c21" alt=""
data:image/s3,"s3://crabby-images/e218c/e218cc07c899b6ed8a9127dbfd4767a9565aef23" alt=""
因为 5000 端口并没有服务在监听:
data:image/s3,"s3://crabby-images/ad93a/ad93a9cd188f64eeff11125eadb9e4366a1bd08c" alt=""
data:image/s3,"s3://crabby-images/ec65c/ec65c8b185620d886de75e4d99ecf138ed693c66" alt=""
因此,修改 proxy.js 里的配置。
在 appRoute 里,将 target 指向真实的 SAP UI5 服务器运行地址:
data:image/s3,"s3://crabby-images/40eaf/40eafc7aa9c27f9675080ed66982255b384a72e4" alt=""
然后通过代理服务器访问整个应用。这样,8080 端口就被隐藏掉,起作用的是代理服务器监听的 8085 端口。
Chrome 开发者工具 network 里观察到的全是 8085 端口:
data:image/s3,"s3://crabby-images/c85ef/c85efd4dca9c5adc2333b57ac012d8b916f29b7f" alt=""
odata.json 文件里添加一条名为 sap 的配置:
{
"sap": {
"target": "http://10.0.xx.xx"
}
}
data:image/s3,"s3://crabby-images/1d33c/1d33ce7f3334a4b1c4f082e7e9a8469c77124b4a" alt=""
这样如果我在地址栏里输入:
根据这个正则表达式:
var dirname = req.url.replace(/^\/([^\/]*).*$/, '$1');
会自动将 url 里 8085 后出现的一对 //
里的 sap 提取出来。
然后执行:
var route = routing[dirname] || appRoute;
从 odata.json 的配置里,提取新的路由信息:
data:image/s3,"s3://crabby-images/1d389/1d3892666bab686778ebb50b1535fb09bfdd5120" alt=""
从上图可以看出,凡是发送到 http://localhost:8085/sap/ 的请求,会被重定向到 10.0.xx.xx/sap 了。
当一个请求被代理时,它遵循两个不同的管道,它们将转换应用于 req 和 res 对象。 第一个管道(传入)负责创建和操作将客户端连接到目标的流。 第二个管道(传出)负责创建和操作从目标向客户端返回数据的流。
网友评论