@cloudsnorkel/cdk-github-runners 0.15.0 → 0.15.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (42) hide show
  1. package/.jsii +533 -327
  2. package/API.md +384 -174
  3. package/README.md +1 -1
  4. package/assets/delete-failed-runner.lambda/index.js +23 -6
  5. package/assets/idle-runner-repear.lambda/index.js +23 -6
  6. package/assets/setup.lambda/index.html +7 -7
  7. package/assets/setup.lambda/index.js +23 -6
  8. package/assets/status.lambda/index.js +23 -6
  9. package/assets/token-retriever.lambda/index.js +23 -6
  10. package/assets/warm-runner-manager.lambda/index.js +23 -6
  11. package/assets/webhook-handler.lambda/index.js +23 -6
  12. package/assets/webhook-redelivery.lambda/index.js +23 -6
  13. package/lib/access.js +1 -1
  14. package/lib/image-builders/api.js +1 -1
  15. package/lib/image-builders/aws-image-builder/base-image.js +2 -2
  16. package/lib/image-builders/aws-image-builder/builder.js +1 -1
  17. package/lib/image-builders/aws-image-builder/deprecated/ami.js +1 -1
  18. package/lib/image-builders/aws-image-builder/deprecated/container.js +1 -1
  19. package/lib/image-builders/aws-image-builder/deprecated/linux-components.js +1 -1
  20. package/lib/image-builders/aws-image-builder/deprecated/windows-components.js +1 -1
  21. package/lib/image-builders/codebuild-deprecated.js +1 -1
  22. package/lib/image-builders/components.js +1 -1
  23. package/lib/image-builders/static.js +1 -1
  24. package/lib/providers/codebuild.d.ts +2 -2
  25. package/lib/providers/codebuild.js +3 -3
  26. package/lib/providers/common.d.ts +47 -3
  27. package/lib/providers/common.js +29 -5
  28. package/lib/providers/composite.js +14 -19
  29. package/lib/providers/ec2.d.ts +4 -2
  30. package/lib/providers/ec2.js +58 -30
  31. package/lib/providers/ecs.d.ts +2 -2
  32. package/lib/providers/ecs.js +8 -8
  33. package/lib/providers/fargate.d.ts +2 -2
  34. package/lib/providers/fargate.js +13 -8
  35. package/lib/providers/lambda.d.ts +2 -2
  36. package/lib/providers/lambda.js +3 -3
  37. package/lib/runner.d.ts +6 -1
  38. package/lib/runner.js +54 -31
  39. package/lib/secrets.js +1 -1
  40. package/lib/warm-runner.d.ts +15 -7
  41. package/lib/warm-runner.js +19 -12
  42. package/package.json +15 -15
@@ -29,8 +29,9 @@ export interface WarmRunnerBaseProps {
29
29
  readonly owner: string;
30
30
  /**
31
31
  * Registration level — must match how your runners are set up in GitHub. Choose
32
- * 'org' for org-wide runners, 'repo' for repo-level. See the setup wizard or
33
- * {@link SETUP_GITHUB.md} for choosing repo vs org.
32
+ * 'org' for org-wide runners, 'repo' for repo-level. See the setup wizard for choosing repo vs org.
33
+ *
34
+ * @see https://github.com/CloudSnorkel/cdk-github-runners/blob/main/SETUP_GITHUB.md
34
35
  *
35
36
  * @default 'repo'
36
37
  */
@@ -64,7 +65,9 @@ export interface ScheduledWarmRunnerProps extends WarmRunnerBaseProps {
64
65
  *
65
66
  * Runners will be provisioned using the specified provider and registered in the specified repository or organization.
66
67
  *
67
- * Registration level must match the one selected during setup. See {@link SETUP_GITHUB.md} for more information on the selection.
68
+ * Registration level must match the one selected during setup.
69
+ *
70
+ * @see https://github.com/CloudSnorkel/cdk-github-runners/blob/main/SETUP_GITHUB.md
68
71
  *
69
72
  * ## Limitations
70
73
  *
@@ -76,7 +79,7 @@ export interface ScheduledWarmRunnerProps extends WarmRunnerBaseProps {
76
79
  * gap in coverage until the retry succeeds. Current retry mechanism has built-in back-off rate and
77
80
  * can be tweaked using `retryOptions`. This will be improved in the future.
78
81
  *
79
- * @example
82
+ * ```typescript
80
83
  * new AlwaysOnWarmRunner(stack, 'AlwaysOnLinux', {
81
84
  * runners,
82
85
  * provider: myProvider,
@@ -84,6 +87,7 @@ export interface ScheduledWarmRunnerProps extends WarmRunnerBaseProps {
84
87
  * owner: 'my-org',
85
88
  * repo: 'my-repo',
86
89
  * });
90
+ * ```
87
91
  */
88
92
  export declare class AlwaysOnWarmRunner extends Construct {
89
93
  /**
@@ -98,7 +102,9 @@ export declare class AlwaysOnWarmRunner extends Construct {
98
102
  *
99
103
  * Runners will be provisioned using the specified provider and registered in the specified repository or organization.
100
104
  *
101
- * Registration level must match the one selected during setup. See {@link SETUP_GITHUB.md} for more information on the selection.
105
+ * Registration level must match the one selected during setup.
106
+ *
107
+ * @see https://github.com/CloudSnorkel/cdk-github-runners/blob/main/SETUP_GITHUB.md
102
108
  *
103
109
  * ## Limitations
104
110
  *
@@ -113,7 +119,7 @@ export declare class AlwaysOnWarmRunner extends Construct {
113
119
  * gap in coverage until the retry succeeds. Current retry mechanism has built-in back-off rate and
114
120
  * can be tweaked using `retryOptions`. This will be improved in the future.
115
121
  *
116
- * @example
122
+ * ```typescript
117
123
  * // Cron: fill at 1pm on weekdays
118
124
  * new ScheduledWarmRunner(stack, 'Business Hours', {
119
125
  * runners,
@@ -124,8 +130,9 @@ export declare class AlwaysOnWarmRunner extends Construct {
124
130
  * schedule: events.Schedule.cron({ hour: '13', minute: '0', weekDay: 'MON-FRI' }),
125
131
  * duration: cdk.Duration.hours(2),
126
132
  * });
133
+ * ```
127
134
  *
128
- * @example
135
+ * ```typescript
129
136
  * // Rate: fill every 12 hours
130
137
  * new ScheduledWarmRunner(stack, 'Every 12 Hours', {
131
138
  * runners,
@@ -136,6 +143,7 @@ export declare class AlwaysOnWarmRunner extends Construct {
136
143
  * schedule: events.Schedule.rate(cdk.Duration.hours(5)),
137
144
  * duration: cdk.Duration.hours(12),
138
145
  * });
146
+ * ```
139
147
  */
140
148
  export declare class ScheduledWarmRunner extends Construct {
141
149
  /**
@@ -11,17 +11,17 @@ const cron_parser_1 = require("cron-parser");
11
11
  function buildWarmRunner(scope, props, schedule, duration, createInitialFill) {
12
12
  const registrationLevel = props.registrationLevel ?? 'repo';
13
13
  if (registrationLevel === 'org' && props.repo) {
14
- throw new Error('Do not specify repo when registrationLevel is \'org\'');
14
+ cdk.Annotations.of(scope).addError('Do not specify repo when registrationLevel is \'org\'');
15
15
  }
16
16
  if (registrationLevel === 'repo' && !props.repo) {
17
- throw new Error('repo is required when registrationLevel is \'repo\'');
17
+ cdk.Annotations.of(scope).addError('repo is required when registrationLevel is \'repo\'');
18
18
  }
19
19
  const providerPath = props.provider.node.path;
20
20
  if (!props.runners.providers.some(p => p.node.path === providerPath)) {
21
- throw new Error(`Provider ${providerPath} is not in the providers list of the GitHubRunners construct`);
21
+ cdk.Annotations.of(scope).addError(`Provider ${providerPath} is not in the providers list of the GitHubRunners construct`);
22
22
  }
23
23
  const labels = props.provider.labels;
24
- const repo = registrationLevel === 'repo' ? props.repo : '';
24
+ const repo = registrationLevel === 'repo' ? (props.repo ?? '') : '';
25
25
  const configHash = crypto.createHash('sha256')
26
26
  .update(JSON.stringify({ providerPath, providerLabels: labels, count: props.count, duration, owner: props.owner, repo }))
27
27
  .digest('hex')
@@ -61,7 +61,9 @@ function buildWarmRunner(scope, props, schedule, duration, createInitialFill) {
61
61
  *
62
62
  * Runners will be provisioned using the specified provider and registered in the specified repository or organization.
63
63
  *
64
- * Registration level must match the one selected during setup. See {@link SETUP_GITHUB.md} for more information on the selection.
64
+ * Registration level must match the one selected during setup.
65
+ *
66
+ * @see https://github.com/CloudSnorkel/cdk-github-runners/blob/main/SETUP_GITHUB.md
65
67
  *
66
68
  * ## Limitations
67
69
  *
@@ -73,7 +75,7 @@ function buildWarmRunner(scope, props, schedule, duration, createInitialFill) {
73
75
  * gap in coverage until the retry succeeds. Current retry mechanism has built-in back-off rate and
74
76
  * can be tweaked using `retryOptions`. This will be improved in the future.
75
77
  *
76
- * @example
78
+ * ```typescript
77
79
  * new AlwaysOnWarmRunner(stack, 'AlwaysOnLinux', {
78
80
  * runners,
79
81
  * provider: myProvider,
@@ -81,6 +83,7 @@ function buildWarmRunner(scope, props, schedule, duration, createInitialFill) {
81
83
  * owner: 'my-org',
82
84
  * repo: 'my-repo',
83
85
  * });
86
+ * ```
84
87
  */
85
88
  class AlwaysOnWarmRunner extends constructs_1.Construct {
86
89
  constructor(scope, id, props) {
@@ -90,7 +93,7 @@ class AlwaysOnWarmRunner extends constructs_1.Construct {
90
93
  }
91
94
  exports.AlwaysOnWarmRunner = AlwaysOnWarmRunner;
92
95
  _a = JSII_RTTI_SYMBOL_1;
93
- AlwaysOnWarmRunner[_a] = { fqn: "@cloudsnorkel/cdk-github-runners.AlwaysOnWarmRunner", version: "0.15.0" };
96
+ AlwaysOnWarmRunner[_a] = { fqn: "@cloudsnorkel/cdk-github-runners.AlwaysOnWarmRunner", version: "0.15.1" };
94
97
  /**
95
98
  * Convert AWS EventBridge cron format to cron-parser format.
96
99
  * AWS: cron(min hour dom month dow year), cron-parser: sec min hour dom month dow
@@ -150,7 +153,9 @@ function getScheduleIntervalSeconds(expressionString) {
150
153
  *
151
154
  * Runners will be provisioned using the specified provider and registered in the specified repository or organization.
152
155
  *
153
- * Registration level must match the one selected during setup. See {@link SETUP_GITHUB.md} for more information on the selection.
156
+ * Registration level must match the one selected during setup.
157
+ *
158
+ * @see https://github.com/CloudSnorkel/cdk-github-runners/blob/main/SETUP_GITHUB.md
154
159
  *
155
160
  * ## Limitations
156
161
  *
@@ -165,7 +170,7 @@ function getScheduleIntervalSeconds(expressionString) {
165
170
  * gap in coverage until the retry succeeds. Current retry mechanism has built-in back-off rate and
166
171
  * can be tweaked using `retryOptions`. This will be improved in the future.
167
172
  *
168
- * @example
173
+ * ```typescript
169
174
  * // Cron: fill at 1pm on weekdays
170
175
  * new ScheduledWarmRunner(stack, 'Business Hours', {
171
176
  * runners,
@@ -176,8 +181,9 @@ function getScheduleIntervalSeconds(expressionString) {
176
181
  * schedule: events.Schedule.cron({ hour: '13', minute: '0', weekDay: 'MON-FRI' }),
177
182
  * duration: cdk.Duration.hours(2),
178
183
  * });
184
+ * ```
179
185
  *
180
- * @example
186
+ * ```typescript
181
187
  * // Rate: fill every 12 hours
182
188
  * new ScheduledWarmRunner(stack, 'Every 12 Hours', {
183
189
  * runners,
@@ -188,6 +194,7 @@ function getScheduleIntervalSeconds(expressionString) {
188
194
  * schedule: events.Schedule.rate(cdk.Duration.hours(5)),
189
195
  * duration: cdk.Duration.hours(12),
190
196
  * });
197
+ * ```
191
198
  */
192
199
  class ScheduledWarmRunner extends constructs_1.Construct {
193
200
  constructor(scope, id, props) {
@@ -206,5 +213,5 @@ class ScheduledWarmRunner extends constructs_1.Construct {
206
213
  }
207
214
  exports.ScheduledWarmRunner = ScheduledWarmRunner;
208
215
  _b = JSII_RTTI_SYMBOL_1;
209
- ScheduledWarmRunner[_b] = { fqn: "@cloudsnorkel/cdk-github-runners.ScheduledWarmRunner", version: "0.15.0" };
210
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"warm-runner.js","sourceRoot":"","sources":["../src/warm-runner.ts"],"names":[],"mappings":"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","sourcesContent":["import * as crypto from 'crypto';\nimport * as cdk from 'aws-cdk-lib';\nimport {\n  aws_events as events,\n  aws_events_targets as events_targets,\n} from 'aws-cdk-lib';\nimport { Construct } from 'constructs';\nimport { CronExpressionParser } from 'cron-parser';\nimport { ICompositeProvider, IRunnerProvider } from './providers';\nimport { GitHubRunners } from './runner';\nimport { WarmRunnerFillPayload } from './warm-runner-manager.lambda';\n\n/**\n * Common properties for warm runner constructs.\n *\n * @internal\n */\nexport interface WarmRunnerBaseProps {\n  /**\n   * The GitHubRunners construct that owns the shared warm runner infrastructure.\n   */\n  readonly runners: GitHubRunners;\n\n  /**\n   * Provider to use. Warm runners bypass the provider selector — they always use\n   * this provider, regardless of job characteristics. Labels cannot be modified.\n   */\n  readonly provider: IRunnerProvider | ICompositeProvider;\n\n  /**\n   * Number of warm runners to maintain.\n   */\n  readonly count: number;\n\n  /**\n   * GitHub owner where runners will be registered (org or user login).\n   */\n  readonly owner: string;\n\n  /**\n   * Registration level — must match how your runners are set up in GitHub. Choose\n   * 'org' for org-wide runners, 'repo' for repo-level. See the setup wizard or\n   * {@link SETUP_GITHUB.md} for choosing repo vs org.\n   *\n   * @default 'repo'\n   */\n  readonly registrationLevel?: 'org' | 'repo';\n\n  /**\n   * Repository name (without owner) where runners will be registered. Required when `registrationLevel` is 'repo'.\n   */\n  readonly repo?: string;\n}\n\n/**\n * Properties for always on warm runners.\n */\nexport interface AlwaysOnWarmRunnerProps extends WarmRunnerBaseProps { }\n\n/**\n * Properties for scheduled warm runners.\n */\nexport interface ScheduledWarmRunnerProps extends WarmRunnerBaseProps {\n  /**\n   * When to start filling the pool (e.g. start of business hours).\n   */\n  readonly schedule: events.Schedule;\n\n  /**\n   * How long the warm runners should be maintained from the fill time (schedule). Defines the end of the\n   * window (schedule time + duration).\n   */\n  readonly duration: cdk.Duration;\n}\n\nfunction buildWarmRunner(scope: Construct, props: WarmRunnerBaseProps, schedule: events.Schedule, duration: number, createInitialFill: boolean) {\n  const registrationLevel = props.registrationLevel ?? 'repo';\n  if (registrationLevel === 'org' && props.repo) {\n    throw new Error('Do not specify repo when registrationLevel is \\'org\\'');\n  }\n  if (registrationLevel === 'repo' && !props.repo) {\n    throw new Error('repo is required when registrationLevel is \\'repo\\'');\n  }\n\n  const providerPath = props.provider.node.path;\n  if (!props.runners.providers.some(p => p.node.path === providerPath)) {\n    throw new Error(`Provider ${providerPath} is not in the providers list of the GitHubRunners construct`);\n  }\n\n  const labels = props.provider.labels;\n\n  const repo = registrationLevel === 'repo' ? props.repo! : '';\n  const configHash = crypto.createHash('sha256')\n    .update(JSON.stringify({ providerPath, providerLabels: labels, count: props.count, duration, owner: props.owner, repo }))\n    .digest('hex')\n    .slice(0, 16);\n\n  const fillPayload: WarmRunnerFillPayload = {\n    action: 'fill' as const,\n    providerPath,\n    providerLabels: labels,\n    count: props.count,\n    duration,\n    owner: props.owner,\n    repo,\n    configHash,\n  };\n\n  const { lambda: managerFn, queue } = props.runners._ensureWarmRunnerInfra();\n  props.runners._registerWarmConfigHash(configHash);\n\n  // Schedule to fill the warm pool (usually daily). Sends to SQS so we get stable messageId for idempotent fills.\n  new events.Rule(scope, 'Schedule', {\n    schedule,\n    targets: [new events_targets.SqsQueue(queue, {\n      message: events.RuleTargetInput.fromObject(fillPayload),\n    })],\n  });\n\n  // Fill the warm pool immediately on deploy (AlwaysOnWarmRunner only).\n  // ScheduledWarmRunner does not get deployment-fill. First fill happens at the next schedule fire.\n  if (createInitialFill) {\n    new cdk.CustomResource(scope, 'Initial Fill', {\n      serviceToken: managerFn.functionArn,\n      resourceType: 'Custom::WarmRunnerFill',\n      properties: fillPayload,\n    });\n  }\n\n  return fillPayload;\n}\n\n/**\n * Warm runners that run 24/7. Fills at midnight UTC and each runner stays alive for 24 hours.\n *\n * Runners will be provisioned using the specified provider and registered in the specified repository or organization.\n *\n * Registration level must match the one selected during setup. See {@link SETUP_GITHUB.md} for more information on the selection.\n *\n * ## Limitations\n *\n * - Jobs will still trigger provisioning of on-demand runners, even if a warm runner ends up being used.\n * - You may briefly see more than `count` runners when changing config or at rotation.\n * - To remove: set `count` to 0, deploy, wait for warm runners to stop, then remove and deploy again.\n *   If you don't follow this procedure, warm runners may linger until they expire.\n * - Provider failures or timeouts (like Lambda provider timing out after 15 minutes) will result in a\n *   gap in coverage until the retry succeeds. Current retry mechanism has built-in back-off rate and\n *   can be tweaked using `retryOptions`. This will be improved in the future.\n *\n * @example\n * new AlwaysOnWarmRunner(stack, 'AlwaysOnLinux', {\n *   runners,\n *   provider: myProvider,\n *   count: 3,\n *   owner: 'my-org',\n *   repo: 'my-repo',\n * });\n */\nexport class AlwaysOnWarmRunner extends Construct {\n  /**\n   * The fill payload for this warm runner configuration.\n   * @internal\n   */\n  public readonly _fillPayload: WarmRunnerFillPayload;\n\n  constructor(scope: Construct, id: string, props: AlwaysOnWarmRunnerProps) {\n    super(scope, id);\n    this._fillPayload = buildWarmRunner(this, props, events.Schedule.cron({ hour: '0', minute: '0' }), cdk.Duration.days(1).toSeconds(), true);\n  }\n}\n\n/**\n * Convert AWS EventBridge cron format to cron-parser format.\n * AWS: cron(min hour dom month dow year), cron-parser: sec min hour dom month dow\n */\nfunction awsCronToParserFormat(expressionString: string): string {\n  const match = expressionString.match(/^cron\\((.+)\\)$/);\n  if (!match) return expressionString;\n  const [, inner] = match;\n  const parts = inner.trim().split(/\\s+/);\n  if (parts.length !== 6) return expressionString;\n  const [minute, hour, dom, month, dow] = parts;\n  return `0 ${minute} ${hour} ${dom} ${month} ${dow}`;\n}\n\n/**\n * Parse AWS EventBridge rate expression and return interval in seconds.\n * Format: rate(value unit) e.g. rate(2 hours), rate(5 minutes), rate(1 day)\n */\nfunction parseRateInterval(expressionString: string): number | undefined {\n  const match = expressionString.match(/^rate\\((\\d+)\\s+(minute|minutes|hour|hours|day|days)\\)$/i);\n  if (!match) return undefined;\n  const value = parseInt(match[1], 10);\n  const unit = match[2].toLowerCase();\n  if (value <= 0) return undefined;\n  const secondsPerUnit: Record<string, number> = {\n    minute: 60,\n    minutes: 60,\n    hour: 3600,\n    hours: 3600,\n    day: 86400,\n    days: 86400,\n  };\n  return value * secondsPerUnit[unit];\n}\n\n/**\n * Get the interval between schedule occurrences in seconds.\n * Supports both cron and rate expressions.\n */\nfunction getScheduleIntervalSeconds(expressionString: string): number | undefined {\n  const rateInterval = parseRateInterval(expressionString);\n  if (rateInterval !== undefined) return rateInterval;\n\n  try {\n    const cronExpression = CronExpressionParser.parse(awsCronToParserFormat(expressionString));\n    const next = cronExpression.take(2);\n    return (next[1].getTime() - next[0].getTime()) / 1000;\n  } catch {\n    return undefined;\n  }\n}\n\n/**\n * Warm runners active during a time window specified by start time (`schedule`) and duration (`duration`).\n *\n * Runners will be provisioned using the specified provider and registered in the specified repository or organization.\n *\n * Registration level must match the one selected during setup. See {@link SETUP_GITHUB.md} for more information on the selection.\n *\n * ## Limitations\n *\n * - **No deployment-fill**: Unlike `AlwaysOnWarmRunner`, scheduled warm runners do not get an initial\n *   fill on deploy. The first fill happens at the next schedule occurrence. If you deploy at 1pm for\n *   a 2pm schedule, runners will not appear until 2pm.\n * - Jobs will still trigger provisioning of on-demand runners, even if a warm runner ends up being used.\n * - You may briefly see more than `count` runners when changing config or at rotation.\n * - To remove: set `count` to 0, deploy, wait for warm runners to stop, then remove and deploy again.\n *   If you don't follow this procedure, warm runners may linger until they expire.\n * - Provider failures or timeouts (like Lambda provider timing out after 15 minutes) will result in a\n *   gap in coverage until the retry succeeds. Current retry mechanism has built-in back-off rate and\n *   can be tweaked using `retryOptions`. This will be improved in the future.\n *\n * @example\n * // Cron: fill at 1pm on weekdays\n * new ScheduledWarmRunner(stack, 'Business Hours', {\n *   runners,\n *   provider: myProvider,\n *   count: 3,\n *   owner: 'my-org',\n *   repo: 'my-repo',\n *   schedule: events.Schedule.cron({ hour: '13', minute: '0', weekDay: 'MON-FRI' }),\n *   duration: cdk.Duration.hours(2),\n * });\n *\n * @example\n * // Rate: fill every 12 hours\n * new ScheduledWarmRunner(stack, 'Every 12 Hours', {\n *   runners,\n *   provider: myProvider,\n *   count: 2,\n *   owner: 'my-org',\n *   repo: 'my-repo',\n *   schedule: events.Schedule.rate(cdk.Duration.hours(5)),\n *   duration: cdk.Duration.hours(12),\n * });\n */\nexport class ScheduledWarmRunner extends Construct {\n  /**\n   * The fill payload for this warm runner configuration.\n   * @internal\n   */\n  public readonly _fillPayload: WarmRunnerFillPayload;\n\n  constructor(scope: Construct, id: string, props: ScheduledWarmRunnerProps) {\n    super(scope, id);\n\n    // make sure the duration is not longer than the interval between next two schedule occurrences\n    const interval = getScheduleIntervalSeconds(props.schedule.expressionString);\n    if (interval !== undefined && interval < props.duration.toSeconds()) {\n      cdk.Annotations.of(this).addError(`ScheduledWarmRunner duration ${props.duration.toHumanString()} is longer than the interval ${cdk.Duration.seconds(interval).toHumanString()} between next two schedule occurrences. This will result in overlapping warm runners at the start of the next schedule occurrence.`);\n    }\n\n    // warn for short interval\n    if (interval !== undefined && interval < cdk.Duration.hours(1).toSeconds()) {\n      cdk.Annotations.of(this).addWarningV2(\n        '@cloudsnorkel/cdk-github-runners:ScheduledWarmRunner.intervalTooShort',\n        `ScheduledWarmRunner interval ${cdk.Duration.seconds(interval).toHumanString()} is less than 1 hour, which may result in more warm runners than expected`,\n      );\n    }\n\n    this._fillPayload = buildWarmRunner(this, props, props.schedule, props.duration.toSeconds(), false);\n  }\n}\n"]}
216
+ ScheduledWarmRunner[_b] = { fqn: "@cloudsnorkel/cdk-github-runners.ScheduledWarmRunner", version: "0.15.1" };
217
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"warm-runner.js","sourceRoot":"","sources":["../src/warm-runner.ts"],"names":[],"mappings":"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","sourcesContent":["import * as crypto from 'crypto';\nimport * as cdk from 'aws-cdk-lib';\nimport {\n  aws_events as events,\n  aws_events_targets as events_targets,\n} from 'aws-cdk-lib';\nimport { Construct } from 'constructs';\nimport { CronExpressionParser } from 'cron-parser';\nimport { ICompositeProvider, IRunnerProvider } from './providers';\nimport { GitHubRunners } from './runner';\nimport { WarmRunnerFillPayload } from './warm-runner-manager.lambda';\n\n/**\n * Common properties for warm runner constructs.\n *\n * @internal\n */\nexport interface WarmRunnerBaseProps {\n  /**\n   * The GitHubRunners construct that owns the shared warm runner infrastructure.\n   */\n  readonly runners: GitHubRunners;\n\n  /**\n   * Provider to use. Warm runners bypass the provider selector — they always use\n   * this provider, regardless of job characteristics. Labels cannot be modified.\n   */\n  readonly provider: IRunnerProvider | ICompositeProvider;\n\n  /**\n   * Number of warm runners to maintain.\n   */\n  readonly count: number;\n\n  /**\n   * GitHub owner where runners will be registered (org or user login).\n   */\n  readonly owner: string;\n\n  /**\n   * Registration level — must match how your runners are set up in GitHub. Choose\n   * 'org' for org-wide runners, 'repo' for repo-level. See the setup wizard for choosing repo vs org.\n   *\n   * @see https://github.com/CloudSnorkel/cdk-github-runners/blob/main/SETUP_GITHUB.md\n   *\n   * @default 'repo'\n   */\n  readonly registrationLevel?: 'org' | 'repo';\n\n  /**\n   * Repository name (without owner) where runners will be registered. Required when `registrationLevel` is 'repo'.\n   */\n  readonly repo?: string;\n}\n\n/**\n * Properties for always on warm runners.\n */\nexport interface AlwaysOnWarmRunnerProps extends WarmRunnerBaseProps { }\n\n/**\n * Properties for scheduled warm runners.\n */\nexport interface ScheduledWarmRunnerProps extends WarmRunnerBaseProps {\n  /**\n   * When to start filling the pool (e.g. start of business hours).\n   */\n  readonly schedule: events.Schedule;\n\n  /**\n   * How long the warm runners should be maintained from the fill time (schedule). Defines the end of the\n   * window (schedule time + duration).\n   */\n  readonly duration: cdk.Duration;\n}\n\nfunction buildWarmRunner(scope: Construct, props: WarmRunnerBaseProps, schedule: events.Schedule, duration: number, createInitialFill: boolean) {\n  const registrationLevel = props.registrationLevel ?? 'repo';\n  if (registrationLevel === 'org' && props.repo) {\n    cdk.Annotations.of(scope).addError('Do not specify repo when registrationLevel is \\'org\\'');\n  }\n  if (registrationLevel === 'repo' && !props.repo) {\n    cdk.Annotations.of(scope).addError('repo is required when registrationLevel is \\'repo\\'');\n  }\n\n  const providerPath = props.provider.node.path;\n  if (!props.runners.providers.some(p => p.node.path === providerPath)) {\n    cdk.Annotations.of(scope).addError(`Provider ${providerPath} is not in the providers list of the GitHubRunners construct`);\n  }\n\n  const labels = props.provider.labels;\n\n  const repo = registrationLevel === 'repo' ? (props.repo ?? '') : '';\n  const configHash = crypto.createHash('sha256')\n    .update(JSON.stringify({ providerPath, providerLabels: labels, count: props.count, duration, owner: props.owner, repo }))\n    .digest('hex')\n    .slice(0, 16);\n\n  const fillPayload: WarmRunnerFillPayload = {\n    action: 'fill' as const,\n    providerPath,\n    providerLabels: labels,\n    count: props.count,\n    duration,\n    owner: props.owner,\n    repo,\n    configHash,\n  };\n\n  const { lambda: managerFn, queue } = props.runners._ensureWarmRunnerInfra();\n  props.runners._registerWarmConfigHash(configHash);\n\n  // Schedule to fill the warm pool (usually daily). Sends to SQS so we get stable messageId for idempotent fills.\n  new events.Rule(scope, 'Schedule', {\n    schedule,\n    targets: [new events_targets.SqsQueue(queue, {\n      message: events.RuleTargetInput.fromObject(fillPayload),\n    })],\n  });\n\n  // Fill the warm pool immediately on deploy (AlwaysOnWarmRunner only).\n  // ScheduledWarmRunner does not get deployment-fill. First fill happens at the next schedule fire.\n  if (createInitialFill) {\n    new cdk.CustomResource(scope, 'Initial Fill', {\n      serviceToken: managerFn.functionArn,\n      resourceType: 'Custom::WarmRunnerFill',\n      properties: fillPayload,\n    });\n  }\n\n  return fillPayload;\n}\n\n/**\n * Warm runners that run 24/7. Fills at midnight UTC and each runner stays alive for 24 hours.\n *\n * Runners will be provisioned using the specified provider and registered in the specified repository or organization.\n *\n * Registration level must match the one selected during setup.\n *\n * @see https://github.com/CloudSnorkel/cdk-github-runners/blob/main/SETUP_GITHUB.md\n *\n * ## Limitations\n *\n * - Jobs will still trigger provisioning of on-demand runners, even if a warm runner ends up being used.\n * - You may briefly see more than `count` runners when changing config or at rotation.\n * - To remove: set `count` to 0, deploy, wait for warm runners to stop, then remove and deploy again.\n *   If you don't follow this procedure, warm runners may linger until they expire.\n * - Provider failures or timeouts (like Lambda provider timing out after 15 minutes) will result in a\n *   gap in coverage until the retry succeeds. Current retry mechanism has built-in back-off rate and\n *   can be tweaked using `retryOptions`. This will be improved in the future.\n *\n * ```typescript\n * new AlwaysOnWarmRunner(stack, 'AlwaysOnLinux', {\n *   runners,\n *   provider: myProvider,\n *   count: 3,\n *   owner: 'my-org',\n *   repo: 'my-repo',\n * });\n * ```\n */\nexport class AlwaysOnWarmRunner extends Construct {\n  /**\n   * The fill payload for this warm runner configuration.\n   * @internal\n   */\n  public readonly _fillPayload: WarmRunnerFillPayload;\n\n  constructor(scope: Construct, id: string, props: AlwaysOnWarmRunnerProps) {\n    super(scope, id);\n    this._fillPayload = buildWarmRunner(this, props, events.Schedule.cron({ hour: '0', minute: '0' }), cdk.Duration.days(1).toSeconds(), true);\n  }\n}\n\n/**\n * Convert AWS EventBridge cron format to cron-parser format.\n * AWS: cron(min hour dom month dow year), cron-parser: sec min hour dom month dow\n */\nfunction awsCronToParserFormat(expressionString: string): string {\n  const match = expressionString.match(/^cron\\((.+)\\)$/);\n  if (!match) return expressionString;\n  const [, inner] = match;\n  const parts = inner.trim().split(/\\s+/);\n  if (parts.length !== 6) return expressionString;\n  const [minute, hour, dom, month, dow] = parts;\n  return `0 ${minute} ${hour} ${dom} ${month} ${dow}`;\n}\n\n/**\n * Parse AWS EventBridge rate expression and return interval in seconds.\n * Format: rate(value unit) e.g. rate(2 hours), rate(5 minutes), rate(1 day)\n */\nfunction parseRateInterval(expressionString: string): number | undefined {\n  const match = expressionString.match(/^rate\\((\\d+)\\s+(minute|minutes|hour|hours|day|days)\\)$/i);\n  if (!match) return undefined;\n  const value = parseInt(match[1], 10);\n  const unit = match[2].toLowerCase();\n  if (value <= 0) return undefined;\n  const secondsPerUnit: Record<string, number> = {\n    minute: 60,\n    minutes: 60,\n    hour: 3600,\n    hours: 3600,\n    day: 86400,\n    days: 86400,\n  };\n  return value * secondsPerUnit[unit];\n}\n\n/**\n * Get the interval between schedule occurrences in seconds.\n * Supports both cron and rate expressions.\n */\nfunction getScheduleIntervalSeconds(expressionString: string): number | undefined {\n  const rateInterval = parseRateInterval(expressionString);\n  if (rateInterval !== undefined) return rateInterval;\n\n  try {\n    const cronExpression = CronExpressionParser.parse(awsCronToParserFormat(expressionString));\n    const next = cronExpression.take(2);\n    return (next[1].getTime() - next[0].getTime()) / 1000;\n  } catch {\n    return undefined;\n  }\n}\n\n/**\n * Warm runners active during a time window specified by start time (`schedule`) and duration (`duration`).\n *\n * Runners will be provisioned using the specified provider and registered in the specified repository or organization.\n *\n * Registration level must match the one selected during setup.\n *\n * @see https://github.com/CloudSnorkel/cdk-github-runners/blob/main/SETUP_GITHUB.md\n *\n * ## Limitations\n *\n * - **No deployment-fill**: Unlike `AlwaysOnWarmRunner`, scheduled warm runners do not get an initial\n *   fill on deploy. The first fill happens at the next schedule occurrence. If you deploy at 1pm for\n *   a 2pm schedule, runners will not appear until 2pm.\n * - Jobs will still trigger provisioning of on-demand runners, even if a warm runner ends up being used.\n * - You may briefly see more than `count` runners when changing config or at rotation.\n * - To remove: set `count` to 0, deploy, wait for warm runners to stop, then remove and deploy again.\n *   If you don't follow this procedure, warm runners may linger until they expire.\n * - Provider failures or timeouts (like Lambda provider timing out after 15 minutes) will result in a\n *   gap in coverage until the retry succeeds. Current retry mechanism has built-in back-off rate and\n *   can be tweaked using `retryOptions`. This will be improved in the future.\n *\n * ```typescript\n * // Cron: fill at 1pm on weekdays\n * new ScheduledWarmRunner(stack, 'Business Hours', {\n *   runners,\n *   provider: myProvider,\n *   count: 3,\n *   owner: 'my-org',\n *   repo: 'my-repo',\n *   schedule: events.Schedule.cron({ hour: '13', minute: '0', weekDay: 'MON-FRI' }),\n *   duration: cdk.Duration.hours(2),\n * });\n * ```\n *\n * ```typescript\n * // Rate: fill every 12 hours\n * new ScheduledWarmRunner(stack, 'Every 12 Hours', {\n *   runners,\n *   provider: myProvider,\n *   count: 2,\n *   owner: 'my-org',\n *   repo: 'my-repo',\n *   schedule: events.Schedule.rate(cdk.Duration.hours(5)),\n *   duration: cdk.Duration.hours(12),\n * });\n * ```\n */\nexport class ScheduledWarmRunner extends Construct {\n  /**\n   * The fill payload for this warm runner configuration.\n   * @internal\n   */\n  public readonly _fillPayload: WarmRunnerFillPayload;\n\n  constructor(scope: Construct, id: string, props: ScheduledWarmRunnerProps) {\n    super(scope, id);\n\n    // make sure the duration is not longer than the interval between next two schedule occurrences\n    const interval = getScheduleIntervalSeconds(props.schedule.expressionString);\n    if (interval !== undefined && interval < props.duration.toSeconds()) {\n      cdk.Annotations.of(this).addError(`ScheduledWarmRunner duration ${props.duration.toHumanString()} is longer than the interval ${cdk.Duration.seconds(interval).toHumanString()} between next two schedule occurrences. This will result in overlapping warm runners at the start of the next schedule occurrence.`);\n    }\n\n    // warn for short interval\n    if (interval !== undefined && interval < cdk.Duration.hours(1).toSeconds()) {\n      cdk.Annotations.of(this).addWarningV2(\n        '@cloudsnorkel/cdk-github-runners:ScheduledWarmRunner.intervalTooShort',\n        `ScheduledWarmRunner interval ${cdk.Duration.seconds(interval).toHumanString()} is less than 1 hour, which may result in more warm runners than expected`,\n      );\n    }\n\n    this._fillPayload = buildWarmRunner(this, props, props.schedule, props.duration.toSeconds(), false);\n  }\n}\n"]}
package/package.json CHANGED
@@ -72,17 +72,17 @@
72
72
  "organization": false
73
73
  },
74
74
  "devDependencies": {
75
- "@aws-sdk/client-cloudformation": "^3.1000.0",
76
- "@aws-sdk/client-codebuild": "^3.1000.0",
77
- "@aws-sdk/client-ec2": "^3.1000.0",
78
- "@aws-sdk/client-ecr": "^3.1000.0",
79
- "@aws-sdk/client-imagebuilder": "^3.1000.0",
80
- "@aws-sdk/client-lambda": "^3.1000.0",
81
- "@aws-sdk/client-secrets-manager": "^3.1000.0",
82
- "@aws-sdk/client-sfn": "^3.1000.0",
83
- "@aws-sdk/client-sns": "^3.1000.0",
84
- "@aws-sdk/client-sqs": "^3.1004.0",
85
- "@aws-sdk/client-ssm": "^3.1000.0",
75
+ "@aws-sdk/client-cloudformation": "^3.1014.0",
76
+ "@aws-sdk/client-codebuild": "^3.1014.0",
77
+ "@aws-sdk/client-ec2": "^3.1014.0",
78
+ "@aws-sdk/client-ecr": "^3.1014.0",
79
+ "@aws-sdk/client-imagebuilder": "^3.1014.0",
80
+ "@aws-sdk/client-lambda": "^3.1014.0",
81
+ "@aws-sdk/client-secrets-manager": "^3.1014.0",
82
+ "@aws-sdk/client-sfn": "^3.1014.0",
83
+ "@aws-sdk/client-sns": "^3.1014.0",
84
+ "@aws-sdk/client-sqs": "^3.1014.0",
85
+ "@aws-sdk/client-ssm": "^3.1014.0",
86
86
  "@octokit/auth-app": "^8.2.0",
87
87
  "@octokit/core": "^7.0.6",
88
88
  "@octokit/request-error": "^7.1.0",
@@ -92,7 +92,7 @@
92
92
  "@tsconfig/svelte": "^5",
93
93
  "@types/aws-lambda": "^8.10.161",
94
94
  "@types/jest": "^29",
95
- "@types/node": "^25.3.3",
95
+ "@types/node": "^25.5.0",
96
96
  "@typescript-eslint/eslint-plugin": "^8",
97
97
  "@typescript-eslint/parser": "^8",
98
98
  "aws-cdk": "^2",
@@ -100,7 +100,7 @@
100
100
  "bootstrap": "^5.2.0",
101
101
  "commit-and-tag-version": "^12",
102
102
  "constructs": "10.5.1",
103
- "esbuild": "^0.27.3",
103
+ "esbuild": "^0.27.4",
104
104
  "eslint": "^9",
105
105
  "eslint-import-resolver-typescript": "^2.7.1",
106
106
  "eslint-plugin-import": "^2.32.0",
@@ -112,7 +112,7 @@
112
112
  "jsii-docgen": "^10.5.0",
113
113
  "jsii-pacmak": "^1.127.0",
114
114
  "jsii-rosetta": "~5.9.0",
115
- "projen": "^0.99.17",
115
+ "projen": "^0.99.21",
116
116
  "sass": "^1.54.0",
117
117
  "svelte": "^5",
118
118
  "svelte-check": "^4",
@@ -150,7 +150,7 @@
150
150
  "publishConfig": {
151
151
  "access": "public"
152
152
  },
153
- "version": "0.15.0",
153
+ "version": "0.15.1",
154
154
  "jest": {
155
155
  "coverageProvider": "v8",
156
156
  "testMatch": [