redbrick/index.js

628 lines
26 KiB
JavaScript
Raw Normal View History

2023-07-29 23:44:21 +02:00
var https = require("https");
var os = require("os");
var http = require("http");
var url = require("url");
var fs = require("fs");
var path = require("path");
2023-07-30 02:24:20 +02:00
var childProcess = require("child_process");
var readline = require("readline");
2023-07-29 23:44:21 +02:00
var version = "UNKNOWN";
try {
version = JSON.parse(fs.readFileSync(__dirname + "/mod.info")).version;
2023-07-30 02:24:20 +02:00
} catch (ex) {
// Can't determine version
2023-07-29 23:44:21 +02:00
}
var configJSONS = JSON.parse(fs.readFileSync(__dirname + "/../../../config.json")); // Read configuration JSON
2023-07-29 23:44:21 +02:00
function Mod() {}
Mod.prototype.callback = function (req, res, serverconsole, responseEnd, href, ext, uobject, search, defaultpage, users, page404, head, foot, fd, elseCallback, configJSON, callServerError, getCustomHeaders, origHref, redirect, parsePostData) {
2023-07-29 23:44:21 +02:00
return function () {
if (!configJSON) {
configJSON = configJSONS;
}
2023-07-29 23:44:21 +02:00
if (!getCustomHeaders) {
var bheaders = JSON.parse(JSON.stringify(configJSON.customHeaders));
} else {
var bheaders = getCustomHeaders();
}
bheaders["Content-Type"] = "text/html"; // HTML output
if (!getCustomHeaders) {
bheaders["Server"] =
"SVR.JS/" +
configJSON.version +
" (" +
os.platform()[0].toUpperCase() +
os.platform().slice(1) +
")"; // Add Server header
}
2023-07-30 02:24:20 +02:00
var abheaders = JSON.parse(JSON.stringify(bheaders));
2023-07-29 23:44:21 +02:00
function executeCGI(fname, req, res, dh, nEnv) {
// Function to execute CGI scripts
2023-07-29 23:44:21 +02:00
var env = JSON.parse(JSON.stringify(process.env));
var nEnvKeys = Object.keys(nEnv);
2023-07-30 02:24:20 +02:00
for (var i = 0; i < nEnvKeys.length; i++) {
2023-07-29 23:44:21 +02:00
env[nEnvKeys[i]] = nEnv[nEnvKeys[i]];
}
var exttointerpreter = {
".pl": ["perl"],
".py": ["python"],
".sh": ["bash"],
".ksh": ["ksh"],
".csh": ["csh"],
".rb": ["ruby"],
".php": ["php-cgi"]
};
if(os.platform() == "win32") {
exttointerpreter[".exe"] = [];
exttointerpreter[".bat"] = ["cmd", "/c"];
exttointerpreter[".vbs"] = ["cscript"];
}
2023-07-29 23:44:21 +02:00
var exttointerpreteruser = {};
fs.readFile(__dirname + "/../../../redbrick-interpreters.json", function (err, data) {
if (!err) {
try {
exttointerpreteruser = JSON.parse(data.toString());
} catch (ex) {}
2023-07-29 23:44:21 +02:00
}
fs.stat(fname, function (err, stats) {
if (err) {
if (!callServerError) {
res.writeHead(500, {
"Content-Type": "text/html",
"Server": "RedBrick/" + version
});
res.end("<html><head></head><body><h1>RedBrick Error!</h1><p>Reason: " + err.message + "</p></body></html>");
2023-07-29 23:44:21 +02:00
} else {
callServerError(500, "RedBrick/" + version, err);
2023-07-29 23:44:21 +02:00
}
return;
2023-07-29 23:44:21 +02:00
}
if (stats.size == 0) {
afterShebangCallback(false);
return;
}
var s = fs.createReadStream(fname);
s.on("error", function (err) {
if (!callServerError) {
res.writeHead(500, {
"Content-Type": "text/html",
"Server": "RedBrick/" + version
});
res.end("<html><head></head><body><h1>RedBrick Error!</h1><p>Reason: " + err.message + "</p></body></html>");
} else {
callServerError(500, "RedBrick/" + version, err);
}
return;
}).on("open", function () {
var rl = readline.createInterface({
input: s,
});
rl.once("line", function (line) {
rl.close();
if (line.substr(0, 2) == "#!") {
var args = line.substr(2).match(/(?:[^" ]|\\(?:.|$)|"[^"]*(?:"|$))+/g);
if (!args) args = [];
args = args.map(function (arg) {
return arg.replace(/"/g, "");
});
afterShebangCallback(args);
} else if (os.platform() != "win32" && line.substr(0, 4) == "\x7fELF") {
afterShebangCallback("binary");
} else {
afterShebangCallback(false);
}
});
});
});
function afterShebangCallback(passedArgs) {
var ext = path.extname(fname);
var args = [];
var buffer = "";
var stderr = "";
var headerendline = -1;
var cned = false;
args.push(fname);
if (passedArgs == "binary") {
filename = (process.cwd() + (os.platform() == "win32" ? "\\" + fname.replace(/\//g, "\\") : "/" + fname)).replace(os.platform() == "win32" ? /\\+/ : /\/+/, os.platform() == "win32" ? "\\" : "/");
args = [];
} else if (passedArgs) {
if (os.platform() == "win32") {
args = passedArgs;
args.push((process.cwd() + ("\\" + fname.replace(/\//g, "\\"))).replace(/\\+/, "\\"));
filename = args.shift();
} else {
filename = (process.cwd() + (os.platform() == "win32" ? "\\" + fname.replace(/\//g, "\\") : "/" + fname)).replace(os.platform() == "win32" ? /\\+/ : /\/+/, os.platform() == "win32" ? "\\" : "/");
args = [];
}
} else {
args = exttointerpreteruser[ext];
if (args === null) {
elseCallback();
return;
} else if (!args) {
args = exttointerpreter[ext];
if (!args) {
elseCallback();
return;
}
}
args.push((process.cwd() + (os.platform() == "win32" ? "\\" + fname.replace(/\//g, "\\") : "/" + fname)).replace(os.platform() == "win32" ? /\\+/ : /\/+/, os.platform() == "win32" ? "\\" : "/"));
filename = args.shift();
2023-07-29 23:44:21 +02:00
}
2023-07-30 02:24:20 +02:00
var wd = fname.split("/");
wd[0] = "";
wd[wd.length - 1] = "";
wd = wd.join(os.platform() == "win32" ? "\\" : "/");
var interpreter = childProcess.spawn(filename, args, {
cwd: (process.cwd() + wd).replace(os.platform() == "win32" ? /\\+/ : /\/+/, os.platform() == "win32" ? "\\" : "/"),
env: env
});
interpreter.on("error", (error) => {
2023-07-29 23:44:21 +02:00
if (!callServerError) {
res.writeHead(500, {
"Content-Type": "text/html",
"Server": "RedBrick/" + version
});
res.end("<html><head></head><body><h1>RedBrick Error!</h1><p>Reason: " + error.message + "</p></body></html>");
2023-07-29 23:44:21 +02:00
} else {
callServerError(500, "RedBrick/" + version, error);
2023-07-29 23:44:21 +02:00
}
});
var dataHandler = function (data) {
buffer += data.toString("latin1");
var m = null;
2023-08-21 23:28:37 +02:00
if (!cned) m = buffer.match(/(?:\r\n\r\n|\n\r\n\r|\n\n|\r\r)/);
if (!cned && m) {
cned = true;
eol = m[0];
headerendline = m.index;
2023-08-21 23:28:37 +02:00
var bheaders = buffer.substr(0, headerendline).split(/(?:\r\n|\n\r|\n|\r)/);
var bheaderso = {};
if (dh) bheaderso = dh;
var code = 200;
var msg = "OK";
2023-08-30 18:13:30 +02:00
var httpMatch = bheaders[0].match(/^HTTP\/[^ ]+ ([0-9]{3})(?: (.*))?$/);
if (httpMatch) {
bheaders.shift();
code = parseInt(httpMatch[1]);
if (httpMatch[2]) msg = httpMatch[2];
else msg = http.STATUS_CODES[code];
} else if (bheaders[0].indexOf(":") == -1) {
var heada = bheaders.shift();
2023-08-30 18:13:30 +02:00
var hso = heada.match(/^([0-9]{3})(?: (.*))?$/);
if (hso) {
code = parseInt(hso[1]);
if (hso[2]) msg = hso[2];
else msg = http.STATUS_CODES[code];
2023-08-02 14:33:36 +02:00
}
}
for (var i = 0; i < bheaders.length; i++) {
2023-08-30 18:13:30 +02:00
var headerp = bheaders[i].match(/^([^:]*)(?:: (.*))?/);
if(!headerp) headerp = [];
var headern = headerp[1];
var headerv = headerp[2];
if (headern.toLowerCase() == "status") {
2023-08-30 18:13:30 +02:00
var httpMatch = headerv.match(/^([0-9]{3})(?: (.*))?$/);
if(httpMatch) {
code = parseInt(httpMatch[1]);
if (httpMatch[2]) msg = httpMatch[2];
else msg = http.STATUS_CODES[code];
}
} else if (headern.toLowerCase() == "set-cookie") {
if (!bheaderso["Set-Cookie"]) bheaderso["Set-Cookie"] = [];
bheaderso["Set-Cookie"].push(headerv);
} else {
bheaderso[headern] = headerv;
}
}
if (code == 200 && (bheaderso["Location"] || bheaderso["location"])) {
code = 302;
msg = "Found";
}
try {
res.writeHead(code, msg, bheaderso);
res.write(buffer.substr(headerendline + eol.length), "latin1");
} catch (ex) {
if (!callServerError) {
res.writeHead(500);
res.end(ex.stack);
} else {
callServerError(500, "RedBrick/" + version, ex);
}
return;
}
2023-07-29 23:44:21 +02:00
} else {
if (cned && !res.finished) res.write(data);
2023-07-29 23:44:21 +02:00
}
};
if (interpreter.stdout) {
interpreter.stdout.on("data", dataHandler);
interpreter.stderr.on("data", function (data) {
stderr += data.toString();
});
req.pipe(interpreter.stdin);
interpreter.on("exit", (code, signal) => {
if (!cned && (signal || code !== 0)) {
var ex = new Error("Process execution failed!" + (stderr ? " Reason: " + stderr.trim() : ""));
if (!callServerError) {
res.writeHead(500);
res.end(ex.stack);
} else {
callServerError(500, "RedBrick/" + version, ex);
}
} else {
res.end();
}
});
2023-07-30 02:24:20 +02:00
}
}
});
2023-07-29 23:44:21 +02:00
}
function executeCGIWithEnv(a, b, req, res, pubip, port, software, dh) {
// Function to set up environment variables and execute CGI scripts
2023-07-29 23:44:21 +02:00
var nEnv = {};
if (req.headers.authorization) {
nEnv["AUTH_TYPE"] = req.headers.authorization.split(" ")[0];
if (nEnv["AUTH_TYPE"] == "Basic") {
var remoteCred = req.headers.authorization.split(" ")[1];
if (!remoteCred) {
nEnv["REMOTE_USER"] = "redbrick_cgi_invalid_user";
} else {
var remoteCredDecoded = Buffer.from(remoteCred, "base64").toString("utf8");
nEnv["REMOTE_USER"] = remoteCredDecoded.split(":")[0];
}
} else {
nEnv["REMOTE_USER"] = "svrjs_this_property_is_not_yet_supported_by_redbrick_cgi";
}
}
2023-07-29 23:44:21 +02:00
nEnv["QUERY_STRING"] = req.url.split("?")[1];
if (nEnv["QUERY_STRING"] == undefined || nEnv["QUERY_STRING"] == "undefined") nEnv["QUERY_STRING"] = "";
nEnv["SERVER_SOFTWARE"] = software;
nEnv["SERVER_PROTOCOL"] = "HTTP/" + req.httpVersion;
2023-08-02 14:33:36 +02:00
if (pubip && (port !== null && port !== undefined)) {
nEnv["SERVER_PORT"] = port;
nEnv["SERVER_ADDR"] = pubip.replace(/^::ffff:/i, "");
if (nEnv["SERVER_ADDR"].indexOf(":") != -1) nEnv["SERVER_ADDR"] = "[" + nEnv["SERVER_ADDR"] + "]";
}
if(configJSON.serverAdministratorEmail && configJSON.serverAdministratorEmail != "[no contact information]") {
nEnv["SERVER_ADMIN"] = configJSON.serverAdministratorEmail;
}
2023-07-29 23:44:21 +02:00
nEnv["SERVER_NAME"] = req.headers.host;
nEnv["DOCUMENT_ROOT"] = process.cwd();
nEnv["PATH_INFO"] = decodeURI(b);
nEnv["PATH_TRANSLATED"] = b ? decodeURI((process.cwd() + (require("os").platform == "win32" ? b.replace(/\//g, "\\") : b)).replace((require("os").platform == "win32" ? /\\\\/g : /\/\//g), (require("os").platform == "win32" ? "\\" : "/"))) : "";
2023-07-29 23:44:21 +02:00
nEnv["REQUEST_METHOD"] = req.method;
nEnv["GATEWAY_INTERFACE"] = "CGI/1.1";
nEnv["REQUEST_URI"] = (!origHref || origHref == href) ? req.url : (origHref + (uobject.search ? ("?" + uobject.search) : ""));
2023-07-30 02:24:20 +02:00
nEnv["REMOTE_ADDR"] = (req.socket.realRemoteAddress ? req.socket.realRemoteAddress : ((req.headers["x-forwarded-for"] && configJSON.enableIPSpoofing) ? req.headers["x-forwarded-for"].split(",")[0].replace(/ /g, "") : req.socket.remoteAddress)).replace(/^::ffff:/i, "");
2023-09-03 11:31:18 +02:00
if(req.socket.realRemoteAddress && req.socket.realRemotePort) {
nEnv["REMOTE_PORT"] = req.socket.realRemotePort;
} else if(!(req.socket.realRemoteAddress && !req.socket.realRemotePort)) {
nEnv["REMOTE_PORT"] = req.socket.remotePort;
}
2023-07-29 23:44:21 +02:00
nEnv["SCRIPT_NAME"] = a;
nEnv["SCRIPT_FILENAME"] = (process.cwd() + (require("os").platform == "win32" ? a.replace(/\//g, "\\") : a)).replace((require("os").platform == "win32" ? /\\\\/g : /\/\//g), (require("os").platform == "win32" ? "\\" : "/"));
if (req.socket.encrypted) nEnv["HTTPS"] = "ON";
2023-07-29 23:44:21 +02:00
if (req.headers["content-type"]) nEnv["CONTENT_TYPE"] = req.headers["content-type"];
if (req.headers["content-length"]) nEnv["CONTENT_LENGTH"] = req.headers["content-length"];
var nh = JSON.parse(JSON.stringify(req.headers));
delete nh["content-type"];
delete nh["content-length"];
var nhKeys = Object.keys(nh);
for (var i = 0; i < nhKeys.length; i++) {
nEnv["HTTP_" + nhKeys[i].replace(/[^0-9A-Za-z]+/g, "_").toUpperCase()] = req.headers[nhKeys[i]];
}
executeCGI("." + a, req, res, dh, nEnv);
}
if (href.match(new RegExp("/cgi-bin(?:$|[?#/])", os.platform() == "win32" ? "i" : ""))) {
fs.stat("." + href, function (err, stats) {
2023-08-02 14:33:36 +02:00
if (!err) {
if (!stats.isFile()) {
fs.stat("." + href + "/index.php", function (e2, s2) {
if (!e2 && s2.isFile()) {
try {
executeCGIWithEnv(
(href + "/index.php").replace(/\/+/g, "/"),
"",
req,
res,
req.socket.localAddress,
req.socket.localPort,
getCustomHeaders ?
getCustomHeaders()["Server"] +
" RedBrick/" +
version :
"SVR.JS/" +
configJSON.version +
" (" +
os.platform()[0].toUpperCase() +
os.platform().slice(1) +
"; Node.JS/" +
process.version +
") RedBrick/" +
version,
bheaders
);
} catch (ex) {
if (!callServerError) {
res.writeHead(500, "Internal Server Error", abheaders);
res.write(
"<html><head><title>500 Internal Server Error</title></head><body><h1>500 Internal Server Error</h1><p>A server had unexpected exception. Below, the stack trace of the error is shown:</p><code>" +
ex.stack.replace(/\r\n/g, "<br/>").replace(/\n/g, "<br/>").replace(/\r/g, "<br/>").replace(/ /g, "&nbsp;") +
"</code><p>Please contact the developer/administrator of the website.</p><p style=\"font-style: italic; font-weight: normal;\">SVR.JS " +
configJSON.version +
" (" +
os.platform()[0].toUpperCase() +
os.platform().slice(1) +
"; Node.JS/" +
process.version +
") RedBrick/" +
version +
" " +
(req.headers.host == undefined ? "" : " on " + req.headers.host) +
"</p></body></html>"
);
res.end();
} else {
callServerError(500, "RedBrick/" + version, ex);
}
}
} else {
fs.stat("." + href + "/index.cgi", function (e3, s3) {
if (!e3 && s3.isFile()) {
try {
executeCGIWithEnv(
(href + "/index.cgi").replace(/\/+/g, "/"),
"",
req,
res,
req.socket.localAddress,
req.socket.localPort,
getCustomHeaders ?
getCustomHeaders()["Server"] +
" RedBrick/" +
version :
"SVR.JS/" +
configJSON.version +
" (" +
os.platform()[0].toUpperCase() +
os.platform().slice(1) +
"; Node.JS/" +
process.version +
") RedBrick/" +
version,
bheaders
);
} catch (ex) {
if (!callServerError) {
res.writeHead(500, "Internal Server Error", abheaders);
res.write(
"<html><head><title>500 Internal Server Error</title></head><body><h1>500 Internal Server Error</h1><p>A server had unexpected exception. Below, the stack trace of the error is shown:</p><code>" +
ex.stack.replace(/\r\n/g, "<br/>").replace(/\n/g, "<br/>").replace(/\r/g, "<br/>").replace(/ /g, "&nbsp;") +
"</code><p>Please contact the developer/administrator of the website.</p><p style=\"font-style: italic; font-weight: normal;\">SVR.JS " +
configJSON.version +
" (" +
os.platform()[0].toUpperCase() +
os.platform().slice(1) +
"; Node.JS/" +
process.version +
") RedBrick/" +
version +
" " +
(req.headers.host == undefined ? "" : " on " + req.headers.host) +
"</p></body></html>"
);
res.end();
} else {
callServerError(500, "RedBrick/" + version, ex);
}
}
} else {
elseCallback();
}
});
2023-08-02 14:33:36 +02:00
}
});
2023-07-29 23:44:21 +02:00
} else {
try {
executeCGIWithEnv(
href,
"",
req,
res,
req.socket.localAddress,
req.socket.localPort,
getCustomHeaders ?
getCustomHeaders()["Server"] +
" RedBrick/" +
version :
"SVR.JS/" +
configJSON.version +
" (" +
os.platform()[0].toUpperCase() +
os.platform().slice(1) +
"; Node.JS/" +
process.version +
") RedBrick/" +
version,
bheaders
);
2023-07-29 23:44:21 +02:00
} catch (ex) {
if (!callServerError) {
res.writeHead(500, "Internal Server Error", abheaders);
res.write(
"<html><head><title>500 Internal Server Error</title></head><body><h1>500 Internal Server Error</h1><p>A server had unexpected exception. Below, the stack trace of the error is shown:</p><code>" +
ex.stack.replace(/\r\n/g, "<br/>").replace(/\n/g, "<br/>").replace(/\r/g, "<br/>").replace(/ /g, "&nbsp;") +
"</code><p>Please contact the developer/administrator of the website.</p><p style=\"font-style: italic; font-weight: normal;\">SVR.JS " +
configJSON.version +
" (" +
os.platform()[0].toUpperCase() +
os.platform().slice(1) +
"; Node.JS/" +
process.version +
") RedBrick/" +
version +
" " +
(req.headers.host == undefined ? "" : " on " + req.headers.host) +
"</p></body></html>"
);
res.end();
2023-07-29 23:44:21 +02:00
} else {
callServerError(500, "RedBrick/" + version, ex);
}
}
}
2023-08-02 14:33:36 +02:00
} else if (err && err.code == "ENOTDIR") {
function checkPath(pth, cb, a) {
// Function to check the path of the file and execute CGI script
2023-07-29 23:44:21 +02:00
var cpth = pth.split("/");
if (cpth.length < 3) {
cb(false);
return;
}
2023-07-30 02:24:20 +02:00
if (!a) b = [];
2023-07-29 23:44:21 +02:00
else var b = a.split("/");
var isFile = false;
fs.stat(pth, function (err, stats) {
if (!err && stats.isFile()) {
cb({
fpth: pth,
rpth: (a !== undefined ? "/" + a : "")
})
} else {
2023-08-02 14:33:36 +02:00
fs.stat(pth + "/index.php", function (e2, s2) {
if (!e2 && s2.isFile()) {
cb({
fpth: (pth + "/index.php").replace(/\/+/g, "/"),
rpth: (a !== undefined ? "/" + a : "")
})
} else {
fs.stat(pth + "/index.cgi", function (e3, s3) {
if (!e3 && s3.isFile()) {
cb({
fpth: (pth + "/index.cgi").replace(/\/+/g, "/"),
rpth: (a !== undefined ? "/" + a : "")
})
} else {
b.unshift(cpth.pop());
return checkPath(cpth.join("/"), cb, b.join("/"));
}
});
2023-08-02 14:33:36 +02:00
}
});
2023-07-29 23:44:21 +02:00
}
});
2023-07-29 23:44:21 +02:00
}
checkPath("." + href, function (pathp) {
if (!pathp) {
elseCallback();
} else {
try {
executeCGIWithEnv(
pathp.fpth.substr(1),
pathp.rpth,
req,
res,
req.socket.localAddress,
req.socket.localPort,
getCustomHeaders ?
getCustomHeaders()["Server"] +
" RedBrick/" +
version :
"SVR.JS/" +
configJSON.version +
" (" +
os.platform()[0].toUpperCase() +
os.platform().slice(1) +
"; Node.JS/" +
process.version +
") RedBrick/" +
version,
bheaders
);
} catch (ex) {
if (!callServerError) {
res.writeHead(500, "Internal Server Error", abheaders);
res.write(
"<html><head><title>500 Internal Server Error</title></head><body><h1>500 Internal Server Error</h1><p>A server had unexpected exception. Below, the stack trace of the error is shown:</p><code>" +
ex.stack.replace(/\r\n/g, "<br/>").replace(/\n/g, "<br/>").replace(/\r/g, "<br/>").replace(/ /g, "&nbsp;") +
"</code><p>Please contact the developer/administrator of the website.</p><p style=\"font-style: italic; font-weight: normal;\">SVR.JS " +
configJSON.version +
" (" +
os.platform()[0].toUpperCase() +
os.platform().slice(1) +
"; Node.JS/" +
process.version +
") RedBrick/" +
version +
" " +
(req.headers.host == undefined ? "" : " on " + req.headers.host) +
"</p></body></html>"
);
res.end();
} else {
callServerError(500, "RedBrick/" + version, ex);
}
2023-07-29 23:44:21 +02:00
}
}
});
2023-08-02 14:33:36 +02:00
} else if (err && err.code == "ENOENT") {
elseCallback(); //Invoke default error handler
2023-07-29 23:44:21 +02:00
} else {
if (!callServerError) {
res.writeHead(500, "Internal Server Error", abheaders);
res.write(
"<html><head><title>500 Internal Server Error</title></head><body><h1>500 Internal Server Error</h1><p>A server had unexpected exception. Below, the stack trace of the error is shown:</p><code>" +
err.stack.replace(/\r\n/g, "<br/>").replace(/\n/g, "<br/>").replace(/\r/g, "<br/>").replace(/ /g, "&nbsp;") +
"</code><p>Please contact the developer/administrator of the website.</p><p style=\"font-style: italic; font-weight: normal;\">SVR.JS " +
configJSON.version +
" (" +
os.platform()[0].toUpperCase() +
os.platform().slice(1) +
"; Node.JS/" +
process.version +
") RedBrick/" +
version +
" " +
(req.headers.host == undefined ? "" : " on " + req.headers.host) +
"</p></body></html>"
);
res.end();
} else {
callServerError(500, "RedBrick/" + version, err);
}
2023-07-29 23:44:21 +02:00
}
});
} else if ((href == "/redbrick-interpreters.json" || (os.platform() == "win32" && href.toLowerCase() == "/redbrick-interpreters.json")) && path.normalize(__dirname + "/../../..") == process.cwd()) {
2023-07-30 02:24:20 +02:00
if (!callServerError) {
res.writeHead(200, "OK", {
"Content-Type": "application/json",
"Server": "RedBrick/" + version
});
res.end(JSON.stringify(exttointerpreteruser, null, 2));
2023-07-30 02:24:20 +02:00
} else {
callServerError(200, "RedBrick/" + version, exttointerpreteruser);
2023-07-30 02:24:20 +02:00
}
2023-07-29 23:44:21 +02:00
} else {
elseCallback();
}
}
}
2023-07-29 23:44:21 +02:00
module.exports = Mod;