@backstage/backend-defaults 0.3.0-next.2 → 0.3.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (72) hide show
  1. package/CHANGELOG.md +62 -0
  2. package/auth/package.json +6 -0
  3. package/cache/package.json +1 -1
  4. package/config.d.ts +277 -0
  5. package/database/package.json +1 -1
  6. package/discovery/package.json +1 -1
  7. package/dist/auth.cjs.js +1025 -0
  8. package/dist/auth.cjs.js.map +1 -0
  9. package/dist/auth.d.ts +14 -0
  10. package/dist/cache.cjs.js.map +1 -1
  11. package/dist/cache.d.ts +31 -37
  12. package/dist/cjs/config-BDOwXIyo.cjs.js +64 -0
  13. package/dist/cjs/config-BDOwXIyo.cjs.js.map +1 -0
  14. package/dist/cjs/createConfigSecretEnumerator-DShyoWWL.cjs.js +33 -0
  15. package/dist/cjs/createConfigSecretEnumerator-DShyoWWL.cjs.js.map +1 -0
  16. package/dist/cjs/helpers-D2f1CG0o.cjs.js +53 -0
  17. package/dist/cjs/helpers-D2f1CG0o.cjs.js.map +1 -0
  18. package/dist/database.cjs.js +59 -145
  19. package/dist/database.cjs.js.map +1 -1
  20. package/dist/database.d.ts +7 -2
  21. package/dist/discovery.cjs.js +6 -6
  22. package/dist/discovery.cjs.js.map +1 -1
  23. package/dist/discovery.d.ts +9 -1
  24. package/dist/httpAuth.cjs.js +192 -0
  25. package/dist/httpAuth.cjs.js.map +1 -0
  26. package/dist/httpAuth.d.ts +15 -0
  27. package/dist/httpRouter.cjs.js +191 -0
  28. package/dist/httpRouter.cjs.js.map +1 -0
  29. package/dist/httpRouter.d.ts +55 -0
  30. package/dist/index.cjs.js +14 -8
  31. package/dist/index.cjs.js.map +1 -1
  32. package/dist/lifecycle.cjs.js.map +1 -1
  33. package/dist/lifecycle.d.ts +5 -1
  34. package/dist/logger.cjs.js +17 -0
  35. package/dist/logger.cjs.js.map +1 -0
  36. package/dist/logger.d.ts +14 -0
  37. package/dist/permissions.cjs.js.map +1 -1
  38. package/dist/permissions.d.ts +6 -0
  39. package/dist/rootConfig.cjs.js +3 -0
  40. package/dist/rootConfig.cjs.js.map +1 -1
  41. package/dist/rootConfig.d.ts +17 -2
  42. package/dist/rootHttpRouter.cjs.js +629 -0
  43. package/dist/rootHttpRouter.cjs.js.map +1 -0
  44. package/dist/rootHttpRouter.d.ts +283 -0
  45. package/dist/rootLifecycle.cjs.js.map +1 -1
  46. package/dist/rootLifecycle.d.ts +5 -1
  47. package/dist/rootLogger.cjs.js +143 -0
  48. package/dist/rootLogger.cjs.js.map +1 -0
  49. package/dist/rootLogger.d.ts +58 -0
  50. package/dist/scheduler.cjs.js +11 -40
  51. package/dist/scheduler.cjs.js.map +1 -1
  52. package/dist/scheduler.d.ts +19 -2
  53. package/dist/urlReader.cjs.js +2932 -2
  54. package/dist/urlReader.cjs.js.map +1 -1
  55. package/dist/urlReader.d.ts +422 -4
  56. package/dist/userInfo.cjs.js +70 -0
  57. package/dist/userInfo.cjs.js.map +1 -0
  58. package/dist/userInfo.d.ts +14 -0
  59. package/httpAuth/package.json +6 -0
  60. package/httpRouter/package.json +6 -0
  61. package/lifecycle/package.json +1 -1
  62. package/logger/package.json +6 -0
  63. package/migrations/auth/20240327104803_public_keys.js +50 -0
  64. package/package.json +103 -11
  65. package/permissions/package.json +1 -1
  66. package/rootConfig/package.json +1 -1
  67. package/rootHttpRouter/package.json +6 -0
  68. package/rootLifecycle/package.json +1 -1
  69. package/rootLogger/package.json +6 -0
  70. package/scheduler/package.json +1 -1
  71. package/urlReader/package.json +1 -1
  72. package/userInfo/package.json +6 -0
@@ -0,0 +1,629 @@
1
+ 'use strict';
2
+
3
+ var express = require('express');
4
+ var trimEnd = require('lodash/trimEnd');
5
+ var config = require('./cjs/config-BDOwXIyo.cjs.js');
6
+ var http = require('http');
7
+ var https = require('https');
8
+ var stoppableServer = require('stoppable');
9
+ var fs = require('fs-extra');
10
+ var platformPath = require('path');
11
+ var forge = require('node-forge');
12
+ var cors = require('cors');
13
+ var helmet = require('helmet');
14
+ var morgan = require('morgan');
15
+ var compression = require('compression');
16
+ var kebabCase = require('lodash/kebabCase');
17
+ var minimatch = require('minimatch');
18
+ var errors = require('@backstage/errors');
19
+ var crypto = require('crypto');
20
+ var backendPluginApi = require('@backstage/backend-plugin-api');
21
+
22
+ function _interopDefaultCompat (e) { return e && typeof e === 'object' && 'default' in e ? e : { default: e }; }
23
+
24
+ function _interopNamespaceCompat(e) {
25
+ if (e && typeof e === 'object' && 'default' in e) return e;
26
+ var n = Object.create(null);
27
+ if (e) {
28
+ Object.keys(e).forEach(function (k) {
29
+ if (k !== 'default') {
30
+ var d = Object.getOwnPropertyDescriptor(e, k);
31
+ Object.defineProperty(n, k, d.get ? d : {
32
+ enumerable: true,
33
+ get: function () { return e[k]; }
34
+ });
35
+ }
36
+ });
37
+ }
38
+ n.default = e;
39
+ return Object.freeze(n);
40
+ }
41
+
42
+ var express__default = /*#__PURE__*/_interopDefaultCompat(express);
43
+ var trimEnd__default = /*#__PURE__*/_interopDefaultCompat(trimEnd);
44
+ var http__namespace = /*#__PURE__*/_interopNamespaceCompat(http);
45
+ var https__namespace = /*#__PURE__*/_interopNamespaceCompat(https);
46
+ var stoppableServer__default = /*#__PURE__*/_interopDefaultCompat(stoppableServer);
47
+ var fs__default = /*#__PURE__*/_interopDefaultCompat(fs);
48
+ var forge__default = /*#__PURE__*/_interopDefaultCompat(forge);
49
+ var cors__default = /*#__PURE__*/_interopDefaultCompat(cors);
50
+ var helmet__default = /*#__PURE__*/_interopDefaultCompat(helmet);
51
+ var morgan__default = /*#__PURE__*/_interopDefaultCompat(morgan);
52
+ var compression__default = /*#__PURE__*/_interopDefaultCompat(compression);
53
+ var kebabCase__default = /*#__PURE__*/_interopDefaultCompat(kebabCase);
54
+
55
+ function normalizePath(path) {
56
+ return `${trimEnd__default.default(path, "/")}/`;
57
+ }
58
+ class DefaultRootHttpRouter {
59
+ #indexPath;
60
+ #router = express.Router();
61
+ #namedRoutes = express.Router();
62
+ #indexRouter = express.Router();
63
+ #existingPaths = new Array();
64
+ static create(options) {
65
+ let indexPath;
66
+ if (options?.indexPath === false) {
67
+ indexPath = void 0;
68
+ } else if (options?.indexPath === void 0) {
69
+ indexPath = "/api/app";
70
+ } else if (options?.indexPath === "") {
71
+ throw new Error("indexPath option may not be an empty string");
72
+ } else {
73
+ indexPath = options.indexPath;
74
+ }
75
+ return new DefaultRootHttpRouter(indexPath);
76
+ }
77
+ constructor(indexPath) {
78
+ this.#indexPath = indexPath;
79
+ this.#router.use(this.#namedRoutes);
80
+ this.#router.use("/api/", (_req, _res, next) => {
81
+ next("router");
82
+ });
83
+ if (this.#indexPath) {
84
+ this.#router.use(this.#indexRouter);
85
+ }
86
+ }
87
+ use(path, handler) {
88
+ if (path.match(/^[/\s]*$/)) {
89
+ throw new Error(`Root router path may not be empty`);
90
+ }
91
+ const conflictingPath = this.#findConflictingPath(path);
92
+ if (conflictingPath) {
93
+ throw new Error(
94
+ `Path ${path} conflicts with the existing path ${conflictingPath}`
95
+ );
96
+ }
97
+ this.#existingPaths.push(path);
98
+ this.#namedRoutes.use(path, handler);
99
+ if (this.#indexPath === path) {
100
+ this.#indexRouter.use(handler);
101
+ }
102
+ }
103
+ handler() {
104
+ return this.#router;
105
+ }
106
+ #findConflictingPath(newPath) {
107
+ const normalizedNewPath = normalizePath(newPath);
108
+ for (const path of this.#existingPaths) {
109
+ const normalizedPath = normalizePath(path);
110
+ if (normalizedPath.startsWith(normalizedNewPath)) {
111
+ return path;
112
+ }
113
+ if (normalizedNewPath.startsWith(normalizedPath)) {
114
+ return path;
115
+ }
116
+ }
117
+ return void 0;
118
+ }
119
+ }
120
+
121
+ const FIVE_DAYS_IN_MS = 5 * 24 * 60 * 60 * 1e3;
122
+ const IP_HOSTNAME_REGEX = /:|^\d+\.\d+\.\d+\.\d+$/;
123
+ async function getGeneratedCertificate(hostname, logger) {
124
+ const hasModules = await fs__default.default.pathExists("node_modules");
125
+ let certPath;
126
+ if (hasModules) {
127
+ certPath = platformPath.resolve(
128
+ "node_modules/.cache/backstage-backend/dev-cert.pem"
129
+ );
130
+ await fs__default.default.ensureDir(platformPath.dirname(certPath));
131
+ } else {
132
+ certPath = platformPath.resolve(".dev-cert.pem");
133
+ }
134
+ if (await fs__default.default.pathExists(certPath)) {
135
+ try {
136
+ const cert = await fs__default.default.readFile(certPath);
137
+ const crt = forge__default.default.pki.certificateFromPem(cert.toString());
138
+ const remainingMs = crt.validity.notAfter.getTime() - Date.now();
139
+ if (remainingMs > FIVE_DAYS_IN_MS) {
140
+ logger.info("Using existing self-signed certificate");
141
+ return {
142
+ key: cert,
143
+ cert
144
+ };
145
+ }
146
+ } catch (error) {
147
+ logger.warn(`Unable to use existing self-signed certificate, ${error}`);
148
+ }
149
+ }
150
+ logger.info("Generating new self-signed certificate");
151
+ const newCert = await generateCertificate(hostname);
152
+ await fs__default.default.writeFile(certPath, newCert.cert + newCert.key, "utf8");
153
+ return newCert;
154
+ }
155
+ async function generateCertificate(hostname) {
156
+ const attributes = [
157
+ {
158
+ name: "commonName",
159
+ value: "dev-cert"
160
+ }
161
+ ];
162
+ const sans = [
163
+ {
164
+ type: 2,
165
+ // DNS
166
+ value: "localhost"
167
+ },
168
+ {
169
+ type: 2,
170
+ value: "localhost.localdomain"
171
+ },
172
+ {
173
+ type: 2,
174
+ value: "[::1]"
175
+ },
176
+ {
177
+ type: 7,
178
+ // IP
179
+ ip: "127.0.0.1"
180
+ },
181
+ {
182
+ type: 7,
183
+ ip: "fe80::1"
184
+ }
185
+ ];
186
+ if (!sans.find(({ value, ip }) => value === hostname || ip === hostname)) {
187
+ sans.push(
188
+ IP_HOSTNAME_REGEX.test(hostname) ? {
189
+ type: 7,
190
+ ip: hostname
191
+ } : {
192
+ type: 2,
193
+ value: hostname
194
+ }
195
+ );
196
+ }
197
+ const params = {
198
+ algorithm: "sha256",
199
+ keySize: 2048,
200
+ days: 30,
201
+ extensions: [
202
+ {
203
+ name: "keyUsage",
204
+ keyCertSign: true,
205
+ digitalSignature: true,
206
+ nonRepudiation: true,
207
+ keyEncipherment: true,
208
+ dataEncipherment: true
209
+ },
210
+ {
211
+ name: "extKeyUsage",
212
+ serverAuth: true,
213
+ clientAuth: true,
214
+ codeSigning: true,
215
+ timeStamping: true
216
+ },
217
+ {
218
+ name: "subjectAltName",
219
+ altNames: sans
220
+ }
221
+ ]
222
+ };
223
+ return new Promise(
224
+ (resolve, reject) => require("selfsigned").generate(
225
+ attributes,
226
+ params,
227
+ (err, bundle) => {
228
+ if (err) {
229
+ reject(err);
230
+ } else {
231
+ resolve({ key: bundle.private, cert: bundle.cert });
232
+ }
233
+ }
234
+ )
235
+ );
236
+ }
237
+
238
+ async function createHttpServer(listener, options, deps) {
239
+ const server = await createServer(listener, options, deps);
240
+ const stopper = stoppableServer__default.default(server, 0);
241
+ const stopServer = stopper.stop.bind(stopper);
242
+ return Object.assign(server, {
243
+ start() {
244
+ return new Promise((resolve, reject) => {
245
+ const handleStartupError = (error) => {
246
+ server.close();
247
+ reject(error);
248
+ };
249
+ server.on("error", handleStartupError);
250
+ const { host, port } = options.listen;
251
+ server.listen(port, host, () => {
252
+ server.off("error", handleStartupError);
253
+ deps.logger.info(`Listening on ${host}:${port}`);
254
+ resolve();
255
+ });
256
+ });
257
+ },
258
+ stop() {
259
+ return new Promise((resolve, reject) => {
260
+ stopServer((error) => {
261
+ if (error) {
262
+ reject(error);
263
+ } else {
264
+ resolve();
265
+ }
266
+ });
267
+ });
268
+ },
269
+ port() {
270
+ const address = server.address();
271
+ if (typeof address === "string" || address === null) {
272
+ throw new Error(`Unexpected server address '${address}'`);
273
+ }
274
+ return address.port;
275
+ }
276
+ });
277
+ }
278
+ async function createServer(listener, options, deps) {
279
+ if (options.https) {
280
+ const { certificate } = options.https;
281
+ if (certificate.type === "generated") {
282
+ const credentials = await getGeneratedCertificate(
283
+ certificate.hostname,
284
+ deps.logger
285
+ );
286
+ return https__namespace.createServer(credentials, listener);
287
+ }
288
+ return https__namespace.createServer(certificate, listener);
289
+ }
290
+ return http__namespace.createServer(listener);
291
+ }
292
+
293
+ function readHelmetOptions(config) {
294
+ const cspOptions = readCspDirectives(config);
295
+ return {
296
+ contentSecurityPolicy: {
297
+ useDefaults: false,
298
+ directives: applyCspDirectives(cspOptions)
299
+ },
300
+ // These are all disabled in order to maintain backwards compatibility
301
+ // when bumping helmet v5. We can't enable these by default because
302
+ // there is no way for users to configure them.
303
+ // TODO(Rugvip): We should give control of this setup to consumers
304
+ crossOriginEmbedderPolicy: false,
305
+ crossOriginOpenerPolicy: false,
306
+ crossOriginResourcePolicy: false,
307
+ originAgentCluster: false
308
+ };
309
+ }
310
+ function readCspDirectives(config) {
311
+ const cc = config?.getOptionalConfig("csp");
312
+ if (!cc) {
313
+ return void 0;
314
+ }
315
+ const result = {};
316
+ for (const key of cc.keys()) {
317
+ if (cc.get(key) === false) {
318
+ result[key] = false;
319
+ } else {
320
+ result[key] = cc.getStringArray(key);
321
+ }
322
+ }
323
+ return result;
324
+ }
325
+ function applyCspDirectives(directives) {
326
+ const result = helmet__default.default.contentSecurityPolicy.getDefaultDirectives();
327
+ result["script-src"] = ["'self'", "'unsafe-eval'"];
328
+ delete result["form-action"];
329
+ if (directives) {
330
+ for (const [key, value] of Object.entries(directives)) {
331
+ const kebabCaseKey = kebabCase__default.default(key);
332
+ if (value === false) {
333
+ delete result[kebabCaseKey];
334
+ } else {
335
+ result[kebabCaseKey] = value;
336
+ }
337
+ }
338
+ }
339
+ return result;
340
+ }
341
+
342
+ function readCorsOptions(config) {
343
+ const cc = config?.getOptionalConfig("cors");
344
+ if (!cc) {
345
+ return { origin: false };
346
+ }
347
+ return removeUnknown({
348
+ origin: createCorsOriginMatcher(readStringArray(cc, "origin")),
349
+ methods: readStringArray(cc, "methods"),
350
+ allowedHeaders: readStringArray(cc, "allowedHeaders"),
351
+ exposedHeaders: readStringArray(cc, "exposedHeaders"),
352
+ credentials: cc.getOptionalBoolean("credentials"),
353
+ maxAge: cc.getOptionalNumber("maxAge"),
354
+ preflightContinue: cc.getOptionalBoolean("preflightContinue"),
355
+ optionsSuccessStatus: cc.getOptionalNumber("optionsSuccessStatus")
356
+ });
357
+ }
358
+ function removeUnknown(obj) {
359
+ return Object.fromEntries(
360
+ Object.entries(obj).filter(([, v]) => v !== void 0)
361
+ );
362
+ }
363
+ function readStringArray(config, key) {
364
+ const value = config.getOptional(key);
365
+ if (typeof value === "string") {
366
+ return [value];
367
+ } else if (!value) {
368
+ return void 0;
369
+ }
370
+ return config.getStringArray(key);
371
+ }
372
+ function createCorsOriginMatcher(allowedOriginPatterns) {
373
+ if (!allowedOriginPatterns) {
374
+ return void 0;
375
+ }
376
+ const allowedOriginMatchers = allowedOriginPatterns.map(
377
+ (pattern) => new minimatch.Minimatch(pattern, { nocase: true, noglobstar: true })
378
+ );
379
+ return (origin, callback) => {
380
+ return callback(
381
+ null,
382
+ allowedOriginMatchers.some((pattern) => pattern.match(origin ?? ""))
383
+ );
384
+ };
385
+ }
386
+
387
+ function handleBadError(error, logger) {
388
+ const logId = crypto.randomBytes(10).toString("hex");
389
+ logger.child({ logId }).error(`Filtered internal error with logId=${logId} from response`, error);
390
+ const newError = new Error(`An internal error occurred logId=${logId}`);
391
+ delete newError.stack;
392
+ return newError;
393
+ }
394
+ function applyInternalErrorFilter(error, logger) {
395
+ try {
396
+ errors.assertError(error);
397
+ } catch (assertionError) {
398
+ errors.assertError(assertionError);
399
+ return handleBadError(assertionError, logger);
400
+ }
401
+ const constructorName = error.constructor.name;
402
+ if (constructorName === "DatabaseError") {
403
+ return handleBadError(error, logger);
404
+ }
405
+ return error;
406
+ }
407
+
408
+ class MiddlewareFactory {
409
+ #config;
410
+ #logger;
411
+ /**
412
+ * Creates a new {@link MiddlewareFactory}.
413
+ */
414
+ static create(options) {
415
+ return new MiddlewareFactory(options);
416
+ }
417
+ constructor(options) {
418
+ this.#config = options.config;
419
+ this.#logger = options.logger;
420
+ }
421
+ /**
422
+ * Returns a middleware that unconditionally produces a 404 error response.
423
+ *
424
+ * @remarks
425
+ *
426
+ * Typically you want to place this middleware at the end of the chain, such
427
+ * that it's the last one attempted after no other routes matched.
428
+ *
429
+ * @returns An Express request handler
430
+ */
431
+ notFound() {
432
+ return (_req, res) => {
433
+ res.status(404).end();
434
+ };
435
+ }
436
+ /**
437
+ * Returns the compression middleware.
438
+ *
439
+ * @remarks
440
+ *
441
+ * The middleware will attempt to compress response bodies for all requests
442
+ * that traverse through the middleware.
443
+ */
444
+ compression() {
445
+ return compression__default.default();
446
+ }
447
+ /**
448
+ * Returns a request logging middleware.
449
+ *
450
+ * @remarks
451
+ *
452
+ * Typically you want to place this middleware at the start of the chain, such
453
+ * that it always logs requests whether they are "caught" by handlers farther
454
+ * down or not.
455
+ *
456
+ * @returns An Express request handler
457
+ */
458
+ logging() {
459
+ const logger = this.#logger.child({
460
+ type: "incomingRequest"
461
+ });
462
+ return morgan__default.default("combined", {
463
+ stream: {
464
+ write(message) {
465
+ logger.info(message.trimEnd());
466
+ }
467
+ }
468
+ });
469
+ }
470
+ /**
471
+ * Returns a middleware that implements the helmet library.
472
+ *
473
+ * @remarks
474
+ *
475
+ * This middleware applies security policies to incoming requests and outgoing
476
+ * responses. It is configured using config keys such as `backend.csp`.
477
+ *
478
+ * @see {@link https://helmetjs.github.io/}
479
+ *
480
+ * @returns An Express request handler
481
+ */
482
+ helmet() {
483
+ return helmet__default.default(readHelmetOptions(this.#config.getOptionalConfig("backend")));
484
+ }
485
+ /**
486
+ * Returns a middleware that implements the cors library.
487
+ *
488
+ * @remarks
489
+ *
490
+ * This middleware handles CORS. It is configured using the config key
491
+ * `backend.cors`.
492
+ *
493
+ * @see {@link https://github.com/expressjs/cors}
494
+ *
495
+ * @returns An Express request handler
496
+ */
497
+ cors() {
498
+ return cors__default.default(readCorsOptions(this.#config.getOptionalConfig("backend")));
499
+ }
500
+ /**
501
+ * Express middleware to handle errors during request processing.
502
+ *
503
+ * @remarks
504
+ *
505
+ * This is commonly the very last middleware in the chain.
506
+ *
507
+ * Its primary purpose is not to do translation of business logic exceptions,
508
+ * but rather to be a global catch-all for uncaught "fatal" errors that are
509
+ * expected to result in a 500 error. However, it also does handle some common
510
+ * error types (such as http-error exceptions, and the well-known error types
511
+ * in the `@backstage/errors` package) and returns the enclosed status code
512
+ * accordingly.
513
+ *
514
+ * It will also produce a response body with a serialized form of the error,
515
+ * unless a previous handler already did send a body. See
516
+ * {@link @backstage/errors#ErrorResponseBody} for the response shape used.
517
+ *
518
+ * @returns An Express error request handler
519
+ */
520
+ error(options = {}) {
521
+ const showStackTraces = options.showStackTraces ?? process.env.NODE_ENV === "development";
522
+ const logger = this.#logger.child({
523
+ type: "errorHandler"
524
+ });
525
+ return (rawError, req, res, next) => {
526
+ const error = applyInternalErrorFilter(rawError, logger);
527
+ const statusCode = getStatusCode(error);
528
+ if (options.logAllErrors || statusCode >= 500) {
529
+ logger.error(`Request failed with status ${statusCode}`, error);
530
+ }
531
+ if (res.headersSent) {
532
+ next(error);
533
+ return;
534
+ }
535
+ const body = {
536
+ error: errors.serializeError(error, { includeStack: showStackTraces }),
537
+ request: { method: req.method, url: req.url },
538
+ response: { statusCode }
539
+ };
540
+ res.status(statusCode).json(body);
541
+ };
542
+ }
543
+ }
544
+ function getStatusCode(error) {
545
+ const knownStatusCodeFields = ["statusCode", "status"];
546
+ for (const field of knownStatusCodeFields) {
547
+ const statusCode = error[field];
548
+ if (typeof statusCode === "number" && (statusCode | 0) === statusCode && // is whole integer
549
+ statusCode >= 100 && statusCode <= 599) {
550
+ return statusCode;
551
+ }
552
+ }
553
+ switch (error.name) {
554
+ case errors.NotModifiedError.name:
555
+ return 304;
556
+ case errors.InputError.name:
557
+ return 400;
558
+ case errors.AuthenticationError.name:
559
+ return 401;
560
+ case errors.NotAllowedError.name:
561
+ return 403;
562
+ case errors.NotFoundError.name:
563
+ return 404;
564
+ case errors.ConflictError.name:
565
+ return 409;
566
+ case errors.NotImplementedError.name:
567
+ return 501;
568
+ case errors.ServiceUnavailableError.name:
569
+ return 503;
570
+ }
571
+ return 500;
572
+ }
573
+
574
+ function defaultConfigure({ applyDefaults }) {
575
+ applyDefaults();
576
+ }
577
+ const rootHttpRouterServiceFactory = backendPluginApi.createServiceFactory(
578
+ (options) => ({
579
+ service: backendPluginApi.coreServices.rootHttpRouter,
580
+ deps: {
581
+ config: backendPluginApi.coreServices.rootConfig,
582
+ rootLogger: backendPluginApi.coreServices.rootLogger,
583
+ lifecycle: backendPluginApi.coreServices.rootLifecycle
584
+ },
585
+ async factory({ config: config$1, rootLogger, lifecycle }) {
586
+ const { indexPath, configure = defaultConfigure } = options ?? {};
587
+ const logger = rootLogger.child({ service: "rootHttpRouter" });
588
+ const app = express__default.default();
589
+ const router = DefaultRootHttpRouter.create({ indexPath });
590
+ const middleware = MiddlewareFactory.create({ config: config$1, logger });
591
+ const routes = router.handler();
592
+ const server = await createHttpServer(
593
+ app,
594
+ config.readHttpServerOptions(config$1.getOptionalConfig("backend")),
595
+ { logger }
596
+ );
597
+ configure({
598
+ app,
599
+ server,
600
+ routes,
601
+ middleware,
602
+ config: config$1,
603
+ logger,
604
+ lifecycle,
605
+ applyDefaults() {
606
+ app.use(middleware.helmet());
607
+ app.use(middleware.cors());
608
+ app.use(middleware.compression());
609
+ app.use(middleware.logging());
610
+ app.use(routes);
611
+ app.use(middleware.notFound());
612
+ app.use(middleware.error());
613
+ }
614
+ });
615
+ lifecycle.addShutdownHook(() => server.stop());
616
+ await server.start();
617
+ return router;
618
+ }
619
+ })
620
+ );
621
+
622
+ exports.readHttpServerOptions = config.readHttpServerOptions;
623
+ exports.DefaultRootHttpRouter = DefaultRootHttpRouter;
624
+ exports.MiddlewareFactory = MiddlewareFactory;
625
+ exports.createHttpServer = createHttpServer;
626
+ exports.readCorsOptions = readCorsOptions;
627
+ exports.readHelmetOptions = readHelmetOptions;
628
+ exports.rootHttpRouterServiceFactory = rootHttpRouterServiceFactory;
629
+ //# sourceMappingURL=rootHttpRouter.cjs.js.map