-
Notifications
You must be signed in to change notification settings - Fork 0
/
http-proxy-server-in-80-lines.js
81 lines (78 loc) · 3.26 KB
/
http-proxy-server-in-80-lines.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
//https://qiita.com/LightSpeedC/items/5c1edc2c974206c743f4
'use strict';
const http = require('http'), url = require('url'), net = require('net');
const HTTP_PORT = process.argv[2] || 8080; // internal proxy server port
const PROXY_URL = process.argv[3] || null; // external proxy server URL
const PROXY_HOST = PROXY_URL ? url.parse(PROXY_URL).hostname : null;
const PROXY_PORT = PROXY_URL ? (url.parse(PROXY_URL).port || 80) : null;
//console.log("test");
const server = http.createServer(function onCliReq(cliReq, cliRes) {
let svrSoc;
const cliSoc = cliReq.socket, x = url.parse(cliReq.url);
console.log('cliReq.url=%s', cliReq.url);
const svrReq = http.request({host: PROXY_HOST || x.hostname,
port: PROXY_PORT || x.port || 80,
path: PROXY_URL ? cliReq.url : x.path,
method: cliReq.method, headers: cliReq.headers,
agent: cliSoc.$agent}, function onSvrRes(svrRes) {
svrSoc = svrRes.socket;
cliRes.writeHead(svrRes.statusCode, svrRes.headers);
svrRes.pipe(cliRes);
});
cliReq.pipe(svrReq);
svrReq.on('error', function onSvrReqErr(err) {
cliRes.writeHead(400, err.message, {'content-type': 'text/html'});
cliRes.end('<h1>' + err.message + '<br/>' + cliReq.url + '</h1>');
onErr(err, 'svrReq', x.hostname + ':' + (x.port || 80), svrSoc);
});
})
.on('clientError', (err, soc) => onErr(err, 'cliErr', '', soc))
.on('connect', function onCliConn(cliReq, cliSoc, cliHead) {
//console.log("connected");
//sys.log("syslog");
console.log('cliReq.url=%s', cliReq.url);
const x = url.parse('https://' + cliReq.url);
//console.log('x.href=%s', x.href);
let svrSoc;
//console.log("test4");
if (PROXY_URL) {
//console.log("test2");
const svrReq = http.request({host: PROXY_HOST, port: PROXY_PORT,
path: cliReq.url, method: cliReq.method, headers: cliReq.headers,
agent: cliSoc.$agent});
svrReq.end();
svrReq.on('connect', function onSvrConn(svrRes, svrSoc2, svrHead) {
svrSoc = svrSoc2;
cliSoc.write('HTTP/1.0 200 Connection established\r\n\r\n');
if (cliHead && cliHead.length) svrSoc.write(cliHead);
if (svrHead && svrHead.length) cliSoc.write(svrHead);
svrSoc.pipe(cliSoc);
cliSoc.pipe(svrSoc);
svrSoc.on('error', err => onErr(err, 'svrSoc', cliReq.url, cliSoc));
});
svrReq.on('error', err => onErr(err, 'svrRq2', cliReq.url, cliSoc));
}
else {
//console.log("test3");
svrSoc = net.connect(x.port || 443, x.hostname, function onSvrConn() {
cliSoc.write('HTTP/1.0 200 Connection established\r\n\r\n');
if (cliHead && cliHead.length) svrSoc.write(cliHead);
cliSoc.pipe(svrSoc);
});
svrSoc.pipe(cliSoc);
svrSoc.on('error', err => onErr(err, 'svrSoc', cliReq.url, cliSoc));
}
cliSoc.on('error', err => onErr(err, 'cliSoc', cliReq.url, svrSoc));
})
.on('connection', function onConn(cliSoc) {
//console.log("aaa");
cliSoc.$agent = new http.Agent({keepAlive: true});
cliSoc.$agent.on('error', err => console.log('agent:', err));
})
.listen(HTTP_PORT, () =>
console.log('http proxy server started on port ' + HTTP_PORT +
(PROXY_URL ? ' -> ' + PROXY_HOST + ':' + PROXY_PORT : '')));
function onErr(err, msg, url, soc) {
if (soc) soc.end();
console.log('%s %s: %s', new Date().toLocaleTimeString(), msg, url, err + '');
}