Page MenuHomePhorge

commonjs.c
No OneTemporary

Size
6 KB
Referenced Files
None
Subscribers
None

commonjs.c

#include "esm/commonjs.h"
#include "esm/loader.h"
#include "internal.h"
#include "reactor.h"
#include "errors.h"
#include "silver/compiler.h"
#include "silver/engine.h"
#include <libgen.h>
#include <string.h>
#include <stdlib.h>
static ant_value_t esm_cjs_require(ant_t *js, ant_value_t *args, int nargs) {
if (nargs < 1 || vtype(args[0]) != T_STR)
return js_mkerr(js, "require() expects a string specifier");
ant_value_t fn = js_getcurrentfunc(js);
ant_value_t data = js_get_slot(fn, SLOT_DATA);
const char *base_path = js_module_eval_active_filename(js);
if (vtype(data) == T_STR) {
ant_offset_t path_len = 0;
ant_offset_t path_off = vstr(js, data, &path_len);
base_path = (const char *)(uintptr_t)(path_off);
}
ant_value_t ns = js_esm_import_sync_from_require(js, args[0], base_path);
if (is_err(ns)) return ns;
if (vtype(ns) == T_OBJ) {
ant_value_t default_export = js_get_slot(ns, SLOT_DEFAULT);
if (vtype(default_export) != T_UNDEF) return default_export;
}
return ns;
}
static ant_value_t esm_cjs_require_resolve(ant_t *js, ant_value_t *args, int nargs) {
if (nargs < 1 || vtype(args[0]) != T_STR) {
return js_mkerr(js, "require.resolve() expects a string specifier");
}
ant_value_t fn = js_getcurrentfunc(js);
ant_value_t data = js_get_slot(fn, SLOT_DATA);
const char *base_path = js_module_eval_active_filename(js);
if (vtype(data) == T_STR) {
ant_offset_t data_len = 0;
ant_offset_t data_off = vstr(js, data, &data_len);
base_path = (const char *)(uintptr_t)(data_off);
}
ant_value_t resolved = js_esm_resolve_specifier_require(js, args[0], base_path);
if (is_err(resolved)) return resolved;
if (vtype(resolved) != T_STR) return resolved;
ant_offset_t len = 0;
ant_offset_t off = vstr(js, resolved, &len);
const char *s = (const char *)(uintptr_t)(off);
static const char *prefix = "file://";
if ((size_t)len >= strlen(prefix) && strncmp(s, prefix, strlen(prefix)) == 0) {
const char *path_part = s + strlen(prefix);
size_t path_len = (size_t)len - strlen(prefix);
return js_mkstr(js, path_part, path_len);
}
return resolved;
}
static bool copy_own_prop(
ant_t *js, ant_value_t dst, ant_value_t src,
const char *key, size_t key_len, ant_value_t *err
) {
ant_value_t value = js_mkundef();
if (js_try_get_own_data_prop(js, src, key, key_len, &value)) {
ant_value_t res = setprop_cstr(js, dst, key, key_len, value);
if (is_err(res)) { *err = res; return false; }
return true;
}
prop_meta_t meta;
if (lookup_string_prop_meta(js, src, key, key_len, &meta) && (meta.has_getter || meta.has_setter)) {
ant_value_t ns = js_as_obj(dst);
int flags = (meta.enumerable ? JS_DESC_E : 0) | (meta.configurable ? JS_DESC_C : 0);
if (meta.has_getter) js_set_getter_desc(js, ns, key, key_len, meta.getter, flags);
if (meta.has_setter) js_set_setter_desc(js, ns, key, key_len, meta.setter, flags);
return true;
}
value = js_get(js, src, key);
if (is_err(value)) { *err = value; return false; }
if (vtype(value) == T_UNDEF) return true;
ant_value_t res = setprop_cstr(js, dst, key, key_len, value);
if (is_err(res)) { *err = res; return false; }
return true;
}
static ant_value_t esm_populate_cjs_namespace(ant_t *js, ant_value_t ns, ant_value_t exports_val) {
ant_value_t set_default = setprop_cstr(js, ns, "default", 7, exports_val);
if (is_err(set_default)) return set_default;
js_set_slot_wb(js, ns, SLOT_DEFAULT, exports_val);
if (!is_object_type(exports_val)) return js_mkundef();
ant_iter_t iter = js_prop_iter_begin(js, exports_val);
const char *key = NULL;
size_t key_len = 0;
while (js_prop_iter_next(&iter, &key, &key_len, NULL)) {
if (key_len == 7 && memcmp(key, "default", 7) == 0) continue;
ant_value_t err = js_mkundef();
if (!copy_own_prop(js, ns, exports_val, key, key_len, &err)) {
if (is_err(err)) { js_prop_iter_end(&iter); return err; }
}}
js_prop_iter_end(&iter);
return js_mkundef();
}
static ant_value_t esm_eval_commonjs_function(
ant_t *js,
const char *code,
size_t code_len,
ant_value_t require_fn,
ant_value_t module_obj,
ant_value_t exports_obj,
ant_value_t filename_val,
ant_value_t dirname_val
) {
static const sv_param_t cjs_params[] = {
SV_PARAM("require"),
SV_PARAM("module"),
SV_PARAM("exports"),
SV_PARAM("__filename"),
SV_PARAM("__dirname"),
};
sv_func_t *compiled = sv_compile_function_with_params(
js, cjs_params,
(int)(sizeof(cjs_params) / sizeof(cjs_params[0])),
code, code_len, false
);
if (!compiled) {
if (js->thrown_exists) return mkval(T_ERR, 0);
return js_mkerr_typed(js, JS_ERR_INTERNAL | JS_ERR_NO_STACK, "Unexpected compile error");
}
js_clear_error_site(js);
ant_value_t args[] = {require_fn, module_obj, exports_obj, filename_val, dirname_val};
return sv_execute_entry(js->vm, compiled, exports_obj, args, 5);
}
ant_value_t esm_load_commonjs_module(
ant_t *js,
const char *module_path, const char *code,
size_t code_len, ant_value_t ns
) {
char *path_copy = strdup(module_path);
if (!path_copy) return js_mkerr(js, "OOM loading CommonJS module");
ant_value_t object_proto = js->sym.object_proto;
ant_value_t module_obj = js_mkobj(js);
ant_value_t exports_obj = js_mkobj(js);
if (is_object_type(object_proto)) {
js_set_proto_init(module_obj, object_proto);
js_set_proto_init(exports_obj, object_proto);
}
js_set(js, module_obj, "exports", exports_obj);
js_set(js, module_obj, "loaded", js_false);
js_set(js, module_obj, "id", js_mkstr(js, module_path, strlen(module_path)));
js_set(js, module_obj, "filename", js_mkstr(js, module_path, strlen(module_path)));
ant_value_t require_fn = js_heavy_mkfun(
js, esm_cjs_require,
js_mkstr(js, module_path, strlen(module_path))
);
ant_value_t require_resolve_fn = js_heavy_mkfun(
js, esm_cjs_require_resolve,
js_mkstr(js, module_path, strlen(module_path))
);
js_set(js, require_fn, "resolve", require_resolve_fn);
char *dir = dirname(path_copy);
ant_value_t dirname_val = js_mkstr(js, dir, strlen(dir));
ant_value_t filename_val = js_mkstr(js, module_path, strlen(module_path));
const char *prev_filename = js->filename;
js_set_filename(js, module_path);
ant_value_t result = esm_eval_commonjs_function(
js, code, code_len,
require_fn, module_obj, exports_obj,
filename_val, dirname_val
);
if (vtype(result) == T_PROMISE) js_run_event_loop(js);
js_set(js, module_obj, "loaded", js_true);
ant_value_t exports_val = js_get(js, module_obj, "exports");
if (!is_err(result) && !js->thrown_exists) {
ant_value_t ns_res = esm_populate_cjs_namespace(js, ns, exports_val);
if (is_err(ns_res)) result = ns_res;
}
js_set_filename(js, prev_filename);
free(path_copy);
if (is_err(result)) return result;
return exports_val;
}

File Metadata

Mime Type
text/x-c
Expires
Sat, May 2, 2:13 AM (1 d, 19 h)
Storage Engine
blob
Storage Format
Raw Data
Storage Handle
541473
Default Alt Text
commonjs.c (6 KB)

Event Timeline