#include #include "js_modules.h" #include #include "modules/js_flipper.h" #define TAG "JS modules" // Absolute path is used to make possible plugin load from CLI #define MODULES_PATH "/ext/apps_data/js_app/plugins" typedef struct { JsModeConstructor create; JsModeDestructor destroy; void* context; } JsModuleData; DICT_DEF2(JsModuleDict, FuriString*, FURI_STRING_OPLIST, JsModuleData, M_POD_OPLIST); static const JsModuleDescriptor modules_builtin[] = { {"flipper", js_flipper_create, NULL}, }; struct JsModules { struct mjs* mjs; JsModuleDict_t module_dict; PluginManager* plugin_manager; }; JsModules* js_modules_create(struct mjs* mjs, CompositeApiResolver* resolver) { JsModules* modules = malloc(sizeof(JsModules)); modules->mjs = mjs; JsModuleDict_init(modules->module_dict); modules->plugin_manager = plugin_manager_alloc( PLUGIN_APP_ID, PLUGIN_API_VERSION, composite_api_resolver_get(resolver)); return modules; } void js_modules_destroy(JsModules* modules) { JsModuleDict_it_t it; for(JsModuleDict_it(it, modules->module_dict); !JsModuleDict_end_p(it); JsModuleDict_next(it)) { const JsModuleDict_itref_t* module_itref = JsModuleDict_cref(it); if(module_itref->value.destroy) { module_itref->value.destroy(module_itref->value.context); } } plugin_manager_free(modules->plugin_manager); JsModuleDict_clear(modules->module_dict); free(modules); } mjs_val_t js_module_require(JsModules* modules, const char* name, size_t name_len) { FuriString* module_name = furi_string_alloc_set_str(name); // Check if module is already installed JsModuleData* module_inst = JsModuleDict_get(modules->module_dict, module_name); if(module_inst) { //-V547 furi_string_free(module_name); mjs_prepend_errorf( modules->mjs, MJS_BAD_ARGS_ERROR, "\"%s\" module is already installed", name); return MJS_UNDEFINED; } bool module_found = false; // Check built-in modules for(size_t i = 0; i < COUNT_OF(modules_builtin); i++) { //-V1008 size_t name_compare_len = strlen(modules_builtin[i].name); if(name_compare_len != name_len) { continue; } if(strncmp(name, modules_builtin[i].name, name_compare_len) == 0) { JsModuleData module = { .create = modules_builtin[i].create, .destroy = modules_builtin[i].destroy}; JsModuleDict_set_at(modules->module_dict, module_name, module); module_found = true; FURI_LOG_I(TAG, "Using built-in module %s", name); break; } } // External module load if(!module_found) { FuriString* module_path = furi_string_alloc(); furi_string_printf(module_path, "%s/js_%s.fal", MODULES_PATH, name); FURI_LOG_I(TAG, "Loading external module %s", furi_string_get_cstr(module_path)); do { uint32_t plugin_cnt_last = plugin_manager_get_count(modules->plugin_manager); PluginManagerError load_error = plugin_manager_load_single( modules->plugin_manager, furi_string_get_cstr(module_path)); if(load_error != PluginManagerErrorNone) { break; } const JsModuleDescriptor* plugin = plugin_manager_get_ep(modules->plugin_manager, plugin_cnt_last); furi_assert(plugin); if(strncmp(name, plugin->name, name_len) != 0) { FURI_LOG_E(TAG, "Module name missmatch %s", plugin->name); break; } JsModuleData module = {.create = plugin->create, .destroy = plugin->destroy}; JsModuleDict_set_at(modules->module_dict, module_name, module); module_found = true; } while(0); furi_string_free(module_path); } // Run module constructor mjs_val_t module_object = MJS_UNDEFINED; if(module_found) { module_inst = JsModuleDict_get(modules->module_dict, module_name); furi_assert(module_inst); if(module_inst->create) { //-V779 module_inst->context = module_inst->create(modules->mjs, &module_object); } } if(module_object == MJS_UNDEFINED) { //-V547 mjs_prepend_errorf(modules->mjs, MJS_BAD_ARGS_ERROR, "\"%s\" module load fail", name); } furi_string_free(module_name); return module_object; }