egg 4.1.0-beta.2 → 4.1.0-beta.21

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.
Files changed (86) hide show
  1. package/dist/agent.d.ts +3 -7
  2. package/dist/agent.js +6 -10
  3. package/dist/app/extend/context.d.ts +166 -193
  4. package/dist/app/extend/context.js +259 -244
  5. package/dist/app/extend/helper.d.ts +31 -36
  6. package/dist/app/extend/helper.js +41 -45
  7. package/dist/app/extend/request.d.ts +127 -140
  8. package/dist/app/extend/request.js +258 -219
  9. package/dist/app/extend/response.d.ts +22 -32
  10. package/dist/app/extend/response.js +34 -36
  11. package/dist/app/middleware/body_parser.d.ts +2 -2
  12. package/dist/app/middleware/body_parser.js +3 -7
  13. package/dist/app/middleware/meta.d.ts +8 -9
  14. package/dist/app/middleware/meta.js +19 -13
  15. package/dist/app/middleware/notfound.d.ts +5 -9
  16. package/dist/app/middleware/notfound.js +28 -25
  17. package/dist/app/middleware/override_method.d.ts +2 -2
  18. package/dist/app/middleware/override_method.js +3 -7
  19. package/dist/app/middleware/site_file.d.ts +7 -11
  20. package/dist/app/middleware/site_file.js +52 -37
  21. package/dist/config/config.default.d.ts +2 -8
  22. package/dist/config/config.default.js +376 -256
  23. package/dist/config/config.local.d.ts +5 -7
  24. package/dist/config/config.local.js +9 -6
  25. package/dist/config/config.unittest.d.ts +5 -7
  26. package/dist/config/config.unittest.js +8 -9
  27. package/dist/config/favicon.png +0 -0
  28. package/dist/config/plugin.d.ts +120 -122
  29. package/dist/config/plugin.js +121 -53
  30. package/dist/index.d.ts +66 -19
  31. package/dist/index.js +68 -19
  32. package/dist/lib/agent.d.ts +13 -16
  33. package/dist/lib/agent.js +46 -54
  34. package/dist/lib/application.d.ts +46 -61
  35. package/dist/lib/application.js +249 -197
  36. package/dist/lib/core/base_context_class.d.ts +12 -17
  37. package/dist/lib/core/base_context_class.js +15 -17
  38. package/dist/lib/core/base_context_logger.d.ts +35 -39
  39. package/dist/lib/core/base_context_logger.js +60 -58
  40. package/dist/lib/core/base_hook_class.d.ts +10 -17
  41. package/dist/lib/core/base_hook_class.js +22 -26
  42. package/dist/lib/core/context_httpclient.d.ts +16 -20
  43. package/dist/lib/core/context_httpclient.js +26 -29
  44. package/dist/lib/core/httpclient.d.ts +12 -15
  45. package/dist/lib/core/httpclient.js +37 -34
  46. package/dist/lib/core/logger.d.ts +3 -7
  47. package/dist/lib/core/logger.js +36 -30
  48. package/dist/lib/core/messenger/IMessenger.d.ts +49 -53
  49. package/dist/lib/core/messenger/IMessenger.js +2 -1
  50. package/dist/lib/core/messenger/base.d.ts +7 -11
  51. package/dist/lib/core/messenger/base.js +28 -29
  52. package/dist/lib/core/messenger/index.d.ts +4 -8
  53. package/dist/lib/core/messenger/index.js +8 -11
  54. package/dist/lib/core/messenger/ipc.d.ts +55 -59
  55. package/dist/lib/core/messenger/ipc.js +149 -120
  56. package/dist/lib/core/messenger/local.d.ts +56 -61
  57. package/dist/lib/core/messenger/local.js +126 -124
  58. package/dist/lib/core/utils.d.ts +2 -5
  59. package/dist/lib/core/utils.js +66 -44
  60. package/dist/lib/egg.d.ts +271 -292
  61. package/dist/lib/egg.js +576 -506
  62. package/dist/lib/error/CookieLimitExceedError.d.ts +4 -7
  63. package/dist/lib/error/CookieLimitExceedError.js +12 -15
  64. package/dist/lib/error/MessageUnhandledRejectionError.d.ts +4 -7
  65. package/dist/lib/error/MessageUnhandledRejectionError.js +12 -15
  66. package/dist/lib/error/index.d.ts +2 -3
  67. package/dist/lib/error/index.js +3 -4
  68. package/dist/lib/loader/AgentWorkerLoader.d.ts +7 -12
  69. package/dist/lib/loader/AgentWorkerLoader.js +18 -22
  70. package/dist/lib/loader/AppWorkerLoader.d.ts +12 -17
  71. package/dist/lib/loader/AppWorkerLoader.js +37 -35
  72. package/dist/lib/loader/EggApplicationLoader.d.ts +3 -7
  73. package/dist/lib/loader/EggApplicationLoader.js +4 -7
  74. package/dist/lib/loader/index.d.ts +3 -4
  75. package/dist/lib/loader/index.js +4 -5
  76. package/dist/lib/start.d.ts +18 -22
  77. package/dist/lib/start.js +42 -32
  78. package/dist/lib/types.d.ts +252 -230
  79. package/dist/lib/types.js +31 -14
  80. package/dist/lib/types.plugin.d.ts +12 -0
  81. package/dist/lib/types.plugin.js +14 -0
  82. package/dist/lib/utils.d.ts +2 -5
  83. package/dist/lib/utils.js +12 -12
  84. package/dist/urllib.d.ts +1 -2
  85. package/dist/urllib.js +2 -3
  86. package/package.json +31 -31
package/dist/lib/egg.js CHANGED
@@ -1,512 +1,582 @@
1
- import { BaseHookClass } from "./core/base_hook_class.js";
2
- import { BaseContextClass as BaseContextClass$1 } from "./core/base_context_class.js";
1
+ import { performance } from 'node:perf_hooks';
2
+ import path from 'node:path';
3
+ import fs from 'node:fs';
4
+ import http, {} from 'node:http';
5
+ import inspector from 'node:inspector';
6
+ import { AsyncLocalStorage } from 'node:async_hooks';
7
+ import { EggCore, Router } from '@eggjs/core';
8
+ import { utils as eggUtils } from '@eggjs/core';
9
+ import createClusterClient, { close as closeClusterClient,
10
+ // @ts-ignore Could not find a declaration file for module 'cluster-client'
11
+ } from 'cluster-client';
12
+ import { extend } from '@eggjs/extend2';
13
+ import { EggContextLogger as ContextLogger, EggLoggers, EggLogger } from 'egg-logger';
14
+ import { Cookies as ContextCookies } from '@eggjs/cookies';
15
+ // eslint-disable-next-line @typescript-eslint/ban-ts-comment
16
+ // @ts-ignore
17
+ import CircularJSON from 'circular-json-for-egg';
3
18
  import Context from "../app/extend/context.js";
4
- import { create } from "./core/messenger/index.js";
19
+ import Request from "../app/extend/request.js";
20
+ import Response from "../app/extend/response.js";
21
+ import { create as createMessenger } from "./core/messenger/index.js";
5
22
  import { ContextHttpClient } from "./core/context_httpclient.js";
6
- import { HttpClient } from "./core/httpclient.js";
23
+ import { HttpClient, } from "./core/httpclient.js";
7
24
  import { createLoggers } from "./core/logger.js";
8
25
  import { convertObject } from "./core/utils.js";
26
+ import { BaseContextClass } from "./core/base_context_class.js";
27
+ import { BaseHookClass } from "./core/base_hook_class.js";
9
28
  import { getSourceDirname } from "./utils.js";
10
- import { EggCore, Request as Request$1, Response as Response$1, Router, utils } from "@eggjs/core";
11
- import path from "node:path";
12
- import { performance } from "node:perf_hooks";
13
- import fs from "node:fs";
14
- import http from "node:http";
15
- import inspector from "node:inspector";
16
- import { AsyncLocalStorage } from "node:async_hooks";
17
- import createClusterClient, { close } from "cluster-client";
18
- import { extend } from "@eggjs/extend2";
19
- import { EggContextLogger, EggLogger, EggLoggers } from "egg-logger";
20
- import { Cookies } from "@eggjs/cookies";
21
- import CircularJSON from "circular-json-for-egg";
22
-
23
- //#region src/lib/egg.ts
24
- const EGG_PATH = Symbol.for("egg#eggPath");
25
- var Request = class extends Request$1 {};
26
- var Response = class extends Response$1 {};
29
+ const EGG_PATH = Symbol.for('egg#eggPath');
30
+ export { Request, Response };
31
+ // export egg classes
32
+ export { Context, Router };
27
33
  /**
28
- * Based on koa's Application
29
- * @see https://github.com/eggjs/egg-core
30
- * @see https://github.com/eggjs/koa/blob/master/src/application.ts
31
- * @augments EggCore
32
- */
33
- var EggApplicationCore = class extends EggCore {
34
- ContextCookies = Cookies;
35
- ContextLogger = EggContextLogger;
36
- ContextHttpClient = ContextHttpClient;
37
- HttpClient = HttpClient;
38
- HttpClientNext = HttpClient;
39
- /**
40
- * Retrieve base context class
41
- * @member {BaseContextClass} BaseContextClass
42
- * @since 1.0.0
43
- */
44
- BaseContextClass = BaseContextClass$1;
45
- /**
46
- * Retrieve base controller
47
- * @member {Controller} Controller
48
- * @since 1.0.0
49
- */
50
- Controller = BaseContextClass$1;
51
- /**
52
- * Retrieve base service
53
- * @member {Service} Service
54
- * @since 1.0.0
55
- */
56
- Service = BaseContextClass$1;
57
- /**
58
- * Retrieve base subscription
59
- * @member {Subscription} Subscription
60
- * @since 2.12.0
61
- */
62
- Subscription = BaseContextClass$1;
63
- /**
64
- * Retrieve base context class
65
- * @member {BaseHookClass} BaseHookClass
66
- */
67
- BaseHookClass = BaseHookClass;
68
- /**
69
- * Retrieve base boot
70
- * @member {Boot}
71
- * @alias BaseHookClass
72
- */
73
- Boot = BaseHookClass;
74
- #httpClient;
75
- #loggers;
76
- #clusterClients = [];
77
- messenger;
78
- agent;
79
- application;
80
- /**
81
- * @class
82
- * @param {Object} options
83
- * - {Object} [type] - type of instance, Agent and Application both extend koa, type can determine what it is.
84
- * - {String} [baseDir] - app root dir, default is `process.cwd()`
85
- * - {Object} [plugins] - custom plugin config, use it in unittest
86
- * - {String} [mode] - process mode, can be cluster / single, default is `cluster`
87
- */
88
- constructor(options) {
89
- options = {
90
- mode: "cluster",
91
- type: "application",
92
- baseDir: process.cwd(),
93
- ...options
94
- };
95
- super(options);
96
- /**
97
- * messenger instance
98
- * @member {Messenger}
99
- * @since 1.0.0
100
- */
101
- this.messenger = create(this);
102
- this.messenger.once("egg-ready", () => {
103
- this.lifecycle.triggerServerDidReady();
104
- });
105
- this.lifecycle.registerBeforeStart(async () => {
106
- await this.load();
107
- }, "load files");
108
- }
109
- /**
110
- * @deprecated please use `options` property instead
111
- */
112
- get _options() {
113
- return this.options;
114
- }
115
- async loadConfig() {
116
- await this.loader.loadConfig();
117
- }
118
- async load() {
119
- await this.loadConfig();
120
- this.ready(() => process.nextTick(() => {
121
- const dumpStartTime = Date.now();
122
- this.dumpConfig();
123
- this.dumpTiming();
124
- this.coreLogger.info("[egg] dump config after ready, %sms", Date.now() - dumpStartTime);
125
- }));
126
- this.#setupTimeoutTimer();
127
- this.console.info("[egg] App root: %s", this.baseDir);
128
- this.console.info("[egg] All *.log files save on %j", this.config.logger.dir);
129
- this.console.info("[egg] Loaded enabled plugin %j", this.loader.orderPlugins);
130
- this._unhandledRejectionHandler = this._unhandledRejectionHandler.bind(this);
131
- process.on("unhandledRejection", this._unhandledRejectionHandler);
132
- this.lifecycle.registerBeforeClose(async () => {
133
- for (const clusterClient of this.#clusterClients) await close(clusterClient);
134
- this.#clusterClients = [];
135
- if (this.type === "application" && this.options.mode === "single") await this.agent.close();
136
- for (const logger of this.loggers.values()) logger.close();
137
- this.messenger.close();
138
- process.removeListener("unhandledRejection", this._unhandledRejectionHandler);
139
- });
140
- await this.loader.load();
141
- }
142
- /**
143
- * Usage: new ApiClient({ cluster: app.cluster })
144
- */
145
- get cluster() {
146
- return this.clusterWrapper.bind(this);
147
- }
148
- /**
149
- * Wrap the Client with Leader/Follower Pattern
150
- *
151
- * @description almost the same as Agent.cluster API, the only different is that this method create Follower.
152
- *
153
- * @see https://github.com/node-modules/cluster-client
154
- * @param {Function} clientClass - client class function
155
- * @param {Object} [options]
156
- * - {Boolean} [autoGenerate] - whether generate delegate rule automatically, default is true
157
- * - {Function} [formatKey] - a method to transform the subscription info into a string,default is JSON.stringify
158
- * - {Object} [transcode|JSON.stringify/parse]
159
- * - {Function} encode - custom serialize method
160
- * - {Function} decode - custom deserialize method
161
- * - {Boolean} [isBroadcast] - whether broadcast subscription result to all followers or just one, default is true
162
- * - {Number} [responseTimeout] - response timeout, default is 3 seconds
163
- * - {Number} [maxWaitTime|30000] - leader startup max time, default is 30 seconds
164
- * @return {ClientWrapper} wrapper
165
- */
166
- clusterWrapper(clientClass, options) {
167
- const clientClassOptions = {
168
- ...this.config.clusterClient,
169
- ...options,
170
- singleMode: this.options.mode === "single",
171
- port: this.options.clusterPort,
172
- isLeader: this.type === "agent",
173
- logger: this.coreLogger,
174
- isCheckHeartbeat: this.config.env === "prod" ? true : inspector.url() === void 0
175
- };
176
- const client = createClusterClient(clientClass, clientClassOptions);
177
- this.#patchClusterClient(client);
178
- return client;
179
- }
180
- /**
181
- * print the information when console.log(app)
182
- * @return {Object} inspected app.
183
- * @since 1.0.0
184
- * @example
185
- * ```js
186
- * console.log(app);
187
- * =>
188
- * {
189
- * name: 'mock-app',
190
- * env: 'test',
191
- * subdomainOffset: 2,
192
- * config: '<egg config>',
193
- * controller: '<egg controller>',
194
- * service: '<egg service>',
195
- * middlewares: '<egg middlewares>',
196
- * urllib: '<egg urllib>',
197
- * loggers: '<egg loggers>'
198
- * }
199
- * ```
200
- */
201
- inspect() {
202
- const res = { env: this.config.env };
203
- function delegate(res$1, app, keys) {
204
- for (const key of keys) if (app[key]) res$1[key] = app[key];
205
- }
206
- function abbr(res$1, app, keys) {
207
- for (const key of keys) if (app[key]) res$1[key] = `<egg ${key}>`;
208
- }
209
- delegate(res, this, [
210
- "name",
211
- "baseDir",
212
- "subdomainOffset"
213
- ]);
214
- abbr(res, this, [
215
- "config",
216
- "controller",
217
- "httpclient",
218
- "loggers",
219
- "middlewares",
220
- "router",
221
- "serviceClasses"
222
- ]);
223
- return res;
224
- }
225
- toJSON() {
226
- return this.inspect();
227
- }
228
- /**
229
- * http request helper base on {@link httpclient}, it will auto save httpclient log.
230
- * Keep the same api with `httpclient.request(url, args)`.
231
- *
232
- * See https://github.com/node-modules/urllib#api-doc for more details.
233
- *
234
- * @param {String} url request url address.
235
- * @param {Object} options
236
- * - method {String} - Request method, defaults to GET. Could be GET, POST, DELETE or PUT. Alias 'type'.
237
- * - data {Object} - Data to be sent. Will be stringify automatically.
238
- * - dataType {String} - String - Type of response data. Could be `text` or `json`.
239
- * If it's `text`, the callback data would be a String.
240
- * If it's `json`, the data of callback would be a parsed JSON Object.
241
- * Default callback data would be a Buffer.
242
- * - headers {Object} - Request headers.
243
- * - timeout {Number} - Request timeout in milliseconds. Defaults to exports.TIMEOUT.
244
- * Include remote server connecting timeout and response timeout.
245
- * When timeout happen, will return ConnectionTimeout or ResponseTimeout.
246
- * - auth {String} - `username:password` used in HTTP Basic Authorization.
247
- * - followRedirect {Boolean} - follow HTTP 3xx responses as redirects. defaults to false.
248
- * - gzip {Boolean} - let you get the res object when request connected, default false. alias customResponse
249
- * - nestedQuerystring {Boolean} - urllib default use querystring to stringify form data which don't
250
- * support nested object, will use qs instead of querystring to support nested object by set this option to true.
251
- * - more options see https://github.com/node-modules/urllib
252
- * @return {Object}
253
- * - status {Number} - HTTP response status
254
- * - headers {Object} - HTTP response headers
255
- * - res {Object} - HTTP response meta
256
- * - data {Object} - HTTP response body
257
- *
258
- * @example
259
- * ```js
260
- * const result = await app.curl('http://example.com/foo.json', {
261
- * method: 'GET',
262
- * dataType: 'json',
263
- * });
264
- * console.log(result.status, result.headers, result.data);
265
- * ```
266
- */
267
- async curl(url, options) {
268
- return await this.httpClient.request(url, options);
269
- }
270
- /**
271
- * Create a new HttpClient instance with custom options
272
- * @param {Object} [options] HttpClient init options
273
- */
274
- createHttpClient(options) {
275
- return new this.HttpClient(this, options);
276
- }
277
- /**
278
- * HttpClient instance
279
- * @see https://github.com/node-modules/urllib
280
- * @member {HttpClient}
281
- */
282
- get httpClient() {
283
- if (!this.#httpClient) this.#httpClient = this.createHttpClient();
284
- return this.#httpClient;
285
- }
286
- /**
287
- * @deprecated please use httpClient instead
288
- * @alias httpClient
289
- * @member {HttpClient}
290
- */
291
- get httpclient() {
292
- return this.httpClient;
293
- }
294
- /**
295
- * All loggers contain logger, coreLogger and customLogger
296
- * @member {Object}
297
- * @since 1.0.0
298
- */
299
- get loggers() {
300
- if (!this.#loggers) this.#loggers = createLoggers(this);
301
- return this.#loggers;
302
- }
303
- /**
304
- * Get logger by name, it's equal to app.loggers['name'],
305
- * but you can extend it with your own logical.
306
- * @param {String} name - logger name
307
- * @return {Logger} logger
308
- */
309
- getLogger(name) {
310
- return this.loggers[name] || null;
311
- }
312
- /**
313
- * application logger, log file is `$HOME/logs/{appname}/{appname}-web`
314
- * @member {Logger}
315
- * @since 1.0.0
316
- */
317
- get logger() {
318
- return this.getLogger("logger");
319
- }
320
- /**
321
- * core logger for framework and plugins, log file is `$HOME/logs/{appname}/egg-web`
322
- * @member {Logger}
323
- * @since 1.0.0
324
- */
325
- get coreLogger() {
326
- return this.getLogger("coreLogger");
327
- }
328
- _unhandledRejectionHandler(err) {
329
- this.coreLogger.error("[egg:unhandledRejection] %s", err && err.message || err);
330
- if (!(err instanceof Error)) {
331
- const newError = new Error(String(err));
332
- if (err) {
333
- if (err.name) newError.name = err.name;
334
- if (err.message) newError.message = err.message;
335
- if (err.stack) newError.stack = err.stack;
336
- }
337
- err = newError;
338
- }
339
- if (err.name === "Error") err.name = "unhandledRejectionError";
340
- this.coreLogger.error(err);
341
- }
342
- /**
343
- * dump out the config and meta object
344
- * @private
345
- */
346
- dumpConfigToObject() {
347
- let ignoreList;
348
- try {
349
- ignoreList = Array.from(this.config.dump.ignore);
350
- } catch {
351
- ignoreList = [];
352
- }
353
- const config = extend(true, {}, {
354
- config: this.config,
355
- plugins: this.loader.allPlugins,
356
- appInfo: this.loader.appInfo
357
- });
358
- convertObject(config, ignoreList);
359
- return {
360
- config,
361
- meta: this.loader.configMeta
362
- };
363
- }
364
- /**
365
- * save app.config to `run/${type}_config.json`
366
- * @private
367
- */
368
- dumpConfig() {
369
- const rundir = this.config.rundir;
370
- try {
371
- if (!fs.existsSync(rundir)) fs.mkdirSync(rundir);
372
- const { config, meta } = this.dumpConfigToObject();
373
- const dumpFile = path.join(rundir, `${this.type}_config.json`);
374
- fs.writeFileSync(dumpFile, CircularJSON.stringify(config, null, 2));
375
- const dumpMetaFile = path.join(rundir, `${this.type}_config_meta.json`);
376
- fs.writeFileSync(dumpMetaFile, CircularJSON.stringify(meta, null, 2));
377
- } catch (err) {
378
- this.coreLogger.warn(`[egg] dumpConfig error: ${err.message}`);
379
- }
380
- }
381
- dumpTiming() {
382
- try {
383
- const items = this.timing.toJSON();
384
- const rundir = this.config.rundir;
385
- const dumpFile = path.join(rundir, `${this.type}_timing_${process.pid}.json`);
386
- fs.writeFileSync(dumpFile, CircularJSON.stringify(items, null, 2));
387
- this.coreLogger.info(this.timing.toString());
388
- this.timing.disable();
389
- for (const item of items) if (item.index > 0 && item.duration && item.duration >= this.config.dump.timing.slowBootActionMinDuration) this.coreLogger.warn("[egg][dumpTiming][slow-boot-action] #%d %dms, name: %s", item.index, item.duration, item.name);
390
- } catch (err) {
391
- this.coreLogger.warn(`[egg] dumpTiming error: ${err.message}`);
392
- }
393
- }
394
- get [EGG_PATH]() {
395
- return getSourceDirname();
396
- }
397
- #setupTimeoutTimer() {
398
- const startTimeoutTimer = setTimeout(() => {
399
- this.coreLogger.error(this.timing.toString());
400
- this.coreLogger.error(`${this.type} still doesn't ready after ${this.config.workerStartTimeout} ms.`);
401
- const items = this.timing.toJSON();
402
- for (const item of items) {
403
- if (item.end) continue;
404
- this.coreLogger.error(`unfinished timing item: ${CircularJSON.stringify(item)}`);
405
- }
406
- this.coreLogger.error("[egg][setupTimeoutTimer] check run/%s_timing_%s.json for more details.", this.type, process.pid);
407
- this.emit("startTimeout");
408
- this.dumpConfig();
409
- this.dumpTiming();
410
- }, this.config.workerStartTimeout);
411
- this.ready(() => clearTimeout(startTimeoutTimer));
412
- }
413
- get config() {
414
- return super.config;
415
- }
416
- /**
417
- * app.env delegate app.config.env
418
- * @deprecated
419
- */
420
- get env() {
421
- this.deprecate("please use app.config.env instead");
422
- return this.config.env;
423
- }
424
- set env(_) {}
425
- /**
426
- * app.proxy delegate app.config.proxy
427
- * @deprecated
428
- */
429
- get proxy() {
430
- return this.config.proxy;
431
- }
432
- set proxy(_) {}
433
- #patchClusterClient(client) {
434
- const rawCreate = client.create;
435
- client.create = (...args) => {
436
- const realClient = rawCreate.apply(client, args);
437
- this.#clusterClients.push(realClient);
438
- return realClient;
439
- };
440
- }
441
- /**
442
- * Create an anonymous context, the context isn't request level, so the request is mocked.
443
- * then you can use context level API like `ctx.service`
444
- * @member {String} EggApplication#createAnonymousContext
445
- * @param {Request} [req] - if you want to mock request like querystring, you can pass an object to this function.
446
- * @return {Context} context
447
- */
448
- createAnonymousContext(req) {
449
- const request = {
450
- headers: {
451
- host: "127.0.0.1",
452
- "x-forwarded-for": "127.0.0.1"
453
- },
454
- query: {},
455
- querystring: "",
456
- host: "127.0.0.1",
457
- hostname: "127.0.0.1",
458
- protocol: "http",
459
- secure: "false",
460
- method: "GET",
461
- url: "/",
462
- path: "/",
463
- socket: {
464
- remoteAddress: "127.0.0.1",
465
- remotePort: 7001
466
- }
467
- };
468
- if (req) for (const key in req) if (key === "headers" || key === "query" || key === "socket") Object.assign(request[key], req[key]);
469
- else request[key] = req[key];
470
- const response = new http.ServerResponse(request);
471
- return this.createContext(request, response);
472
- }
473
- /**
474
- * Run async function in the anonymous context scope
475
- * @see Context#runInAnonymousContextScope
476
- * @param {Function} scope - the first args is an anonymous ctx, scope should be async function
477
- * @param {Request} [req] - if you want to mock request like querystring, you can pass an object to this function.
478
- */
479
- async runInAnonymousContextScope(scope, req) {
480
- const ctx = this.createAnonymousContext(req);
481
- if (!scope.name) Reflect.set(scope, "_name", utils.getCalleeFromStack(true));
482
- return await this.ctxStorage.run(ctx, async () => {
483
- return await scope(ctx);
484
- });
485
- }
486
- /**
487
- * Create egg context
488
- * @function EggApplication#createContext
489
- * @param {Req} req - node native Request object
490
- * @param {Res} res - node native Response object
491
- * @return {Context} context object
492
- */
493
- createContext(req, res) {
494
- const context = Object.create(this.context);
495
- const request = context.request = Object.create(this.request);
496
- const response = context.response = Object.create(this.response);
497
- context.app = request.app = response.app = this;
498
- context.req = request.req = response.req = req;
499
- context.res = request.res = response.res = res;
500
- request.ctx = response.ctx = context;
501
- request.response = response;
502
- response.request = request;
503
- context.onerror = context.onerror.bind(context);
504
- context.originalUrl = request.originalUrl = req.url;
505
- context.starttime = Date.now();
506
- context.performanceStarttime = performance.now();
507
- return context;
508
- }
509
- };
510
-
511
- //#endregion
512
- export { Context, EggApplicationCore, Request, Response, Router };
34
+ * Based on koa's Application
35
+ * @see https://github.com/eggjs/egg-core
36
+ * @see https://github.com/eggjs/koa/blob/master/src/application.ts
37
+ * @augments EggCore
38
+ */
39
+ export class EggApplicationCore extends EggCore {
40
+ /**
41
+ * Get the current request context from AsyncLocalStorage.
42
+ * This provides access to the context object for the current request lifecycle.
43
+ * @returns {Context | undefined} The current request context, or undefined if not in a request scope.
44
+ */
45
+ get currentContext() {
46
+ return this.ctxStorage.getStore();
47
+ }
48
+ // export context base classes, let framework can impl sub class and over context extend easily.
49
+ ContextCookies = ContextCookies;
50
+ ContextLogger = ContextLogger;
51
+ ContextHttpClient = ContextHttpClient;
52
+ HttpClient = HttpClient;
53
+ // keep compatible with egg version 3.x
54
+ HttpClientNext = HttpClient;
55
+ /**
56
+ * Retrieve base context class
57
+ * @member {BaseContextClass} BaseContextClass
58
+ * @since 1.0.0
59
+ */
60
+ BaseContextClass = BaseContextClass;
61
+ /**
62
+ * Retrieve base controller
63
+ * @member {Controller} Controller
64
+ * @since 1.0.0
65
+ */
66
+ Controller = BaseContextClass;
67
+ /**
68
+ * Retrieve base service
69
+ * @member {Service} Service
70
+ * @since 1.0.0
71
+ */
72
+ Service = BaseContextClass;
73
+ /**
74
+ * Retrieve base subscription
75
+ * @member {Subscription} Subscription
76
+ * @since 2.12.0
77
+ */
78
+ Subscription = BaseContextClass;
79
+ /**
80
+ * Retrieve base context class
81
+ * @member {BaseHookClass} BaseHookClass
82
+ */
83
+ BaseHookClass = BaseHookClass;
84
+ /**
85
+ * Retrieve base boot
86
+ * @member {Boot}
87
+ * @alias BaseHookClass
88
+ */
89
+ Boot = BaseHookClass;
90
+ #httpClient;
91
+ #loggers;
92
+ #clusterClients = [];
93
+ messenger;
94
+ agent;
95
+ application;
96
+ /**
97
+ * @class
98
+ * @param {Object} options
99
+ * - {Object} [type] - type of instance, Agent and Application both extend koa, type can determine what it is.
100
+ * - {String} [baseDir] - app root dir, default is `process.cwd()`
101
+ * - {Object} [plugins] - custom plugin config, use it in unittest
102
+ * - {String} [mode] - process mode, can be cluster / single, default is `cluster`
103
+ */
104
+ constructor(options) {
105
+ options = {
106
+ mode: 'cluster',
107
+ type: 'application',
108
+ baseDir: process.cwd(),
109
+ ...options,
110
+ };
111
+ super(options);
112
+ /**
113
+ * messenger instance
114
+ * @member {Messenger}
115
+ * @since 1.0.0
116
+ */
117
+ this.messenger = createMessenger(this);
118
+ // trigger `serverDidReady` hook when all the app workers
119
+ // and agent worker are ready
120
+ this.messenger.once('egg-ready', () => {
121
+ this.lifecycle.triggerServerDidReady();
122
+ });
123
+ this.lifecycle.registerBeforeStart(async () => {
124
+ await this.load();
125
+ }, 'load files');
126
+ }
127
+ /**
128
+ * @deprecated please use `options` property instead
129
+ */
130
+ get _options() {
131
+ return this.options;
132
+ }
133
+ async loadConfig() {
134
+ await this.loader.loadConfig();
135
+ }
136
+ async load() {
137
+ await this.loadConfig();
138
+ // dump config after ready, ensure all the modifications during start will be recorded
139
+ // make sure dumpConfig is the last ready callback
140
+ this.ready(() => process.nextTick(() => {
141
+ const dumpStartTime = Date.now();
142
+ this.dumpConfig();
143
+ this.dumpTiming();
144
+ this.coreLogger.info('[egg] dump config after ready, %sms', Date.now() - dumpStartTime);
145
+ }));
146
+ this.#setupTimeoutTimer();
147
+ this.console.info('[egg] App root: %s', this.baseDir);
148
+ this.console.info('[egg] All *.log files save on %j', this.config.logger.dir);
149
+ this.console.info('[egg] Loaded enabled plugin %j', this.loader.orderPlugins);
150
+ // Listen the error that promise had not catch, then log it in common-error
151
+ this._unhandledRejectionHandler = this._unhandledRejectionHandler.bind(this);
152
+ process.on('unhandledRejection', this._unhandledRejectionHandler);
153
+ // register close function
154
+ this.lifecycle.registerBeforeClose(async () => {
155
+ // close all cluster clients
156
+ for (const clusterClient of this.#clusterClients) {
157
+ await closeClusterClient(clusterClient);
158
+ }
159
+ this.#clusterClients = [];
160
+ // single process mode will close agent before app close
161
+ if (this.type === 'application' && this.options.mode === 'single') {
162
+ await this.agent.close();
163
+ }
164
+ for (const logger of this.loggers.values()) {
165
+ logger.close();
166
+ }
167
+ this.messenger.close();
168
+ process.removeListener('unhandledRejection', this._unhandledRejectionHandler);
169
+ });
170
+ await this.loader.load();
171
+ }
172
+ /**
173
+ * Usage: new ApiClient({ cluster: app.cluster })
174
+ */
175
+ get cluster() {
176
+ return this.clusterWrapper.bind(this);
177
+ }
178
+ /**
179
+ * Wrap the Client with Leader/Follower Pattern
180
+ *
181
+ * @description almost the same as Agent.cluster API, the only different is that this method create Follower.
182
+ *
183
+ * @see https://github.com/node-modules/cluster-client
184
+ * @param {Function} clientClass - client class function
185
+ * @param {Object} [options]
186
+ * - {Boolean} [autoGenerate] - whether generate delegate rule automatically, default is true
187
+ * - {Function} [formatKey] - a method to transform the subscription info into a string,default is JSON.stringify
188
+ * - {Object} [transcode|JSON.stringify/parse]
189
+ * - {Function} encode - custom serialize method
190
+ * - {Function} decode - custom deserialize method
191
+ * - {Boolean} [isBroadcast] - whether broadcast subscription result to all followers or just one, default is true
192
+ * - {Number} [responseTimeout] - response timeout, default is 3 seconds
193
+ * - {Number} [maxWaitTime|30000] - leader startup max time, default is 30 seconds
194
+ * @return {ClientWrapper} wrapper
195
+ */
196
+ clusterWrapper(clientClass, options) {
197
+ const clientClassOptions = {
198
+ ...this.config.clusterClient,
199
+ ...options,
200
+ singleMode: this.options.mode === 'single',
201
+ // cluster need a port that can't conflict on the environment
202
+ port: this.options.clusterPort,
203
+ // agent worker is leader, app workers are follower
204
+ isLeader: this.type === 'agent',
205
+ logger: this.coreLogger,
206
+ // debug mode does not check heartbeat
207
+ isCheckHeartbeat: this.config.env === 'prod' ? true : inspector.url() === undefined,
208
+ };
209
+ const client = createClusterClient(clientClass, clientClassOptions);
210
+ this.#patchClusterClient(client);
211
+ return client;
212
+ }
213
+ /**
214
+ * print the information when console.log(app)
215
+ * @return {Object} inspected app.
216
+ * @since 1.0.0
217
+ * @example
218
+ * ```js
219
+ * console.log(app);
220
+ * =>
221
+ * {
222
+ * name: 'mock-app',
223
+ * env: 'test',
224
+ * subdomainOffset: 2,
225
+ * config: '<egg config>',
226
+ * controller: '<egg controller>',
227
+ * service: '<egg service>',
228
+ * middlewares: '<egg middlewares>',
229
+ * urllib: '<egg urllib>',
230
+ * loggers: '<egg loggers>'
231
+ * }
232
+ * ```
233
+ */
234
+ inspect() {
235
+ const res = {
236
+ env: this.config.env,
237
+ };
238
+ function delegate(res, app, keys) {
239
+ for (const key of keys) {
240
+ if (app[key]) {
241
+ res[key] = app[key];
242
+ }
243
+ }
244
+ }
245
+ function abbr(res, app, keys) {
246
+ for (const key of keys) {
247
+ if (app[key]) {
248
+ res[key] = `<egg ${key}>`;
249
+ }
250
+ }
251
+ }
252
+ delegate(res, this, ['name', 'baseDir', 'subdomainOffset']);
253
+ abbr(res, this, ['config', 'controller', 'httpclient', 'loggers', 'middlewares', 'router', 'serviceClasses']);
254
+ return res;
255
+ }
256
+ toJSON() {
257
+ return this.inspect();
258
+ }
259
+ /**
260
+ * http request helper base on {@link httpclient}, it will auto save httpclient log.
261
+ * Keep the same api with `httpclient.request(url, args)`.
262
+ *
263
+ * See https://github.com/node-modules/urllib#api-doc for more details.
264
+ *
265
+ * @param {String} url request url address.
266
+ * @param {Object} options
267
+ * - method {String} - Request method, defaults to GET. Could be GET, POST, DELETE or PUT. Alias 'type'.
268
+ * - data {Object} - Data to be sent. Will be stringify automatically.
269
+ * - dataType {String} - String - Type of response data. Could be `text` or `json`.
270
+ * If it's `text`, the callback data would be a String.
271
+ * If it's `json`, the data of callback would be a parsed JSON Object.
272
+ * Default callback data would be a Buffer.
273
+ * - headers {Object} - Request headers.
274
+ * - timeout {Number} - Request timeout in milliseconds. Defaults to exports.TIMEOUT.
275
+ * Include remote server connecting timeout and response timeout.
276
+ * When timeout happen, will return ConnectionTimeout or ResponseTimeout.
277
+ * - auth {String} - `username:password` used in HTTP Basic Authorization.
278
+ * - followRedirect {Boolean} - follow HTTP 3xx responses as redirects. defaults to false.
279
+ * - gzip {Boolean} - let you get the res object when request connected, default false. alias customResponse
280
+ * - nestedQuerystring {Boolean} - urllib default use querystring to stringify form data which don't
281
+ * support nested object, will use qs instead of querystring to support nested object by set this option to true.
282
+ * - more options see https://github.com/node-modules/urllib
283
+ * @return {Object}
284
+ * - status {Number} - HTTP response status
285
+ * - headers {Object} - HTTP response headers
286
+ * - res {Object} - HTTP response meta
287
+ * - data {Object} - HTTP response body
288
+ *
289
+ * @example
290
+ * ```js
291
+ * const result = await app.curl('http://example.com/foo.json', {
292
+ * method: 'GET',
293
+ * dataType: 'json',
294
+ * });
295
+ * console.log(result.status, result.headers, result.data);
296
+ * ```
297
+ */
298
+ async curl(url, options) {
299
+ return await this.httpClient.request(url, options);
300
+ }
301
+ /**
302
+ * Create a new HttpClient instance with custom options
303
+ * @param {Object} [options] HttpClient init options
304
+ */
305
+ createHttpClient(options) {
306
+ return new this.HttpClient(this, options);
307
+ }
308
+ /**
309
+ * HttpClient instance
310
+ * @see https://github.com/node-modules/urllib
311
+ * @member {HttpClient}
312
+ */
313
+ get httpClient() {
314
+ if (!this.#httpClient) {
315
+ this.#httpClient = this.createHttpClient();
316
+ }
317
+ return this.#httpClient;
318
+ }
319
+ /**
320
+ * @deprecated please use httpClient instead
321
+ * @alias httpClient
322
+ * @member {HttpClient}
323
+ */
324
+ get httpclient() {
325
+ return this.httpClient;
326
+ }
327
+ /**
328
+ * All loggers contain logger, coreLogger and customLogger
329
+ * @member {Object}
330
+ * @since 1.0.0
331
+ */
332
+ get loggers() {
333
+ if (!this.#loggers) {
334
+ this.#loggers = createLoggers(this);
335
+ }
336
+ return this.#loggers;
337
+ }
338
+ /**
339
+ * Get logger by name, it's equal to app.loggers['name'],
340
+ * but you can extend it with your own logical.
341
+ * @param {String} name - logger name
342
+ * @return {Logger} logger
343
+ */
344
+ getLogger(name) {
345
+ return this.loggers[name] || null;
346
+ }
347
+ /**
348
+ * application logger, log file is `$HOME/logs/{appname}/{appname}-web`
349
+ * @member {Logger}
350
+ * @since 1.0.0
351
+ */
352
+ get logger() {
353
+ return this.getLogger('logger');
354
+ }
355
+ /**
356
+ * core logger for framework and plugins, log file is `$HOME/logs/{appname}/egg-web`
357
+ * @member {Logger}
358
+ * @since 1.0.0
359
+ */
360
+ get coreLogger() {
361
+ return this.getLogger('coreLogger');
362
+ }
363
+ _unhandledRejectionHandler(err) {
364
+ this.coreLogger.error('[egg:unhandledRejection] %s', (err && err.message) || err);
365
+ if (!(err instanceof Error)) {
366
+ const newError = new Error(String(err));
367
+ // err maybe an object, try to copy the name, message and stack to the new error instance
368
+ if (err) {
369
+ if (err.name)
370
+ newError.name = err.name;
371
+ if (err.message)
372
+ newError.message = err.message;
373
+ if (err.stack)
374
+ newError.stack = err.stack;
375
+ }
376
+ err = newError;
377
+ }
378
+ if (err.name === 'Error') {
379
+ err.name = 'unhandledRejectionError';
380
+ }
381
+ this.coreLogger.error(err);
382
+ }
383
+ /**
384
+ * dump out the config and meta object
385
+ * @private
386
+ */
387
+ dumpConfigToObject() {
388
+ let ignoreList;
389
+ try {
390
+ // support array and set
391
+ ignoreList = Array.from(this.config.dump.ignore);
392
+ }
393
+ catch {
394
+ ignoreList = [];
395
+ }
396
+ const config = extend(true, {}, {
397
+ config: this.config,
398
+ plugins: this.loader.allPlugins,
399
+ appInfo: this.loader.appInfo,
400
+ });
401
+ convertObject(config, ignoreList);
402
+ return {
403
+ config,
404
+ meta: this.loader.configMeta,
405
+ };
406
+ }
407
+ /**
408
+ * save app.config to `run/${type}_config.json`
409
+ * @private
410
+ */
411
+ dumpConfig() {
412
+ const rundir = this.config.rundir;
413
+ try {
414
+ if (!fs.existsSync(rundir)) {
415
+ fs.mkdirSync(rundir);
416
+ }
417
+ // get dumped object
418
+ const { config, meta } = this.dumpConfigToObject();
419
+ // dump config
420
+ const dumpFile = path.join(rundir, `${this.type}_config.json`);
421
+ fs.writeFileSync(dumpFile, CircularJSON.stringify(config, null, 2));
422
+ // dump config meta
423
+ const dumpMetaFile = path.join(rundir, `${this.type}_config_meta.json`);
424
+ fs.writeFileSync(dumpMetaFile, CircularJSON.stringify(meta, null, 2));
425
+ }
426
+ catch (err) {
427
+ this.coreLogger.warn(`[egg] dumpConfig error: ${err.message}`);
428
+ }
429
+ }
430
+ dumpTiming() {
431
+ try {
432
+ const items = this.timing.toJSON();
433
+ const rundir = this.config.rundir;
434
+ const dumpFile = path.join(rundir, `${this.type}_timing_${process.pid}.json`);
435
+ fs.writeFileSync(dumpFile, CircularJSON.stringify(items, null, 2));
436
+ this.coreLogger.info(this.timing.toString());
437
+ // only disable, not clear bootstrap timing data.
438
+ this.timing.disable();
439
+ // show duration >= ${slowBootActionMinDuration}ms action to warning log
440
+ for (const item of items) {
441
+ // ignore #0 name: Process Start
442
+ if (item.index > 0 && item.duration && item.duration >= this.config.dump.timing.slowBootActionMinDuration) {
443
+ this.coreLogger.warn('[egg][dumpTiming][slow-boot-action] #%d %dms, name: %s', item.index, item.duration, item.name);
444
+ }
445
+ }
446
+ }
447
+ catch (err) {
448
+ this.coreLogger.warn(`[egg] dumpTiming error: ${err.message}`);
449
+ }
450
+ }
451
+ get [EGG_PATH]() {
452
+ return getSourceDirname();
453
+ }
454
+ #setupTimeoutTimer() {
455
+ const startTimeoutTimer = setTimeout(() => {
456
+ this.coreLogger.error(this.timing.toString());
457
+ this.coreLogger.error(`${this.type} still doesn't ready after ${this.config.workerStartTimeout} ms.`);
458
+ // log unfinished
459
+ const items = this.timing.toJSON();
460
+ for (const item of items) {
461
+ if (item.end)
462
+ continue;
463
+ this.coreLogger.error(`unfinished timing item: ${CircularJSON.stringify(item)}`);
464
+ }
465
+ this.coreLogger.error('[egg][setupTimeoutTimer] check run/%s_timing_%s.json for more details.', this.type, process.pid);
466
+ this.emit('startTimeout');
467
+ this.dumpConfig();
468
+ this.dumpTiming();
469
+ }, this.config.workerStartTimeout);
470
+ this.ready(() => clearTimeout(startTimeoutTimer));
471
+ }
472
+ get config() {
473
+ return super.config;
474
+ }
475
+ /**
476
+ * app.env delegate app.config.env
477
+ * @deprecated
478
+ */
479
+ get env() {
480
+ this.deprecate('please use app.config.env instead');
481
+ return this.config.env;
482
+ }
483
+ /* eslint no-empty-function: off */
484
+ set env(_) { }
485
+ /**
486
+ * app.proxy delegate app.config.proxy
487
+ * @deprecated
488
+ */
489
+ get proxy() {
490
+ // this.deprecate('please use app.config.proxy instead');
491
+ return this.config.proxy;
492
+ }
493
+ /* eslint no-empty-function: off */
494
+ set proxy(_) { }
495
+ #patchClusterClient(client) {
496
+ const rawCreate = client.create;
497
+ client.create = (...args) => {
498
+ const realClient = rawCreate.apply(client, args);
499
+ this.#clusterClients.push(realClient);
500
+ return realClient;
501
+ };
502
+ }
503
+ /**
504
+ * Create an anonymous context, the context isn't request level, so the request is mocked.
505
+ * then you can use context level API like `ctx.service`
506
+ * @member {String} EggApplication#createAnonymousContext
507
+ * @param {Request} [req] - if you want to mock request like querystring, you can pass an object to this function.
508
+ * @return {Context} context
509
+ */
510
+ createAnonymousContext(req) {
511
+ const request = {
512
+ headers: {
513
+ host: '127.0.0.1',
514
+ 'x-forwarded-for': '127.0.0.1',
515
+ },
516
+ query: {},
517
+ querystring: '',
518
+ host: '127.0.0.1',
519
+ hostname: '127.0.0.1',
520
+ protocol: 'http',
521
+ secure: 'false',
522
+ method: 'GET',
523
+ url: '/',
524
+ path: '/',
525
+ socket: {
526
+ remoteAddress: '127.0.0.1',
527
+ remotePort: 7001,
528
+ },
529
+ };
530
+ if (req) {
531
+ for (const key in req) {
532
+ if (key === 'headers' || key === 'query' || key === 'socket') {
533
+ Object.assign(request[key], req[key]);
534
+ }
535
+ else {
536
+ request[key] = req[key];
537
+ }
538
+ }
539
+ }
540
+ const response = new http.ServerResponse(request);
541
+ return this.createContext(request, response);
542
+ }
543
+ /**
544
+ * Run async function in the anonymous context scope
545
+ * @see Context#runInAnonymousContextScope
546
+ * @param {Function} scope - the first args is an anonymous ctx, scope should be async function
547
+ * @param {Request} [req] - if you want to mock request like querystring, you can pass an object to this function.
548
+ */
549
+ async runInAnonymousContextScope(scope, req) {
550
+ const ctx = this.createAnonymousContext(req);
551
+ if (!scope.name) {
552
+ Reflect.set(scope, '_name', eggUtils.getCalleeFromStack(true));
553
+ }
554
+ return await this.ctxStorage.run(ctx, async () => {
555
+ return await scope(ctx);
556
+ });
557
+ }
558
+ /**
559
+ * Create egg context
560
+ * @function EggApplication#createContext
561
+ * @param {Req} req - node native Request object
562
+ * @param {Res} res - node native Response object
563
+ * @return {Context} context object
564
+ */
565
+ createContext(req, res) {
566
+ const context = Object.create(this.context);
567
+ const request = (context.request = Object.create(this.request));
568
+ const response = (context.response = Object.create(this.response));
569
+ context.app = request.app = response.app = this;
570
+ context.req = request.req = response.req = req;
571
+ context.res = request.res = response.res = res;
572
+ request.ctx = response.ctx = context;
573
+ request.response = response;
574
+ response.request = request;
575
+ context.onerror = context.onerror.bind(context);
576
+ context.originalUrl = request.originalUrl = req.url;
577
+ context.starttime = Date.now();
578
+ context.performanceStarttime = performance.now();
579
+ return context;
580
+ }
581
+ }
582
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZWdnLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vc3JjL2xpYi9lZ2cudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLFdBQVcsRUFBRSxNQUFNLGlCQUFpQixDQUFDO0FBQzlDLE9BQU8sSUFBSSxNQUFNLFdBQVcsQ0FBQztBQUM3QixPQUFPLEVBQUUsTUFBTSxTQUFTLENBQUM7QUFDekIsT0FBTyxJQUFJLEVBQUUsRUFBNkMsTUFBTSxXQUFXLENBQUM7QUFDNUUsT0FBTyxTQUFTLE1BQU0sZ0JBQWdCLENBQUM7QUFDdkMsT0FBTyxFQUFFLGlCQUFpQixFQUFFLE1BQU0sa0JBQWtCLENBQUM7QUFFckQsT0FBTyxFQUFFLE9BQU8sRUFBRSxNQUFNLEVBQUUsTUFBTSxhQUFhLENBQUM7QUFFOUMsT0FBTyxFQUFFLEtBQUssSUFBSSxRQUFRLEVBQUUsTUFBTSxhQUFhLENBQUM7QUFDaEQsT0FBTyxtQkFBbUIsRUFBRSxFQUMxQixLQUFLLElBQUksa0JBQWtCO0FBQzNCLDJFQUEyRTtFQUM1RSxNQUFNLGdCQUFnQixDQUFDO0FBQ3hCLE9BQU8sRUFBRSxNQUFNLEVBQUUsTUFBTSxnQkFBZ0IsQ0FBQztBQUN4QyxPQUFPLEVBQUUsZ0JBQWdCLElBQUksYUFBYSxFQUFFLFVBQVUsRUFBRSxTQUFTLEVBQUUsTUFBTSxZQUFZLENBQUM7QUFDdEYsT0FBTyxFQUFFLE9BQU8sSUFBSSxjQUFjLEVBQUUsTUFBTSxnQkFBZ0IsQ0FBQztBQUMzRCw2REFBNkQ7QUFDN0QsYUFBYTtBQUNiLE9BQU8sWUFBWSxNQUFNLHVCQUF1QixDQUFDO0FBSWpELE9BQU8sT0FBTyxNQUFNLDBCQUEwQixDQUFDO0FBQy9DLE9BQU8sT0FBTyxNQUFNLDBCQUEwQixDQUFDO0FBQy9DLE9BQU8sUUFBUSxNQUFNLDJCQUEyQixDQUFDO0FBRWpELE9BQU8sRUFBRSxNQUFNLElBQUksZUFBZSxFQUFtQixNQUFNLDJCQUEyQixDQUFDO0FBQ3ZGLE9BQU8s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