@pipeline-builder/pipeline-manager 3.3.5 → 3.3.7

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/dist/cli.d.ts.map CHANGED
@@ -1 +1 @@
1
- {"version":3,"file":"cli.d.ts","sourceRoot":"","sources":["../src/cli.ts"],"names":[],"mappings":";AAgCA;;GAEG;AACH,UAAU,UAAU;IAClB;;;OAGG;IACH,UAAU,CAAC,EAAE,OAAO,CAAC;IAErB;;;OAGG;IACH,aAAa,CAAC,EAAE,OAAO,CAAC;IAExB;;;OAGG;IACH,KAAK,CAAC,EAAE,OAAO,CAAC;IAEhB;;;OAGG;IACH,OAAO,CAAC,EAAE,OAAO,CAAC;IAElB;;;OAGG;IACH,KAAK,CAAC,EAAE,OAAO,CAAC;IAEhB;;;OAGG;IACH,OAAO,CAAC,EAAE,OAAO,CAAC;CACnB;AA6ND;;;;GAIG;AACH,wBAAgB,aAAa,CAAC,OAAO,GAAE,UAAe,GAAG,IAAI,CAyD5D;AAiDD;;;;;;;;;;;;;;;;GAgBG;AACH,wBAAgB,IAAI,CAAC,OAAO,GAAE,UAAe,GAAG,IAAI,CA0BnD"}
1
+ {"version":3,"file":"cli.d.ts","sourceRoot":"","sources":["../src/cli.ts"],"names":[],"mappings":";AAkCA;;GAEG;AACH,UAAU,UAAU;IAClB;;;OAGG;IACH,UAAU,CAAC,EAAE,OAAO,CAAC;IAErB;;;OAGG;IACH,aAAa,CAAC,EAAE,OAAO,CAAC;IAExB;;;OAGG;IACH,KAAK,CAAC,EAAE,OAAO,CAAC;IAEhB;;;OAGG;IACH,OAAO,CAAC,EAAE,OAAO,CAAC;IAElB;;;OAGG;IACH,KAAK,CAAC,EAAE,OAAO,CAAC;IAEhB;;;OAGG;IACH,OAAO,CAAC,EAAE,OAAO,CAAC;CACnB;AAkOD;;;;GAIG;AACH,wBAAgB,aAAa,CAAC,OAAO,GAAE,UAAe,GAAG,IAAI,CAyD5D;AAiDD;;;;;;;;;;;;;;;;GAgBG;AACH,wBAAgB,IAAI,CAAC,OAAO,GAAE,UAAe,GAAG,IAAI,CA0BnD"}
package/dist/cli.js CHANGED
@@ -6,6 +6,8 @@ Object.defineProperty(exports, "__esModule", { value: true });
6
6
  exports.initializeCli = initializeCli;
7
7
  exports.main = main;
8
8
  const commander_1 = require("commander");
9
+ const audit_stacks_1 = require("./commands/audit-stacks");
10
+ const audit_tokens_1 = require("./commands/audit-tokens");
9
11
  const bootstrap_1 = require("./commands/bootstrap");
10
12
  const create_pipeline_1 = require("./commands/create-pipeline");
11
13
  const deploy_1 = require("./commands/deploy");
@@ -135,6 +137,10 @@ Examples:
135
137
  (0, setup_events_1.setupEvents)(commander_1.program); // Deploy EventBridge event ingestion infrastructure
136
138
  (0, bootstrap_1.bootstrap)(commander_1.program); // Bootstrap CDK toolkit stack
137
139
  (0, deploy_1.deploy)(commander_1.program); // Deploy pipeline with CDK (--app prints boilerplate path)
140
+ // Operator audit commands (cron-friendly: exit 1 on findings)
141
+ (0, output_utils_1.printDebug)('Registering audit commands');
142
+ (0, audit_tokens_1.auditTokens)(commander_1.program); // Scan Secrets Manager for expiring platform tokens
143
+ (0, audit_stacks_1.auditStacks)(commander_1.program); // Diff CFN stacks vs pipeline_registry to find drift
138
144
  (0, synth_1.synth)(commander_1.program); // Run CDK synthesis
139
145
  (0, validate_templates_1.validateTemplatesCommand)(commander_1.program); // Validate {{ ... }} templates in a pipeline or plugin
140
146
  // Shell completions
@@ -371,4 +377,4 @@ if (require.main === module) {
371
377
  };
372
378
  main(options);
373
379
  }
374
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cli.js","sourceRoot":"","sources":["../src/cli.ts"],"names":[],"mappings":";;AACA,+CAA+C;AAC/C,sCAAsC;;AAuStC,sCAyDC;AAkED,oBA0BC;AA1bD,yCAAoC;AACpC,oDAAiD;AACjD,gEAA4D;AAC5D,8CAA2C;AAC3C,0DAAsD;AACtD,sDAAkD;AAClD,8DAA0D;AAC1D,0DAAsD;AACtD,4CAAyC;AACzC,0DAAsD;AACtD,8CAA2C;AAC3C,wDAAoD;AACpD,4CAAyC;AACzC,4DAAwD;AACxD,sEAAyE;AACzE,gDAA6C;AAC7C,0DAOgC;AAChC,2CAAoD;AACpD,yDAAiE;AACjE,uDAAqG;AA2CrG;;GAEG;AACH,SAAS,gBAAgB;IACvB,MAAM,QAAQ,GAAa,EAAE,CAAC;IAE9B,2CAA2C;IAC3C,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,wBAAQ,CAAC,cAAc,CAAC,EAAE,CAAC;QAC1C,QAAQ,CAAC,IAAI,CAAC,gDAAgD,CAAC,CAAC;QAChE,QAAQ,CAAC,IAAI,CAAC,6CAA6C,CAAC,CAAC;IAC/D,CAAC;IAED,qBAAqB;IACrB,MAAM,WAAW,GAAG,OAAO,CAAC,OAAO,CAAC;IACpC,MAAM,YAAY,GAAG,QAAQ,CAAC,WAAW,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,IAAI,GAAG,EAAE,EAAE,CAAC,CAAC;IAC7E,IAAI,YAAY,GAAG,EAAE,EAAE,CAAC;QACtB,QAAQ,CAAC,IAAI,CAAC,WAAW,WAAW,qCAAqC,CAAC,CAAC;IAC7E,CAAC;IAED,mBAAmB;IACnB,IAAI,QAAQ,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;QACxB,IAAA,2BAAY,EAAC,sBAAsB,CAAC,CAAC;QACrC,QAAQ,CAAC,OAAO,CAAC,OAAO,CAAC,EAAE;YACzB,IAAA,2BAAY,EAAC,OAAO,CAAC,CAAC;QACxB,CAAC,CAAC,CAAC;QACH,OAAO,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC;IAClB,CAAC;AACH,CAAC;AAED;;GAEG;AACH,SAAS,kBAAkB,CAAC,OAAmB;IAC7C,IAAI,OAAO,CAAC,KAAK;QAAE,OAAO;IAE1B,MAAM,WAAW,GAAG,IAAA,mCAAmB,GAAE,CAAC;IAE1C,IAAI,OAAO,CAAC,KAAK,IAAI,OAAO,CAAC,OAAO,EAAE,CAAC;QACrC,IAAA,yBAAU,EAAC,mBAAmB,EAAE;YAC9B,IAAI,EAAE,wBAAQ;YACd,OAAO,EAAE,2BAAW;YACpB,WAAW;YACX,IAAI,EAAE,OAAO,CAAC,OAAO;YACrB,QAAQ,EAAE,OAAO,CAAC,QAAQ;YAC1B,IAAI,EAAE,OAAO,CAAC,IAAI;YAClB,GAAG,EAAE,OAAO,CAAC,GAAG,EAAE;YAClB,GAAG,EAAE;gBACH,KAAK,EAAE,OAAO,CAAC,GAAG,CAAC,KAAK;gBACxB,KAAK,EAAE,OAAO,CAAC,GAAG,CAAC,wBAAQ,CAAC,cAAc,CAAC,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC,CAAC,SAAS;gBAC/D,GAAG,EAAE,OAAO,CAAC,GAAG,CAAC,wBAAQ,CAAC,iBAAiB,CAAC,IAAI,SAAS;aAC1D;SACF,CAAC,CAAC;IACL,CAAC;SAAM,CAAC;QACN,IAAA,yBAAU,EAAC,cAAc,EAAE;YACzB,OAAO,EAAE,2BAAW;YACpB,WAAW;SACZ,CAAC,CAAC;IACL,CAAC;AACH,CAAC;AAED;;GAEG;AACH,SAAS,gBAAgB;IACvB,IAAA,yBAAU,EAAC,sBAAsB,CAAC,CAAC;IAEnC,oBAAoB;IACpB,mBAAO;SACJ,IAAI,CAAC,wBAAQ,CAAC;SACd,WAAW,CAAC,+BAAe,CAAC;SAC5B,OAAO,CAAC,2BAAW,EAAE,eAAe,EAAE,kBAAkB,CAAC;SACzD,MAAM,CAAC,SAAS,EAAE,uCAAuC,EAAE,KAAK,CAAC;SACjE,MAAM,CAAC,WAAW,EAAE,2BAA2B,EAAE,KAAK,CAAC;SACvD,MAAM,CAAC,SAAS,EAAE,8BAA8B,EAAE,KAAK,CAAC;SACxD,MAAM,CAAC,YAAY,EAAE,wBAAwB,EAAE,KAAK,CAAC;SACrD,WAAW,CAAC,OAAO,EAAE;;IAEtB,wBAAQ,CAAC,cAAc;IACvB,wBAAQ,CAAC,iBAAiB;IAC1B,wBAAQ,CAAC,eAAe;IACxB,wBAAQ,CAAC,uBAAuB;IAChC,wBAAQ,CAAC,KAAK;;;MAGZ,wBAAQ;MACR,wBAAQ;MACR,wBAAQ;MACR,wBAAQ;MACR,wBAAQ;MACR,wBAAQ;;CAEb,CAAC,CAAC;IAED,qCAAqC;IACrC,IAAA,iBAAO,EAAC,mBAAO,CAAC,CAAC;IAEjB,0BAA0B;IAC1B,IAAA,yBAAU,EAAC,qCAAqC,CAAC,CAAC;IAClD,IAAA,aAAK,EAAC,mBAAO,CAAC,CAAC,CAAC,4DAA4D;IAE5E,iBAAiB;IACjB,IAAA,yBAAU,EAAC,4BAA4B,CAAC,CAAC;IACzC,IAAA,sBAAS,EAAC,mBAAO,CAAC,CAAC,CAAC,sBAAsB;IAC1C,IAAA,0BAAW,EAAC,mBAAO,CAAC,CAAC,CAAC,gCAAgC;IACtD,IAAA,0BAAW,EAAC,mBAAO,CAAC,CAAC,CAAC,wBAAwB;IAC9C,IAAA,8BAAa,EAAC,mBAAO,CAAC,CAAC,CAAC,kCAAkC;IAE1D,yBAAyB;IACzB,IAAA,yBAAU,EAAC,oCAAoC,CAAC,CAAC;IACjD,IAAA,gCAAc,EAAC,mBAAO,CAAC,CAAC,CAAC,gCAAgC;IACzD,IAAA,4BAAY,EAAC,mBAAO,CAAC,CAAC,CAAC,2BAA2B;IAElD,iBAAiB;IACjB,IAAA,yBAAU,EAAC,4BAA4B,CAAC,CAAC;IACzC,IAAA,eAAM,EAAC,mBAAO,CAAC,CAAC,CAAC,2CAA2C;IAE5D,sBAAsB;IACtB,IAAA,yBAAU,EAAC,iCAAiC,CAAC,CAAC;IAC9C,IAAA,wBAAU,EAAC,mBAAO,CAAC,CAAC,CAAC,kDAAkD;IACvE,IAAA,0BAAW,EAAC,mBAAO,CAAC,CAAC,CAAC,oDAAoD;IAC1E,IAAA,qBAAS,EAAC,mBAAO,CAAC,CAAC,CAAC,8BAA8B;IAClD,IAAA,eAAM,EAAC,mBAAO,CAAC,CAAC,CAAC,2DAA2D;IAC5E,IAAA,aAAK,EAAC,mBAAO,CAAC,CAAC,CAAC,oBAAoB;IACpC,IAAA,6CAAwB,EAAC,mBAAO,CAAC,CAAC,CAAC,uDAAuD;IAE1F,oBAAoB;IACpB,IAAA,yBAAU,EAAC,iCAAiC,CAAC,CAAC;IAC9C,mBAAO;SACJ,OAAO,CAAC,aAAa,CAAC;SACtB,WAAW,CAAC,8CAA8C,CAAC;SAC3D,QAAQ,CAAC,SAAS,EAAE,gCAAgC,CAAC;SACrD,MAAM,CAAC,CAAC,KAAa,EAAE,EAAE;QACxB,QAAQ,KAAK,EAAE,CAAC;YACd,KAAK,MAAM;gBACT,OAAO,CAAC,GAAG,CAAC;;;;;;2DAMqC,CAAC,CAAC;gBACnD,MAAM;YACR,KAAK,KAAK;gBACR,OAAO,CAAC,GAAG,CAAC;;;;;2CAKqB,CAAC,CAAC;gBACnC,MAAM;YACR,KAAK,MAAM;gBACT,OAAO,CAAC,GAAG,CAAC;4KACsJ,CAAC,CAAC;gBACpK,MAAM;YACR;gBACE,OAAO,CAAC,KAAK,CAAC,kBAAkB,KAAK,2BAA2B,CAAC,CAAC;gBAClE,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;QACpB,CAAC;IACH,CAAC,CAAC,CAAC;IAEL,IAAA,yBAAU,EAAC,sCAAsC,CAAC,CAAC;AACrD,CAAC;AAED;;GAEG;AACH,SAAS,kBAAkB;IACzB,6BAA6B;IAC7B,OAAO,CAAC,EAAE,CAAC,mBAAmB,EAAE,CAAC,KAAY,EAAE,EAAE;QAC/C,OAAO,CAAC,KAAK,CAAC,EAAE,CAAC,CAAC,CAAC,aAAa;QAChC,IAAA,yBAAU,EAAC,oBAAoB,EAAE;YAC/B,KAAK,EAAE,KAAK,CAAC,OAAO;YACpB,IAAI,EAAE,KAAK,CAAC,IAAI;SACjB,CAAC,CAAC;QAEH,IAAA,2BAAW,EAAC,KAAK,EAAE,2BAAW,CAAC,OAAO,EAAE;YACtC,KAAK,EAAE,IAAA,2BAAW,EAAC,mBAAO,CAAC,IAAI,EAAE,CAAC;YAClC,IAAI,EAAE,IAAI;YACV,OAAO,EAAE;gBACP,IAAI,EAAE,mBAAmB;aAC1B;SACF,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;IAEH,sCAAsC;IACtC,OAAO,CAAC,EAAE,CAAC,oBAAoB,EAAE,CAAC,MAAe,EAAE,EAAE;QACnD,OAAO,CAAC,KAAK,CAAC,EAAE,CAAC,CAAC,CAAC,aAAa;QAChC,IAAA,yBAAU,EAAC,6BAA6B,EAAE;YACxC,MAAM,EAAE,MAAM,YAAY,KAAK,CAAC,CAAC,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC,CAAC,MAAM,CAAC,MAAM,CAAC;SAClE,CAAC,CAAC;QAEH,MAAM,KAAK,GAAG,MAAM,YAAY,KAAK,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,IAAI,KAAK,CAAC,MAAM,CAAC,MAAM,CAAC,CAAC,CAAC;QAE3E,IAAA,2BAAW,EAAC,KAAK,EAAE,2BAAW,CAAC,OAAO,EAAE;YACtC,KAAK,EAAE,IAAA,2BAAW,EAAC,mBAAO,CAAC,IAAI,EAAE,CAAC;YAClC,IAAI,EAAE,IAAI;YACV,OAAO,EAAE;gBACP,IAAI,EAAE,oBAAoB;aAC3B;SACF,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;IAEH,yBAAyB;IACzB,OAAO,CAAC,EAAE,CAAC,QAAQ,EAAE,GAAG,EAAE;QACxB,OAAO,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC,CAAC,aAAa;QAC9B,IAAA,2BAAY,EAAC,sCAAsC,CAAC,CAAC;QACrD,OAAO,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC,CAAC,aAAa;QAC9B,OAAO,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC,CAAC,gCAAgC;IACrD,CAAC,CAAC,CAAC;IAEH,iBAAiB;IACjB,OAAO,CAAC,EAAE,CAAC,SAAS,EAAE,GAAG,EAAE;QACzB,OAAO,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC,CAAC,aAAa;QAC9B,IAAA,2BAAY,EAAC,8BAA8B,CAAC,CAAC;QAC7C,OAAO,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC,CAAC,aAAa;QAC9B,OAAO,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC,CAAC,iCAAiC;IACtD,CAAC,CAAC,CAAC;AACL,CAAC;AAED;;;;GAIG;AACH,SAAgB,aAAa,CAAC,UAAsB,EAAE;IACpD,MAAM,EACJ,UAAU,GAAG,IAAI,EACjB,aAAa,GAAG,KAAK,EACrB,KAAK,GAAG,KAAK,EACb,OAAO,GAAG,KAAK,EACf,KAAK,GAAG,KAAK,EACb,OAAO,GAAG,KAAK,GAChB,GAAG,OAAO,CAAC;IAEZ,sBAAsB;IACtB,IAAI,OAAO,EAAE,CAAC;QACZ,OAAO,CAAC,GAAG,CAAC,QAAQ,GAAG,GAAG,CAAC;IAC7B,CAAC;IAED,iBAAiB;IACjB,IAAI,UAAU,IAAI,CAAC,KAAK,EAAE,CAAC;QACzB,IAAI,aAAa,EAAE,CAAC;YAClB,IAAA,mBAAU,GAAE,CAAC;QACf,CAAC;aAAM,CAAC;YACN,IAAA,eAAM,EAAC;gBACL,OAAO,EAAE,KAAK;gBACd,eAAe,EAAE,IAAI;gBACrB,aAAa,EAAE,KAAK,IAAI,OAAO;gBAC/B,eAAe,EAAE,KAAK;aACvB,CAAC,CAAC;QACL,CAAC;IACH,CAAC;IAED,uBAAuB;IACvB,kBAAkB,CAAC,EAAE,KAAK,EAAE,OAAO,EAAE,KAAK,EAAE,CAAC,CAAC;IAE9C,oBAAoB;IACpB,IAAI,CAAC,KAAK,EAAE,CAAC;QACX,gBAAgB,EAAE,CAAC;IACrB,CAAC;IAED,uBAAuB;IACvB,kBAAkB,EAAE,CAAC;IAErB,oBAAoB;IACpB,IAAI,CAAC;QACH,gBAAgB,EAAE,CAAC;QACnB,IAAA,yBAAU,EAAC,6BAA6B,CAAC,CAAC;IAC5C,CAAC;IAAC,OAAO,KAAK,EAAE,CAAC;QACf,IAAA,yBAAU,EAAC,2BAA2B,EAAE;YACtC,KAAK,EAAE,KAAK,YAAY,KAAK,CAAC,CAAC,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC;SAC9D,CAAC,CAAC;QAEH,IAAA,2BAAW,EAAC,KAAK,EAAE,2BAAW,CAAC,aAAa,EAAE;YAC5C,KAAK,EAAE,KAAK,IAAI,IAAA,2BAAW,GAAE;YAC7B,IAAI,EAAE,IAAI;YACV,OAAO,EAAE;gBACP,KAAK,EAAE,gBAAgB;aACxB;SACF,CAAC,CAAC;IACL,CAAC;AACH,CAAC;AAED;;GAEG;AACH,SAAS,cAAc;IACrB,IAAI,CAAC;QACH,mBAAO,CAAC,KAAK,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC;QAE5B,mCAAmC;QACnC,MAAM,IAAI,GAAG,OAAO,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC;QACnC,IAAI,IAAI,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;YACtB,IAAA,wBAAS,EAAC,wCAAwC,CAAC,CAAC;YACpD,OAAO,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC,CAAC,aAAa;YAC9B,mBAAO,CAAC,UAAU,EAAE,CAAC;YACrB,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;QAClB,CAAC;QAED,0BAA0B;QAC1B,MAAM,OAAO,GAAG,IAAI,CAAC,CAAC,CAAC,CAAC;QACxB,IAAI,OAAO,IAAI,CAAC,OAAO,CAAC,UAAU,CAAC,GAAG,CAAC,EAAE,CAAC;YACxC,MAAM,aAAa,GAAG,mBAAO,CAAC,QAAQ,CAAC,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,CAAC,IAAI,EAAE,KAAK,OAAO,CAAC,CAAC;YAE3E,IAAI,CAAC,aAAa,EAAE,CAAC;gBACnB,OAAO,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC,CAAC,aAAa;gBAC9B,IAAA,yBAAU,EAAC,oBAAoB,OAAO,EAAE,CAAC,CAAC;gBAC1C,OAAO,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC,CAAC,aAAa;gBAC9B,IAAA,wBAAS,EAAC,qBAAqB,CAAC,CAAC;gBACjC,mBAAO,CAAC,QAAQ,CAAC,OAAO,CAAC,GAAG,CAAC,EAAE;oBAC7B,OAAO,CAAC,GAAG,CAAC,OAAO,GAAG,CAAC,IAAI,EAAE,MAAM,GAAG,CAAC,WAAW,EAAE,EAAE,CAAC,CAAC;gBAC1D,CAAC,CAAC,CAAC;gBACH,OAAO,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC,CAAC,aAAa;gBAC9B,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;YAClB,CAAC;QACH,CAAC;QAED,IAAA,yBAAU,EAAC,4CAA4C,CAAC,CAAC;IAC3D,CAAC;IAAC,OAAO,KAAK,EAAE,CAAC;QACf,IAAA,2BAAW,EAAC,KAAK,EAAE,2BAAW,CAAC,OAAO,EAAE;YACtC,KAAK,EAAE,IAAA,2BAAW,EAAC,mBAAO,CAAC,IAAI,EAAE,CAAC;YAClC,IAAI,EAAE,IAAI;YACV,OAAO,EAAE;gBACP,KAAK,EAAE,kBAAkB;gBACzB,IAAI,EAAE,OAAO,CAAC,IAAI;aACnB;SACF,CAAC,CAAC;IACL,CAAC;AACH,CAAC;AAED;;;;;;;;;;;;;;;;GAgBG;AACH,SAAgB,IAAI,CAAC,UAAsB,EAAE;IAC3C,IAAI,CAAC;QACH,iBAAiB;QACjB,aAAa,CAAC,OAAO,CAAC,CAAC;QAEvB,sCAAsC;QACtC,cAAc,EAAE,CAAC;QAEjB,mDAAmD;QACnD,IAAA,yBAAU,EAAC,sCAAsC,CAAC,CAAC;IACrD,CAAC;IAAC,OAAO,KAAK,EAAE,CAAC;QACf,gCAAgC;QAChC,OAAO,CAAC,KAAK,CAAC,EAAE,CAAC,CAAC,CAAC,aAAa;QAChC,IAAA,yBAAU,EAAC,iBAAiB,EAAE;YAC5B,KAAK,EAAE,KAAK,YAAY,KAAK,CAAC,CAAC,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC;SAC9D,CAAC,CAAC;QAEH,IAAA,2BAAW,EAAC,KAAK,EAAE,2BAAW,CAAC,OAAO,EAAE;YACtC,KAAK,EAAE,IAAA,2BAAW,EAAC,OAAO,CAAC;YAC3B,IAAI,EAAE,IAAI;YACV,OAAO,EAAE;gBACP,KAAK,EAAE,MAAM;gBACb,OAAO;aACR;SACF,CAAC,CAAC;IACL,CAAC;AACH,CAAC;AAED;;GAEG;AACH,IAAI,OAAO,CAAC,IAAI,KAAK,MAAM,EAAE,CAAC;IAC5B,qDAAqD;IACrD,MAAM,OAAO,GAAe;QAC1B,KAAK,EAAE,OAAO,CAAC,GAAG,CAAC,KAAK,KAAK,MAAM;QACnC,KAAK,EAAE,OAAO,CAAC,IAAI,CAAC,QAAQ,CAAC,SAAS,CAAC;QACvC,OAAO,EAAE,OAAO,CAAC,IAAI,CAAC,QAAQ,CAAC,WAAW,CAAC;QAC3C,OAAO,EAAE,OAAO,CAAC,IAAI,CAAC,QAAQ,CAAC,YAAY,CAAC,IAAI,OAAO,CAAC,GAAG,CAAC,QAAQ,KAAK,GAAG;KAC7E,CAAC;IAEF,IAAI,CAAC,OAAO,CAAC,CAAC;AAChB,CAAC","sourcesContent":["#!/usr/bin/env node\n// Copyright 2026 Pipeline Builder Contributors\n// SPDX-License-Identifier: Apache-2.0\n\nimport { program } from 'commander';\nimport { bootstrap } from './commands/bootstrap';\nimport { createPipeline } from './commands/create-pipeline';\nimport { deploy } from './commands/deploy';\nimport { getPipeline } from './commands/get-pipeline';\nimport { getPlugin } from './commands/get-plugin';\nimport { listPipelines } from './commands/list-pipelines';\nimport { listPlugins } from './commands/list-plugins';\nimport { login } from './commands/login';\nimport { setupEvents } from './commands/setup-events';\nimport { status } from './commands/status';\nimport { storeToken } from './commands/store-token';\nimport { synth } from './commands/synth';\nimport { uploadPlugin } from './commands/upload-plugin';\nimport { validateTemplatesCommand } from './commands/validate-templates';\nimport { version } from './commands/version';\nimport {\n  APP_NAME,\n  APP_DESCRIPTION,\n  APP_VERSION,\n  ENV_VARS,\n  isDebugMode,\n  generateExecutionId,\n} from './config/cli.constants';\nimport { banner, miniBanner } from './utils/banner';\nimport { ERROR_CODES, handleError } from './utils/error-handler';\nimport { printInfo, printError, printWarning, printDebug, printSection } from './utils/output-utils';\n\n/**\n * CLI initialization options\n */\ninterface CliOptions {\n  /**\n   * Show banner on startup\n   * @default true\n   */\n  showBanner?: boolean;\n\n  /**\n   * Minimal banner (no ASCII art)\n   * @default false\n   */\n  minimalBanner?: boolean;\n\n  /**\n   * Debug mode\n   * @default false\n   */\n  debug?: boolean;\n\n  /**\n   * Verbose output\n   * @default false\n   */\n  verbose?: boolean;\n\n  /**\n   * Quiet mode (minimal output)\n   * @default false\n   */\n  quiet?: boolean;\n\n  /**\n   * No color output\n   * @default false\n   */\n  noColor?: boolean;\n}\n\n/**\n * Check environment and display warnings\n */\nfunction checkEnvironment(): void {\n  const warnings: string[] = [];\n\n  // Check for required environment variables\n  if (!process.env[ENV_VARS.PLATFORM_TOKEN]) {\n    warnings.push('PLATFORM_TOKEN environment variable is not set');\n    warnings.push('Authentication will fail for API operations');\n  }\n\n  // Check Node version\n  const nodeVersion = process.version;\n  const majorVersion = parseInt(nodeVersion.slice(1).split('.')[0] || '0', 10);\n  if (majorVersion < 18) {\n    warnings.push(`Node.js ${nodeVersion} detected - version 18+ recommended`);\n  }\n\n  // Display warnings\n  if (warnings.length > 0) {\n    printSection('Environment Warnings');\n    warnings.forEach(warning => {\n      printWarning(warning);\n    });\n    console.log('');\n  }\n}\n\n/**\n * Display startup information\n */\nfunction displayStartupInfo(options: CliOptions): void {\n  if (options.quiet) return;\n\n  const executionId = generateExecutionId();\n\n  if (options.debug || options.verbose) {\n    printDebug('CLI Configuration', {\n      name: APP_NAME,\n      version: APP_VERSION,\n      executionId,\n      node: process.version,\n      platform: process.platform,\n      arch: process.arch,\n      cwd: process.cwd(),\n      env: {\n        debug: process.env.DEBUG,\n        token: process.env[ENV_VARS.PLATFORM_TOKEN] ? 'set' : 'not set',\n        url: process.env[ENV_VARS.PLATFORM_BASE_URL] || 'default',\n      },\n    });\n  } else {\n    printDebug('Starting CLI', {\n      version: APP_VERSION,\n      executionId,\n    });\n  }\n}\n\n/**\n * Register all CLI commands\n */\nfunction registerCommands(): void {\n  printDebug('Registering commands');\n\n  // Configure program\n  program\n    .name(APP_NAME)\n    .description(APP_DESCRIPTION)\n    .version(APP_VERSION, '-v, --version', 'Show CLI version')\n    .option('--debug', 'Enable debug output with stack traces', false)\n    .option('--verbose', 'Show detailed information', false)\n    .option('--quiet', 'Minimal output (errors only)', false)\n    .option('--no-color', 'Disable colored output', false)\n    .addHelpText('after', `\nEnvironment Variables:\n  ${ENV_VARS.PLATFORM_TOKEN}              Authentication token (required)\n  ${ENV_VARS.PLATFORM_BASE_URL}                 API base URL (optional)\n  ${ENV_VARS.CLI_CONFIG_PATH}              Config file path (optional)\n  ${ENV_VARS.TLS_REJECT_UNAUTHORIZED}      Disable SSL verification if '0'\n  ${ENV_VARS.DEBUG}                        Enable debug mode if 'true'\n\nExamples:\n  $ ${APP_NAME} version\n  $ ${APP_NAME} list-pipelines --project my-app\n  $ ${APP_NAME} get-pipeline --id pipe-123 --format json\n  $ ${APP_NAME} store-token --days 30 --region us-east-1\n  $ ${APP_NAME} bootstrap --account 123456789012 --region us-east-1\n  $ ${APP_NAME} deploy --id pipe-123 --profile production\n\n`);\n\n  // Version command (special handling)\n  version(program);\n\n  // Authentication commands\n  printDebug('Registering authentication commands');\n  login(program); // Login and obtain PLATFORM_TOKEN (also supports --refresh)\n\n  // Query commands\n  printDebug('Registering query commands');\n  getPlugin(program); // Single plugin by ID\n  listPlugins(program); // Multiple plugins with filters\n  getPipeline(program); // Single pipeline by ID\n  listPipelines(program); // Multiple pipelines with filters\n\n  // Create/Upload commands\n  printDebug('Registering create/upload commands');\n  createPipeline(program); // Create pipeline configuration\n  uploadPlugin(program); // Upload and deploy plugin\n\n  // Status command\n  printDebug('Registering status command');\n  status(program); // Show environment and connectivity status\n\n  // Deployment commands\n  printDebug('Registering deployment commands');\n  storeToken(program); // Generate JWT token and store in Secrets Manager\n  setupEvents(program); // Deploy EventBridge event ingestion infrastructure\n  bootstrap(program); // Bootstrap CDK toolkit stack\n  deploy(program); // Deploy pipeline with CDK (--app prints boilerplate path)\n  synth(program); // Run CDK synthesis\n  validateTemplatesCommand(program); // Validate {{ ... }} templates in a pipeline or plugin\n\n  // Shell completions\n  printDebug('Registering completions command');\n  program\n    .command('completions')\n    .description('Generate shell completions (bash, zsh, fish)')\n    .argument('<shell>', 'Shell type: bash, zsh, or fish')\n    .action((shell: string) => {\n      switch (shell) {\n        case 'bash':\n          console.log(`# pipeline-manager bash completions\n_pipeline_manager_completions() {\n  local cur=\"\\${COMP_WORDS[COMP_CWORD]}\"\n  local commands=\"login deploy synth upload-plugin create-pipeline status version bootstrap setup-events store-token completions\"\n  COMPREPLY=($(compgen -W \"\\${commands}\" -- \"\\${cur}\"))\n}\ncomplete -F _pipeline_manager_completions pipeline-manager`);\n          break;\n        case 'zsh':\n          console.log(`# pipeline-manager zsh completions\n_pipeline_manager() {\n  local commands=(login deploy synth upload-plugin create-pipeline status version bootstrap setup-events store-token completions)\n  _describe 'command' commands\n}\ncompdef _pipeline_manager pipeline-manager`);\n          break;\n        case 'fish':\n          console.log(`# pipeline-manager fish completions\ncomplete -c pipeline-manager -n '__fish_use_subcommand' -a 'login deploy synth upload-plugin create-pipeline status version bootstrap setup-events store-token completions'`);\n          break;\n        default:\n          console.error(`Unknown shell: ${shell}. Use bash, zsh, or fish.`);\n          process.exit(1);\n      }\n    });\n\n  printDebug('All commands registered successfully');\n}\n\n/**\n * Handle program errors\n */\nfunction setupErrorHandlers(): void {\n  // Handle uncaught exceptions\n  process.on('uncaughtException', (error: Error) => {\n    console.error(''); // Empty line\n    printError('Uncaught exception', {\n      error: error.message,\n      name: error.name,\n    });\n\n    handleError(error, ERROR_CODES.GENERAL, {\n      debug: isDebugMode(program.opts()),\n      exit: true,\n      context: {\n        type: 'uncaughtException',\n      },\n    });\n  });\n\n  // Handle unhandled promise rejections\n  process.on('unhandledRejection', (reason: unknown) => {\n    console.error(''); // Empty line\n    printError('Unhandled promise rejection', {\n      reason: reason instanceof Error ? reason.message : String(reason),\n    });\n\n    const error = reason instanceof Error ? reason : new Error(String(reason));\n\n    handleError(error, ERROR_CODES.GENERAL, {\n      debug: isDebugMode(program.opts()),\n      exit: true,\n      context: {\n        type: 'unhandledRejection',\n      },\n    });\n  });\n\n  // Handle SIGINT (Ctrl+C)\n  process.on('SIGINT', () => {\n    console.log(''); // Empty line\n    printWarning('Process interrupted by user (SIGINT)');\n    console.log(''); // Empty line\n    process.exit(130); // Standard exit code for SIGINT\n  });\n\n  // Handle SIGTERM\n  process.on('SIGTERM', () => {\n    console.log(''); // Empty line\n    printWarning('Process terminated (SIGTERM)');\n    console.log(''); // Empty line\n    process.exit(143); // Standard exit code for SIGTERM\n  });\n}\n\n/**\n * Initialize CLI with options\n *\n * @param options - CLI initialization options\n */\nexport function initializeCli(options: CliOptions = {}): void {\n  const {\n    showBanner = true,\n    minimalBanner = false,\n    debug = false,\n    verbose = false,\n    quiet = false,\n    noColor = false,\n  } = options;\n\n  // Handle color output\n  if (noColor) {\n    process.env.NO_COLOR = '1';\n  }\n\n  // Display banner\n  if (showBanner && !quiet) {\n    if (minimalBanner) {\n      miniBanner();\n    } else {\n      banner({\n        minimal: false,\n        showDescription: true,\n        showTimestamp: debug || verbose,\n        showExecutionId: debug,\n      });\n    }\n  }\n\n  // Display startup info\n  displayStartupInfo({ debug, verbose, quiet });\n\n  // Check environment\n  if (!quiet) {\n    checkEnvironment();\n  }\n\n  // Setup error handlers\n  setupErrorHandlers();\n\n  // Register commands\n  try {\n    registerCommands();\n    printDebug('CLI initialization complete');\n  } catch (error) {\n    printError('CLI initialization failed', {\n      error: error instanceof Error ? error.message : String(error),\n    });\n\n    handleError(error, ERROR_CODES.CONFIGURATION, {\n      debug: debug || isDebugMode(),\n      exit: true,\n      context: {\n        stage: 'initialization',\n      },\n    });\n  }\n}\n\n/**\n * Parse command line arguments\n */\nfunction parseArguments(): void {\n  try {\n    program.parse(process.argv);\n\n    // Show help if no command provided\n    const args = process.argv.slice(2);\n    if (args.length === 0) {\n      printInfo('No command specified - displaying help');\n      console.log(''); // Empty line\n      program.outputHelp();\n      process.exit(0);\n    }\n\n    // Check if command exists\n    const command = args[0];\n    if (command && !command.startsWith('-')) {\n      const commandExists = program.commands.some(cmd => cmd.name() === command);\n\n      if (!commandExists) {\n        console.log(''); // Empty line\n        printError(`Unknown command: ${command}`);\n        console.log(''); // Empty line\n        printInfo('Available commands:');\n        program.commands.forEach(cmd => {\n          console.log(`  • ${cmd.name()} - ${cmd.description()}`);\n        });\n        console.log(''); // Empty line\n        process.exit(1);\n      }\n    }\n\n    printDebug('Command line arguments parsed successfully');\n  } catch (error) {\n    handleError(error, ERROR_CODES.GENERAL, {\n      debug: isDebugMode(program.opts()),\n      exit: true,\n      context: {\n        stage: 'argument-parsing',\n        argv: process.argv,\n      },\n    });\n  }\n}\n\n/**\n * Main CLI entry point\n *\n * @param options - CLI initialization options\n *\n * @example\n * ```typescript\n * // Standard initialization\n * main();\n *\n * // Minimal mode\n * main({ minimalBanner: true, quiet: true });\n *\n * // Debug mode\n * main({ debug: true, verbose: true });\n * ```\n */\nexport function main(options: CliOptions = {}): void {\n  try {\n    // Initialize CLI\n    initializeCli(options);\n\n    // Parse arguments and execute command\n    parseArguments();\n\n    // If we reach here, command completed successfully\n    printDebug('CLI execution completed successfully');\n  } catch (error) {\n    // Final catch-all error handler\n    console.error(''); // Empty line\n    printError('Fatal CLI error', {\n      error: error instanceof Error ? error.message : String(error),\n    });\n\n    handleError(error, ERROR_CODES.GENERAL, {\n      debug: isDebugMode(options),\n      exit: true,\n      context: {\n        stage: 'main',\n        options,\n      },\n    });\n  }\n}\n\n/**\n * Run CLI if executed directly\n */\nif (require.main === module) {\n  // Parse CLI options from environment or command line\n  const options: CliOptions = {\n    debug: process.env.DEBUG === 'true',\n    quiet: process.argv.includes('--quiet'),\n    verbose: process.argv.includes('--verbose'),\n    noColor: process.argv.includes('--no-color') || process.env.NO_COLOR === '1',\n  };\n\n  main(options);\n}"]}
380
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cli.js","sourceRoot":"","sources":["../src/cli.ts"],"names":[],"mappings":"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","sourcesContent":["#!/usr/bin/env node\n// Copyright 2026 Pipeline Builder Contributors\n// SPDX-License-Identifier: Apache-2.0\n\nimport { program } from 'commander';\nimport { auditStacks } from './commands/audit-stacks';\nimport { auditTokens } from './commands/audit-tokens';\nimport { bootstrap } from './commands/bootstrap';\nimport { createPipeline } from './commands/create-pipeline';\nimport { deploy } from './commands/deploy';\nimport { getPipeline } from './commands/get-pipeline';\nimport { getPlugin } from './commands/get-plugin';\nimport { listPipelines } from './commands/list-pipelines';\nimport { listPlugins } from './commands/list-plugins';\nimport { login } from './commands/login';\nimport { setupEvents } from './commands/setup-events';\nimport { status } from './commands/status';\nimport { storeToken } from './commands/store-token';\nimport { synth } from './commands/synth';\nimport { uploadPlugin } from './commands/upload-plugin';\nimport { validateTemplatesCommand } from './commands/validate-templates';\nimport { version } from './commands/version';\nimport {\n  APP_NAME,\n  APP_DESCRIPTION,\n  APP_VERSION,\n  ENV_VARS,\n  isDebugMode,\n  generateExecutionId,\n} from './config/cli.constants';\nimport { banner, miniBanner } from './utils/banner';\nimport { ERROR_CODES, handleError } from './utils/error-handler';\nimport { printInfo, printError, printWarning, printDebug, printSection } from './utils/output-utils';\n\n/**\n * CLI initialization options\n */\ninterface CliOptions {\n  /**\n   * Show banner on startup\n   * @default true\n   */\n  showBanner?: boolean;\n\n  /**\n   * Minimal banner (no ASCII art)\n   * @default false\n   */\n  minimalBanner?: boolean;\n\n  /**\n   * Debug mode\n   * @default false\n   */\n  debug?: boolean;\n\n  /**\n   * Verbose output\n   * @default false\n   */\n  verbose?: boolean;\n\n  /**\n   * Quiet mode (minimal output)\n   * @default false\n   */\n  quiet?: boolean;\n\n  /**\n   * No color output\n   * @default false\n   */\n  noColor?: boolean;\n}\n\n/**\n * Check environment and display warnings\n */\nfunction checkEnvironment(): void {\n  const warnings: string[] = [];\n\n  // Check for required environment variables\n  if (!process.env[ENV_VARS.PLATFORM_TOKEN]) {\n    warnings.push('PLATFORM_TOKEN environment variable is not set');\n    warnings.push('Authentication will fail for API operations');\n  }\n\n  // Check Node version\n  const nodeVersion = process.version;\n  const majorVersion = parseInt(nodeVersion.slice(1).split('.')[0] || '0', 10);\n  if (majorVersion < 18) {\n    warnings.push(`Node.js ${nodeVersion} detected - version 18+ recommended`);\n  }\n\n  // Display warnings\n  if (warnings.length > 0) {\n    printSection('Environment Warnings');\n    warnings.forEach(warning => {\n      printWarning(warning);\n    });\n    console.log('');\n  }\n}\n\n/**\n * Display startup information\n */\nfunction displayStartupInfo(options: CliOptions): void {\n  if (options.quiet) return;\n\n  const executionId = generateExecutionId();\n\n  if (options.debug || options.verbose) {\n    printDebug('CLI Configuration', {\n      name: APP_NAME,\n      version: APP_VERSION,\n      executionId,\n      node: process.version,\n      platform: process.platform,\n      arch: process.arch,\n      cwd: process.cwd(),\n      env: {\n        debug: process.env.DEBUG,\n        token: process.env[ENV_VARS.PLATFORM_TOKEN] ? 'set' : 'not set',\n        url: process.env[ENV_VARS.PLATFORM_BASE_URL] || 'default',\n      },\n    });\n  } else {\n    printDebug('Starting CLI', {\n      version: APP_VERSION,\n      executionId,\n    });\n  }\n}\n\n/**\n * Register all CLI commands\n */\nfunction registerCommands(): void {\n  printDebug('Registering commands');\n\n  // Configure program\n  program\n    .name(APP_NAME)\n    .description(APP_DESCRIPTION)\n    .version(APP_VERSION, '-v, --version', 'Show CLI version')\n    .option('--debug', 'Enable debug output with stack traces', false)\n    .option('--verbose', 'Show detailed information', false)\n    .option('--quiet', 'Minimal output (errors only)', false)\n    .option('--no-color', 'Disable colored output', false)\n    .addHelpText('after', `\nEnvironment Variables:\n  ${ENV_VARS.PLATFORM_TOKEN}              Authentication token (required)\n  ${ENV_VARS.PLATFORM_BASE_URL}                 API base URL (optional)\n  ${ENV_VARS.CLI_CONFIG_PATH}              Config file path (optional)\n  ${ENV_VARS.TLS_REJECT_UNAUTHORIZED}      Disable SSL verification if '0'\n  ${ENV_VARS.DEBUG}                        Enable debug mode if 'true'\n\nExamples:\n  $ ${APP_NAME} version\n  $ ${APP_NAME} list-pipelines --project my-app\n  $ ${APP_NAME} get-pipeline --id pipe-123 --format json\n  $ ${APP_NAME} store-token --days 30 --region us-east-1\n  $ ${APP_NAME} bootstrap --account 123456789012 --region us-east-1\n  $ ${APP_NAME} deploy --id pipe-123 --profile production\n\n`);\n\n  // Version command (special handling)\n  version(program);\n\n  // Authentication commands\n  printDebug('Registering authentication commands');\n  login(program); // Login and obtain PLATFORM_TOKEN (also supports --refresh)\n\n  // Query commands\n  printDebug('Registering query commands');\n  getPlugin(program); // Single plugin by ID\n  listPlugins(program); // Multiple plugins with filters\n  getPipeline(program); // Single pipeline by ID\n  listPipelines(program); // Multiple pipelines with filters\n\n  // Create/Upload commands\n  printDebug('Registering create/upload commands');\n  createPipeline(program); // Create pipeline configuration\n  uploadPlugin(program); // Upload and deploy plugin\n\n  // Status command\n  printDebug('Registering status command');\n  status(program); // Show environment and connectivity status\n\n  // Deployment commands\n  printDebug('Registering deployment commands');\n  storeToken(program); // Generate JWT token and store in Secrets Manager\n  setupEvents(program); // Deploy EventBridge event ingestion infrastructure\n  bootstrap(program); // Bootstrap CDK toolkit stack\n  deploy(program); // Deploy pipeline with CDK (--app prints boilerplate path)\n\n  // Operator audit commands (cron-friendly: exit 1 on findings)\n  printDebug('Registering audit commands');\n  auditTokens(program); // Scan Secrets Manager for expiring platform tokens\n  auditStacks(program); // Diff CFN stacks vs pipeline_registry to find drift\n  synth(program); // Run CDK synthesis\n  validateTemplatesCommand(program); // Validate {{ ... }} templates in a pipeline or plugin\n\n  // Shell completions\n  printDebug('Registering completions command');\n  program\n    .command('completions')\n    .description('Generate shell completions (bash, zsh, fish)')\n    .argument('<shell>', 'Shell type: bash, zsh, or fish')\n    .action((shell: string) => {\n      switch (shell) {\n        case 'bash':\n          console.log(`# pipeline-manager bash completions\n_pipeline_manager_completions() {\n  local cur=\"\\${COMP_WORDS[COMP_CWORD]}\"\n  local commands=\"login deploy synth upload-plugin create-pipeline status version bootstrap setup-events store-token completions\"\n  COMPREPLY=($(compgen -W \"\\${commands}\" -- \"\\${cur}\"))\n}\ncomplete -F _pipeline_manager_completions pipeline-manager`);\n          break;\n        case 'zsh':\n          console.log(`# pipeline-manager zsh completions\n_pipeline_manager() {\n  local commands=(login deploy synth upload-plugin create-pipeline status version bootstrap setup-events store-token completions)\n  _describe 'command' commands\n}\ncompdef _pipeline_manager pipeline-manager`);\n          break;\n        case 'fish':\n          console.log(`# pipeline-manager fish completions\ncomplete -c pipeline-manager -n '__fish_use_subcommand' -a 'login deploy synth upload-plugin create-pipeline status version bootstrap setup-events store-token completions'`);\n          break;\n        default:\n          console.error(`Unknown shell: ${shell}. Use bash, zsh, or fish.`);\n          process.exit(1);\n      }\n    });\n\n  printDebug('All commands registered successfully');\n}\n\n/**\n * Handle program errors\n */\nfunction setupErrorHandlers(): void {\n  // Handle uncaught exceptions\n  process.on('uncaughtException', (error: Error) => {\n    console.error(''); // Empty line\n    printError('Uncaught exception', {\n      error: error.message,\n      name: error.name,\n    });\n\n    handleError(error, ERROR_CODES.GENERAL, {\n      debug: isDebugMode(program.opts()),\n      exit: true,\n      context: {\n        type: 'uncaughtException',\n      },\n    });\n  });\n\n  // Handle unhandled promise rejections\n  process.on('unhandledRejection', (reason: unknown) => {\n    console.error(''); // Empty line\n    printError('Unhandled promise rejection', {\n      reason: reason instanceof Error ? reason.message : String(reason),\n    });\n\n    const error = reason instanceof Error ? reason : new Error(String(reason));\n\n    handleError(error, ERROR_CODES.GENERAL, {\n      debug: isDebugMode(program.opts()),\n      exit: true,\n      context: {\n        type: 'unhandledRejection',\n      },\n    });\n  });\n\n  // Handle SIGINT (Ctrl+C)\n  process.on('SIGINT', () => {\n    console.log(''); // Empty line\n    printWarning('Process interrupted by user (SIGINT)');\n    console.log(''); // Empty line\n    process.exit(130); // Standard exit code for SIGINT\n  });\n\n  // Handle SIGTERM\n  process.on('SIGTERM', () => {\n    console.log(''); // Empty line\n    printWarning('Process terminated (SIGTERM)');\n    console.log(''); // Empty line\n    process.exit(143); // Standard exit code for SIGTERM\n  });\n}\n\n/**\n * Initialize CLI with options\n *\n * @param options - CLI initialization options\n */\nexport function initializeCli(options: CliOptions = {}): void {\n  const {\n    showBanner = true,\n    minimalBanner = false,\n    debug = false,\n    verbose = false,\n    quiet = false,\n    noColor = false,\n  } = options;\n\n  // Handle color output\n  if (noColor) {\n    process.env.NO_COLOR = '1';\n  }\n\n  // Display banner\n  if (showBanner && !quiet) {\n    if (minimalBanner) {\n      miniBanner();\n    } else {\n      banner({\n        minimal: false,\n        showDescription: true,\n        showTimestamp: debug || verbose,\n        showExecutionId: debug,\n      });\n    }\n  }\n\n  // Display startup info\n  displayStartupInfo({ debug, verbose, quiet });\n\n  // Check environment\n  if (!quiet) {\n    checkEnvironment();\n  }\n\n  // Setup error handlers\n  setupErrorHandlers();\n\n  // Register commands\n  try {\n    registerCommands();\n    printDebug('CLI initialization complete');\n  } catch (error) {\n    printError('CLI initialization failed', {\n      error: error instanceof Error ? error.message : String(error),\n    });\n\n    handleError(error, ERROR_CODES.CONFIGURATION, {\n      debug: debug || isDebugMode(),\n      exit: true,\n      context: {\n        stage: 'initialization',\n      },\n    });\n  }\n}\n\n/**\n * Parse command line arguments\n */\nfunction parseArguments(): void {\n  try {\n    program.parse(process.argv);\n\n    // Show help if no command provided\n    const args = process.argv.slice(2);\n    if (args.length === 0) {\n      printInfo('No command specified - displaying help');\n      console.log(''); // Empty line\n      program.outputHelp();\n      process.exit(0);\n    }\n\n    // Check if command exists\n    const command = args[0];\n    if (command && !command.startsWith('-')) {\n      const commandExists = program.commands.some(cmd => cmd.name() === command);\n\n      if (!commandExists) {\n        console.log(''); // Empty line\n        printError(`Unknown command: ${command}`);\n        console.log(''); // Empty line\n        printInfo('Available commands:');\n        program.commands.forEach(cmd => {\n          console.log(`  • ${cmd.name()} - ${cmd.description()}`);\n        });\n        console.log(''); // Empty line\n        process.exit(1);\n      }\n    }\n\n    printDebug('Command line arguments parsed successfully');\n  } catch (error) {\n    handleError(error, ERROR_CODES.GENERAL, {\n      debug: isDebugMode(program.opts()),\n      exit: true,\n      context: {\n        stage: 'argument-parsing',\n        argv: process.argv,\n      },\n    });\n  }\n}\n\n/**\n * Main CLI entry point\n *\n * @param options - CLI initialization options\n *\n * @example\n * ```typescript\n * // Standard initialization\n * main();\n *\n * // Minimal mode\n * main({ minimalBanner: true, quiet: true });\n *\n * // Debug mode\n * main({ debug: true, verbose: true });\n * ```\n */\nexport function main(options: CliOptions = {}): void {\n  try {\n    // Initialize CLI\n    initializeCli(options);\n\n    // Parse arguments and execute command\n    parseArguments();\n\n    // If we reach here, command completed successfully\n    printDebug('CLI execution completed successfully');\n  } catch (error) {\n    // Final catch-all error handler\n    console.error(''); // Empty line\n    printError('Fatal CLI error', {\n      error: error instanceof Error ? error.message : String(error),\n    });\n\n    handleError(error, ERROR_CODES.GENERAL, {\n      debug: isDebugMode(options),\n      exit: true,\n      context: {\n        stage: 'main',\n        options,\n      },\n    });\n  }\n}\n\n/**\n * Run CLI if executed directly\n */\nif (require.main === module) {\n  // Parse CLI options from environment or command line\n  const options: CliOptions = {\n    debug: process.env.DEBUG === 'true',\n    quiet: process.argv.includes('--quiet'),\n    verbose: process.argv.includes('--verbose'),\n    noColor: process.argv.includes('--no-color') || process.env.NO_COLOR === '1',\n  };\n\n  main(options);\n}"]}
@@ -0,0 +1,24 @@
1
+ import { Command } from 'commander';
2
+ /**
3
+ * Registers the `audit-stacks` command with the CLI program.
4
+ *
5
+ * Diffs CloudFormation stacks (filtered to those tagged with `pipeline-builder`)
6
+ * against the platform's `pipeline_registry` table. Surfaces:
7
+ * - **Orphaned stacks**: tagged stacks with no DB record (likely leftovers after
8
+ * a pipeline was deleted from the dashboard but the CDK stack stayed in AWS).
9
+ * - **Missing stacks**: registry entries pointing to stacks that no longer exist
10
+ * in CloudFormation (likely after a manual `aws cloudformation delete-stack`).
11
+ *
12
+ * Designed for cron use:
13
+ * - Exits 0 when there are no findings.
14
+ * - Exits 1 when at least one orphaned or missing stack is found.
15
+ * - Exits 2 on AWS errors / scan failures.
16
+ *
17
+ * @example
18
+ * ```bash
19
+ * pipeline-manager audit-stacks --region us-east-1
20
+ * pipeline-manager audit-stacks --org acme --json
21
+ * ```
22
+ */
23
+ export declare function auditStacks(program: Command): void;
24
+ //# sourceMappingURL=audit-stacks.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"audit-stacks.d.ts","sourceRoot":"","sources":["../../src/commands/audit-stacks.ts"],"names":[],"mappings":"AAIA,OAAO,EAAE,OAAO,EAAE,MAAM,WAAW,CAAC;AAoCpC;;;;;;;;;;;;;;;;;;;;GAoBG;AACH,wBAAgB,WAAW,CAAC,OAAO,EAAE,OAAO,GAAG,IAAI,CAgIlD"}
@@ -0,0 +1,162 @@
1
+ "use strict";
2
+ // Copyright 2026 Pipeline Builder Contributors
3
+ // SPDX-License-Identifier: Apache-2.0
4
+ Object.defineProperty(exports, "__esModule", { value: true });
5
+ exports.auditStacks = auditStacks;
6
+ const client_cloudformation_1 = require("@aws-sdk/client-cloudformation");
7
+ const command_utils_1 = require("../utils/command-utils");
8
+ const error_handler_1 = require("../utils/error-handler");
9
+ const output_utils_1 = require("../utils/output-utils");
10
+ const ACTIVE_STATUSES = [
11
+ 'CREATE_COMPLETE', 'UPDATE_COMPLETE', 'UPDATE_ROLLBACK_COMPLETE',
12
+ 'IMPORT_COMPLETE', 'IMPORT_ROLLBACK_COMPLETE',
13
+ ];
14
+ /**
15
+ * Registers the `audit-stacks` command with the CLI program.
16
+ *
17
+ * Diffs CloudFormation stacks (filtered to those tagged with `pipeline-builder`)
18
+ * against the platform's `pipeline_registry` table. Surfaces:
19
+ * - **Orphaned stacks**: tagged stacks with no DB record (likely leftovers after
20
+ * a pipeline was deleted from the dashboard but the CDK stack stayed in AWS).
21
+ * - **Missing stacks**: registry entries pointing to stacks that no longer exist
22
+ * in CloudFormation (likely after a manual `aws cloudformation delete-stack`).
23
+ *
24
+ * Designed for cron use:
25
+ * - Exits 0 when there are no findings.
26
+ * - Exits 1 when at least one orphaned or missing stack is found.
27
+ * - Exits 2 on AWS errors / scan failures.
28
+ *
29
+ * @example
30
+ * ```bash
31
+ * pipeline-manager audit-stacks --region us-east-1
32
+ * pipeline-manager audit-stacks --org acme --json
33
+ * ```
34
+ */
35
+ function auditStacks(program) {
36
+ program
37
+ .command('audit-stacks')
38
+ .description('Diff CloudFormation stacks vs pipeline_registry to find orphaned or missing CDK deployments')
39
+ .option('--region <region>', 'AWS region (defaults to AWS_REGION env)')
40
+ .option('--profile <profile>', 'AWS CLI profile', 'default')
41
+ .option('--org <orgId>', 'Filter both registry and stack scan to a single org')
42
+ .option('--json', 'Output results as JSON', false)
43
+ .option('--verify-ssl', 'Enable SSL certificate verification')
44
+ .option('--no-verify-ssl', 'Disable SSL certificate verification')
45
+ .action(async (options) => {
46
+ const executionId = (0, command_utils_1.printCommandHeader)('Audit Stacks');
47
+ (0, command_utils_1.printSslWarning)(options.verifySsl);
48
+ const region = options.region || process.env.AWS_REGION || process.env.CDK_DEFAULT_REGION || 'us-east-1';
49
+ try {
50
+ // Fetch the platform's view of registered pipelines.
51
+ (0, output_utils_1.printInfo)('Fetching pipeline registry from platform', { org: options.org ?? '(all)' });
52
+ const client = await (0, command_utils_1.createAuthenticatedClientAsync)(options);
53
+ const registryRes = await client.get(`/api/pipelines/registry${options.org ? `?orgId=${encodeURIComponent(options.org)}` : ''}`);
54
+ const entries = registryRes?.entries
55
+ ?? registryRes?.data?.entries
56
+ ?? [];
57
+ // Fetch stacks from CloudFormation.
58
+ (0, output_utils_1.printInfo)('Listing CloudFormation stacks', { region });
59
+ const cfn = new client_cloudformation_1.CloudFormationClient({ region });
60
+ const stacks = [];
61
+ let nextToken;
62
+ let pages = 0;
63
+ do {
64
+ const listResp = await cfn.send(new client_cloudformation_1.ListStacksCommand({
65
+ StackStatusFilter: ACTIVE_STATUSES,
66
+ NextToken: nextToken,
67
+ }));
68
+ for (const summary of listResp.StackSummaries ?? []) {
69
+ if (!summary.StackName)
70
+ continue;
71
+ // Pull tags via DescribeStacks (ListStacks doesn't include them).
72
+ const desc = await cfn.send(new client_cloudformation_1.DescribeStacksCommand({ StackName: summary.StackName }));
73
+ const tags = desc.Stacks?.[0]?.Tags ?? [];
74
+ const pbTag = tags.find((t) => t.Key === 'pipeline-builder')?.Value;
75
+ const orgTag = tags.find((t) => t.Key === 'OrgId' || t.Key === 'orgId')?.Value;
76
+ if (!pbTag)
77
+ continue; // Not a pipeline-builder stack.
78
+ if (options.org && orgTag && orgTag !== options.org)
79
+ continue; // Wrong org filter.
80
+ stacks.push({
81
+ stackName: summary.StackName,
82
+ stackStatus: summary.StackStatus ?? 'UNKNOWN',
83
+ region,
84
+ pipelineBuilderTag: pbTag,
85
+ orgIdTag: orgTag,
86
+ creationTime: summary.CreationTime,
87
+ });
88
+ }
89
+ nextToken = listResp.NextToken;
90
+ pages++;
91
+ } while (nextToken && pages < 20);
92
+ // Diff.
93
+ const registryStackNames = new Set(entries.map((e) => e.stackName).filter((n) => !!n));
94
+ const cfnStackNames = new Set(stacks.map((s) => s.stackName));
95
+ const findings = [];
96
+ for (const stack of stacks) {
97
+ if (!registryStackNames.has(stack.stackName)) {
98
+ findings.push({
99
+ type: 'orphaned-stack',
100
+ stackName: stack.stackName,
101
+ detail: `CloudFormation stack tagged pipeline-builder=${stack.pipelineBuilderTag ?? '(unset)'} has no matching pipeline_registry entry`,
102
+ });
103
+ }
104
+ }
105
+ for (const entry of entries) {
106
+ if (entry.stackName && !cfnStackNames.has(entry.stackName)) {
107
+ findings.push({
108
+ type: 'missing-stack',
109
+ stackName: entry.stackName,
110
+ pipelineId: entry.pipelineId,
111
+ pipelineName: entry.pipelineName,
112
+ detail: `Registry entry references stack ${entry.stackName} but no such stack exists in CloudFormation`,
113
+ });
114
+ }
115
+ }
116
+ const exitCode = findings.length > 0 ? 1 : 0;
117
+ if (options.json) {
118
+ console.log(JSON.stringify({
119
+ scannedAt: new Date().toISOString(),
120
+ region,
121
+ orgFilter: options.org,
122
+ registryEntries: entries.length,
123
+ cfnStacks: stacks.length,
124
+ findings,
125
+ executionId,
126
+ }, null, 2));
127
+ }
128
+ else {
129
+ (0, output_utils_1.printSection)('Audit Results');
130
+ (0, output_utils_1.printInfo)('Counts', {
131
+ registryEntries: String(entries.length),
132
+ cfnStacks: String(stacks.length),
133
+ findings: String(findings.length),
134
+ });
135
+ if (findings.length === 0) {
136
+ (0, output_utils_1.printSuccess)('No drift between pipeline_registry and CloudFormation');
137
+ }
138
+ else {
139
+ for (const f of findings) {
140
+ if (f.type === 'orphaned-stack') {
141
+ (0, output_utils_1.printWarning)(`ORPHANED ${f.stackName} — ${f.detail}`);
142
+ }
143
+ else {
144
+ (0, output_utils_1.printWarning)(`MISSING ${f.stackName} (pipeline ${f.pipelineId ?? '?'}) — ${f.detail}`);
145
+ }
146
+ }
147
+ (0, output_utils_1.printError)(`${findings.length} drift finding${findings.length === 1 ? '' : 's'}. Review and reconcile manually.`);
148
+ }
149
+ }
150
+ process.exit(exitCode);
151
+ }
152
+ catch (error) {
153
+ (0, error_handler_1.handleError)(error, error_handler_1.ERROR_CODES.API_REQUEST, {
154
+ debug: program.opts().debug,
155
+ exit: false,
156
+ context: { command: 'audit-stacks', executionId },
157
+ });
158
+ process.exit(2);
159
+ }
160
+ });
161
+ }
162
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"audit-stacks.js","sourceRoot":"","sources":["../../src/commands/audit-stacks.ts"],"names":[],"mappings":"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","sourcesContent":["// Copyright 2026 Pipeline Builder Contributors\n// SPDX-License-Identifier: Apache-2.0\n\nimport { CloudFormationClient, ListStacksCommand, DescribeStacksCommand, type StackStatus } from '@aws-sdk/client-cloudformation';\nimport { Command } from 'commander';\nimport { createAuthenticatedClientAsync, printCommandHeader, printSslWarning } from '../utils/command-utils';\nimport { ERROR_CODES, handleError } from '../utils/error-handler';\nimport { printError, printInfo, printSection, printSuccess, printWarning } from '../utils/output-utils';\n\ninterface RegistryEntry {\n  pipelineId: string;\n  pipelineArn: string;\n  pipelineName: string;\n  stackName?: string;\n  accountId?: string;\n  region?: string;\n}\n\ninterface StackInfo {\n  stackName: string;\n  stackStatus: string;\n  region: string;\n  pipelineBuilderTag?: string;\n  orgIdTag?: string;\n  creationTime?: Date;\n}\n\ninterface AuditFinding {\n  type: 'orphaned-stack' | 'missing-stack';\n  stackName?: string;\n  pipelineId?: string;\n  pipelineName?: string;\n  detail: string;\n}\n\nconst ACTIVE_STATUSES = [\n  'CREATE_COMPLETE', 'UPDATE_COMPLETE', 'UPDATE_ROLLBACK_COMPLETE',\n  'IMPORT_COMPLETE', 'IMPORT_ROLLBACK_COMPLETE',\n];\n\n/**\n * Registers the `audit-stacks` command with the CLI program.\n *\n * Diffs CloudFormation stacks (filtered to those tagged with `pipeline-builder`)\n * against the platform's `pipeline_registry` table. Surfaces:\n *   - **Orphaned stacks**: tagged stacks with no DB record (likely leftovers after\n *     a pipeline was deleted from the dashboard but the CDK stack stayed in AWS).\n *   - **Missing stacks**: registry entries pointing to stacks that no longer exist\n *     in CloudFormation (likely after a manual `aws cloudformation delete-stack`).\n *\n * Designed for cron use:\n *   - Exits 0 when there are no findings.\n *   - Exits 1 when at least one orphaned or missing stack is found.\n *   - Exits 2 on AWS errors / scan failures.\n *\n * @example\n * ```bash\n * pipeline-manager audit-stacks --region us-east-1\n * pipeline-manager audit-stacks --org acme --json\n * ```\n */\nexport function auditStacks(program: Command): void {\n  program\n    .command('audit-stacks')\n    .description('Diff CloudFormation stacks vs pipeline_registry to find orphaned or missing CDK deployments')\n    .option('--region <region>', 'AWS region (defaults to AWS_REGION env)')\n    .option('--profile <profile>', 'AWS CLI profile', 'default')\n    .option('--org <orgId>', 'Filter both registry and stack scan to a single org')\n    .option('--json', 'Output results as JSON', false)\n    .option('--verify-ssl', 'Enable SSL certificate verification')\n    .option('--no-verify-ssl', 'Disable SSL certificate verification')\n    .action(async (options) => {\n      const executionId = printCommandHeader('Audit Stacks');\n      printSslWarning(options.verifySsl);\n\n      const region = options.region || process.env.AWS_REGION || process.env.CDK_DEFAULT_REGION || 'us-east-1';\n\n      try {\n        // Fetch the platform's view of registered pipelines.\n        printInfo('Fetching pipeline registry from platform', { org: options.org ?? '(all)' });\n        const client = await createAuthenticatedClientAsync(options);\n        const registryRes = await client.get<{ entries: RegistryEntry[] }>(\n          `/api/pipelines/registry${options.org ? `?orgId=${encodeURIComponent(options.org)}` : ''}`,\n        );\n        const entries: RegistryEntry[] = (registryRes as { entries?: RegistryEntry[] })?.entries\n          ?? (registryRes as { data?: { entries?: RegistryEntry[] } })?.data?.entries\n          ?? [];\n\n        // Fetch stacks from CloudFormation.\n        printInfo('Listing CloudFormation stacks', { region });\n        const cfn = new CloudFormationClient({ region });\n        const stacks: StackInfo[] = [];\n        let nextToken: string | undefined;\n        let pages = 0;\n        do {\n          const listResp = await cfn.send(new ListStacksCommand({\n            StackStatusFilter: ACTIVE_STATUSES as StackStatus[],\n            NextToken: nextToken,\n          }));\n          for (const summary of listResp.StackSummaries ?? []) {\n            if (!summary.StackName) continue;\n            // Pull tags via DescribeStacks (ListStacks doesn't include them).\n            const desc = await cfn.send(new DescribeStacksCommand({ StackName: summary.StackName }));\n            const tags = desc.Stacks?.[0]?.Tags ?? [];\n            const pbTag = tags.find((t) => t.Key === 'pipeline-builder')?.Value;\n            const orgTag = tags.find((t) => t.Key === 'OrgId' || t.Key === 'orgId')?.Value;\n            if (!pbTag) continue; // Not a pipeline-builder stack.\n            if (options.org && orgTag && orgTag !== options.org) continue; // Wrong org filter.\n            stacks.push({\n              stackName: summary.StackName,\n              stackStatus: summary.StackStatus ?? 'UNKNOWN',\n              region,\n              pipelineBuilderTag: pbTag,\n              orgIdTag: orgTag,\n              creationTime: summary.CreationTime,\n            });\n          }\n          nextToken = listResp.NextToken;\n          pages++;\n        } while (nextToken && pages < 20);\n\n        // Diff.\n        const registryStackNames = new Set(entries.map((e) => e.stackName).filter((n): n is string => !!n));\n        const cfnStackNames = new Set(stacks.map((s) => s.stackName));\n\n        const findings: AuditFinding[] = [];\n        for (const stack of stacks) {\n          if (!registryStackNames.has(stack.stackName)) {\n            findings.push({\n              type: 'orphaned-stack',\n              stackName: stack.stackName,\n              detail: `CloudFormation stack tagged pipeline-builder=${stack.pipelineBuilderTag ?? '(unset)'} has no matching pipeline_registry entry`,\n            });\n          }\n        }\n        for (const entry of entries) {\n          if (entry.stackName && !cfnStackNames.has(entry.stackName)) {\n            findings.push({\n              type: 'missing-stack',\n              stackName: entry.stackName,\n              pipelineId: entry.pipelineId,\n              pipelineName: entry.pipelineName,\n              detail: `Registry entry references stack ${entry.stackName} but no such stack exists in CloudFormation`,\n            });\n          }\n        }\n\n        const exitCode = findings.length > 0 ? 1 : 0;\n\n        if (options.json) {\n          console.log(JSON.stringify({\n            scannedAt: new Date().toISOString(),\n            region,\n            orgFilter: options.org,\n            registryEntries: entries.length,\n            cfnStacks: stacks.length,\n            findings,\n            executionId,\n          }, null, 2));\n        } else {\n          printSection('Audit Results');\n          printInfo('Counts', {\n            registryEntries: String(entries.length),\n            cfnStacks: String(stacks.length),\n            findings: String(findings.length),\n          });\n          if (findings.length === 0) {\n            printSuccess('No drift between pipeline_registry and CloudFormation');\n          } else {\n            for (const f of findings) {\n              if (f.type === 'orphaned-stack') {\n                printWarning(`ORPHANED  ${f.stackName} — ${f.detail}`);\n              } else {\n                printWarning(`MISSING   ${f.stackName} (pipeline ${f.pipelineId ?? '?'}) — ${f.detail}`);\n              }\n            }\n            printError(`${findings.length} drift finding${findings.length === 1 ? '' : 's'}. Review and reconcile manually.`);\n          }\n        }\n        process.exit(exitCode);\n      } catch (error) {\n        handleError(error, ERROR_CODES.API_REQUEST, {\n          debug: program.opts().debug,\n          exit: false,\n          context: { command: 'audit-stacks', executionId },\n        });\n        process.exit(2);\n      }\n    });\n}\n"]}
@@ -0,0 +1,21 @@
1
+ import { Command } from 'commander';
2
+ /**
3
+ * Registers the `audit-tokens` command with the CLI program.
4
+ *
5
+ * Scans AWS Secrets Manager for stored platform JWTs (matching the
6
+ * `pipeline-builder/<orgId>/platform` naming convention used by `store-token`)
7
+ * and reports which secrets are expired or close to expiring.
8
+ *
9
+ * Designed for cron use:
10
+ * - Exits 0 when nothing is at-risk.
11
+ * - Exits 1 when at least one secret is expired or expires within `--warn-days`.
12
+ * - Exits 2 on AWS errors / scan failures.
13
+ *
14
+ * @example
15
+ * ```bash
16
+ * pipeline-manager audit-tokens --region us-east-1 --warn-days 7
17
+ * pipeline-manager audit-tokens --region us-east-1 --json
18
+ * ```
19
+ */
20
+ export declare function auditTokens(program: Command): void;
21
+ //# sourceMappingURL=audit-tokens.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"audit-tokens.d.ts","sourceRoot":"","sources":["../../src/commands/audit-tokens.ts"],"names":[],"mappings":"AAIA,OAAO,EAAE,OAAO,EAAE,MAAM,WAAW,CAAC;AAepC;;;;;;;;;;;;;;;;;GAiBG;AACH,wBAAgB,WAAW,CAAC,OAAO,EAAE,OAAO,GAAG,IAAI,CA8GlD"}
@@ -0,0 +1,137 @@
1
+ "use strict";
2
+ // Copyright 2026 Pipeline Builder Contributors
3
+ // SPDX-License-Identifier: Apache-2.0
4
+ Object.defineProperty(exports, "__esModule", { value: true });
5
+ exports.auditTokens = auditTokens;
6
+ const pipeline_core_1 = require("@pipeline-builder/pipeline-core");
7
+ const auth_guard_1 = require("../utils/auth-guard");
8
+ const aws_secrets_1 = require("../utils/aws-secrets");
9
+ const command_utils_1 = require("../utils/command-utils");
10
+ const error_handler_1 = require("../utils/error-handler");
11
+ const output_utils_1 = require("../utils/output-utils");
12
+ /**
13
+ * Registers the `audit-tokens` command with the CLI program.
14
+ *
15
+ * Scans AWS Secrets Manager for stored platform JWTs (matching the
16
+ * `pipeline-builder/<orgId>/platform` naming convention used by `store-token`)
17
+ * and reports which secrets are expired or close to expiring.
18
+ *
19
+ * Designed for cron use:
20
+ * - Exits 0 when nothing is at-risk.
21
+ * - Exits 1 when at least one secret is expired or expires within `--warn-days`.
22
+ * - Exits 2 on AWS errors / scan failures.
23
+ *
24
+ * @example
25
+ * ```bash
26
+ * pipeline-manager audit-tokens --region us-east-1 --warn-days 7
27
+ * pipeline-manager audit-tokens --region us-east-1 --json
28
+ * ```
29
+ */
30
+ function auditTokens(program) {
31
+ program
32
+ .command('audit-tokens')
33
+ .description('Scan stored platform tokens in AWS Secrets Manager and report upcoming expirations')
34
+ .option('--region <region>', 'AWS region (defaults to AWS_REGION env)')
35
+ .option('--profile <profile>', 'AWS CLI profile', 'default')
36
+ .option('--warn-days <days>', 'Flag tokens expiring within N days as at-risk', '7')
37
+ .option('--prefix <prefix>', 'Secrets Manager name prefix to scan', `${pipeline_core_1.CoreConstants.SECRETS_PATH_PREFIX}/`)
38
+ .option('--json', 'Output results as JSON', false)
39
+ .action(async (options) => {
40
+ const executionId = (0, command_utils_1.printCommandHeader)('Audit Tokens');
41
+ const region = options.region || process.env.AWS_REGION || process.env.CDK_DEFAULT_REGION || 'us-east-1';
42
+ const warnDays = parseInt(options.warnDays, 10);
43
+ if (!Number.isFinite(warnDays) || warnDays < 0) {
44
+ (0, output_utils_1.printError)('Invalid --warn-days value', { provided: options.warnDays });
45
+ process.exit(2);
46
+ }
47
+ try {
48
+ (0, output_utils_1.printInfo)('Listing secrets', { region, prefix: options.prefix });
49
+ const secrets = await (0, aws_secrets_1.listSecrets)(options.prefix, { region, profile: options.profile });
50
+ // Filter to ones following the `<prefix>/<orgId>/platform` pattern.
51
+ const platformSecrets = secrets.filter((s) => s.name.endsWith('/platform'));
52
+ const entries = [];
53
+ const now = Date.now();
54
+ const warnCutoff = now + warnDays * 24 * 60 * 60 * 1000;
55
+ for (const s of platformSecrets) {
56
+ let raw;
57
+ try {
58
+ raw = await (0, aws_secrets_1.getSecretValue)(s.name, { region, profile: options.profile });
59
+ }
60
+ catch (err) {
61
+ (0, output_utils_1.printWarning)(`Could not read ${s.name}: ${err instanceof Error ? err.message : String(err)}`);
62
+ continue;
63
+ }
64
+ let parsed;
65
+ try {
66
+ parsed = JSON.parse(raw);
67
+ }
68
+ catch {
69
+ (0, output_utils_1.printWarning)(`Secret ${s.name} is not valid JSON, skipping`);
70
+ continue;
71
+ }
72
+ // Prefer the explicit expiresAt field written by store-token; fall back
73
+ // to decoding the JWT's exp claim if the field is missing.
74
+ let expiresAt;
75
+ if (parsed.expiresAt) {
76
+ const d = new Date(parsed.expiresAt);
77
+ if (!Number.isNaN(d.getTime()))
78
+ expiresAt = d;
79
+ }
80
+ if (!expiresAt && parsed.accessToken) {
81
+ const payload = (0, auth_guard_1.decodeTokenPayload)(parsed.accessToken);
82
+ if (payload?.exp && typeof payload.exp === 'number') {
83
+ expiresAt = new Date(payload.exp * 1000);
84
+ }
85
+ }
86
+ if (!expiresAt) {
87
+ (0, output_utils_1.printWarning)(`Secret ${s.name} has no expiry information, skipping`);
88
+ continue;
89
+ }
90
+ const daysUntilExpiry = Math.floor((expiresAt.getTime() - now) / (24 * 60 * 60 * 1000));
91
+ const status = expiresAt.getTime() < now ? 'expired'
92
+ : expiresAt.getTime() < warnCutoff ? 'expiring-soon'
93
+ : 'ok';
94
+ entries.push({ secretName: s.name, arn: s.arn, expiresAt, daysUntilExpiry, status });
95
+ }
96
+ const atRisk = entries.filter((e) => e.status !== 'ok');
97
+ const exitCode = atRisk.length > 0 ? 1 : 0;
98
+ if (options.json) {
99
+ console.log(JSON.stringify({
100
+ scannedAt: new Date().toISOString(),
101
+ region,
102
+ warnDays,
103
+ totalScanned: entries.length,
104
+ atRiskCount: atRisk.length,
105
+ entries: entries.map((e) => ({ ...e, expiresAt: e.expiresAt.toISOString() })),
106
+ executionId,
107
+ }, null, 2));
108
+ }
109
+ else {
110
+ (0, output_utils_1.printSection)('Audit Results');
111
+ (0, output_utils_1.printInfo)(`Scanned ${entries.length} platform secret${entries.length === 1 ? '' : 's'}`);
112
+ if (atRisk.length === 0) {
113
+ (0, output_utils_1.printSuccess)(`All tokens valid for at least ${warnDays} days`);
114
+ }
115
+ else {
116
+ for (const e of atRisk) {
117
+ const label = e.status === 'expired'
118
+ ? `EXPIRED ${Math.abs(e.daysUntilExpiry)} day${e.daysUntilExpiry === -1 ? '' : 's'} ago`
119
+ : `expires in ${e.daysUntilExpiry} day${e.daysUntilExpiry === 1 ? '' : 's'}`;
120
+ (0, output_utils_1.printWarning)(`${e.secretName} — ${label}`);
121
+ }
122
+ (0, output_utils_1.printError)(`${atRisk.length} secret${atRisk.length === 1 ? '' : 's'} need rotation. Run \`pipeline-manager store-token --days <N>\` to refresh.`);
123
+ }
124
+ }
125
+ process.exit(exitCode);
126
+ }
127
+ catch (error) {
128
+ (0, error_handler_1.handleError)(error, error_handler_1.ERROR_CODES.API_REQUEST, {
129
+ debug: program.opts().debug,
130
+ exit: false,
131
+ context: { command: 'audit-tokens', executionId },
132
+ });
133
+ process.exit(2);
134
+ }
135
+ });
136
+ }
137
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"audit-tokens.js","sourceRoot":"","sources":["../../src/commands/audit-tokens.ts"],"names":[],"mappings":"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","sourcesContent":["// Copyright 2026 Pipeline Builder Contributors\n// SPDX-License-Identifier: Apache-2.0\n\nimport { CoreConstants } from '@pipeline-builder/pipeline-core';\nimport { Command } from 'commander';\nimport { decodeTokenPayload } from '../utils/auth-guard';\nimport { getSecretValue, listSecrets } from '../utils/aws-secrets';\nimport { printCommandHeader } from '../utils/command-utils';\nimport { ERROR_CODES, handleError } from '../utils/error-handler';\nimport { printError, printInfo, printSection, printSuccess, printWarning } from '../utils/output-utils';\n\ninterface TokenAuditEntry {\n  secretName: string;\n  arn: string;\n  expiresAt: Date;\n  daysUntilExpiry: number;\n  status: 'expired' | 'expiring-soon' | 'ok';\n}\n\n/**\n * Registers the `audit-tokens` command with the CLI program.\n *\n * Scans AWS Secrets Manager for stored platform JWTs (matching the\n * `pipeline-builder/<orgId>/platform` naming convention used by `store-token`)\n * and reports which secrets are expired or close to expiring.\n *\n * Designed for cron use:\n *   - Exits 0 when nothing is at-risk.\n *   - Exits 1 when at least one secret is expired or expires within `--warn-days`.\n *   - Exits 2 on AWS errors / scan failures.\n *\n * @example\n * ```bash\n * pipeline-manager audit-tokens --region us-east-1 --warn-days 7\n * pipeline-manager audit-tokens --region us-east-1 --json\n * ```\n */\nexport function auditTokens(program: Command): void {\n  program\n    .command('audit-tokens')\n    .description('Scan stored platform tokens in AWS Secrets Manager and report upcoming expirations')\n    .option('--region <region>', 'AWS region (defaults to AWS_REGION env)')\n    .option('--profile <profile>', 'AWS CLI profile', 'default')\n    .option('--warn-days <days>', 'Flag tokens expiring within N days as at-risk', '7')\n    .option('--prefix <prefix>', 'Secrets Manager name prefix to scan', `${CoreConstants.SECRETS_PATH_PREFIX}/`)\n    .option('--json', 'Output results as JSON', false)\n    .action(async (options) => {\n      const executionId = printCommandHeader('Audit Tokens');\n      const region = options.region || process.env.AWS_REGION || process.env.CDK_DEFAULT_REGION || 'us-east-1';\n      const warnDays = parseInt(options.warnDays, 10);\n      if (!Number.isFinite(warnDays) || warnDays < 0) {\n        printError('Invalid --warn-days value', { provided: options.warnDays });\n        process.exit(2);\n      }\n\n      try {\n        printInfo('Listing secrets', { region, prefix: options.prefix });\n        const secrets = await listSecrets(options.prefix, { region, profile: options.profile });\n\n        // Filter to ones following the `<prefix>/<orgId>/platform` pattern.\n        const platformSecrets = secrets.filter((s) => s.name.endsWith('/platform'));\n\n        const entries: TokenAuditEntry[] = [];\n        const now = Date.now();\n        const warnCutoff = now + warnDays * 24 * 60 * 60 * 1000;\n\n        for (const s of platformSecrets) {\n          let raw: string;\n          try {\n            raw = await getSecretValue(s.name, { region, profile: options.profile });\n          } catch (err) {\n            printWarning(`Could not read ${s.name}: ${err instanceof Error ? err.message : String(err)}`);\n            continue;\n          }\n\n          let parsed: { accessToken?: string; expiresAt?: string };\n          try {\n            parsed = JSON.parse(raw);\n          } catch {\n            printWarning(`Secret ${s.name} is not valid JSON, skipping`);\n            continue;\n          }\n\n          // Prefer the explicit expiresAt field written by store-token; fall back\n          // to decoding the JWT's exp claim if the field is missing.\n          let expiresAt: Date | undefined;\n          if (parsed.expiresAt) {\n            const d = new Date(parsed.expiresAt);\n            if (!Number.isNaN(d.getTime())) expiresAt = d;\n          }\n          if (!expiresAt && parsed.accessToken) {\n            const payload = decodeTokenPayload(parsed.accessToken);\n            if (payload?.exp && typeof payload.exp === 'number') {\n              expiresAt = new Date(payload.exp * 1000);\n            }\n          }\n          if (!expiresAt) {\n            printWarning(`Secret ${s.name} has no expiry information, skipping`);\n            continue;\n          }\n\n          const daysUntilExpiry = Math.floor((expiresAt.getTime() - now) / (24 * 60 * 60 * 1000));\n          const status: TokenAuditEntry['status'] =\n            expiresAt.getTime() < now ? 'expired'\n              : expiresAt.getTime() < warnCutoff ? 'expiring-soon'\n              : 'ok';\n          entries.push({ secretName: s.name, arn: s.arn, expiresAt, daysUntilExpiry, status });\n        }\n\n        const atRisk = entries.filter((e) => e.status !== 'ok');\n        const exitCode = atRisk.length > 0 ? 1 : 0;\n\n        if (options.json) {\n          console.log(JSON.stringify({\n            scannedAt: new Date().toISOString(),\n            region,\n            warnDays,\n            totalScanned: entries.length,\n            atRiskCount: atRisk.length,\n            entries: entries.map((e) => ({ ...e, expiresAt: e.expiresAt.toISOString() })),\n            executionId,\n          }, null, 2));\n        } else {\n          printSection('Audit Results');\n          printInfo(`Scanned ${entries.length} platform secret${entries.length === 1 ? '' : 's'}`);\n          if (atRisk.length === 0) {\n            printSuccess(`All tokens valid for at least ${warnDays} days`);\n          } else {\n            for (const e of atRisk) {\n              const label = e.status === 'expired'\n                ? `EXPIRED ${Math.abs(e.daysUntilExpiry)} day${e.daysUntilExpiry === -1 ? '' : 's'} ago`\n                : `expires in ${e.daysUntilExpiry} day${e.daysUntilExpiry === 1 ? '' : 's'}`;\n              printWarning(`${e.secretName} — ${label}`);\n            }\n            printError(`${atRisk.length} secret${atRisk.length === 1 ? '' : 's'} need rotation. Run \\`pipeline-manager store-token --days <N>\\` to refresh.`);\n          }\n        }\n        process.exit(exitCode);\n      } catch (error) {\n        handleError(error, ERROR_CODES.API_REQUEST, {\n          debug: program.opts().debug,\n          exit: false,\n          context: { command: 'audit-tokens', executionId },\n        });\n        process.exit(2);\n      }\n    });\n}\n"]}
@@ -3,6 +3,10 @@ interface TokenPayload {
3
3
  isAdmin?: boolean;
4
4
  sub?: string;
5
5
  organizationId?: string;
6
+ /** Standard JWT expiry claim, in seconds since epoch. */
7
+ exp?: number;
8
+ /** Standard JWT issued-at claim, in seconds since epoch. */
9
+ iat?: number;
6
10
  }
7
11
  /**
8
12
  * Decode a JWT token payload without signature verification.
@@ -1 +1 @@
1
- {"version":3,"file":"auth-guard.d.ts","sourceRoot":"","sources":["../../src/utils/auth-guard.ts"],"names":[],"mappings":"AAGA,UAAU,YAAY;IACpB,IAAI,CAAC,EAAE,MAAM,CAAC;IACd,OAAO,CAAC,EAAE,OAAO,CAAC;IAClB,GAAG,CAAC,EAAE,MAAM,CAAC;IACb,cAAc,CAAC,EAAE,MAAM,CAAC;CACzB;AAED;;;;;;GAMG;AACH,wBAAgB,kBAAkB,CAAC,KAAK,EAAE,MAAM,GAAG,YAAY,GAAG,IAAI,CASrE"}
1
+ {"version":3,"file":"auth-guard.d.ts","sourceRoot":"","sources":["../../src/utils/auth-guard.ts"],"names":[],"mappings":"AAGA,UAAU,YAAY;IACpB,IAAI,CAAC,EAAE,MAAM,CAAC;IACd,OAAO,CAAC,EAAE,OAAO,CAAC;IAClB,GAAG,CAAC,EAAE,MAAM,CAAC;IACb,cAAc,CAAC,EAAE,MAAM,CAAC;IACxB,yDAAyD;IACzD,GAAG,CAAC,EAAE,MAAM,CAAC;IACb,4DAA4D;IAC5D,GAAG,CAAC,EAAE,MAAM,CAAC;CACd;AAED;;;;;;GAMG;AACH,wBAAgB,kBAAkB,CAAC,KAAK,EAAE,MAAM,GAAG,YAAY,GAAG,IAAI,CASrE"}
@@ -22,4 +22,4 @@ function decodeTokenPayload(token) {
22
22
  return null;
23
23
  }
24
24
  }
25
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYXV0aC1ndWFyZC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy91dGlscy9hdXRoLWd1YXJkLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7QUFBQSwrQ0FBK0M7QUFDL0Msc0NBQXNDOztBQWdCdEMsZ0RBU0M7QUFoQkQ7Ozs7OztHQU1HO0FBQ0gsU0FBZ0Isa0JBQWtCLENBQUMsS0FBYTtJQUM5QyxJQUFJLENBQUM7UUFDSCxNQUFNLEtBQUssR0FBRyxLQUFLLENBQUMsS0FBSyxDQUFDLEdBQUcsQ0FBQyxDQUFDO1FBQy9CLElBQUksS0FBSyxDQUFDLE1BQU0sS0FBSyxDQUFDLElBQUksQ0FBQyxLQUFLLENBQUMsQ0FBQyxDQUFDO1lBQUUsT0FBTyxJQUFJLENBQUM7UUFDakQsTUFBTSxPQUFPLEdBQUcsTUFBTSxDQUFDLElBQUksQ0FBQyxLQUFLLENBQUMsQ0FBQyxDQUFDLEVBQUUsV0FBVyxDQUFDLENBQUMsUUFBUSxDQUFDLE9BQU8sQ0FBQyxDQUFDO1FBQ3JFLE9BQU8sSUFBSSxDQUFDLEtBQUssQ0FBQyxPQUFPLENBQWlCLENBQUM7SUFDN0MsQ0FBQztJQUFDLE1BQU0sQ0FBQztRQUNQLE9BQU8sSUFBSSxDQUFDO0lBQ2QsQ0FBQztBQUNILENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyIvLyBDb3B5cmlnaHQgMjAyNiBQaXBlbGluZSBCdWlsZGVyIENvbnRyaWJ1dG9yc1xuLy8gU1BEWC1MaWNlbnNlLUlkZW50aWZpZXI6IEFwYWNoZS0yLjBcblxuaW50ZXJmYWNlIFRva2VuUGF5bG9hZCB7XG4gIHJvbGU/OiBzdHJpbmc7XG4gIGlzQWRtaW4/OiBib29sZWFuO1xuICBzdWI/OiBzdHJpbmc7XG4gIG9yZ2FuaXphdGlvbklkPzogc3RyaW5nO1xufVxuXG4vKipcbiAqIERlY29kZSBhIEpXVCB0b2tlbiBwYXlsb2FkIHdpdGhvdXQgc2lnbmF0dXJlIHZlcmlmaWNhdGlvbi5cbiAqXG4gKiBTRUNVUklUWTogTk8gc2lnbmF0dXJlIHZlcmlmaWNhdGlvbi4gVGhlIGRlY29kZWQgcGF5bG9hZCBNVVNUIE5PVCBiZSB1c2VkXG4gKiBmb3IgYXV0aG9yaXphdGlvbiBkZWNpc2lvbnMg4oCUIHRoZSBzZXJ2ZXIgdmFsaWRhdGVzIHRoZSB0b2tlbiBvbiBldmVyeSBBUElcbiAqIGNhbGwuIFVzZSBvbmx5IGZvciBhZHZpc29yeSBVWCBoaW50cy5cbiAqL1xuZXhwb3J0IGZ1bmN0aW9uIGRlY29kZVRva2VuUGF5bG9hZCh0b2tlbjogc3RyaW5nKTogVG9rZW5QYXlsb2FkIHwgbnVsbCB7XG4gIHRyeSB7XG4gICAgY29uc3QgcGFydHMgPSB0b2tlbi5zcGxpdCgnLicpO1xuICAgIGlmIChwYXJ0cy5sZW5ndGggIT09IDMgfHwgIXBhcnRzWzFdKSByZXR1cm4gbnVsbDtcbiAgICBjb25zdCBwYXlsb2FkID0gQnVmZmVyLmZyb20ocGFydHNbMV0sICdiYXNlNjR1cmwnKS50b1N0cmluZygndXRmLTgnKTtcbiAgICByZXR1cm4gSlNPTi5wYXJzZShwYXlsb2FkKSBhcyBUb2tlblBheWxvYWQ7XG4gIH0gY2F0Y2gge1xuICAgIHJldHVybiBudWxsO1xuICB9XG59XG4iXX0=
25
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYXV0aC1ndWFyZC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy91dGlscy9hdXRoLWd1YXJkLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7QUFBQSwrQ0FBK0M7QUFDL0Msc0NBQXNDOztBQW9CdEMsZ0RBU0M7QUFoQkQ7Ozs7OztHQU1HO0FBQ0gsU0FBZ0Isa0JBQWtCLENBQUMsS0FBYTtJQUM5QyxJQUFJLENBQUM7UUFDSCxNQUFNLEtBQUssR0FBRyxLQUFLLENBQUMsS0FBSyxDQUFDLEdBQUcsQ0FBQyxDQUFDO1FBQy9CLElBQUksS0FBSyxDQUFDLE1BQU0sS0FBSyxDQUFDLElBQUksQ0FBQyxLQUFLLENBQUMsQ0FBQyxDQUFDO1lBQUUsT0FBTyxJQUFJLENBQUM7UUFDakQsTUFBTSxPQUFPLEdBQUcsTUFBTSxDQUFDLElBQUksQ0FBQyxLQUFLLENBQUMsQ0FBQyxDQUFDLEVBQUUsV0FBVyxDQUFDLENBQUMsUUFBUSxDQUFDLE9BQU8sQ0FBQyxDQUFDO1FBQ3JFLE9BQU8sSUFBSSxDQUFDLEtBQUssQ0FBQyxPQUFPLENBQWlCLENBQUM7SUFDN0MsQ0FBQztJQUFDLE1BQU0sQ0FBQztRQUNQLE9BQU8sSUFBSSxDQUFDO0lBQ2QsQ0FBQztBQUNILENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyIvLyBDb3B5cmlnaHQgMjAyNiBQaXBlbGluZSBCdWlsZGVyIENvbnRyaWJ1dG9yc1xuLy8gU1BEWC1MaWNlbnNlLUlkZW50aWZpZXI6IEFwYWNoZS0yLjBcblxuaW50ZXJmYWNlIFRva2VuUGF5bG9hZCB7XG4gIHJvbGU/OiBzdHJpbmc7XG4gIGlzQWRtaW4/OiBib29sZWFuO1xuICBzdWI/OiBzdHJpbmc7XG4gIG9yZ2FuaXphdGlvbklkPzogc3RyaW5nO1xuICAvKiogU3RhbmRhcmQgSldUIGV4cGlyeSBjbGFpbSwgaW4gc2Vjb25kcyBzaW5jZSBlcG9jaC4gKi9cbiAgZXhwPzogbnVtYmVyO1xuICAvKiogU3RhbmRhcmQgSldUIGlzc3VlZC1hdCBjbGFpbSwgaW4gc2Vjb25kcyBzaW5jZSBlcG9jaC4gKi9cbiAgaWF0PzogbnVtYmVyO1xufVxuXG4vKipcbiAqIERlY29kZSBhIEpXVCB0b2tlbiBwYXlsb2FkIHdpdGhvdXQgc2lnbmF0dXJlIHZlcmlmaWNhdGlvbi5cbiAqXG4gKiBTRUNVUklUWTogTk8gc2lnbmF0dXJlIHZlcmlmaWNhdGlvbi4gVGhlIGRlY29kZWQgcGF5bG9hZCBNVVNUIE5PVCBiZSB1c2VkXG4gKiBmb3IgYXV0aG9yaXphdGlvbiBkZWNpc2lvbnMg4oCUIHRoZSBzZXJ2ZXIgdmFsaWRhdGVzIHRoZSB0b2tlbiBvbiBldmVyeSBBUElcbiAqIGNhbGwuIFVzZSBvbmx5IGZvciBhZHZpc29yeSBVWCBoaW50cy5cbiAqL1xuZXhwb3J0IGZ1bmN0aW9uIGRlY29kZVRva2VuUGF5bG9hZCh0b2tlbjogc3RyaW5nKTogVG9rZW5QYXlsb2FkIHwgbnVsbCB7XG4gIHRyeSB7XG4gICAgY29uc3QgcGFydHMgPSB0b2tlbi5zcGxpdCgnLicpO1xuICAgIGlmIChwYXJ0cy5sZW5ndGggIT09IDMgfHwgIXBhcnRzWzFdKSByZXR1cm4gbnVsbDtcbiAgICBjb25zdCBwYXlsb2FkID0gQnVmZmVyLmZyb20ocGFydHNbMV0sICdiYXNlNjR1cmwnKS50b1N0cmluZygndXRmLTgnKTtcbiAgICByZXR1cm4gSlNPTi5wYXJzZShwYXlsb2FkKSBhcyBUb2tlblBheWxvYWQ7XG4gIH0gY2F0Y2gge1xuICAgIHJldHVybiBudWxsO1xuICB9XG59XG4iXX0=
@@ -18,4 +18,18 @@ export declare function getSecretArn(secretName: string, options: SecretsOptions
18
18
  * Get the value of a secret.
19
19
  */
20
20
  export declare function getSecretValue(secretName: string, options: SecretsOptions): Promise<string>;
21
+ /** Minimal secret summary returned by listSecrets. */
22
+ export interface SecretSummary {
23
+ name: string;
24
+ arn: string;
25
+ description?: string;
26
+ lastChangedDate?: Date;
27
+ }
28
+ /**
29
+ * List secrets matching a name prefix. Pages internally up to `maxPages` (default 10).
30
+ *
31
+ * @param namePrefix - Filter by `Name` prefix (case-sensitive). Falls back to
32
+ * listing all secrets when omitted (use sparingly).
33
+ */
34
+ export declare function listSecrets(namePrefix: string | undefined, options: SecretsOptions, maxPages?: number): Promise<SecretSummary[]>;
21
35
  //# sourceMappingURL=aws-secrets.d.ts.map
@@ -1 +1 @@
1
- {"version":3,"file":"aws-secrets.d.ts","sourceRoot":"","sources":["../../src/utils/aws-secrets.ts"],"names":[],"mappings":"AAaA;;GAEG;AACH,MAAM,WAAW,cAAc;IAC7B,MAAM,CAAC,EAAE,MAAM,CAAC;IAChB,OAAO,CAAC,EAAE,MAAM,CAAC;CAClB;AAWD;;;GAGG;AACH,wBAAsB,YAAY,CAChC,UAAU,EAAE,MAAM,EAClB,WAAW,EAAE,MAAM,EACnB,WAAW,EAAE,MAAM,EACnB,OAAO,EAAE,cAAc,GACtB,OAAO,CAAC,IAAI,CAAC,CA2Bf;AAED;;GAEG;AACH,wBAAsB,YAAY,CAChC,UAAU,EAAE,MAAM,EAClB,OAAO,EAAE,cAAc,GACtB,OAAO,CAAC,MAAM,CAAC,CAMjB;AAED;;GAEG;AACH,wBAAsB,cAAc,CAClC,UAAU,EAAE,MAAM,EAClB,OAAO,EAAE,cAAc,GACtB,OAAO,CAAC,MAAM,CAAC,CASjB"}
1
+ {"version":3,"file":"aws-secrets.d.ts","sourceRoot":"","sources":["../../src/utils/aws-secrets.ts"],"names":[],"mappings":"AAcA;;GAEG;AACH,MAAM,WAAW,cAAc;IAC7B,MAAM,CAAC,EAAE,MAAM,CAAC;IAChB,OAAO,CAAC,EAAE,MAAM,CAAC;CAClB;AAWD;;;GAGG;AACH,wBAAsB,YAAY,CAChC,UAAU,EAAE,MAAM,EAClB,WAAW,EAAE,MAAM,EACnB,WAAW,EAAE,MAAM,EACnB,OAAO,EAAE,cAAc,GACtB,OAAO,CAAC,IAAI,CAAC,CA2Bf;AAED;;GAEG;AACH,wBAAsB,YAAY,CAChC,UAAU,EAAE,MAAM,EAClB,OAAO,EAAE,cAAc,GACtB,OAAO,CAAC,MAAM,CAAC,CAMjB;AAED;;GAEG;AACH,wBAAsB,cAAc,CAClC,UAAU,EAAE,MAAM,EAClB,OAAO,EAAE,cAAc,GACtB,OAAO,CAAC,MAAM,CAAC,CASjB;AAED,sDAAsD;AACtD,MAAM,WAAW,aAAa;IAC5B,IAAI,EAAE,MAAM,CAAC;IACb,GAAG,EAAE,MAAM,CAAC;IACZ,WAAW,CAAC,EAAE,MAAM,CAAC;IACrB,eAAe,CAAC,EAAE,IAAI,CAAC;CACxB;AAED;;;;;GAKG;AACH,wBAAsB,WAAW,CAC/B,UAAU,EAAE,MAAM,GAAG,SAAS,EAC9B,OAAO,EAAE,cAAc,EACvB,QAAQ,SAAK,GACZ,OAAO,CAAC,aAAa,EAAE,CAAC,CA0B1B"}
@@ -5,6 +5,7 @@ Object.defineProperty(exports, "__esModule", { value: true });
5
5
  exports.upsertSecret = upsertSecret;
6
6
  exports.getSecretArn = getSecretArn;
7
7
  exports.getSecretValue = getSecretValue;
8
+ exports.listSecrets = listSecrets;
8
9
  const client_secrets_manager_1 = require("@aws-sdk/client-secrets-manager");
9
10
  const output_utils_1 = require("./output-utils");
10
11
  function createClient(options) {
@@ -71,4 +72,38 @@ async function getSecretValue(secretName, options) {
71
72
  }
72
73
  return response.SecretString;
73
74
  }
74
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYXdzLXNlY3JldHMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvdXRpbHMvYXdzLXNlY3JldHMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IjtBQUFBLCtDQUErQztBQUMvQyxzQ0FBc0M7O0FBaUN0QyxvQ0FnQ0M7QUFLRCxvQ0FTQztBQUtELHdDQVlDO0FBOUZELDRFQU95QztBQUN6QyxpREFBeUQ7QUFVekQsU0FBUyxZQUFZLENBQUMsT0FBdUI7SUFDM0MsT0FBTyxJQUFJLDZDQUFvQixDQUFDO1FBQzlCLE1BQU0sRUFBRSxPQUFPLENBQUMsTUFBTSxJQUFJLE9BQU8sQ0FBQyxHQUFHLENBQUMsVUFBVSxJQUFJLE9BQU8sQ0FBQyxHQUFHLENBQUMsa0JBQWtCO1FBQ2xGLEdBQUcsQ0FBQyxPQUFPLENBQUMsT0FBTyxJQUFJO1lBQ3JCLFdBQVcsRUFBRSxTQUFTLEVBQUUsaURBQWlEO1NBQzFFLENBQUM7S0FDSCxDQUFDLENBQUM7QUFDTCxDQUFDO0FBRUQ7OztHQUdHO0FBQ0ksS0FBSyxVQUFVLFlBQVksQ0FDaEMsVUFBa0IsRUFDbEIsV0FBbUIsRUFDbkIsV0FBbUIsRUFDbkIsT0FBdUI7SUFFdkIsTUFBTSxNQUFNLEdBQUcsWUFBWSxDQUFDLE9BQU8sQ0FBQyxDQUFDO0lBRXJDLElBQUksQ0FBQztRQUNILE1BQU0sTUFBTSxDQUFDLElBQUksQ0FBQyxJQUFJLDRDQUFtQixDQUFDO1lBQ3hDLElBQUksRUFBRSxVQUFVO1lBQ2hCLFdBQVcsRUFBRSxXQUFXO1lBQ3hCLFlBQVksRUFBRSxXQUFXO1NBQzFCLENBQUMsQ0FBQyxDQUFDO1FBQ0osSUFBQSwyQkFBWSxFQUFDLG1DQUFtQyxDQUFDLENBQUM7SUFDcEQsQ0FBQztJQUFDLE9BQU8sS0FBSyxFQUFFLENBQUM7UUFDZixNQUFNLE1BQU0sR0FBRyxLQUFLLFlBQVksS0FBSyxDQUFDLENBQUMsQ0FBQyxLQUFLLENBQUMsT0FBTyxDQUFDLENBQUMsQ0FBQyxFQUFFLENBQUM7UUFDM0QsSUFBSSxNQUFNLENBQUMsUUFBUSxDQUFDLHlCQUF5QixDQUFDLElBQUksTUFBTSxDQUFDLFFBQVEsQ0FBQyxnQkFBZ0IsQ0FBQyxFQUFFLENBQUM7WUFDcEYsSUFBQSx3QkFBUyxFQUFDLG9DQUFvQyxDQUFDLENBQUM7WUFDaEQsTUFBTSxNQUFNLENBQUMsSUFBSSxDQUFDLElBQUksOENBQXFCLENBQUM7Z0JBQzFDLFFBQVEsRUFBRSxVQUFVO2dCQUNwQixZQUFZLEVBQUUsV0FBVzthQUMxQixDQUFDLENBQUMsQ0FBQztZQUNKLE1BQU0sTUFBTSxDQUFDLElBQUksQ0FBQyxJQUFJLDRDQUFtQixDQUFDO2dCQUN4QyxRQUFRLEVBQUUsVUFBVTtnQkFDcEIsV0FBVyxFQUFFLFdBQVc7YUFDekIsQ0FBQyxDQUFDLENBQUM7WUFDSixJQUFBLDJCQUFZLEVBQUMsbUNBQW1DLENBQUMsQ0FBQztRQUNwRCxDQUFDO2FBQU0sQ0FBQztZQUNOLE1BQU0sS0FBSyxDQUFDO1FBQ2QsQ0FBQztJQUNILENBQUM7QUFDSCxDQUFDO0FBRUQ7O0dBRUc7QUFDSSxLQUFLLFVBQVUsWUFBWSxDQUNoQyxVQUFrQixFQUNsQixPQUF1QjtJQUV2QixNQUFNLE1BQU0sR0FBRyxZQUFZLENBQUMsT0FBTyxDQUFDLENBQUM7SUFDckMsTUFBTSxRQUFRLEdBQUcsTUFBTSxNQUFNLENBQUMsSUFBSSxDQUFDLElBQUksOENBQXFCLENBQUM7UUFDM0QsUUFBUSxFQUFFLFVBQVU7S0FDckIsQ0FBQyxDQUFDLENBQUM7SUFDSixPQUFPLFFBQVEsQ0FBQyxHQUFHLElBQUksV0FBVyxDQUFDO0FBQ3JDLENBQUM7QUFFRDs7R0FFRztBQUNJLEtBQUssVUFBVSxjQUFjLENBQ2xDLFVBQWtCLEVBQ2xCLE9BQXVCO0lBRXZCLE1BQU0sTUFBTSxHQUFHLFlBQVksQ0FBQyxPQUFPLENBQUMsQ0FBQztJQUNyQyxNQUFNLFFBQVEsR0FBRyxNQUFNLE1BQU0sQ0FBQyxJQUFJLENBQUMsSUFBSSw4Q0FBcUIsQ0FBQztRQUMzRCxRQUFRLEVBQUUsVUFBVTtLQUNyQixDQUFDLENBQUMsQ0FBQztJQUNKLElBQUksQ0FBQyxRQUFRLENBQUMsWUFBWSxFQUFFLENBQUM7UUFDM0IsTUFBTSxJQUFJLEtBQUssQ0FBQyxXQUFXLFVBQVUsWUFBWSxDQUFDLENBQUM7SUFDckQsQ0FBQztJQUNELE9BQU8sUUFBUSxDQUFDLFlBQVksQ0FBQztBQUMvQixDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiLy8gQ29weXJpZ2h0IDIwMjYgUGlwZWxpbmUgQnVpbGRlciBDb250cmlidXRvcnNcbi8vIFNQRFgtTGljZW5zZS1JZGVudGlmaWVyOiBBcGFjaGUtMi4wXG5cbmltcG9ydCB7XG4gIFNlY3JldHNNYW5hZ2VyQ2xpZW50LFxuICBDcmVhdGVTZWNyZXRDb21tYW5kLFxuICBQdXRTZWNyZXRWYWx1ZUNvbW1hbmQsXG4gIFVwZGF0ZVNlY3JldENvbW1hbmQsXG4gIERlc2NyaWJlU2VjcmV0Q29tbWFuZCxcbiAgR2V0U2VjcmV0VmFsdWVDb21tYW5kLFxufSBmcm9tICdAYXdzLXNkay9jbGllbnQtc2VjcmV0cy1tYW5hZ2VyJztcbmltcG9ydCB7IHByaW50SW5mbywgcHJpbnRTdWNjZXNzIH0gZnJvbSAnLi9vdXRwdXQtdXRpbHMnO1xuXG4vKipcbiAqIE9wdGlvbnMgZm9yIFNlY3JldHMgTWFuYWdlciBvcGVyYXRpb25zLlxuICovXG5leHBvcnQgaW50ZXJmYWNlIFNlY3JldHNPcHRpb25zIHtcbiAgcmVnaW9uPzogc3RyaW5nO1xuICBwcm9maWxlPzogc3RyaW5nO1xufVxuXG5mdW5jdGlvbiBjcmVhdGVDbGllbnQob3B0aW9uczogU2VjcmV0c09wdGlvbnMpOiBTZWNyZXRzTWFuYWdlckNsaWVudCB7XG4gIHJldHVybiBuZXcgU2VjcmV0c01hbmFnZXJDbGllbnQoe1xuICAgIHJlZ2lvbjogb3B0aW9ucy5yZWdpb24gfHwgcHJvY2Vzcy5lbnYuQVdTX1JFR0lPTiB8fCBwcm9jZXNzLmVudi5DREtfREVGQVVMVF9SRUdJT04sXG4gICAgLi4uKG9wdGlvbnMucHJvZmlsZSAmJiB7XG4gICAgICBjcmVkZW50aWFsczogdW5kZWZpbmVkLCAvLyBwcm9maWxlIGlzIGhhbmRsZWQgYnkgQVdTIFNESyBjcmVkZW50aWFsIGNoYWluXG4gICAgfSksXG4gIH0pO1xufVxuXG4vKipcbiAqIENyZWF0ZSBvciB1cGRhdGUgYSBzZWNyZXQgaW4gQVdTIFNlY3JldHMgTWFuYWdlci5cbiAqIFRyaWVzIGNyZWF0ZSBmaXJzdDsgaWYgaXQgZXhpc3RzLCB1cGRhdGVzIHRoZSB2YWx1ZSBhbmQgZGVzY3JpcHRpb24uXG4gKi9cbmV4cG9ydCBhc3luYyBmdW5jdGlvbiB1cHNlcnRTZWNyZXQoXG4gIHNlY3JldE5hbWU6IHN0cmluZyxcbiAgc2VjcmV0VmFsdWU6IHN0cmluZyxcbiAgZGVzY3JpcHRpb246IHN0cmluZyxcbiAgb3B0aW9uczogU2VjcmV0c09wdGlvbnMsXG4pOiBQcm9taXNlPHZvaWQ+IHtcbiAgY29uc3QgY2xpZW50ID0gY3JlYXRlQ2xpZW50KG9wdGlvbnMpO1xuXG4gIHRyeSB7XG4gICAgYXdhaXQgY2xpZW50LnNlbmQobmV3IENyZWF0ZVNlY3JldENvbW1hbmQoe1xuICAgICAgTmFtZTogc2VjcmV0TmFtZSxcbiAgICAgIERlc2NyaXB0aW9uOiBkZXNjcmlwdGlvbixcbiAgICAgIFNlY3JldFN0cmluZzogc2VjcmV0VmFsdWUsXG4gICAgfSkpO1xuICAgIHByaW50U3VjY2VzcygnU2VjcmV0IGNyZWF0ZWQgaW4gU2VjcmV0cyBNYW5hZ2VyJyk7XG4gIH0gY2F0Y2ggKGVycm9yKSB7XG4gICAgY29uc3QgZXJyTXNnID0gZXJyb3IgaW5zdGFuY2VvZiBFcnJvciA/IGVycm9yLm1lc3NhZ2UgOiAnJztcbiAgICBpZiAoZXJyTXNnLmluY2x1ZGVzKCdSZXNvdXJjZUV4aXN0c0V4Y2VwdGlvbicpIHx8IGVyck1zZy5pbmNsdWRlcygnYWxyZWFkeSBleGlzdHMnKSkge1xuICAgICAgcHJpbnRJbmZvKCdTZWNyZXQgYWxyZWFkeSBleGlzdHMsIHVwZGF0aW5nLi4uJyk7XG4gICAgICBhd2FpdCBjbGllbnQuc2VuZChuZXcgUHV0U2VjcmV0VmFsdWVDb21tYW5kKHtcbiAgICAgICAgU2VjcmV0SWQ6IHNlY3JldE5hbWUsXG4gICAgICAgIFNlY3JldFN0cmluZzogc2VjcmV0VmFsdWUsXG4gICAgICB9KSk7XG4gICAgICBhd2FpdCBjbGllbnQuc2VuZChuZXcgVXBkYXRlU2VjcmV0Q29tbWFuZCh7XG4gICAgICAgIFNlY3JldElkOiBzZWNyZXROYW1lLFxuICAgICAgICBEZXNjcmlwdGlvbjogZGVzY3JpcHRpb24sXG4gICAgICB9KSk7XG4gICAgICBwcmludFN1Y2Nlc3MoJ1NlY3JldCB1cGRhdGVkIGluIFNlY3JldHMgTWFuYWdlcicpO1xuICAgIH0gZWxzZSB7XG4gICAgICB0aHJvdyBlcnJvcjtcbiAgICB9XG4gIH1cbn1cblxuLyoqXG4gKiBHZXQgdGhlIEFSTiBvZiBhIHNlY3JldC5cbiAqL1xuZXhwb3J0IGFzeW5jIGZ1bmN0aW9uIGdldFNlY3JldEFybihcbiAgc2VjcmV0TmFtZTogc3RyaW5nLFxuICBvcHRpb25zOiBTZWNyZXRzT3B0aW9ucyxcbik6IFByb21pc2U8c3RyaW5nPiB7XG4gIGNvbnN0IGNsaWVudCA9IGNyZWF0ZUNsaWVudChvcHRpb25zKTtcbiAgY29uc3QgcmVzcG9uc2UgPSBhd2FpdCBjbGllbnQuc2VuZChuZXcgRGVzY3JpYmVTZWNyZXRDb21tYW5kKHtcbiAgICBTZWNyZXRJZDogc2VjcmV0TmFtZSxcbiAgfSkpO1xuICByZXR1cm4gcmVzcG9uc2UuQVJOID8/ICcodW5rbm93biknO1xufVxuXG4vKipcbiAqIEdldCB0aGUgdmFsdWUgb2YgYSBzZWNyZXQuXG4gKi9cbmV4cG9ydCBhc3luYyBmdW5jdGlvbiBnZXRTZWNyZXRWYWx1ZShcbiAgc2VjcmV0TmFtZTogc3RyaW5nLFxuICBvcHRpb25zOiBTZWNyZXRzT3B0aW9ucyxcbik6IFByb21pc2U8c3RyaW5nPiB7XG4gIGNvbnN0IGNsaWVudCA9IGNyZWF0ZUNsaWVudChvcHRpb25zKTtcbiAgY29uc3QgcmVzcG9uc2UgPSBhd2FpdCBjbGllbnQuc2VuZChuZXcgR2V0U2VjcmV0VmFsdWVDb21tYW5kKHtcbiAgICBTZWNyZXRJZDogc2VjcmV0TmFtZSxcbiAgfSkpO1xuICBpZiAoIXJlc3BvbnNlLlNlY3JldFN0cmluZykge1xuICAgIHRocm93IG5ldyBFcnJvcihgU2VjcmV0IFwiJHtzZWNyZXROYW1lfVwiIGlzIGVtcHR5YCk7XG4gIH1cbiAgcmV0dXJuIHJlc3BvbnNlLlNlY3JldFN0cmluZztcbn1cbiJdfQ==
75
+ /**
76
+ * List secrets matching a name prefix. Pages internally up to `maxPages` (default 10).
77
+ *
78
+ * @param namePrefix - Filter by `Name` prefix (case-sensitive). Falls back to
79
+ * listing all secrets when omitted (use sparingly).
80
+ */
81
+ async function listSecrets(namePrefix, options, maxPages = 10) {
82
+ const client = createClient(options);
83
+ const out = [];
84
+ let nextToken;
85
+ let page = 0;
86
+ do {
87
+ const response = await client.send(new client_secrets_manager_1.ListSecretsCommand({
88
+ MaxResults: 100,
89
+ NextToken: nextToken,
90
+ ...(namePrefix && {
91
+ Filters: [{ Key: 'name', Values: [namePrefix] }],
92
+ }),
93
+ }));
94
+ for (const s of response.SecretList ?? []) {
95
+ if (!s.Name || !s.ARN)
96
+ continue;
97
+ out.push({
98
+ name: s.Name,
99
+ arn: s.ARN,
100
+ description: s.Description,
101
+ lastChangedDate: s.LastChangedDate,
102
+ });
103
+ }
104
+ nextToken = response.NextToken;
105
+ page++;
106
+ } while (nextToken && page < maxPages);
107
+ return out;
108
+ }
109
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"aws-secrets.js","sourceRoot":"","sources":["../../src/utils/aws-secrets.ts"],"names":[],"mappings":"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","sourcesContent":["// Copyright 2026 Pipeline Builder Contributors\n// SPDX-License-Identifier: Apache-2.0\n\nimport {\n  SecretsManagerClient,\n  CreateSecretCommand,\n  PutSecretValueCommand,\n  UpdateSecretCommand,\n  DescribeSecretCommand,\n  GetSecretValueCommand,\n  ListSecretsCommand,\n} from '@aws-sdk/client-secrets-manager';\nimport { printInfo, printSuccess } from './output-utils';\n\n/**\n * Options for Secrets Manager operations.\n */\nexport interface SecretsOptions {\n  region?: string;\n  profile?: string;\n}\n\nfunction createClient(options: SecretsOptions): SecretsManagerClient {\n  return new SecretsManagerClient({\n    region: options.region || process.env.AWS_REGION || process.env.CDK_DEFAULT_REGION,\n    ...(options.profile && {\n      credentials: undefined, // profile is handled by AWS SDK credential chain\n    }),\n  });\n}\n\n/**\n * Create or update a secret in AWS Secrets Manager.\n * Tries create first; if it exists, updates the value and description.\n */\nexport async function upsertSecret(\n  secretName: string,\n  secretValue: string,\n  description: string,\n  options: SecretsOptions,\n): Promise<void> {\n  const client = createClient(options);\n\n  try {\n    await client.send(new CreateSecretCommand({\n      Name: secretName,\n      Description: description,\n      SecretString: secretValue,\n    }));\n    printSuccess('Secret created in Secrets Manager');\n  } catch (error) {\n    const errMsg = error instanceof Error ? error.message : '';\n    if (errMsg.includes('ResourceExistsException') || errMsg.includes('already exists')) {\n      printInfo('Secret already exists, updating...');\n      await client.send(new PutSecretValueCommand({\n        SecretId: secretName,\n        SecretString: secretValue,\n      }));\n      await client.send(new UpdateSecretCommand({\n        SecretId: secretName,\n        Description: description,\n      }));\n      printSuccess('Secret updated in Secrets Manager');\n    } else {\n      throw error;\n    }\n  }\n}\n\n/**\n * Get the ARN of a secret.\n */\nexport async function getSecretArn(\n  secretName: string,\n  options: SecretsOptions,\n): Promise<string> {\n  const client = createClient(options);\n  const response = await client.send(new DescribeSecretCommand({\n    SecretId: secretName,\n  }));\n  return response.ARN ?? '(unknown)';\n}\n\n/**\n * Get the value of a secret.\n */\nexport async function getSecretValue(\n  secretName: string,\n  options: SecretsOptions,\n): Promise<string> {\n  const client = createClient(options);\n  const response = await client.send(new GetSecretValueCommand({\n    SecretId: secretName,\n  }));\n  if (!response.SecretString) {\n    throw new Error(`Secret \"${secretName}\" is empty`);\n  }\n  return response.SecretString;\n}\n\n/** Minimal secret summary returned by listSecrets. */\nexport interface SecretSummary {\n  name: string;\n  arn: string;\n  description?: string;\n  lastChangedDate?: Date;\n}\n\n/**\n * List secrets matching a name prefix. Pages internally up to `maxPages` (default 10).\n *\n * @param namePrefix - Filter by `Name` prefix (case-sensitive). Falls back to\n *                    listing all secrets when omitted (use sparingly).\n */\nexport async function listSecrets(\n  namePrefix: string | undefined,\n  options: SecretsOptions,\n  maxPages = 10,\n): Promise<SecretSummary[]> {\n  const client = createClient(options);\n  const out: SecretSummary[] = [];\n  let nextToken: string | undefined;\n  let page = 0;\n  do {\n    const response = await client.send(new ListSecretsCommand({\n      MaxResults: 100,\n      NextToken: nextToken,\n      ...(namePrefix && {\n        Filters: [{ Key: 'name', Values: [namePrefix] }],\n      }),\n    }));\n    for (const s of response.SecretList ?? []) {\n      if (!s.Name || !s.ARN) continue;\n      out.push({\n        name: s.Name,\n        arn: s.ARN,\n        description: s.Description,\n        lastChangedDate: s.LastChangedDate,\n      });\n    }\n    nextToken = response.NextToken;\n    page++;\n  } while (nextToken && page < maxPages);\n  return out;\n}\n"]}
package/package.json CHANGED
@@ -30,8 +30,7 @@
30
30
  "@aws-sdk/client-lambda": "3.821.0",
31
31
  "@aws-sdk/client-secrets-manager": "3.821.0",
32
32
  "@aws-sdk/client-sts": "3.821.0",
33
- "@pipeline-builder/pipeline-core": "3.3.4",
34
- "aws-cdk-lib": "2.240.0",
33
+ "aws-cdk-lib": "2.251.0",
35
34
  "axios": "1.13.5",
36
35
  "commander": "14.0.3",
37
36
  "figlet": "1.10.0",
@@ -40,7 +39,8 @@
40
39
  "picocolors": "1.1.1",
41
40
  "progress": "2.0.3",
42
41
  "typescript": "5.9.3",
43
- "yaml": "2.8.2"
42
+ "yaml": "2.8.2",
43
+ "@pipeline-builder/pipeline-core": "3.3.7"
44
44
  },
45
45
  "keywords": [
46
46
  "aws",
@@ -70,13 +70,20 @@
70
70
  "engines": {
71
71
  "node": ">= 24.14.0"
72
72
  },
73
+ "devEngines": {
74
+ "packageManager": {
75
+ "name": "pnpm",
76
+ "version": ">=10",
77
+ "onFail": "ignore"
78
+ }
79
+ },
73
80
  "license": "Apache-2.0",
74
81
  "homepage": "https://mwashburn160.github.io/pipeline-builder/",
75
82
  "publishConfig": {
76
83
  "access": "public",
77
84
  "registry": "https://registry.npmjs.org/"
78
85
  },
79
- "version": "3.3.5",
86
+ "version": "3.3.7",
80
87
  "bugs": {
81
88
  "url": "https://github.com/mwashburn160/pipeline-builder/issues"
82
89
  },