skuba 13.0.0-custom-conditions-exports-20250730235248 → 13.0.0-custom-conditions-exports-20250923005538

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 (136) hide show
  1. package/README.md +1 -2
  2. package/config/tsconfig.json +2 -1
  3. package/jest/moduleNameMapper.js +3 -0
  4. package/lib/cli/build/assets.js +1 -1
  5. package/lib/cli/build/assets.js.map +2 -2
  6. package/lib/cli/build/esbuild.js +1 -1
  7. package/lib/cli/build/esbuild.js.map +2 -2
  8. package/lib/cli/build/index.js +1 -1
  9. package/lib/cli/build/index.js.map +2 -2
  10. package/lib/cli/build/tsc.d.ts +9 -2
  11. package/lib/cli/build/tsc.js +76 -40
  12. package/lib/cli/build/tsc.js.map +3 -3
  13. package/lib/cli/configure/analyseDependencies.d.ts +2 -2
  14. package/lib/cli/configure/analyseDependencies.js.map +1 -1
  15. package/lib/cli/configure/analysis/package.d.ts +1 -1
  16. package/lib/cli/configure/analysis/package.js +1 -1
  17. package/lib/cli/configure/analysis/package.js.map +2 -2
  18. package/lib/cli/configure/ensureTemplateCompletion.d.ts +2 -2
  19. package/lib/cli/configure/ensureTemplateCompletion.js.map +1 -1
  20. package/lib/cli/configure/getEntryPoint.d.ts +2 -2
  21. package/lib/cli/configure/getEntryPoint.js.map +1 -1
  22. package/lib/cli/configure/getProjectType.d.ts +2 -2
  23. package/lib/cli/configure/getProjectType.js.map +1 -1
  24. package/lib/cli/configure/processing/package.js +8 -2
  25. package/lib/cli/configure/processing/package.js.map +2 -2
  26. package/lib/cli/init/getConfig.js +1 -1
  27. package/lib/cli/init/getConfig.js.map +2 -2
  28. package/lib/cli/lint/internal.js +1 -1
  29. package/lib/cli/lint/internal.js.map +2 -2
  30. package/lib/cli/lint/internalLints/patchRenovateConfig.js +2 -2
  31. package/lib/cli/lint/internalLints/patchRenovateConfig.js.map +1 -1
  32. package/lib/cli/lint/internalLints/upgrade/index.d.ts +2 -2
  33. package/lib/cli/lint/internalLints/upgrade/index.js.map +1 -1
  34. package/lib/cli/lint/internalLints/upgrade/patches/12.0.2/index.js +3 -8
  35. package/lib/cli/lint/internalLints/upgrade/patches/12.0.2/index.js.map +2 -2
  36. package/lib/cli/lint/internalLints/upgrade/patches/12.0.2/unhandledRejections.d.ts +4 -0
  37. package/lib/cli/lint/internalLints/upgrade/patches/12.0.2/unhandledRejections.js +162 -0
  38. package/lib/cli/lint/internalLints/upgrade/patches/12.0.2/unhandledRejections.js.map +7 -0
  39. package/lib/cli/lint/internalLints/upgrade/patches/12.1.1/index.d.ts +2 -0
  40. package/lib/cli/lint/internalLints/upgrade/patches/12.1.1/index.js +35 -0
  41. package/lib/cli/lint/internalLints/upgrade/patches/12.1.1/index.js.map +7 -0
  42. package/lib/cli/lint/internalLints/upgrade/patches/12.1.1/patchJestSnapshots.d.ts +3 -0
  43. package/lib/cli/lint/internalLints/upgrade/patches/12.1.1/patchJestSnapshots.js +105 -0
  44. package/lib/cli/lint/internalLints/upgrade/patches/12.1.1/patchJestSnapshots.js.map +7 -0
  45. package/lib/cli/lint/internalLints/upgrade/patches/12.2.0/configureTsConfigForESM.d.ts +23 -0
  46. package/lib/cli/lint/internalLints/upgrade/patches/12.2.0/configureTsConfigForESM.js +321 -0
  47. package/lib/cli/lint/internalLints/upgrade/patches/12.2.0/configureTsConfigForESM.js.map +7 -0
  48. package/lib/cli/lint/internalLints/upgrade/patches/12.2.0/index.d.ts +2 -0
  49. package/lib/cli/lint/internalLints/upgrade/patches/12.2.0/index.js +40 -0
  50. package/lib/cli/lint/internalLints/upgrade/patches/12.2.0/index.js.map +7 -0
  51. package/lib/cli/lint/internalLints/upgrade/patches/{12.0.2 → 12.2.0}/rewriteSrcImports.js +7 -4
  52. package/lib/cli/lint/internalLints/upgrade/patches/{12.0.2 → 12.2.0}/rewriteSrcImports.js.map +3 -3
  53. package/lib/cli/lint/internalLints/upgrade/patches/8.2.1/upgradeESLint.js +1 -1
  54. package/lib/cli/lint/internalLints/upgrade/patches/8.2.1/upgradeESLint.js.map +2 -2
  55. package/lib/cli/node/index.js +6 -0
  56. package/lib/cli/node/index.js.map +2 -2
  57. package/lib/cli/start/index.js +6 -0
  58. package/lib/cli/start/index.js.map +2 -2
  59. package/lib/cli/test/index.d.ts +1 -1
  60. package/lib/cli/test/index.js +18 -4
  61. package/lib/cli/test/index.js.map +2 -2
  62. package/lib/utils/args.d.ts +2 -0
  63. package/lib/utils/args.js +5 -0
  64. package/lib/utils/args.js.map +2 -2
  65. package/lib/utils/manifest.d.ts +1 -1
  66. package/lib/utils/manifest.js +1 -1
  67. package/lib/utils/manifest.js.map +2 -2
  68. package/lib/utils/template.d.ts +1 -1
  69. package/package.json +17 -20
  70. package/template/base/_.prettierrc.js +1 -1
  71. package/template/base/_eslint.config.js +1 -1
  72. package/template/base/_pnpm-workspace.yaml +1 -0
  73. package/template/base/jest.setup.ts +1 -1
  74. package/template/base/tsconfig.build.json +3 -0
  75. package/template/base/tsconfig.json +0 -1
  76. package/template/express-rest-api/.buildkite/pipeline.yml +6 -0
  77. package/template/express-rest-api/.env +1 -1
  78. package/template/express-rest-api/.gantry/dev.yml +5 -1
  79. package/template/express-rest-api/.gantry/prod.yml +5 -1
  80. package/template/express-rest-api/Dockerfile +1 -1
  81. package/template/express-rest-api/Dockerfile.dev-deps +1 -1
  82. package/template/express-rest-api/README.md +5 -5
  83. package/template/express-rest-api/gantry.apply.yml +17 -1
  84. package/template/express-rest-api/package.json +11 -5
  85. package/template/express-rest-api/src/api/healthCheck.ts +2 -2
  86. package/template/express-rest-api/src/config.ts +7 -7
  87. package/template/express-rest-api/src/framework/logging.ts +11 -7
  88. package/template/express-rest-api/src/framework/metrics.ts +1 -1
  89. package/template/express-rest-api/src/listen.ts +6 -0
  90. package/template/express-rest-api/src/tracing.ts +56 -0
  91. package/template/greeter/Dockerfile +1 -1
  92. package/template/greeter/README.md +2 -2
  93. package/template/greeter/package.json +2 -2
  94. package/template/koa-rest-api/.buildkite/pipeline.yml +6 -0
  95. package/template/koa-rest-api/.env +1 -1
  96. package/template/koa-rest-api/.gantry/dev.yml +3 -3
  97. package/template/koa-rest-api/.gantry/prod.yml +3 -3
  98. package/template/koa-rest-api/Dockerfile.dev-deps +1 -1
  99. package/template/koa-rest-api/README.md +6 -6
  100. package/template/koa-rest-api/gantry.apply.yml +15 -3
  101. package/template/koa-rest-api/package.json +10 -11
  102. package/template/koa-rest-api/skuba.template.js +1 -1
  103. package/template/koa-rest-api/src/api/healthCheck.ts +2 -2
  104. package/template/koa-rest-api/src/config.ts +7 -7
  105. package/template/koa-rest-api/src/framework/logging.ts +12 -8
  106. package/template/koa-rest-api/src/framework/metrics.ts +1 -1
  107. package/template/koa-rest-api/src/framework/server.test.ts +7 -8
  108. package/template/koa-rest-api/src/framework/server.ts +1 -4
  109. package/template/koa-rest-api/src/listen.ts +6 -0
  110. package/template/lambda-sqs-worker-cdk/.buildkite/pipeline.yml +6 -2
  111. package/template/lambda-sqs-worker-cdk/.env +1 -1
  112. package/template/lambda-sqs-worker-cdk/Dockerfile +1 -1
  113. package/template/lambda-sqs-worker-cdk/README.md +8 -8
  114. package/template/lambda-sqs-worker-cdk/infra/__snapshots__/appStack.test.ts.snap +54 -26
  115. package/template/lambda-sqs-worker-cdk/infra/appStack.test.ts +5 -22
  116. package/template/lambda-sqs-worker-cdk/infra/appStack.ts +16 -9
  117. package/template/lambda-sqs-worker-cdk/infra/config.ts +34 -20
  118. package/template/lambda-sqs-worker-cdk/infra/index.ts +1 -1
  119. package/template/lambda-sqs-worker-cdk/package.json +7 -9
  120. package/template/lambda-sqs-worker-cdk/src/app.test.ts +91 -51
  121. package/template/lambda-sqs-worker-cdk/src/app.ts +10 -9
  122. package/template/lambda-sqs-worker-cdk/src/config.ts +11 -16
  123. package/template/lambda-sqs-worker-cdk/src/framework/handler.test.ts +10 -5
  124. package/template/lambda-sqs-worker-cdk/src/framework/handler.ts +48 -24
  125. package/template/lambda-sqs-worker-cdk/src/framework/logging.ts +23 -11
  126. package/template/lambda-sqs-worker-cdk/src/framework/metrics.ts +1 -4
  127. package/template/lambda-sqs-worker-cdk/src/testing/handler.ts +4 -1
  128. package/template/lambda-sqs-worker-cdk/tsconfig.json +1 -2
  129. package/template/oss-npm-package/.github/workflows/release.yml +5 -6
  130. package/template/oss-npm-package/.github/workflows/validate.yml +5 -5
  131. package/template/oss-npm-package/_package.json +0 -3
  132. package/template/oss-npm-package/skuba.template.js +1 -1
  133. package/lib/cli/lint/internalLints/upgrade/patches/12.0.2/configureTsConfigForESM.d.ts +0 -6
  134. package/lib/cli/lint/internalLints/upgrade/patches/12.0.2/configureTsConfigForESM.js +0 -248
  135. package/lib/cli/lint/internalLints/upgrade/patches/12.0.2/configureTsConfigForESM.js.map +0 -7
  136. /package/lib/cli/lint/internalLints/upgrade/patches/{12.0.2 → 12.2.0}/rewriteSrcImports.d.ts +0 -0
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "version": 3,
3
3
  "sources": ["../../../src/cli/start/index.ts"],
4
- "sourcesContent": ["import path from 'path';\n\nimport getPort from 'get-port';\n\nimport { parseRunArgs } from '../../utils/args.js';\nimport { createExec } from '../../utils/exec.js';\nimport { getEntryPointFromManifest } from '../../utils/manifest.js';\nimport { isIpPort } from '../../utils/validation.js';\n\nexport const start = async () => {\n const [args, availablePort] = await Promise.all([\n parseRunArgs(process.argv.slice(2)),\n getPort(),\n ]);\n\n args.entryPoint ??= await getEntryPointFromManifest();\n\n const execProcess = createExec({\n env: {\n __SKUBA_ENTRY_POINT: args.entryPoint,\n __SKUBA_PORT: String(isIpPort(args.port) ? args.port : availablePort),\n },\n });\n\n return execProcess(\n 'tsx',\n 'watch',\n '--clear-screen=false',\n ...args.node,\n '--env-file-if-exists',\n '.env',\n '--require',\n 'tsconfig-paths/register',\n path.join(__dirname, '..', '..', 'wrapper', 'index.js'),\n ...args.script,\n );\n};\n"],
5
- "mappings": ";;;;;;;;;;;;;;;;;;;;;;;;;;;;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,kBAAiB;AAEjB,sBAAoB;AAEpB,kBAA6B;AAC7B,kBAA2B;AAC3B,sBAA0C;AAC1C,wBAAyB;AAElB,MAAM,QAAQ,YAAY;AAC/B,QAAM,CAAC,MAAM,aAAa,IAAI,MAAM,QAAQ,IAAI;AAAA,QAC9C,0BAAa,QAAQ,KAAK,MAAM,CAAC,CAAC;AAAA,QAClC,gBAAAA,SAAQ;AAAA,EACV,CAAC;AAED,OAAK,eAAe,UAAM,2CAA0B;AAEpD,QAAM,kBAAc,wBAAW;AAAA,IAC7B,KAAK;AAAA,MACH,qBAAqB,KAAK;AAAA,MAC1B,cAAc,WAAO,4BAAS,KAAK,IAAI,IAAI,KAAK,OAAO,aAAa;AAAA,IACtE;AAAA,EACF,CAAC;AAED,SAAO;AAAA,IACL;AAAA,IACA;AAAA,IACA;AAAA,IACA,GAAG,KAAK;AAAA,IACR;AAAA,IACA;AAAA,IACA;AAAA,IACA;AAAA,IACA,YAAAC,QAAK,KAAK,WAAW,MAAM,MAAM,WAAW,UAAU;AAAA,IACtD,GAAG,KAAK;AAAA,EACV;AACF;",
4
+ "sourcesContent": ["import path from 'path';\n\nimport getPort from 'get-port';\n\nimport { parseRunArgs } from '../../utils/args.js';\nimport { createExec } from '../../utils/exec.js';\nimport { getEntryPointFromManifest } from '../../utils/manifest.js';\nimport { isIpPort } from '../../utils/validation.js';\nimport { getCustomConditions } from '../build/tsc.js';\n\nexport const start = async () => {\n const customConditions = getCustomConditions();\n const [args, availablePort] = await Promise.all([\n parseRunArgs(process.argv.slice(2)),\n getPort(),\n ]);\n\n const uniqueConditions = [\n ...new Set([...(args.conditions ?? []), ...customConditions]),\n ];\n\n args.entryPoint ??= await getEntryPointFromManifest();\n\n const execProcess = createExec({\n env: {\n __SKUBA_ENTRY_POINT: args.entryPoint,\n __SKUBA_PORT: String(isIpPort(args.port) ? args.port : availablePort),\n },\n });\n\n return execProcess(\n 'tsx',\n 'watch',\n '--clear-screen=false',\n ...args.node,\n ...uniqueConditions.map((condition) => `--conditions=${condition}`),\n '--env-file-if-exists',\n '.env',\n '--require',\n 'tsconfig-paths/register',\n path.join(__dirname, '..', '..', 'wrapper', 'index.js'),\n ...args.script,\n );\n};\n"],
5
+ "mappings": ";;;;;;;;;;;;;;;;;;;;;;;;;;;;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,kBAAiB;AAEjB,sBAAoB;AAEpB,kBAA6B;AAC7B,kBAA2B;AAC3B,sBAA0C;AAC1C,wBAAyB;AACzB,iBAAoC;AAE7B,MAAM,QAAQ,YAAY;AAC/B,QAAM,uBAAmB,gCAAoB;AAC7C,QAAM,CAAC,MAAM,aAAa,IAAI,MAAM,QAAQ,IAAI;AAAA,QAC9C,0BAAa,QAAQ,KAAK,MAAM,CAAC,CAAC;AAAA,QAClC,gBAAAA,SAAQ;AAAA,EACV,CAAC;AAED,QAAM,mBAAmB;AAAA,IACvB,GAAG,oBAAI,IAAI,CAAC,GAAI,KAAK,cAAc,CAAC,GAAI,GAAG,gBAAgB,CAAC;AAAA,EAC9D;AAEA,OAAK,eAAe,UAAM,2CAA0B;AAEpD,QAAM,kBAAc,wBAAW;AAAA,IAC7B,KAAK;AAAA,MACH,qBAAqB,KAAK;AAAA,MAC1B,cAAc,WAAO,4BAAS,KAAK,IAAI,IAAI,KAAK,OAAO,aAAa;AAAA,IACtE;AAAA,EACF,CAAC;AAED,SAAO;AAAA,IACL;AAAA,IACA;AAAA,IACA;AAAA,IACA,GAAG,KAAK;AAAA,IACR,GAAG,iBAAiB,IAAI,CAAC,cAAc,gBAAgB,SAAS,EAAE;AAAA,IAClE;AAAA,IACA;AAAA,IACA;AAAA,IACA;AAAA,IACA,YAAAC,QAAK,KAAK,WAAW,MAAM,MAAM,WAAW,UAAU;AAAA,IACtD,GAAG,KAAK;AAAA,EACV;AACF;",
6
6
  "names": ["getPort", "path"]
7
7
  }
@@ -1 +1 @@
1
- export declare const test: () => Promise<void>;
1
+ export declare const test: () => Promise<import("execa").ExecaReturnValue<string>>;
@@ -1,7 +1,9 @@
1
1
  "use strict";
2
+ var __create = Object.create;
2
3
  var __defProp = Object.defineProperty;
3
4
  var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
4
5
  var __getOwnPropNames = Object.getOwnPropertyNames;
6
+ var __getProtoOf = Object.getPrototypeOf;
5
7
  var __hasOwnProp = Object.prototype.hasOwnProperty;
6
8
  var __export = (target, all) => {
7
9
  for (var name in all)
@@ -15,18 +17,30 @@ var __copyProps = (to, from, except, desc) => {
15
17
  }
16
18
  return to;
17
19
  };
20
+ var __toESM = (mod, isNodeMode, target) => (target = mod != null ? __create(__getProtoOf(mod)) : {}, __copyProps(
21
+ // If the importer is in node compatibility mode or this is not an ESM
22
+ // file that has been converted to a CommonJS file using a Babel-
23
+ // compatible transform (i.e. "__esModule" has not been set), then set
24
+ // "default" to the CommonJS "module.exports" for node compatibility.
25
+ isNodeMode || !mod || !mod.__esModule ? __defProp(target, "default", { value: mod, enumerable: true }) : target,
26
+ mod
27
+ ));
18
28
  var __toCommonJS = (mod) => __copyProps(__defProp({}, "__esModule", { value: true }), mod);
19
29
  var test_exports = {};
20
30
  __export(test_exports, {
21
31
  test: () => test
22
32
  });
23
33
  module.exports = __toCommonJS(test_exports);
24
- var import_jest = require("jest");
34
+ var import_exec = require("../../utils/exec.js");
25
35
  const test = async () => {
26
- process.env.NODE_ENV ??= "test";
27
- process.env.TS_JEST_LOG ??= "stdout:error";
28
36
  const argv = process.argv.slice(2);
29
- return (0, import_jest.run)(argv);
37
+ const nodeOptions = process.env.NODE_OPTIONS ?? "";
38
+ const execWithEnv = (0, import_exec.createExec)({
39
+ env: {
40
+ NODE_OPTIONS: !nodeOptions.includes("--experimental-vm-modules") ? `${nodeOptions} --experimental-vm-modules --no-warnings=ExperimentalWarning` : nodeOptions
41
+ }
42
+ });
43
+ return execWithEnv(require.resolve("jest/bin/jest"), ...argv);
30
44
  };
31
45
  // Annotate the CommonJS export names for ESM import in node:
32
46
  0 && (module.exports = {
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "version": 3,
3
3
  "sources": ["../../../src/cli/test/index.ts"],
4
- "sourcesContent": ["import { run } from 'jest';\n\nexport const test = async () => {\n // This is usually set in `jest-cli`'s binary wrapper\n process.env.NODE_ENV ??= 'test';\n\n // ts-jest is logging a warning about `isolatedModules`.\n // This is a workaround until we can remove the `isolatedModules` option.\n // https://github.com/seek-oss/skuba/issues/1841\n process.env.TS_JEST_LOG ??= 'stdout:error';\n\n const argv = process.argv.slice(2);\n\n return run(argv);\n};\n"],
5
- "mappings": ";;;;;;;;;;;;;;;;;;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,kBAAoB;AAEb,MAAM,OAAO,YAAY;AAE9B,UAAQ,IAAI,aAAa;AAKzB,UAAQ,IAAI,gBAAgB;AAE5B,QAAM,OAAO,QAAQ,KAAK,MAAM,CAAC;AAEjC,aAAO,iBAAI,IAAI;AACjB;",
4
+ "sourcesContent": ["import { createExec } from '../../utils/exec.js';\n\nexport const test = async () => {\n const argv = process.argv.slice(2);\n\n const nodeOptions = process.env.NODE_OPTIONS ?? '';\n\n const execWithEnv = createExec({\n env: {\n NODE_OPTIONS: !nodeOptions.includes('--experimental-vm-modules')\n ? `${nodeOptions} --experimental-vm-modules --no-warnings=ExperimentalWarning`\n : nodeOptions,\n },\n });\n\n // Run Jest in a child process with proper environment\n return execWithEnv(require.resolve('jest/bin/jest'), ...argv);\n};\n"],
5
+ "mappings": ";;;;;;;;;;;;;;;;;;;;;;;;;;;;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,kBAA2B;AAEpB,MAAM,OAAO,YAAY;AAC9B,QAAM,OAAO,QAAQ,KAAK,MAAM,CAAC;AAEjC,QAAM,cAAc,QAAQ,IAAI,gBAAgB;AAEhD,QAAM,kBAAc,wBAAW;AAAA,IAC7B,KAAK;AAAA,MACH,cAAc,CAAC,YAAY,SAAS,2BAA2B,IAC3D,GAAG,WAAW,iEACd;AAAA,IACN;AAAA,EACF,CAAC;AAGD,SAAO,YAAY,gBAAgB,eAAe,GAAG,GAAG,IAAI;AAC9D;",
6
6
  "names": []
7
7
  }
@@ -18,6 +18,8 @@ export declare const parseProcessArgs: (args?: string[]) => {
18
18
  args: string[];
19
19
  };
20
20
  interface RunArgs {
21
+ /** The conditions to pass to the entry point script. */
22
+ conditions?: string[];
21
23
  /** The path to the entry point script. */
22
24
  entryPoint?: string;
23
25
  /** The port to start an HTTP server on. */
package/lib/utils/args.js CHANGED
@@ -100,6 +100,11 @@ const parseRunArgsIteration = (state, args) => {
100
100
  }
101
101
  return args.slice(1);
102
102
  }
103
+ if (arg1.startsWith("--conditions=")) {
104
+ state.conditions ??= [];
105
+ state.conditions.push(arg1.slice(13));
106
+ return args.slice(1);
107
+ }
103
108
  state.entryPoint = arg1;
104
109
  state.script.push(...args.slice(1));
105
110
  return [];
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "version": 3,
3
3
  "sources": ["../../src/utils/args.ts"],
4
- "sourcesContent": ["import assert from 'assert';\n\nimport { COMMAND_ALIASES } from './command.js';\n\nexport const hasDebugFlag = (args = process.argv) =>\n args.some((arg) => arg.toLocaleLowerCase() === '--debug');\n\nexport const hasSerialFlag = (args = process.argv, env = process.env) =>\n args.some((arg) => arg.toLocaleLowerCase() === '--serial') ||\n Boolean(\n // Run serially on SEEK's central npm publishing pipeline.\n // Exhausting agents here can cause grief.\n env.BUILDKITE_AGENT_META_DATA_QUEUE?.split(',').some((queueName) =>\n queueName.startsWith('artefacts:npm'),\n ),\n );\n\n/**\n * Parse process arguments.\n *\n * This function mutates the input list by removing the command name element.\n * Downstream commands can access their args with an ordinary\n * `process.argv.slice(2)`.\n *\n * Example input:\n *\n * ```typescript\n * ['/bin/node', 'node_modules/.bin/skuba', 'test', '--foo', '--bar']\n * ```\n */\nexport const parseProcessArgs = (args = process.argv) => {\n const skubaIdx = args.findIndex((chunk) => /skuba(\\.[jt]s)?$/.test(chunk));\n\n assert(skubaIdx >= 0, 'Cannot parse args for `skuba`');\n\n const rawCommand = (args[skubaIdx + 1] ?? 'help').toLocaleLowerCase();\n\n const commandName = COMMAND_ALIASES[rawCommand] ?? rawCommand;\n\n const payload = {\n commandName,\n args: args.slice(skubaIdx + 2),\n };\n\n args.splice(skubaIdx + 1, 1);\n\n return payload;\n};\n\ninterface RunArgs {\n /** The path to the entry point script. */\n entryPoint?: string;\n\n /** The port to start an HTTP server on. */\n port?: number;\n\n /** Arguments passed through to the Node.js executable. */\n node: string[];\n\n /** Arguments passed through to the entry point script. */\n script: string[];\n}\n\n/**\n * Make a best effort to parse \"run\" args.\n *\n * These are arguments that would be passed to `skuba node` or `skuba start`.\n * Parsing is handrolled because we support some weird and wonderful behaviour:\n *\n * - The `--port` option may be intended for skuba itself\n * - The `--inspect` options may be intended for the Node.js inspector\n * - The entry point may be omitted in favour of `package.json` inference\n * - Other args may be intended for propagation to the entry point\n */\nexport const parseRunArgs = (argv: string[]): RunArgs => {\n const state: RunArgs = {\n node: [],\n script: [],\n };\n\n let args = argv.filter((element) => element.length);\n\n while (args.length) {\n args = parseRunArgsIteration(state, args);\n }\n\n return state;\n};\n\nconst isDigits = (arg: unknown): arg is string =>\n typeof arg === 'string' && /^\\d+$/.test(arg);\n\nconst parseRunArgsIteration = (state: RunArgs, args: string[]): string[] => {\n const [arg1, arg2] = args;\n\n if (!arg1) {\n return [];\n }\n\n if (/^--inspect(-brk)?=\\d+$/.test(arg1)) {\n state.node.push(arg1);\n return args.slice(1);\n }\n\n // Node.js inspector options that are optionally followed by a numeric port.\n if (['--inspect', '--inspect-brk'].includes(arg1)) {\n if (isDigits(arg2)) {\n state.node.push(`${arg1}=${arg2}`);\n return args.slice(2);\n }\n\n state.node.push(arg1);\n\n if (arg2) {\n // Some other string that doesn't relate to the Node.js inspector option.\n // This is presumably the entry point script to run.\n state.entryPoint = arg2;\n state.script.push(...args.slice(2));\n }\n\n return [];\n }\n\n if (/^--port=\\d+$/.test(arg1)) {\n state.port = Number(arg1.slice(7));\n return args.slice(1);\n }\n\n if (arg1 === '--port') {\n if (isDigits(arg2)) {\n state.port = Number(arg2);\n return args.slice(2);\n }\n\n // Invalid port argument; eat it.\n return args.slice(1);\n }\n\n state.entryPoint = arg1;\n state.script.push(...args.slice(1));\n return [];\n};\n"],
5
- "mappings": ";;;;;;;;;;;;;;;;;;;;;;;;;;;;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,oBAAmB;AAEnB,qBAAgC;AAEzB,MAAM,eAAe,CAAC,OAAO,QAAQ,SAC1C,KAAK,KAAK,CAAC,QAAQ,IAAI,kBAAkB,MAAM,SAAS;AAEnD,MAAM,gBAAgB,CAAC,OAAO,QAAQ,MAAM,MAAM,QAAQ,QAC/D,KAAK,KAAK,CAAC,QAAQ,IAAI,kBAAkB,MAAM,UAAU,KACzD;AAAA;AAAA;AAAA,EAGE,IAAI,iCAAiC,MAAM,GAAG,EAAE;AAAA,IAAK,CAAC,cACpD,UAAU,WAAW,eAAe;AAAA,EACtC;AACF;AAeK,MAAM,mBAAmB,CAAC,OAAO,QAAQ,SAAS;AACvD,QAAM,WAAW,KAAK,UAAU,CAAC,UAAU,mBAAmB,KAAK,KAAK,CAAC;AAEzE,oBAAAA,SAAO,YAAY,GAAG,+BAA+B;AAErD,QAAM,cAAc,KAAK,WAAW,CAAC,KAAK,QAAQ,kBAAkB;AAEpE,QAAM,cAAc,+BAAgB,UAAU,KAAK;AAEnD,QAAM,UAAU;AAAA,IACd;AAAA,IACA,MAAM,KAAK,MAAM,WAAW,CAAC;AAAA,EAC/B;AAEA,OAAK,OAAO,WAAW,GAAG,CAAC;AAE3B,SAAO;AACT;AA2BO,MAAM,eAAe,CAAC,SAA4B;AACvD,QAAM,QAAiB;AAAA,IACrB,MAAM,CAAC;AAAA,IACP,QAAQ,CAAC;AAAA,EACX;AAEA,MAAI,OAAO,KAAK,OAAO,CAAC,YAAY,QAAQ,MAAM;AAElD,SAAO,KAAK,QAAQ;AAClB,WAAO,sBAAsB,OAAO,IAAI;AAAA,EAC1C;AAEA,SAAO;AACT;AAEA,MAAM,WAAW,CAAC,QAChB,OAAO,QAAQ,YAAY,QAAQ,KAAK,GAAG;AAE7C,MAAM,wBAAwB,CAAC,OAAgB,SAA6B;AAC1E,QAAM,CAAC,MAAM,IAAI,IAAI;AAErB,MAAI,CAAC,MAAM;AACT,WAAO,CAAC;AAAA,EACV;AAEA,MAAI,yBAAyB,KAAK,IAAI,GAAG;AACvC,UAAM,KAAK,KAAK,IAAI;AACpB,WAAO,KAAK,MAAM,CAAC;AAAA,EACrB;AAGA,MAAI,CAAC,aAAa,eAAe,EAAE,SAAS,IAAI,GAAG;AACjD,QAAI,SAAS,IAAI,GAAG;AAClB,YAAM,KAAK,KAAK,GAAG,IAAI,IAAI,IAAI,EAAE;AACjC,aAAO,KAAK,MAAM,CAAC;AAAA,IACrB;AAEA,UAAM,KAAK,KAAK,IAAI;AAEpB,QAAI,MAAM;AAGR,YAAM,aAAa;AACnB,YAAM,OAAO,KAAK,GAAG,KAAK,MAAM,CAAC,CAAC;AAAA,IACpC;AAEA,WAAO,CAAC;AAAA,EACV;AAEA,MAAI,eAAe,KAAK,IAAI,GAAG;AAC7B,UAAM,OAAO,OAAO,KAAK,MAAM,CAAC,CAAC;AACjC,WAAO,KAAK,MAAM,CAAC;AAAA,EACrB;AAEA,MAAI,SAAS,UAAU;AACrB,QAAI,SAAS,IAAI,GAAG;AAClB,YAAM,OAAO,OAAO,IAAI;AACxB,aAAO,KAAK,MAAM,CAAC;AAAA,IACrB;AAGA,WAAO,KAAK,MAAM,CAAC;AAAA,EACrB;AAEA,QAAM,aAAa;AACnB,QAAM,OAAO,KAAK,GAAG,KAAK,MAAM,CAAC,CAAC;AAClC,SAAO,CAAC;AACV;",
4
+ "sourcesContent": ["import assert from 'assert';\n\nimport { COMMAND_ALIASES } from './command.js';\n\nexport const hasDebugFlag = (args = process.argv) =>\n args.some((arg) => arg.toLocaleLowerCase() === '--debug');\n\nexport const hasSerialFlag = (args = process.argv, env = process.env) =>\n args.some((arg) => arg.toLocaleLowerCase() === '--serial') ||\n Boolean(\n // Run serially on SEEK's central npm publishing pipeline.\n // Exhausting agents here can cause grief.\n env.BUILDKITE_AGENT_META_DATA_QUEUE?.split(',').some((queueName) =>\n queueName.startsWith('artefacts:npm'),\n ),\n );\n\n/**\n * Parse process arguments.\n *\n * This function mutates the input list by removing the command name element.\n * Downstream commands can access their args with an ordinary\n * `process.argv.slice(2)`.\n *\n * Example input:\n *\n * ```typescript\n * ['/bin/node', 'node_modules/.bin/skuba', 'test', '--foo', '--bar']\n * ```\n */\nexport const parseProcessArgs = (args = process.argv) => {\n const skubaIdx = args.findIndex((chunk) => /skuba(\\.[jt]s)?$/.test(chunk));\n\n assert(skubaIdx >= 0, 'Cannot parse args for `skuba`');\n\n const rawCommand = (args[skubaIdx + 1] ?? 'help').toLocaleLowerCase();\n\n const commandName = COMMAND_ALIASES[rawCommand] ?? rawCommand;\n\n const payload = {\n commandName,\n args: args.slice(skubaIdx + 2),\n };\n\n args.splice(skubaIdx + 1, 1);\n\n return payload;\n};\n\ninterface RunArgs {\n /** The conditions to pass to the entry point script. */\n conditions?: string[];\n\n /** The path to the entry point script. */\n entryPoint?: string;\n\n /** The port to start an HTTP server on. */\n port?: number;\n\n /** Arguments passed through to the Node.js executable. */\n node: string[];\n\n /** Arguments passed through to the entry point script. */\n script: string[];\n}\n\n/**\n * Make a best effort to parse \"run\" args.\n *\n * These are arguments that would be passed to `skuba node` or `skuba start`.\n * Parsing is handrolled because we support some weird and wonderful behaviour:\n *\n * - The `--port` option may be intended for skuba itself\n * - The `--inspect` options may be intended for the Node.js inspector\n * - The entry point may be omitted in favour of `package.json` inference\n * - Other args may be intended for propagation to the entry point\n */\nexport const parseRunArgs = (argv: string[]): RunArgs => {\n const state: RunArgs = {\n node: [],\n script: [],\n };\n\n let args = argv.filter((element) => element.length);\n\n while (args.length) {\n args = parseRunArgsIteration(state, args);\n }\n\n return state;\n};\n\nconst isDigits = (arg: unknown): arg is string =>\n typeof arg === 'string' && /^\\d+$/.test(arg);\n\nconst parseRunArgsIteration = (state: RunArgs, args: string[]): string[] => {\n const [arg1, arg2] = args;\n\n if (!arg1) {\n return [];\n }\n\n if (/^--inspect(-brk)?=\\d+$/.test(arg1)) {\n state.node.push(arg1);\n return args.slice(1);\n }\n\n // Node.js inspector options that are optionally followed by a numeric port.\n if (['--inspect', '--inspect-brk'].includes(arg1)) {\n if (isDigits(arg2)) {\n state.node.push(`${arg1}=${arg2}`);\n return args.slice(2);\n }\n\n state.node.push(arg1);\n\n if (arg2) {\n // Some other string that doesn't relate to the Node.js inspector option.\n // This is presumably the entry point script to run.\n state.entryPoint = arg2;\n state.script.push(...args.slice(2));\n }\n\n return [];\n }\n\n if (/^--port=\\d+$/.test(arg1)) {\n state.port = Number(arg1.slice(7));\n return args.slice(1);\n }\n\n if (arg1 === '--port') {\n if (isDigits(arg2)) {\n state.port = Number(arg2);\n return args.slice(2);\n }\n\n // Invalid port argument; eat it.\n return args.slice(1);\n }\n\n if (arg1.startsWith('--conditions=')) {\n state.conditions ??= [];\n state.conditions.push(arg1.slice(13));\n return args.slice(1);\n }\n\n state.entryPoint = arg1;\n state.script.push(...args.slice(1));\n return [];\n};\n"],
5
+ "mappings": ";;;;;;;;;;;;;;;;;;;;;;;;;;;;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,oBAAmB;AAEnB,qBAAgC;AAEzB,MAAM,eAAe,CAAC,OAAO,QAAQ,SAC1C,KAAK,KAAK,CAAC,QAAQ,IAAI,kBAAkB,MAAM,SAAS;AAEnD,MAAM,gBAAgB,CAAC,OAAO,QAAQ,MAAM,MAAM,QAAQ,QAC/D,KAAK,KAAK,CAAC,QAAQ,IAAI,kBAAkB,MAAM,UAAU,KACzD;AAAA;AAAA;AAAA,EAGE,IAAI,iCAAiC,MAAM,GAAG,EAAE;AAAA,IAAK,CAAC,cACpD,UAAU,WAAW,eAAe;AAAA,EACtC;AACF;AAeK,MAAM,mBAAmB,CAAC,OAAO,QAAQ,SAAS;AACvD,QAAM,WAAW,KAAK,UAAU,CAAC,UAAU,mBAAmB,KAAK,KAAK,CAAC;AAEzE,oBAAAA,SAAO,YAAY,GAAG,+BAA+B;AAErD,QAAM,cAAc,KAAK,WAAW,CAAC,KAAK,QAAQ,kBAAkB;AAEpE,QAAM,cAAc,+BAAgB,UAAU,KAAK;AAEnD,QAAM,UAAU;AAAA,IACd;AAAA,IACA,MAAM,KAAK,MAAM,WAAW,CAAC;AAAA,EAC/B;AAEA,OAAK,OAAO,WAAW,GAAG,CAAC;AAE3B,SAAO;AACT;AA8BO,MAAM,eAAe,CAAC,SAA4B;AACvD,QAAM,QAAiB;AAAA,IACrB,MAAM,CAAC;AAAA,IACP,QAAQ,CAAC;AAAA,EACX;AAEA,MAAI,OAAO,KAAK,OAAO,CAAC,YAAY,QAAQ,MAAM;AAElD,SAAO,KAAK,QAAQ;AAClB,WAAO,sBAAsB,OAAO,IAAI;AAAA,EAC1C;AAEA,SAAO;AACT;AAEA,MAAM,WAAW,CAAC,QAChB,OAAO,QAAQ,YAAY,QAAQ,KAAK,GAAG;AAE7C,MAAM,wBAAwB,CAAC,OAAgB,SAA6B;AAC1E,QAAM,CAAC,MAAM,IAAI,IAAI;AAErB,MAAI,CAAC,MAAM;AACT,WAAO,CAAC;AAAA,EACV;AAEA,MAAI,yBAAyB,KAAK,IAAI,GAAG;AACvC,UAAM,KAAK,KAAK,IAAI;AACpB,WAAO,KAAK,MAAM,CAAC;AAAA,EACrB;AAGA,MAAI,CAAC,aAAa,eAAe,EAAE,SAAS,IAAI,GAAG;AACjD,QAAI,SAAS,IAAI,GAAG;AAClB,YAAM,KAAK,KAAK,GAAG,IAAI,IAAI,IAAI,EAAE;AACjC,aAAO,KAAK,MAAM,CAAC;AAAA,IACrB;AAEA,UAAM,KAAK,KAAK,IAAI;AAEpB,QAAI,MAAM;AAGR,YAAM,aAAa;AACnB,YAAM,OAAO,KAAK,GAAG,KAAK,MAAM,CAAC,CAAC;AAAA,IACpC;AAEA,WAAO,CAAC;AAAA,EACV;AAEA,MAAI,eAAe,KAAK,IAAI,GAAG;AAC7B,UAAM,OAAO,OAAO,KAAK,MAAM,CAAC,CAAC;AACjC,WAAO,KAAK,MAAM,CAAC;AAAA,EACrB;AAEA,MAAI,SAAS,UAAU;AACrB,QAAI,SAAS,IAAI,GAAG;AAClB,YAAM,OAAO,OAAO,IAAI;AACxB,aAAO,KAAK,MAAM,CAAC;AAAA,IACrB;AAGA,WAAO,KAAK,MAAM,CAAC;AAAA,EACrB;AAEA,MAAI,KAAK,WAAW,eAAe,GAAG;AACpC,UAAM,eAAe,CAAC;AACtB,UAAM,WAAW,KAAK,KAAK,MAAM,EAAE,CAAC;AACpC,WAAO,KAAK,MAAM,CAAC;AAAA,EACrB;AAEA,QAAM,aAAa;AACnB,QAAM,OAAO,KAAK,GAAG,KAAK,MAAM,CAAC,CAAC;AAClC,SAAO,CAAC;AACV;",
6
6
  "names": ["assert"]
7
7
  }
@@ -4,7 +4,7 @@ export type ProjectType = z.infer<typeof projectTypeSchema>;
4
4
  export declare const projectTypeSchema: z.ZodUnion<readonly [z.ZodLiteral<"application">, z.ZodLiteral<"package">]>;
5
5
  export declare const PROJECT_TYPES: readonly ["application", "package"];
6
6
  export declare const getSkubaManifest: () => Promise<NormalizedPackageJson>;
7
- export declare const getConsumerManifest: (cwd?: string) => Promise<readPkgUp.NormalizedReadResult | undefined>;
7
+ export declare const getConsumerManifest: (cwd?: string) => Promise<readPkgUp.ReadResult | undefined>;
8
8
  export declare const getPropFromConsumerManifest: <T extends string, V = unknown>(prop: T) => Promise<V | undefined>;
9
9
  export declare const getStringPropFromConsumerManifest: <T extends string>(prop: T) => Promise<string | undefined>;
10
10
  export declare const getEntryPointFromManifest: () => Promise<string>;
@@ -57,7 +57,7 @@ const getSkubaManifest = async () => {
57
57
  }
58
58
  return skubaManifest = result.packageJson;
59
59
  };
60
- const getConsumerManifest = (cwd) => (0, import_read_pkg_up.default)({ cwd });
60
+ const getConsumerManifest = (cwd) => (0, import_read_pkg_up.default)({ cwd, normalize: false });
61
61
  const getPropFromConsumerManifest = async (prop) => {
62
62
  const result = await getConsumerManifest();
63
63
  return result !== void 0 && (0, import_validation.hasProp)(result.packageJson.skuba, prop) ? result.packageJson.skuba[prop] : void 0;
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "version": 3,
3
3
  "sources": ["../../src/utils/manifest.ts"],
4
- "sourcesContent": ["import readPkgUp, { type NormalizedPackageJson } from 'read-pkg-up';\nimport * as z from 'zod/v4';\n\nimport { hasProp } from './validation.js';\n\nexport type ProjectType = z.infer<typeof projectTypeSchema>;\n\nexport const projectTypeSchema = z.union([\n z.literal('application'),\n z.literal('package'),\n]);\n\nexport const PROJECT_TYPES = ['application', 'package'] as const;\n\nconst DEFAULT_ENTRY_POINT = 'src/app.ts';\n\nlet skubaManifest: NormalizedPackageJson | undefined;\n\nexport const getSkubaManifest = async (): Promise<NormalizedPackageJson> => {\n if (skubaManifest !== undefined) {\n return skubaManifest;\n }\n\n const result = await readPkgUp({ cwd: __dirname });\n\n if (result === undefined) {\n throw Error('skuba could not find its own manifest');\n }\n\n return (skubaManifest = result.packageJson);\n};\n\nexport const getConsumerManifest = (cwd?: string) => readPkgUp({ cwd });\n\nexport const getPropFromConsumerManifest = async <\n T extends string,\n V = unknown,\n>(\n prop: T,\n): Promise<V | undefined> => {\n const result = await getConsumerManifest();\n\n return result !== undefined && hasProp<T, V>(result.packageJson.skuba, prop)\n ? result.packageJson.skuba[prop]\n : undefined;\n};\n\nexport const getStringPropFromConsumerManifest = async <T extends string>(\n prop: T,\n): Promise<string | undefined> => {\n const result = await getPropFromConsumerManifest(prop);\n\n return typeof result === 'string' ? result : undefined;\n};\n\nexport const getEntryPointFromManifest = async (): Promise<string> => {\n const entryPoint = await getStringPropFromConsumerManifest('entryPoint');\n\n return entryPoint ?? DEFAULT_ENTRY_POINT;\n};\n"],
5
- "mappings": ";;;;;;;;;;;;;;;;;;;;;;;;;;;;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,yBAAsD;AACtD,QAAmB;AAEnB,wBAAwB;AAIjB,MAAM,oBAAoB,EAAE,MAAM;AAAA,EACvC,EAAE,QAAQ,aAAa;AAAA,EACvB,EAAE,QAAQ,SAAS;AACrB,CAAC;AAEM,MAAM,gBAAgB,CAAC,eAAe,SAAS;AAEtD,MAAM,sBAAsB;AAE5B,IAAI;AAEG,MAAM,mBAAmB,YAA4C;AAC1E,MAAI,kBAAkB,QAAW;AAC/B,WAAO;AAAA,EACT;AAEA,QAAM,SAAS,UAAM,mBAAAA,SAAU,EAAE,KAAK,UAAU,CAAC;AAEjD,MAAI,WAAW,QAAW;AACxB,UAAM,MAAM,uCAAuC;AAAA,EACrD;AAEA,SAAQ,gBAAgB,OAAO;AACjC;AAEO,MAAM,sBAAsB,CAAC,YAAiB,mBAAAA,SAAU,EAAE,IAAI,CAAC;AAE/D,MAAM,8BAA8B,OAIzC,SAC2B;AAC3B,QAAM,SAAS,MAAM,oBAAoB;AAEzC,SAAO,WAAW,cAAa,2BAAc,OAAO,YAAY,OAAO,IAAI,IACvE,OAAO,YAAY,MAAM,IAAI,IAC7B;AACN;AAEO,MAAM,oCAAoC,OAC/C,SACgC;AAChC,QAAM,SAAS,MAAM,4BAA4B,IAAI;AAErD,SAAO,OAAO,WAAW,WAAW,SAAS;AAC/C;AAEO,MAAM,4BAA4B,YAA6B;AACpE,QAAM,aAAa,MAAM,kCAAkC,YAAY;AAEvE,SAAO,cAAc;AACvB;",
4
+ "sourcesContent": ["import readPkgUp, { type NormalizedPackageJson } from 'read-pkg-up';\nimport * as z from 'zod/v4';\n\nimport { hasProp } from './validation.js';\n\nexport type ProjectType = z.infer<typeof projectTypeSchema>;\n\nexport const projectTypeSchema = z.union([\n z.literal('application'),\n z.literal('package'),\n]);\n\nexport const PROJECT_TYPES = ['application', 'package'] as const;\n\nconst DEFAULT_ENTRY_POINT = 'src/app.ts';\n\nlet skubaManifest: NormalizedPackageJson | undefined;\n\nexport const getSkubaManifest = async (): Promise<NormalizedPackageJson> => {\n if (skubaManifest !== undefined) {\n return skubaManifest;\n }\n\n const result = await readPkgUp({ cwd: __dirname });\n\n if (result === undefined) {\n throw Error('skuba could not find its own manifest');\n }\n\n return (skubaManifest = result.packageJson);\n};\n\nexport const getConsumerManifest = (cwd?: string) =>\n readPkgUp({ cwd, normalize: false });\n\nexport const getPropFromConsumerManifest = async <\n T extends string,\n V = unknown,\n>(\n prop: T,\n): Promise<V | undefined> => {\n const result = await getConsumerManifest();\n\n return result !== undefined && hasProp<T, V>(result.packageJson.skuba, prop)\n ? result.packageJson.skuba[prop]\n : undefined;\n};\n\nexport const getStringPropFromConsumerManifest = async <T extends string>(\n prop: T,\n): Promise<string | undefined> => {\n const result = await getPropFromConsumerManifest(prop);\n\n return typeof result === 'string' ? result : undefined;\n};\n\nexport const getEntryPointFromManifest = async (): Promise<string> => {\n const entryPoint = await getStringPropFromConsumerManifest('entryPoint');\n\n return entryPoint ?? DEFAULT_ENTRY_POINT;\n};\n"],
5
+ "mappings": ";;;;;;;;;;;;;;;;;;;;;;;;;;;;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,yBAAsD;AACtD,QAAmB;AAEnB,wBAAwB;AAIjB,MAAM,oBAAoB,EAAE,MAAM;AAAA,EACvC,EAAE,QAAQ,aAAa;AAAA,EACvB,EAAE,QAAQ,SAAS;AACrB,CAAC;AAEM,MAAM,gBAAgB,CAAC,eAAe,SAAS;AAEtD,MAAM,sBAAsB;AAE5B,IAAI;AAEG,MAAM,mBAAmB,YAA4C;AAC1E,MAAI,kBAAkB,QAAW;AAC/B,WAAO;AAAA,EACT;AAEA,QAAM,SAAS,UAAM,mBAAAA,SAAU,EAAE,KAAK,UAAU,CAAC;AAEjD,MAAI,WAAW,QAAW;AACxB,UAAM,MAAM,uCAAuC;AAAA,EACrD;AAEA,SAAQ,gBAAgB,OAAO;AACjC;AAEO,MAAM,sBAAsB,CAAC,YAClC,mBAAAA,SAAU,EAAE,KAAK,WAAW,MAAM,CAAC;AAE9B,MAAM,8BAA8B,OAIzC,SAC2B;AAC3B,QAAM,SAAS,MAAM,oBAAoB;AAEzC,SAAO,WAAW,cAAa,2BAAc,OAAO,YAAY,OAAO,IAAI,IACvE,OAAO,YAAY,MAAM,IAAI,IAC7B;AACN;AAEO,MAAM,oCAAoC,OAC/C,SACgC;AAChC,QAAM,SAAS,MAAM,4BAA4B,IAAI;AAErD,SAAO,OAAO,WAAW,WAAW,SAAS;AAC/C;AAEO,MAAM,4BAA4B,YAA6B;AACpE,QAAM,aAAa,MAAM,kCAAkC,YAAY;AAEvE,SAAO,cAAc;AACvB;",
6
6
  "names": ["readPkgUp"]
7
7
  }
@@ -23,7 +23,7 @@ export declare const templateConfigSchema: z.ZodObject<{
23
23
  name: z.ZodString;
24
24
  message: z.ZodString;
25
25
  initial: z.ZodString;
26
- validate: z.ZodOptional<z.ZodPipe<z.ZodCustom<z.core.$InferInnerFunctionType<z.core.$ZodTuple<[z.ZodString], null>, z.ZodUnion<readonly [z.ZodBoolean, z.ZodString]>>, z.core.$InferInnerFunctionType<z.core.$ZodTuple<[z.ZodString], null>, z.ZodUnion<readonly [z.ZodBoolean, z.ZodString]>>>, z.ZodTransform<(args_0: string) => string | boolean, z.core.$InferInnerFunctionType<z.core.$ZodTuple<[z.ZodString], null>, z.ZodUnion<readonly [z.ZodBoolean, z.ZodString]>>>>>;
26
+ validate: z.ZodOptional<z.ZodPipe<z.ZodCustom<z.core.$InferInnerFunctionType<z.ZodTuple<readonly [z.ZodString], null>, z.ZodUnion<readonly [z.ZodBoolean, z.ZodString]>>, z.core.$InferInnerFunctionType<z.ZodTuple<readonly [z.ZodString], null>, z.ZodUnion<readonly [z.ZodBoolean, z.ZodString]>>>, z.ZodTransform<(args_0: string) => string | boolean, z.core.$InferInnerFunctionType<z.ZodTuple<readonly [z.ZodString], null>, z.ZodUnion<readonly [z.ZodBoolean, z.ZodString]>>>>>;
27
27
  }, z.core.$strip>>;
28
28
  entryPoint: z.ZodOptional<z.ZodString>;
29
29
  noSkip: z.ZodOptional<z.ZodBoolean>;
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "skuba",
3
- "version": "13.0.0-custom-conditions-exports-20250730235248",
3
+ "version": "13.0.0-custom-conditions-exports-20250923005538",
4
4
  "private": false,
5
5
  "description": "SEEK development toolkit for backend applications and packages",
6
6
  "homepage": "https://github.com/seek-oss/skuba#readme",
@@ -101,15 +101,15 @@
101
101
  "tsconfig-paths": "^4.0.0",
102
102
  "tsconfig-seek": "2.0.0",
103
103
  "tsx": "^4.16.2",
104
- "typescript": "~5.8.0",
105
- "zod": "^3.25.67",
106
- "eslint-config-skuba": "7.0.2"
104
+ "typescript": "~5.9.0",
105
+ "zod": "^4.0.0",
106
+ "eslint-config-skuba": "7.1.1"
107
107
  },
108
108
  "devDependencies": {
109
- "@changesets/cli": "2.29.5",
109
+ "@changesets/cli": "2.29.6",
110
110
  "@changesets/get-github-info": "0.6.0",
111
- "@jest/reporters": "30.0.4",
112
- "@jest/test-result": "30.0.4",
111
+ "@jest/reporters": "30.1.3",
112
+ "@jest/test-result": "30.1.3",
113
113
  "@types/ejs": "3.1.5",
114
114
  "@types/express": "5.0.3",
115
115
  "@types/fs-extra": "11.0.4",
@@ -119,20 +119,20 @@
119
119
  "@types/module-alias": "2.0.4",
120
120
  "@types/npm-registry-fetch": "8.0.8",
121
121
  "@types/npm-which": "3.0.4",
122
- "@types/picomatch": "4.0.0",
123
- "@types/semver": "7.7.0",
122
+ "@types/picomatch": "4.0.2",
123
+ "@types/semver": "7.7.1",
124
124
  "@types/supertest": "6.0.3",
125
- "enhanced-resolve": "5.18.2",
125
+ "enhanced-resolve": "5.18.3",
126
126
  "express": "5.1.0",
127
- "fastify": "5.4.0",
128
- "jest-diff": "30.0.4",
129
- "jsonfile": "6.1.0",
127
+ "fastify": "5.5.0",
128
+ "jest-diff": "30.1.2",
129
+ "jsonfile": "6.2.0",
130
130
  "koa": "3.0.1",
131
- "memfs": "4.17.2",
131
+ "memfs": "4.38.2",
132
132
  "remark-cli": "12.0.1",
133
133
  "remark-preset-lint-recommended": "7.0.1",
134
134
  "semver": "7.7.2",
135
- "supertest": "7.1.3",
135
+ "supertest": "7.1.4",
136
136
  "type-fest": "2.19.0"
137
137
  },
138
138
  "peerDependencies": {
@@ -146,15 +146,12 @@
146
146
  "engines": {
147
147
  "node": ">=20.9.0"
148
148
  },
149
- "publishConfig": {
150
- "provenance": true
151
- },
152
149
  "skuba": {
153
150
  "build": "esbuild",
154
151
  "entryPoint": "src/index.ts",
155
152
  "template": null,
156
153
  "type": "package",
157
- "version": "12.0.2"
154
+ "version": "12.2.0"
158
155
  },
159
156
  "scripts": {
160
157
  "build": "scripts/build.sh",
@@ -168,7 +165,7 @@
168
165
  "lint:packages": "pnpm --filter '!./template/**' lint",
169
166
  "release": "pnpm --silent build && changeset publish",
170
167
  "skuba": "pnpm --silent build && pnpm --silent skuba:exec",
171
- "skuba:exec": "node --experimental-vm-modules --no-warnings=ExperimentalWarning lib/skuba",
168
+ "skuba:exec": "node lib/skuba",
172
169
  "stage": "changeset version && node ./.changeset/inject.js && pnpm format",
173
170
  "test": "pnpm --silent skuba test --selectProjects unit",
174
171
  "test:ci": "pnpm --silent skuba test --runInBand",
@@ -1 +1 @@
1
- export { default } from 'skuba/config/prettier.js';
1
+ module.exports = require('skuba/config/prettier');
@@ -1 +1 @@
1
- export { default } from 'eslint-config-skuba';
1
+ module.exports = require('eslint-config-skuba');
@@ -8,4 +8,5 @@ publicHoistPattern:
8
8
  - esbuild
9
9
  - jest
10
10
  - tsconfig-seek
11
+ - typescript
11
12
  # end managed by skuba
@@ -1,3 +1,3 @@
1
- process.env.ENVIRONMENT = 'test';
1
+ process.env.DEPLOYMENT = 'test';
2
2
 
3
3
  export {};
@@ -1,4 +1,7 @@
1
1
  {
2
+ "compilerOptions": {
3
+ "rootDir": "src"
4
+ },
2
5
  "exclude": ["**/__mocks__/**/*", "**/*.test.ts", "src/testing/**/*"],
3
6
  "extends": "./tsconfig.json",
4
7
  "include": ["src/**/*"]
@@ -1,6 +1,5 @@
1
1
  {
2
2
  "compilerOptions": {
3
- "baseUrl": ".",
4
3
  "customConditions": ["@seek/<%- serviceName %>/source"],
5
4
  "lib": ["ES2024"],
6
5
  "outDir": "lib",
@@ -77,6 +77,9 @@ steps:
77
77
  queue: <%- devBuildkiteQueueName %>
78
78
  label: 🤞 Deploy Dev
79
79
  concurrency_group: <%- teamName %>/deploy/gantry/<%- devGantryEnvironmentName %>
80
+ env:
81
+ DD_DEPLOYMENT_ENVIRONMENT: development
82
+ DD_DEPLOYMENT_SERVICE: <%- serviceName %>
80
83
  key: deploy-dev
81
84
  plugins:
82
85
  - seek-jobs/gantry#v4.0.0:
@@ -93,6 +96,9 @@ steps:
93
96
  branches: ${BUILDKITE_PIPELINE_DEFAULT_BRANCH}
94
97
  concurrency_group: <%- teamName %>/deploy/gantry/<%- prodGantryEnvironmentName %>
95
98
  depends_on: deploy-dev
99
+ env:
100
+ DD_DEPLOYMENT_ENVIRONMENT: production
101
+ DD_DEPLOYMENT_SERVICE: <%- serviceName %>
96
102
  plugins:
97
103
  - seek-jobs/gantry#v4.0.0:
98
104
  command: apply
@@ -1 +1 @@
1
- ENVIRONMENT=local
1
+ DEPLOYMENT=local
@@ -1,7 +1,11 @@
1
- environment: dev
1
+ deployment: dev
2
+ env: development
2
3
 
3
4
  maxInstanceCount: 1
4
5
  minInstanceCount: 1
5
6
 
6
7
  # Disable dashboard for cost savings
7
8
  cloudwatchDashboardDisabled: true
9
+
10
+ openTelemetry:
11
+ enabled: true
@@ -1,4 +1,8 @@
1
- environment: prod
1
+ deployment: prod
2
+ env: production
2
3
 
3
4
  maxInstanceCount: 10
4
5
  minInstanceCount: 3
6
+
7
+ openTelemetry:
8
+ enabled: true
@@ -28,4 +28,4 @@ ARG PORT=8001
28
28
  ENV PORT=${PORT}
29
29
  EXPOSE ${PORT}
30
30
 
31
- CMD ["lib/listen.js"]
31
+ CMD ["--require", "./lib/tracing.js", "./lib/listen.js"]
@@ -1,4 +1,4 @@
1
- # syntax=docker/dockerfile:1.17
1
+ # syntax=docker/dockerfile:1.18
2
2
 
3
3
  FROM public.ecr.aws/docker/library/node:22-alpine AS dev-deps
4
4
 
@@ -70,19 +70,19 @@ pnpm start:debug
70
70
 
71
71
  This project is deployed through a [Buildkite pipeline](.buildkite/pipeline.yml).
72
72
 
73
- - Commits to a feature branch can be deployed to the dev environment by unblocking a step in the Buildkite UI
74
- - Commits to the default branch are automatically deployed to the dev and prod environments in sequence
73
+ - Commits to a feature branch can be deployed to the development environment by unblocking a step in the Buildkite UI
74
+ - Commits to the default branch are automatically deployed to the development and production environments in sequence
75
75
 
76
76
  To rapidly roll back a change,
77
77
  retry an individual deployment step from the previous build in Buildkite.
78
- Note that this will introduce drift between the head of the default Git branch and the live environment;
78
+ Note that this will introduce drift between the head of the default Git branch and the live deployments;
79
79
  use with caution and always follow up with a proper revert or fix in Git history.
80
80
 
81
81
  ## Support
82
82
 
83
83
  ### Dev
84
84
 
85
- TODO: add support links for the dev environment.
85
+ TODO: add support links for the dev deployment.
86
86
 
87
87
  <!--
88
88
  - CloudWatch dashboard
@@ -92,7 +92,7 @@ TODO: add support links for the dev environment.
92
92
 
93
93
  ### Prod
94
94
 
95
- TODO: add support links for the prod environment.
95
+ TODO: add support links for the prod deployment.
96
96
 
97
97
  <!--
98
98
  - CloudWatch dashboard
@@ -9,9 +9,14 @@ owner: '{{values "owner"}}'
9
9
  image: '{{values "image"}}'
10
10
 
11
11
  env:
12
- ENVIRONMENT: '{{values "environment"}}'
12
+ DEPLOYMENT: '{{values "deployment"}}'
13
13
  SERVICE: '{{values "service"}}'
14
14
 
15
+ DD_ENV: '{{values "env"}}'
16
+ DD_SERVICE: '{{values "service"}}'
17
+
18
+ OPENTELEMETRY_ENABLED: '{{values "openTelemetry.enabled"}}'
19
+
15
20
  {{if .Values.cloudwatchDashboardDisabled}}
16
21
  cloudwatchDashboardDisabled: {{values "cloudwatchDashboardDisabled"}}
17
22
  {{end}}
@@ -20,6 +25,17 @@ cloudwatchDashboardDisabled: {{values "cloudwatchDashboardDisabled"}}
20
25
  datadogSecretId: '{{values "datadogSecretId"}}'
21
26
  {{end}}
22
27
 
28
+ # Uncomment if the `tin` tier is acceptable for your SEEK Auth Sidecar logs
29
+ # The eeeoh log forwarder bypasses Amazon CloudWatch for cost savings
30
+ # logSink:
31
+ # forwarder: eeeoh
32
+ # splunkIndex: '{{values "logSink.splunkIndex"}}'
33
+
34
+ openTelemetry:
35
+ datadogEnvironmentName: '{{values "env"}}'
36
+ enabled: {{values "openTelemetry.enabled"}}
37
+ useGantryServiceName: true
38
+
23
39
  {{if .Values.pagerDutyEndpoint}}
24
40
  pagerDutyEndpoint: '{{values "pagerDutyEndpoint"}}'
25
41
  {{end}}
@@ -19,22 +19,28 @@
19
19
  "test:watch": "skuba test --watch"
20
20
  },
21
21
  "dependencies": {
22
- "@seek/logger": "^10.0.0",
22
+ "@opentelemetry/api": "^1.9.0",
23
+ "@opentelemetry/core": "^2.0.0",
24
+ "@opentelemetry/exporter-trace-otlp-grpc": "^0.204.0",
25
+ "@opentelemetry/instrumentation-aws-sdk": "^0.58.0",
26
+ "@opentelemetry/instrumentation-http": "^0.204.0",
27
+ "@opentelemetry/propagator-b3": "^2.0.0",
28
+ "@opentelemetry/sdk-node": "^0.204.0",
29
+ "@seek/logger": "^11.1.0",
23
30
  "express": "^5.0.0",
24
- "hot-shots": "^10.0.0",
25
- "seek-datadog-custom-metrics": "^4.6.3",
31
+ "hot-shots": "^11.0.0",
32
+ "seek-datadog-custom-metrics": "^6.0.0",
26
33
  "skuba-dive": "^2.0.0"
27
34
  },
28
35
  "devDependencies": {
29
36
  "@types/express": "^5.0.0",
30
37
  "@types/node": "^22.13.10",
31
38
  "@types/supertest": "^6.0.0",
32
- "mime": "^4.0.1",
33
39
  "pino-pretty": "^13.0.0",
34
40
  "skuba": "*",
35
41
  "supertest": "^7.0.0"
36
42
  },
37
- "packageManager": "pnpm@10.12.4",
43
+ "packageManager": "pnpm@10.15.1",
38
44
  "engines": {
39
45
  "node": ">=22"
40
46
  }
@@ -3,8 +3,8 @@ import type { Handler } from 'express';
3
3
  /**
4
4
  * Signifies that the API is available to serve requests.
5
5
  *
6
- * The deployment environment calls this endpoint to see if the container is
7
- * unhealthy and needs to be recycled.
6
+ * The workload hosting environment calls this endpoint to see if the container
7
+ * is unhealthy and needs to be recycled.
8
8
  */
9
9
  export const healthCheckHandler: Handler = (_req, res) => {
10
10
  res.send('');
@@ -1,7 +1,7 @@
1
1
  import { Env } from 'skuba-dive';
2
2
 
3
3
  interface Config {
4
- environment: Environment;
4
+ deployment: Deployment;
5
5
 
6
6
  logLevel: string;
7
7
  name: string;
@@ -11,17 +11,17 @@ interface Config {
11
11
  port: number | null;
12
12
  }
13
13
 
14
- type Environment = (typeof environments)[number];
14
+ type Deployment = (typeof deployments)[number];
15
15
 
16
16
  const dev = '<%- devGantryEnvironmentName %>';
17
17
  const prod = '<%- prodGantryEnvironmentName %>';
18
18
 
19
- const environments = ['local', 'test', dev, prod] as const;
19
+ const deployments = ['local', 'test', dev, prod] as const;
20
20
 
21
- const environment = Env.oneOf(environments)('ENVIRONMENT');
21
+ const deployment = Env.oneOf(deployments)('DEPLOYMENT');
22
22
 
23
23
  /* istanbul ignore next: config verification makes more sense in a smoke test */
24
- const configs: Record<Environment, () => Omit<Config, 'environment'>> = {
24
+ const configs: Record<Deployment, () => Omit<Config, 'deployment'>> = {
25
25
  local: () => ({
26
26
  logLevel: 'debug',
27
27
  name: '<%- serviceName %>',
@@ -60,6 +60,6 @@ const configs: Record<Environment, () => Omit<Config, 'environment'>> = {
60
60
  };
61
61
 
62
62
  export const config: Config = {
63
- ...configs[environment](),
64
- environment,
63
+ ...configs[deployment](),
64
+ deployment,
65
65
  };
@@ -1,17 +1,21 @@
1
- import createLogger from '@seek/logger';
1
+ import { createLogger } from '@seek/logger';
2
2
 
3
3
  import { config } from '#src/config.js';
4
4
 
5
5
  export const logger = createLogger({
6
- base: {
7
- environment: config.environment,
8
- version: config.version,
6
+ eeeoh: {
7
+ /**
8
+ * TODO: choose an appropriate Datadog log tier.
9
+ *
10
+ * https://github.com/seek-oss/logger/blob/master/docs/eeeoh.md#datadog-log-tiers
11
+ */
12
+ datadog: 'tin',
13
+ team: '<%- teamName %>',
14
+ use: 'environment',
9
15
  },
10
16
 
11
17
  level: config.logLevel,
12
18
 
13
- name: config.name,
14
-
15
19
  transport:
16
- config.environment === 'local' ? { target: 'pino-pretty' } : undefined,
20
+ config.deployment === 'local' ? { target: 'pino-pretty' } : undefined,
17
21
  });
@@ -7,5 +7,5 @@ import { config } from '#src/config.js';
7
7
 
8
8
  /* istanbul ignore next: StatsD client is not our responsibility */
9
9
  export const metricsClient = createStatsDClient(StatsD, config, (err) =>
10
- logger.error({ err }, 'StatsD error'),
10
+ logger.error(err, 'StatsD error'),
11
11
  );
@@ -19,3 +19,9 @@ const listener = app.listen(config.port, () => {
19
19
  // https://docs.aws.amazon.com/elasticloadbalancing/latest/application/application-load-balancers.html#connection-idle-timeout
20
20
  // AWS recommends setting an application timeout larger than the load balancer
21
21
  listener.keepAliveTimeout = 31000;
22
+
23
+ // Report unhandled rejections instead of crashing the process
24
+ // Make sure to monitor these reports and alert as appropriate
25
+ process.on('unhandledRejection', (err) =>
26
+ logger.error(err, 'Unhandled promise rejection'),
27
+ );