static int es_faprovider_register(duk_context *ctx) { es_context_t *ec = es_get(ctx); const char *name = duk_require_string(ctx, 0); es_fap_t *ef = es_resource_alloc(&es_resource_fap); ef->fap.fap_opaque = ef; if(es_prop_is_true(ctx, 1, "cachable")) ef->fap.fap_flags |= FAP_ALLOW_CACHE, ef->fap.fap_fini = es_fap_fini; ef->fap.fap_open = es_fap_open; ef->fap.fap_seek = es_fap_seek; ef->fap.fap_read = es_fap_read; ef->fap.fap_close = es_fap_close; ef->fap.fap_fsize = es_fap_fsize; ef->fap.fap_stat = es_fap_stat; ef->name = strdup(name); ef->fap.fap_name = ef->name; es_resource_retain(&ef->super); // Refcount owned by FAP fileaccess_register_dynamic(&ef->fap); es_resource_link(&ef->super, ec, 1); es_root_register(ctx, 1, ef); es_resource_push(ctx, &ef->super); return 1; }
static int es_route_create(duk_context *ctx) { const char *str = duk_safe_to_string(ctx, 0); if(str[0] != '^') { int l = strlen(str); char *s = alloca(l + 2); s[0] = '^'; memcpy(s+1, str, l+1); str = s; } es_context_t *ec = es_get(ctx); hts_mutex_lock(&route_mutex); es_route_t *er; LIST_FOREACH(er, &routes, er_link) if(!strcmp(er->er_pattern, str)) break; if(er != NULL) { hts_mutex_unlock(&route_mutex); duk_error(ctx, DUK_ERR_ERROR, "Route %s already exist", str); } er = es_resource_alloc(&es_resource_route); if(hts_regcomp(&er->er_regex, str)) { hts_mutex_unlock(&route_mutex); free(er); duk_error(ctx, DUK_ERR_ERROR, "Invalid regular expression for route %s", str); } er->er_pattern = strdup(str); er->er_prio = strcspn(str, "()[].*?+$") ?: INT32_MAX; LIST_INSERT_SORTED(&routes, er, er_link, er_cmp, es_route_t); es_resource_link(&er->super, ec, 1); hts_mutex_unlock(&route_mutex); es_root_register(ctx, 1, er); es_resource_push(ctx, &er->super); return 1; }