-
Notifications
You must be signed in to change notification settings - Fork 4
/
Copy pathlivecode.mjs
102 lines (85 loc) · 4.3 KB
/
livecode.mjs
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
92
93
94
95
96
97
98
99
100
101
102
let debugFirebase = undefined;
import * as observable from './observable.mjs';
export const setLivecodeFirebase = (firebase) => debugFirebase = firebase;
export async function livecodeMiddleware(req, res, next) {
try {
// Priority given to anonymous paths
const tunnelPath = req.requestConfig.correlation
? `/services/http/debug/${req.requestConfig.correlation}`
: req.cachedConfig?.debugger?.path
if (tunnelPath) {
// TODO, this watcher is introducing 70ms of latency, we should really just track the status with the dynamic config
const status = await debugFirebase.database().ref(tunnelPath + "/status").once('value');
if (/* old way */ status.val() === 'online' || /* new way */ status.val()?.state === 'online') {
// todo unsubscribe when status leaves offline
console.log("Tunneling request over", tunnelPath)
const cellReq = observable.createCellRequest(req);
const id = req.id;
// SECURITY: Now we ensure all the secrets resolve and they are keyed by the domain being executed
const namespace = req.cachedConfig.namespace;
Object.keys(req.pendingSecrets || {}).map(key => {
if (!key.startsWith(namespace)) {
const err = new Error(`Notebooks by ${namespace} cannot access ${key}`);
err.status = 403;
throw err;
}
});
// Resolve all outstanding secret fetches
const secrets = await resolveObject(req.pendingSecrets || {});
// ergonomics improvement, strip namespace_ prefix of all secrets
Object.keys(secrets).forEach(
secretName => secrets[secretName.replace(`${namespace}_`, '')] = secrets[secretName]);
// mixin api_key
if (req.cachedConfig.api_key) secrets.api_key = req.cachedConfig.api_key;
debugFirebase.database().ref(tunnelPath + "/requests/" + id).set({
request: cellReq,
context: {
serverless: false,
namespace,
secrets
}
});
debugFirebase.database().ref(tunnelPath + "/requests/" + id + "/headers").on('child_added', snap => {
res.header(snap.key, snap.val())
});
debugFirebase.database().ref(tunnelPath + "/requests/" + id + "/status").on('value', snap => {
if (snap.val() === null) return; // ignore first null result
res.status(snap.val())
});
debugFirebase.database().ref(tunnelPath + "/requests/" + id + "/writes").on('child_added', snap => {
const chunk = snap.val();
if (chunk.ARuRQygChDsaTvPRztEb === "bufferBase64") {
chunk = Buffer.from(chunk.value, 'base64')
}
res.write(chunk);
});
debugFirebase.database().ref(tunnelPath + "/requests/" + id + "/response").on('value', snap => {
if (snap.val() === null) return; // ignore first null result
const result = JSON.parse(snap.val());
result.json ? res.json(result.json) : null;
if (result.send) {
if (result.send.ARuRQygChDsaTvPRztEb === "bufferBase64") {
res.send(Buffer.from(result.send.value, 'base64'))
} else {
res.send(result.send)
}
}
result.end ? res.end() : null;
});
} else {
console.log("Debugging receiver is not online");
next();
}
} else {
next();
}
} catch (err) {
console.error(err);
next();
}
}
function resolveObject(obj) {
return Promise.all(
Object.entries(obj).map(async ([k, v]) => [k, await v])
).then(Object.fromEntries);
}