forked from Expensify/App
-
Notifications
You must be signed in to change notification settings - Fork 0
/
proxy.ts
91 lines (80 loc) · 3.41 KB
/
proxy.ts
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
82
83
84
85
86
87
88
89
90
91
import dotenv from 'dotenv';
import http from 'http';
import type {IncomingMessage, ServerResponse} from 'http';
import https from 'https';
import proxyConfig from '../config/proxyConfig';
dotenv.config();
if (process.env.USE_WEB_PROXY === 'false') {
process.stdout.write('Skipping proxy as USE_WEB_PROXY was set to false.\n');
process.exit();
}
const host = 'www.expensify.com';
const stagingHost = 'staging.expensify.com';
const stagingSecureHost = 'staging-secure.expensify.com';
// eslint-disable-next-line no-console
console.log(`Creating proxy with host: ${host} for production API and ${stagingHost} for staging API`);
/**
* Local proxy server that hits the production endpoint
* to get around CORS issues. We use this so that it's
* possible to work on the app within a limited development
* environment that has no local API.
*/
const server = http.createServer((request: IncomingMessage, response: ServerResponse) => {
let hostname = host;
let requestPath = request.url;
// Add CORS headers
response.setHeader('Access-Control-Allow-Origin', '*');
response.setHeader('Access-Control-Allow-Methods', 'GET, POST, PUT, DELETE, OPTIONS');
response.setHeader('Access-Control-Allow-Headers', 'Content-Type, Authorization, X-Requested-With');
// Handle preflight requests
if (request.method === 'OPTIONS') {
response.writeHead(200);
response.end();
return;
}
/**
* When a request is matching a proxy config path we might direct it to a different host (e.g. staging)
* For requests matching proxy config patterns we replace the mapping url (prefix) with the actual path.
* This is done because the staging api root is only intended for the proxy,
* the actual server request must use the /api path.
* For example,
* /api/OpenReport => request sent to production server
* /staging/api/OpenReport => request sent to staging server
* /staging-secure/api/OpenReport => request sent to secure staging server
* /chat-attachments/46545... => request sent to production server
* /receipts/w_... => request sent to production server
* /staging/chat-attachments/46545... => request sent to staging server
*/
if (request.url?.startsWith(proxyConfig.STAGING_SECURE)) {
hostname = stagingSecureHost;
requestPath = request.url.replace(proxyConfig.STAGING_SECURE, '/');
} else if (request.url?.startsWith(proxyConfig.STAGING)) {
hostname = stagingHost;
requestPath = request.url.replace(proxyConfig.STAGING, '/');
}
const proxyRequest = https.request({
hostname,
method: request.method,
path: requestPath,
headers: {
...request.headers,
host: hostname,
// eslint-disable-next-line @typescript-eslint/naming-convention
'user-agent': request.headers['user-agent']?.concat(' Development-NewDot/1.0'),
},
port: 443,
});
request.pipe(proxyRequest);
proxyRequest.on('response', (proxyResponse) => {
response.writeHead(proxyResponse.statusCode ?? 0, proxyResponse.headers);
proxyResponse.pipe(response);
});
proxyRequest.on('error', (error) => {
/* eslint-disable-next-line no-console */
console.log(error);
});
});
server.listen(9000, () => {
/* eslint-disable-next-line no-console */
console.log('Proxy server listening at http://localhost:9000');
});