@carno.js/core 0.2.3
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/LICENSE +674 -0
- package/dist/Carno.d.ts +65 -0
- package/dist/Carno.js +307 -0
- package/dist/Cheetah.d.ts +65 -0
- package/dist/Cheetah.js +307 -0
- package/dist/cache/bento-cache.driver.d.ts +13 -0
- package/dist/cache/bento-cache.driver.js +55 -0
- package/dist/cache/cache.service.d.ts +8 -0
- package/dist/cache/cache.service.js +6 -0
- package/dist/commons/decorators/Injectable.decorator.d.ts +20 -0
- package/dist/commons/decorators/Injectable.decorator.js +33 -0
- package/dist/commons/decorators/controller.decorator.d.ts +8 -0
- package/dist/commons/decorators/controller.decorator.js +22 -0
- package/dist/commons/decorators/http.decorators.d.ts +13 -0
- package/dist/commons/decorators/http.decorators.js +44 -0
- package/dist/commons/decorators/index.d.ts +5 -0
- package/dist/commons/decorators/index.js +21 -0
- package/dist/commons/decorators/inject.decorator.d.ts +1 -0
- package/dist/commons/decorators/inject.decorator.js +5 -0
- package/dist/commons/decorators/middleware.decorator.d.ts +2 -0
- package/dist/commons/decorators/middleware.decorator.js +30 -0
- package/dist/commons/decorators/service.decorator.d.ts +2 -0
- package/dist/commons/decorators/service.decorator.js +7 -0
- package/dist/commons/http-code.enum.d.ts +50 -0
- package/dist/commons/http-code.enum.js +54 -0
- package/dist/commons/index.d.ts +3 -0
- package/dist/commons/index.js +19 -0
- package/dist/commons/registries/ProviderControl.d.ts +77 -0
- package/dist/commons/registries/ProviderControl.js +112 -0
- package/dist/commons/registries/ProviderRegistry.d.ts +7 -0
- package/dist/commons/registries/ProviderRegistry.js +20 -0
- package/dist/constants.d.ts +7 -0
- package/dist/constants.js +10 -0
- package/dist/container/ContainerConfiguration.d.ts +45 -0
- package/dist/container/ContainerConfiguration.js +121 -0
- package/dist/container/DependencyResolver.d.ts +18 -0
- package/dist/container/DependencyResolver.js +81 -0
- package/dist/container/InjectorService.d.ts +34 -0
- package/dist/container/InjectorService.js +157 -0
- package/dist/container/MethodInvoker.d.ts +17 -0
- package/dist/container/MethodInvoker.js +69 -0
- package/dist/container/RouteResolver.d.ts +27 -0
- package/dist/container/RouteResolver.js +173 -0
- package/dist/container/container.d.ts +41 -0
- package/dist/container/container.js +71 -0
- package/dist/container/createContainer.d.ts +3 -0
- package/dist/container/createContainer.js +12 -0
- package/dist/container/createInjector.d.ts +2 -0
- package/dist/container/createInjector.js +7 -0
- package/dist/container/index.d.ts +6 -0
- package/dist/container/index.js +22 -0
- package/dist/container/middleware.resolver.d.ts +9 -0
- package/dist/container/middleware.resolver.js +35 -0
- package/dist/default-routes-carno.d.ts +3 -0
- package/dist/default-routes-carno.js +29 -0
- package/dist/default-routes-cheetah.d.ts +3 -0
- package/dist/default-routes-cheetah.js +29 -0
- package/dist/domain/CarnoClosure.d.ts +1 -0
- package/dist/domain/CarnoClosure.js +2 -0
- package/dist/domain/CarnoMiddleware.d.ts +5 -0
- package/dist/domain/CarnoMiddleware.js +2 -0
- package/dist/domain/CheetahClosure.d.ts +1 -0
- package/dist/domain/CheetahClosure.js +2 -0
- package/dist/domain/CheetahMiddleware.d.ts +5 -0
- package/dist/domain/CheetahMiddleware.js +2 -0
- package/dist/domain/Context.d.ts +26 -0
- package/dist/domain/Context.js +107 -0
- package/dist/domain/LocalsContainer.d.ts +4 -0
- package/dist/domain/LocalsContainer.js +10 -0
- package/dist/domain/Metadata.d.ts +449 -0
- package/dist/domain/Metadata.js +511 -0
- package/dist/domain/cors-config.d.ts +12 -0
- package/dist/domain/cors-config.js +18 -0
- package/dist/domain/http-method.d.ts +7 -0
- package/dist/domain/http-method.js +11 -0
- package/dist/domain/index.d.ts +10 -0
- package/dist/domain/index.js +26 -0
- package/dist/domain/provider-scope.d.ts +5 -0
- package/dist/domain/provider-scope.js +9 -0
- package/dist/domain/provider-type.d.ts +6 -0
- package/dist/domain/provider-type.js +10 -0
- package/dist/domain/provider.d.ts +37 -0
- package/dist/domain/provider.js +70 -0
- package/dist/events/hooks.decorator.d.ts +3 -0
- package/dist/events/hooks.decorator.js +29 -0
- package/dist/events/index.d.ts +2 -0
- package/dist/events/index.js +18 -0
- package/dist/events/on-event.d.ts +13 -0
- package/dist/events/on-event.js +11 -0
- package/dist/exceptions/HttpException.d.ts +9 -0
- package/dist/exceptions/HttpException.js +26 -0
- package/dist/exceptions/index.d.ts +1 -0
- package/dist/exceptions/index.js +17 -0
- package/dist/index.d.ts +16 -0
- package/dist/index.js +32 -0
- package/dist/route/Matcher.d.ts +16 -0
- package/dist/route/Matcher.js +48 -0
- package/dist/route/RouteExecutor.d.ts +13 -0
- package/dist/route/RouteExecutor.js +70 -0
- package/dist/route/memoirist.d.ts +25 -0
- package/dist/route/memoirist.js +250 -0
- package/dist/services/logger.service.d.ts +23 -0
- package/dist/services/logger.service.js +54 -0
- package/dist/services/request-logger.service.d.ts +15 -0
- package/dist/services/request-logger.service.js +50 -0
- package/dist/testing/core-testing.d.ts +24 -0
- package/dist/testing/core-testing.js +98 -0
- package/dist/testing/index.d.ts +1 -0
- package/dist/testing/index.js +17 -0
- package/dist/utils/ancestorOf.d.ts +2 -0
- package/dist/utils/ancestorOf.js +10 -0
- package/dist/utils/ancestorsOf.d.ts +6 -0
- package/dist/utils/ancestorsOf.js +20 -0
- package/dist/utils/classOf.d.ts +13 -0
- package/dist/utils/classOf.js +21 -0
- package/dist/utils/cleanObject.d.ts +6 -0
- package/dist/utils/cleanObject.js +22 -0
- package/dist/utils/constructorOf.d.ts +11 -0
- package/dist/utils/constructorOf.js +18 -0
- package/dist/utils/createInstance.d.ts +1 -0
- package/dist/utils/createInstance.js +7 -0
- package/dist/utils/decoratorTypeOf.d.ts +11 -0
- package/dist/utils/decoratorTypeOf.js +32 -0
- package/dist/utils/deepClone.d.ts +6 -0
- package/dist/utils/deepClone.js +63 -0
- package/dist/utils/deepMerge.d.ts +9 -0
- package/dist/utils/deepMerge.js +62 -0
- package/dist/utils/descriptorOf.d.ts +8 -0
- package/dist/utils/descriptorOf.js +16 -0
- package/dist/utils/getClassOrSymbol.d.ts +1 -0
- package/dist/utils/getClassOrSymbol.js +8 -0
- package/dist/utils/getConstructorArgNames.d.ts +1 -0
- package/dist/utils/getConstructorArgNames.js +12 -0
- package/dist/utils/getMethodArgTypes.d.ts +1 -0
- package/dist/utils/getMethodArgTypes.js +9 -0
- package/dist/utils/getValue.d.ts +32 -0
- package/dist/utils/getValue.js +47 -0
- package/dist/utils/hasJsonMethod.d.ts +1 -0
- package/dist/utils/hasJsonMethod.js +6 -0
- package/dist/utils/index.d.ts +15 -0
- package/dist/utils/index.js +31 -0
- package/dist/utils/isArray.d.ts +13 -0
- package/dist/utils/isArray.js +21 -0
- package/dist/utils/isArrowFn.d.ts +1 -0
- package/dist/utils/isArrowFn.js +7 -0
- package/dist/utils/isBoolean.d.ts +7 -0
- package/dist/utils/isBoolean.js +15 -0
- package/dist/utils/isBuffer.d.ts +7 -0
- package/dist/utils/isBuffer.js +19 -0
- package/dist/utils/isClass.d.ts +1 -0
- package/dist/utils/isClass.js +26 -0
- package/dist/utils/isClassValidator.d.ts +6 -0
- package/dist/utils/isClassValidator.js +13 -0
- package/dist/utils/isCollection.d.ts +6 -0
- package/dist/utils/isCollection.js +20 -0
- package/dist/utils/isDate.d.ts +6 -0
- package/dist/utils/isDate.js +11 -0
- package/dist/utils/isEmpty.d.ts +6 -0
- package/dist/utils/isEmpty.js +12 -0
- package/dist/utils/isFunction.d.ts +1 -0
- package/dist/utils/isFunction.js +6 -0
- package/dist/utils/isInheritedFrom.d.ts +1 -0
- package/dist/utils/isInheritedFrom.js +24 -0
- package/dist/utils/isMomentObject.d.ts +1 -0
- package/dist/utils/isMomentObject.js +6 -0
- package/dist/utils/isMongooseObject.d.ts +2 -0
- package/dist/utils/isMongooseObject.js +11 -0
- package/dist/utils/isNil.d.ts +1 -0
- package/dist/utils/isNil.js +6 -0
- package/dist/utils/isNumber.d.ts +7 -0
- package/dist/utils/isNumber.js +15 -0
- package/dist/utils/isObject.d.ts +1 -0
- package/dist/utils/isObject.js +6 -0
- package/dist/utils/isObservable.d.ts +1 -0
- package/dist/utils/isObservable.js +6 -0
- package/dist/utils/isPlainObject.d.ts +7 -0
- package/dist/utils/isPlainObject.js +16 -0
- package/dist/utils/isPrimitive.d.ts +14 -0
- package/dist/utils/isPrimitive.js +28 -0
- package/dist/utils/isPrimitiveType.d.ts +1 -0
- package/dist/utils/isPrimitiveType.js +11 -0
- package/dist/utils/isPromise.d.ts +7 -0
- package/dist/utils/isPromise.js +14 -0
- package/dist/utils/isProtectedKey.d.ts +5 -0
- package/dist/utils/isProtectedKey.js +10 -0
- package/dist/utils/isRegExp.d.ts +1 -0
- package/dist/utils/isRegExp.js +6 -0
- package/dist/utils/isRequestScope.d.ts +11 -0
- package/dist/utils/isRequestScope.js +23 -0
- package/dist/utils/isSerializable.d.ts +1 -0
- package/dist/utils/isSerializable.js +11 -0
- package/dist/utils/isStream.d.ts +1 -0
- package/dist/utils/isStream.js +6 -0
- package/dist/utils/isString.d.ts +6 -0
- package/dist/utils/isString.js +14 -0
- package/dist/utils/isSymbol.d.ts +6 -0
- package/dist/utils/isSymbol.js +14 -0
- package/dist/utils/methodsOf.d.ts +9 -0
- package/dist/utils/methodsOf.js +24 -0
- package/dist/utils/nameOf.d.ts +14 -0
- package/dist/utils/nameOf.js +31 -0
- package/dist/utils/objectKeys.d.ts +1 -0
- package/dist/utils/objectKeys.js +7 -0
- package/dist/utils/primitiveOf.d.ts +1 -0
- package/dist/utils/primitiveOf.js +18 -0
- package/dist/utils/prototypeOf.d.ts +6 -0
- package/dist/utils/prototypeOf.js +12 -0
- package/dist/utils/setValue.d.ts +1 -0
- package/dist/utils/setValue.js +32 -0
- package/dist/utils/toMap.d.ts +3 -0
- package/dist/utils/toMap.js +34 -0
- package/dist/utils/toStringConstructor.d.ts +1 -0
- package/dist/utils/toStringConstructor.js +10 -0
- package/package.json +55 -0
package/dist/Carno.d.ts
ADDED
|
@@ -0,0 +1,65 @@
|
|
|
1
|
+
import { Server } from "bun";
|
|
2
|
+
import { ValidatorOptions } from "class-validator";
|
|
3
|
+
import * as pino from "pino";
|
|
4
|
+
import { TokenRouteWithProvider } from "./container/ContainerConfiguration";
|
|
5
|
+
import { CorsConfig } from "./domain/cors-config";
|
|
6
|
+
import Memoirist from "./route/memoirist";
|
|
7
|
+
export interface ApplicationConfig {
|
|
8
|
+
validation?: ValidatorOptions;
|
|
9
|
+
logger?: pino.LoggerOptions;
|
|
10
|
+
exports?: any[];
|
|
11
|
+
providers?: any[];
|
|
12
|
+
cors?: CorsConfig;
|
|
13
|
+
globalMiddlewares?: any[];
|
|
14
|
+
}
|
|
15
|
+
export declare class Carno {
|
|
16
|
+
config: ApplicationConfig;
|
|
17
|
+
router: Memoirist<TokenRouteWithProvider>;
|
|
18
|
+
private injector;
|
|
19
|
+
private fetch;
|
|
20
|
+
private server;
|
|
21
|
+
constructor(config?: ApplicationConfig);
|
|
22
|
+
/**
|
|
23
|
+
* Use the Carno plugin.
|
|
24
|
+
*
|
|
25
|
+
* @param plugin
|
|
26
|
+
*/
|
|
27
|
+
use(plugin: Carno): this;
|
|
28
|
+
private findProviderInConfig;
|
|
29
|
+
private getProviderToken;
|
|
30
|
+
private shouldCloneProvider;
|
|
31
|
+
private cloneProvider;
|
|
32
|
+
/**
|
|
33
|
+
* Set the custom logger provider.
|
|
34
|
+
* The provider must be a class with the @Service() decorator.
|
|
35
|
+
* The provider must extend the LoggerService class.
|
|
36
|
+
*
|
|
37
|
+
* @param provider
|
|
38
|
+
*/
|
|
39
|
+
useLogger(provider: any): this;
|
|
40
|
+
private loadProvidersAndControllers;
|
|
41
|
+
private registerControllers;
|
|
42
|
+
private registerMetadataProviders;
|
|
43
|
+
private registerConfigProviders;
|
|
44
|
+
private normalizeProvider;
|
|
45
|
+
init(): Promise<void>;
|
|
46
|
+
listen(port?: number): Promise<void>;
|
|
47
|
+
private registerShutdownHandlers;
|
|
48
|
+
getHttpServer(): Server<any>;
|
|
49
|
+
getInjector(): import(".").InjectorService;
|
|
50
|
+
private createHttpServer;
|
|
51
|
+
private fetcher;
|
|
52
|
+
private catcher;
|
|
53
|
+
private bootstrapApplication;
|
|
54
|
+
private handleShutdownHook;
|
|
55
|
+
private closeHttpServer;
|
|
56
|
+
private exitProcess;
|
|
57
|
+
private reportHookFailure;
|
|
58
|
+
private isCorsEnabled;
|
|
59
|
+
private isOriginAllowed;
|
|
60
|
+
private buildCorsHeaders;
|
|
61
|
+
private handlePreflightRequest;
|
|
62
|
+
private applyCorsHeaders;
|
|
63
|
+
close(closeActiveConnections?: boolean): void;
|
|
64
|
+
private discoverRoutePath;
|
|
65
|
+
}
|
package/dist/Carno.js
ADDED
|
@@ -0,0 +1,307 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
var __importDefault = (this && this.__importDefault) || function (mod) {
|
|
3
|
+
return (mod && mod.__esModule) ? mod : { "default": mod };
|
|
4
|
+
};
|
|
5
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
6
|
+
exports.Carno = void 0;
|
|
7
|
+
const node_process_1 = __importDefault(require("node:process"));
|
|
8
|
+
const index_1 = require("./commons/index");
|
|
9
|
+
const constants_1 = require("./constants");
|
|
10
|
+
const createContainer_1 = require("./container/createContainer");
|
|
11
|
+
const createInjector_1 = require("./container/createInjector");
|
|
12
|
+
const domain_1 = require("./domain");
|
|
13
|
+
const Context_1 = require("./domain/Context");
|
|
14
|
+
const LocalsContainer_1 = require("./domain/LocalsContainer");
|
|
15
|
+
const cors_config_1 = require("./domain/cors-config");
|
|
16
|
+
const on_event_1 = require("./events/on-event");
|
|
17
|
+
const HttpException_1 = require("./exceptions/HttpException");
|
|
18
|
+
const RouteExecutor_1 = require("./route/RouteExecutor");
|
|
19
|
+
const memoirist_1 = __importDefault(require("./route/memoirist"));
|
|
20
|
+
const logger_service_1 = require("./services/logger.service");
|
|
21
|
+
const parseUrl = require("parseurl-fast");
|
|
22
|
+
// todo: change console.log for LoggerService.
|
|
23
|
+
class Carno {
|
|
24
|
+
constructor(config = {}) {
|
|
25
|
+
this.config = config;
|
|
26
|
+
this.router = new memoirist_1.default();
|
|
27
|
+
this.injector = (0, createInjector_1.createInjector)();
|
|
28
|
+
this.fetch = async (request, server) => {
|
|
29
|
+
try {
|
|
30
|
+
return await this.fetcher(request, server);
|
|
31
|
+
}
|
|
32
|
+
catch (error) {
|
|
33
|
+
if (error instanceof HttpException_1.HttpException) {
|
|
34
|
+
let response = new Response(JSON.stringify({
|
|
35
|
+
message: error.getResponse(),
|
|
36
|
+
statusCode: error.getStatus(),
|
|
37
|
+
}), {
|
|
38
|
+
status: error.statusCode,
|
|
39
|
+
headers: { "Content-Type": "application/json" },
|
|
40
|
+
});
|
|
41
|
+
if (this.isCorsEnabled()) {
|
|
42
|
+
const origin = request.headers.get("origin");
|
|
43
|
+
if (origin && this.isOriginAllowed(origin)) {
|
|
44
|
+
response = this.applyCorsHeaders(response, origin);
|
|
45
|
+
}
|
|
46
|
+
}
|
|
47
|
+
return response;
|
|
48
|
+
}
|
|
49
|
+
throw error;
|
|
50
|
+
}
|
|
51
|
+
};
|
|
52
|
+
this.catcher = (error) => {
|
|
53
|
+
console.error("Unhandled error:", error);
|
|
54
|
+
return new Response("Internal Server Error", { status: 500 });
|
|
55
|
+
};
|
|
56
|
+
void this.bootstrapApplication();
|
|
57
|
+
}
|
|
58
|
+
/**
|
|
59
|
+
* Use the Carno plugin.
|
|
60
|
+
*
|
|
61
|
+
* @param plugin
|
|
62
|
+
*/
|
|
63
|
+
use(plugin) {
|
|
64
|
+
if (!this.config.providers) {
|
|
65
|
+
this.config.providers = [];
|
|
66
|
+
}
|
|
67
|
+
if (!this.config.globalMiddlewares) {
|
|
68
|
+
this.config.globalMiddlewares = [];
|
|
69
|
+
}
|
|
70
|
+
for (const exportProvider of plugin.config.exports || []) {
|
|
71
|
+
const existingProvider = this.findProviderInConfig(plugin, exportProvider);
|
|
72
|
+
const providerToAdd = this.shouldCloneProvider(existingProvider)
|
|
73
|
+
? this.cloneProvider(existingProvider)
|
|
74
|
+
: exportProvider;
|
|
75
|
+
this.config.providers.push(providerToAdd);
|
|
76
|
+
}
|
|
77
|
+
if (plugin.config.globalMiddlewares) {
|
|
78
|
+
this.config.globalMiddlewares.push(...plugin.config.globalMiddlewares);
|
|
79
|
+
}
|
|
80
|
+
return this;
|
|
81
|
+
}
|
|
82
|
+
findProviderInConfig(plugin, exportProvider) {
|
|
83
|
+
return plugin.config.providers?.find((p) => this.getProviderToken(p) === this.getProviderToken(exportProvider));
|
|
84
|
+
}
|
|
85
|
+
getProviderToken(provider) {
|
|
86
|
+
return provider?.provide || provider;
|
|
87
|
+
}
|
|
88
|
+
shouldCloneProvider(provider) {
|
|
89
|
+
return !!(provider?.useValue !== undefined || provider?.useClass);
|
|
90
|
+
}
|
|
91
|
+
cloneProvider(provider) {
|
|
92
|
+
return { ...provider };
|
|
93
|
+
}
|
|
94
|
+
/**
|
|
95
|
+
* Set the custom logger provider.
|
|
96
|
+
* The provider must be a class with the @Service() decorator.
|
|
97
|
+
* The provider must extend the LoggerService class.
|
|
98
|
+
*
|
|
99
|
+
* @param provider
|
|
100
|
+
*/
|
|
101
|
+
useLogger(provider) {
|
|
102
|
+
(0, index_1.registerProvider)({ provide: logger_service_1.LoggerService, useClass: provider });
|
|
103
|
+
return this;
|
|
104
|
+
}
|
|
105
|
+
loadProvidersAndControllers() {
|
|
106
|
+
const providers = domain_1.Metadata.get(constants_1.PROVIDER, Reflect) || [];
|
|
107
|
+
const controllers = domain_1.Metadata.get(constants_1.CONTROLLER, Reflect) || [];
|
|
108
|
+
this.registerControllers(controllers);
|
|
109
|
+
this.registerMetadataProviders(providers);
|
|
110
|
+
this.registerConfigProviders();
|
|
111
|
+
}
|
|
112
|
+
registerControllers(controllers) {
|
|
113
|
+
for (const controller of controllers) {
|
|
114
|
+
(0, index_1.registerController)(controller);
|
|
115
|
+
controller.options?.children &&
|
|
116
|
+
controller.options.children.forEach((child) => {
|
|
117
|
+
(0, index_1.registerController)({ provide: child, parent: controller.provide });
|
|
118
|
+
});
|
|
119
|
+
}
|
|
120
|
+
}
|
|
121
|
+
registerMetadataProviders(providers) {
|
|
122
|
+
for (const provider of providers) {
|
|
123
|
+
(0, index_1.registerProvider)(provider);
|
|
124
|
+
}
|
|
125
|
+
}
|
|
126
|
+
registerConfigProviders() {
|
|
127
|
+
if (!this.config.providers) {
|
|
128
|
+
return;
|
|
129
|
+
}
|
|
130
|
+
for (const provider of this.config.providers) {
|
|
131
|
+
const normalized = this.normalizeProvider(provider);
|
|
132
|
+
(0, index_1.registerProvider)(normalized);
|
|
133
|
+
}
|
|
134
|
+
}
|
|
135
|
+
normalizeProvider(provider) {
|
|
136
|
+
if (provider?.provide) {
|
|
137
|
+
return provider;
|
|
138
|
+
}
|
|
139
|
+
return {
|
|
140
|
+
provide: provider,
|
|
141
|
+
useClass: provider,
|
|
142
|
+
};
|
|
143
|
+
}
|
|
144
|
+
async init() {
|
|
145
|
+
this.loadProvidersAndControllers();
|
|
146
|
+
await this.injector.loadModule((0, createContainer_1.createContainer)(), this.config, this.router);
|
|
147
|
+
}
|
|
148
|
+
async listen(port = 3000) {
|
|
149
|
+
this.registerShutdownHandlers();
|
|
150
|
+
await this.init();
|
|
151
|
+
this.createHttpServer(port);
|
|
152
|
+
}
|
|
153
|
+
registerShutdownHandlers() {
|
|
154
|
+
const shutdown = async (signal) => {
|
|
155
|
+
console.log(`\nReceived ${signal}, starting graceful shutdown...`);
|
|
156
|
+
await this.handleShutdownHook();
|
|
157
|
+
};
|
|
158
|
+
node_process_1.default.on("SIGTERM", () => shutdown("SIGTERM"));
|
|
159
|
+
node_process_1.default.on("SIGINT", () => shutdown("SIGINT"));
|
|
160
|
+
}
|
|
161
|
+
getHttpServer() {
|
|
162
|
+
return this.server;
|
|
163
|
+
}
|
|
164
|
+
getInjector() {
|
|
165
|
+
return this.injector;
|
|
166
|
+
}
|
|
167
|
+
createHttpServer(port) {
|
|
168
|
+
this.server = Bun.serve({ port, fetch: this.fetch, error: this.catcher });
|
|
169
|
+
console.log(`Server running on port ${port}`);
|
|
170
|
+
}
|
|
171
|
+
async fetcher(request, server) {
|
|
172
|
+
if (this.isCorsEnabled()) {
|
|
173
|
+
const origin = request.headers.get("origin");
|
|
174
|
+
if (request.method === "OPTIONS" && origin) {
|
|
175
|
+
return this.handlePreflightRequest(request);
|
|
176
|
+
}
|
|
177
|
+
}
|
|
178
|
+
const urlParsed = parseUrl(request);
|
|
179
|
+
const context = await Context_1.Context.createFromRequest(urlParsed, request, server);
|
|
180
|
+
await this.injector.callHook(on_event_1.EventType.OnRequest, { context });
|
|
181
|
+
const local = new LocalsContainer_1.LocalsContainer();
|
|
182
|
+
const routePath = this.discoverRoutePath(urlParsed);
|
|
183
|
+
const route = this.router.find(request.method.toLowerCase(), routePath);
|
|
184
|
+
if (!route) {
|
|
185
|
+
throw new HttpException_1.HttpException("Method not allowed", 404);
|
|
186
|
+
}
|
|
187
|
+
context.param = route.params;
|
|
188
|
+
local.set(Context_1.Context, context);
|
|
189
|
+
let response = await RouteExecutor_1.RouteExecutor.executeRoute(route.store, this.injector, context, local);
|
|
190
|
+
if (this.isCorsEnabled()) {
|
|
191
|
+
const origin = request.headers.get("origin");
|
|
192
|
+
if (origin && this.isOriginAllowed(origin)) {
|
|
193
|
+
response = this.applyCorsHeaders(response, origin);
|
|
194
|
+
}
|
|
195
|
+
}
|
|
196
|
+
return response;
|
|
197
|
+
}
|
|
198
|
+
async bootstrapApplication() {
|
|
199
|
+
try {
|
|
200
|
+
await this.injector.callHook(on_event_1.EventType.OnApplicationBoot, {});
|
|
201
|
+
}
|
|
202
|
+
catch (error) {
|
|
203
|
+
this.reportHookFailure(on_event_1.EventType.OnApplicationBoot, error);
|
|
204
|
+
}
|
|
205
|
+
}
|
|
206
|
+
async handleShutdownHook() {
|
|
207
|
+
try {
|
|
208
|
+
await this.injector.callHook(on_event_1.EventType.OnApplicationShutdown);
|
|
209
|
+
this.closeHttpServer();
|
|
210
|
+
this.exitProcess();
|
|
211
|
+
}
|
|
212
|
+
catch (error) {
|
|
213
|
+
this.reportHookFailure(on_event_1.EventType.OnApplicationShutdown, error);
|
|
214
|
+
this.exitProcess(1);
|
|
215
|
+
}
|
|
216
|
+
}
|
|
217
|
+
closeHttpServer() {
|
|
218
|
+
if (this.server) {
|
|
219
|
+
console.log("Closing HTTP server...");
|
|
220
|
+
this.server.stop(true);
|
|
221
|
+
}
|
|
222
|
+
}
|
|
223
|
+
exitProcess(code = 0) {
|
|
224
|
+
console.log("Shutdown complete.");
|
|
225
|
+
node_process_1.default.exit(code);
|
|
226
|
+
}
|
|
227
|
+
reportHookFailure(event, error) {
|
|
228
|
+
console.error(`Lifecycle hook ${event} failed`, error);
|
|
229
|
+
}
|
|
230
|
+
isCorsEnabled() {
|
|
231
|
+
return !!this.config.cors;
|
|
232
|
+
}
|
|
233
|
+
isOriginAllowed(origin) {
|
|
234
|
+
if (!origin || !this.config.cors) {
|
|
235
|
+
return false;
|
|
236
|
+
}
|
|
237
|
+
const { origins } = this.config.cors;
|
|
238
|
+
if (typeof origins === "string") {
|
|
239
|
+
return origins === "*" || origins === origin;
|
|
240
|
+
}
|
|
241
|
+
if (Array.isArray(origins)) {
|
|
242
|
+
return origins.includes(origin);
|
|
243
|
+
}
|
|
244
|
+
if (origins instanceof RegExp) {
|
|
245
|
+
return origins.test(origin);
|
|
246
|
+
}
|
|
247
|
+
if (typeof origins === "function") {
|
|
248
|
+
return origins(origin);
|
|
249
|
+
}
|
|
250
|
+
return false;
|
|
251
|
+
}
|
|
252
|
+
buildCorsHeaders(origin) {
|
|
253
|
+
const cors = this.config.cors;
|
|
254
|
+
const headers = {};
|
|
255
|
+
const allowedOrigin = typeof cors.origins === "string" && cors.origins === "*"
|
|
256
|
+
? "*"
|
|
257
|
+
: origin;
|
|
258
|
+
headers["Access-Control-Allow-Origin"] = allowedOrigin;
|
|
259
|
+
if (cors.credentials) {
|
|
260
|
+
headers["Access-Control-Allow-Credentials"] = "true";
|
|
261
|
+
}
|
|
262
|
+
const methods = cors.methods || cors_config_1.DEFAULT_CORS_METHODS;
|
|
263
|
+
headers["Access-Control-Allow-Methods"] = methods.join(", ");
|
|
264
|
+
const allowedHeaders = cors.allowedHeaders || cors_config_1.DEFAULT_CORS_ALLOWED_HEADERS;
|
|
265
|
+
headers["Access-Control-Allow-Headers"] = allowedHeaders.join(", ");
|
|
266
|
+
if (cors.exposedHeaders && cors.exposedHeaders.length > 0) {
|
|
267
|
+
headers["Access-Control-Expose-Headers"] = cors.exposedHeaders.join(", ");
|
|
268
|
+
}
|
|
269
|
+
if (cors.maxAge !== undefined) {
|
|
270
|
+
headers["Access-Control-Max-Age"] = cors.maxAge.toString();
|
|
271
|
+
}
|
|
272
|
+
return headers;
|
|
273
|
+
}
|
|
274
|
+
handlePreflightRequest(request) {
|
|
275
|
+
const origin = request.headers.get("origin");
|
|
276
|
+
if (!this.isOriginAllowed(origin)) {
|
|
277
|
+
return new Response(null, { status: 403 });
|
|
278
|
+
}
|
|
279
|
+
const corsHeaders = this.buildCorsHeaders(origin);
|
|
280
|
+
return new Response(null, {
|
|
281
|
+
status: 204,
|
|
282
|
+
headers: corsHeaders,
|
|
283
|
+
});
|
|
284
|
+
}
|
|
285
|
+
applyCorsHeaders(response, origin) {
|
|
286
|
+
const corsHeaders = this.buildCorsHeaders(origin);
|
|
287
|
+
const newHeaders = new Headers(response.headers);
|
|
288
|
+
for (const [key, value] of Object.entries(corsHeaders)) {
|
|
289
|
+
newHeaders.set(key, value);
|
|
290
|
+
}
|
|
291
|
+
return new Response(response.body, {
|
|
292
|
+
status: response.status,
|
|
293
|
+
statusText: response.statusText,
|
|
294
|
+
headers: newHeaders,
|
|
295
|
+
});
|
|
296
|
+
}
|
|
297
|
+
close(closeActiveConnections = false) {
|
|
298
|
+
this.server?.stop(closeActiveConnections);
|
|
299
|
+
}
|
|
300
|
+
discoverRoutePath(url) {
|
|
301
|
+
if (url?.pathname) {
|
|
302
|
+
return url.pathname;
|
|
303
|
+
}
|
|
304
|
+
return url?.path || "/";
|
|
305
|
+
}
|
|
306
|
+
}
|
|
307
|
+
exports.Carno = Carno;
|
|
@@ -0,0 +1,65 @@
|
|
|
1
|
+
import { Server } from "bun";
|
|
2
|
+
import { ValidatorOptions } from "class-validator";
|
|
3
|
+
import * as pino from "pino";
|
|
4
|
+
import { TokenRouteWithProvider } from "./container/ContainerConfiguration";
|
|
5
|
+
import { CorsConfig } from "./domain/cors-config";
|
|
6
|
+
import Memoirist from "./route/memoirist";
|
|
7
|
+
export interface ApplicationConfig {
|
|
8
|
+
validation?: ValidatorOptions;
|
|
9
|
+
logger?: pino.LoggerOptions;
|
|
10
|
+
exports?: any[];
|
|
11
|
+
providers?: any[];
|
|
12
|
+
cors?: CorsConfig;
|
|
13
|
+
globalMiddlewares?: any[];
|
|
14
|
+
}
|
|
15
|
+
export declare class Cheetah {
|
|
16
|
+
config: ApplicationConfig;
|
|
17
|
+
router: Memoirist<TokenRouteWithProvider>;
|
|
18
|
+
private injector;
|
|
19
|
+
private fetch;
|
|
20
|
+
private server;
|
|
21
|
+
constructor(config?: ApplicationConfig);
|
|
22
|
+
/**
|
|
23
|
+
* Use the Cheetah plugin.
|
|
24
|
+
*
|
|
25
|
+
* @param plugin
|
|
26
|
+
*/
|
|
27
|
+
use(plugin: Cheetah): this;
|
|
28
|
+
private findProviderInConfig;
|
|
29
|
+
private getProviderToken;
|
|
30
|
+
private shouldCloneProvider;
|
|
31
|
+
private cloneProvider;
|
|
32
|
+
/**
|
|
33
|
+
* Set the custom logger provider.
|
|
34
|
+
* The provider must be a class with the @Service() decorator.
|
|
35
|
+
* The provider must extend the LoggerService class.
|
|
36
|
+
*
|
|
37
|
+
* @param provider
|
|
38
|
+
*/
|
|
39
|
+
useLogger(provider: any): this;
|
|
40
|
+
private loadProvidersAndControllers;
|
|
41
|
+
private registerControllers;
|
|
42
|
+
private registerMetadataProviders;
|
|
43
|
+
private registerConfigProviders;
|
|
44
|
+
private normalizeProvider;
|
|
45
|
+
init(): Promise<void>;
|
|
46
|
+
listen(port?: number): Promise<void>;
|
|
47
|
+
private registerShutdownHandlers;
|
|
48
|
+
getHttpServer(): Server<any>;
|
|
49
|
+
getInjector(): import(".").InjectorService;
|
|
50
|
+
private createHttpServer;
|
|
51
|
+
private fetcher;
|
|
52
|
+
private catcher;
|
|
53
|
+
private bootstrapApplication;
|
|
54
|
+
private handleShutdownHook;
|
|
55
|
+
private closeHttpServer;
|
|
56
|
+
private exitProcess;
|
|
57
|
+
private reportHookFailure;
|
|
58
|
+
private isCorsEnabled;
|
|
59
|
+
private isOriginAllowed;
|
|
60
|
+
private buildCorsHeaders;
|
|
61
|
+
private handlePreflightRequest;
|
|
62
|
+
private applyCorsHeaders;
|
|
63
|
+
close(closeActiveConnections?: boolean): void;
|
|
64
|
+
private discoverRoutePath;
|
|
65
|
+
}
|