Files
puter/src/backend/src/Kernel.js
2024-12-05 14:00:21 -05:00

412 lines
14 KiB
JavaScript

/*
* Copyright (C) 2024 Puter Technologies Inc.
*
* This file is part of Puter.
*
* Puter is free software: you can redistribute it and/or modify
* it under the terms of the GNU Affero General Public License as published
* by the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU Affero General Public License for more details.
*
* You should have received a copy of the GNU Affero General Public License
* along with this program. If not, see <https://www.gnu.org/licenses/>.
*/
const { AdvancedBase, libs } = require("@heyputer/putility");
const { Context } = require('./util/context');
const BaseService = require("./services/BaseService");
const useapi = require('useapi');
const yargs = require('yargs/yargs')
const { hideBin } = require('yargs/helpers');
const { Extension } = require("./Extension");
const { ExtensionModule } = require("./ExtensionModule");
const { spawn } = require("node:child_process");
const { quot } = libs.string;
class Kernel extends AdvancedBase {
constructor ({ entry_path } = {}) {
super();
this.modules = [];
this.useapi = useapi();
this.useapi.withuse(() => {
def('Module', AdvancedBase);
def('Service', BaseService);
});
this.entry_path = entry_path;
}
add_module (module) {
this.modules.push(module);
}
_runtime_init (boot_parameters) {
const kvjs = require('@heyputer/kv.js');
const kv = new kvjs();
global.kv = kv;
global.cl = console.log;
const { RuntimeEnvironment } = require('./boot/RuntimeEnvironment');
const { BootLogger } = require('./boot/BootLogger');
// Temporary logger for boot process;
// LoggerService will be initialized in app.js
const bootLogger = new BootLogger();
this.bootLogger = bootLogger;
// Determine config and runtime locations
const runtimeEnv = new RuntimeEnvironment({
entry_path: this.entry_path,
logger: bootLogger,
boot_parameters,
});
const environment = runtimeEnv.init();
this.environment = environment;
// polyfills
require('./polyfill/to-string-higher-radix');
}
boot () {
const args = yargs(hideBin(process.argv)).argv
this._runtime_init({ args });
// const express = require('express')
// const app = express();
const config = require('./config');
globalThis.ll = o => o;
globalThis.xtra_log = () => {};
if ( config.env === 'dev' ) {
globalThis.ll = o => {
console.log('debug: ' + require('node:util').inspect(o));
return o;
};
globalThis.xtra_log = (...args) => {
// append to file in temp
const fs = require('fs');
const path = require('path');
const log_path = path.join('/tmp/xtra_log.txt');
fs.appendFileSync(log_path, args.join(' ') + '\n');
}
}
const { consoleLogManager } = require('./util/consolelog');
consoleLogManager.initialize_proxy_methods();
// TODO: temporary dependency inversion; requires moving:
// - rm, so we can move mv
// - mv, so we can move mkdir
// - generate_default_fsentries, so we can move mkdir
// - mkdir, which needs an fs provider
// === START: Initialize Service Registry ===
const { Container } = require('./services/Container');
const services = new Container({ logger: this.bootLogger });
this.services = services;
// app.set('services', services);
const root_context = Context.create({
environment: this.environment,
useapi: this.useapi,
services,
config,
logger: this.bootLogger,
args,
}, 'app');
globalThis.root_context = root_context;
root_context.arun(async () => {
await this._install_modules();
await this._boot_services();
});
// Error.stackTraceLimit = Infinity;
Error.stackTraceLimit = 200;
}
async _install_modules () {
const { services } = this;
// Internal modules
for ( const module_ of this.modules ) {
services.registerModule(module_.constructor.name, module_);
const mod_context = this._create_mod_context(Context.get(), {
name: module_.constructor.name,
['module']: module_,
external: false,
});
await module_.install(mod_context);
}
// External modules
await this.install_extern_mods_();
try {
await services.init();
} catch (e) {
// First we'll try to mark the system as invalid via
// SystemValidationService. This might fail because this service
// may not be initialized yet.
const svc_systemValidation = (() => {
try {
return services.get('system-validation');
} catch (e) {
return null;
}
})();
if ( ! svc_systemValidation ) {
// If we can't mark the system as invalid, we'll just have to
// throw the error and let the server crash.
throw e;
}
await svc_systemValidation.mark_invalid(
'failed to initialize services',
e,
);
}
for ( const module of this.modules ) {
await module.install_legacy?.(Context.get());
}
services.ready.resolve();
// provide services to helpers
const { tmp_provide_services } = require('./helpers');
tmp_provide_services(services);
}
async _boot_services () {
const { services } = this;
await services.ready;
await services.emit('boot.consolidation');
// === END: Initialize Service Registry ===
// self check
(async () => {
await services.ready;
globalThis.services = services;
const log = services.get('log-service').create('init');
log.info('services ready');
log.system('server ready', {
deployment_type: globalThis.deployment_type,
});
})();
await services.emit('boot.activation');
await services.emit('boot.ready');
}
async install_extern_mods_ () {
const path_ = require('path');
const fs = require('fs');
// In runtime directory, we'll create a `mod_packages` directory.`
if ( fs.existsSync('mod_packages') ) {
fs.rmSync('mod_packages', { recursive: true, force: true });
}
fs.mkdirSync('mod_packages');
const mod_install_root_context = Context.get();
const mod_paths = this.environment.mod_paths;
for ( const mods_dirpath of mod_paths ) {
if ( ! fs.existsSync(mods_dirpath) ) {
this.services.logger.error(
`mod directory not found: ${quot(mods_dirpath)}; skipping...`
);
// intentional delay so error is seen
this.services.logger.info('boot will continue in 4 seconds');
await new Promise(rslv => setTimeout(rslv, 4000));
continue;
}
const mod_dirnames = fs.readdirSync(mods_dirpath);
for ( const mod_dirname of mod_dirnames ) {
let mod_path = path_.join(mods_dirpath, mod_dirname);
let stat = fs.lstatSync(mod_path);
while ( stat.isSymbolicLink() ) {
mod_path = fs.readlinkSync(mod_path);
stat = fs.lstatSync(mod_path);
}
if ( ! stat.isDirectory() && !(mod_dirname.endsWith('.js')) ) {
continue;
}
const mod_name = path_.parse(mod_path).name;
const mod_package_dir = `mod_packages/${mod_name}`;
fs.mkdirSync(mod_package_dir);
if ( ! stat.isDirectory() ) {
this.create_mod_package_json(mod_package_dir, {
name: mod_name,
});
fs.copyFileSync(mod_path, path_.join(mod_package_dir, 'main.js'));
} else {
if ( ! fs.existsSync(path_.join(mod_path, 'package.json')) ) {
// Expect main.js or index.js to exist
const options = ['main.js', 'index.js'];
let entry_file = null;
for ( const option of options ) {
if ( fs.existsSync(path_.join(mod_path, option)) ) {
entry_file = option;
break;
}
}
if ( ! entry_file ) {
// If directory is empty, we'll just skip it
if ( fs.readdirSync(mod_path).length === 0 ) {
this.bootLogger.warn(`Empty mod directory ${quot(mod_path)}; skipping...`);
continue;
}
// Other wise, we'll throw an error
this.bootLogger.error(`Expected main.js or index.js in ${quot(mod_path)}`);
if ( ! process.env.SKIP_INVALID_MODS ) {
this.bootLogger.error(`Set SKIP_INVALID_MODS=1 (environment variable) to run anyway.`);
process.exit(1);
} else {
continue;
}
}
this.create_mod_package_json(mod_package_dir, {
name: mod_name,
entry: entry_file,
});
}
fs.cpSync(mod_path, mod_package_dir, {
recursive: true,
});
}
const mod_require_dir = path_.join(process.cwd(), mod_package_dir);
await this.run_npm_install(mod_require_dir);
const mod = new ExtensionModule();
mod.extension = new Extension();
// This is where the module gets the 'use' and 'def' globals
await this.useapi.awithuse(async () => {
// This is where the module gets the 'extension' global
await useapi.aglobalwith({
extension: mod.extension,
}, async () => {
const maybe_promise = require(mod_require_dir);
if ( maybe_promise && maybe_promise instanceof Promise ) {
await maybe_promise;
}
});
});
const mod_context = this._create_mod_context(mod_install_root_context, {
name: mod_dirname,
['module']: mod,
external: true,
mod_path,
});
// TODO: DRY `awithuse` and `aglobalwith` with above
await this.useapi.awithuse(async () => {
await useapi.aglobalwith({
extension: mod.extension,
}, async () => {
// This is where the 'install' event gets triggered
await mod.install(mod_context);
});
});
}
}
}
_create_mod_context (parent, options) {
const path_ = require('path');
const fs = require('fs');
const modapi = {};
let mod_path = options.mod_path;
if ( ! mod_path && options.module.dirname ) {
mod_path = options.module.dirname();
}
if ( mod_path ) {
modapi.libdir = (prefix, directory) => {
const fullpath = path_.join(mod_path, directory);
const fsitems = fs.readdirSync(fullpath);
for ( const item of fsitems ) {
if ( ! item.endsWith('.js') ) {
continue;
}
const stat = fs.statSync(path_.join(fullpath, item));
if ( ! stat.isFile() ) {
continue;
}
const name = item.slice(0, -3);
const path = path_.join(fullpath, item);
let lib = require(path);
// TODO: This context can be made dynamic by adding a
// getter-like behavior to useapi.
this.useapi.def(`${prefix}.${name}`, lib);
}
}
}
const mod_context = parent.sub({ modapi }, `mod:${options.name}`);
return mod_context;
}
create_mod_package_json (mod_path, { name, entry }) {
const fs = require('fs');
const path_ = require('path');
const data = JSON.stringify({
name,
version: '1.0.0',
main: entry ?? 'main.js',
});
console.log('WRITING TO', path_.join(mod_path, 'package.json'));
fs.writeFileSync(path_.join(mod_path, 'package.json'), data);
}
async run_npm_install (path) {
const proc = spawn('npm', ['install'], { cwd: path, stdio: 'inherit' });
return new Promise((rslv, rjct) => {
proc.on('close', code => {
if ( code !== 0 ) {
throw new Error(`exit code: ${code}`);
}
rslv();
});
proc.on('error', err => {
rjct(err);
})
})
}
}
module.exports = { Kernel };