@aws-cdk/integ-runner 2.191.2 → 2.192.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.
package/README.md CHANGED
@@ -73,6 +73,8 @@ If not, changes cannot be compared across systems and the [update workflow](#upd
73
73
  List of AWS Profiles to use when running tests in parallel
74
74
  - `--exclude` (default=`false`)
75
75
  If this is set to `true` then the list of tests provided will be excluded
76
+ - `--strict` (default=`false`)
77
+ Fail if any specified tests are not found. Cannot be used with `--exclude`
76
78
  - `--from-file`
77
79
  Read the list of tests from this file
78
80
  - `--disable-update-workflow` (default=`false`)
package/lib/cli.d.ts CHANGED
@@ -20,6 +20,7 @@ export declare function parseCliArgs(args?: string[]): {
20
20
  disableUpdateWorkflow: boolean;
21
21
  language: string[] | undefined;
22
22
  watch: boolean;
23
+ strict: boolean;
23
24
  unstable: string[];
24
25
  };
25
26
  export declare function main(args: string[]): Promise<void>;
package/lib/cli.js CHANGED
@@ -37,6 +37,7 @@ function parseCliArgs(args = []) {
37
37
  .options('profiles', { type: 'array', desc: 'list of AWS profiles to use. Tests will be run in parallel across each profile+regions', default: [] })
38
38
  .options('max-workers', { type: 'number', desc: 'The max number of workerpool workers to use when running integration tests in parallel', default: 16 })
39
39
  .options('exclude', { type: 'boolean', desc: 'Run all tests in the directory, except the specified TESTs', default: false })
40
+ .option('strict', { type: 'boolean', default: false, desc: 'Fail if any specified tests are not found' })
40
41
  .options('from-file', { type: 'string', desc: 'Read TEST names from a file (one TEST per line)' })
41
42
  .option('inspect-failures', { type: 'boolean', desc: 'Keep the integ test cloud assembly if a failure occurs for inspection', default: false })
42
43
  .option('disable-update-workflow', { type: 'boolean', default: false, desc: 'If this is "true" then the stack update workflow will be disabled' })
@@ -68,6 +69,9 @@ function parseCliArgs(args = []) {
68
69
  if (tests.length > 0 && fromFile) {
69
70
  throw new Error('A list of tests cannot be provided if "--from-file" is provided');
70
71
  }
72
+ if (argv.strict && argv.exclude) {
73
+ throw new Error('Cannot use --strict with --exclude');
74
+ }
71
75
  const requestedTests = fromFile
72
76
  ? (fs.readFileSync(fromFile, { encoding: 'utf8' })).split('\n').filter(x => x)
73
77
  : (tests.length > 0 ? tests : undefined); // 'undefined' means no request
@@ -93,6 +97,7 @@ function parseCliArgs(args = []) {
93
97
  disableUpdateWorkflow: argv['disable-update-workflow'],
94
98
  language: arrayFromYargs(argv.language),
95
99
  watch: argv.watch,
100
+ strict: argv.strict,
96
101
  unstable: arrayFromYargs(argv.unstable) ?? [],
97
102
  };
98
103
  }
@@ -297,4 +302,4 @@ function engineFromOptions(options) {
297
302
  }
298
303
  return { engine: 'toolkit-lib' };
299
304
  }
300
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cli.js","sourceRoot":"","sources":["cli.ts"],"names":[],"mappings":";;AAkBA,oCAmFC;AAED,oBAcC;AAsLD,kBAIC;AA/SD,gFAAgF;AAChF,yBAAyB;AACzB,6BAA6B;AAC7B,+BAA+B;AAC/B,yCAAyC;AACzC,mCAAmC;AAGnC,kEAA8D;AAE9D,uCAAkE;AAClE,qEAAoE;AAEpE,6CAA6C;AAC7C,+CAA+C;AAC/C,iEAAiE;AACjE,MAAM,KAAK,GAAG,OAAO,CAAC,OAAO,CAAC,CAAC;AAE/B,SAAgB,YAAY,CAAC,OAAiB,EAAE;IAC9C,MAAM,IAAI,GAAG,KAAK;SACf,KAAK,CAAC,+BAA+B,CAAC;SACtC,MAAM,CAAC,QAAQ,EAAE;QAChB,MAAM,EAAE,IAAI;QACZ,YAAY,EAAE,cAAc;QAC5B,OAAO,EAAE,mBAAmB;QAC5B,IAAI,EAAE,yFAAyF;KAChG,CAAC;SACD,MAAM,CAAC,OAAO,EAAE,EAAE,IAAI,EAAE,SAAS,EAAE,OAAO,EAAE,KAAK,EAAE,IAAI,EAAE,mCAAmC,EAAE,CAAC;SAC/F,MAAM,CAAC,MAAM,EAAE,EAAE,IAAI,EAAE,SAAS,EAAE,OAAO,EAAE,KAAK,EAAE,IAAI,EAAE,oCAAoC,EAAE,CAAC;SAC/F,MAAM,CAAC,OAAO,EAAE,EAAE,IAAI,EAAE,SAAS,EAAE,OAAO,EAAE,IAAI,EAAE,IAAI,EAAE,8EAA8E,EAAE,CAAC;SACzI,MAAM,CAAC,SAAS,EAAE,EAAE,IAAI,EAAE,SAAS,EAAE,OAAO,EAAE,KAAK,EAAE,KAAK,EAAE,GAAG,EAAE,KAAK,EAAE,IAAI,EAAE,IAAI,EAAE,wGAAwG,EAAE,CAAC;SAC/L,MAAM,CAAC,SAAS,EAAE,EAAE,IAAI,EAAE,SAAS,EAAE,OAAO,EAAE,KAAK,EAAE,IAAI,EAAE,+EAA+E,EAAE,CAAC;SAC7I,MAAM,CAAC,kBAAkB,EAAE,EAAE,IAAI,EAAE,SAAS,EAAE,OAAO,EAAE,KAAK,EAAE,IAAI,EAAE,gEAAgE,EAAE,CAAC;SACvI,MAAM,CAAC,OAAO,EAAE,EAAE,IAAI,EAAE,SAAS,EAAE,OAAO,EAAE,KAAK,EAAE,IAAI,EAAE,uDAAuD,EAAE,CAAC;SACnH,MAAM,CAAC,kBAAkB,EAAE,EAAE,IAAI,EAAE,OAAO,EAAE,IAAI,EAAE,yHAAyH,EAAE,OAAO,EAAE,EAAE,EAAE,CAAC;SAC3L,OAAO,CAAC,WAAW,EAAE,EAAE,IAAI,EAAE,QAAQ,EAAE,OAAO,EAAE,MAAM,EAAE,IAAI,EAAE,4GAA4G,EAAE,CAAC;SAC7K,OAAO,CAAC,UAAU,EAAE,EAAE,IAAI,EAAE,OAAO,EAAE,IAAI,EAAE,wFAAwF,EAAE,OAAO,EAAE,EAAE,EAAE,CAAC;SACnJ,OAAO,CAAC,aAAa,EAAE,EAAE,IAAI,EAAE,QAAQ,EAAE,IAAI,EAAE,wFAAwF,EAAE,OAAO,EAAE,EAAE,EAAE,CAAC;SACvJ,OAAO,CAAC,SAAS,EAAE,EAAE,IAAI,EAAE,SAAS,EAAE,IAAI,EAAE,4DAA4D,EAAE,OAAO,EAAE,KAAK,EAAE,CAAC;SAC3H,OAAO,CAAC,WAAW,EAAE,EAAE,IAAI,EAAE,QAAQ,EAAE,IAAI,EAAE,iDAAiD,EAAE,CAAC;SACjG,MAAM,CAAC,kBAAkB,EAAE,EAAE,IAAI,EAAE,SAAS,EAAE,IAAI,EAAE,uEAAuE,EAAE,OAAO,EAAE,KAAK,EAAE,CAAC;SAC9I,MAAM,CAAC,yBAAyB,EAAE,EAAE,IAAI,EAAE,SAAS,EAAE,OAAO,EAAE,KAAK,EAAE,IAAI,EAAE,mEAAmE,EAAE,CAAC;SACjJ,MAAM,CAAC,UAAU,EAAE;QAClB,KAAK,EAAE,GAAG;QACV,OAAO,EAAE,CAAC,YAAY,EAAE,YAAY,EAAE,QAAQ,EAAE,IAAI,CAAC;QACrD,OAAO,EAAE,CAAC,YAAY,EAAE,YAAY,EAAE,QAAQ,EAAE,IAAI,CAAC;QACrD,IAAI,EAAE,OAAO;QACb,KAAK,EAAE,CAAC;QACR,IAAI,EAAE,uEAAuE;KAC9E,CAAC;SACD,MAAM,CAAC,KAAK,EAAE,EAAE,IAAI,EAAE,QAAQ,EAAE,OAAO,EAAE,SAAS,EAAE,IAAI,EAAE,0MAA0M,EAAE,CAAC;SACvQ,MAAM,CAAC,YAAY,EAAE,EAAE,IAAI,EAAE,OAAO,EAAE,IAAI,EAAE,wJAAwJ,EAAE,OAAO,EAAE,EAAE,EAAE,CAAC;SACpN,MAAM,CAAC,UAAU,EAAE,EAAE,IAAI,EAAE,OAAO,EAAE,IAAI,EAAE,sNAAsN,EAAE,KAAK,EAAE,CAAC,EAAE,OAAO,EAAE,CAAC,oBAAoB,EAAE,uBAAuB,CAAC,EAAE,OAAO,EAAE,EAAE,EAAE,CAAC;SACpV,MAAM,EAAE;SACR,KAAK,CAAC,IAAI,CAAC,CAAC;IAEf,MAAM,KAAK,GAAa,IAAI,CAAC,CAAC,CAAC;IAC/B,MAAM,eAAe,GAAG,cAAc,CAAC,IAAI,CAAC,kBAAkB,CAAC,CAAC,CAAC;IACjE,MAAM,WAAW,GAAa,eAAe,IAAI,CAAC,WAAW,EAAE,WAAW,EAAE,WAAW,CAAC,CAAC;IACzF,MAAM,QAAQ,GAAG,cAAc,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;IAC/C,MAAM,QAAQ,GAAuB,IAAI,CAAC,WAAW,CAAC,CAAC;IACvD,MAAM,UAAU,GAAW,IAAI,CAAC,aAAa,CAAC,CAAC;IAC/C,MAAM,SAAS,GAAW,IAAI,CAAC,OAAO,CAAC;IACvC,MAAM,OAAO,GAAY,SAAS,IAAI,CAAC,CAAC;IAExC,MAAM,QAAQ,GAAG,WAAW,CAAC,MAAM,GAAG,CAAC,QAAQ,IAAI,CAAC,CAAC,CAAC,CAAC,CAAC,MAAM,CAAC;IAC/D,IAAI,UAAU,GAAG,QAAQ,EAAE,CAAC;QAC1B,MAAM,CAAC,OAAO,CAAC,6HAA6H,EAAE,QAAQ,EAAE,UAAU,CAAC,CAAC;IACtK,CAAC;IAED,IAAI,KAAK,CAAC,MAAM,GAAG,CAAC,IAAI,QAAQ,EAAE,CAAC;QACjC,MAAM,IAAI,KAAK,CAAC,iEAAiE,CAAC,CAAC;IACrF,CAAC;IACD,MAAM,cAAc,GAAG,QAAQ;QAC7B,CAAC,CAAC,CAAC,EAAE,CAAC,YAAY,CAAC,QAAQ,EAAE,EAAE,QAAQ,EAAE,MAAM,EAAE,CAAC,CAAC,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC;QAC9E,CAAC,CAAC,CAAC,KAAK,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC,CAAC,SAAS,CAAC,CAAC,CAAC,+BAA+B;IAE3E,OAAO;QACL,KAAK,EAAE,cAAc;QACrB,GAAG,EAAE,IAAI,CAAC,GAA2B;QACrC,SAAS,EAAE,cAAc,CAAC,IAAI,CAAC,YAAY,CAAC,CAAC;QAC7C,WAAW;QACX,eAAe,EAAE,eAAe;QAChC,QAAQ;QACR,iBAAiB,EAAE,CAAC,IAAI,CAAC,kBAAkB,CAAC,IAAI,KAAK,CAAY;QACjE,QAAQ;QACR,OAAO,EAAE,IAAI,CAAC,OAAkB;QAChC,UAAU;QACV,IAAI,EAAE,IAAI,CAAC,IAAe;QAC1B,SAAS,EAAE,IAAI,CAAC,SAAmB;QACnC,eAAe,EAAE,IAAI,CAAC,kBAAkB,CAAY;QACpD,SAAS;QACT,OAAO;QACP,KAAK,EAAE,IAAI,CAAC,KAAgB;QAC5B,KAAK,EAAE,IAAI,CAAC,KAAgB;QAC5B,MAAM,EAAE,IAAI,CAAC,SAAS,CAAY;QAClC,qBAAqB,EAAE,IAAI,CAAC,yBAAyB,CAAY;QACjE,QAAQ,EAAE,cAAc,CAAC,IAAI,CAAC,QAAQ,CAAC;QACvC,KAAK,EAAE,IAAI,CAAC,KAAgB;QAC5B,QAAQ,EAAE,cAAc,CAAC,IAAI,CAAC,QAAQ,CAAC,IAAI,EAAE;KAC9C,CAAC;AACJ,CAAC;AAEM,KAAK,UAAU,IAAI,CAAC,IAAc;IACvC,IAAI,cAAc,CAAC;IACnB,IAAI,CAAC;QACH,MAAM,OAAO,GAAG,YAAY,CAAC,IAAI,CAAC,CAAC;QACnC,MAAM,MAAM,GAAG,iBAAiB,CAAC,OAAO,CAAC,CAAC;QAC1C,cAAc,GAAG,MAAM,CAAC,MAAM,CAAC;QAC/B,MAAM,GAAG,CAAC,OAAO,EAAE,MAAM,CAAC,CAAC;IAC7B,CAAC;IAAC,OAAO,GAAQ,EAAE,CAAC;QAClB,MAAM,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC;QAClB,IAAI,cAAc,KAAK,aAAa,EAAE,CAAC;YACrC,MAAM,CAAC,OAAO,CAAC,whBAAwhB,CAAC,CAAC;QAC3iB,CAAC;QACD,MAAM,GAAG,CAAC;IACZ,CAAC;AACH,CAAC;AAED,KAAK,UAAU,GAAG,CAAC,OAAwC,EAAE,EAAE,MAAM,EAAiB;IACpF,MAAM,aAAa,GAAG,MAAM,IAAI,oCAAgB,CAAC,IAAI,CAAC,OAAO,CAAC,OAAO,CAAC,SAAS,CAAC,CAAC,CAAC,cAAc,CAAC,OAAO,CAAC,CAAC;IAE1G,wCAAwC;IACxC,IAAI,OAAO,CAAC,IAAI,EAAE,CAAC;QACjB,OAAO,CAAC,MAAM,CAAC,KAAK,CAAC,aAAa,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,yBAAyB,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,GAAG,IAAI,CAAC,CAAC;QAC5F,OAAO;IACT,CAAC;IAED,MAAM,IAAI,GAAG,UAAU,CAAC,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,SAAS,EAAE,IAAI,EAAE,KAAK,EAAE,SAAS,EAAE,SAAS,EAAE,UAAU,CAAC,EAAE;QAChG,UAAU,EAAE,OAAO,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,OAAO,CAAC,UAAU;KACnD,CAAC,CAAC;IAEH,MAAM,UAAU,GAA4B,EAAE,CAAC;IAC/C,IAAI,kBAAkB,GAAY,KAAK,CAAC;IACxC,IAAI,eAAe,GAA4B,EAAE,CAAC;IAClD,IAAI,cAAc,GAAG,KAAK,CAAC;IAC3B,iBAAiB,CAAC;QAChB,GAAG,OAAO;QACV,WAAW,EAAE,OAAO,CAAC,eAAe;QACpC,KAAK,EAAE,aAAa;KACrB,CAAC,CAAC;IAEH,IAAI,CAAC;QACH,IAAI,CAAC,OAAO,CAAC,KAAK,EAAE,CAAC;YACnB,6DAA6D;YAC7D,iEAAiE;YACjE,wBAAwB;YACxB,eAAe,GAAG,MAAM,IAAA,0BAAgB,EAAC,IAAI,EAAE,aAAa,EAAE;gBAC5D,MAAM,EAAE,OAAO,CAAC,eAAe;gBAC/B,OAAO,EAAE,OAAO,CAAC,OAAO;gBACxB,MAAM;aACP,CAAC,CAAC;YACH,KAAK,MAAM,OAAO,IAAI,eAAe,EAAE,CAAC;gBACtC,MAAM,CAAC,OAAO,CAAC,WAAW,OAAO,CAAC,QAAQ,EAAE,CAAC,CAAC;gBAC9C,IAAI,OAAO,CAAC,kBAAkB,IAAI,OAAO,CAAC,kBAAkB,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;oBACxE,uBAAuB,CAAC,OAAO,CAAC,kBAAkB,CAAC,CAAC;oBACpD,kBAAkB,GAAG,IAAI,CAAC;gBAC5B,CAAC;YACH,CAAC;YACD,IAAI,CAAC,OAAO,CAAC,KAAK,EAAE,CAAC;gBACnB,UAAU,CAAC,IAAI,CAAC,GAAG,eAAe,CAAC,CAAC;YACtC,CAAC;iBAAM,CAAC;gBACN,+DAA+D;gBAC/D,iDAAiD;gBACjD,UAAU,CAAC,IAAI,CAAC,GAAG,UAAU,CAAC,aAAa,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,IAAI,CAAC,EAAE,eAAe,CAAC,CAAC,CAAC;YAClF,CAAC;QACH,CAAC;aAAM,CAAC;YACN,UAAU,CAAC,IAAI,CAAC,GAAG,aAAa,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC;QACrD,CAAC;QAED,qEAAqE;QACrE,IAAI,OAAO,CAAC,iBAAiB,IAAI,OAAO,CAAC,KAAK,EAAE,CAAC;YAC/C,MAAM,EAAE,OAAO,EAAE,OAAO,EAAE,GAAG,MAAM,IAAA,6BAAmB,EAAC;gBACrD,IAAI;gBACJ,MAAM;gBACN,KAAK,EAAE,UAAU;gBACjB,OAAO,EAAE,OAAO,CAAC,WAAW;gBAC5B,QAAQ,EAAE,OAAO,CAAC,QAAQ;gBAC1B,KAAK,EAAE,OAAO,CAAC,KAAK;gBACpB,MAAM,EAAE,OAAO,CAAC,MAAM;gBACtB,SAAS,EAAE,OAAO,CAAC,SAAS;gBAC5B,cAAc,EAAE,CAAC,OAAO,CAAC,qBAAqB;gBAC9C,KAAK,EAAE,OAAO,CAAC,KAAK;aACrB,CAAC,CAAC;YACH,cAAc,GAAG,OAAO,CAAC;YAEzB,IAAI,OAAO,CAAC,KAAK,KAAK,KAAK,EAAE,CAAC;gBAC5B,MAAM,CAAC,OAAO,CAAC,oDAAoD,CAAC,CAAC;YACvE,CAAC;YAED,IAAI,OAAO,CAAC,OAAO,CAAC,OAAO,CAAC,EAAE,CAAC;gBAC7B,YAAY,CAAC,OAAO,CAAC,CAAC;YACxB,CAAC;YAED,IAAI,CAAC,OAAO,EAAE,CAAC;gBACb,MAAM,IAAI,KAAK,CAAC,gCAAgC,CAAC,CAAC;YACpD,CAAC;QACH,CAAC;aAAM,IAAI,OAAO,CAAC,KAAK,EAAE,CAAC;YACzB,MAAM,IAAA,yCAAoB,EAAC,IAAI,EAAE;gBAC/B,KAAK,EAAE,IAAI;gBACX,MAAM;gBACN,SAAS,EAAE,OAAO,CAAC,SAAS;gBAC5B,GAAG,UAAU,CAAC,CAAC,CAAC;gBAChB,OAAO,EAAE,OAAO,CAAC,QAAQ,CAAC,CAAC,CAAC,OAAO,CAAC,QAAQ,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,SAAS;gBAC3D,MAAM,EAAE,OAAO,CAAC,WAAW,CAAC,CAAC,CAAC;aAC/B,CAAC,CAAC;QACL,CAAC;IACH,CAAC;YAAS,CAAC;QACT,KAAK,IAAI,CAAC,SAAS,EAAE,CAAC;IACxB,CAAC;IAED,IAAI,kBAAkB,EAAE,CAAC;QACvB,MAAM,IAAI,KAAK,CAAC,gCAAgC,CAAC,CAAC;IACpD,CAAC;IACD,IAAI,eAAe,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;QAC/B,IAAI,OAAO,GAAG,EAAE,CAAC;QACjB,IAAI,CAAC,OAAO,CAAC,iBAAiB,EAAE,CAAC;YAC/B,OAAO,GAAG,6DAA6D,CAAC;QAC1E,CAAC;QACD,IAAI,CAAC,cAAc,EAAE,CAAC;YACpB,MAAM,IAAI,KAAK,CAAC,uBAAuB,OAAO,EAAE,CAAC,CAAC;QACpD,CAAC;IACH,CAAC;AACH,CAAC;AAED,SAAS,iBAAiB,CAAC,IAU1B;IACC,IAAI,IAAI,CAAC,KAAK,EAAE,CAAC;QACf,IACE,CAAC,IAAI,CAAC,WAAW,IAAI,IAAI,CAAC,WAAW,CAAC,MAAM,GAAG,CAAC,CAAC;eAC5C,CAAC,IAAI,CAAC,QAAQ,IAAI,IAAI,CAAC,QAAQ,CAAC,MAAM,GAAG,CAAC,CAAC;eAC3C,IAAI,CAAC,KAAK,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;YAC7B,MAAM,IAAI,KAAK,CAAC,8EAA8E;gBAC5F,qDAAqD,CAAC,CAAC;QAC3D,CAAC;QAED,IAAI,IAAI,CAAC,iBAAiB,IAAI,IAAI,CAAC,qBAAqB,IAAI,IAAI,CAAC,KAAK,IAAI,IAAI,CAAC,MAAM,EAAE,CAAC;YACtF,MAAM,CAAC,OAAO,CAAC,2HAA2H,CAAC,CAAC;QAC9I,CAAC;IACH,CAAC;AACH,CAAC;AAED,SAAS,uBAAuB,CAAC,OAA4B;IAC3D,IAAI,OAAO,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;QACvB,MAAM,CAAC,OAAO,CAAC,+BAA+B,EAAE,KAAK,CAAC,IAAI,CAAC,qBAAqB,CAAC,CAAC,CAAC;QACnF,OAAO,CAAC,OAAO,CAAC,MAAM,CAAC,EAAE;YACvB,MAAM,CAAC,OAAO,CAAC,2CAA2C,EAAE,MAAM,CAAC,SAAS,EAAE,MAAM,CAAC,SAAS,EAAE,MAAM,CAAC,MAAM,CAAC,CAAC;QACjH,CAAC,CAAC,CAAC;QACH,MAAM,CAAC,OAAO,CAAC,oFAAoF,CAAC,CAAC;IACvG,CAAC;AACH,CAAC;AAED,SAAS,YAAY,CAAC,OAA6B;IACjD,MAAM,CAAC,SAAS,CAAC,qCAAqC,CAAC,CAAC;IACxD,MAAM,aAAa,GAAG,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,QAAQ,GAAG,CAAC,CAAC,QAAQ,CAAC,CAAC;IACtE,aAAa,CAAC,OAAO,CAAC,MAAM,CAAC,EAAE;QAC7B,MAAM,CAAC,KAAK,CAAC,uCAAuC,EAAE,MAAM,CAAC,OAAO,EAAE,MAAM,CAAC,MAAM,EAAE,MAAM,CAAC,QAAQ,CAAC,CAAC;QACtG,MAAM,WAAW,GAAG,MAAM,CAAC,OAAO,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;QAC7E,WAAW,CAAC,OAAO,CAAC,IAAI,CAAC,EAAE,CAAC,MAAM,CAAC,KAAK,CAAC,UAAU,EAAE,IAAI,CAAC,CAAC,CAAC,EAAE,IAAI,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;IAC1E,CAAC,CAAC,CAAC;AACL,CAAC;AAED;;;;;;;;GAQG;AACH,SAAS,cAAc,CAAC,EAAY;IAClC,IAAI,EAAE,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;QACpB,OAAO,SAAS,CAAC;IACnB,CAAC;IACD,OAAO,EAAE,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,KAAK,EAAE,CAAC,CAAC;AAClC,CAAC;AAED;;;;GAIG;AACH,SAAS,UAAU,CAAC,YAA6B,EAAE,mBAA4C;IAC7F,MAAM,eAAe,GAAG,IAAI,GAAG,CAAC,mBAAmB,CAAC,GAAG,CAAC,IAAI,CAAC,EAAE,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC,CAAC;IAChF,MAAM,KAAK,GAA4B,mBAAmB,CAAC;IAC3D,KAAK,CAAC,IAAI,CAAC,GAAG,YAAY,CAAC,MAAM,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC,eAAe,CAAC,GAAG,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC,CAAC,CAAC;IAChF,OAAO,KAAK,CAAC;AACf,CAAC;AAED,SAAgB,GAAG,CAAC,OAAiB,OAAO,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC,CAAC;IACxD,IAAI,CAAC,IAAI,CAAC,CAAC,IAAI,EAAE,CAAC,KAAK,CAAC,GAAG,EAAE;QAC3B,OAAO,CAAC,QAAQ,GAAG,CAAC,CAAC;IACvB,CAAC,CAAC,CAAC;AACL,CAAC;AAED;;;;GAIG;AACH,SAAS,cAAc,CAAC,QAAiB;IACvC,IAAI,CAAC,QAAQ,EAAE,CAAC;QACd,OAAO,EAAE,CAAC;IACZ,CAAC;IAED,IAAI,CAAC;QACH,OAAO,IAAI,CAAC,KAAK,CAAC,EAAE,CAAC,YAAY,CAAC,QAAQ,EAAE,EAAE,QAAQ,EAAE,OAAO,EAAE,CAAC,CAAC,CAAC;IACtE,CAAC;IAAC,MAAM,CAAC;QACP,OAAO,EAAE,CAAC;IACZ,CAAC;AACH,CAAC;AAED,SAAS,iBAAiB,CAAC,OAAgC;IACzD,IAAI,OAAO,CAAC,QAAQ,EAAE,QAAQ,CAAC,uBAAuB,CAAC,EAAE,CAAC;QACxD,MAAM,CAAC,OAAO,CAAC,qXAAqX,CAAC,CAAC;QACtY,OAAO,EAAE,MAAM,EAAE,aAAa,EAAE,CAAC;IACnC,CAAC;IACD,OAAO,EAAE,MAAM,EAAE,aAAa,EAAE,CAAC;AACnC,CAAC","sourcesContent":["// Exercise all integ stacks and if they deploy, update the expected synth files\nimport * as fs from 'fs';\nimport * as path from 'path';\nimport * as chalk from 'chalk';\nimport * as workerpool from 'workerpool';\nimport * as logger from './logger';\nimport type { EngineOptions } from './runner/engine';\nimport type { IntegTest, IntegTestInfo } from './runner/integration-tests';\nimport { IntegrationTests } from './runner/integration-tests';\nimport type { IntegRunnerMetrics, IntegTestWorkerConfig, DestructiveChange } from './workers';\nimport { runSnapshotTests, runIntegrationTests } from './workers';\nimport { watchIntegrationTest } from './workers/integ-watch-worker';\n\n// https://github.com/yargs/yargs/issues/1929\n// https://github.com/evanw/esbuild/issues/1492\n// eslint-disable-next-line @typescript-eslint/no-require-imports\nconst yargs = require('yargs');\n\nexport function parseCliArgs(args: string[] = []) {\n  const argv = yargs\n    .usage('Usage: integ-runner [TEST...]')\n    .option('config', {\n      config: true,\n      configParser: configFromFile,\n      default: 'integ.config.json',\n      desc: 'Load options from a JSON config file. Options provided as CLI arguments take precedent.',\n    })\n    .option('watch', { type: 'boolean', default: false, desc: 'Perform integ tests in watch mode' })\n    .option('list', { type: 'boolean', default: false, desc: 'List tests instead of running them' })\n    .option('clean', { type: 'boolean', default: true, desc: 'Clean up and delete stack after test is completed (use --no-clean to negate)' })\n    .option('verbose', { type: 'boolean', default: false, alias: 'v', count: true, desc: 'Verbose logs and metrics on integration tests durations (specify multiple times to increase verbosity)' })\n    .option('dry-run', { type: 'boolean', default: false, desc: 'do not actually deploy the stack. just update the snapshot (not recommended!)' })\n    .option('update-on-failed', { type: 'boolean', default: false, desc: 'rerun integration tests and update snapshots for failed tests.' })\n    .option('force', { type: 'boolean', default: false, desc: 'Rerun all integration tests even if tests are passing' })\n    .option('parallel-regions', { type: 'array', desc: 'Tests are run in parallel across these regions. To prevent tests from running in parallel, provide only a single region', default: [] })\n    .options('directory', { type: 'string', default: 'test', desc: 'starting directory to discover integration tests. Tests will be discovered recursively from this directory' })\n    .options('profiles', { type: 'array', desc: 'list of AWS profiles to use. Tests will be run in parallel across each profile+regions', default: [] })\n    .options('max-workers', { type: 'number', desc: 'The max number of workerpool workers to use when running integration tests in parallel', default: 16 })\n    .options('exclude', { type: 'boolean', desc: 'Run all tests in the directory, except the specified TESTs', default: false })\n    .options('from-file', { type: 'string', desc: 'Read TEST names from a file (one TEST per line)' })\n    .option('inspect-failures', { type: 'boolean', desc: 'Keep the integ test cloud assembly if a failure occurs for inspection', default: false })\n    .option('disable-update-workflow', { type: 'boolean', default: false, desc: 'If this is \"true\" then the stack update workflow will be disabled' })\n    .option('language', {\n      alias: 'l',\n      default: ['javascript', 'typescript', 'python', 'go'],\n      choices: ['javascript', 'typescript', 'python', 'go'],\n      type: 'array',\n      nargs: 1,\n      desc: 'Use these presets to run integration tests for the selected languages',\n    })\n    .option('app', { type: 'string', default: undefined, desc: 'The custom CLI command that will be used to run the test files. You can include {filePath} to specify where in the command the test file path should be inserted. Example: --app=\"python3.8 {filePath}\".' })\n    .option('test-regex', { type: 'array', desc: 'Detect integration test files matching this JavaScript regex pattern. If used multiple times, all files matching any one of the patterns are detected.', default: [] })\n    .option('unstable', { type: 'array', desc: 'Opt-in to using unstable features. By using these flags you acknowledges that scope and APIs of unstable features may change without notice. Specify multiple times for each unstable feature you want to opt-in to.', nargs: 1, choices: ['toolkit-lib-engine', 'deprecated-cli-engine'], default: [] })\n    .strict()\n    .parse(args);\n\n  const tests: string[] = argv._;\n  const parallelRegions = arrayFromYargs(argv['parallel-regions']);\n  const testRegions: string[] = parallelRegions ?? ['us-east-1', 'us-east-2', 'us-west-2'];\n  const profiles = arrayFromYargs(argv.profiles);\n  const fromFile: string | undefined = argv['from-file'];\n  const maxWorkers: number = argv['max-workers'];\n  const verbosity: number = argv.verbose;\n  const verbose: boolean = verbosity >= 1;\n\n  const numTests = testRegions.length * (profiles ?? [1]).length;\n  if (maxWorkers < numTests) {\n    logger.warning('You are attempting to run %s tests in parallel, but only have %s workers. Not all of your profiles+regions will be utilized', numTests, maxWorkers);\n  }\n\n  if (tests.length > 0 && fromFile) {\n    throw new Error('A list of tests cannot be provided if \"--from-file\" is provided');\n  }\n  const requestedTests = fromFile\n    ? (fs.readFileSync(fromFile, { encoding: 'utf8' })).split('\\n').filter(x => x)\n    : (tests.length > 0 ? tests : undefined); // 'undefined' means no request\n\n  return {\n    tests: requestedTests,\n    app: argv.app as (string | undefined),\n    testRegex: arrayFromYargs(argv['test-regex']),\n    testRegions,\n    originalRegions: parallelRegions,\n    profiles,\n    runUpdateOnFailed: (argv['update-on-failed'] ?? false) as boolean,\n    fromFile,\n    exclude: argv.exclude as boolean,\n    maxWorkers,\n    list: argv.list as boolean,\n    directory: argv.directory as string,\n    inspectFailures: argv['inspect-failures'] as boolean,\n    verbosity,\n    verbose,\n    clean: argv.clean as boolean,\n    force: argv.force as boolean,\n    dryRun: argv['dry-run'] as boolean,\n    disableUpdateWorkflow: argv['disable-update-workflow'] as boolean,\n    language: arrayFromYargs(argv.language),\n    watch: argv.watch as boolean,\n    unstable: arrayFromYargs(argv.unstable) ?? [],\n  };\n}\n\nexport async function main(args: string[]) {\n  let engineForError;\n  try {\n    const options = parseCliArgs(args);\n    const engine = engineFromOptions(options);\n    engineForError = engine.engine;\n    await run(options, engine);\n  } catch (err: any) {\n    logger.error(err);\n    if (engineForError === 'toolkit-lib') {\n      logger.warning('\\n[Notice] You are using the new default engine to run integration tests. If you think the above failure has been caused by the new engine, you may choose to temporarily revert to the old engine by adding the `--unstable=deprecated-cli-engine` option. Please note that this engine is deprecated and scheduled to be removed in January 2026.\\n\\nIf reverting to the old engine resolves an issue for you, please let us know so we can address this in the new engine. Report issues here: https://github.com/aws/aws-cdk-cli/issues/new/choose');\n    }\n    throw err;\n  }\n}\n\nasync function run(options: ReturnType<typeof parseCliArgs>, { engine }: EngineOptions) {\n  const testsFromArgs = await new IntegrationTests(path.resolve(options.directory)).fromCliOptions(options);\n\n  // List only prints the discovered tests\n  if (options.list) {\n    process.stdout.write(testsFromArgs.map(t => t.discoveryRelativeFileName).join('\\n') + '\\n');\n    return;\n  }\n\n  const pool = workerpool.pool(path.join(__dirname, '..', 'lib', 'workers', 'extract', 'index.js'), {\n    maxWorkers: options.watch ? 1 : options.maxWorkers,\n  });\n\n  const testsToRun: IntegTestWorkerConfig[] = [];\n  let destructiveChanges: boolean = false;\n  let failedSnapshots: IntegTestWorkerConfig[] = [];\n  let testsSucceeded = false;\n  validateWatchArgs({\n    ...options,\n    testRegions: options.originalRegions,\n    tests: testsFromArgs,\n  });\n\n  try {\n    if (!options.watch) {\n      // always run snapshot tests, but if '--force' is passed then\n      // run integration tests on all failed tests, not just those that\n      // failed snapshot tests\n      failedSnapshots = await runSnapshotTests(pool, testsFromArgs, {\n        retain: options.inspectFailures,\n        verbose: options.verbose,\n        engine,\n      });\n      for (const failure of failedSnapshots) {\n        logger.warning(`Failed: ${failure.fileName}`);\n        if (failure.destructiveChanges && failure.destructiveChanges.length > 0) {\n          printDestructiveChanges(failure.destructiveChanges);\n          destructiveChanges = true;\n        }\n      }\n      if (!options.force) {\n        testsToRun.push(...failedSnapshots);\n      } else {\n        // if any of the test failed snapshot tests, keep those results\n        // and merge with the rest of the tests from args\n        testsToRun.push(...mergeTests(testsFromArgs.map(t => t.info), failedSnapshots));\n      }\n    } else {\n      testsToRun.push(...testsFromArgs.map(t => t.info));\n    }\n\n    // run integration tests if `--update-on-failed` OR `--force` is used\n    if (options.runUpdateOnFailed || options.force) {\n      const { success, metrics } = await runIntegrationTests({\n        pool,\n        engine,\n        tests: testsToRun,\n        regions: options.testRegions,\n        profiles: options.profiles,\n        clean: options.clean,\n        dryRun: options.dryRun,\n        verbosity: options.verbosity,\n        updateWorkflow: !options.disableUpdateWorkflow,\n        watch: options.watch,\n      });\n      testsSucceeded = success;\n\n      if (options.clean === false) {\n        logger.warning('Not cleaning up stacks since \"--no-clean\" was used');\n      }\n\n      if (Boolean(options.verbose)) {\n        printMetrics(metrics);\n      }\n\n      if (!success) {\n        throw new Error('Some integration tests failed!');\n      }\n    } else if (options.watch) {\n      await watchIntegrationTest(pool, {\n        watch: true,\n        engine,\n        verbosity: options.verbosity,\n        ...testsToRun[0],\n        profile: options.profiles ? options.profiles[0] : undefined,\n        region: options.testRegions[0],\n      });\n    }\n  } finally {\n    void pool.terminate();\n  }\n\n  if (destructiveChanges) {\n    throw new Error('Some changes were destructive!');\n  }\n  if (failedSnapshots.length > 0) {\n    let message = '';\n    if (!options.runUpdateOnFailed) {\n      message = 'To re-run failed tests run: integ-runner --update-on-failed';\n    }\n    if (!testsSucceeded) {\n      throw new Error(`Some tests failed!\\n${message}`);\n    }\n  }\n}\n\nfunction validateWatchArgs(args: {\n  tests: IntegTest[];\n  testRegions?: string[];\n  profiles?: string[];\n  maxWorkers: number;\n  force: boolean;\n  dryRun: boolean;\n  disableUpdateWorkflow: boolean;\n  runUpdateOnFailed: boolean;\n  watch: boolean;\n}) {\n  if (args.watch) {\n    if (\n      (args.testRegions && args.testRegions.length > 1)\n        || (args.profiles && args.profiles.length > 1)\n        || args.tests.length > 1) {\n      throw new Error('Running with watch only supports a single test. Only provide a single option'+\n        'to `--profiles` `--parallel-regions` `--max-workers');\n    }\n\n    if (args.runUpdateOnFailed || args.disableUpdateWorkflow || args.force || args.dryRun) {\n      logger.warning('args `--update-on-failed`, `--disable-update-workflow`, `--force`, `--dry-run` have no effect when running with `--watch`');\n    }\n  }\n}\n\nfunction printDestructiveChanges(changes: DestructiveChange[]): void {\n  if (changes.length > 0) {\n    logger.warning('!!! This test contains %s !!!', chalk.bold('destructive changes'));\n    changes.forEach(change => {\n      logger.warning('    Stack: %s - Resource: %s - Impact: %s', change.stackName, change.logicalId, change.impact);\n    });\n    logger.warning('!!! If these destructive changes are necessary, please indicate this on the PR !!!');\n  }\n}\n\nfunction printMetrics(metrics: IntegRunnerMetrics[]): void {\n  logger.highlight('   --- Integration test metrics ---');\n  const sortedMetrics = metrics.sort((a, b) => a.duration - b.duration);\n  sortedMetrics.forEach(metric => {\n    logger.print('Profile %s + Region %s total time: %s', metric.profile, metric.region, metric.duration);\n    const sortedTests = Object.entries(metric.tests).sort((a, b) => a[1] - b[1]);\n    sortedTests.forEach(test => logger.print('  %s: %s', test[0], test[1]));\n  });\n}\n\n/**\n * Translate a Yargs input array to something that makes more sense in a programming language\n * model (telling the difference between absence and an empty array)\n *\n * - An empty array is the default case, meaning the user didn't pass any arguments. We return\n *   undefined.\n * - If the user passed a single empty string, they did something like `--array=`, which we'll\n *   take to mean they passed an empty array.\n */\nfunction arrayFromYargs(xs: string[]): string[] | undefined {\n  if (xs.length === 0) {\n    return undefined;\n  }\n  return xs.filter(x => x !== '');\n}\n\n/**\n * Merge the tests we received from command line arguments with\n * tests that failed snapshot tests. The failed snapshot tests have additional\n * information that we want to keep so this should override any test from args\n */\nfunction mergeTests(testFromArgs: IntegTestInfo[], failedSnapshotTests: IntegTestWorkerConfig[]): IntegTestWorkerConfig[] {\n  const failedTestNames = new Set(failedSnapshotTests.map(test => test.fileName));\n  const final: IntegTestWorkerConfig[] = failedSnapshotTests;\n  final.push(...testFromArgs.filter(test => !failedTestNames.has(test.fileName)));\n  return final;\n}\n\nexport function cli(args: string[] = process.argv.slice(2)) {\n  main(args).then().catch(() => {\n    process.exitCode = 1;\n  });\n}\n\n/**\n * Read CLI options from a config file if provided.\n *\n * @returns parsed CLI config options\n */\nfunction configFromFile(fileName?: string): Record<string, any> {\n  if (!fileName) {\n    return {};\n  }\n\n  try {\n    return JSON.parse(fs.readFileSync(fileName, { encoding: 'utf-8' }));\n  } catch {\n    return {};\n  }\n}\n\nfunction engineFromOptions(options: { unstable?: string[] }): Required<EngineOptions> {\n  if (options.unstable?.includes('deprecated-cli-engine')) {\n    logger.warning('[Deprecation Notice] You have opted-in to use the deprecated CLI engine which is scheduled to be removed in January 2026. If you have encountered blockers while using the new default engine, please let us know by opening an issue: https://github.com/aws/aws-cdk-cli/issues/new/choose\\n\\nTo use the new default engine, remove the `--unstable=deprecated-cli-engine` option.');\n    return { engine: 'cli-wrapper' };\n  }\n  return { engine: 'toolkit-lib' };\n}\n"]}
305
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cli.js","sourceRoot":"","sources":["cli.ts"],"names":[],"mappings":";;AAkBA,oCA0FC;AAED,oBAcC;AAsLD,kBAIC;AAtTD,gFAAgF;AAChF,yBAAyB;AACzB,6BAA6B;AAC7B,+BAA+B;AAC/B,yCAAyC;AACzC,mCAAmC;AAGnC,kEAA8D;AAE9D,uCAAkE;AAClE,qEAAoE;AAEpE,6CAA6C;AAC7C,+CAA+C;AAC/C,iEAAiE;AACjE,MAAM,KAAK,GAAG,OAAO,CAAC,OAAO,CAAC,CAAC;AAE/B,SAAgB,YAAY,CAAC,OAAiB,EAAE;IAC9C,MAAM,IAAI,GAAG,KAAK;SACf,KAAK,CAAC,+BAA+B,CAAC;SACtC,MAAM,CAAC,QAAQ,EAAE;QAChB,MAAM,EAAE,IAAI;QACZ,YAAY,EAAE,cAAc;QAC5B,OAAO,EAAE,mBAAmB;QAC5B,IAAI,EAAE,yFAAyF;KAChG,CAAC;SACD,MAAM,CAAC,OAAO,EAAE,EAAE,IAAI,EAAE,SAAS,EAAE,OAAO,EAAE,KAAK,EAAE,IAAI,EAAE,mCAAmC,EAAE,CAAC;SAC/F,MAAM,CAAC,MAAM,EAAE,EAAE,IAAI,EAAE,SAAS,EAAE,OAAO,EAAE,KAAK,EAAE,IAAI,EAAE,oCAAoC,EAAE,CAAC;SAC/F,MAAM,CAAC,OAAO,EAAE,EAAE,IAAI,EAAE,SAAS,EAAE,OAAO,EAAE,IAAI,EAAE,IAAI,EAAE,8EAA8E,EAAE,CAAC;SACzI,MAAM,CAAC,SAAS,EAAE,EAAE,IAAI,EAAE,SAAS,EAAE,OAAO,EAAE,KAAK,EAAE,KAAK,EAAE,GAAG,EAAE,KAAK,EAAE,IAAI,EAAE,IAAI,EAAE,wGAAwG,EAAE,CAAC;SAC/L,MAAM,CAAC,SAAS,EAAE,EAAE,IAAI,EAAE,SAAS,EAAE,OAAO,EAAE,KAAK,EAAE,IAAI,EAAE,+EAA+E,EAAE,CAAC;SAC7I,MAAM,CAAC,kBAAkB,EAAE,EAAE,IAAI,EAAE,SAAS,EAAE,OAAO,EAAE,KAAK,EAAE,IAAI,EAAE,gEAAgE,EAAE,CAAC;SACvI,MAAM,CAAC,OAAO,EAAE,EAAE,IAAI,EAAE,SAAS,EAAE,OAAO,EAAE,KAAK,EAAE,IAAI,EAAE,uDAAuD,EAAE,CAAC;SACnH,MAAM,CAAC,kBAAkB,EAAE,EAAE,IAAI,EAAE,OAAO,EAAE,IAAI,EAAE,yHAAyH,EAAE,OAAO,EAAE,EAAE,EAAE,CAAC;SAC3L,OAAO,CAAC,WAAW,EAAE,EAAE,IAAI,EAAE,QAAQ,EAAE,OAAO,EAAE,MAAM,EAAE,IAAI,EAAE,4GAA4G,EAAE,CAAC;SAC7K,OAAO,CAAC,UAAU,EAAE,EAAE,IAAI,EAAE,OAAO,EAAE,IAAI,EAAE,wFAAwF,EAAE,OAAO,EAAE,EAAE,EAAE,CAAC;SACnJ,OAAO,CAAC,aAAa,EAAE,EAAE,IAAI,EAAE,QAAQ,EAAE,IAAI,EAAE,wFAAwF,EAAE,OAAO,EAAE,EAAE,EAAE,CAAC;SACvJ,OAAO,CAAC,SAAS,EAAE,EAAE,IAAI,EAAE,SAAS,EAAE,IAAI,EAAE,4DAA4D,EAAE,OAAO,EAAE,KAAK,EAAE,CAAC;SAC3H,MAAM,CAAC,QAAQ,EAAE,EAAE,IAAI,EAAE,SAAS,EAAE,OAAO,EAAE,KAAK,EAAE,IAAI,EAAE,2CAA2C,EAAE,CAAC;SACxG,OAAO,CAAC,WAAW,EAAE,EAAE,IAAI,EAAE,QAAQ,EAAE,IAAI,EAAE,iDAAiD,EAAE,CAAC;SACjG,MAAM,CAAC,kBAAkB,EAAE,EAAE,IAAI,EAAE,SAAS,EAAE,IAAI,EAAE,uEAAuE,EAAE,OAAO,EAAE,KAAK,EAAE,CAAC;SAC9I,MAAM,CAAC,yBAAyB,EAAE,EAAE,IAAI,EAAE,SAAS,EAAE,OAAO,EAAE,KAAK,EAAE,IAAI,EAAE,mEAAmE,EAAE,CAAC;SACjJ,MAAM,CAAC,UAAU,EAAE;QAClB,KAAK,EAAE,GAAG;QACV,OAAO,EAAE,CAAC,YAAY,EAAE,YAAY,EAAE,QAAQ,EAAE,IAAI,CAAC;QACrD,OAAO,EAAE,CAAC,YAAY,EAAE,YAAY,EAAE,QAAQ,EAAE,IAAI,CAAC;QACrD,IAAI,EAAE,OAAO;QACb,KAAK,EAAE,CAAC;QACR,IAAI,EAAE,uEAAuE;KAC9E,CAAC;SACD,MAAM,CAAC,KAAK,EAAE,EAAE,IAAI,EAAE,QAAQ,EAAE,OAAO,EAAE,SAAS,EAAE,IAAI,EAAE,0MAA0M,EAAE,CAAC;SACvQ,MAAM,CAAC,YAAY,EAAE,EAAE,IAAI,EAAE,OAAO,EAAE,IAAI,EAAE,wJAAwJ,EAAE,OAAO,EAAE,EAAE,EAAE,CAAC;SACpN,MAAM,CAAC,UAAU,EAAE,EAAE,IAAI,EAAE,OAAO,EAAE,IAAI,EAAE,sNAAsN,EAAE,KAAK,EAAE,CAAC,EAAE,OAAO,EAAE,CAAC,oBAAoB,EAAE,uBAAuB,CAAC,EAAE,OAAO,EAAE,EAAE,EAAE,CAAC;SACpV,MAAM,EAAE;SACR,KAAK,CAAC,IAAI,CAAC,CAAC;IAEf,MAAM,KAAK,GAAa,IAAI,CAAC,CAAC,CAAC;IAC/B,MAAM,eAAe,GAAG,cAAc,CAAC,IAAI,CAAC,kBAAkB,CAAC,CAAC,CAAC;IACjE,MAAM,WAAW,GAAa,eAAe,IAAI,CAAC,WAAW,EAAE,WAAW,EAAE,WAAW,CAAC,CAAC;IACzF,MAAM,QAAQ,GAAG,cAAc,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;IAC/C,MAAM,QAAQ,GAAuB,IAAI,CAAC,WAAW,CAAC,CAAC;IACvD,MAAM,UAAU,GAAW,IAAI,CAAC,aAAa,CAAC,CAAC;IAC/C,MAAM,SAAS,GAAW,IAAI,CAAC,OAAO,CAAC;IACvC,MAAM,OAAO,GAAY,SAAS,IAAI,CAAC,CAAC;IAExC,MAAM,QAAQ,GAAG,WAAW,CAAC,MAAM,GAAG,CAAC,QAAQ,IAAI,CAAC,CAAC,CAAC,CAAC,CAAC,MAAM,CAAC;IAC/D,IAAI,UAAU,GAAG,QAAQ,EAAE,CAAC;QAC1B,MAAM,CAAC,OAAO,CAAC,6HAA6H,EAAE,QAAQ,EAAE,UAAU,CAAC,CAAC;IACtK,CAAC;IAED,IAAI,KAAK,CAAC,MAAM,GAAG,CAAC,IAAI,QAAQ,EAAE,CAAC;QACjC,MAAM,IAAI,KAAK,CAAC,iEAAiE,CAAC,CAAC;IACrF,CAAC;IAED,IAAI,IAAI,CAAC,MAAM,IAAI,IAAI,CAAC,OAAO,EAAE,CAAC;QAChC,MAAM,IAAI,KAAK,CAAC,oCAAoC,CAAC,CAAC;IACxD,CAAC;IAED,MAAM,cAAc,GAAG,QAAQ;QAC7B,CAAC,CAAC,CAAC,EAAE,CAAC,YAAY,CAAC,QAAQ,EAAE,EAAE,QAAQ,EAAE,MAAM,EAAE,CAAC,CAAC,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC;QAC9E,CAAC,CAAC,CAAC,KAAK,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC,CAAC,SAAS,CAAC,CAAC,CAAC,+BAA+B;IAE3E,OAAO;QACL,KAAK,EAAE,cAAc;QACrB,GAAG,EAAE,IAAI,CAAC,GAA2B;QACrC,SAAS,EAAE,cAAc,CAAC,IAAI,CAAC,YAAY,CAAC,CAAC;QAC7C,WAAW;QACX,eAAe,EAAE,eAAe;QAChC,QAAQ;QACR,iBAAiB,EAAE,CAAC,IAAI,CAAC,kBAAkB,CAAC,IAAI,KAAK,CAAY;QACjE,QAAQ;QACR,OAAO,EAAE,IAAI,CAAC,OAAkB;QAChC,UAAU;QACV,IAAI,EAAE,IAAI,CAAC,IAAe;QAC1B,SAAS,EAAE,IAAI,CAAC,SAAmB;QACnC,eAAe,EAAE,IAAI,CAAC,kBAAkB,CAAY;QACpD,SAAS;QACT,OAAO;QACP,KAAK,EAAE,IAAI,CAAC,KAAgB;QAC5B,KAAK,EAAE,IAAI,CAAC,KAAgB;QAC5B,MAAM,EAAE,IAAI,CAAC,SAAS,CAAY;QAClC,qBAAqB,EAAE,IAAI,CAAC,yBAAyB,CAAY;QACjE,QAAQ,EAAE,cAAc,CAAC,IAAI,CAAC,QAAQ,CAAC;QACvC,KAAK,EAAE,IAAI,CAAC,KAAgB;QAC5B,MAAM,EAAE,IAAI,CAAC,MAAiB;QAC9B,QAAQ,EAAE,cAAc,CAAC,IAAI,CAAC,QAAQ,CAAC,IAAI,EAAE;KAC9C,CAAC;AACJ,CAAC;AAEM,KAAK,UAAU,IAAI,CAAC,IAAc;IACvC,IAAI,cAAc,CAAC;IACnB,IAAI,CAAC;QACH,MAAM,OAAO,GAAG,YAAY,CAAC,IAAI,CAAC,CAAC;QACnC,MAAM,MAAM,GAAG,iBAAiB,CAAC,OAAO,CAAC,CAAC;QAC1C,cAAc,GAAG,MAAM,CAAC,MAAM,CAAC;QAC/B,MAAM,GAAG,CAAC,OAAO,EAAE,MAAM,CAAC,CAAC;IAC7B,CAAC;IAAC,OAAO,GAAQ,EAAE,CAAC;QAClB,MAAM,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC;QAClB,IAAI,cAAc,KAAK,aAAa,EAAE,CAAC;YACrC,MAAM,CAAC,OAAO,CAAC,whBAAwhB,CAAC,CAAC;QAC3iB,CAAC;QACD,MAAM,GAAG,CAAC;IACZ,CAAC;AACH,CAAC;AAED,KAAK,UAAU,GAAG,CAAC,OAAwC,EAAE,EAAE,MAAM,EAAiB;IACpF,MAAM,aAAa,GAAG,MAAM,IAAI,oCAAgB,CAAC,IAAI,CAAC,OAAO,CAAC,OAAO,CAAC,SAAS,CAAC,CAAC,CAAC,cAAc,CAAC,OAAO,CAAC,CAAC;IAE1G,wCAAwC;IACxC,IAAI,OAAO,CAAC,IAAI,EAAE,CAAC;QACjB,OAAO,CAAC,MAAM,CAAC,KAAK,CAAC,aAAa,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,yBAAyB,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,GAAG,IAAI,CAAC,CAAC;QAC5F,OAAO;IACT,CAAC;IAED,MAAM,IAAI,GAAG,UAAU,CAAC,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,SAAS,EAAE,IAAI,EAAE,KAAK,EAAE,SAAS,EAAE,SAAS,EAAE,UAAU,CAAC,EAAE;QAChG,UAAU,EAAE,OAAO,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,OAAO,CAAC,UAAU;KACnD,CAAC,CAAC;IAEH,MAAM,UAAU,GAA4B,EAAE,CAAC;IAC/C,IAAI,kBAAkB,GAAY,KAAK,CAAC;IACxC,IAAI,eAAe,GAA4B,EAAE,CAAC;IAClD,IAAI,cAAc,GAAG,KAAK,CAAC;IAC3B,iBAAiB,CAAC;QAChB,GAAG,OAAO;QACV,WAAW,EAAE,OAAO,CAAC,eAAe;QACpC,KAAK,EAAE,aAAa;KACrB,CAAC,CAAC;IAEH,IAAI,CAAC;QACH,IAAI,CAAC,OAAO,CAAC,KAAK,EAAE,CAAC;YACnB,6DAA6D;YAC7D,iEAAiE;YACjE,wBAAwB;YACxB,eAAe,GAAG,MAAM,IAAA,0BAAgB,EAAC,IAAI,EAAE,aAAa,EAAE;gBAC5D,MAAM,EAAE,OAAO,CAAC,eAAe;gBAC/B,OAAO,EAAE,OAAO,CAAC,OAAO;gBACxB,MAAM;aACP,CAAC,CAAC;YACH,KAAK,MAAM,OAAO,IAAI,eAAe,EAAE,CAAC;gBACtC,MAAM,CAAC,OAAO,CAAC,WAAW,OAAO,CAAC,QAAQ,EAAE,CAAC,CAAC;gBAC9C,IAAI,OAAO,CAAC,kBAAkB,IAAI,OAAO,CAAC,kBAAkB,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;oBACxE,uBAAuB,CAAC,OAAO,CAAC,kBAAkB,CAAC,CAAC;oBACpD,kBAAkB,GAAG,IAAI,CAAC;gBAC5B,CAAC;YACH,CAAC;YACD,IAAI,CAAC,OAAO,CAAC,KAAK,EAAE,CAAC;gBACnB,UAAU,CAAC,IAAI,CAAC,GAAG,eAAe,CAAC,CAAC;YACtC,CAAC;iBAAM,CAAC;gBACN,+DAA+D;gBAC/D,iDAAiD;gBACjD,UAAU,CAAC,IAAI,CAAC,GAAG,UAAU,CAAC,aAAa,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,IAAI,CAAC,EAAE,eAAe,CAAC,CAAC,CAAC;YAClF,CAAC;QACH,CAAC;aAAM,CAAC;YACN,UAAU,CAAC,IAAI,CAAC,GAAG,aAAa,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC;QACrD,CAAC;QAED,qEAAqE;QACrE,IAAI,OAAO,CAAC,iBAAiB,IAAI,OAAO,CAAC,KAAK,EAAE,CAAC;YAC/C,MAAM,EAAE,OAAO,EAAE,OAAO,EAAE,GAAG,MAAM,IAAA,6BAAmB,EAAC;gBACrD,IAAI;gBACJ,MAAM;gBACN,KAAK,EAAE,UAAU;gBACjB,OAAO,EAAE,OAAO,CAAC,WAAW;gBAC5B,QAAQ,EAAE,OAAO,CAAC,QAAQ;gBAC1B,KAAK,EAAE,OAAO,CAAC,KAAK;gBACpB,MAAM,EAAE,OAAO,CAAC,MAAM;gBACtB,SAAS,EAAE,OAAO,CAAC,SAAS;gBAC5B,cAAc,EAAE,CAAC,OAAO,CAAC,qBAAqB;gBAC9C,KAAK,EAAE,OAAO,CAAC,KAAK;aACrB,CAAC,CAAC;YACH,cAAc,GAAG,OAAO,CAAC;YAEzB,IAAI,OAAO,CAAC,KAAK,KAAK,KAAK,EAAE,CAAC;gBAC5B,MAAM,CAAC,OAAO,CAAC,oDAAoD,CAAC,CAAC;YACvE,CAAC;YAED,IAAI,OAAO,CAAC,OAAO,CAAC,OAAO,CAAC,EAAE,CAAC;gBAC7B,YAAY,CAAC,OAAO,CAAC,CAAC;YACxB,CAAC;YAED,IAAI,CAAC,OAAO,EAAE,CAAC;gBACb,MAAM,IAAI,KAAK,CAAC,gCAAgC,CAAC,CAAC;YACpD,CAAC;QACH,CAAC;aAAM,IAAI,OAAO,CAAC,KAAK,EAAE,CAAC;YACzB,MAAM,IAAA,yCAAoB,EAAC,IAAI,EAAE;gBAC/B,KAAK,EAAE,IAAI;gBACX,MAAM;gBACN,SAAS,EAAE,OAAO,CAAC,SAAS;gBAC5B,GAAG,UAAU,CAAC,CAAC,CAAC;gBAChB,OAAO,EAAE,OAAO,CAAC,QAAQ,CAAC,CAAC,CAAC,OAAO,CAAC,QAAQ,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,SAAS;gBAC3D,MAAM,EAAE,OAAO,CAAC,WAAW,CAAC,CAAC,CAAC;aAC/B,CAAC,CAAC;QACL,CAAC;IACH,CAAC;YAAS,CAAC;QACT,KAAK,IAAI,CAAC,SAAS,EAAE,CAAC;IACxB,CAAC;IAED,IAAI,kBAAkB,EAAE,CAAC;QACvB,MAAM,IAAI,KAAK,CAAC,gCAAgC,CAAC,CAAC;IACpD,CAAC;IACD,IAAI,eAAe,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;QAC/B,IAAI,OAAO,GAAG,EAAE,CAAC;QACjB,IAAI,CAAC,OAAO,CAAC,iBAAiB,EAAE,CAAC;YAC/B,OAAO,GAAG,6DAA6D,CAAC;QAC1E,CAAC;QACD,IAAI,CAAC,cAAc,EAAE,CAAC;YACpB,MAAM,IAAI,KAAK,CAAC,uBAAuB,OAAO,EAAE,CAAC,CAAC;QACpD,CAAC;IACH,CAAC;AACH,CAAC;AAED,SAAS,iBAAiB,CAAC,IAU1B;IACC,IAAI,IAAI,CAAC,KAAK,EAAE,CAAC;QACf,IACE,CAAC,IAAI,CAAC,WAAW,IAAI,IAAI,CAAC,WAAW,CAAC,MAAM,GAAG,CAAC,CAAC;eAC5C,CAAC,IAAI,CAAC,QAAQ,IAAI,IAAI,CAAC,QAAQ,CAAC,MAAM,GAAG,CAAC,CAAC;eAC3C,IAAI,CAAC,KAAK,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;YAC7B,MAAM,IAAI,KAAK,CAAC,8EAA8E;gBAC5F,qDAAqD,CAAC,CAAC;QAC3D,CAAC;QAED,IAAI,IAAI,CAAC,iBAAiB,IAAI,IAAI,CAAC,qBAAqB,IAAI,IAAI,CAAC,KAAK,IAAI,IAAI,CAAC,MAAM,EAAE,CAAC;YACtF,MAAM,CAAC,OAAO,CAAC,2HAA2H,CAAC,CAAC;QAC9I,CAAC;IACH,CAAC;AACH,CAAC;AAED,SAAS,uBAAuB,CAAC,OAA4B;IAC3D,IAAI,OAAO,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;QACvB,MAAM,CAAC,OAAO,CAAC,+BAA+B,EAAE,KAAK,CAAC,IAAI,CAAC,qBAAqB,CAAC,CAAC,CAAC;QACnF,OAAO,CAAC,OAAO,CAAC,MAAM,CAAC,EAAE;YACvB,MAAM,CAAC,OAAO,CAAC,2CAA2C,EAAE,MAAM,CAAC,SAAS,EAAE,MAAM,CAAC,SAAS,EAAE,MAAM,CAAC,MAAM,CAAC,CAAC;QACjH,CAAC,CAAC,CAAC;QACH,MAAM,CAAC,OAAO,CAAC,oFAAoF,CAAC,CAAC;IACvG,CAAC;AACH,CAAC;AAED,SAAS,YAAY,CAAC,OAA6B;IACjD,MAAM,CAAC,SAAS,CAAC,qCAAqC,CAAC,CAAC;IACxD,MAAM,aAAa,GAAG,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,QAAQ,GAAG,CAAC,CAAC,QAAQ,CAAC,CAAC;IACtE,aAAa,CAAC,OAAO,CAAC,MAAM,CAAC,EAAE;QAC7B,MAAM,CAAC,KAAK,CAAC,uCAAuC,EAAE,MAAM,CAAC,OAAO,EAAE,MAAM,CAAC,MAAM,EAAE,MAAM,CAAC,QAAQ,CAAC,CAAC;QACtG,MAAM,WAAW,GAAG,MAAM,CAAC,OAAO,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;QAC7E,WAAW,CAAC,OAAO,CAAC,IAAI,CAAC,EAAE,CAAC,MAAM,CAAC,KAAK,CAAC,UAAU,EAAE,IAAI,CAAC,CAAC,CAAC,EAAE,IAAI,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;IAC1E,CAAC,CAAC,CAAC;AACL,CAAC;AAED;;;;;;;;GAQG;AACH,SAAS,cAAc,CAAC,EAAY;IAClC,IAAI,EAAE,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;QACpB,OAAO,SAAS,CAAC;IACnB,CAAC;IACD,OAAO,EAAE,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,KAAK,EAAE,CAAC,CAAC;AAClC,CAAC;AAED;;;;GAIG;AACH,SAAS,UAAU,CAAC,YAA6B,EAAE,mBAA4C;IAC7F,MAAM,eAAe,GAAG,IAAI,GAAG,CAAC,mBAAmB,CAAC,GAAG,CAAC,IAAI,CAAC,EAAE,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC,CAAC;IAChF,MAAM,KAAK,GAA4B,mBAAmB,CAAC;IAC3D,KAAK,CAAC,IAAI,CAAC,GAAG,YAAY,CAAC,MAAM,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC,eAAe,CAAC,GAAG,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC,CAAC,CAAC;IAChF,OAAO,KAAK,CAAC;AACf,CAAC;AAED,SAAgB,GAAG,CAAC,OAAiB,OAAO,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC,CAAC;IACxD,IAAI,CAAC,IAAI,CAAC,CAAC,IAAI,EAAE,CAAC,KAAK,CAAC,GAAG,EAAE;QAC3B,OAAO,CAAC,QAAQ,GAAG,CAAC,CAAC;IACvB,CAAC,CAAC,CAAC;AACL,CAAC;AAED;;;;GAIG;AACH,SAAS,cAAc,CAAC,QAAiB;IACvC,IAAI,CAAC,QAAQ,EAAE,CAAC;QACd,OAAO,EAAE,CAAC;IACZ,CAAC;IAED,IAAI,CAAC;QACH,OAAO,IAAI,CAAC,KAAK,CAAC,EAAE,CAAC,YAAY,CAAC,QAAQ,EAAE,EAAE,QAAQ,EAAE,OAAO,EAAE,CAAC,CAAC,CAAC;IACtE,CAAC;IAAC,MAAM,CAAC;QACP,OAAO,EAAE,CAAC;IACZ,CAAC;AACH,CAAC;AAED,SAAS,iBAAiB,CAAC,OAAgC;IACzD,IAAI,OAAO,CAAC,QAAQ,EAAE,QAAQ,CAAC,uBAAuB,CAAC,EAAE,CAAC;QACxD,MAAM,CAAC,OAAO,CAAC,qXAAqX,CAAC,CAAC;QACtY,OAAO,EAAE,MAAM,EAAE,aAAa,EAAE,CAAC;IACnC,CAAC;IACD,OAAO,EAAE,MAAM,EAAE,aAAa,EAAE,CAAC;AACnC,CAAC","sourcesContent":["// Exercise all integ stacks and if they deploy, update the expected synth files\nimport * as fs from 'fs';\nimport * as path from 'path';\nimport * as chalk from 'chalk';\nimport * as workerpool from 'workerpool';\nimport * as logger from './logger';\nimport type { EngineOptions } from './runner/engine';\nimport type { IntegTest, IntegTestInfo } from './runner/integration-tests';\nimport { IntegrationTests } from './runner/integration-tests';\nimport type { IntegRunnerMetrics, IntegTestWorkerConfig, DestructiveChange } from './workers';\nimport { runSnapshotTests, runIntegrationTests } from './workers';\nimport { watchIntegrationTest } from './workers/integ-watch-worker';\n\n// https://github.com/yargs/yargs/issues/1929\n// https://github.com/evanw/esbuild/issues/1492\n// eslint-disable-next-line @typescript-eslint/no-require-imports\nconst yargs = require('yargs');\n\nexport function parseCliArgs(args: string[] = []) {\n  const argv = yargs\n    .usage('Usage: integ-runner [TEST...]')\n    .option('config', {\n      config: true,\n      configParser: configFromFile,\n      default: 'integ.config.json',\n      desc: 'Load options from a JSON config file. Options provided as CLI arguments take precedent.',\n    })\n    .option('watch', { type: 'boolean', default: false, desc: 'Perform integ tests in watch mode' })\n    .option('list', { type: 'boolean', default: false, desc: 'List tests instead of running them' })\n    .option('clean', { type: 'boolean', default: true, desc: 'Clean up and delete stack after test is completed (use --no-clean to negate)' })\n    .option('verbose', { type: 'boolean', default: false, alias: 'v', count: true, desc: 'Verbose logs and metrics on integration tests durations (specify multiple times to increase verbosity)' })\n    .option('dry-run', { type: 'boolean', default: false, desc: 'do not actually deploy the stack. just update the snapshot (not recommended!)' })\n    .option('update-on-failed', { type: 'boolean', default: false, desc: 'rerun integration tests and update snapshots for failed tests.' })\n    .option('force', { type: 'boolean', default: false, desc: 'Rerun all integration tests even if tests are passing' })\n    .option('parallel-regions', { type: 'array', desc: 'Tests are run in parallel across these regions. To prevent tests from running in parallel, provide only a single region', default: [] })\n    .options('directory', { type: 'string', default: 'test', desc: 'starting directory to discover integration tests. Tests will be discovered recursively from this directory' })\n    .options('profiles', { type: 'array', desc: 'list of AWS profiles to use. Tests will be run in parallel across each profile+regions', default: [] })\n    .options('max-workers', { type: 'number', desc: 'The max number of workerpool workers to use when running integration tests in parallel', default: 16 })\n    .options('exclude', { type: 'boolean', desc: 'Run all tests in the directory, except the specified TESTs', default: false })\n    .option('strict', { type: 'boolean', default: false, desc: 'Fail if any specified tests are not found' })\n    .options('from-file', { type: 'string', desc: 'Read TEST names from a file (one TEST per line)' })\n    .option('inspect-failures', { type: 'boolean', desc: 'Keep the integ test cloud assembly if a failure occurs for inspection', default: false })\n    .option('disable-update-workflow', { type: 'boolean', default: false, desc: 'If this is \"true\" then the stack update workflow will be disabled' })\n    .option('language', {\n      alias: 'l',\n      default: ['javascript', 'typescript', 'python', 'go'],\n      choices: ['javascript', 'typescript', 'python', 'go'],\n      type: 'array',\n      nargs: 1,\n      desc: 'Use these presets to run integration tests for the selected languages',\n    })\n    .option('app', { type: 'string', default: undefined, desc: 'The custom CLI command that will be used to run the test files. You can include {filePath} to specify where in the command the test file path should be inserted. Example: --app=\"python3.8 {filePath}\".' })\n    .option('test-regex', { type: 'array', desc: 'Detect integration test files matching this JavaScript regex pattern. If used multiple times, all files matching any one of the patterns are detected.', default: [] })\n    .option('unstable', { type: 'array', desc: 'Opt-in to using unstable features. By using these flags you acknowledges that scope and APIs of unstable features may change without notice. Specify multiple times for each unstable feature you want to opt-in to.', nargs: 1, choices: ['toolkit-lib-engine', 'deprecated-cli-engine'], default: [] })\n    .strict()\n    .parse(args);\n\n  const tests: string[] = argv._;\n  const parallelRegions = arrayFromYargs(argv['parallel-regions']);\n  const testRegions: string[] = parallelRegions ?? ['us-east-1', 'us-east-2', 'us-west-2'];\n  const profiles = arrayFromYargs(argv.profiles);\n  const fromFile: string | undefined = argv['from-file'];\n  const maxWorkers: number = argv['max-workers'];\n  const verbosity: number = argv.verbose;\n  const verbose: boolean = verbosity >= 1;\n\n  const numTests = testRegions.length * (profiles ?? [1]).length;\n  if (maxWorkers < numTests) {\n    logger.warning('You are attempting to run %s tests in parallel, but only have %s workers. Not all of your profiles+regions will be utilized', numTests, maxWorkers);\n  }\n\n  if (tests.length > 0 && fromFile) {\n    throw new Error('A list of tests cannot be provided if \"--from-file\" is provided');\n  }\n\n  if (argv.strict && argv.exclude) {\n    throw new Error('Cannot use --strict with --exclude');\n  }\n\n  const requestedTests = fromFile\n    ? (fs.readFileSync(fromFile, { encoding: 'utf8' })).split('\\n').filter(x => x)\n    : (tests.length > 0 ? tests : undefined); // 'undefined' means no request\n\n  return {\n    tests: requestedTests,\n    app: argv.app as (string | undefined),\n    testRegex: arrayFromYargs(argv['test-regex']),\n    testRegions,\n    originalRegions: parallelRegions,\n    profiles,\n    runUpdateOnFailed: (argv['update-on-failed'] ?? false) as boolean,\n    fromFile,\n    exclude: argv.exclude as boolean,\n    maxWorkers,\n    list: argv.list as boolean,\n    directory: argv.directory as string,\n    inspectFailures: argv['inspect-failures'] as boolean,\n    verbosity,\n    verbose,\n    clean: argv.clean as boolean,\n    force: argv.force as boolean,\n    dryRun: argv['dry-run'] as boolean,\n    disableUpdateWorkflow: argv['disable-update-workflow'] as boolean,\n    language: arrayFromYargs(argv.language),\n    watch: argv.watch as boolean,\n    strict: argv.strict as boolean,\n    unstable: arrayFromYargs(argv.unstable) ?? [],\n  };\n}\n\nexport async function main(args: string[]) {\n  let engineForError;\n  try {\n    const options = parseCliArgs(args);\n    const engine = engineFromOptions(options);\n    engineForError = engine.engine;\n    await run(options, engine);\n  } catch (err: any) {\n    logger.error(err);\n    if (engineForError === 'toolkit-lib') {\n      logger.warning('\\n[Notice] You are using the new default engine to run integration tests. If you think the above failure has been caused by the new engine, you may choose to temporarily revert to the old engine by adding the `--unstable=deprecated-cli-engine` option. Please note that this engine is deprecated and scheduled to be removed in January 2026.\\n\\nIf reverting to the old engine resolves an issue for you, please let us know so we can address this in the new engine. Report issues here: https://github.com/aws/aws-cdk-cli/issues/new/choose');\n    }\n    throw err;\n  }\n}\n\nasync function run(options: ReturnType<typeof parseCliArgs>, { engine }: EngineOptions) {\n  const testsFromArgs = await new IntegrationTests(path.resolve(options.directory)).fromCliOptions(options);\n\n  // List only prints the discovered tests\n  if (options.list) {\n    process.stdout.write(testsFromArgs.map(t => t.discoveryRelativeFileName).join('\\n') + '\\n');\n    return;\n  }\n\n  const pool = workerpool.pool(path.join(__dirname, '..', 'lib', 'workers', 'extract', 'index.js'), {\n    maxWorkers: options.watch ? 1 : options.maxWorkers,\n  });\n\n  const testsToRun: IntegTestWorkerConfig[] = [];\n  let destructiveChanges: boolean = false;\n  let failedSnapshots: IntegTestWorkerConfig[] = [];\n  let testsSucceeded = false;\n  validateWatchArgs({\n    ...options,\n    testRegions: options.originalRegions,\n    tests: testsFromArgs,\n  });\n\n  try {\n    if (!options.watch) {\n      // always run snapshot tests, but if '--force' is passed then\n      // run integration tests on all failed tests, not just those that\n      // failed snapshot tests\n      failedSnapshots = await runSnapshotTests(pool, testsFromArgs, {\n        retain: options.inspectFailures,\n        verbose: options.verbose,\n        engine,\n      });\n      for (const failure of failedSnapshots) {\n        logger.warning(`Failed: ${failure.fileName}`);\n        if (failure.destructiveChanges && failure.destructiveChanges.length > 0) {\n          printDestructiveChanges(failure.destructiveChanges);\n          destructiveChanges = true;\n        }\n      }\n      if (!options.force) {\n        testsToRun.push(...failedSnapshots);\n      } else {\n        // if any of the test failed snapshot tests, keep those results\n        // and merge with the rest of the tests from args\n        testsToRun.push(...mergeTests(testsFromArgs.map(t => t.info), failedSnapshots));\n      }\n    } else {\n      testsToRun.push(...testsFromArgs.map(t => t.info));\n    }\n\n    // run integration tests if `--update-on-failed` OR `--force` is used\n    if (options.runUpdateOnFailed || options.force) {\n      const { success, metrics } = await runIntegrationTests({\n        pool,\n        engine,\n        tests: testsToRun,\n        regions: options.testRegions,\n        profiles: options.profiles,\n        clean: options.clean,\n        dryRun: options.dryRun,\n        verbosity: options.verbosity,\n        updateWorkflow: !options.disableUpdateWorkflow,\n        watch: options.watch,\n      });\n      testsSucceeded = success;\n\n      if (options.clean === false) {\n        logger.warning('Not cleaning up stacks since \"--no-clean\" was used');\n      }\n\n      if (Boolean(options.verbose)) {\n        printMetrics(metrics);\n      }\n\n      if (!success) {\n        throw new Error('Some integration tests failed!');\n      }\n    } else if (options.watch) {\n      await watchIntegrationTest(pool, {\n        watch: true,\n        engine,\n        verbosity: options.verbosity,\n        ...testsToRun[0],\n        profile: options.profiles ? options.profiles[0] : undefined,\n        region: options.testRegions[0],\n      });\n    }\n  } finally {\n    void pool.terminate();\n  }\n\n  if (destructiveChanges) {\n    throw new Error('Some changes were destructive!');\n  }\n  if (failedSnapshots.length > 0) {\n    let message = '';\n    if (!options.runUpdateOnFailed) {\n      message = 'To re-run failed tests run: integ-runner --update-on-failed';\n    }\n    if (!testsSucceeded) {\n      throw new Error(`Some tests failed!\\n${message}`);\n    }\n  }\n}\n\nfunction validateWatchArgs(args: {\n  tests: IntegTest[];\n  testRegions?: string[];\n  profiles?: string[];\n  maxWorkers: number;\n  force: boolean;\n  dryRun: boolean;\n  disableUpdateWorkflow: boolean;\n  runUpdateOnFailed: boolean;\n  watch: boolean;\n}) {\n  if (args.watch) {\n    if (\n      (args.testRegions && args.testRegions.length > 1)\n        || (args.profiles && args.profiles.length > 1)\n        || args.tests.length > 1) {\n      throw new Error('Running with watch only supports a single test. Only provide a single option'+\n        'to `--profiles` `--parallel-regions` `--max-workers');\n    }\n\n    if (args.runUpdateOnFailed || args.disableUpdateWorkflow || args.force || args.dryRun) {\n      logger.warning('args `--update-on-failed`, `--disable-update-workflow`, `--force`, `--dry-run` have no effect when running with `--watch`');\n    }\n  }\n}\n\nfunction printDestructiveChanges(changes: DestructiveChange[]): void {\n  if (changes.length > 0) {\n    logger.warning('!!! This test contains %s !!!', chalk.bold('destructive changes'));\n    changes.forEach(change => {\n      logger.warning('    Stack: %s - Resource: %s - Impact: %s', change.stackName, change.logicalId, change.impact);\n    });\n    logger.warning('!!! If these destructive changes are necessary, please indicate this on the PR !!!');\n  }\n}\n\nfunction printMetrics(metrics: IntegRunnerMetrics[]): void {\n  logger.highlight('   --- Integration test metrics ---');\n  const sortedMetrics = metrics.sort((a, b) => a.duration - b.duration);\n  sortedMetrics.forEach(metric => {\n    logger.print('Profile %s + Region %s total time: %s', metric.profile, metric.region, metric.duration);\n    const sortedTests = Object.entries(metric.tests).sort((a, b) => a[1] - b[1]);\n    sortedTests.forEach(test => logger.print('  %s: %s', test[0], test[1]));\n  });\n}\n\n/**\n * Translate a Yargs input array to something that makes more sense in a programming language\n * model (telling the difference between absence and an empty array)\n *\n * - An empty array is the default case, meaning the user didn't pass any arguments. We return\n *   undefined.\n * - If the user passed a single empty string, they did something like `--array=`, which we'll\n *   take to mean they passed an empty array.\n */\nfunction arrayFromYargs(xs: string[]): string[] | undefined {\n  if (xs.length === 0) {\n    return undefined;\n  }\n  return xs.filter(x => x !== '');\n}\n\n/**\n * Merge the tests we received from command line arguments with\n * tests that failed snapshot tests. The failed snapshot tests have additional\n * information that we want to keep so this should override any test from args\n */\nfunction mergeTests(testFromArgs: IntegTestInfo[], failedSnapshotTests: IntegTestWorkerConfig[]): IntegTestWorkerConfig[] {\n  const failedTestNames = new Set(failedSnapshotTests.map(test => test.fileName));\n  const final: IntegTestWorkerConfig[] = failedSnapshotTests;\n  final.push(...testFromArgs.filter(test => !failedTestNames.has(test.fileName)));\n  return final;\n}\n\nexport function cli(args: string[] = process.argv.slice(2)) {\n  main(args).then().catch(() => {\n    process.exitCode = 1;\n  });\n}\n\n/**\n * Read CLI options from a config file if provided.\n *\n * @returns parsed CLI config options\n */\nfunction configFromFile(fileName?: string): Record<string, any> {\n  if (!fileName) {\n    return {};\n  }\n\n  try {\n    return JSON.parse(fs.readFileSync(fileName, { encoding: 'utf-8' }));\n  } catch {\n    return {};\n  }\n}\n\nfunction engineFromOptions(options: { unstable?: string[] }): Required<EngineOptions> {\n  if (options.unstable?.includes('deprecated-cli-engine')) {\n    logger.warning('[Deprecation Notice] You have opted-in to use the deprecated CLI engine which is scheduled to be removed in January 2026. If you have encountered blockers while using the new default engine, please let us know by opening an issue: https://github.com/aws/aws-cdk-cli/issues/new/choose\\n\\nTo use the new default engine, remove the `--unstable=deprecated-cli-engine` option.');\n    return { engine: 'cli-wrapper' };\n  }\n  return { engine: 'toolkit-lib' };\n}\n"]}
package/lib/index.js CHANGED
@@ -5414,7 +5414,8 @@ var init_integration_tests = __esm({
5414
5414
  async fromCliOptions(options) {
5415
5415
  const baseOptions = {
5416
5416
  tests: options.tests,
5417
- exclude: options.exclude
5417
+ exclude: options.exclude,
5418
+ strict: options.strict
5418
5419
  };
5419
5420
  if (options.app && options.testRegex) {
5420
5421
  return this.discover({
@@ -5470,7 +5471,7 @@ var init_integration_tests = __esm({
5470
5471
  * If they have provided a test name that we don't find, then we write out that error message.
5471
5472
  * - If it is a list of tests to exclude, then we discover all available tests and filter out the tests that were provided by the user.
5472
5473
  */
5473
- filterTests(discoveredTests, requestedTests, exclude) {
5474
+ filterTests(discoveredTests, requestedTests, exclude, strict) {
5474
5475
  if (!requestedTests) {
5475
5476
  return discoveredTests;
5476
5477
  }
@@ -5487,6 +5488,9 @@ var init_integration_tests = __esm({
5487
5488
  if (unmatchedPatterns.length > 0) {
5488
5489
  process.stderr.write(`Available tests: ${discoveredTests.map((t) => t.discoveryRelativeFileName).join(" ")}
5489
5490
  `);
5491
+ if (strict) {
5492
+ throw new Error(`Strict mode: ${unmatchedPatterns.length} test(s) not found: ${unmatchedPatterns.join(", ")}`);
5493
+ }
5490
5494
  return [];
5491
5495
  }
5492
5496
  }
@@ -5512,7 +5516,7 @@ var init_integration_tests = __esm({
5512
5516
  }))
5513
5517
  );
5514
5518
  const discoveredTests = ignoreUncompiledTypeScript ? this.filterUncompiledTypeScript(testCases) : testCases;
5515
- return this.filterTests(discoveredTests, options.tests, options.exclude);
5519
+ return this.filterTests(discoveredTests, options.tests, options.exclude, options.strict);
5516
5520
  }
5517
5521
  filterUncompiledTypeScript(testCases) {
5518
5522
  const jsTestCases = testCases.filter((t) => t.fileName.endsWith(".js"));
@@ -10487,7 +10491,7 @@ function parseCliArgs(args = []) {
10487
10491
  configParser: configFromFile,
10488
10492
  default: "integ.config.json",
10489
10493
  desc: "Load options from a JSON config file. Options provided as CLI arguments take precedent."
10490
- }).option("watch", { type: "boolean", default: false, desc: "Perform integ tests in watch mode" }).option("list", { type: "boolean", default: false, desc: "List tests instead of running them" }).option("clean", { type: "boolean", default: true, desc: "Clean up and delete stack after test is completed (use --no-clean to negate)" }).option("verbose", { type: "boolean", default: false, alias: "v", count: true, desc: "Verbose logs and metrics on integration tests durations (specify multiple times to increase verbosity)" }).option("dry-run", { type: "boolean", default: false, desc: "do not actually deploy the stack. just update the snapshot (not recommended!)" }).option("update-on-failed", { type: "boolean", default: false, desc: "rerun integration tests and update snapshots for failed tests." }).option("force", { type: "boolean", default: false, desc: "Rerun all integration tests even if tests are passing" }).option("parallel-regions", { type: "array", desc: "Tests are run in parallel across these regions. To prevent tests from running in parallel, provide only a single region", default: [] }).options("directory", { type: "string", default: "test", desc: "starting directory to discover integration tests. Tests will be discovered recursively from this directory" }).options("profiles", { type: "array", desc: "list of AWS profiles to use. Tests will be run in parallel across each profile+regions", default: [] }).options("max-workers", { type: "number", desc: "The max number of workerpool workers to use when running integration tests in parallel", default: 16 }).options("exclude", { type: "boolean", desc: "Run all tests in the directory, except the specified TESTs", default: false }).options("from-file", { type: "string", desc: "Read TEST names from a file (one TEST per line)" }).option("inspect-failures", { type: "boolean", desc: "Keep the integ test cloud assembly if a failure occurs for inspection", default: false }).option("disable-update-workflow", { type: "boolean", default: false, desc: 'If this is "true" then the stack update workflow will be disabled' }).option("language", {
10494
+ }).option("watch", { type: "boolean", default: false, desc: "Perform integ tests in watch mode" }).option("list", { type: "boolean", default: false, desc: "List tests instead of running them" }).option("clean", { type: "boolean", default: true, desc: "Clean up and delete stack after test is completed (use --no-clean to negate)" }).option("verbose", { type: "boolean", default: false, alias: "v", count: true, desc: "Verbose logs and metrics on integration tests durations (specify multiple times to increase verbosity)" }).option("dry-run", { type: "boolean", default: false, desc: "do not actually deploy the stack. just update the snapshot (not recommended!)" }).option("update-on-failed", { type: "boolean", default: false, desc: "rerun integration tests and update snapshots for failed tests." }).option("force", { type: "boolean", default: false, desc: "Rerun all integration tests even if tests are passing" }).option("parallel-regions", { type: "array", desc: "Tests are run in parallel across these regions. To prevent tests from running in parallel, provide only a single region", default: [] }).options("directory", { type: "string", default: "test", desc: "starting directory to discover integration tests. Tests will be discovered recursively from this directory" }).options("profiles", { type: "array", desc: "list of AWS profiles to use. Tests will be run in parallel across each profile+regions", default: [] }).options("max-workers", { type: "number", desc: "The max number of workerpool workers to use when running integration tests in parallel", default: 16 }).options("exclude", { type: "boolean", desc: "Run all tests in the directory, except the specified TESTs", default: false }).option("strict", { type: "boolean", default: false, desc: "Fail if any specified tests are not found" }).options("from-file", { type: "string", desc: "Read TEST names from a file (one TEST per line)" }).option("inspect-failures", { type: "boolean", desc: "Keep the integ test cloud assembly if a failure occurs for inspection", default: false }).option("disable-update-workflow", { type: "boolean", default: false, desc: 'If this is "true" then the stack update workflow will be disabled' }).option("language", {
10491
10495
  alias: "l",
10492
10496
  default: ["javascript", "typescript", "python", "go"],
10493
10497
  choices: ["javascript", "typescript", "python", "go"],
@@ -10510,6 +10514,9 @@ function parseCliArgs(args = []) {
10510
10514
  if (tests.length > 0 && fromFile) {
10511
10515
  throw new Error('A list of tests cannot be provided if "--from-file" is provided');
10512
10516
  }
10517
+ if (argv.strict && argv.exclude) {
10518
+ throw new Error("Cannot use --strict with --exclude");
10519
+ }
10513
10520
  const requestedTests = fromFile ? fs2.readFileSync(fromFile, { encoding: "utf8" }).split("\n").filter((x) => x) : tests.length > 0 ? tests : void 0;
10514
10521
  return {
10515
10522
  tests: requestedTests,
@@ -10533,6 +10540,7 @@ function parseCliArgs(args = []) {
10533
10540
  disableUpdateWorkflow: argv["disable-update-workflow"],
10534
10541
  language: arrayFromYargs(argv.language),
10535
10542
  watch: argv.watch,
10543
+ strict: argv.strict,
10536
10544
  unstable: arrayFromYargs(argv.unstable) ?? []
10537
10545
  };
10538
10546
  }
@@ -109,6 +109,12 @@ export interface IntegrationTestsDiscoveryOptions {
109
109
  * @default false
110
110
  */
111
111
  readonly exclude?: boolean;
112
+ /**
113
+ * If this is set to true, throw an error if any specified tests are not found
114
+ *
115
+ * @default false
116
+ */
117
+ readonly strict?: boolean;
112
118
  /**
113
119
  * List of tests to include (or exclude if `exclude=true`)
114
120
  *
@@ -140,6 +146,7 @@ export declare class IntegrationTests {
140
146
  language?: string[];
141
147
  testRegex?: string[];
142
148
  tests?: string[];
149
+ strict?: boolean;
143
150
  }): Promise<IntegTest[]>;
144
151
  /**
145
152
  * Get the default configuration for a language
@@ -74,6 +74,7 @@ class IntegrationTests {
74
74
  const baseOptions = {
75
75
  tests: options.tests,
76
76
  exclude: options.exclude,
77
+ strict: options.strict,
77
78
  };
78
79
  // Explicitly set both, app and test-regex
79
80
  if (options.app && options.testRegex) {
@@ -136,7 +137,7 @@ class IntegrationTests {
136
137
  * If they have provided a test name that we don't find, then we write out that error message.
137
138
  * - If it is a list of tests to exclude, then we discover all available tests and filter out the tests that were provided by the user.
138
139
  */
139
- filterTests(discoveredTests, requestedTests, exclude) {
140
+ filterTests(discoveredTests, requestedTests, exclude, strict) {
140
141
  if (!requestedTests) {
141
142
  return discoveredTests;
142
143
  }
@@ -152,6 +153,9 @@ class IntegrationTests {
152
153
  }
153
154
  if (unmatchedPatterns.length > 0) {
154
155
  process.stderr.write(`Available tests: ${discoveredTests.map(t => t.discoveryRelativeFileName).join(' ')}\n`);
156
+ if (strict) {
157
+ throw new Error(`Strict mode: ${unmatchedPatterns.length} test(s) not found: ${unmatchedPatterns.join(', ')}`);
158
+ }
155
159
  return [];
156
160
  }
157
161
  }
@@ -178,7 +182,7 @@ class IntegrationTests {
178
182
  appCommand,
179
183
  })));
180
184
  const discoveredTests = ignoreUncompiledTypeScript ? this.filterUncompiledTypeScript(testCases) : testCases;
181
- return this.filterTests(discoveredTests, options.tests, options.exclude);
185
+ return this.filterTests(discoveredTests, options.tests, options.exclude, options.strict);
182
186
  }
183
187
  filterUncompiledTypeScript(testCases) {
184
188
  const jsTestCases = testCases.filter(t => t.fileName.endsWith('.js'));
@@ -212,4 +216,4 @@ class IntegrationTests {
212
216
  }
213
217
  }
214
218
  exports.IntegrationTests = IntegrationTests;
215
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"integration-tests.js","sourceRoot":"","sources":["integration-tests.ts"],"names":[],"mappings":"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","sourcesContent":["import * as path from 'path';\nimport * as fs from 'fs-extra';\n\nconst CDK_OUTDIR_PREFIX = 'cdk-integ.out';\n\n/**\n * Represents a single integration test\n *\n * This type is a data-only structure, so it can trivially be passed to workers.\n * Derived attributes are calculated using the `IntegTest` class.\n */\nexport interface IntegTestInfo {\n  /**\n   * Path to the file to run\n   *\n   * Path is relative to the current working directory.\n   */\n  readonly fileName: string;\n\n  /**\n   * The root directory we discovered this test from\n   *\n   * Path is relative to the current working directory.\n   */\n  readonly discoveryRoot: string;\n\n  /**\n   * The CLI command used to run this test.\n   * If it contains {filePath}, the test file names will be substituted at that place in the command for each run.\n   *\n   * @default - test run command will be `node {filePath}`\n   */\n  readonly appCommand?: string;\n\n  /**\n   * true if this test is running in watch mode\n   *\n   * @default false\n   */\n  readonly watch?: boolean;\n}\n\n/**\n * Derived information for IntegTests\n */\nexport class IntegTest {\n  /**\n   * The name of the file to run\n   *\n   * Path is relative to the current working directory.\n   */\n  public readonly fileName: string;\n\n  /**\n   * Relative path to the file to run\n   *\n   * Relative from the \"discovery root\".\n   */\n  public readonly discoveryRelativeFileName: string;\n\n  /**\n   * The absolute path to the file\n   */\n  public readonly absoluteFileName: string;\n\n  /**\n   * The normalized name of the test. This name\n   * will be the same regardless of what directory the tool\n   * is run from.\n   */\n  public readonly normalizedTestName: string;\n\n  /**\n   * Directory the test is in\n   */\n  public readonly directory: string;\n\n  /**\n   * Display name for the test\n   *\n   * Depends on the discovery directory.\n   *\n   * Looks like `integ.mytest` or `package/test/integ.mytest`.\n   */\n  public readonly testName: string;\n\n  /**\n   * Path of the snapshot directory for this test\n   */\n  public readonly snapshotDir: string;\n\n  /**\n   * Path to the temporary output directory for this test\n   */\n  public readonly temporaryOutputDir: string;\n\n  /**\n   * The CLI command used to run this test.\n   * If it contains {filePath}, the test file names will be substituted at that place in the command for each run.\n   *\n   * @default - test run command will be `node {filePath}`\n   */\n  readonly appCommand: string;\n\n  constructor(public readonly info: IntegTestInfo) {\n    this.appCommand = info.appCommand ?? 'node {filePath}';\n    this.absoluteFileName = path.resolve(info.fileName);\n    this.fileName = path.relative(process.cwd(), info.fileName);\n\n    const parsed = path.parse(this.fileName);\n    this.discoveryRelativeFileName = path.relative(info.discoveryRoot, info.fileName);\n    // if `--watch` then we need the directory to be the cwd\n    this.directory = info.watch ? process.cwd() : parsed.dir;\n\n    // if we are running in a package directory then just use the fileName\n    // as the testname, but if we are running in a parent directory with\n    // multiple packages then use the directory/filename as the testname\n    //\n    // Looks either like `integ.mytest` or `package/test/integ.mytest`.\n    const relDiscoveryRoot = path.relative(process.cwd(), info.discoveryRoot);\n    this.testName = this.directory === path.join(relDiscoveryRoot, 'test') || this.directory === path.join(relDiscoveryRoot)\n      ? parsed.name\n      : path.join(path.relative(this.info.discoveryRoot, parsed.dir), parsed.name);\n\n    this.normalizedTestName = parsed.name;\n    this.snapshotDir = path.join(parsed.dir, `${parsed.base}.snapshot`);\n    this.temporaryOutputDir = path.join(parsed.dir, `${CDK_OUTDIR_PREFIX}.${parsed.base}.snapshot`);\n  }\n\n  /**\n   * Whether this test matches the user-given name\n   *\n   * We are very lenient here. A name matches if it matches:\n   *\n   * - The CWD-relative filename\n   * - The discovery root-relative filename\n   * - The suite name\n   * - The absolute filename\n   */\n  public matches(name: string) {\n    return [\n      this.fileName,\n      this.discoveryRelativeFileName,\n      this.testName,\n      this.absoluteFileName,\n    ].includes(name);\n  }\n}\n\n/**\n * Configuration options how integration test files are discovered\n */\nexport interface IntegrationTestsDiscoveryOptions {\n  /**\n   * If this is set to true then the list of tests\n   * provided will be excluded\n   *\n   * @default false\n   */\n  readonly exclude?: boolean;\n\n  /**\n   * List of tests to include (or exclude if `exclude=true`)\n   *\n   * @default - all matched files\n   */\n  readonly tests?: string[];\n\n  /**\n   * A map of of the app commands to run integration tests with,\n   * and the regex patterns matching the integration test files each app command.\n   *\n   * If the app command contains {filePath}, the test file names will be substituted at that place in the command for each run.\n   */\n  readonly testCases: {\n    [app: string]: string[];\n  };\n}\n\n/**\n * Returns the name of the Python executable for the current OS\n */\nfunction pythonExecutable() {\n  let python = 'python3';\n  if (process.platform === 'win32') {\n    python = 'python';\n  }\n  return python;\n}\n\n/**\n * Discover integration tests\n */\nexport class IntegrationTests {\n  constructor(private readonly directory: string) {\n  }\n\n  /**\n   * Get integration tests discovery options from CLI options\n   */\n  public async fromCliOptions(options: {\n    app?: string;\n    exclude?: boolean;\n    language?: string[];\n    testRegex?: string[];\n    tests?: string[];\n  }): Promise<IntegTest[]> {\n    const baseOptions = {\n      tests: options.tests,\n      exclude: options.exclude,\n    };\n\n    // Explicitly set both, app and test-regex\n    if (options.app && options.testRegex) {\n      return this.discover({\n        testCases: {\n          [options.app]: options.testRegex,\n        },\n        ...baseOptions,\n      });\n    }\n\n    // Use the selected presets\n    if (!options.app && !options.testRegex) {\n      // Only case with multiple languages, i.e. the only time we need to check the special case\n      const ignoreUncompiledTypeScript = options.language?.includes('javascript') && options.language?.includes('typescript');\n\n      return this.discover({\n        testCases: this.getLanguagePresets(options.language),\n        ...baseOptions,\n      }, ignoreUncompiledTypeScript);\n    }\n\n    // Only one of app or test-regex is set, with a single preset selected\n    // => override either app or test-regex\n    if (options.language?.length === 1) {\n      const [presetApp, presetTestRegex] = this.getLanguagePreset(options.language[0]);\n      return this.discover({\n        testCases: {\n          [options.app ?? presetApp]: options.testRegex ?? presetTestRegex,\n        },\n        ...baseOptions,\n      });\n    }\n\n    // Only one of app or test-regex is set, with multiple presets\n    // => impossible to resolve\n    const option = options.app ? '--app' : '--test-regex';\n    throw new Error(`Only a single \"--language\" can be used with \"${option}\". Alternatively provide both \"--app\" and \"--test-regex\" to fully customize the configuration.`);\n  }\n\n  /**\n   * Get the default configuration for a language\n   */\n  private getLanguagePreset(language: string) {\n    const languagePresets: {\n      [language: string]: [string, string[]];\n    } = {\n      javascript: ['node {filePath}', ['^integ\\\\..*\\\\.js$']],\n      typescript: ['node -r ts-node/register {filePath}', ['^integ\\\\.(?!.*\\\\.d\\\\.ts$).*\\\\.ts$']],\n      python: [`${pythonExecutable()} {filePath}`, ['^integ_.*\\\\.py$']],\n      go: ['go run {filePath}', ['^integ_.*\\\\.go$']],\n    };\n\n    return languagePresets[language];\n  }\n\n  /**\n   * Get the config for all selected languages\n   */\n  private getLanguagePresets(languages: string[] = []) {\n    return Object.fromEntries(\n      languages\n        .map(language => this.getLanguagePreset(language))\n        .filter(Boolean),\n    );\n  }\n\n  /**\n   * If the user provides a list of tests, these can either be a list of tests to include or a list of tests to exclude.\n   *\n   * - If it is a list of tests to include then we discover all available tests and check whether they have provided valid tests.\n   *   If they have provided a test name that we don't find, then we write out that error message.\n   * - If it is a list of tests to exclude, then we discover all available tests and filter out the tests that were provided by the user.\n   */\n  private filterTests(discoveredTests: IntegTest[], requestedTests?: string[], exclude?: boolean): IntegTest[] {\n    if (!requestedTests) {\n      return discoveredTests;\n    }\n\n    const allTests = discoveredTests.filter(t => {\n      const matches = requestedTests.some(pattern => t.matches(pattern));\n      return matches !== !!exclude; // Looks weird but is equal to (matches && !exclude) || (!matches && exclude)\n    });\n\n    // If not excluding, all patterns must have matched at least one test\n    if (!exclude) {\n      const unmatchedPatterns = requestedTests.filter(pattern => !discoveredTests.some(t => t.matches(pattern)));\n      for (const unmatched of unmatchedPatterns) {\n        process.stderr.write(`No such integ test: ${unmatched}\\n`);\n      }\n      if (unmatchedPatterns.length > 0) {\n        process.stderr.write(`Available tests: ${discoveredTests.map(t => t.discoveryRelativeFileName).join(' ')}\\n`);\n        return [];\n      }\n    }\n\n    return allTests;\n  }\n\n  /**\n   * Takes an optional list of tests to look for, otherwise\n   * it will look for all tests from the directory\n   *\n   * @param tests - Tests to include or exclude, undefined means include all tests.\n   * @param exclude - Whether the 'tests' list is inclusive or exclusive (inclusive by default).\n   */\n  private async discover(options: IntegrationTestsDiscoveryOptions, ignoreUncompiledTypeScript: boolean = false): Promise<IntegTest[]> {\n    const files = await this.readTree();\n\n    const testCases = Object.entries(options.testCases)\n      .flatMap(([appCommand, patterns]) => files\n        .filter(fileName => patterns.some((pattern) => {\n          const regex = new RegExp(pattern);\n          return regex.test(fileName) || regex.test(path.basename(fileName));\n        }))\n        .map(fileName => new IntegTest({\n          discoveryRoot: this.directory,\n          fileName,\n          appCommand,\n        })),\n      );\n\n    const discoveredTests = ignoreUncompiledTypeScript ? this.filterUncompiledTypeScript(testCases) : testCases;\n\n    return this.filterTests(discoveredTests, options.tests, options.exclude);\n  }\n\n  private filterUncompiledTypeScript(testCases: IntegTest[]): IntegTest[] {\n    const jsTestCases = testCases.filter(t => t.fileName.endsWith('.js'));\n\n    return testCases\n      // Remove all TypeScript test cases (ending in .ts)\n      // for which a compiled version is present (same name, ending in .js)\n      .filter((tsCandidate) => {\n        if (!tsCandidate.fileName.endsWith('.ts')) {\n          return true;\n        }\n        return jsTestCases.findIndex(jsTest => jsTest.testName === tsCandidate.testName) === -1;\n      });\n  }\n\n  private async readTree(): Promise<string[]> {\n    const ret = new Array<string>();\n\n    async function recurse(dir: string) {\n      const files = await fs.readdir(dir);\n      for (const file of files) {\n        const fullPath = path.join(dir, file);\n        const statf = await fs.stat(fullPath);\n        if (statf.isFile()) {\n          ret.push(fullPath);\n        }\n        if (statf.isDirectory()) {\n          await recurse(fullPath);\n        }\n      }\n    }\n\n    await recurse(this.directory);\n    return ret;\n  }\n}\n"]}
219
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"integration-tests.js","sourceRoot":"","sources":["integration-tests.ts"],"names":[],"mappings":"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","sourcesContent":["import * as path from 'path';\nimport * as fs from 'fs-extra';\n\nconst CDK_OUTDIR_PREFIX = 'cdk-integ.out';\n\n/**\n * Represents a single integration test\n *\n * This type is a data-only structure, so it can trivially be passed to workers.\n * Derived attributes are calculated using the `IntegTest` class.\n */\nexport interface IntegTestInfo {\n  /**\n   * Path to the file to run\n   *\n   * Path is relative to the current working directory.\n   */\n  readonly fileName: string;\n\n  /**\n   * The root directory we discovered this test from\n   *\n   * Path is relative to the current working directory.\n   */\n  readonly discoveryRoot: string;\n\n  /**\n   * The CLI command used to run this test.\n   * If it contains {filePath}, the test file names will be substituted at that place in the command for each run.\n   *\n   * @default - test run command will be `node {filePath}`\n   */\n  readonly appCommand?: string;\n\n  /**\n   * true if this test is running in watch mode\n   *\n   * @default false\n   */\n  readonly watch?: boolean;\n}\n\n/**\n * Derived information for IntegTests\n */\nexport class IntegTest {\n  /**\n   * The name of the file to run\n   *\n   * Path is relative to the current working directory.\n   */\n  public readonly fileName: string;\n\n  /**\n   * Relative path to the file to run\n   *\n   * Relative from the \"discovery root\".\n   */\n  public readonly discoveryRelativeFileName: string;\n\n  /**\n   * The absolute path to the file\n   */\n  public readonly absoluteFileName: string;\n\n  /**\n   * The normalized name of the test. This name\n   * will be the same regardless of what directory the tool\n   * is run from.\n   */\n  public readonly normalizedTestName: string;\n\n  /**\n   * Directory the test is in\n   */\n  public readonly directory: string;\n\n  /**\n   * Display name for the test\n   *\n   * Depends on the discovery directory.\n   *\n   * Looks like `integ.mytest` or `package/test/integ.mytest`.\n   */\n  public readonly testName: string;\n\n  /**\n   * Path of the snapshot directory for this test\n   */\n  public readonly snapshotDir: string;\n\n  /**\n   * Path to the temporary output directory for this test\n   */\n  public readonly temporaryOutputDir: string;\n\n  /**\n   * The CLI command used to run this test.\n   * If it contains {filePath}, the test file names will be substituted at that place in the command for each run.\n   *\n   * @default - test run command will be `node {filePath}`\n   */\n  readonly appCommand: string;\n\n  constructor(public readonly info: IntegTestInfo) {\n    this.appCommand = info.appCommand ?? 'node {filePath}';\n    this.absoluteFileName = path.resolve(info.fileName);\n    this.fileName = path.relative(process.cwd(), info.fileName);\n\n    const parsed = path.parse(this.fileName);\n    this.discoveryRelativeFileName = path.relative(info.discoveryRoot, info.fileName);\n    // if `--watch` then we need the directory to be the cwd\n    this.directory = info.watch ? process.cwd() : parsed.dir;\n\n    // if we are running in a package directory then just use the fileName\n    // as the testname, but if we are running in a parent directory with\n    // multiple packages then use the directory/filename as the testname\n    //\n    // Looks either like `integ.mytest` or `package/test/integ.mytest`.\n    const relDiscoveryRoot = path.relative(process.cwd(), info.discoveryRoot);\n    this.testName = this.directory === path.join(relDiscoveryRoot, 'test') || this.directory === path.join(relDiscoveryRoot)\n      ? parsed.name\n      : path.join(path.relative(this.info.discoveryRoot, parsed.dir), parsed.name);\n\n    this.normalizedTestName = parsed.name;\n    this.snapshotDir = path.join(parsed.dir, `${parsed.base}.snapshot`);\n    this.temporaryOutputDir = path.join(parsed.dir, `${CDK_OUTDIR_PREFIX}.${parsed.base}.snapshot`);\n  }\n\n  /**\n   * Whether this test matches the user-given name\n   *\n   * We are very lenient here. A name matches if it matches:\n   *\n   * - The CWD-relative filename\n   * - The discovery root-relative filename\n   * - The suite name\n   * - The absolute filename\n   */\n  public matches(name: string) {\n    return [\n      this.fileName,\n      this.discoveryRelativeFileName,\n      this.testName,\n      this.absoluteFileName,\n    ].includes(name);\n  }\n}\n\n/**\n * Configuration options how integration test files are discovered\n */\nexport interface IntegrationTestsDiscoveryOptions {\n  /**\n   * If this is set to true then the list of tests\n   * provided will be excluded\n   *\n   * @default false\n   */\n  readonly exclude?: boolean;\n\n  /**\n   * If this is set to true, throw an error if any specified tests are not found\n   *\n   * @default false\n   */\n  readonly strict?: boolean;\n\n  /**\n   * List of tests to include (or exclude if `exclude=true`)\n   *\n   * @default - all matched files\n   */\n  readonly tests?: string[];\n\n  /**\n   * A map of of the app commands to run integration tests with,\n   * and the regex patterns matching the integration test files each app command.\n   *\n   * If the app command contains {filePath}, the test file names will be substituted at that place in the command for each run.\n   */\n  readonly testCases: {\n    [app: string]: string[];\n  };\n}\n\n/**\n * Returns the name of the Python executable for the current OS\n */\nfunction pythonExecutable() {\n  let python = 'python3';\n  if (process.platform === 'win32') {\n    python = 'python';\n  }\n  return python;\n}\n\n/**\n * Discover integration tests\n */\nexport class IntegrationTests {\n  constructor(private readonly directory: string) {\n  }\n\n  /**\n   * Get integration tests discovery options from CLI options\n   */\n  public async fromCliOptions(options: {\n    app?: string;\n    exclude?: boolean;\n    language?: string[];\n    testRegex?: string[];\n    tests?: string[];\n    strict?: boolean;\n  }): Promise<IntegTest[]> {\n    const baseOptions = {\n      tests: options.tests,\n      exclude: options.exclude,\n      strict: options.strict,\n    };\n\n    // Explicitly set both, app and test-regex\n    if (options.app && options.testRegex) {\n      return this.discover({\n        testCases: {\n          [options.app]: options.testRegex,\n        },\n        ...baseOptions,\n      });\n    }\n\n    // Use the selected presets\n    if (!options.app && !options.testRegex) {\n      // Only case with multiple languages, i.e. the only time we need to check the special case\n      const ignoreUncompiledTypeScript = options.language?.includes('javascript') && options.language?.includes('typescript');\n\n      return this.discover({\n        testCases: this.getLanguagePresets(options.language),\n        ...baseOptions,\n      }, ignoreUncompiledTypeScript);\n    }\n\n    // Only one of app or test-regex is set, with a single preset selected\n    // => override either app or test-regex\n    if (options.language?.length === 1) {\n      const [presetApp, presetTestRegex] = this.getLanguagePreset(options.language[0]);\n      return this.discover({\n        testCases: {\n          [options.app ?? presetApp]: options.testRegex ?? presetTestRegex,\n        },\n        ...baseOptions,\n      });\n    }\n\n    // Only one of app or test-regex is set, with multiple presets\n    // => impossible to resolve\n    const option = options.app ? '--app' : '--test-regex';\n    throw new Error(`Only a single \"--language\" can be used with \"${option}\". Alternatively provide both \"--app\" and \"--test-regex\" to fully customize the configuration.`);\n  }\n\n  /**\n   * Get the default configuration for a language\n   */\n  private getLanguagePreset(language: string) {\n    const languagePresets: {\n      [language: string]: [string, string[]];\n    } = {\n      javascript: ['node {filePath}', ['^integ\\\\..*\\\\.js$']],\n      typescript: ['node -r ts-node/register {filePath}', ['^integ\\\\.(?!.*\\\\.d\\\\.ts$).*\\\\.ts$']],\n      python: [`${pythonExecutable()} {filePath}`, ['^integ_.*\\\\.py$']],\n      go: ['go run {filePath}', ['^integ_.*\\\\.go$']],\n    };\n\n    return languagePresets[language];\n  }\n\n  /**\n   * Get the config for all selected languages\n   */\n  private getLanguagePresets(languages: string[] = []) {\n    return Object.fromEntries(\n      languages\n        .map(language => this.getLanguagePreset(language))\n        .filter(Boolean),\n    );\n  }\n\n  /**\n   * If the user provides a list of tests, these can either be a list of tests to include or a list of tests to exclude.\n   *\n   * - If it is a list of tests to include then we discover all available tests and check whether they have provided valid tests.\n   *   If they have provided a test name that we don't find, then we write out that error message.\n   * - If it is a list of tests to exclude, then we discover all available tests and filter out the tests that were provided by the user.\n   */\n  private filterTests(discoveredTests: IntegTest[], requestedTests?: string[], exclude?: boolean, strict?: boolean): IntegTest[] {\n    if (!requestedTests) {\n      return discoveredTests;\n    }\n\n    const allTests = discoveredTests.filter(t => {\n      const matches = requestedTests.some(pattern => t.matches(pattern));\n      return matches !== !!exclude; // Looks weird but is equal to (matches && !exclude) || (!matches && exclude)\n    });\n\n    // If not excluding, all patterns must have matched at least one test\n    if (!exclude) {\n      const unmatchedPatterns = requestedTests.filter(pattern => !discoveredTests.some(t => t.matches(pattern)));\n      for (const unmatched of unmatchedPatterns) {\n        process.stderr.write(`No such integ test: ${unmatched}\\n`);\n      }\n      if (unmatchedPatterns.length > 0) {\n        process.stderr.write(`Available tests: ${discoveredTests.map(t => t.discoveryRelativeFileName).join(' ')}\\n`);\n        if (strict) {\n          throw new Error(`Strict mode: ${unmatchedPatterns.length} test(s) not found: ${unmatchedPatterns.join(', ')}`);\n        }\n        return [];\n      }\n    }\n\n    return allTests;\n  }\n\n  /**\n   * Takes an optional list of tests to look for, otherwise\n   * it will look for all tests from the directory\n   *\n   * @param tests - Tests to include or exclude, undefined means include all tests.\n   * @param exclude - Whether the 'tests' list is inclusive or exclusive (inclusive by default).\n   */\n  private async discover(options: IntegrationTestsDiscoveryOptions, ignoreUncompiledTypeScript: boolean = false): Promise<IntegTest[]> {\n    const files = await this.readTree();\n\n    const testCases = Object.entries(options.testCases)\n      .flatMap(([appCommand, patterns]) => files\n        .filter(fileName => patterns.some((pattern) => {\n          const regex = new RegExp(pattern);\n          return regex.test(fileName) || regex.test(path.basename(fileName));\n        }))\n        .map(fileName => new IntegTest({\n          discoveryRoot: this.directory,\n          fileName,\n          appCommand,\n        })),\n      );\n\n    const discoveredTests = ignoreUncompiledTypeScript ? this.filterUncompiledTypeScript(testCases) : testCases;\n\n    return this.filterTests(discoveredTests, options.tests, options.exclude, options.strict);\n  }\n\n  private filterUncompiledTypeScript(testCases: IntegTest[]): IntegTest[] {\n    const jsTestCases = testCases.filter(t => t.fileName.endsWith('.js'));\n\n    return testCases\n      // Remove all TypeScript test cases (ending in .ts)\n      // for which a compiled version is present (same name, ending in .js)\n      .filter((tsCandidate) => {\n        if (!tsCandidate.fileName.endsWith('.ts')) {\n          return true;\n        }\n        return jsTestCases.findIndex(jsTest => jsTest.testName === tsCandidate.testName) === -1;\n      });\n  }\n\n  private async readTree(): Promise<string[]> {\n    const ret = new Array<string>();\n\n    async function recurse(dir: string) {\n      const files = await fs.readdir(dir);\n      for (const file of files) {\n        const fullPath = path.join(dir, file);\n        const statf = await fs.stat(fullPath);\n        if (statf.isFile()) {\n          ret.push(fullPath);\n        }\n        if (statf.isDirectory()) {\n          await recurse(fullPath);\n        }\n      }\n    }\n\n    await recurse(this.directory);\n    return ret;\n  }\n}\n"]}
@@ -4878,7 +4878,7 @@ var require_semver2 = __commonJS({
4878
4878
  // ../cloud-assembly-schema/cli-version.json
4879
4879
  var require_cli_version = __commonJS({
4880
4880
  "../cloud-assembly-schema/cli-version.json"(exports2, module2) {
4881
- module2.exports = { version: "2.1031.1" };
4881
+ module2.exports = { version: "2.1031.2" };
4882
4882
  }
4883
4883
  });
4884
4884
 
@@ -322994,9 +322994,9 @@ var require_drawRow = __commonJS({
322994
322994
  }
322995
322995
  });
322996
322996
 
322997
- // ../../../node_modules/table/node_modules/ajv/dist/runtime/equal.js
322997
+ // ../../../node_modules/ajv/dist/runtime/equal.js
322998
322998
  var require_equal = __commonJS({
322999
- "../../../node_modules/table/node_modules/ajv/dist/runtime/equal.js"(exports2) {
322999
+ "../../../node_modules/ajv/dist/runtime/equal.js"(exports2) {
323000
323000
  "use strict";
323001
323001
  Object.defineProperty(exports2, "__esModule", { value: true });
323002
323002
  var equal = require_fast_deep_equal();
@@ -333695,6 +333695,15 @@ var init_digest = __esm({
333695
333695
  }
333696
333696
  });
333697
333697
 
333698
+ // ../toolkit-lib/lib/util/concurrency.ts
333699
+ var pLimit3;
333700
+ var init_concurrency = __esm({
333701
+ "../toolkit-lib/lib/util/concurrency.ts"() {
333702
+ "use strict";
333703
+ pLimit3 = require_p_limit();
333704
+ }
333705
+ });
333706
+
333698
333707
  // ../toolkit-lib/lib/api/refactoring/exclude.ts
333699
333708
  var import_cloud_assembly_schema6;
333700
333709
  var init_exclude = __esm({
@@ -334005,7 +334014,8 @@ async function getDeployedStacks(sdkProvider, environment) {
334005
334014
  template
334006
334015
  };
334007
334016
  }, "normalize");
334008
- return Promise.all(summaries.map(normalize3));
334017
+ const limit = pLimit3(20);
334018
+ return Promise.all(summaries.map((s3) => limit(() => normalize3(s3))));
334009
334019
  }
334010
334020
  function formatEnvironmentSectionHeader2(environment) {
334011
334021
  const env = `aws://${environment.account}/${environment.region}`;
@@ -334059,6 +334069,7 @@ var init_refactoring = __esm({
334059
334069
  init_cloudformation3();
334060
334070
  init_digest();
334061
334071
  init_toolkit_error();
334072
+ init_concurrency();
334062
334073
  init_exclude();
334063
334074
  init_context3();
334064
334075
  __name(getDeployedStacks, "getDeployedStacks");
@@ -336834,15 +336845,6 @@ var init_work_graph2 = __esm({
336834
336845
  }
336835
336846
  });
336836
336847
 
336837
- // ../toolkit-lib/lib/util/concurrency.ts
336838
- var pLimit3;
336839
- var init_concurrency = __esm({
336840
- "../toolkit-lib/lib/util/concurrency.ts"() {
336841
- "use strict";
336842
- pLimit3 = require_p_limit();
336843
- }
336844
- });
336845
-
336846
336848
  // ../toolkit-lib/lib/util/promises.ts
336847
336849
  function promiseWithResolvers2() {
336848
336850
  let resolve10, reject;
package/package.json CHANGED
@@ -62,16 +62,16 @@
62
62
  "node-backpack": "^1.1.7",
63
63
  "prettier": "^2.8",
64
64
  "ts-jest": "^29.4.4",
65
- "typescript": "5.8",
65
+ "typescript": "5.9",
66
66
  "@aws-cdk/aws-service-spec": "^0.1.98",
67
67
  "@aws-cdk/cdk-assets-lib": "1.0.4",
68
68
  "@aws-cdk/cdk-cli-wrapper": "0.0.0",
69
- "@aws-cdk/cloud-assembly-schema": ">=48.17.0",
69
+ "@aws-cdk/cloud-assembly-schema": ">=48.18.0",
70
70
  "@aws-cdk/cloudformation-diff": "2.184.1",
71
71
  "@aws-cdk/cx-api": "^2.215.0",
72
- "@aws-cdk/toolkit-lib": "1.10.2",
72
+ "@aws-cdk/toolkit-lib": "1.10.3",
73
73
  "@aws-sdk/client-cloudformation": "^3",
74
- "aws-cdk": "2.1031.1",
74
+ "aws-cdk": "2.1031.2",
75
75
  "chalk": "^4",
76
76
  "chokidar": "^3",
77
77
  "fs-extra": "^9",
@@ -80,7 +80,7 @@
80
80
  },
81
81
  "dependencies": {
82
82
  "@aws-cdk/aws-service-spec": "0.1.98",
83
- "aws-cdk": "2.1031.1"
83
+ "aws-cdk": "2.1031.2"
84
84
  },
85
85
  "keywords": [
86
86
  "aws",
@@ -95,7 +95,7 @@
95
95
  "publishConfig": {
96
96
  "access": "public"
97
97
  },
98
- "version": "2.191.2",
98
+ "version": "2.192.0",
99
99
  "types": "lib/index.d.ts",
100
100
  "//": "~~ Generated by projen. To modify, edit .projenrc.js and run \"npx projen\".",
101
101
  "optionalDependencies": {