2021-01-02 18:10:00 +00:00
|
|
|
"use strict";
|
|
|
|
|
2022-03-18 01:24:29 +00:00
|
|
|
var auth = require('auth');
|
|
|
|
var core = require('core');
|
|
|
|
|
|
|
|
var gSessionIndex = 0;
|
|
|
|
|
|
|
|
function makeSessionId() {
|
|
|
|
return (gSessionIndex++).toString();
|
|
|
|
}
|
|
|
|
|
2021-01-02 18:10:00 +00:00
|
|
|
function App() {
|
|
|
|
this._on_output = null;
|
|
|
|
this._send_queue = [];
|
|
|
|
return this;
|
|
|
|
}
|
|
|
|
|
|
|
|
App.prototype.readOutput = function(callback) {
|
|
|
|
this._on_output = callback;
|
|
|
|
}
|
|
|
|
|
|
|
|
App.prototype.makeFunction = function(api) {
|
|
|
|
let self = this;
|
|
|
|
let result = function() {
|
|
|
|
let message = {action: api[0]};
|
|
|
|
for (let i = 1; i < api.length; i++) {
|
|
|
|
message[api[i]] = arguments[i - 1];
|
|
|
|
}
|
|
|
|
self.send(message);
|
|
|
|
};
|
|
|
|
Object.defineProperty(result, 'name', {value: api[0], writable: false});
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
App.prototype.send = function(message) {
|
|
|
|
if (message) {
|
|
|
|
this._send_queue.push(message);
|
|
|
|
}
|
|
|
|
if (this._on_output) {
|
|
|
|
this._send_queue.forEach(message => this._on_output(message));
|
|
|
|
this._send_queue = [];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
function socket(request, response, client) {
|
|
|
|
var process;
|
|
|
|
var options = {};
|
|
|
|
var credentials = auth.query(request.headers);
|
|
|
|
|
2022-01-21 00:49:03 +00:00
|
|
|
response.onClose = async function() {
|
|
|
|
if (process && process.task) {
|
|
|
|
process.task.kill();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
response.onError = async function(error) {
|
|
|
|
if (process && process.task) {
|
|
|
|
process.task.kill();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-01-02 18:10:00 +00:00
|
|
|
response.onMessage = async function(event) {
|
|
|
|
if (event.opCode == 0x1 || event.opCode == 0x2) {
|
|
|
|
var message;
|
|
|
|
try {
|
|
|
|
message = JSON.parse(event.data);
|
|
|
|
} catch (error) {
|
|
|
|
print("ERROR", error, event.data, event.data.length, event.opCode);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (message.action == "hello") {
|
|
|
|
var packageOwner;
|
|
|
|
var packageName;
|
|
|
|
var blobId;
|
|
|
|
var match;
|
2022-01-30 14:51:09 +00:00
|
|
|
var parentApp;
|
2021-01-09 23:06:33 +00:00
|
|
|
if (match = /^\/([&%][^\.]{44}(?:\.\w+)?)(\/?.*)/.exec(message.path)) {
|
2021-01-02 18:10:00 +00:00
|
|
|
blobId = match[1];
|
|
|
|
} else if (match = /^\/\~([^\/]+)\/([^\/]+)\/$/.exec(message.path)) {
|
2022-03-16 00:23:14 +00:00
|
|
|
packageOwner = match[1];
|
|
|
|
packageName = match[2];
|
|
|
|
blobId = await new Database(packageOwner).get('path:' + packageName);
|
2021-01-02 18:10:00 +00:00
|
|
|
if (!blobId) {
|
|
|
|
response.send(JSON.stringify({action: "error", error: message.path + ' not found'}), 0x1);
|
2022-01-18 02:50:46 +00:00
|
|
|
return;
|
2021-01-02 18:10:00 +00:00
|
|
|
}
|
2022-03-16 00:23:14 +00:00
|
|
|
if (packageOwner != 'core') {
|
|
|
|
var coreId = await new Database('core').get('path:' + packageName);
|
2022-01-30 14:51:09 +00:00
|
|
|
parentApp = {
|
2022-03-16 00:23:14 +00:00
|
|
|
path: '/~core/' + packageName + '/',
|
2022-01-30 14:51:09 +00:00
|
|
|
id: coreId,
|
|
|
|
};
|
|
|
|
}
|
2021-01-02 18:10:00 +00:00
|
|
|
}
|
2022-01-30 14:51:09 +00:00
|
|
|
response.send(JSON.stringify({
|
|
|
|
action: "session",
|
|
|
|
credentials: credentials,
|
|
|
|
parentApp: parentApp,
|
|
|
|
id: blobId,
|
|
|
|
}), 0x1);
|
2021-01-02 18:10:00 +00:00
|
|
|
|
|
|
|
options.api = message.api || [];
|
|
|
|
options.credentials = credentials;
|
2022-03-16 00:23:14 +00:00
|
|
|
options.packageOwner = packageOwner;
|
|
|
|
options.packageName = packageName;
|
2021-01-02 18:10:00 +00:00
|
|
|
var sessionId = makeSessionId();
|
|
|
|
if (blobId) {
|
2022-03-18 01:24:29 +00:00
|
|
|
process = await core.getSessionProcessBlob(blobId, sessionId, options);
|
2021-01-02 18:10:00 +00:00
|
|
|
}
|
|
|
|
if (process) {
|
|
|
|
process.app.readOutput(function(message) {
|
|
|
|
response.send(JSON.stringify(message), 0x1);
|
|
|
|
});
|
|
|
|
process.app.send();
|
|
|
|
}
|
|
|
|
|
|
|
|
var ping = function() {
|
|
|
|
var now = Date.now();
|
|
|
|
var again = true;
|
|
|
|
if (now - process.lastActive < process.timeout) {
|
|
|
|
// Active.
|
|
|
|
} else if (process.lastPing > process.lastActive) {
|
|
|
|
// We lost them.
|
|
|
|
if (process.task) {
|
|
|
|
process.task.kill();
|
|
|
|
}
|
|
|
|
again = false;
|
|
|
|
} else {
|
|
|
|
// Idle. Ping them.
|
|
|
|
response.send("", 0x9);
|
|
|
|
process.lastPing = now;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (again) {
|
|
|
|
setTimeout(ping, process.timeout);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (process && process.timeout > 0) {
|
|
|
|
setTimeout(ping, process.timeout);
|
|
|
|
}
|
2022-02-26 21:17:15 +00:00
|
|
|
} else if (message.action == 'enableStats') {
|
2022-03-07 21:06:20 +00:00
|
|
|
if (process) {
|
2022-04-20 23:45:17 +00:00
|
|
|
core.enableStats(process, message.enabled);
|
2022-02-26 21:17:15 +00:00
|
|
|
}
|
2021-01-02 18:10:00 +00:00
|
|
|
} else {
|
|
|
|
if (process && process.eventHandlers['message']) {
|
2022-03-18 01:24:29 +00:00
|
|
|
await core.invoke(process.eventHandlers['message'], [message]);
|
2021-01-02 18:10:00 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
} else if (event.opCode == 0x8) {
|
|
|
|
// Close.
|
2022-01-21 00:49:03 +00:00
|
|
|
if (process && process.task) {
|
2021-01-02 18:10:00 +00:00
|
|
|
process.task.kill();
|
|
|
|
}
|
|
|
|
response.send(event.data, 0x8);
|
|
|
|
} else if (event.opCode == 0xa) {
|
|
|
|
// PONG
|
|
|
|
}
|
|
|
|
|
|
|
|
if (process) {
|
|
|
|
process.lastActive = Date.now();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
exports.socket = socket;
|
2022-03-18 01:24:29 +00:00
|
|
|
exports.App = App;
|