@aerokit/sdk 12.56.0 → 12.58.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/cjs/db/decorators.js +1 -1
- package/dist/cjs/db/repository.js +1 -1
- package/dist/cjs/db/store.js +79 -12
- package/dist/cjs/http/decorators.js +33 -21
- package/dist/cjs/http/rs/resource-http-controller.js +2 -1
- package/dist/dts/db/decorators.d.ts +4 -1
- package/dist/dts/db/store.d.ts +6 -0
- package/dist/dts/http/decorators.d.ts +8 -8
- package/dist/esm/db/decorators.mjs +1 -1
- package/dist/esm/db/repository.mjs +1 -1
- package/dist/esm/db/store.mjs +79 -12
- package/dist/esm/http/decorators.mjs +33 -21
- package/dist/esm/http/rs/resource-http-controller.mjs +2 -1
- package/package.json +1 -1
|
@@ -155,6 +155,7 @@ class HttpController {
|
|
|
155
155
|
}
|
|
156
156
|
} else {
|
|
157
157
|
logger.error("No suitable resource handler for Resource[{}], Method[{}], Content-Type[{}], Accept[{}] found", resourcePath, method.toUpperCase(), contentTypeHeader, acceptsHeader);
|
|
158
|
+
logger.error("Registered resource handlers for Resource[{}] are [{}]", resourcePath, JSON.stringify(_oConfiguration));
|
|
158
159
|
this.sendError(response2.BAD_REQUEST, void 0, "Bad Request", "No suitable processor for this request.");
|
|
159
160
|
}
|
|
160
161
|
}
|
|
@@ -303,4 +304,4 @@ const matchMediaType = function(request2, producesMediaTypes, consumesMediaTypes
|
|
|
303
304
|
}
|
|
304
305
|
return isProduceMatched && isConsumeMatched;
|
|
305
306
|
};
|
|
306
|
-
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../../src/http/rs/resource-http-controller.ts"],
  "sourcesContent": ["import { Response as response } from \"../response\";\nimport { Request as request } from \"../request\";\nimport { ResourceMappings } from \"./resource-mappings\";\nimport { Logging } from \"@aerokit/sdk/log\";\n\n// Declaration for the external dirigibleRequire function and its dependency\ndeclare function dirigibleRequire(module: string): any;\nconst { match } = dirigibleRequire(\"modules/src/http/path-to-regexp/6.2.1/index.js\");\n\nconst logger: any = Logging.getLogger('http.rs.controller');\n\n/**\n * Interface for the context object passed to handler functions (before, serve, catch).\n */\ninterface RequestContext {\n    pathParameters: { [key: string]: any };\n    queryParameters: { [key: string]: any };\n    response: any;\n    res: any;\n    request: any;\n    req: any;\n    // Context properties for error handling might also be attached here\n    suppressStack?: boolean;\n    httpErrorCode?: number;\n    errorMessage?: string;\n    errorName?: string;\n    errorCode?: any;\n}\n\nfunction getRequest(): any {\n    return request;\n}\nfunction getResponse(): any {\n    return response;\n}\n\n/**\n * Creates a service (HttpController) instance, optionally initialized with oMappings.\n *\n * @param oConfig Configuration object or configuration builder with configuration() getter function.\n * @returns A new HttpController instance.\n */\nexport function service(oConfig?: any): HttpController {\n    let config: ResourceMappings | any;\n    if (oConfig !== undefined) {\n        if (typeof oConfig === 'object' || oConfig instanceof ResourceMappings) {\n            config = oConfig;\n        }\n    }\n    return new HttpController(config);\n}\n\n/**\n * The main class for handling HTTP requests and routing them to the correct resource handlers.\n */\nexport class HttpController {\n\n    resource: Function;\n    resourcePath: Function;\n    resourceMappings: ResourceMappings;\n\n    // Index signature to allow dynamic method assignment in the constructor\n    [key: string]: any;\n\n    /**\n     * Constructor function for HttpController instances.\n     *\n     * @param oMappings The mappings configuration for this controller.\n     */\n    constructor(oMappings?: ResourceMappings | any) {\n        if (oMappings instanceof ResourceMappings) {\n            this.resourceMappings = oMappings;\n        } else if (typeof oMappings === 'object' || oMappings === undefined) {\n            this.resourceMappings = new ResourceMappings(oMappings, this);\n        } else {\n            // Default initialization if input is unexpected\n            this.resourceMappings = new ResourceMappings({}, this);\n        }\n\n        // Alias for resourceMappings.resource\n        this.resource = this.resourcePath = this.resourceMappings.resourcePath.bind(this.resourceMappings);\n\n        // weave-in HTTP method-based factory functions - shortcut for service().resource(sPath).method\n        ['get', 'post', 'put', 'delete', 'remove', 'method'].forEach((sMethodName: string) => {\n            this[sMethodName] = (...allArguments: any[]): HttpController => {\n                if (allArguments.length < 1)\n                    throw Error('Insufficient arguments provided to HttpController method ' + sMethodName + '.');\n\n                // sPath is always the first argument\n                let sPath = allArguments[0];\n                if (sPath === undefined)\n                    sPath = \"\";\n\n                // The next arguments (sVerb, arrConsumes, arrProduces) are used by ResourceMappings.find/resource\n                const sVerb = allArguments[1];\n                const arrConsumes = allArguments[2];\n                const arrProduces = allArguments[3];\n\n                // Find existing resource or create a new one based on the path (and other optional constraints)\n                const resource: any = this.resourceMappings.find(sPath, sVerb, arrConsumes, arrProduces) || this.resourceMappings.resource(sPath);\n\n                // Pass all arguments *except* the sPath to the corresponding method of the resource\n                const resourceMethodArgs = allArguments.slice(1);\n                \n                // Execute the method on the resource instance\n                resource[sMethodName].apply(resource, resourceMethodArgs);\n                \n                return this;\n            };\n        });\n    }\n\n    /**\n     * Alias for execute.\n     */\n    listen(request: any, response: any): void {\n        this.execute(request, response);\n    }\n\n    /**\n     * Executes the request handling logic, finding the best matching resource and handler.\n     */\n    execute(request?: any, response?: any): void {\n        request = request || getRequest();\n        const requestPath: string = request.getResourcePath();\n        const method: string = request.getMethod().toLowerCase();\n        const _oConfiguration: any = this.resourceMappings.configuration();\n\n        const matches: any[] = matchRequestUrl(requestPath, method, _oConfiguration);\n        let resourceHandler: any;\n\n        if (matches && matches[0]) {\n            const verbHandlers: any[] = _oConfiguration[matches[0].d][method];\n            if (verbHandlers) {\n                resourceHandler = verbHandlers.filter((handlerDef) => {\n                    return matchMediaType(request, handlerDef.produces, handlerDef.consumes);\n                })[0];\n            }\n        }\n\n        response = response || getResponse();\n        const queryParams: { [key: string]: any } = request.getQueryParametersMap() || {};\n        const acceptsHeader: string[] = normalizeMediaTypeHeaderValue(request.getHeader('Accept')) || [];\n        const contentTypeHeader: string[] = normalizeMediaTypeHeaderValue(request.getHeader('Content-Type')) || [];\n        const resourcePath: string = requestPath;\n\n        if (resourceHandler) {\n            const ctx: RequestContext = {\n                \"pathParameters\": {},\n                \"queryParameters\": {},\n                \"response\": response,\n                \"res\": response,\n                \"request\": request,\n                \"req\": request\n            };\n            if (matches[0].pathParams) {\n                ctx.pathParameters = matches[0].pathParams;\n            }\n            ctx.queryParameters = queryParams;\n\n            const noop = function () { };\n            let _before: Function, _serve: Function, _catch: Function, _finally: Function;\n            _before = resourceHandler.before || noop;\n            _serve = resourceHandler.handler || resourceHandler.serve || noop;\n            _catch = resourceHandler.catch || catchErrorHandler.bind(this, {\n                path: resourcePath,\n                method: method.toUpperCase(),\n                contentType: contentTypeHeader.join(','),\n                accepts: acceptsHeader.join(',')\n            });\n            _finally = resourceHandler.finally || noop;\n\n            const callbackArgs: any[] = [ctx, request, response, resourceHandler, this];\n\n            try {\n                logger.trace('Before serving request for Resource[{}], Method[{}], Content-Type[{}], Accept[{}]', resourcePath, method.toUpperCase(), contentTypeHeader, acceptsHeader);\n                _before.apply(this, callbackArgs);\n\n                if (!response.isCommitted()) {\n                    logger.trace('Serving request for Resource[{}], Method[{}], Content-Type[{}], Accept[{}]', resourcePath, method.toUpperCase(), contentTypeHeader, acceptsHeader);\n                    _serve.apply(this, callbackArgs);\n                    logger.trace('Serving request for Resource[{}], Method[{}], Content-Type[{}], Accept[{}] finished', resourcePath, method.toUpperCase(), contentTypeHeader, acceptsHeader);\n                }\n            } catch (err: any) {\n                try {\n                    callbackArgs.splice(1, 0, err);\n                    _catch.apply(this, callbackArgs);\n                } catch (_catchErr) {\n                    logger.error('Serving request for Resource[{}], Method[{}], Content-Type[{}], Accept[{}] error handler threw error', _catchErr);\n                    throw _catchErr;\n                }\n            } finally {\n                HttpController.prototype.closeResponse.call(this);\n                try {\n                    _finally.apply(this, []);\n                } catch (_finallyErr) {\n                    logger.error('Serving request for Resource[{}], Method[{}], Content-Type[{}], Accept[{}] post handler threw error', _finallyErr);\n                }\n            }\n        } else {\n            logger.error('No suitable resource handler for Resource[{}], Method[{}], Content-Type[{}], Accept[{}] found', resourcePath, method.toUpperCase(), contentTypeHeader, acceptsHeader);\n            this.sendError(response.BAD_REQUEST, undefined, 'Bad Request', 'No suitable processor for this request.');\n        }\n    }\n\n    /**\n     * Returns the ResourceMappings instance of this controller.\n     */\n    mappings(): ResourceMappings {\n        return this.resourceMappings;\n    };\n\n    /**\n     * Sends an error response to the client, formatted based on the accepted media type.\n     */\n    sendError(httpErrorCode: number, applicationErrorCode: any, errorName: string, errorDetails: string): void {\n        const clientAcceptMediaTypes: string[] = normalizeMediaTypeHeaderValue(request.getHeader('Accept')) || ['application/json'];\n        const isHtml: boolean = clientAcceptMediaTypes.some((acceptMediaType) => isMimeTypeCompatible('*/html', acceptMediaType));\n\n        response.setStatus(httpErrorCode || response.INTERNAL_SERVER_ERROR);\n\n        if (isHtml) {\n            const message: string = errorName + (applicationErrorCode !== undefined ? '[' + applicationErrorCode + ']' : '') + (errorDetails ? ': ' + errorDetails : '');\n            response.sendError(httpErrorCode || response.INTERNAL_SERVER_ERROR, message);\n        } else {\n            const body = {\n                \"code\": applicationErrorCode,\n                \"error\": errorName,\n                \"details\": errorDetails\n            };\n            response.setHeader(\"Content-Type\", \"application/json\");\n            response.print(JSON.stringify(body, null, 2));\n        }\n        this.closeResponse();\n    };\n\n    /**\n     * Flushes and closes the HTTP response stream.\n     */\n    closeResponse(): void {\n        response.flush();\n        response.close();\n    };\n}\n\n/**\n * Custom sort function for matched route definitions, preferring exact matches over those with placeholders.\n */\nfunction matchedRouteDefinitionsSorter(p: any, n: any): number {\n    p.w = calculateMatchedRouteWeight(p);\n    n.w = calculateMatchedRouteWeight(n);\n\n    if (n.w === p.w) {\n        // The one with less placeholders wins\n        const m1 = p.d.match(/{(.*?)}/g);\n        const placeholdersCount1 = m1 !== null ? m1.length : 0;\n        const m2 = n.d.match(/{(.*?)}/g);\n        const placeholdersCount2 = m2 !== null ? m2.length : 0;\n        if (placeholdersCount1 > placeholdersCount2) {\n            n.w = n.w + 1;\n        } else if (placeholdersCount1 < placeholdersCount2) {\n            p.w = p.w + 1;\n        }\n    }\n    return n.w - p.w;\n}\n\n/**\n * Calculates the initial weight of a matched route definition.\n */\nfunction calculateMatchedRouteWeight(matchedRoute: any): number {\n    return (matchedRoute.params && Object.keys(matchedRoute.params).length > 0) ? 0 : 1; // always prefer exact route definitions - set weight to 1\n}\n\n/**\n * Transforms path parameters declared in braces (e.g., '/api/{pathParam}') to path-to-regexp format (e.g., '/api/:pathParam').\n */\nfunction transformPathParamsDeclaredInBraces(pathDefinition: string): string {\n    const pathParamsInBracesMatcher = /({(\\w*\\*?)})/g; // matches cases like '/api/{pathParam}' or '/api/{pathParam*}'\n    return pathDefinition.replace(pathParamsInBracesMatcher, \":$2\"); // transforms matched cases to '/api/:pathParam' or '/api/:pathParam*'\n}\n\n/**\n * Finds all routes in the configuration that match the request path and method.\n */\nfunction matchRequestUrl(requestPath: string, method: string, cfg: any): any[] {\n    return Object.entries(cfg)\n        .filter(([_, handlers]) => handlers && (handlers as any)[method])\n        .map(([path, _]) => path)\n        .reduce((matches: any[], path: string) => matchingRouteDefinitionsReducer(matches, path, requestPath), [])\n        .sort(matchedRouteDefinitionsSorter);\n}\n\n/**\n * Reducer function to attempt matching a defined path against the request path using path-to-regexp.\n */\nfunction matchingRouteDefinitionsReducer(matchedDefinitions: any[], definedPath: string, requestPath: string): any[] {\n    // 'match' from path-to-regexp is used here\n    const pathMatcher = match(transformPathParamsDeclaredInBraces(definedPath));\n    const matched = pathMatcher(requestPath);\n\n    if (matched) {\n        // Ensure pathParams is an object of key/value pairs\n        const pathParams = matched.params;\n\n        const matchedDefinition = {\n            p: requestPath,\n            d: definedPath,\n            pathParams: pathParams\n        };\n        matchedDefinitions.push(matchedDefinition);\n    }\n    return matchedDefinitions;\n}\n\n/**\n * Normalizes an HTTP media type header value (e.g., \"text/plain; q=0.9, application/json\").\n */\nfunction normalizeMediaTypeHeaderValue(sMediaType: string | undefined | null): string[] | undefined {\n    if (sMediaType === undefined || sMediaType === null)\n        return;\n    // convert to array of individual types\n    let arrMediaType = sMediaType.split(',');\n    arrMediaType = arrMediaType.map((mimeTypeEntry) => {\n        // remove escaping, remove quality or other attributes (e.g., '; q=0.9')\n        return mimeTypeEntry.replace('\\\\', '').split(';')[0].trim();\n    });\n    return arrMediaType.filter(type => type.length > 0);\n}\n\n/**\n * Checks if a source MIME type is compatible with a target MIME type, supporting wildcards (*).\n */\nfunction isMimeTypeCompatible(source: string, target: string): boolean {\n    if (source === target)\n        return true;\n\n    const targetM = target.split('/');\n    const sourceM = source.split('/');\n\n    // Target is wildcard type, Source has a specific subtype (e.g., target=*/json, source=application/json)\n    if (targetM[0] === '*' && targetM[1] === sourceM[1])\n        return true;\n    // Source is wildcard type, Target has a specific subtype (e.g., source=*/json, target=application/json)\n    if (sourceM[0] === '*' && targetM[1] === sourceM[1])\n        return true;\n\n    // Target is wildcard subtype, Source has a specific type (e.g., target=application/*, source=application/json)\n    if (targetM[1] === '*' && targetM[0] === sourceM[0])\n        return true;\n    // Source is wildcard subtype, Target has a specific type (e.g., source=application/*, target=application/json)\n    if (sourceM[1] === '*' && targetM[0] === sourceM[0])\n        return true;\n\n    return false;\n}\n\n/**\n * Default error handler function.\n */\nconst catchErrorHandler = function (this: HttpController, logctx: any, ctx: RequestContext, err: any, request: any, response: any): void {\n    if (ctx.suppressStack) {\n        const detailsMsg = (ctx.errorName || \"\") + (ctx.errorCode ? \" [\" + ctx.errorCode + \"]\" : \"\") + (ctx.errorMessage ? \": \" + ctx.errorMessage : \"\");\n        logger.info('Serving resource[{}], Verb[{}], Content-Type[{}], Accept[{}] finished in error. {}', logctx.path, logctx.method, logctx.contentType, logctx.accepts, detailsMsg);\n    } else {\n        logger.error('Serving resource[{}], Verb[{}], Content-Type[{}], Accept[{}] finished in error', logctx.path, logctx.method, logctx.contentType, logctx.accepts, err);\n    }\n\n    const httpErrorCode = ctx.httpErrorCode || response.INTERNAL_SERVER_ERROR;\n    const errorMessage = ctx.errorMessage || (err && err.message);\n    const errorName = ctx.errorName || (err && err.name);\n    const errorCode = ctx.errorCode;\n\n    this.sendError(httpErrorCode, errorCode, errorName, errorMessage);\n};\n\n/**\n * Checks if the request media types match the resource handler's consumes and produces constraints.\n */\nconst matchMediaType = function (request: any, producesMediaTypes: string[], consumesMediaTypes: string[]): boolean {\n    let isProduceMatched = false;\n    const acceptsMediaTypes = normalizeMediaTypeHeaderValue(request.getHeader('Accept'));\n\n    // 1. Check Produces (Accepts header)\n    if (!acceptsMediaTypes || acceptsMediaTypes.length === 0 || acceptsMediaTypes.includes('*/*')) {\n        // Output media type is not restricted by client or client accepts anything\n        isProduceMatched = true;\n    } else {\n        if (producesMediaTypes && producesMediaTypes.length) {\n            const matchedProducesMIME = acceptsMediaTypes.filter((acceptsMediaType) => {\n                return producesMediaTypes.some((producesMediaType) => {\n                    return isMimeTypeCompatible(acceptsMediaType, producesMediaType);\n                });\n            });\n            isProduceMatched = matchedProducesMIME && matchedProducesMIME.length > 0;\n        } else {\n            // Resource doesn't specify produces, so it matches if the client accepts anything or if produces is an empty array\n            isProduceMatched = true;\n        }\n    }\n\n    // 2. Check Consumes (Content-Type header)\n    let isConsumeMatched = false;\n    const contentTypeMediaTypes = normalizeMediaTypeHeaderValue(request.getContentType());\n\n    if (!consumesMediaTypes || consumesMediaTypes.length === 0 || consumesMediaTypes.includes('*/*')) {\n        // Input media type is not restricted by resource or resource accepts anything\n        isConsumeMatched = true;\n    } else {\n        if (contentTypeMediaTypes && consumesMediaTypes && consumesMediaTypes.length) {\n            const matchedConsumesMIME = contentTypeMediaTypes.filter((contentTypeMediaType) => {\n                return consumesMediaTypes.some((consumesMediaType) => {\n                    return isMimeTypeCompatible(contentTypeMediaType, consumesMediaType);\n                });\n            });\n            isConsumeMatched = matchedConsumesMIME && matchedConsumesMIME.length > 0;\n        }\n    }\n\n    return isProduceMatched && isConsumeMatched;\n};\n"],
  "mappings": "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",
  "names": ["request", "response"]
}

|
|
307
|
+
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../../src/http/rs/resource-http-controller.ts"],
  "sourcesContent": ["import { Response as response } from \"../response\";\nimport { Request as request } from \"../request\";\nimport { ResourceMappings } from \"./resource-mappings\";\nimport { Logging } from \"@aerokit/sdk/log\";\n\n// Declaration for the external dirigibleRequire function and its dependency\ndeclare function dirigibleRequire(module: string): any;\nconst { match } = dirigibleRequire(\"modules/src/http/path-to-regexp/6.2.1/index.js\");\n\nconst logger: any = Logging.getLogger('http.rs.controller');\n\n/**\n * Interface for the context object passed to handler functions (before, serve, catch).\n */\ninterface RequestContext {\n    pathParameters: { [key: string]: any };\n    queryParameters: { [key: string]: any };\n    response: any;\n    res: any;\n    request: any;\n    req: any;\n    // Context properties for error handling might also be attached here\n    suppressStack?: boolean;\n    httpErrorCode?: number;\n    errorMessage?: string;\n    errorName?: string;\n    errorCode?: any;\n}\n\nfunction getRequest(): any {\n    return request;\n}\nfunction getResponse(): any {\n    return response;\n}\n\n/**\n * Creates a service (HttpController) instance, optionally initialized with oMappings.\n *\n * @param oConfig Configuration object or configuration builder with configuration() getter function.\n * @returns A new HttpController instance.\n */\nexport function service(oConfig?: any): HttpController {\n    let config: ResourceMappings | any;\n    if (oConfig !== undefined) {\n        if (typeof oConfig === 'object' || oConfig instanceof ResourceMappings) {\n            config = oConfig;\n        }\n    }\n    return new HttpController(config);\n}\n\n/**\n * The main class for handling HTTP requests and routing them to the correct resource handlers.\n */\nexport class HttpController {\n\n    resource: Function;\n    resourcePath: Function;\n    resourceMappings: ResourceMappings;\n\n    // Index signature to allow dynamic method assignment in the constructor\n    [key: string]: any;\n\n    /**\n     * Constructor function for HttpController instances.\n     *\n     * @param oMappings The mappings configuration for this controller.\n     */\n    constructor(oMappings?: ResourceMappings | any) {\n        if (oMappings instanceof ResourceMappings) {\n            this.resourceMappings = oMappings;\n        } else if (typeof oMappings === 'object' || oMappings === undefined) {\n            this.resourceMappings = new ResourceMappings(oMappings, this);\n        } else {\n            // Default initialization if input is unexpected\n            this.resourceMappings = new ResourceMappings({}, this);\n        }\n\n        // Alias for resourceMappings.resource\n        this.resource = this.resourcePath = this.resourceMappings.resourcePath.bind(this.resourceMappings);\n\n        // weave-in HTTP method-based factory functions - shortcut for service().resource(sPath).method\n        ['get', 'post', 'put', 'delete', 'remove', 'method'].forEach((sMethodName: string) => {\n            this[sMethodName] = (...allArguments: any[]): HttpController => {\n                if (allArguments.length < 1)\n                    throw Error('Insufficient arguments provided to HttpController method ' + sMethodName + '.');\n\n                // sPath is always the first argument\n                let sPath = allArguments[0];\n                if (sPath === undefined)\n                    sPath = \"\";\n\n                // The next arguments (sVerb, arrConsumes, arrProduces) are used by ResourceMappings.find/resource\n                const sVerb = allArguments[1];\n                const arrConsumes = allArguments[2];\n                const arrProduces = allArguments[3];\n\n                // Find existing resource or create a new one based on the path (and other optional constraints)\n                const resource: any = this.resourceMappings.find(sPath, sVerb, arrConsumes, arrProduces) || this.resourceMappings.resource(sPath);\n\n                // Pass all arguments *except* the sPath to the corresponding method of the resource\n                const resourceMethodArgs = allArguments.slice(1);\n                \n                // Execute the method on the resource instance\n                resource[sMethodName].apply(resource, resourceMethodArgs);\n                \n                return this;\n            };\n        });\n    }\n\n    /**\n     * Alias for execute.\n     */\n    listen(request: any, response: any): void {\n        this.execute(request, response);\n    }\n\n    /**\n     * Executes the request handling logic, finding the best matching resource and handler.\n     */\n    execute(request?: any, response?: any): void {\n        request = request || getRequest();\n        const requestPath: string = request.getResourcePath();\n        const method: string = request.getMethod().toLowerCase();\n        const _oConfiguration: any = this.resourceMappings.configuration();\n\n        const matches: any[] = matchRequestUrl(requestPath, method, _oConfiguration);\n        let resourceHandler: any;\n\n        if (matches && matches[0]) {\n            const verbHandlers: any[] = _oConfiguration[matches[0].d][method];\n            if (verbHandlers) {\n                resourceHandler = verbHandlers.filter((handlerDef) => {\n                    return matchMediaType(request, handlerDef.produces, handlerDef.consumes);\n                })[0];\n            }\n        }\n\n        response = response || getResponse();\n        const queryParams: { [key: string]: any } = request.getQueryParametersMap() || {};\n        const acceptsHeader: string[] = normalizeMediaTypeHeaderValue(request.getHeader('Accept')) || [];\n        const contentTypeHeader: string[] = normalizeMediaTypeHeaderValue(request.getHeader('Content-Type')) || [];\n        const resourcePath: string = requestPath;\n\n        if (resourceHandler) {\n            const ctx: RequestContext = {\n                \"pathParameters\": {},\n                \"queryParameters\": {},\n                \"response\": response,\n                \"res\": response,\n                \"request\": request,\n                \"req\": request\n            };\n            if (matches[0].pathParams) {\n                ctx.pathParameters = matches[0].pathParams;\n            }\n            ctx.queryParameters = queryParams;\n\n            const noop = function () { };\n            let _before: Function, _serve: Function, _catch: Function, _finally: Function;\n            _before = resourceHandler.before || noop;\n            _serve = resourceHandler.handler || resourceHandler.serve || noop;\n            _catch = resourceHandler.catch || catchErrorHandler.bind(this, {\n                path: resourcePath,\n                method: method.toUpperCase(),\n                contentType: contentTypeHeader.join(','),\n                accepts: acceptsHeader.join(',')\n            });\n            _finally = resourceHandler.finally || noop;\n\n            const callbackArgs: any[] = [ctx, request, response, resourceHandler, this];\n\n            try {\n                logger.trace('Before serving request for Resource[{}], Method[{}], Content-Type[{}], Accept[{}]', resourcePath, method.toUpperCase(), contentTypeHeader, acceptsHeader);\n                _before.apply(this, callbackArgs);\n\n                if (!response.isCommitted()) {\n                    logger.trace('Serving request for Resource[{}], Method[{}], Content-Type[{}], Accept[{}]', resourcePath, method.toUpperCase(), contentTypeHeader, acceptsHeader);\n                    _serve.apply(this, callbackArgs);\n                    logger.trace('Serving request for Resource[{}], Method[{}], Content-Type[{}], Accept[{}] finished', resourcePath, method.toUpperCase(), contentTypeHeader, acceptsHeader);\n                }\n            } catch (err: any) {\n                try {\n                    callbackArgs.splice(1, 0, err);\n                    _catch.apply(this, callbackArgs);\n                } catch (_catchErr) {\n                    logger.error('Serving request for Resource[{}], Method[{}], Content-Type[{}], Accept[{}] error handler threw error', _catchErr);\n                    throw _catchErr;\n                }\n            } finally {\n                HttpController.prototype.closeResponse.call(this);\n                try {\n                    _finally.apply(this, []);\n                } catch (_finallyErr) {\n                    logger.error('Serving request for Resource[{}], Method[{}], Content-Type[{}], Accept[{}] post handler threw error', _finallyErr);\n                }\n            }\n        } else {\n            logger.error('No suitable resource handler for Resource[{}], Method[{}], Content-Type[{}], Accept[{}] found', resourcePath, method.toUpperCase(), contentTypeHeader, acceptsHeader);\n\t\t\tlogger.error('Registered resource handlers for Resource[{}] are [{}]', resourcePath, JSON.stringify(_oConfiguration));\n            this.sendError(response.BAD_REQUEST, undefined, 'Bad Request', 'No suitable processor for this request.');\n        }\n    }\n\n    /**\n     * Returns the ResourceMappings instance of this controller.\n     */\n    mappings(): ResourceMappings {\n        return this.resourceMappings;\n    };\n\n    /**\n     * Sends an error response to the client, formatted based on the accepted media type.\n     */\n    sendError(httpErrorCode: number, applicationErrorCode: any, errorName: string, errorDetails: string): void {\n        const clientAcceptMediaTypes: string[] = normalizeMediaTypeHeaderValue(request.getHeader('Accept')) || ['application/json'];\n        const isHtml: boolean = clientAcceptMediaTypes.some((acceptMediaType) => isMimeTypeCompatible('*/html', acceptMediaType));\n\n        response.setStatus(httpErrorCode || response.INTERNAL_SERVER_ERROR);\n\n        if (isHtml) {\n            const message: string = errorName + (applicationErrorCode !== undefined ? '[' + applicationErrorCode + ']' : '') + (errorDetails ? ': ' + errorDetails : '');\n            response.sendError(httpErrorCode || response.INTERNAL_SERVER_ERROR, message);\n        } else {\n            const body = {\n                \"code\": applicationErrorCode,\n                \"error\": errorName,\n                \"details\": errorDetails\n            };\n            response.setHeader(\"Content-Type\", \"application/json\");\n            response.print(JSON.stringify(body, null, 2));\n        }\n        this.closeResponse();\n    };\n\n    /**\n     * Flushes and closes the HTTP response stream.\n     */\n    closeResponse(): void {\n        response.flush();\n        response.close();\n    };\n}\n\n/**\n * Custom sort function for matched route definitions, preferring exact matches over those with placeholders.\n */\nfunction matchedRouteDefinitionsSorter(p: any, n: any): number {\n    p.w = calculateMatchedRouteWeight(p);\n    n.w = calculateMatchedRouteWeight(n);\n\n    if (n.w === p.w) {\n        // The one with less placeholders wins\n        const m1 = p.d.match(/{(.*?)}/g);\n        const placeholdersCount1 = m1 !== null ? m1.length : 0;\n        const m2 = n.d.match(/{(.*?)}/g);\n        const placeholdersCount2 = m2 !== null ? m2.length : 0;\n        if (placeholdersCount1 > placeholdersCount2) {\n            n.w = n.w + 1;\n        } else if (placeholdersCount1 < placeholdersCount2) {\n            p.w = p.w + 1;\n        }\n    }\n    return n.w - p.w;\n}\n\n/**\n * Calculates the initial weight of a matched route definition.\n */\nfunction calculateMatchedRouteWeight(matchedRoute: any): number {\n    return (matchedRoute.params && Object.keys(matchedRoute.params).length > 0) ? 0 : 1; // always prefer exact route definitions - set weight to 1\n}\n\n/**\n * Transforms path parameters declared in braces (e.g., '/api/{pathParam}') to path-to-regexp format (e.g., '/api/:pathParam').\n */\nfunction transformPathParamsDeclaredInBraces(pathDefinition: string): string {\n    const pathParamsInBracesMatcher = /({(\\w*\\*?)})/g; // matches cases like '/api/{pathParam}' or '/api/{pathParam*}'\n    return pathDefinition.replace(pathParamsInBracesMatcher, \":$2\"); // transforms matched cases to '/api/:pathParam' or '/api/:pathParam*'\n}\n\n/**\n * Finds all routes in the configuration that match the request path and method.\n */\nfunction matchRequestUrl(requestPath: string, method: string, cfg: any): any[] {\n    return Object.entries(cfg)\n        .filter(([_, handlers]) => handlers && (handlers as any)[method])\n        .map(([path, _]) => path)\n        .reduce((matches: any[], path: string) => matchingRouteDefinitionsReducer(matches, path, requestPath), [])\n        .sort(matchedRouteDefinitionsSorter);\n}\n\n/**\n * Reducer function to attempt matching a defined path against the request path using path-to-regexp.\n */\nfunction matchingRouteDefinitionsReducer(matchedDefinitions: any[], definedPath: string, requestPath: string): any[] {\n    // 'match' from path-to-regexp is used here\n    const pathMatcher = match(transformPathParamsDeclaredInBraces(definedPath));\n    const matched = pathMatcher(requestPath);\n\n    if (matched) {\n        // Ensure pathParams is an object of key/value pairs\n        const pathParams = matched.params;\n\n        const matchedDefinition = {\n            p: requestPath,\n            d: definedPath,\n            pathParams: pathParams\n        };\n        matchedDefinitions.push(matchedDefinition);\n    }\n    return matchedDefinitions;\n}\n\n/**\n * Normalizes an HTTP media type header value (e.g., \"text/plain; q=0.9, application/json\").\n */\nfunction normalizeMediaTypeHeaderValue(sMediaType: string | undefined | null): string[] | undefined {\n    if (sMediaType === undefined || sMediaType === null)\n        return;\n    // convert to array of individual types\n    let arrMediaType = sMediaType.split(',');\n    arrMediaType = arrMediaType.map((mimeTypeEntry) => {\n        // remove escaping, remove quality or other attributes (e.g., '; q=0.9')\n        return mimeTypeEntry.replace('\\\\', '').split(';')[0].trim();\n    });\n    return arrMediaType.filter(type => type.length > 0);\n}\n\n/**\n * Checks if a source MIME type is compatible with a target MIME type, supporting wildcards (*).\n */\nfunction isMimeTypeCompatible(source: string, target: string): boolean {\n    if (source === target)\n        return true;\n\n    const targetM = target.split('/');\n    const sourceM = source.split('/');\n\n    // Target is wildcard type, Source has a specific subtype (e.g., target=*/json, source=application/json)\n    if (targetM[0] === '*' && targetM[1] === sourceM[1])\n        return true;\n    // Source is wildcard type, Target has a specific subtype (e.g., source=*/json, target=application/json)\n    if (sourceM[0] === '*' && targetM[1] === sourceM[1])\n        return true;\n\n    // Target is wildcard subtype, Source has a specific type (e.g., target=application/*, source=application/json)\n    if (targetM[1] === '*' && targetM[0] === sourceM[0])\n        return true;\n    // Source is wildcard subtype, Target has a specific type (e.g., source=application/*, target=application/json)\n    if (sourceM[1] === '*' && targetM[0] === sourceM[0])\n        return true;\n\n    return false;\n}\n\n/**\n * Default error handler function.\n */\nconst catchErrorHandler = function (this: HttpController, logctx: any, ctx: RequestContext, err: any, request: any, response: any): void {\n    if (ctx.suppressStack) {\n        const detailsMsg = (ctx.errorName || \"\") + (ctx.errorCode ? \" [\" + ctx.errorCode + \"]\" : \"\") + (ctx.errorMessage ? \": \" + ctx.errorMessage : \"\");\n        logger.info('Serving resource[{}], Verb[{}], Content-Type[{}], Accept[{}] finished in error. {}', logctx.path, logctx.method, logctx.contentType, logctx.accepts, detailsMsg);\n    } else {\n        logger.error('Serving resource[{}], Verb[{}], Content-Type[{}], Accept[{}] finished in error', logctx.path, logctx.method, logctx.contentType, logctx.accepts, err);\n    }\n\n    const httpErrorCode = ctx.httpErrorCode || response.INTERNAL_SERVER_ERROR;\n    const errorMessage = ctx.errorMessage || (err && err.message);\n    const errorName = ctx.errorName || (err && err.name);\n    const errorCode = ctx.errorCode;\n\n    this.sendError(httpErrorCode, errorCode, errorName, errorMessage);\n};\n\n/**\n * Checks if the request media types match the resource handler's consumes and produces constraints.\n */\nconst matchMediaType = function (request: any, producesMediaTypes: string[], consumesMediaTypes: string[]): boolean {\n    let isProduceMatched = false;\n    const acceptsMediaTypes = normalizeMediaTypeHeaderValue(request.getHeader('Accept'));\n\n    // 1. Check Produces (Accepts header)\n    if (!acceptsMediaTypes || acceptsMediaTypes.length === 0 || acceptsMediaTypes.includes('*/*')) {\n        // Output media type is not restricted by client or client accepts anything\n        isProduceMatched = true;\n    } else {\n        if (producesMediaTypes && producesMediaTypes.length) {\n            const matchedProducesMIME = acceptsMediaTypes.filter((acceptsMediaType) => {\n                return producesMediaTypes.some((producesMediaType) => {\n                    return isMimeTypeCompatible(acceptsMediaType, producesMediaType);\n                });\n            });\n            isProduceMatched = matchedProducesMIME && matchedProducesMIME.length > 0;\n        } else {\n            // Resource doesn't specify produces, so it matches if the client accepts anything or if produces is an empty array\n            isProduceMatched = true;\n        }\n    }\n\n    // 2. Check Consumes (Content-Type header)\n    let isConsumeMatched = false;\n    const contentTypeMediaTypes = normalizeMediaTypeHeaderValue(request.getContentType());\n\n    if (!consumesMediaTypes || consumesMediaTypes.length === 0 || consumesMediaTypes.includes('*/*')) {\n        // Input media type is not restricted by resource or resource accepts anything\n        isConsumeMatched = true;\n    } else {\n        if (contentTypeMediaTypes && consumesMediaTypes && consumesMediaTypes.length) {\n            const matchedConsumesMIME = contentTypeMediaTypes.filter((contentTypeMediaType) => {\n                return consumesMediaTypes.some((consumesMediaType) => {\n                    return isMimeTypeCompatible(contentTypeMediaType, consumesMediaType);\n                });\n            });\n            isConsumeMatched = matchedConsumesMIME && matchedConsumesMIME.length > 0;\n        }\n    }\n\n    return isProduceMatched && isConsumeMatched;\n};\n"],
  "mappings": "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",
  "names": ["request", "response"]
}

|
|
@@ -19,12 +19,15 @@ type ClassDecoratorContext = {
|
|
|
19
19
|
name?: string | symbol;
|
|
20
20
|
addInitializer(fn: () => void): void;
|
|
21
21
|
};
|
|
22
|
+
export type ColumnTypes = 'integer' | 'long' | 'short' | 'byte' | 'float' | 'double' | 'big_integer' | 'big_decimal' | 'string' | 'char' | 'text' | 'nstring' | 'ntext' | 'date' | 'time' | 'timestamp' | 'calendar' | 'calendar_date' | 'instant' | 'boolean' | 'true_false' | 'yes_no' | 'numeric_boolean' | 'binary' | 'blob' | 'clob' | 'materialized_blob' | 'materialized_clob' | 'serializable' | 'any' | 'object' | 'uuid-char' | 'uuid-binary' | 'json' | 'jsonb' | 'xml';
|
|
22
23
|
export interface ColumnOptions {
|
|
23
24
|
name?: string;
|
|
24
|
-
type?: string;
|
|
25
|
+
type?: ColumnTypes | (string & {});
|
|
25
26
|
length?: number;
|
|
26
27
|
nullable?: boolean;
|
|
27
28
|
defaultValue?: string;
|
|
29
|
+
precision?: number;
|
|
30
|
+
scale?: number;
|
|
28
31
|
}
|
|
29
32
|
export interface OneToManyOptions {
|
|
30
33
|
table?: string;
|
package/dist/dts/db/store.d.ts
CHANGED
|
@@ -136,4 +136,10 @@ export declare class Store {
|
|
|
136
136
|
* Gets the underlying database column name used for the ID field.
|
|
137
137
|
*/
|
|
138
138
|
static getIdColumn(name: string): string;
|
|
139
|
+
/**
|
|
140
|
+
* Parse a JSON string and revive ISO date strings into JS Date objects.
|
|
141
|
+
* It handles both full ISO timestamps (with timezone) and date-only strings (YYYY-MM-DD).
|
|
142
|
+
* Returns undefined for null/empty inputs.
|
|
143
|
+
*/
|
|
144
|
+
private static parseResult;
|
|
139
145
|
}
|
|
@@ -1,11 +1,11 @@
|
|
|
1
1
|
export declare function Controller(ctr: {
|
|
2
2
|
new (): any;
|
|
3
|
-
}
|
|
3
|
+
}): void;
|
|
4
4
|
export declare function Documentation(documentation: string): (value: any, context: ClassDecoratorContext | ClassFieldDecoratorContext | ClassMethodDecoratorContext) => void;
|
|
5
|
-
export declare const Get: (path: string
|
|
6
|
-
export declare const Post: (path: string
|
|
7
|
-
export declare const Put: (path: string
|
|
8
|
-
export declare const Patch: (path: string
|
|
9
|
-
export declare const Delete: (path: string
|
|
10
|
-
export declare const Head: (path: string
|
|
11
|
-
export declare const Options: (path: string
|
|
5
|
+
export declare const Get: (path: string) => (target: any, propertyKey: string) => void;
|
|
6
|
+
export declare const Post: (path: string) => (target: any, propertyKey: string) => void;
|
|
7
|
+
export declare const Put: (path: string) => (target: any, propertyKey: string) => void;
|
|
8
|
+
export declare const Patch: (path: string) => (target: any, propertyKey: string) => void;
|
|
9
|
+
export declare const Delete: (path: string) => (target: any, propertyKey: string) => void;
|
|
10
|
+
export declare const Head: (path: string) => (target: any, propertyKey: string) => void;
|
|
11
|
+
export declare const Options: (path: string) => (target: any, propertyKey: string) => void;
|
|
@@ -145,4 +145,4 @@ export {
|
|
|
145
145
|
OneToMany,
|
|
146
146
|
Table
|
|
147
147
|
};
|
|
148
|
-
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../src/db/decorators.ts"],
  "sourcesContent": ["/**\n * * ECMAScript 2025-compliant ORM decorator implementation\n * Compatible with GraalJS runtime.\n * * Features:\n * - Uses context.addInitializer for stable decorator timing\n * - Stores metadata in a global WeakMap cache\n * - Finalizes entity metadata once per class\n * - Defers finalization via microtask (Promise.resolve().then)\n */\n\n// --- Decorator Context Types (based on ECMAScript Decorators proposal) ---\ntype ClassFieldDecoratorContext = {\n  kind: \"field\";\n  name: string | symbol;\n  static: boolean;\n  private: boolean;\n  addInitializer(fn: () => void): void;\n};\n\ntype ClassDecoratorContext = {\n  kind: \"class\";\n  name?: string | symbol;\n  addInitializer(fn: () => void): void;\n};\n\n// --- Metadata Models ---\nexport interface ColumnOptions {\n  name?: string;\n  type?: string;\n  length?: number;\n  nullable?: boolean;\n  defaultValue?: string;\n}\n\nexport interface OneToManyOptions {\n  table?: string;\n  joinColumn: string;\n  cascade?: \"all\" | \"none\" | \"persist\" | \"merge\" | \"remove\";\n  inverse?: boolean;\n  lazy?: boolean;\n  fetch?: \"select\" | \"join\";\n  joinColumnNotNull?: boolean;\n}\n\nexport interface ManyToOneOptions {\n  joinColumn?: string;\n  cascade?: \"all\" | \"none\" | \"persist\" | \"merge\" | \"remove\";\n  nullable?: boolean;\n  lazy?: boolean;\n  fetch?: \"select\" | \"join\";\n}\n\ninterface PropertyMetadata {\n  propertyName: string;\n  isId: boolean;\n  isGenerated: boolean;\n  documentation?: string;\n  columnOptions?: ColumnOptions;\n  oneToManyOptions?: { type: Function; options: OneToManyOptions };\n  manyToOneOptions?: { type: Function; options: ManyToOneOptions };\n}\n\nexport interface EntityConstructor extends Function {\n  new (...args: any[]): any;\n  $entity_name: string;\n  $table_name: string;\n  $id_name: string;\n  $id_column: string;\n  $initialized?: boolean;\n  $documentation?: string;\n}\n\n// --- Global Metadata Cache ---\nconst globalCache: WeakMap<Function, PropertyMetadata[]> =\n  (globalThis as any).__decorator_metadata_cache__ ||\n  ((globalThis as any).__decorator_metadata_cache__ = new WeakMap());\n\nfunction getMetadataArray(constructor: Function): PropertyMetadata[] {\n  if (!globalCache.has(constructor)) {\n    globalCache.set(constructor, []);\n  }\n  return globalCache.get(constructor)!;\n}\n\n// --- Defer Helper (GraalJS-safe microtask) ---\nfunction defer(fn: () => void): void {\n  // Uses Promise microtask queue for deferred execution\n  Promise.resolve().then(fn);\n}\n\n// --- Core Property Decorator Factory ---\nfunction createPropertyDecorator(\n  kind: \"column\" | \"id\" | \"generated\",\n  options?: ColumnOptions\n) {\n  return function (_: any, context: ClassFieldDecoratorContext) {\n    if (context.kind !== \"field\") {\n      throw new Error(`@${kind} must apply to fields`);\n    }\n\n    context.addInitializer(function () {\n      const ctor = (this as any).constructor as EntityConstructor;\n      const propertyName = context.name.toString();\n      const metadataArray = getMetadataArray(ctor);\n\n      let metadata = metadataArray.find(\n        (m) => m.propertyName === propertyName\n      );\n\n      if (!metadata) {\n        metadata = {\n          propertyName: propertyName,\n          isId: false,\n          isGenerated: false,\n        };\n        metadataArray.push(metadata);\n      }\n\n      if (kind === \"id\") metadata.isId = true;\n      if (kind === \"generated\") metadata.isGenerated = true;\n      if (kind === \"column\") metadata.columnOptions = options;\n    });\n  };\n}\n\n// --- @Documentation Decorator (Dual-Purpose) ---\n/**\n * Adds documentation metadata to a class or a field.\n */\nexport function Documentation(description: string) {\n  return function (\n    value: Function | any,\n    context: ClassDecoratorContext | ClassFieldDecoratorContext\n  ) {\n    if (context.kind === \"class\") {\n      context.addInitializer(function () {\n        (value as EntityConstructor).$documentation = description;\n      });\n      return value;\n    } else if (context.kind === \"field\") {\n      context.addInitializer(function () {\n        const ctor = (this as any).constructor as EntityConstructor;\n        const propertyName = context.name.toString();\n        const metadataArray = getMetadataArray(ctor);\n\n        let metadata = metadataArray.find(\n          (m) => m.propertyName === propertyName\n        );\n\n        if (!metadata) {\n          metadata = {\n            propertyName: propertyName,\n            isId: false,\n            isGenerated: false,\n          };\n          metadataArray.push(metadata);\n        }\n\n        metadata.documentation = description;\n      });\n    }\n  };\n}\n\n\n// --- @Entity Decorator ---\n/**\n * Marks a class as an entity and initiates metadata finalization.\n * @param entityName The name of the entity (defaults to class name).\n */\nexport function Entity(entityName?: string) {\n  return function (value: Function, context: ClassDecoratorContext) {\n    context.addInitializer(function () {\n      // Defer execution to ensure all field decorators have run\n      defer(() => {\n        const ctor = value as EntityConstructor;\n\n        // Prevent duplicate registration (idempotency)\n        if (ctor.$initialized) return;\n        ctor.$initialized = true;\n\n        ctor.$entity_name = entityName || ctor.name;\n\n        const metadataArray = getMetadataArray(ctor);\n        const idMetadata = metadataArray.find((m) => m.isId);\n\n        if (idMetadata) {\n          ctor.$id_name = idMetadata.propertyName;\n          // Determine ID column name: use explicit name or convert property name to upper case\n          ctor.$id_column =\n            idMetadata.columnOptions?.name ||\n            idMetadata.propertyName.toUpperCase();\n        }\n\n        // Future: Logic to map all properties to columns/relations goes here\n      });\n    });\n  };\n}\n\n// --- @Table Decorator ---\n/**\n * Specifies the database table name for the entity.\n * @param tableName The table name (defaults to uppercase class name).\n */\nexport function Table(tableName?: string) {\n  return function <T extends EntityConstructor>(\n    value: T,\n    context: ClassDecoratorContext\n  ) {\n    context.addInitializer(function () {\n      (value as EntityConstructor).$table_name =\n        tableName || (context.name?.toString() ?? value.name.toUpperCase());\n    });\n\n    return value;\n  };\n}\n\n// --- Exported Property Decorators ---\n/**\n * Marks a property as a standard database column.\n */\nexport const Column = (options?: ColumnOptions) =>\n  createPropertyDecorator(\"column\", options);\n\n/**\n * Marks a property as the entity's primary key.\n */\nexport const Id = () => createPropertyDecorator(\"id\");\n\n/**\n * Marks a property as a generated value (e.g., auto-increment).\n * @param strategy The generation strategy (e.g., \"IDENTITY\"). Parameter is currently unused in logic.\n */\nexport const Generated = (strategy: string) =>\n  createPropertyDecorator(\"generated\");\n\n/**\n * Defines a one-to-many relationship.\n */\nexport function OneToMany(\n  typeFunction: () => Function,\n  options: OneToManyOptions\n) {\n  return function (_: any, context: ClassFieldDecoratorContext) {\n    if (context.kind !== \"field\") return;\n\n    context.addInitializer(function () {\n      const ctor = (this as any).constructor as EntityConstructor;\n      const propertyName = context.name.toString();\n      const metadataArray = getMetadataArray(ctor);\n\n      let metadata = metadataArray.find(\n        (m) => m.propertyName === propertyName\n      );\n\n      if (!metadata) {\n        metadata = {\n          propertyName: propertyName,\n          isId: false,\n          isGenerated: false,\n        };\n        metadataArray.push(metadata);\n      }\n\n      metadata.oneToManyOptions = { type: typeFunction(), options };\n    });\n  };\n}\n\n/**\n * Defines a many-to-one relationship.\n */\nexport function ManyToOne(\n  typeFunction: () => Function,\n  options: ManyToOneOptions = {}\n) {\n  return function (_: any, context: ClassFieldDecoratorContext) {\n    if (context.kind !== \"field\") return;\n\n    context.addInitializer(function () {\n      const ctor = (this as any).constructor as EntityConstructor;\n      const propertyName = context.name.toString();\n      const metadataArray = getMetadataArray(ctor);\n\n      let metadata = metadataArray.find(\n        (m) => m.propertyName === propertyName\n      );\n\n      if (!metadata) {\n        metadata = {\n          propertyName: propertyName,\n          isId: false,\n          isGenerated: false,\n        };\n        metadataArray.push(metadata);\n      }\n\n      metadata.manyToOneOptions = { type: typeFunction(), options };\n    });\n  };\n}"],
  "mappings": "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",
  "names": []
}

|
|
148
|
+
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../src/db/decorators.ts"],
  "sourcesContent": ["/**\n * * ECMAScript 2025-compliant ORM decorator implementation\n * Compatible with GraalJS runtime.\n * * Features:\n * - Uses context.addInitializer for stable decorator timing\n * - Stores metadata in a global WeakMap cache\n * - Finalizes entity metadata once per class\n * - Defers finalization via microtask (Promise.resolve().then)\n */\n\n// --- Decorator Context Types (based on ECMAScript Decorators proposal) ---\ntype ClassFieldDecoratorContext = {\n  kind: \"field\";\n  name: string | symbol;\n  static: boolean;\n  private: boolean;\n  addInitializer(fn: () => void): void;\n};\n\ntype ClassDecoratorContext = {\n  kind: \"class\";\n  name?: string | symbol;\n  addInitializer(fn: () => void): void;\n};\n\nexport type ColumnTypes =\n  // Numeric types\n  | 'integer' | 'long' | 'short' | 'byte' | 'float' | 'double' | 'big_integer' | 'big_decimal'\n  // String types\n  | 'string' | 'char' | 'text' | 'nstring' | 'ntext'\n  // Date/Time types\n  | 'date' | 'time' | 'timestamp' | 'calendar' | 'calendar_date' | 'instant'\n  // Boolean types\n  | 'boolean' | 'true_false' | 'yes_no' | 'numeric_boolean' \n  // Binary types\n  | 'binary' | 'blob' | 'clob' | 'materialized_blob' | 'materialized_clob'\n  // Other types\n  | 'serializable' | 'any' | 'object' | 'uuid-char' | 'uuid-binary' | 'json' | 'jsonb' | 'xml';\n\n// --- Metadata Models ---\nexport interface ColumnOptions {\n  name?: string;\n  type?: ColumnTypes | (string & {});\n  length?: number;\n  nullable?: boolean;\n  defaultValue?: string;\n  precision?: number;\n  scale?: number;\n}\n\nexport interface OneToManyOptions {\n  table?: string;\n  joinColumn: string;\n  cascade?: \"all\" | \"none\" | \"persist\" | \"merge\" | \"remove\";\n  inverse?: boolean;\n  lazy?: boolean;\n  fetch?: \"select\" | \"join\";\n  joinColumnNotNull?: boolean;\n}\n\nexport interface ManyToOneOptions {\n  joinColumn?: string;\n  cascade?: \"all\" | \"none\" | \"persist\" | \"merge\" | \"remove\";\n  nullable?: boolean;\n  lazy?: boolean;\n  fetch?: \"select\" | \"join\";\n}\n\ninterface PropertyMetadata {\n  propertyName: string;\n  isId: boolean;\n  isGenerated: boolean;\n  documentation?: string;\n  columnOptions?: ColumnOptions;\n  oneToManyOptions?: { type: Function; options: OneToManyOptions };\n  manyToOneOptions?: { type: Function; options: ManyToOneOptions };\n}\n\nexport interface EntityConstructor extends Function {\n  new(...args: any[]): any;\n  $entity_name: string;\n  $table_name: string;\n  $id_name: string;\n  $id_column: string;\n  $initialized?: boolean;\n  $documentation?: string;\n}\n\n// --- Global Metadata Cache ---\nconst globalCache: WeakMap<Function, PropertyMetadata[]> =\n  (globalThis as any).__decorator_metadata_cache__ ||\n  ((globalThis as any).__decorator_metadata_cache__ = new WeakMap());\n\nfunction getMetadataArray(constructor: Function): PropertyMetadata[] {\n  if (!globalCache.has(constructor)) {\n    globalCache.set(constructor, []);\n  }\n  return globalCache.get(constructor)!;\n}\n\n// --- Defer Helper (GraalJS-safe microtask) ---\nfunction defer(fn: () => void): void {\n  // Uses Promise microtask queue for deferred execution\n  Promise.resolve().then(fn);\n}\n\n// --- Core Property Decorator Factory ---\nfunction createPropertyDecorator(\n  kind: \"column\" | \"id\" | \"generated\",\n  options?: ColumnOptions\n) {\n  return function (_: any, context: ClassFieldDecoratorContext) {\n    if (context.kind !== \"field\") {\n      throw new Error(`@${kind} must apply to fields`);\n    }\n\n    context.addInitializer(function () {\n      const ctor = (this as any).constructor as EntityConstructor;\n      const propertyName = context.name.toString();\n      const metadataArray = getMetadataArray(ctor);\n\n      let metadata = metadataArray.find(\n        (m) => m.propertyName === propertyName\n      );\n\n      if (!metadata) {\n        metadata = {\n          propertyName: propertyName,\n          isId: false,\n          isGenerated: false,\n        };\n        metadataArray.push(metadata);\n      }\n\n      if (kind === \"id\") metadata.isId = true;\n      if (kind === \"generated\") metadata.isGenerated = true;\n      if (kind === \"column\") metadata.columnOptions = options;\n    });\n  };\n}\n\n// --- @Documentation Decorator (Dual-Purpose) ---\n/**\n * Adds documentation metadata to a class or a field.\n */\nexport function Documentation(description: string) {\n  return function (\n    value: Function | any,\n    context: ClassDecoratorContext | ClassFieldDecoratorContext\n  ) {\n    if (context.kind === \"class\") {\n      context.addInitializer(function () {\n        (value as EntityConstructor).$documentation = description;\n      });\n      return value;\n    } else if (context.kind === \"field\") {\n      context.addInitializer(function () {\n        const ctor = (this as any).constructor as EntityConstructor;\n        const propertyName = context.name.toString();\n        const metadataArray = getMetadataArray(ctor);\n\n        let metadata = metadataArray.find(\n          (m) => m.propertyName === propertyName\n        );\n\n        if (!metadata) {\n          metadata = {\n            propertyName: propertyName,\n            isId: false,\n            isGenerated: false,\n          };\n          metadataArray.push(metadata);\n        }\n\n        metadata.documentation = description;\n      });\n    }\n  };\n}\n\n\n// --- @Entity Decorator ---\n/**\n * Marks a class as an entity and initiates metadata finalization.\n * @param entityName The name of the entity (defaults to class name).\n */\nexport function Entity(entityName?: string) {\n  return function (value: Function, context: ClassDecoratorContext) {\n    context.addInitializer(function () {\n      // Defer execution to ensure all field decorators have run\n      defer(() => {\n        const ctor = value as EntityConstructor;\n\n        // Prevent duplicate registration (idempotency)\n        if (ctor.$initialized) return;\n        ctor.$initialized = true;\n\n        ctor.$entity_name = entityName || ctor.name;\n\n        const metadataArray = getMetadataArray(ctor);\n        const idMetadata = metadataArray.find((m) => m.isId);\n\n        if (idMetadata) {\n          ctor.$id_name = idMetadata.propertyName;\n          // Determine ID column name: use explicit name or convert property name to upper case\n          ctor.$id_column =\n            idMetadata.columnOptions?.name ||\n            idMetadata.propertyName.toUpperCase();\n        }\n\n        // Future: Logic to map all properties to columns/relations goes here\n      });\n    });\n  };\n}\n\n// --- @Table Decorator ---\n/**\n * Specifies the database table name for the entity.\n * @param tableName The table name (defaults to uppercase class name).\n */\nexport function Table(tableName?: string) {\n  return function <T extends EntityConstructor>(\n    value: T,\n    context: ClassDecoratorContext\n  ) {\n    context.addInitializer(function () {\n      (value as EntityConstructor).$table_name =\n        tableName || (context.name?.toString() ?? value.name.toUpperCase());\n    });\n\n    return value;\n  };\n}\n\n// --- Exported Property Decorators ---\n/**\n * Marks a property as a standard database column.\n */\nexport const Column = (options?: ColumnOptions) =>\n  createPropertyDecorator(\"column\", options);\n\n/**\n * Marks a property as the entity's primary key.\n */\nexport const Id = () => createPropertyDecorator(\"id\");\n\n/**\n * Marks a property as a generated value (e.g., auto-increment).\n * @param strategy The generation strategy (e.g., \"IDENTITY\"). Parameter is currently unused in logic.\n */\nexport const Generated = (strategy: string) =>\n  createPropertyDecorator(\"generated\");\n\n/**\n * Defines a one-to-many relationship.\n */\nexport function OneToMany(\n  typeFunction: () => Function,\n  options: OneToManyOptions\n) {\n  return function (_: any, context: ClassFieldDecoratorContext) {\n    if (context.kind !== \"field\") return;\n\n    context.addInitializer(function () {\n      const ctor = (this as any).constructor as EntityConstructor;\n      const propertyName = context.name.toString();\n      const metadataArray = getMetadataArray(ctor);\n\n      let metadata = metadataArray.find(\n        (m) => m.propertyName === propertyName\n      );\n\n      if (!metadata) {\n        metadata = {\n          propertyName: propertyName,\n          isId: false,\n          isGenerated: false,\n        };\n        metadataArray.push(metadata);\n      }\n\n      metadata.oneToManyOptions = { type: typeFunction(), options };\n    });\n  };\n}\n\n/**\n * Defines a many-to-one relationship.\n */\nexport function ManyToOne(\n  typeFunction: () => Function,\n  options: ManyToOneOptions = {}\n) {\n  return function (_: any, context: ClassFieldDecoratorContext) {\n    if (context.kind !== \"field\") return;\n\n    context.addInitializer(function () {\n      const ctor = (this as any).constructor as EntityConstructor;\n      const propertyName = context.name.toString();\n      const metadataArray = getMetadataArray(ctor);\n\n      let metadata = metadataArray.find(\n        (m) => m.propertyName === propertyName\n      );\n\n      if (!metadata) {\n        metadata = {\n          propertyName: propertyName,\n          isId: false,\n          isGenerated: false,\n        };\n        metadataArray.push(metadata);\n      }\n\n      metadata.manyToOneOptions = { type: typeFunction(), options };\n    });\n  };\n}"],
  "mappings": "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",
  "names": []
}

|
|
@@ -126,4 +126,4 @@ class Repository {
|
|
|
126
126
|
export {
|
|
127
127
|
Repository
|
|
128
128
|
};
|
|
129
|
-
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../src/db/repository.ts"],
  "sourcesContent": ["import { store, translator, EntityConstructor, Options } from \"@aerokit/sdk/db\";\n\n/**\n * Represents the data structure passed to the event trigger method before/after an operation.\n */\nexport interface EntityEvent<T> {\n    readonly operation: 'create' | 'update' | 'delete';\n    readonly table: string;\n    readonly entity: Partial<T>; // Use Partial<T> for create/delete where some fields might be missing\n    readonly key: {\n        name: string;\n        column: string;\n        value: string | number;\n    }\n    readonly previousEntity?: T;\n}\n\n\n// --- Repository Class ---\n\n/**\n * Abstract base class for data access/business logic, wrapping the `store` API.\n * It handles entity metadata lookup, CRUD operations, translation, and event triggering.\n * @template T The entity type (must be an object).\n */\nexport abstract class Repository<T extends Record<string, any>> {\n\n    private entityConstructor: EntityConstructor;\n\n    constructor(entityConstructor: EntityConstructor) {\n        this.entityConstructor = entityConstructor;\n        \n        // Caches entity metadata (name, table, id) onto the constructor function for static access\n\t\tif (!this.entityConstructor.$entity_name) {\n            // Assumes store methods return non-null strings\n\t\t\tthis.entityConstructor.$entity_name = (store as any).getEntityName(this.entityConstructor.name);\n\t\t\tthis.entityConstructor.$table_name = (store as any).getTableName(this.entityConstructor.name);\n\t\t\tthis.entityConstructor.$id_name = (store as any).getIdName(this.entityConstructor.name);\n\t\t\tthis.entityConstructor.$id_column = (store as any).getIdColumn(this.entityConstructor.name);\n\t\t}\n    }\n\n    protected getEntityName(): string {\n        // Use non-null assertion since the constructor guarantees these properties exist\n        return this.entityConstructor.$entity_name!;\n    }\n\n    protected getTableName(): string {\n        return this.entityConstructor.$table_name!;\n    }\n\n    protected getIdName(): string {\n        return this.entityConstructor.$id_name!;\n    }\n\n    protected getIdColumn(): string {\n        return this.entityConstructor.$id_column!;\n    }\n\n    /**\n     * Finds all entities matching the given options.\n     */\n    public findAll(options: Options = {}): T[] {\n        // Assume store.list returns T[] but we explicitly cast it to T[]\n        const list: T[] = (store as any).list(this.getEntityName(), options);\n        (translator as any).translateList(list, options.language, this.getTableName());\n        return list;\n    }\n\n    /**\n     * Finds a single entity by its primary key ID.\n     */\n    public findById(id: number | string, options: Options = {}): T | undefined {\n        // Assume store.get returns T or null/undefined\n        const entity: T | null = (store as any).get(this.getEntityName(), id);\n        (translator as any).translateEntity(entity, id, options.language, this.getTableName());\n        return entity ?? undefined;\n    }\n\n    /**\n     * Creates a new entity in the database.\n     * @returns The generated ID (string or number).\n     */\n    public create(entity: T): string | number {\n        const id = (store as any).save(this.getEntityName(), entity);\n        this.triggerEvent({\n            operation: \"create\",\n            table: this.getTableName(),\n            entity: entity,\n            key: {\n                name: this.getIdName(),\n                column: this.getIdColumn(),\n                value: id\n            }\n        });\n        return id;\n    }\n\n    /**\n     * Updates an existing entity.\n     * The entity must contain the primary key.\n     */\n    public update(entity: T): void {\n        const idName = this.getIdName();\n        const id = entity[idName] as (number | string);\n        \n        // Retrieve the entity state before update for the event payload\n        const previousEntity = this.findById(id);\n\n        (store as any).update(this.getEntityName(), entity);\n        \n        this.triggerEvent({\n            operation: \"update\",\n            table: this.getTableName(),\n            entity: entity,\n            previousEntity: previousEntity,\n            key: {\n                name: idName,\n                column: this.getIdColumn(),\n                value: id\n            }\n        });\n    }\n\n    /**\n     * Creates the entity if the ID is null/undefined, otherwise updates it.\n     * If an ID is provided but the entity doesn't exist, it creates it.\n     * @returns The entity's ID.\n     */\n    public upsert(entity: T): string | number {\n        const id = entity[this.getIdName()];\n        \n        // If no ID is present, save (create)\n        if (id === null || id === undefined) {\n            return (store as any).save(this.getEntityName(), entity);\n        }\n\n        // If ID is present, check existence\n        const existingEntity = (store as any).get(this.getEntityName(), id);\n        \n        if (existingEntity) {\n            this.update(entity);\n            return id;\n        } else {\n            // ID exists, but entity does not -> save (create with provided ID)\n            return (store as any).save(this.getEntityName(), entity);\n        }\n    }\n\n    /**\n     * Deletes an entity by its primary key ID.\n     */\n    public deleteById(id: number | string): void {\n        // Retrieve entity before removal for the event payload\n        const entity = (store as any).get(this.getEntityName(), id);\n        \n        (store as any).remove(this.getEntityName(), id);\n\n        this.triggerEvent({\n            operation: \"delete\",\n            table: this.getTableName(),\n            entity: entity,\n            key: {\n                name: this.getIdName(),\n                column: this.getIdColumn(),\n                value: id\n            }\n        });\n    }\n\n    /**\n     * Counts the number of entities matching the given options.\n     */\n    public count(options?: Options): number {\n        return (store as any).count(this.getEntityName(), options);\n    }\n\n    /**\n     * Protected method intended for subclass overriding or internal event handling.\n     */\n    protected async triggerEvent(_data: EntityEvent<T>): Promise<void> {\n        // Empty body as in the original code\n    }\n}\n"],
  "mappings": "AAAA,SAAS,OAAO,kBAA8C;AAyBvD,MAAe,WAA0C;AAAA,EAI5D,YAAY,mBAAsC;AAC9C,SAAK,oBAAoB;AAG/B,QAAI,CAAC,KAAK,kBAAkB,cAAc;AAEzC,WAAK,kBAAkB,eAAgB,MAAc,cAAc,KAAK,kBAAkB,IAAI;AAC9F,WAAK,kBAAkB,cAAe,MAAc,aAAa,KAAK,kBAAkB,IAAI;AAC5F,WAAK,kBAAkB,WAAY,MAAc,UAAU,KAAK,kBAAkB,IAAI;AACtF,WAAK,kBAAkB,aAAc,MAAc,YAAY,KAAK,kBAAkB,IAAI;AAAA,IAC3F;AAAA,EACE;AAAA,EAEU,gBAAwB;AAE9B,WAAO,KAAK,kBAAkB;AAAA,EAClC;AAAA,EAEU,eAAuB;AAC7B,WAAO,KAAK,kBAAkB;AAAA,EAClC;AAAA,EAEU,YAAoB;AAC1B,WAAO,KAAK,kBAAkB;AAAA,EAClC;AAAA,EAEU,cAAsB;AAC5B,WAAO,KAAK,kBAAkB;AAAA,EAClC;AAAA;AAAA;AAAA;AAAA,EAKO,QAAQ,UAAmB,CAAC,GAAQ;AAEvC,UAAM,OAAa,MAAc,KAAK,KAAK,cAAc,GAAG,OAAO;AACnE,IAAC,WAAmB,cAAc,MAAM,QAAQ,UAAU,KAAK,aAAa,CAAC;AAC7E,WAAO;AAAA,EACX;AAAA;AAAA;AAAA;AAAA,EAKO,SAAS,IAAqB,UAAmB,CAAC,GAAkB;AAEvE,UAAM,SAAoB,MAAc,IAAI,KAAK,cAAc,GAAG,EAAE;AACpE,IAAC,WAAmB,gBAAgB,QAAQ,IAAI,QAAQ,UAAU,KAAK,aAAa,CAAC;AACrF,WAAO,UAAU;AAAA,EACrB;AAAA;AAAA;AAAA;AAAA;AAAA,EAMO,OAAO,QAA4B;AACtC,UAAM,KAAM,MAAc,KAAK,KAAK,cAAc,GAAG,MAAM;AAC3D,SAAK,aAAa;AAAA,MACd,WAAW;AAAA,MACX,OAAO,KAAK,aAAa;AAAA,MACzB;AAAA,MACA,KAAK;AAAA,QACD,MAAM,KAAK,UAAU;AAAA,QACrB,QAAQ,KAAK,YAAY;AAAA,QACzB,OAAO;AAAA,MACX;AAAA,IACJ,CAAC;AACD,WAAO;AAAA,EACX;AAAA;AAAA;AAAA;AAAA;AAAA,EAMO,OAAO,QAAiB;AAC3B,UAAM,SAAS,KAAK,UAAU;AAC9B,UAAM,KAAK,OAAO,MAAM;AAGxB,UAAM,iBAAiB,KAAK,SAAS,EAAE;AAEvC,IAAC,MAAc,OAAO,KAAK,cAAc,GAAG,MAAM;AAElD,SAAK,aAAa;AAAA,MACd,WAAW;AAAA,MACX,OAAO,KAAK,aAAa;AAAA,MACzB;AAAA,MACA;AAAA,MACA,KAAK;AAAA,QACD,MAAM;AAAA,QACN,QAAQ,KAAK,YAAY;AAAA,QACzB,OAAO;AAAA,MACX;AAAA,IACJ,CAAC;AAAA,EACL;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAOO,OAAO,QAA4B;AACtC,UAAM,KAAK,OAAO,KAAK,UAAU,CAAC;AAGlC,QAAI,OAAO,QAAQ,OAAO,QAAW;AACjC,aAAQ,MAAc,KAAK,KAAK,cAAc,GAAG,MAAM;AAAA,IAC3D;AAGA,UAAM,iBAAkB,MAAc,IAAI,KAAK,cAAc,GAAG,EAAE;AAElE,QAAI,gBAAgB;AAChB,WAAK,OAAO,MAAM;AAClB,aAAO;AAAA,IACX,OAAO;AAEH,aAAQ,MAAc,KAAK,KAAK,cAAc,GAAG,MAAM;AAAA,IAC3D;AAAA,EACJ;AAAA;AAAA;AAAA;AAAA,EAKO,WAAW,IAA2B;AAEzC,UAAM,SAAU,MAAc,IAAI,KAAK,cAAc,GAAG,EAAE;AAE1D,IAAC,MAAc,OAAO,KAAK,cAAc,GAAG,EAAE;AAE9C,SAAK,aAAa;AAAA,MACd,WAAW;AAAA,MACX,OAAO,KAAK,aAAa;AAAA,MACzB;AAAA,MACA,KAAK;AAAA,QACD,MAAM,KAAK,UAAU;AAAA,QACrB,QAAQ,KAAK,YAAY;AAAA,QACzB,OAAO;AAAA,MACX;AAAA,IACJ,CAAC;AAAA,EACL;AAAA;AAAA;AAAA;AAAA,EAKO,MAAM,SAA2B;AACpC,WAAQ,MAAc,MAAM,KAAK,cAAc,GAAG,OAAO;AAAA,EAC7D;AAAA;AAAA;AAAA;AAAA,EAKA,MAAgB,aAAa,OAAsC;AAAA,EAEnE;AACJ;",
  "names": []
}

|
|
129
|
+
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../src/db/repository.ts"],
  "sourcesContent": ["import { store, translator, EntityConstructor, Options } from \"@aerokit/sdk/db\";\n\n/**\n * Represents the data structure passed to the event trigger method before/after an operation.\n */\nexport interface EntityEvent<T> {\n    readonly operation: 'create' | 'update' | 'delete';\n    readonly table: string;\n    readonly entity: Partial<T>; // Use Partial<T> for create/delete where some fields might be missing\n    readonly key: {\n        name: string;\n        column: string;\n        value: string | number;\n    }\n    readonly previousEntity?: T;\n}\n\n\n// --- Repository Class ---\n\n/**\n * Abstract base class for data access/business logic, wrapping the `store` API.\n * It handles entity metadata lookup, CRUD operations, translation, and event triggering.\n * @template T The entity type (must be an object).\n */\nexport abstract class Repository<T extends Record<string, any>> {\n\n    private entityConstructor: EntityConstructor;\n\n    constructor(entityConstructor: EntityConstructor) {\n        this.entityConstructor = entityConstructor;\n        \n        // Caches entity metadata (name, table, id) onto the constructor function for static access\n\t\tif (!this.entityConstructor.$entity_name) {\n            // Assumes store methods return non-null strings\n\t\t\tthis.entityConstructor.$entity_name = store.getEntityName(this.entityConstructor.name);\n\t\t\tthis.entityConstructor.$table_name = store.getTableName(this.entityConstructor.name);\n\t\t\tthis.entityConstructor.$id_name = store.getIdName(this.entityConstructor.name);\n\t\t\tthis.entityConstructor.$id_column = store.getIdColumn(this.entityConstructor.name);\n\t\t}\n    }\n\n    protected getEntityName(): string {\n        // Use non-null assertion since the constructor guarantees these properties exist\n        return this.entityConstructor.$entity_name!;\n    }\n\n    protected getTableName(): string {\n        return this.entityConstructor.$table_name!;\n    }\n\n    protected getIdName(): string {\n        return this.entityConstructor.$id_name!;\n    }\n\n    protected getIdColumn(): string {\n        return this.entityConstructor.$id_column!;\n    }\n\n    /**\n     * Finds all entities matching the given options.\n     */\n    public findAll(options: Options = {}): T[] {\n        // Assume store.list returns T[] but we explicitly cast it to T[]\n        const list: T[] = store.list(this.getEntityName(), options);\n        translator.translateList(list, options.language, this.getTableName());\n        return list;\n    }\n\n    /**\n     * Finds a single entity by its primary key ID.\n     */\n    public findById(id: number | string, options: Options = {}): T | undefined {\n        // Assume store.get returns T or null/undefined\n        const entity: T | null = store.get(this.getEntityName(), id);\n        translator.translateEntity(entity, id, options.language, this.getTableName());\n        return entity ?? undefined;\n    }\n\n    /**\n     * Creates a new entity in the database.\n     * @returns The generated ID (string or number).\n     */\n    public create(entity: T): string | number {\n        const id = store.save(this.getEntityName(), entity);\n        this.triggerEvent({\n            operation: \"create\",\n            table: this.getTableName(),\n            entity: entity,\n            key: {\n                name: this.getIdName(),\n                column: this.getIdColumn(),\n                value: id\n            }\n        });\n        return id;\n    }\n\n    /**\n     * Updates an existing entity.\n     * The entity must contain the primary key.\n     */\n    public update(entity: T): void {\n        const idName = this.getIdName();\n        const id = entity[idName] as (number | string);\n        \n        // Retrieve the entity state before update for the event payload\n        const previousEntity = this.findById(id);\n\n        store.update(this.getEntityName(), entity);\n        \n        this.triggerEvent({\n            operation: \"update\",\n            table: this.getTableName(),\n            entity: entity,\n            previousEntity: previousEntity,\n            key: {\n                name: idName,\n                column: this.getIdColumn(),\n                value: id\n            }\n        });\n    }\n\n    /**\n     * Creates the entity if the ID is null/undefined, otherwise updates it.\n     * If an ID is provided but the entity doesn't exist, it creates it.\n     * @returns The entity's ID.\n     */\n    public upsert(entity: T): string | number {\n        const id = entity[this.getIdName()];\n        \n        // If no ID is present, save (create)\n        if (id === null || id === undefined) {\n            return store.save(this.getEntityName(), entity);\n        }\n\n        // If ID is present, check existence\n        const existingEntity = store.get(this.getEntityName(), id);\n        \n        if (existingEntity) {\n            this.update(entity);\n            return id;\n        } else {\n            // ID exists, but entity does not -> save (create with provided ID)\n            return store.save(this.getEntityName(), entity);\n        }\n    }\n\n    /**\n     * Deletes an entity by its primary key ID.\n     */\n    public deleteById(id: number | string): void {\n        // Retrieve entity before removal for the event payload\n        const entity = store.get(this.getEntityName(), id);\n        \n        store.remove(this.getEntityName(), id);\n\n        this.triggerEvent({\n            operation: \"delete\",\n            table: this.getTableName(),\n            entity: entity,\n            key: {\n                name: this.getIdName(),\n                column: this.getIdColumn(),\n                value: id\n            }\n        });\n    }\n\n    /**\n     * Counts the number of entities matching the given options.\n     */\n    public count(options?: Options): number {\n        return store.count(this.getEntityName(), options);\n    }\n\n    /**\n     * Protected method intended for subclass overriding or internal event handling.\n     */\n    protected async triggerEvent(_data: EntityEvent<T>): Promise<void> {\n        // Empty body as in the original code\n    }\n}\n"],
  "mappings": "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",
  "names": []
}

|