// Matrix Construct // // Copyright (C) Matrix Construct Developers, Authors & Contributors // Copyright (C) 2016-2018 Jason Volk <jason@zemos.net> // // Permission to use, copy, modify, and/or distribute this software for any // purpose with or without fee is hereby granted, provided that the above // copyright notice and this permission notice is present in all copies. The // full license for this software is available in the LICENSE file. #include <ircd/js/js.h> struct assertions { assertions(); } static assertions; namespace ircd { namespace js { struct kernel :trap { JSPrincipals *principals; static const char *const source; std::shared_ptr<task> process; value on_call(object::handle, value::handle, const args &) override; value on_set(object::handle, id::handle, value::handle) override; value on_get(object::handle, id::handle, value::handle) override; void on_add(object::handle, id::handle, value::handle) override; bool on_del(object::handle, id::handle) override; bool on_has(object::handle, id::handle) override; void on_enu(object::handle) override; void on_new(object::handle, object &, const args &) override; void on_gc(JSObject *const &) override; void main() noexcept; kernel(); ~kernel() noexcept; } static kernel; } // namespace js } // namespace ircd /////////////////////////////////////////////////////////////////////////////// // // Kernel source // const char *const ircd::js::kernel::source {R"( 'use strict'; import * as console from "server.console"; import * as listener from "server.listener"; var ircd = { }; ircd.opts = { }; ircd.opts.listener = { host: "127.0.0.1", port: 8448, backlog: 1024, ssl_certificate_file: "/home/jason/newcert.pem", ssl_private_key_file_pem: "/home/jason/privkey.pem", ssl_certificate_chain_file: "/home/jason/newcert.pem", ssl_tmp_dh_file: "/home/jason/dh512.pem", }; let main = async function() { console.debug("IRCd.js Greetings from JavaScript"); listener.listen(ircd.opts.listener); }; let fini = function() { console.info("IRCd.js finished"); } main().then(fini); )"}; /////////////////////////////////////////////////////////////////////////////// ircd::js::kernel::kernel() :trap { "[global]", JSCLASS_GLOBAL_FLAGS_WITH_SLOTS(1) | JSCLASS_HAS_PRIVATE | JSCLASS_EMULATES_UNDEFINED, } ,principals { nullptr } ,process { std::make_shared<task>(source) } { } ircd::js::kernel::~kernel() noexcept { } void ircd::js::kernel::main() noexcept try { std::map<std::string, ircd::module> modules; // These modules host databases and have to be loaded first. modules.emplace("root.so"s, "root.so"s); modules.emplace("client_account.so"s, "client_account.so"s); modules.emplace("client_room.so"s, "client_room.so"s); for(const auto &name : mods::available()) if(startswith(name, "client_")) modules.emplace(name, name); task::enter(process, [](auto &task) { task.main(); }); printf("main finished\n"); ctx::wait(); log.debug("Kernel finished"); } catch(const ircd::ctx::interrupted &e) { log.debug("Kernel interrupted"); return; } catch(const std::exception &e) { log.critical("Kernel PANIC: %s", e.what()); std::terminate(); } void ircd::js::kernel::on_gc(JSObject *const &obj) { //trap::on_gc(obj); } void ircd::js::kernel::on_new(object::handle ca, object &obj, const args &args) { trap::on_new(ca, obj, args); } void ircd::js::kernel::on_enu(object::handle obj) { if(!JS_EnumerateStandardClasses(*cx, obj)) throw internal_error("Failed to enumerate standard classes"); } bool ircd::js::kernel::on_has(object::handle obj, id::handle id) { return trap::on_has(obj, id); } bool ircd::js::kernel::on_del(object::handle obj, id::handle id) { return trap::on_del(obj, id); } void ircd::js::kernel::on_add(object::handle obj, id::handle id, value::handle val) { trap::on_add(obj, id, val); } ircd::js::value ircd::js::kernel::on_get(object::handle obj, id::handle id, value::handle val) { return trap::on_get(obj, id, val); } ircd::js::value ircd::js::kernel::on_set(object::handle obj, id::handle id, value::handle val) { return trap::on_set(obj, id, val); } ircd::js::value ircd::js::kernel::on_call(object::handle ca, value::handle that, const args &args) { return trap::on_call(ca, that, args); } // Unmangled alias to the main task. extern "C" void kmain() { ircd::js::kernel.main(); } assertions::assertions() { using ircd::js::error; if(!ircd::js::cx) throw error("Kernel cannot find required JS context instance on this thread."); } const auto on_load([] { using namespace ircd::js; }); const auto on_unload([] { using namespace ircd::js; }); ircd::mapi::header IRCD_MODULE { "IRCd.js kernel - program which helps programs run", on_load, on_unload };