@eggjs/core 7.0.0-beta.20 → 7.0.0-beta.22
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/base_context_class.d.ts +16 -0
- package/dist/base_context_class.js +37 -0
- package/dist/egg.d.ts +246 -0
- package/dist/egg.js +388 -0
- package/dist/index.d.ts +12 -977
- package/dist/index.js +12 -2425
- package/dist/lifecycle.d.ts +78 -0
- package/dist/lifecycle.js +306 -0
- package/dist/loader/context_loader.d.ts +35 -0
- package/dist/loader/context_loader.js +102 -0
- package/dist/loader/egg_loader.d.ts +368 -0
- package/dist/loader/egg_loader.js +1572 -0
- package/dist/loader/file_loader.d.ts +101 -0
- package/dist/loader/file_loader.js +249 -0
- package/dist/singleton.d.ts +29 -0
- package/dist/singleton.js +117 -0
- package/dist/types.d.ts +53 -0
- package/dist/types.js +2 -0
- package/dist/utils/index.d.ts +17 -0
- package/dist/utils/index.js +113 -0
- package/dist/utils/sequencify.d.ts +13 -0
- package/dist/utils/sequencify.js +60 -0
- package/dist/utils/timing.d.ts +21 -0
- package/dist/utils/timing.js +92 -0
- package/package.json +9 -9
|
@@ -0,0 +1,78 @@
|
|
|
1
|
+
import { EventEmitter } from 'node:events';
|
|
2
|
+
import { type ReadyFunctionArg } from 'get-ready';
|
|
3
|
+
import { Ready } from 'ready-callback';
|
|
4
|
+
import { EggConsoleLogger } from 'egg-logger';
|
|
5
|
+
import type { Fun } from './utils/index.ts';
|
|
6
|
+
import type { EggCore } from './egg.ts';
|
|
7
|
+
export interface ILifecycleBoot {
|
|
8
|
+
fullPath?: string;
|
|
9
|
+
/**
|
|
10
|
+
* Ready to call configDidLoad,
|
|
11
|
+
* Config, plugin files are referred,
|
|
12
|
+
* this is the last chance to modify the config.
|
|
13
|
+
*/
|
|
14
|
+
configWillLoad?(): void;
|
|
15
|
+
/**
|
|
16
|
+
* Config, plugin files have loaded
|
|
17
|
+
*/
|
|
18
|
+
configDidLoad?(): void;
|
|
19
|
+
/**
|
|
20
|
+
* All files have loaded, start plugin here
|
|
21
|
+
*/
|
|
22
|
+
didLoad?(): Promise<void>;
|
|
23
|
+
/**
|
|
24
|
+
* All plugins have started, can do some thing before app ready
|
|
25
|
+
*/
|
|
26
|
+
willReady?(): Promise<void>;
|
|
27
|
+
/**
|
|
28
|
+
* Worker is ready, can do some things,
|
|
29
|
+
* don't need to block the app boot
|
|
30
|
+
*/
|
|
31
|
+
didReady?(err?: Error): Promise<void>;
|
|
32
|
+
/**
|
|
33
|
+
* Server is listening
|
|
34
|
+
*/
|
|
35
|
+
serverDidReady?(): Promise<void>;
|
|
36
|
+
/**
|
|
37
|
+
* Do some thing before app close
|
|
38
|
+
*/
|
|
39
|
+
beforeClose?(): Promise<void>;
|
|
40
|
+
}
|
|
41
|
+
export type BootImplClass<T = ILifecycleBoot> = new (...args: any[]) => T;
|
|
42
|
+
export interface LifecycleOptions {
|
|
43
|
+
baseDir: string;
|
|
44
|
+
app: EggCore;
|
|
45
|
+
logger: EggConsoleLogger;
|
|
46
|
+
}
|
|
47
|
+
export type FunWithFullPath = Fun & {
|
|
48
|
+
fullPath?: string;
|
|
49
|
+
};
|
|
50
|
+
export declare class Lifecycle extends EventEmitter {
|
|
51
|
+
#private;
|
|
52
|
+
loadReady: Ready;
|
|
53
|
+
bootReady: Ready;
|
|
54
|
+
options: LifecycleOptions;
|
|
55
|
+
readyTimeout: number;
|
|
56
|
+
constructor(options: Partial<LifecycleOptions>);
|
|
57
|
+
ready(): Promise<void>;
|
|
58
|
+
ready(flagOrFunction: ReadyFunctionArg): void;
|
|
59
|
+
get app(): EggCore;
|
|
60
|
+
get logger(): EggConsoleLogger<import("egg-logger").LoggerOptions>;
|
|
61
|
+
get timing(): import("./index.ts").Timing;
|
|
62
|
+
legacyReadyCallback(name: string, opt?: object): (...args: unknown[]) => void;
|
|
63
|
+
addBootHook(bootHootOrBootClass: BootImplClass | ILifecycleBoot): void;
|
|
64
|
+
addFunctionAsBootHook<T = EggCore>(hook: (app: T) => void, fullPath?: string): void;
|
|
65
|
+
/**
|
|
66
|
+
* init boots and trigger config did config
|
|
67
|
+
*/
|
|
68
|
+
init(): void;
|
|
69
|
+
registerBeforeStart(scope: Fun, name: string): void;
|
|
70
|
+
registerBeforeClose(fn: FunWithFullPath, fullPath?: string): void;
|
|
71
|
+
close(): Promise<void>;
|
|
72
|
+
triggerConfigWillLoad(): void;
|
|
73
|
+
triggerConfigDidLoad(): void;
|
|
74
|
+
triggerDidLoad(): void;
|
|
75
|
+
triggerWillReady(): void;
|
|
76
|
+
triggerDidReady(err?: Error): Promise<void>;
|
|
77
|
+
triggerServerDidReady(): Promise<void>;
|
|
78
|
+
}
|
|
@@ -0,0 +1,306 @@
|
|
|
1
|
+
import assert from 'node:assert';
|
|
2
|
+
import { EventEmitter } from 'node:events';
|
|
3
|
+
import { debuglog, format } from 'node:util';
|
|
4
|
+
import { isClass } from 'is-type-of';
|
|
5
|
+
import { Ready as ReadyObject } from 'get-ready';
|
|
6
|
+
import { Ready } from 'ready-callback';
|
|
7
|
+
import { EggConsoleLogger } from 'egg-logger';
|
|
8
|
+
import utils from "./utils/index.js";
|
|
9
|
+
const debug = debuglog('egg/core/lifecycle');
|
|
10
|
+
export class Lifecycle extends EventEmitter {
|
|
11
|
+
#init;
|
|
12
|
+
#readyObject;
|
|
13
|
+
#bootHooks;
|
|
14
|
+
#boots;
|
|
15
|
+
#isClosed;
|
|
16
|
+
#closeFunctionSet;
|
|
17
|
+
loadReady;
|
|
18
|
+
bootReady;
|
|
19
|
+
options;
|
|
20
|
+
readyTimeout;
|
|
21
|
+
constructor(options) {
|
|
22
|
+
super();
|
|
23
|
+
options.logger = options.logger ?? new EggConsoleLogger();
|
|
24
|
+
this.options = options;
|
|
25
|
+
this.#readyObject = new ReadyObject();
|
|
26
|
+
this.#bootHooks = [];
|
|
27
|
+
this.#boots = [];
|
|
28
|
+
this.#closeFunctionSet = new Set();
|
|
29
|
+
this.#isClosed = false;
|
|
30
|
+
this.#init = false;
|
|
31
|
+
this.timing.start(`${this.options.app.type} Start`);
|
|
32
|
+
// get app timeout from env or use default timeout 10 second
|
|
33
|
+
const eggReadyTimeoutEnv = Number.parseInt(process.env.EGG_READY_TIMEOUT_ENV || '10000');
|
|
34
|
+
assert(Number.isInteger(eggReadyTimeoutEnv), `process.env.EGG_READY_TIMEOUT_ENV ${process.env.EGG_READY_TIMEOUT_ENV} should be able to parseInt.`);
|
|
35
|
+
this.readyTimeout = eggReadyTimeoutEnv;
|
|
36
|
+
this.#initReady();
|
|
37
|
+
this.on('ready_stat', data => {
|
|
38
|
+
this.logger.info('[egg/core/lifecycle:ready_stat] end ready task %s, remain %j', data.id, data.remain);
|
|
39
|
+
}).on('ready_timeout', id => {
|
|
40
|
+
this.logger.warn('[egg/core/lifecycle:ready_timeout] %s seconds later %s was still unable to finish.', this.readyTimeout / 1000, id);
|
|
41
|
+
});
|
|
42
|
+
this.ready(err => {
|
|
43
|
+
this.triggerDidReady(err);
|
|
44
|
+
debug('app ready');
|
|
45
|
+
this.timing.end(`${this.options.app.type} Start`);
|
|
46
|
+
});
|
|
47
|
+
}
|
|
48
|
+
ready(flagOrFunction) {
|
|
49
|
+
if (flagOrFunction === undefined) {
|
|
50
|
+
return this.#readyObject.ready();
|
|
51
|
+
}
|
|
52
|
+
return this.#readyObject.ready(flagOrFunction);
|
|
53
|
+
}
|
|
54
|
+
get app() {
|
|
55
|
+
return this.options.app;
|
|
56
|
+
}
|
|
57
|
+
get logger() {
|
|
58
|
+
return this.options.logger;
|
|
59
|
+
}
|
|
60
|
+
get timing() {
|
|
61
|
+
return this.app.timing;
|
|
62
|
+
}
|
|
63
|
+
legacyReadyCallback(name, opt) {
|
|
64
|
+
const timingKeyPrefix = 'readyCallback';
|
|
65
|
+
const timing = this.timing;
|
|
66
|
+
const cb = this.loadReady.readyCallback(name, opt);
|
|
67
|
+
const timingKey = `${timingKeyPrefix} in ` + utils.getResolvedFilename(name, this.app.baseDir);
|
|
68
|
+
this.timing.start(timingKey);
|
|
69
|
+
debug('register legacyReadyCallback');
|
|
70
|
+
return function legacyReadyCallback(...args) {
|
|
71
|
+
timing.end(timingKey);
|
|
72
|
+
debug('end legacyReadyCallback');
|
|
73
|
+
cb(...args);
|
|
74
|
+
};
|
|
75
|
+
}
|
|
76
|
+
addBootHook(bootHootOrBootClass) {
|
|
77
|
+
assert(this.#init === false, 'do not add hook when lifecycle has been initialized');
|
|
78
|
+
this.#bootHooks.push(bootHootOrBootClass);
|
|
79
|
+
}
|
|
80
|
+
addFunctionAsBootHook(hook, fullPath) {
|
|
81
|
+
assert(this.#init === false, 'do not add hook when lifecycle has been initialized');
|
|
82
|
+
// app.js is exported as a function
|
|
83
|
+
// call this function in configDidLoad
|
|
84
|
+
class Boot {
|
|
85
|
+
static fullPath;
|
|
86
|
+
app;
|
|
87
|
+
constructor(app) {
|
|
88
|
+
this.app = app;
|
|
89
|
+
}
|
|
90
|
+
configDidLoad() {
|
|
91
|
+
hook(this.app);
|
|
92
|
+
}
|
|
93
|
+
}
|
|
94
|
+
Boot.fullPath = fullPath;
|
|
95
|
+
this.#bootHooks.push(Boot);
|
|
96
|
+
}
|
|
97
|
+
/**
|
|
98
|
+
* init boots and trigger config did config
|
|
99
|
+
*/
|
|
100
|
+
init() {
|
|
101
|
+
debug('%s init lifecycle', this.app.type);
|
|
102
|
+
assert(this.#init === false, 'lifecycle have been init');
|
|
103
|
+
this.#init = true;
|
|
104
|
+
this.#boots = this.#bootHooks.map(BootHootOrBootClass => {
|
|
105
|
+
let instance = BootHootOrBootClass;
|
|
106
|
+
if (isClass(BootHootOrBootClass)) {
|
|
107
|
+
instance = new BootHootOrBootClass(this.app);
|
|
108
|
+
if (!instance.fullPath && 'fullPath' in BootHootOrBootClass) {
|
|
109
|
+
instance.fullPath = BootHootOrBootClass.fullPath;
|
|
110
|
+
}
|
|
111
|
+
}
|
|
112
|
+
debug('[init] add boot instance: %o', instance.fullPath);
|
|
113
|
+
return instance;
|
|
114
|
+
});
|
|
115
|
+
}
|
|
116
|
+
registerBeforeStart(scope, name) {
|
|
117
|
+
debug('%s add registerBeforeStart, name: %o', this.options.app.type, name);
|
|
118
|
+
this.#registerReadyCallback({
|
|
119
|
+
scope,
|
|
120
|
+
ready: this.loadReady,
|
|
121
|
+
timingKeyPrefix: 'Before Start',
|
|
122
|
+
scopeFullName: name,
|
|
123
|
+
});
|
|
124
|
+
}
|
|
125
|
+
registerBeforeClose(fn, fullPath) {
|
|
126
|
+
assert(typeof fn === 'function', 'argument should be function');
|
|
127
|
+
assert(this.#isClosed === false, 'app has been closed');
|
|
128
|
+
if (fullPath) {
|
|
129
|
+
fn.fullPath = fullPath;
|
|
130
|
+
}
|
|
131
|
+
this.#closeFunctionSet.add(fn);
|
|
132
|
+
debug('%s register beforeClose at %o, count: %d', this.app.type, fullPath, this.#closeFunctionSet.size);
|
|
133
|
+
}
|
|
134
|
+
async close() {
|
|
135
|
+
// close in reverse order: first created, last closed
|
|
136
|
+
const closeFns = Array.from(this.#closeFunctionSet);
|
|
137
|
+
debug('%s start trigger %d beforeClose functions', this.app.type, closeFns.length);
|
|
138
|
+
for (const fn of closeFns.reverse()) {
|
|
139
|
+
debug('%s trigger beforeClose at %o', this.app.type, fn.fullPath);
|
|
140
|
+
await utils.callFn(fn);
|
|
141
|
+
this.#closeFunctionSet.delete(fn);
|
|
142
|
+
}
|
|
143
|
+
// Be called after other close callbacks
|
|
144
|
+
this.app.emit('close');
|
|
145
|
+
this.removeAllListeners();
|
|
146
|
+
this.app.removeAllListeners();
|
|
147
|
+
this.#isClosed = true;
|
|
148
|
+
debug('%s closed', this.app.type);
|
|
149
|
+
}
|
|
150
|
+
triggerConfigWillLoad() {
|
|
151
|
+
debug('trigger configWillLoad start');
|
|
152
|
+
for (const boot of this.#boots) {
|
|
153
|
+
if (typeof boot.configWillLoad === 'function') {
|
|
154
|
+
debug('trigger configWillLoad at %o', boot.fullPath);
|
|
155
|
+
boot.configWillLoad();
|
|
156
|
+
}
|
|
157
|
+
}
|
|
158
|
+
debug('trigger configWillLoad end');
|
|
159
|
+
this.triggerConfigDidLoad();
|
|
160
|
+
}
|
|
161
|
+
triggerConfigDidLoad() {
|
|
162
|
+
debug('trigger configDidLoad start');
|
|
163
|
+
for (const boot of this.#boots) {
|
|
164
|
+
if (typeof boot.configDidLoad === 'function') {
|
|
165
|
+
debug('trigger configDidLoad at %o', boot.fullPath);
|
|
166
|
+
boot.configDidLoad();
|
|
167
|
+
}
|
|
168
|
+
// function boot hook register after configDidLoad trigger
|
|
169
|
+
if (typeof boot.beforeClose === 'function') {
|
|
170
|
+
const beforeClose = boot.beforeClose.bind(boot);
|
|
171
|
+
this.registerBeforeClose(beforeClose, boot.fullPath);
|
|
172
|
+
}
|
|
173
|
+
}
|
|
174
|
+
debug('trigger configDidLoad end');
|
|
175
|
+
this.triggerDidLoad();
|
|
176
|
+
}
|
|
177
|
+
triggerDidLoad() {
|
|
178
|
+
debug('trigger didLoad start');
|
|
179
|
+
debug('loadReady start');
|
|
180
|
+
this.loadReady.start();
|
|
181
|
+
for (const boot of this.#boots) {
|
|
182
|
+
if (typeof boot.didLoad === 'function') {
|
|
183
|
+
const didLoad = boot.didLoad.bind(boot);
|
|
184
|
+
this.#registerReadyCallback({
|
|
185
|
+
scope: didLoad,
|
|
186
|
+
ready: this.loadReady,
|
|
187
|
+
timingKeyPrefix: 'Did Load',
|
|
188
|
+
scopeFullName: boot.fullPath + ':didLoad',
|
|
189
|
+
});
|
|
190
|
+
}
|
|
191
|
+
}
|
|
192
|
+
}
|
|
193
|
+
triggerWillReady() {
|
|
194
|
+
debug('trigger willReady start');
|
|
195
|
+
debug('bootReady start');
|
|
196
|
+
this.bootReady.start();
|
|
197
|
+
for (const boot of this.#boots) {
|
|
198
|
+
if (typeof boot.willReady === 'function') {
|
|
199
|
+
const willReady = boot.willReady.bind(boot);
|
|
200
|
+
this.#registerReadyCallback({
|
|
201
|
+
scope: willReady,
|
|
202
|
+
ready: this.bootReady,
|
|
203
|
+
timingKeyPrefix: 'Will Ready',
|
|
204
|
+
scopeFullName: boot.fullPath + ':willReady',
|
|
205
|
+
});
|
|
206
|
+
}
|
|
207
|
+
}
|
|
208
|
+
}
|
|
209
|
+
triggerDidReady(err) {
|
|
210
|
+
debug('trigger didReady start');
|
|
211
|
+
return (async () => {
|
|
212
|
+
for (const boot of this.#boots) {
|
|
213
|
+
if (typeof boot.didReady === 'function') {
|
|
214
|
+
debug('trigger didReady at %o', boot.fullPath);
|
|
215
|
+
try {
|
|
216
|
+
await boot.didReady(err);
|
|
217
|
+
}
|
|
218
|
+
catch (err) {
|
|
219
|
+
debug('trigger didReady error at %o, error: %s', boot.fullPath, err);
|
|
220
|
+
this.emit('error', err);
|
|
221
|
+
}
|
|
222
|
+
}
|
|
223
|
+
}
|
|
224
|
+
debug('trigger didReady end');
|
|
225
|
+
})();
|
|
226
|
+
}
|
|
227
|
+
triggerServerDidReady() {
|
|
228
|
+
debug('trigger serverDidReady start');
|
|
229
|
+
return (async () => {
|
|
230
|
+
for (const boot of this.#boots) {
|
|
231
|
+
if (typeof boot.serverDidReady !== 'function') {
|
|
232
|
+
continue;
|
|
233
|
+
}
|
|
234
|
+
debug('trigger serverDidReady at %o', boot.fullPath);
|
|
235
|
+
try {
|
|
236
|
+
await boot.serverDidReady();
|
|
237
|
+
}
|
|
238
|
+
catch (err) {
|
|
239
|
+
debug('trigger serverDidReady error at %o, error: %s', boot.fullPath, err);
|
|
240
|
+
this.emit('error', err);
|
|
241
|
+
}
|
|
242
|
+
}
|
|
243
|
+
debug('trigger serverDidReady end');
|
|
244
|
+
})();
|
|
245
|
+
}
|
|
246
|
+
#initReady() {
|
|
247
|
+
debug('loadReady init');
|
|
248
|
+
this.loadReady = new Ready({ timeout: this.readyTimeout, lazyStart: true });
|
|
249
|
+
this.#delegateReadyEvent(this.loadReady);
|
|
250
|
+
this.loadReady.ready((err) => {
|
|
251
|
+
debug('loadReady end, err: %o', err);
|
|
252
|
+
debug('trigger didLoad end');
|
|
253
|
+
if (err) {
|
|
254
|
+
this.ready(err);
|
|
255
|
+
}
|
|
256
|
+
else {
|
|
257
|
+
this.triggerWillReady();
|
|
258
|
+
}
|
|
259
|
+
});
|
|
260
|
+
debug('bootReady init');
|
|
261
|
+
this.bootReady = new Ready({ timeout: this.readyTimeout, lazyStart: true });
|
|
262
|
+
this.#delegateReadyEvent(this.bootReady);
|
|
263
|
+
this.bootReady.ready((err) => {
|
|
264
|
+
debug('bootReady end, err: %o', err);
|
|
265
|
+
debug('trigger willReady end');
|
|
266
|
+
this.ready(err || true);
|
|
267
|
+
});
|
|
268
|
+
}
|
|
269
|
+
#delegateReadyEvent(ready) {
|
|
270
|
+
ready.once('error', (err) => ready.ready(err));
|
|
271
|
+
ready.on('ready_timeout', (id) => this.emit('ready_timeout', id));
|
|
272
|
+
ready.on('ready_stat', (data) => this.emit('ready_stat', data));
|
|
273
|
+
ready.on('error', (err) => this.emit('error', err));
|
|
274
|
+
}
|
|
275
|
+
#registerReadyCallback(args) {
|
|
276
|
+
const { scope, ready, timingKeyPrefix, scopeFullName } = args;
|
|
277
|
+
if (typeof scope !== 'function') {
|
|
278
|
+
throw new TypeError('boot only support function');
|
|
279
|
+
}
|
|
280
|
+
// get filename from stack if scopeFullName is undefined
|
|
281
|
+
const name = scopeFullName || utils.getCalleeFromStack(true, 4);
|
|
282
|
+
const timingKey = `${timingKeyPrefix} in ` + utils.getResolvedFilename(name, this.app.baseDir);
|
|
283
|
+
this.timing.start(timingKey);
|
|
284
|
+
debug('[registerReadyCallback] start name: %o', name);
|
|
285
|
+
const done = ready.readyCallback(name);
|
|
286
|
+
// ensure scope executes after load completed
|
|
287
|
+
process.nextTick(async () => {
|
|
288
|
+
try {
|
|
289
|
+
await utils.callFn(scope);
|
|
290
|
+
debug('[registerReadyCallback] end name: %o', name);
|
|
291
|
+
done();
|
|
292
|
+
this.timing.end(timingKey);
|
|
293
|
+
}
|
|
294
|
+
catch (e) {
|
|
295
|
+
let err = e;
|
|
296
|
+
// avoid non-stringify error: TypeError: Cannot convert object to primitive value
|
|
297
|
+
if (!(err instanceof Error)) {
|
|
298
|
+
err = new Error(format('%s', err));
|
|
299
|
+
}
|
|
300
|
+
done(err);
|
|
301
|
+
this.timing.end(timingKey);
|
|
302
|
+
}
|
|
303
|
+
});
|
|
304
|
+
}
|
|
305
|
+
}
|
|
306
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibGlmZWN5Y2xlLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vc3JjL2xpZmVjeWNsZS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLE1BQU0sTUFBTSxhQUFhLENBQUM7QUFDakMsT0FBTyxFQUFFLFlBQVksRUFBRSxNQUFNLGFBQWEsQ0FBQztBQUMzQyxPQUFPLEVBQUUsUUFBUSxFQUFFLE1BQU0sRUFBRSxNQUFNLFdBQVcsQ0FBQztBQUU3QyxPQUFPLEVBQUUsT0FBTyxFQUFFLE1BQU0sWUFBWSxDQUFDO0FBQ3JDLE9BQU8sRUFBRSxLQUFLLElBQUksV0FBVyxFQUF5QixNQUFNLFdBQVcsQ0FBQztBQUN4RSxPQUFPLEVBQUUsS0FBSyxFQUFFLE1BQU0sZ0JBQWdCLENBQUM7QUFDdkMsT0FBTyxFQUFFLGdCQUFnQixFQUFFLE1BQU0sWUFBWSxDQUFDO0FBRTlDLE9BQU8sS0FBSyxNQUFNLGtCQUFrQixDQUFDO0FBSXJDLE1BQU0sS0FBSyxHQUFHLFFBQVEsQ0FBQyxvQkFBb0IsQ0FBQyxDQUFDO0FBc0Q3QyxNQUFNLE9BQU8sU0FBVSxTQUFRLFlBQVk7SUFDekMsS0FBSyxDQUFVO0lBQ2YsWUFBWSxDQUFjO0lBQzFCLFVBQVUsQ0FBcUM7SUFDL0MsTUFBTSxDQUFtQjtJQUN6QixTQUFTLENBQVU7SUFDbkIsaUJBQWlCLENBQXVCO0lBQ3hDLFNBQVMsQ0FBUTtJQUNqQixTQUFTLENBQVE7SUFDakIsT0FBTyxDQUFtQjtJQUMxQixZQUFZLENBQVM7SUFFckIsWUFBWSxPQUFrQztRQUM1QyxLQUFLLEVBQUUsQ0FBQztRQUNSLE9BQU8sQ0FBQyxNQUFNLEdBQUcsT0FBTyxDQUFDLE1BQU0sSUFBSSxJQUFJLGdCQUFnQixFQUFFLENBQUM7UUFDMUQsSUFBSSxDQUFDLE9BQU8sR0FBRyxPQUEyQixDQUFDO1FBQzNDLElBQUksQ0FBQyxZQUFZLEdBQUcsSUFBSSxXQUFXLEVBQUUsQ0FBQztRQUN0QyxJQUFJLENBQUMsVUFBVSxHQUFH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
|
|
@@ -0,0 +1,35 @@
|
|
|
1
|
+
import { FileLoader, type FileLoaderOptions } from './file_loader.ts';
|
|
2
|
+
import type { Context } from '../egg.ts';
|
|
3
|
+
export interface ClassLoaderOptions {
|
|
4
|
+
ctx: Context;
|
|
5
|
+
properties: any;
|
|
6
|
+
}
|
|
7
|
+
export declare class ClassLoader {
|
|
8
|
+
#private;
|
|
9
|
+
readonly _cache: Map<any, any>;
|
|
10
|
+
_ctx: Context;
|
|
11
|
+
constructor(options: ClassLoaderOptions);
|
|
12
|
+
}
|
|
13
|
+
export interface ContextLoaderOptions extends Omit<FileLoaderOptions, 'target'> {
|
|
14
|
+
/** required inject */
|
|
15
|
+
inject: Record<string, any>;
|
|
16
|
+
/** property name defined to target */
|
|
17
|
+
property: string | symbol;
|
|
18
|
+
/** determine the field name of inject object. */
|
|
19
|
+
fieldClass?: string;
|
|
20
|
+
}
|
|
21
|
+
/**
|
|
22
|
+
* Same as {@link FileLoader}, but it will attach file to `inject[fieldClass]`.
|
|
23
|
+
* The exports will be lazy loaded, such as `ctx.group.repository`.
|
|
24
|
+
* @augments FileLoader
|
|
25
|
+
* @since 1.0.0
|
|
26
|
+
*/
|
|
27
|
+
export declare class ContextLoader extends FileLoader {
|
|
28
|
+
#private;
|
|
29
|
+
/**
|
|
30
|
+
* @class
|
|
31
|
+
* @param {Object} options - options same as {@link FileLoader}
|
|
32
|
+
* @param {String} options.fieldClass - determine the field name of inject object.
|
|
33
|
+
*/
|
|
34
|
+
constructor(options: ContextLoaderOptions);
|
|
35
|
+
}
|
|
@@ -0,0 +1,102 @@
|
|
|
1
|
+
import assert from 'node:assert';
|
|
2
|
+
import { isClass, isPrimitive } from 'is-type-of';
|
|
3
|
+
import { FileLoader, EXPORTS } from "./file_loader.js";
|
|
4
|
+
const CLASS_LOADER = Symbol('classLoader');
|
|
5
|
+
export class ClassLoader {
|
|
6
|
+
_cache = new Map();
|
|
7
|
+
_ctx;
|
|
8
|
+
constructor(options) {
|
|
9
|
+
assert(options.ctx, 'options.ctx is required');
|
|
10
|
+
const properties = options.properties;
|
|
11
|
+
this._ctx = options.ctx;
|
|
12
|
+
for (const property in properties) {
|
|
13
|
+
this.#defineProperty(property, properties[property]);
|
|
14
|
+
}
|
|
15
|
+
}
|
|
16
|
+
#defineProperty(property, values) {
|
|
17
|
+
Object.defineProperty(this, property, {
|
|
18
|
+
get() {
|
|
19
|
+
let instance = this._cache.get(property);
|
|
20
|
+
if (!instance) {
|
|
21
|
+
instance = getInstance(values, this._ctx);
|
|
22
|
+
this._cache.set(property, instance);
|
|
23
|
+
}
|
|
24
|
+
return instance;
|
|
25
|
+
},
|
|
26
|
+
});
|
|
27
|
+
}
|
|
28
|
+
}
|
|
29
|
+
/**
|
|
30
|
+
* Same as {@link FileLoader}, but it will attach file to `inject[fieldClass]`.
|
|
31
|
+
* The exports will be lazy loaded, such as `ctx.group.repository`.
|
|
32
|
+
* @augments FileLoader
|
|
33
|
+
* @since 1.0.0
|
|
34
|
+
*/
|
|
35
|
+
export class ContextLoader extends FileLoader {
|
|
36
|
+
#inject;
|
|
37
|
+
/**
|
|
38
|
+
* @class
|
|
39
|
+
* @param {Object} options - options same as {@link FileLoader}
|
|
40
|
+
* @param {String} options.fieldClass - determine the field name of inject object.
|
|
41
|
+
*/
|
|
42
|
+
constructor(options) {
|
|
43
|
+
assert(options.property, 'options.property is required');
|
|
44
|
+
assert(options.inject, 'options.inject is required');
|
|
45
|
+
const target = {};
|
|
46
|
+
if (options.fieldClass) {
|
|
47
|
+
options.inject[options.fieldClass] = target;
|
|
48
|
+
}
|
|
49
|
+
super({
|
|
50
|
+
...options,
|
|
51
|
+
target,
|
|
52
|
+
});
|
|
53
|
+
this.#inject = this.options.inject;
|
|
54
|
+
const app = this.#inject;
|
|
55
|
+
const property = options.property;
|
|
56
|
+
// define ctx.service
|
|
57
|
+
Object.defineProperty(app.context, property, {
|
|
58
|
+
get() {
|
|
59
|
+
// oxlint-disable-next-line unicorn/no-this-assignment, typescript/no-this-alias
|
|
60
|
+
const ctx = this;
|
|
61
|
+
// distinguish property cache,
|
|
62
|
+
// cache's lifecycle is the same with this context instance
|
|
63
|
+
// e.x. ctx.service1 and ctx.service2 have different cache
|
|
64
|
+
if (!ctx[CLASS_LOADER]) {
|
|
65
|
+
ctx[CLASS_LOADER] = new Map();
|
|
66
|
+
}
|
|
67
|
+
const classLoader = ctx[CLASS_LOADER];
|
|
68
|
+
let instance = classLoader.get(property);
|
|
69
|
+
if (!instance) {
|
|
70
|
+
instance = getInstance(target, ctx);
|
|
71
|
+
classLoader.set(property, instance);
|
|
72
|
+
}
|
|
73
|
+
return instance;
|
|
74
|
+
},
|
|
75
|
+
});
|
|
76
|
+
}
|
|
77
|
+
}
|
|
78
|
+
function getInstance(values, ctx) {
|
|
79
|
+
// it's a directory when it has no exports
|
|
80
|
+
// then use ClassLoader
|
|
81
|
+
const Class = values[EXPORTS] ? values : null;
|
|
82
|
+
let instance;
|
|
83
|
+
if (Class) {
|
|
84
|
+
if (isClass(Class)) {
|
|
85
|
+
instance = new Class(ctx);
|
|
86
|
+
}
|
|
87
|
+
else {
|
|
88
|
+
// it's just an object
|
|
89
|
+
instance = Class;
|
|
90
|
+
}
|
|
91
|
+
// Can't set property to primitive, so check again
|
|
92
|
+
// e.x. module.exports = 1;
|
|
93
|
+
}
|
|
94
|
+
else if (isPrimitive(values)) {
|
|
95
|
+
instance = values;
|
|
96
|
+
}
|
|
97
|
+
else {
|
|
98
|
+
instance = new ClassLoader({ ctx, properties: values });
|
|
99
|
+
}
|
|
100
|
+
return instance;
|
|
101
|
+
}
|
|
102
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY29udGV4dF9sb2FkZXIuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvbG9hZGVyL2NvbnRleHRfbG9hZGVyLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sTUFBTSxNQUFNLGFBQWEsQ0FBQztBQUVqQyxPQUFPLEVBQUUsT0FBTyxFQUFFLFdBQVcsRUFBRSxNQUFNLFlBQVksQ0FBQztBQUVsRCxPQUFPLEVBQUUsVUFBVSxFQUFFLE9BQU8sRUFBMEIsTUFBTSxrQkFBa0IsQ0FBQztBQUcvRSxNQUFNLFlBQVksR0FBRyxNQUFNLENBQUMsYUFBYSxDQUFDLENBQUM7QUFPM0MsTUFBTSxPQUFPLFdBQVc7SUFDYixNQUFNLEdBQUcsSUFBSSxHQUFHLEVBQUUsQ0FBQztJQUM1QixJQUFJLENBQVU7SUFFZCxZQUFZLE9BQTJCO1FBQ3JDLE1BQU0sQ0FBQyxPQUFPLENBQUMsR0FBRyxFQUFFLHlCQUF5QixDQUFDLENBQUM7UUFDL0MsTUFBTSxVQUFVLEdBQUcsT0FBTyxDQUFDLFVBQVUsQ0FBQztRQUN0QyxJQUFJLENBQUMsSUFBSSxHQUFHLE9BQU8sQ0FBQyxHQUFHLENBQUM7UUFFeEIsS0FBSyxNQUFNLFFBQVEsSUFBSSxVQUFVLEVBQUUsQ0FBQztZQUNsQyxJQUFJLENBQUMsZUFBZSxDQUFDLFFBQVEsRUFBRSxVQUFVLENBQUMsUUFBUSxDQUFDLENBQUMsQ0FBQztRQUN2RCxDQUFDO0lBQ0gsQ0FBQztJQUVELGVBQWUsQ0FBQyxRQUFnQixFQUFFLE1BQVc7UUFDM0MsTUFBTSxDQUFDLGNBQWMsQ0FBQyxJQUFJLEVBQUUsUUFBUSxFQUFFO1lBQ3BDLEdBQUc7Z0JBQ0QsSUFBSSxRQUFRLEdBQVEsSUFBSSxDQUFDLE1BQU0sQ0FBQyxHQUFHLENBQUMsUUFBUSxDQUFDLENBQUM7Z0JBQzlDLElBQUksQ0FBQyxRQUFRLEVBQUUsQ0FBQztvQkFDZCxRQUFRLEdBQUcsV0FBVyxDQUFDLE1BQU0sRUFBRSxJQUFJLENBQUMsSUFB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
|