aws-ec2-instance-running-scheduler 3.0.10 → 3.1.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.
- package/.jsii +4 -5
- package/README.md +7 -6
- package/assets/funcs/running-scheduler.lambda/index.js +343 -21
- package/lib/constructs/ec2-instance-running-scheduler.js +1 -1
- package/lib/funcs/running-scheduler.lambda.d.ts +1 -1
- package/lib/funcs/running-scheduler.lambda.js +84 -14
- package/lib/stacks/ec2-instance-running-schedule-stack.js +1 -1
- package/package.json +39 -30
package/.jsii
CHANGED
|
@@ -2,7 +2,6 @@
|
|
|
2
2
|
"author": {
|
|
3
3
|
"email": "yicr@users.noreply.github.com",
|
|
4
4
|
"name": "yicr",
|
|
5
|
-
"organization": true,
|
|
6
5
|
"roles": [
|
|
7
6
|
"author"
|
|
8
7
|
]
|
|
@@ -8441,7 +8440,7 @@
|
|
|
8441
8440
|
"stability": "stable"
|
|
8442
8441
|
},
|
|
8443
8442
|
"homepage": "https://github.com/gammarers-aws-cdk-constructs/aws-ec2-instance-running-scheduler.git",
|
|
8444
|
-
"jsiiVersion": "5.9.
|
|
8443
|
+
"jsiiVersion": "5.9.37 (build 5176c0d)",
|
|
8445
8444
|
"keywords": [
|
|
8446
8445
|
"auto",
|
|
8447
8446
|
"aws",
|
|
@@ -8463,7 +8462,7 @@
|
|
|
8463
8462
|
},
|
|
8464
8463
|
"name": "aws-ec2-instance-running-scheduler",
|
|
8465
8464
|
"readme": {
|
|
8466
|
-
"markdown": "# AWS EC2 Instance Running Scheduler\n\n[](https://github.com/gammarers-aws-cdk-constructs/aws-ec2-instance-running-scheduler/blob/main/LICENSE)\n[](https://www.npmjs.com/package/aws-ec2-instance-running-scheduler)\n[](https://github.com/gammarers-aws-cdk-constructs/aws-ec2-instance-running-scheduler/actions/workflows/release.yml)\n[](https://github.com/gammarers-aws-cdk-constructs/aws-ec2-instance-running-scheduler/releases)\n\n[](https://constructs.dev/packages/aws-ec2-instance-running-scheduler)\n\nAWS CDK construct that starts and stops EC2 instances on a cron schedule using **EventBridge Scheduler** and a **Durable Execution Lambda**. The handler discovers instances with the **Resource Groups Tagging API**, runs start/stop and **polls until the instance reaches the target state** (durable `step` / `wait`), processes **multiple instances in parallel** (bounded concurrency), and posts **Slack** summary and per-instance thread messages using a secret from **Secrets Manager
|
|
8465
|
+
"markdown": "# AWS EC2 Instance Running Scheduler\n\n[](https://github.com/gammarers-aws-cdk-constructs/aws-ec2-instance-running-scheduler/blob/main/LICENSE)\n[](https://www.npmjs.com/package/aws-ec2-instance-running-scheduler)\n[](https://github.com/gammarers-aws-cdk-constructs/aws-ec2-instance-running-scheduler/actions/workflows/release.yml)\n[](https://github.com/gammarers-aws-cdk-constructs/aws-ec2-instance-running-scheduler/releases)\n\n[](https://constructs.dev/packages/aws-ec2-instance-running-scheduler)\n\nAWS CDK construct that starts and stops EC2 instances on a cron schedule using **EventBridge Scheduler** and a **Durable Execution Lambda**. The handler discovers instances with the **Resource Groups Tagging API**, runs start/stop and **polls until the instance reaches the target state** (durable `step` / `wait`), processes **multiple instances in parallel** (bounded concurrency), and posts **Slack** summary and per-instance thread messages using a secret from **Secrets Manager**. The Lambda emits **structured application logs** (invocation, EC2 transitions, Slack steps, completion) alongside JSON platform logs.\n\n## Features\n\n- **Tag-based targeting** – Select EC2 instances by tag key and values (e.g. `Schedule` / `YES`) via `tag:GetResources`.\n- **EventBridge Scheduler** – Separate cron rules for start and stop, with per-rule timezone (`aws-cdk-lib` `TimeZone`).\n- **Durable Lambda** – One Lambda with AWS Lambda Durable Execution (`step`, `wait`, `map`, child contexts per instance) for long-running workflows without Step Functions.\n- **Stable-state polling** – After start/stop, the function waits and re-describes instances until `running` (start mode) or `stopped` (stop mode), or until a terminal error.\n- **Slack notifications** – Required for a successful run: parent message plus threaded updates per instance; credentials come from Secrets Manager (`token` and `channel` JSON). The secret name is passed as **`SLACK_SECRET_NAME`** on the function (from `secrets.slackSecretName`) and validated at runtime via **safe-env-getter**.\n- **Structured logging** – The handler uses the durable execution **`ctx.logger`** for traceable JSON application logs (e.g. invocation, describe/start/stop/wait loops, target count, Slack posts, errors before instance state failures).\n- **Scheduling toggle** – Enable or disable both schedules without removing the stack (`enableScheduling`).\n- **Configurable schedules** – Optional cron overrides for start and stop (`minute`, `hour`, `week`, `timezone`); sensible defaults if omitted.\n- **IAM and observability** – EC2 and tagging API permissions, Slack secret read grant, **Parameters and Secrets Lambda Extension** (configured on the function for secret access patterns), JSON logging with configurable system/application levels, and a dedicated log group (construct defaults).\n\n## Installation\n\n**npm**\n\n```bash\nnpm install aws-ec2-instance-running-scheduler\n```\n\n**yarn**\n\n```bash\nyarn add aws-ec2-instance-running-scheduler\n```\n\n**pnpm**\n\n```bash\npnpm add aws-ec2-instance-running-scheduler\n```\n\n## Usage\n\nUse the **construct** `EC2InstanceRunningScheduler` when embedding the scheduler in an existing stack or other CDK scope.\n\n```typescript\nimport * as cdk from 'aws-cdk-lib';\nimport { TimeZone } from 'aws-cdk-lib';\nimport { EC2InstanceRunningScheduler } from 'aws-ec2-instance-running-scheduler';\n\nconst app = new cdk.App();\nconst stack = new cdk.Stack(app, 'MyStack');\n\nnew EC2InstanceRunningScheduler(stack, 'EC2InstanceRunningScheduler', {\n targetResource: {\n tagKey: 'Schedule',\n tagValues: ['YES'],\n },\n secrets: {\n slackSecretName: 'my-slack-secret',\n },\n startSchedule: {\n timezone: TimeZone.ASIA_TOKYO,\n minute: '55',\n hour: '8',\n week: 'MON-FRI',\n },\n stopSchedule: {\n timezone: TimeZone.ASIA_TOKYO,\n minute: '5',\n hour: '19',\n week: 'MON-FRI',\n },\n enableScheduling: true,\n});\n```\n\nUse the **stack** `EC2InstanceRunningScheduleStack` when deploying the scheduler as its own stack. It accepts the **same scheduler options** as the construct (plus standard `StackProps` such as `env`).\n\n```typescript\nimport * as cdk from 'aws-cdk-lib';\nimport { TimeZone } from 'aws-cdk-lib';\nimport { EC2InstanceRunningScheduleStack } from 'aws-ec2-instance-running-scheduler';\n\nconst app = new cdk.App();\n\nnew EC2InstanceRunningScheduleStack(app, 'EC2InstanceRunningScheduleStack', {\n targetResource: {\n tagKey: 'Schedule',\n tagValues: ['YES'],\n },\n secrets: {\n slackSecretName: 'my-slack-secret',\n },\n startSchedule: {\n timezone: TimeZone.ASIA_TOKYO,\n minute: '55',\n hour: '8',\n week: 'MON-FRI',\n },\n stopSchedule: {\n timezone: TimeZone.ASIA_TOKYO,\n minute: '5',\n hour: '19',\n week: 'MON-FRI',\n },\n enableScheduling: true,\n});\n```\n\nEventBridge Scheduler invokes the Lambda with `Params.TagKey`, `Params.TagValues`, and `Params.Mode` (`Start` or `Stop`); the construct wires this for you. The function environment includes **`SLACK_SECRET_NAME`** set to `secrets.slackSecretName`.\n\n## Options\n\nThese options apply to **`EC2InstanceRunningScheduler`** and to **`EC2InstanceRunningScheduleStack`** (stack props include them alongside `StackProps`).\n\n| Option | Type | Required | Description |\n|--------|------|----------|-------------|\n| `targetResource` | `TargetResource` | Yes | Tag key and values used to select EC2 instances. |\n| `secrets` | `Secrets` | Yes | Identifies the Secrets Manager secret used for Slack (`slackSecretName`). |\n| `startSchedule` | `Schedule` | No | Cron for starting instances (default: `50 7 ? * MON-FRI *` in `Etc/UTC`). |\n| `stopSchedule` | `Schedule` | No | Cron for stopping instances (default: `5 19 ? * MON-FRI *` in `Etc/UTC`). |\n| `enableScheduling` | `boolean` | No | Whether both scheduler rules are enabled (default: `true`). |\n\n### TargetResource\n\n- `tagKey` – Tag key used to select instances (e.g. `Schedule`).\n- `tagValues` – Tag values that must match (e.g. `['YES']`).\n\n### Schedule\n\n- `timezone` – `TimeZone` from `aws-cdk-lib` (e.g. `TimeZone.ASIA_TOKYO`, `TimeZone.ETC_UTC`).\n- `minute` – Cron minute (`0`–`59`).\n- `hour` – Cron hour (`0`–`23`).\n- `week` – Cron day-of-week field (e.g. `MON-FRI`).\n\n### Secrets\n\n- `slackSecretName` – Name of the AWS Secrets Manager secret. The Lambda expects JSON with **`token`** (Slack bot token) and **`channel`** (channel ID or name for `chat.postMessage`). The construct sets **`SLACK_SECRET_NAME`** on the function to this value; the handler reads and validates it at runtime (via **safe-env-getter**).\n\n## Requirements\n\n- **Node.js** ≥ 20.0.0 (for developing or synthesizing CDK apps that depend on this package).\n- **aws-cdk-lib** ^2.232.0 and **constructs** ^10.5.1 (peer dependencies).\n- **AWS** – EventBridge Scheduler; Lambda with **Durable Execution** (requires **Node.js 22+** in the deployment region—runtime is the latest Node.js available in the region per the construct), a **live alias**, **Parameters and Secrets Lambda Extension**; EC2 (describe/start/stop); Resource Groups Tagging API; Secrets Manager. The deployed function uses **arm64**, Durable Execution–compatible IAM and settings, and a bundled handler that loads secrets via **aws-lambda-secret-fetcher**.\n\n## License\n\nThis project is licensed under the Apache-2.0 License.\n"
|
|
8467
8466
|
},
|
|
8468
8467
|
"repository": {
|
|
8469
8468
|
"type": "git",
|
|
@@ -8969,6 +8968,6 @@
|
|
|
8969
8968
|
"symbolId": "src/constructs/ec2-instance-running-scheduler:TargetResource"
|
|
8970
8969
|
}
|
|
8971
8970
|
},
|
|
8972
|
-
"version": "3.
|
|
8973
|
-
"fingerprint": "
|
|
8971
|
+
"version": "3.1.1",
|
|
8972
|
+
"fingerprint": "Kh8cXJf3D57ceMbHhtDN1XBQDcLRg5cCROqbixbTBcM="
|
|
8974
8973
|
}
|
package/README.md
CHANGED
|
@@ -7,7 +7,7 @@
|
|
|
7
7
|
|
|
8
8
|
[](https://constructs.dev/packages/aws-ec2-instance-running-scheduler)
|
|
9
9
|
|
|
10
|
-
AWS CDK construct that starts and stops EC2 instances on a cron schedule using **EventBridge Scheduler** and a **Durable Execution Lambda**. The handler discovers instances with the **Resource Groups Tagging API**, runs start/stop and **polls until the instance reaches the target state** (durable `step` / `wait`), processes **multiple instances in parallel** (bounded concurrency), and posts **Slack** summary and per-instance thread messages using a secret from **Secrets Manager**.
|
|
10
|
+
AWS CDK construct that starts and stops EC2 instances on a cron schedule using **EventBridge Scheduler** and a **Durable Execution Lambda**. The handler discovers instances with the **Resource Groups Tagging API**, runs start/stop and **polls until the instance reaches the target state** (durable `step` / `wait`), processes **multiple instances in parallel** (bounded concurrency), and posts **Slack** summary and per-instance thread messages using a secret from **Secrets Manager**. The Lambda emits **structured application logs** (invocation, EC2 transitions, Slack steps, completion) alongside JSON platform logs.
|
|
11
11
|
|
|
12
12
|
## Features
|
|
13
13
|
|
|
@@ -15,10 +15,11 @@ AWS CDK construct that starts and stops EC2 instances on a cron schedule using *
|
|
|
15
15
|
- **EventBridge Scheduler** – Separate cron rules for start and stop, with per-rule timezone (`aws-cdk-lib` `TimeZone`).
|
|
16
16
|
- **Durable Lambda** – One Lambda with AWS Lambda Durable Execution (`step`, `wait`, `map`, child contexts per instance) for long-running workflows without Step Functions.
|
|
17
17
|
- **Stable-state polling** – After start/stop, the function waits and re-describes instances until `running` (start mode) or `stopped` (stop mode), or until a terminal error.
|
|
18
|
-
- **Slack notifications** – Required for a successful run: parent message plus threaded updates per instance; credentials come from Secrets Manager (`token` and `channel` JSON).
|
|
18
|
+
- **Slack notifications** – Required for a successful run: parent message plus threaded updates per instance; credentials come from Secrets Manager (`token` and `channel` JSON). The secret name is passed as **`SLACK_SECRET_NAME`** on the function (from `secrets.slackSecretName`) and validated at runtime via **safe-env-getter**.
|
|
19
|
+
- **Structured logging** – The handler uses the durable execution **`ctx.logger`** for traceable JSON application logs (e.g. invocation, describe/start/stop/wait loops, target count, Slack posts, errors before instance state failures).
|
|
19
20
|
- **Scheduling toggle** – Enable or disable both schedules without removing the stack (`enableScheduling`).
|
|
20
21
|
- **Configurable schedules** – Optional cron overrides for start and stop (`minute`, `hour`, `week`, `timezone`); sensible defaults if omitted.
|
|
21
|
-
- **IAM and observability** – EC2 and tagging API permissions, Slack secret read grant, JSON logging, and a dedicated log group (construct defaults).
|
|
22
|
+
- **IAM and observability** – EC2 and tagging API permissions, Slack secret read grant, **Parameters and Secrets Lambda Extension** (configured on the function for secret access patterns), JSON logging with configurable system/application levels, and a dedicated log group (construct defaults).
|
|
22
23
|
|
|
23
24
|
## Installation
|
|
24
25
|
|
|
@@ -109,7 +110,7 @@ new EC2InstanceRunningScheduleStack(app, 'EC2InstanceRunningScheduleStack', {
|
|
|
109
110
|
});
|
|
110
111
|
```
|
|
111
112
|
|
|
112
|
-
EventBridge Scheduler invokes the Lambda with `Params.TagKey`, `Params.TagValues`, and `Params.Mode` (`Start` or `Stop`); the construct wires this for you.
|
|
113
|
+
EventBridge Scheduler invokes the Lambda with `Params.TagKey`, `Params.TagValues`, and `Params.Mode` (`Start` or `Stop`); the construct wires this for you. The function environment includes **`SLACK_SECRET_NAME`** set to `secrets.slackSecretName`.
|
|
113
114
|
|
|
114
115
|
## Options
|
|
115
116
|
|
|
@@ -137,13 +138,13 @@ These options apply to **`EC2InstanceRunningScheduler`** and to **`EC2InstanceRu
|
|
|
137
138
|
|
|
138
139
|
### Secrets
|
|
139
140
|
|
|
140
|
-
- `slackSecretName` – Name of the AWS Secrets Manager secret. The Lambda expects JSON with **`token`** (Slack bot token) and **`channel`** (channel ID or name for `chat.postMessage`).
|
|
141
|
+
- `slackSecretName` – Name of the AWS Secrets Manager secret. The Lambda expects JSON with **`token`** (Slack bot token) and **`channel`** (channel ID or name for `chat.postMessage`). The construct sets **`SLACK_SECRET_NAME`** on the function to this value; the handler reads and validates it at runtime (via **safe-env-getter**).
|
|
141
142
|
|
|
142
143
|
## Requirements
|
|
143
144
|
|
|
144
145
|
- **Node.js** ≥ 20.0.0 (for developing or synthesizing CDK apps that depend on this package).
|
|
145
146
|
- **aws-cdk-lib** ^2.232.0 and **constructs** ^10.5.1 (peer dependencies).
|
|
146
|
-
- **AWS** – EventBridge Scheduler; Lambda with **Durable Execution**
|
|
147
|
+
- **AWS** – EventBridge Scheduler; Lambda with **Durable Execution** (requires **Node.js 22+** in the deployment region—runtime is the latest Node.js available in the region per the construct), a **live alias**, **Parameters and Secrets Lambda Extension**; EC2 (describe/start/stop); Resource Groups Tagging API; Secrets Manager. The deployed function uses **arm64**, Durable Execution–compatible IAM and settings, and a bundled handler that loads secrets via **aws-lambda-secret-fetcher**.
|
|
147
148
|
|
|
148
149
|
## License
|
|
149
150
|
|
|
@@ -13500,14 +13500,38 @@ var require_axios = __commonJS({
|
|
|
13500
13500
|
return parsed;
|
|
13501
13501
|
};
|
|
13502
13502
|
var $internals = /* @__PURE__ */ Symbol("internals");
|
|
13503
|
+
var isValidHeaderValue = (value) => !/[\r\n]/.test(value);
|
|
13504
|
+
function assertValidHeaderValue(value, header) {
|
|
13505
|
+
if (value === false || value == null) {
|
|
13506
|
+
return;
|
|
13507
|
+
}
|
|
13508
|
+
if (utils$1.isArray(value)) {
|
|
13509
|
+
value.forEach((v) => assertValidHeaderValue(v, header));
|
|
13510
|
+
return;
|
|
13511
|
+
}
|
|
13512
|
+
if (!isValidHeaderValue(String(value))) {
|
|
13513
|
+
throw new Error(`Invalid character in header content ["${header}"]`);
|
|
13514
|
+
}
|
|
13515
|
+
}
|
|
13503
13516
|
function normalizeHeader(header) {
|
|
13504
13517
|
return header && String(header).trim().toLowerCase();
|
|
13505
13518
|
}
|
|
13519
|
+
function stripTrailingCRLF(str) {
|
|
13520
|
+
let end = str.length;
|
|
13521
|
+
while (end > 0) {
|
|
13522
|
+
const charCode = str.charCodeAt(end - 1);
|
|
13523
|
+
if (charCode !== 10 && charCode !== 13) {
|
|
13524
|
+
break;
|
|
13525
|
+
}
|
|
13526
|
+
end -= 1;
|
|
13527
|
+
}
|
|
13528
|
+
return end === str.length ? str : str.slice(0, end);
|
|
13529
|
+
}
|
|
13506
13530
|
function normalizeValue(value) {
|
|
13507
13531
|
if (value === false || value == null) {
|
|
13508
13532
|
return value;
|
|
13509
13533
|
}
|
|
13510
|
-
return utils$1.isArray(value) ? value.map(normalizeValue) : String(value)
|
|
13534
|
+
return utils$1.isArray(value) ? value.map(normalizeValue) : stripTrailingCRLF(String(value));
|
|
13511
13535
|
}
|
|
13512
13536
|
function parseTokens(str) {
|
|
13513
13537
|
const tokens = /* @__PURE__ */ Object.create(null);
|
|
@@ -13563,6 +13587,7 @@ var require_axios = __commonJS({
|
|
|
13563
13587
|
}
|
|
13564
13588
|
const key = utils$1.findKey(self2, lHeader);
|
|
13565
13589
|
if (!key || self2[key] === void 0 || _rewrite === true || _rewrite === void 0 && self2[key] !== false) {
|
|
13590
|
+
assertValidHeaderValue(_value, _header);
|
|
13566
13591
|
self2[key || _header] = normalizeValue(_value);
|
|
13567
13592
|
}
|
|
13568
13593
|
}
|
|
@@ -13781,7 +13806,7 @@ var require_axios = __commonJS({
|
|
|
13781
13806
|
}
|
|
13782
13807
|
return requestedURL;
|
|
13783
13808
|
}
|
|
13784
|
-
var DEFAULT_PORTS = {
|
|
13809
|
+
var DEFAULT_PORTS$1 = {
|
|
13785
13810
|
ftp: 21,
|
|
13786
13811
|
gopher: 70,
|
|
13787
13812
|
http: 80,
|
|
@@ -13806,7 +13831,7 @@ var require_axios = __commonJS({
|
|
|
13806
13831
|
}
|
|
13807
13832
|
proto = proto.split(":", 1)[0];
|
|
13808
13833
|
hostname = hostname.replace(/:\d*$/, "");
|
|
13809
|
-
port = parseInt(port) || DEFAULT_PORTS[proto] || 0;
|
|
13834
|
+
port = parseInt(port) || DEFAULT_PORTS$1[proto] || 0;
|
|
13810
13835
|
if (!shouldProxy(hostname, port)) {
|
|
13811
13836
|
return "";
|
|
13812
13837
|
}
|
|
@@ -13846,7 +13871,7 @@ var require_axios = __commonJS({
|
|
|
13846
13871
|
function getEnv(key) {
|
|
13847
13872
|
return process.env[key.toLowerCase()] || process.env[key.toUpperCase()] || "";
|
|
13848
13873
|
}
|
|
13849
|
-
var VERSION = "1.
|
|
13874
|
+
var VERSION = "1.15.0";
|
|
13850
13875
|
function parseProtocol(url2) {
|
|
13851
13876
|
const match = /^([-+\w]{1,25})(:?\/\/|:)/.exec(url2);
|
|
13852
13877
|
return match && match[1] || "";
|
|
@@ -14115,6 +14140,81 @@ var require_axios = __commonJS({
|
|
|
14115
14140
|
}, cb);
|
|
14116
14141
|
} : fn;
|
|
14117
14142
|
};
|
|
14143
|
+
var DEFAULT_PORTS = {
|
|
14144
|
+
http: 80,
|
|
14145
|
+
https: 443,
|
|
14146
|
+
ws: 80,
|
|
14147
|
+
wss: 443,
|
|
14148
|
+
ftp: 21
|
|
14149
|
+
};
|
|
14150
|
+
var parseNoProxyEntry = (entry) => {
|
|
14151
|
+
let entryHost = entry;
|
|
14152
|
+
let entryPort = 0;
|
|
14153
|
+
if (entryHost.charAt(0) === "[") {
|
|
14154
|
+
const bracketIndex = entryHost.indexOf("]");
|
|
14155
|
+
if (bracketIndex !== -1) {
|
|
14156
|
+
const host = entryHost.slice(1, bracketIndex);
|
|
14157
|
+
const rest = entryHost.slice(bracketIndex + 1);
|
|
14158
|
+
if (rest.charAt(0) === ":" && /^\d+$/.test(rest.slice(1))) {
|
|
14159
|
+
entryPort = Number.parseInt(rest.slice(1), 10);
|
|
14160
|
+
}
|
|
14161
|
+
return [host, entryPort];
|
|
14162
|
+
}
|
|
14163
|
+
}
|
|
14164
|
+
const firstColon = entryHost.indexOf(":");
|
|
14165
|
+
const lastColon = entryHost.lastIndexOf(":");
|
|
14166
|
+
if (firstColon !== -1 && firstColon === lastColon && /^\d+$/.test(entryHost.slice(lastColon + 1))) {
|
|
14167
|
+
entryPort = Number.parseInt(entryHost.slice(lastColon + 1), 10);
|
|
14168
|
+
entryHost = entryHost.slice(0, lastColon);
|
|
14169
|
+
}
|
|
14170
|
+
return [entryHost, entryPort];
|
|
14171
|
+
};
|
|
14172
|
+
var normalizeNoProxyHost = (hostname) => {
|
|
14173
|
+
if (!hostname) {
|
|
14174
|
+
return hostname;
|
|
14175
|
+
}
|
|
14176
|
+
if (hostname.charAt(0) === "[" && hostname.charAt(hostname.length - 1) === "]") {
|
|
14177
|
+
hostname = hostname.slice(1, -1);
|
|
14178
|
+
}
|
|
14179
|
+
return hostname.replace(/\.+$/, "");
|
|
14180
|
+
};
|
|
14181
|
+
function shouldBypassProxy(location) {
|
|
14182
|
+
let parsed;
|
|
14183
|
+
try {
|
|
14184
|
+
parsed = new URL(location);
|
|
14185
|
+
} catch (_err) {
|
|
14186
|
+
return false;
|
|
14187
|
+
}
|
|
14188
|
+
const noProxy = (process.env.no_proxy || process.env.NO_PROXY || "").toLowerCase();
|
|
14189
|
+
if (!noProxy) {
|
|
14190
|
+
return false;
|
|
14191
|
+
}
|
|
14192
|
+
if (noProxy === "*") {
|
|
14193
|
+
return true;
|
|
14194
|
+
}
|
|
14195
|
+
const port = Number.parseInt(parsed.port, 10) || DEFAULT_PORTS[parsed.protocol.split(":", 1)[0]] || 0;
|
|
14196
|
+
const hostname = normalizeNoProxyHost(parsed.hostname.toLowerCase());
|
|
14197
|
+
return noProxy.split(/[\s,]+/).some((entry) => {
|
|
14198
|
+
if (!entry) {
|
|
14199
|
+
return false;
|
|
14200
|
+
}
|
|
14201
|
+
let [entryHost, entryPort] = parseNoProxyEntry(entry);
|
|
14202
|
+
entryHost = normalizeNoProxyHost(entryHost);
|
|
14203
|
+
if (!entryHost) {
|
|
14204
|
+
return false;
|
|
14205
|
+
}
|
|
14206
|
+
if (entryPort && entryPort !== port) {
|
|
14207
|
+
return false;
|
|
14208
|
+
}
|
|
14209
|
+
if (entryHost.charAt(0) === "*") {
|
|
14210
|
+
entryHost = entryHost.slice(1);
|
|
14211
|
+
}
|
|
14212
|
+
if (entryHost.charAt(0) === ".") {
|
|
14213
|
+
return hostname.endsWith(entryHost);
|
|
14214
|
+
}
|
|
14215
|
+
return hostname === entryHost;
|
|
14216
|
+
});
|
|
14217
|
+
}
|
|
14118
14218
|
function speedometer(samplesCount, min) {
|
|
14119
14219
|
samplesCount = samplesCount || 10;
|
|
14120
14220
|
const bytes = new Array(samplesCount);
|
|
@@ -14368,7 +14468,9 @@ var require_axios = __commonJS({
|
|
|
14368
14468
|
if (!proxy && proxy !== false) {
|
|
14369
14469
|
const proxyUrl = getProxyForUrl(location);
|
|
14370
14470
|
if (proxyUrl) {
|
|
14371
|
-
|
|
14471
|
+
if (!shouldBypassProxy(location)) {
|
|
14472
|
+
proxy = new URL(proxyUrl);
|
|
14473
|
+
}
|
|
14372
14474
|
}
|
|
14373
14475
|
}
|
|
14374
14476
|
if (proxy) {
|
|
@@ -15680,12 +15782,23 @@ var require_axios = __commonJS({
|
|
|
15680
15782
|
if (err instanceof Error) {
|
|
15681
15783
|
let dummy = {};
|
|
15682
15784
|
Error.captureStackTrace ? Error.captureStackTrace(dummy) : dummy = new Error();
|
|
15683
|
-
const stack =
|
|
15785
|
+
const stack = (() => {
|
|
15786
|
+
if (!dummy.stack) {
|
|
15787
|
+
return "";
|
|
15788
|
+
}
|
|
15789
|
+
const firstNewlineIndex = dummy.stack.indexOf("\n");
|
|
15790
|
+
return firstNewlineIndex === -1 ? "" : dummy.stack.slice(firstNewlineIndex + 1);
|
|
15791
|
+
})();
|
|
15684
15792
|
try {
|
|
15685
15793
|
if (!err.stack) {
|
|
15686
15794
|
err.stack = stack;
|
|
15687
|
-
} else if (stack
|
|
15688
|
-
|
|
15795
|
+
} else if (stack) {
|
|
15796
|
+
const firstNewlineIndex = stack.indexOf("\n");
|
|
15797
|
+
const secondNewlineIndex = firstNewlineIndex === -1 ? -1 : stack.indexOf("\n", firstNewlineIndex + 1);
|
|
15798
|
+
const stackWithoutTwoTopLines = secondNewlineIndex === -1 ? "" : stack.slice(secondNewlineIndex + 1);
|
|
15799
|
+
if (!String(err.stack).endsWith(stackWithoutTwoTopLines)) {
|
|
15800
|
+
err.stack += "\n" + stack;
|
|
15801
|
+
}
|
|
15689
15802
|
}
|
|
15690
15803
|
} catch (e) {
|
|
15691
15804
|
}
|
|
@@ -18968,6 +19081,146 @@ var require_lib2 = __commonJS({
|
|
|
18968
19081
|
}
|
|
18969
19082
|
});
|
|
18970
19083
|
|
|
19084
|
+
// node_modules/safe-env-getter/lib/index.js
|
|
19085
|
+
var require_lib3 = __commonJS({
|
|
19086
|
+
"node_modules/safe-env-getter/lib/index.js"(exports2) {
|
|
19087
|
+
"use strict";
|
|
19088
|
+
Object.defineProperty(exports2, "__esModule", { value: true });
|
|
19089
|
+
exports2.SafeEnvGetter = exports2.SafeEnvType = exports2.SafeEnvGetterValidationError = exports2.SafeEnvGetterError = void 0;
|
|
19090
|
+
var SafeEnvGetterError = class extends Error {
|
|
19091
|
+
constructor(message) {
|
|
19092
|
+
super(message);
|
|
19093
|
+
this.name = "SafeEnvGetterError";
|
|
19094
|
+
}
|
|
19095
|
+
};
|
|
19096
|
+
exports2.SafeEnvGetterError = SafeEnvGetterError;
|
|
19097
|
+
var SafeEnvGetterValidationError = class _SafeEnvGetterValidationError extends SafeEnvGetterError {
|
|
19098
|
+
/**
|
|
19099
|
+
* Formats validation errors into a human-readable error message.
|
|
19100
|
+
*/
|
|
19101
|
+
static format(errors) {
|
|
19102
|
+
const lines = errors.map((e) => `- ${e.key}: ${e.message}${e.raw == null ? "" : ` (raw="${e.raw}")`}`);
|
|
19103
|
+
return `Invalid environment variables (${errors.length}):
|
|
19104
|
+
${lines.join("\n")}`;
|
|
19105
|
+
}
|
|
19106
|
+
/**
|
|
19107
|
+
* Creates a new validation error from one or more `SafeEnvError` entries.
|
|
19108
|
+
*/
|
|
19109
|
+
constructor(errors) {
|
|
19110
|
+
const msg = _SafeEnvGetterValidationError.format(errors);
|
|
19111
|
+
super(msg);
|
|
19112
|
+
this.name = "SafeEnvGetterValidationError";
|
|
19113
|
+
this.errors = errors;
|
|
19114
|
+
this.keys = errors.map((e) => e.key);
|
|
19115
|
+
}
|
|
19116
|
+
};
|
|
19117
|
+
exports2.SafeEnvGetterValidationError = SafeEnvGetterValidationError;
|
|
19118
|
+
exports2.SafeEnvType = {
|
|
19119
|
+
/** Spec for a string value. */
|
|
19120
|
+
String: { type: "string" },
|
|
19121
|
+
/** Spec for a numeric value. */
|
|
19122
|
+
Number: { type: "number" },
|
|
19123
|
+
/** Spec for a boolean value (1/true/yes/on → true). */
|
|
19124
|
+
Boolean: { type: "boolean" },
|
|
19125
|
+
/**
|
|
19126
|
+
* Returns a spec that restricts the value to one of the given choices.
|
|
19127
|
+
* @param choices - Allowed string literals.
|
|
19128
|
+
* @returns Enum spec for use with `getEnv`.
|
|
19129
|
+
*/
|
|
19130
|
+
Enum: (choices) => ({ type: "enum", choices })
|
|
19131
|
+
};
|
|
19132
|
+
function getEnv(key, spec = exports2.SafeEnvType.String, options) {
|
|
19133
|
+
const raw = process.env[key];
|
|
19134
|
+
const defaultValue = options?.default;
|
|
19135
|
+
const hasDefault = defaultValue !== void 0;
|
|
19136
|
+
if (raw == null || raw === "") {
|
|
19137
|
+
if (!hasDefault) {
|
|
19138
|
+
throw new SafeEnvGetterValidationError([
|
|
19139
|
+
{ key, message: `Missing required environment variable: ${key}`, raw, kind: "missing" }
|
|
19140
|
+
]);
|
|
19141
|
+
}
|
|
19142
|
+
return defaultValue;
|
|
19143
|
+
}
|
|
19144
|
+
switch (spec.type) {
|
|
19145
|
+
case "number": {
|
|
19146
|
+
const n = Number(raw);
|
|
19147
|
+
if (Number.isNaN(n)) {
|
|
19148
|
+
throw new SafeEnvGetterValidationError([
|
|
19149
|
+
{ key, message: `Env ${key}: expected number, got "${raw}"`, raw, kind: "invalid_number" }
|
|
19150
|
+
]);
|
|
19151
|
+
}
|
|
19152
|
+
return n;
|
|
19153
|
+
}
|
|
19154
|
+
case "boolean":
|
|
19155
|
+
return /^(1|true|yes|on)$/i.test(raw) ? true : false;
|
|
19156
|
+
case "enum":
|
|
19157
|
+
if (!spec.choices.includes(raw)) {
|
|
19158
|
+
throw new SafeEnvGetterValidationError([
|
|
19159
|
+
{ key, message: `Env ${key}: must be one of [${spec.choices.join(", ")}]`, raw, kind: "invalid_enum" }
|
|
19160
|
+
]);
|
|
19161
|
+
}
|
|
19162
|
+
return raw;
|
|
19163
|
+
default:
|
|
19164
|
+
return raw;
|
|
19165
|
+
}
|
|
19166
|
+
}
|
|
19167
|
+
var getEnvs = (schema) => {
|
|
19168
|
+
const envs = {};
|
|
19169
|
+
const errors = [];
|
|
19170
|
+
for (const key of Object.keys(schema)) {
|
|
19171
|
+
const entry = schema[key];
|
|
19172
|
+
const spec = Array.isArray(entry) ? entry[0] : entry;
|
|
19173
|
+
const options = Array.isArray(entry) ? entry[1] : void 0;
|
|
19174
|
+
const raw = process.env[key];
|
|
19175
|
+
const defaultValue = options?.default;
|
|
19176
|
+
const hasDefault = defaultValue !== void 0;
|
|
19177
|
+
if (raw == null || raw === "") {
|
|
19178
|
+
if (!hasDefault) {
|
|
19179
|
+
errors.push({ key, message: `Missing required environment variable: ${key}`, raw, kind: "missing" });
|
|
19180
|
+
continue;
|
|
19181
|
+
}
|
|
19182
|
+
envs[key] = defaultValue;
|
|
19183
|
+
continue;
|
|
19184
|
+
}
|
|
19185
|
+
if (spec.type === "number") {
|
|
19186
|
+
const n = Number(raw);
|
|
19187
|
+
if (Number.isNaN(n)) {
|
|
19188
|
+
errors.push({ key, message: `Env ${key}: expected number, got "${raw}"`, raw, kind: "invalid_number" });
|
|
19189
|
+
continue;
|
|
19190
|
+
}
|
|
19191
|
+
envs[key] = n;
|
|
19192
|
+
continue;
|
|
19193
|
+
}
|
|
19194
|
+
if (spec.type === "boolean") {
|
|
19195
|
+
envs[key] = /^(1|true|yes|on)$/i.test(raw) ? true : false;
|
|
19196
|
+
continue;
|
|
19197
|
+
}
|
|
19198
|
+
if (spec.type === "enum") {
|
|
19199
|
+
if (!spec.choices.includes(raw)) {
|
|
19200
|
+
errors.push({
|
|
19201
|
+
key,
|
|
19202
|
+
message: `Env ${key}: must be one of [${spec.choices.join(", ")}]`,
|
|
19203
|
+
raw,
|
|
19204
|
+
kind: "invalid_enum"
|
|
19205
|
+
});
|
|
19206
|
+
continue;
|
|
19207
|
+
}
|
|
19208
|
+
envs[key] = raw;
|
|
19209
|
+
continue;
|
|
19210
|
+
}
|
|
19211
|
+
envs[key] = raw;
|
|
19212
|
+
}
|
|
19213
|
+
if (errors.length > 0)
|
|
19214
|
+
throw new SafeEnvGetterValidationError(errors);
|
|
19215
|
+
return envs;
|
|
19216
|
+
};
|
|
19217
|
+
exports2.SafeEnvGetter = {
|
|
19218
|
+
getEnv,
|
|
19219
|
+
getEnvs
|
|
19220
|
+
};
|
|
19221
|
+
}
|
|
19222
|
+
});
|
|
19223
|
+
|
|
18971
19224
|
// src/funcs/running-scheduler.lambda.ts
|
|
18972
19225
|
var running_scheduler_lambda_exports = {};
|
|
18973
19226
|
__export(running_scheduler_lambda_exports, {
|
|
@@ -22469,6 +22722,7 @@ var import_client_ec2 = require("@aws-sdk/client-ec2");
|
|
|
22469
22722
|
var import_client_resource_groups_tagging_api = require("@aws-sdk/client-resource-groups-tagging-api");
|
|
22470
22723
|
var import_web_api = __toESM(require_dist4());
|
|
22471
22724
|
var import_aws_lambda_secret_fetcher = __toESM(require_lib2());
|
|
22725
|
+
var import_safe_env_getter = __toESM(require_lib3());
|
|
22472
22726
|
|
|
22473
22727
|
// src/funcs/running-scheduler-predicates.ts
|
|
22474
22728
|
var isDesiredStableState = (mode, currentState) => mode === "Start" && currentState === "running" || mode === "Stop" && currentState === "stopped";
|
|
@@ -22490,6 +22744,13 @@ var processOneResource = async (ctx, targetResource, params, resourceIndex) => {
|
|
|
22490
22744
|
const account = arnParts[4] ?? "";
|
|
22491
22745
|
const region = arnParts[3] ?? "";
|
|
22492
22746
|
const stepPrefix = `resource-${resourceIndex}-${identifier}`;
|
|
22747
|
+
ctx.logger.info("processOneResource: start", {
|
|
22748
|
+
resourceIndex,
|
|
22749
|
+
identifier,
|
|
22750
|
+
region,
|
|
22751
|
+
account,
|
|
22752
|
+
mode: params.Mode
|
|
22753
|
+
});
|
|
22493
22754
|
let loopCount = 0;
|
|
22494
22755
|
let currentState = "";
|
|
22495
22756
|
do {
|
|
@@ -22498,21 +22759,37 @@ var processOneResource = async (ctx, targetResource, params, resourceIndex) => {
|
|
|
22498
22759
|
const out = await ec2.send(new import_client_ec2.DescribeInstancesCommand({ InstanceIds: [identifier] }));
|
|
22499
22760
|
return out.Reservations?.[0]?.Instances?.[0]?.State?.Name ?? "unknown";
|
|
22500
22761
|
});
|
|
22762
|
+
ctx.logger.info("processOneResource: described", {
|
|
22763
|
+
identifier,
|
|
22764
|
+
loopCount,
|
|
22765
|
+
currentState,
|
|
22766
|
+
mode: params.Mode
|
|
22767
|
+
});
|
|
22501
22768
|
const mode = params.Mode;
|
|
22502
22769
|
if (mode === "Start" && currentState === "stopped") {
|
|
22770
|
+
ctx.logger.info("processOneResource: starting instance", { identifier, loopCount });
|
|
22503
22771
|
await ctx.step(`${stepPrefix}-start-${loopCount}`, async () => {
|
|
22504
22772
|
const ec2 = new import_client_ec2.EC2Client({});
|
|
22505
22773
|
await ec2.send(new import_client_ec2.StartInstancesCommand({ InstanceIds: [identifier] }));
|
|
22506
22774
|
});
|
|
22775
|
+
ctx.logger.info("processOneResource: wait after start", {
|
|
22776
|
+
identifier,
|
|
22777
|
+
seconds: STATUS_CHANGE_WAIT_SECONDS
|
|
22778
|
+
});
|
|
22507
22779
|
await ctx.wait({ seconds: STATUS_CHANGE_WAIT_SECONDS });
|
|
22508
22780
|
loopCount += 1;
|
|
22509
22781
|
continue;
|
|
22510
22782
|
}
|
|
22511
22783
|
if (mode === "Stop" && currentState === "running") {
|
|
22784
|
+
ctx.logger.info("processOneResource: stopping instance", { identifier, loopCount });
|
|
22512
22785
|
await ctx.step(`${stepPrefix}-stop-${loopCount}`, async () => {
|
|
22513
22786
|
const ec2 = new import_client_ec2.EC2Client({});
|
|
22514
22787
|
await ec2.send(new import_client_ec2.StopInstancesCommand({ InstanceIds: [identifier] }));
|
|
22515
22788
|
});
|
|
22789
|
+
ctx.logger.info("processOneResource: wait after stop", {
|
|
22790
|
+
identifier,
|
|
22791
|
+
seconds: STATUS_CHANGE_WAIT_SECONDS
|
|
22792
|
+
});
|
|
22516
22793
|
await ctx.wait({ seconds: STATUS_CHANGE_WAIT_SECONDS });
|
|
22517
22794
|
loopCount += 1;
|
|
22518
22795
|
continue;
|
|
@@ -22520,13 +22797,32 @@ var processOneResource = async (ctx, targetResource, params, resourceIndex) => {
|
|
|
22520
22797
|
if (!isDesiredStableState(mode, currentState)) {
|
|
22521
22798
|
const transitioning = mode === "Start" && currentState === "pending" || mode === "Stop" && (currentState === "stopping" || currentState === "shutting-down");
|
|
22522
22799
|
if (transitioning) {
|
|
22800
|
+
ctx.logger.info("processOneResource: wait while transitioning", {
|
|
22801
|
+
identifier,
|
|
22802
|
+
loopCount,
|
|
22803
|
+
currentState,
|
|
22804
|
+
mode,
|
|
22805
|
+
seconds: STATUS_CHANGE_WAIT_SECONDS
|
|
22806
|
+
});
|
|
22523
22807
|
await ctx.wait({ seconds: STATUS_CHANGE_WAIT_SECONDS });
|
|
22524
22808
|
loopCount += 1;
|
|
22525
22809
|
continue;
|
|
22526
22810
|
}
|
|
22811
|
+
ctx.logger.error("processOneResource: unexpected state", {
|
|
22812
|
+
identifier,
|
|
22813
|
+
mode,
|
|
22814
|
+
currentState,
|
|
22815
|
+
loopCount
|
|
22816
|
+
});
|
|
22527
22817
|
throw new Error(`instance status fail: mode=${mode} currentState=${currentState}`);
|
|
22528
22818
|
}
|
|
22529
22819
|
} while (!isDesiredStableState(params.Mode, currentState));
|
|
22820
|
+
ctx.logger.info("processOneResource: reached desired stable state", {
|
|
22821
|
+
identifier,
|
|
22822
|
+
finalState: currentState,
|
|
22823
|
+
mode: params.Mode,
|
|
22824
|
+
loopCount
|
|
22825
|
+
});
|
|
22530
22826
|
return {
|
|
22531
22827
|
identifier,
|
|
22532
22828
|
account,
|
|
@@ -22535,22 +22831,26 @@ var processOneResource = async (ctx, targetResource, params, resourceIndex) => {
|
|
|
22535
22831
|
status: currentState
|
|
22536
22832
|
};
|
|
22537
22833
|
};
|
|
22538
|
-
var handler = withDurableExecution(async (event,
|
|
22834
|
+
var handler = withDurableExecution(async (event, ctx) => {
|
|
22539
22835
|
const params = event.Params;
|
|
22836
|
+
ctx.logger.info("running-scheduler: invocation", {
|
|
22837
|
+
mode: params?.Mode,
|
|
22838
|
+
tagKey: params?.TagKey,
|
|
22839
|
+
tagValueCount: params?.TagValues?.length ?? 0
|
|
22840
|
+
});
|
|
22540
22841
|
if (!params?.TagKey || !params?.TagValues || !params?.Mode) {
|
|
22541
22842
|
throw new Error("Invalid event: Params.TagKey, Params.TagValues, Params.Mode are required.");
|
|
22542
22843
|
}
|
|
22543
|
-
const slackSecretName =
|
|
22544
|
-
|
|
22545
|
-
|
|
22546
|
-
}
|
|
22547
|
-
const slackSecretValue = await context.step("fetch-slack-secret", async () => {
|
|
22844
|
+
const slackSecretName = import_safe_env_getter.SafeEnvGetter.getEnv("SLACK_SECRET_NAME");
|
|
22845
|
+
const slackSecretValue = await ctx.step("fetch-slack-secret", async () => {
|
|
22846
|
+
ctx.logger.info("running-scheduler: fetching Slack secret", { secretName: slackSecretName });
|
|
22548
22847
|
return import_aws_lambda_secret_fetcher.secretFetcher.getSecretValue(slackSecretName);
|
|
22549
22848
|
});
|
|
22849
|
+
ctx.logger.info("running-scheduler: Slack secret loaded");
|
|
22550
22850
|
if (!slackSecretValue?.token || !slackSecretValue?.channel) {
|
|
22551
22851
|
throw new Error("Slack secret must contain token and channel.");
|
|
22552
22852
|
}
|
|
22553
|
-
const targetResources = await
|
|
22853
|
+
const targetResources = await ctx.step("get-target-resources", async () => {
|
|
22554
22854
|
const client2 = new import_client_resource_groups_tagging_api.ResourceGroupsTaggingAPIClient({});
|
|
22555
22855
|
const result = await client2.send(
|
|
22556
22856
|
new import_client_resource_groups_tagging_api.GetResourcesCommand({
|
|
@@ -22558,28 +22858,46 @@ var handler = withDurableExecution(async (event, context) => {
|
|
|
22558
22858
|
TagFilters: [{ Key: params.TagKey, Values: params.TagValues }]
|
|
22559
22859
|
})
|
|
22560
22860
|
);
|
|
22561
|
-
|
|
22861
|
+
const arns = (result.ResourceTagMappingList ?? []).map((m) => m.ResourceARN).filter((arn) => arn != null);
|
|
22862
|
+
ctx.logger.info("running-scheduler: get-target-resources done", { count: arns.length });
|
|
22863
|
+
return arns;
|
|
22562
22864
|
});
|
|
22563
22865
|
if (targetResources.length === 0) {
|
|
22866
|
+
ctx.logger.info("running-scheduler: no matching instances", { tagKey: params.TagKey });
|
|
22564
22867
|
return { status: "TargetResourcesNotFound" };
|
|
22565
22868
|
}
|
|
22566
22869
|
const client = new import_web_api.WebClient(slackSecretValue.token);
|
|
22567
22870
|
const channel = slackSecretValue.channel;
|
|
22568
|
-
|
|
22871
|
+
ctx.logger.info("running-scheduler: posting parent Slack message", {
|
|
22872
|
+
instanceCount: targetResources.length
|
|
22873
|
+
});
|
|
22874
|
+
const slackParentMessageResult = await ctx.step("post-slack-messages", async () => {
|
|
22569
22875
|
return client.chat.postMessage({
|
|
22570
22876
|
channel,
|
|
22571
22877
|
text: `${params.Mode === "Start" ? "\u{1F606} Starts" : "\u{1F971} Stops"} the scheduled EC2 Instance.`
|
|
22572
22878
|
});
|
|
22573
22879
|
});
|
|
22574
|
-
|
|
22880
|
+
ctx.logger.info("running-scheduler: parent Slack message posted", {
|
|
22881
|
+
threadTs: slackParentMessageResult?.ts ?? null
|
|
22882
|
+
});
|
|
22883
|
+
ctx.logger.info("running-scheduler: starting parallel instance processing", {
|
|
22884
|
+
count: targetResources.length,
|
|
22885
|
+
maxConcurrency: 10
|
|
22886
|
+
});
|
|
22887
|
+
const results = await ctx.map(
|
|
22575
22888
|
targetResources,
|
|
22576
22889
|
// async (ctx: DurableContext, targetResource: string, index: number) =>
|
|
22577
22890
|
// ctx.step(`process-resource-${index}`, async () =>
|
|
22578
22891
|
// processOneResource(ctx, targetResource, params, index),
|
|
22579
22892
|
// ),
|
|
22580
|
-
async (
|
|
22581
|
-
return
|
|
22893
|
+
async (mapCtx, targetResource, index) => {
|
|
22894
|
+
return mapCtx.runInChildContext(`resource-${index}`, async (childCtx) => {
|
|
22582
22895
|
const result = await processOneResource(childCtx, targetResource, params, index);
|
|
22896
|
+
childCtx.logger.info("running-scheduler: posting thread Slack message", {
|
|
22897
|
+
index,
|
|
22898
|
+
identifier: result.identifier,
|
|
22899
|
+
status: result.status
|
|
22900
|
+
});
|
|
22583
22901
|
await childCtx.step("post-slack-child-messages", async () => {
|
|
22584
22902
|
const display = getStateDisplay(result.status);
|
|
22585
22903
|
return client.chat.postMessage({
|
|
@@ -22605,6 +22923,10 @@ var handler = withDurableExecution(async (event, context) => {
|
|
|
22605
22923
|
{ maxConcurrency: 10 }
|
|
22606
22924
|
);
|
|
22607
22925
|
const resultList = Array.isArray(results) ? results : [];
|
|
22926
|
+
ctx.logger.info("running-scheduler: completed", {
|
|
22927
|
+
processed: targetResources.length,
|
|
22928
|
+
resultCount: resultList.length
|
|
22929
|
+
});
|
|
22608
22930
|
return {
|
|
22609
22931
|
status: "Completed",
|
|
22610
22932
|
processed: targetResources.length,
|
|
@@ -22634,7 +22956,7 @@ mime-types/index.js:
|
|
|
22634
22956
|
*)
|
|
22635
22957
|
|
|
22636
22958
|
axios/dist/node/axios.cjs:
|
|
22637
|
-
(*! Axios v1.
|
|
22959
|
+
(*! Axios v1.15.0 Copyright (c) 2026 Matt Zabriskie and contributors *)
|
|
22638
22960
|
|
|
22639
22961
|
safe-buffer/index.js:
|
|
22640
22962
|
(*! safe-buffer. MIT License. Feross Aboukhadijeh <https://feross.org/opensource> *)
|
|
@@ -139,5 +139,5 @@ class EC2InstanceRunningScheduler extends constructs_1.Construct {
|
|
|
139
139
|
}
|
|
140
140
|
exports.EC2InstanceRunningScheduler = EC2InstanceRunningScheduler;
|
|
141
141
|
_a = JSII_RTTI_SYMBOL_1;
|
|
142
|
-
EC2InstanceRunningScheduler[_a] = { fqn: "aws-ec2-instance-running-scheduler.EC2InstanceRunningScheduler", version: "3.
|
|
142
|
+
EC2InstanceRunningScheduler[_a] = { fqn: "aws-ec2-instance-running-scheduler.EC2InstanceRunningScheduler", version: "3.1.1" };
|
|
143
143
|
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ec2-instance-running-scheduler.js","sourceRoot":"","sources":["../../src/constructs/ec2-instance-running-scheduler.ts"],"names":[],"mappings":"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","sourcesContent":["import { Duration, RemovalPolicy, TimeZone } from 'aws-cdk-lib';\nimport * as iam from 'aws-cdk-lib/aws-iam';\nimport * as lambda from 'aws-cdk-lib/aws-lambda';\nimport * as logs from 'aws-cdk-lib/aws-logs';\nimport * as scheduler from 'aws-cdk-lib/aws-scheduler';\nimport * as targets from 'aws-cdk-lib/aws-scheduler-targets';\nimport { Secret } from 'aws-cdk-lib/aws-secretsmanager';\nimport { Construct } from 'constructs';\nimport { RunningSchedulerFunction } from '../funcs/running-scheduler-function';\n\n/**\n * Cron-style schedule configuration for start/stop actions.\n */\nexport interface Schedule {\n  /** Time zone for the schedule (e.g. ETC_UTC). */\n  readonly timezone: TimeZone;\n  /** Cron minute (0–59). */\n  readonly minute?: string;\n  /** Cron hour (0–23). */\n  readonly hour?: string;\n  /** Cron day of week (e.g. MON-FRI). */\n  readonly week?: string;\n}\n\n/**\n * Defines which EC2 instances are targeted by tag key and values.\n */\nexport interface TargetResource {\n  /** Tag key used to select instances (e.g. Schedule). */\n  readonly tagKey: string;\n  /** Tag values that match instances to include. */\n  readonly tagValues: string[];\n}\n\n/**\n * Secret identifiers required by the scheduler (e.g. Slack).\n */\nexport interface Secrets {\n  /** Name of the Secrets Manager secret containing Slack token and channel. */\n  readonly slackSecretName: string;\n}\n\n/**\n * Properties for creating an EC2 instance running scheduler.\n */\nexport interface EC2InstanceRunningSchedulerProps {\n  /** Tag-based targeting for EC2 instances to start/stop. */\n  readonly targetResource: TargetResource;\n  /** Whether EventBridge Scheduler rules are enabled. Defaults to true if omitted. */\n  readonly enableScheduling?: boolean;\n  /** Secrets (e.g. Slack) used for notifications. */\n  readonly secrets: Secrets;\n  /** Cron schedule for stopping instances. */\n  readonly stopSchedule?: Schedule;\n  /** Cron schedule for starting instances. */\n  readonly startSchedule?: Schedule;\n}\n\n/**\n * Provisions EventBridge Scheduler rules and a Durable Execution Lambda that start/stop tagged EC2 instances.\n *\n * Each schedule invokes the function with `Params` (`TagKey`, `TagValues`, `Mode`). The function uses\n * the Resource Groups Tagging API and EC2 APIs; Slack notifications use the secret named in {@link Secrets.slackSecretName}.\n */\nexport class EC2InstanceRunningScheduler extends Construct {\n  /**\n   * Defines IAM, logging, two cron schedules (start/stop), and the bundled running-scheduler Lambda (Node.js, Durable Execution).\n   *\n   * @param scope - Parent construct.\n   * @param id - Construct id.\n   * @param props - Target tags, optional cron overrides, Slack secret name, and schedule enable flag.\n   */\n  constructor(scope: Construct, id: string, props: EC2InstanceRunningSchedulerProps) {\n    super(scope, id);\n\n    const slackSecret = Secret.fromSecretNameV2(this, 'SlackSecret', props.secrets.slackSecretName);\n\n    // Durable Functions-based Running Scheduler (previous Step Functions logic implemented in Lambda).\n    // Durable Execution requires Node.js 22+.\n    const runningScheduleFunction = new RunningSchedulerFunction(this, 'RunningSchedulerFunction', {\n      description: 'Starts and stops tagged EC2 instances on EventBridge Scheduler schedules.',\n      architecture: lambda.Architecture.ARM_64,\n      timeout: Duration.minutes(15),\n      memorySize: 512,\n      retryAttempts: 2,\n      durableConfig: {\n        executionTimeout: Duration.hours(2),\n        retentionPeriod: Duration.days(1),\n      },\n      environment: {\n        SLACK_SECRET_NAME: props.secrets.slackSecretName,\n      },\n      paramsAndSecrets: lambda.ParamsAndSecretsLayerVersion.fromVersion(lambda.ParamsAndSecretsVersions.V1_0_103, {\n        cacheSize: 500,\n        logLevel: lambda.ParamsAndSecretsLogLevel.INFO,\n      }),\n      role: new iam.Role(this, 'RunningSchedulerFunctionRole', {\n        description: 'Allows the running scheduler to describe, start, and stop EC2 instances and read Slack secrets.',\n        assumedBy: new iam.ServicePrincipal('lambda.amazonaws.com'),\n        managedPolicies: [\n          iam.ManagedPolicy.fromAwsManagedPolicyName('service-role/AWSLambdaBasicExecutionRole'),\n          iam.ManagedPolicy.fromAwsManagedPolicyName('service-role/AWSLambdaBasicDurableExecutionRolePolicy'),\n        ],\n      }),\n      logGroup: new logs.LogGroup(this, 'RunningSchedulerFunctionLogGroup', {\n        retention: logs.RetentionDays.THREE_MONTHS,\n        removalPolicy: RemovalPolicy.DESTROY,\n      }),\n      loggingFormat: lambda.LoggingFormat.JSON,\n      systemLogLevelV2: lambda.SystemLogLevel.INFO,\n      applicationLogLevelV2: lambda.ApplicationLogLevel.INFO,\n    });\n    runningScheduleFunction.addToRolePolicy(new iam.PolicyStatement({\n      sid: 'GetResources',\n      effect: iam.Effect.ALLOW,\n      actions: [\n        'tag:GetResources',\n      ],\n      resources: ['*'],\n    }));\n    // EC2: describe instances and start/stop by instance id\n    runningScheduleFunction.addToRolePolicy(new iam.PolicyStatement({\n      sid: 'Ec2RunningControl',\n      effect: iam.Effect.ALLOW,\n      actions: [\n        'ec2:DescribeInstances',\n        'ec2:StartInstances',\n        'ec2:StopInstances',\n      ],\n      resources: ['*'],\n    }));\n    // Grant read access to the Slack secret\n    slackSecret.grantRead(runningScheduleFunction);\n\n    // See: https://docs.aws.amazon.com/lambda/latest/dg/durable-getting-started-iac.html\n    const runningScheduleFunctionAlias = runningScheduleFunction.addAlias('live');\n\n    // Whether schedules are enabled (default true unless explicitly disabled).\n    const scheduleEnabled: boolean = (() => {\n      if (props.enableScheduling === undefined || props.enableScheduling) {\n        return true;\n      } else {\n        return false;\n      }\n    })();\n\n    // Durable Functions: Lambda performs tag lookup and instance start/stop in a single run.\n    new scheduler.Schedule(this, 'RunningStartSchedule', {\n      description: 'running start schedule',\n      enabled: scheduleEnabled,\n      schedule: scheduler.ScheduleExpression.cron({\n        minute: props.startSchedule?.minute ?? '50',\n        hour: props.startSchedule?.hour ?? '7',\n        weekDay: props.startSchedule?.week ?? 'MON-FRI',\n        timeZone: props.startSchedule?.timezone ?? TimeZone.ETC_UTC,\n      }),\n      target: new targets.LambdaInvoke(runningScheduleFunctionAlias, {\n        input: scheduler.ScheduleTargetInput.fromObject({\n          Params: {\n            TagKey: props.targetResource.tagKey,\n            TagValues: props.targetResource.tagValues,\n            Mode: 'Start',\n          },\n        }),\n      }),\n    });\n\n    new scheduler.Schedule(this, 'RunningStopSchedule', {\n      description: 'running stop schedule',\n      enabled: scheduleEnabled,\n      schedule: scheduler.ScheduleExpression.cron({\n        minute: props.stopSchedule?.minute ?? '5',\n        hour: props.stopSchedule?.hour ?? '19',\n        weekDay: props.stopSchedule?.week ?? 'MON-FRI',\n        timeZone: props.stopSchedule?.timezone ?? TimeZone.ETC_UTC,\n      }),\n      target: new targets.LambdaInvoke(runningScheduleFunctionAlias, {\n        input: scheduler.ScheduleTargetInput.fromObject({\n          Params: {\n            TagKey: props.targetResource.tagKey,\n            TagValues: props.targetResource.tagValues,\n            Mode: 'Stop',\n          },\n        }),\n      }),\n    });\n  }\n}\n"]}
|
|
@@ -25,7 +25,7 @@ export interface SchedulerEvent {
|
|
|
25
25
|
* and uses durable `step` / `wait` / `map` so the run can resume across suspensions.
|
|
26
26
|
*
|
|
27
27
|
* @param event - Payload from EventBridge Scheduler; must include `Params.TagKey`, `Params.TagValues`, `Params.Mode`.
|
|
28
|
-
* @param
|
|
28
|
+
* @param ctx - Root durable execution context.
|
|
29
29
|
* @returns
|
|
30
30
|
* - `{ status: 'TargetResourcesNotFound' }` when no instances match the tag filter.
|
|
31
31
|
* - `{ status: 'Completed', processed, results }` when instances were handled (`results` entries match {@link processOneResource}).
|
|
@@ -14,6 +14,7 @@ const client_ec2_1 = require("@aws-sdk/client-ec2");
|
|
|
14
14
|
const client_resource_groups_tagging_api_1 = require("@aws-sdk/client-resource-groups-tagging-api");
|
|
15
15
|
const web_api_1 = require("@slack/web-api");
|
|
16
16
|
const aws_lambda_secret_fetcher_1 = require("aws-lambda-secret-fetcher");
|
|
17
|
+
const safe_env_getter_1 = require("safe-env-getter");
|
|
17
18
|
const running_scheduler_predicates_1 = require("./running-scheduler-predicates");
|
|
18
19
|
/** Mapping of EC2 instance state to display name and emoji for Slack. */
|
|
19
20
|
const STATE_LIST = [
|
|
@@ -50,6 +51,13 @@ const processOneResource = async (ctx, targetResource, params, resourceIndex) =>
|
|
|
50
51
|
const account = arnParts[4] ?? '';
|
|
51
52
|
const region = arnParts[3] ?? '';
|
|
52
53
|
const stepPrefix = `resource-${resourceIndex}-${identifier}`;
|
|
54
|
+
ctx.logger.info('processOneResource: start', {
|
|
55
|
+
resourceIndex,
|
|
56
|
+
identifier,
|
|
57
|
+
region,
|
|
58
|
+
account,
|
|
59
|
+
mode: params.Mode,
|
|
60
|
+
});
|
|
53
61
|
let loopCount = 0;
|
|
54
62
|
let currentState = '';
|
|
55
63
|
do {
|
|
@@ -58,21 +66,37 @@ const processOneResource = async (ctx, targetResource, params, resourceIndex) =>
|
|
|
58
66
|
const out = await ec2.send(new client_ec2_1.DescribeInstancesCommand({ InstanceIds: [identifier] }));
|
|
59
67
|
return out.Reservations?.[0]?.Instances?.[0]?.State?.Name ?? 'unknown';
|
|
60
68
|
});
|
|
69
|
+
ctx.logger.info('processOneResource: described', {
|
|
70
|
+
identifier,
|
|
71
|
+
loopCount,
|
|
72
|
+
currentState,
|
|
73
|
+
mode: params.Mode,
|
|
74
|
+
});
|
|
61
75
|
const mode = params.Mode;
|
|
62
76
|
if (mode === 'Start' && currentState === 'stopped') {
|
|
77
|
+
ctx.logger.info('processOneResource: starting instance', { identifier, loopCount });
|
|
63
78
|
await ctx.step(`${stepPrefix}-start-${loopCount}`, async () => {
|
|
64
79
|
const ec2 = new client_ec2_1.EC2Client({});
|
|
65
80
|
await ec2.send(new client_ec2_1.StartInstancesCommand({ InstanceIds: [identifier] }));
|
|
66
81
|
});
|
|
82
|
+
ctx.logger.info('processOneResource: wait after start', {
|
|
83
|
+
identifier,
|
|
84
|
+
seconds: STATUS_CHANGE_WAIT_SECONDS,
|
|
85
|
+
});
|
|
67
86
|
await ctx.wait({ seconds: STATUS_CHANGE_WAIT_SECONDS });
|
|
68
87
|
loopCount += 1;
|
|
69
88
|
continue;
|
|
70
89
|
}
|
|
71
90
|
if (mode === 'Stop' && currentState === 'running') {
|
|
91
|
+
ctx.logger.info('processOneResource: stopping instance', { identifier, loopCount });
|
|
72
92
|
await ctx.step(`${stepPrefix}-stop-${loopCount}`, async () => {
|
|
73
93
|
const ec2 = new client_ec2_1.EC2Client({});
|
|
74
94
|
await ec2.send(new client_ec2_1.StopInstancesCommand({ InstanceIds: [identifier] }));
|
|
75
95
|
});
|
|
96
|
+
ctx.logger.info('processOneResource: wait after stop', {
|
|
97
|
+
identifier,
|
|
98
|
+
seconds: STATUS_CHANGE_WAIT_SECONDS,
|
|
99
|
+
});
|
|
76
100
|
await ctx.wait({ seconds: STATUS_CHANGE_WAIT_SECONDS });
|
|
77
101
|
loopCount += 1;
|
|
78
102
|
continue;
|
|
@@ -81,13 +105,32 @@ const processOneResource = async (ctx, targetResource, params, resourceIndex) =>
|
|
|
81
105
|
const transitioning = (mode === 'Start' && currentState === 'pending') ||
|
|
82
106
|
(mode === 'Stop' && (currentState === 'stopping' || currentState === 'shutting-down'));
|
|
83
107
|
if (transitioning) {
|
|
108
|
+
ctx.logger.info('processOneResource: wait while transitioning', {
|
|
109
|
+
identifier,
|
|
110
|
+
loopCount,
|
|
111
|
+
currentState,
|
|
112
|
+
mode,
|
|
113
|
+
seconds: STATUS_CHANGE_WAIT_SECONDS,
|
|
114
|
+
});
|
|
84
115
|
await ctx.wait({ seconds: STATUS_CHANGE_WAIT_SECONDS });
|
|
85
116
|
loopCount += 1;
|
|
86
117
|
continue;
|
|
87
118
|
}
|
|
119
|
+
ctx.logger.error('processOneResource: unexpected state', {
|
|
120
|
+
identifier,
|
|
121
|
+
mode,
|
|
122
|
+
currentState,
|
|
123
|
+
loopCount,
|
|
124
|
+
});
|
|
88
125
|
throw new Error(`instance status fail: mode=${mode} currentState=${currentState}`);
|
|
89
126
|
}
|
|
90
127
|
} while (!(0, running_scheduler_predicates_1.isDesiredStableState)(params.Mode, currentState));
|
|
128
|
+
ctx.logger.info('processOneResource: reached desired stable state', {
|
|
129
|
+
identifier,
|
|
130
|
+
finalState: currentState,
|
|
131
|
+
mode: params.Mode,
|
|
132
|
+
loopCount,
|
|
133
|
+
});
|
|
91
134
|
return {
|
|
92
135
|
identifier,
|
|
93
136
|
account,
|
|
@@ -104,60 +147,83 @@ const processOneResource = async (ctx, targetResource, params, resourceIndex) =>
|
|
|
104
147
|
* and uses durable `step` / `wait` / `map` so the run can resume across suspensions.
|
|
105
148
|
*
|
|
106
149
|
* @param event - Payload from EventBridge Scheduler; must include `Params.TagKey`, `Params.TagValues`, `Params.Mode`.
|
|
107
|
-
* @param
|
|
150
|
+
* @param ctx - Root durable execution context.
|
|
108
151
|
* @returns
|
|
109
152
|
* - `{ status: 'TargetResourcesNotFound' }` when no instances match the tag filter.
|
|
110
153
|
* - `{ status: 'Completed', processed, results }` when instances were handled (`results` entries match {@link processOneResource}).
|
|
111
154
|
* @throws {Error} If `Params` is invalid, `SLACK_SECRET_NAME` is unset, the Slack secret is incomplete, or instance processing fails.
|
|
112
155
|
*/
|
|
113
|
-
exports.handler = (0, durable_execution_sdk_js_1.withDurableExecution)(async (event,
|
|
156
|
+
exports.handler = (0, durable_execution_sdk_js_1.withDurableExecution)(async (event, ctx) => {
|
|
114
157
|
const params = event.Params;
|
|
158
|
+
ctx.logger.info('running-scheduler: invocation', {
|
|
159
|
+
mode: params?.Mode,
|
|
160
|
+
tagKey: params?.TagKey,
|
|
161
|
+
tagValueCount: params?.TagValues?.length ?? 0,
|
|
162
|
+
});
|
|
115
163
|
if (!params?.TagKey || !params?.TagValues || !params?.Mode) {
|
|
116
164
|
throw new Error('Invalid event: Params.TagKey, Params.TagValues, Params.Mode are required.');
|
|
117
165
|
}
|
|
118
|
-
|
|
119
|
-
|
|
120
|
-
|
|
121
|
-
|
|
122
|
-
const slackSecretValue = await context.step('fetch-slack-secret', async () => {
|
|
166
|
+
// safe get Secrets name from environment variable
|
|
167
|
+
const slackSecretName = safe_env_getter_1.SafeEnvGetter.getEnv('SLACK_SECRET_NAME');
|
|
168
|
+
const slackSecretValue = await ctx.step('fetch-slack-secret', async () => {
|
|
169
|
+
ctx.logger.info('running-scheduler: fetching Slack secret', { secretName: slackSecretName });
|
|
123
170
|
return aws_lambda_secret_fetcher_1.secretFetcher.getSecretValue(slackSecretName);
|
|
124
171
|
});
|
|
172
|
+
ctx.logger.info('running-scheduler: Slack secret loaded');
|
|
125
173
|
if (!slackSecretValue?.token || !slackSecretValue?.channel) {
|
|
126
174
|
throw new Error('Slack secret must contain token and channel.');
|
|
127
175
|
}
|
|
128
|
-
const targetResources = await
|
|
176
|
+
const targetResources = await ctx.step('get-target-resources', async () => {
|
|
129
177
|
const client = new client_resource_groups_tagging_api_1.ResourceGroupsTaggingAPIClient({});
|
|
130
178
|
const result = await client.send(new client_resource_groups_tagging_api_1.GetResourcesCommand({
|
|
131
179
|
ResourceTypeFilters: ['ec2:instance'],
|
|
132
180
|
TagFilters: [{ Key: params.TagKey, Values: params.TagValues }],
|
|
133
181
|
}));
|
|
134
|
-
|
|
182
|
+
const arns = (result.ResourceTagMappingList ?? [])
|
|
135
183
|
.map((m) => m.ResourceARN)
|
|
136
184
|
.filter((arn) => arn != null);
|
|
185
|
+
ctx.logger.info('running-scheduler: get-target-resources done', { count: arns.length });
|
|
186
|
+
return arns;
|
|
137
187
|
});
|
|
138
188
|
if (targetResources.length === 0) {
|
|
189
|
+
ctx.logger.info('running-scheduler: no matching instances', { tagKey: params.TagKey });
|
|
139
190
|
return { status: 'TargetResourcesNotFound' };
|
|
140
191
|
}
|
|
141
192
|
const client = new web_api_1.WebClient(slackSecretValue.token);
|
|
142
193
|
const channel = slackSecretValue.channel;
|
|
194
|
+
ctx.logger.info('running-scheduler: posting parent Slack message', {
|
|
195
|
+
instanceCount: targetResources.length,
|
|
196
|
+
});
|
|
143
197
|
// send slack message
|
|
144
|
-
const slackParentMessageResult = await
|
|
198
|
+
const slackParentMessageResult = await ctx.step('post-slack-messages', async () => {
|
|
145
199
|
return client.chat.postMessage({
|
|
146
200
|
channel,
|
|
147
201
|
text: `${params.Mode === 'Start' ? '😆 Starts' : '🥱 Stops'} the scheduled EC2 Instance.`,
|
|
148
202
|
});
|
|
149
203
|
});
|
|
150
|
-
|
|
204
|
+
ctx.logger.info('running-scheduler: parent Slack message posted', {
|
|
205
|
+
threadTs: slackParentMessageResult?.ts ?? null,
|
|
206
|
+
});
|
|
207
|
+
ctx.logger.info('running-scheduler: starting parallel instance processing', {
|
|
208
|
+
count: targetResources.length,
|
|
209
|
+
maxConcurrency: 10,
|
|
210
|
+
});
|
|
211
|
+
const results = await ctx.map(targetResources,
|
|
151
212
|
// async (ctx: DurableContext, targetResource: string, index: number) =>
|
|
152
213
|
// ctx.step(`process-resource-${index}`, async () =>
|
|
153
214
|
// processOneResource(ctx, targetResource, params, index),
|
|
154
215
|
// ),
|
|
155
|
-
async (
|
|
156
|
-
return
|
|
216
|
+
async (mapCtx, targetResource, index) => {
|
|
217
|
+
return mapCtx.runInChildContext(`resource-${index}`, async (childCtx) => {
|
|
157
218
|
const result = await processOneResource(childCtx, targetResource, params, index);
|
|
158
219
|
// if (result.status === 'skipped') {
|
|
159
220
|
// return result;
|
|
160
221
|
// }
|
|
222
|
+
childCtx.logger.info('running-scheduler: posting thread Slack message', {
|
|
223
|
+
index,
|
|
224
|
+
identifier: result.identifier,
|
|
225
|
+
status: result.status,
|
|
226
|
+
});
|
|
161
227
|
// send slack thread message
|
|
162
228
|
await childCtx.step('post-slack-child-messages', async () => {
|
|
163
229
|
const display = getStateDisplay(result.status);
|
|
@@ -182,10 +248,14 @@ exports.handler = (0, durable_execution_sdk_js_1.withDurableExecution)(async (ev
|
|
|
182
248
|
});
|
|
183
249
|
}, { maxConcurrency: 10 });
|
|
184
250
|
const resultList = Array.isArray(results) ? results : [];
|
|
251
|
+
ctx.logger.info('running-scheduler: completed', {
|
|
252
|
+
processed: targetResources.length,
|
|
253
|
+
resultCount: resultList.length,
|
|
254
|
+
});
|
|
185
255
|
return {
|
|
186
256
|
status: 'Completed',
|
|
187
257
|
processed: targetResources.length,
|
|
188
258
|
results: resultList,
|
|
189
259
|
};
|
|
190
260
|
});
|
|
191
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"running-scheduler.lambda.js","sourceRoot":"","sources":["../../src/funcs/running-scheduler.lambda.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * EC2 Running Scheduler – Durable Functions implementation.\n *\n * Implements the running-control flow using AWS Lambda Durable Execution.\n * Step checkpoints, wait (no charge), and parallel map provide a flow equivalent to Step Functions.\n *\n * @see https://docs.aws.amazon.com/lambda/latest/dg/durable-execution-sdk.html\n */\n\nimport { withDurableExecution, type DurableContext } from '@aws/durable-execution-sdk-js';\nimport {\n  EC2Client,\n  DescribeInstancesCommand,\n  StartInstancesCommand,\n  StopInstancesCommand,\n} from '@aws-sdk/client-ec2';\nimport { GetResourcesCommand, ResourceGroupsTaggingAPIClient } from '@aws-sdk/client-resource-groups-tagging-api';\nimport { WebClient } from '@slack/web-api';\nimport { secretFetcher } from 'aws-lambda-secret-fetcher';\nimport { isDesiredStableState } from './running-scheduler-predicates';\n\n/** Mapping of EC2 instance state to display name and emoji for Slack. */\nconst STATE_LIST = [\n  { name: 'RUNNING', emoji: '😆', state: 'running' },\n  { name: 'STOPPED', emoji: '😴', state: 'stopped' },\n] as const;\n\n/** Seconds to wait between polling instance state after start/stop. */\nconst STATUS_CHANGE_WAIT_SECONDS = 20;\n\n/** Event payload from EventBridge Scheduler invoking this Lambda. */\nexport interface SchedulerEvent {\n  Params: {\n    /** Tag key used to select EC2 instances. */\n    TagKey: string;\n    /** Tag values to match. */\n    TagValues: string[];\n    /** Whether to start or stop instances. */\n    Mode: 'Start' | 'Stop';\n  };\n}\n\n/** Slack credentials and default channel loaded from Secrets Manager (`SLACK_SECRET_NAME`). */\ninterface SlackSecret {\n  /** Slack bot token for the Slack `WebClient`. */\n  token: string;\n  /** Channel ID or name passed to `chat.postMessage`. */\n  channel: string;\n}\n\n/**\n * Returns display name and emoji for an EC2 instance state.\n *\n * @param current - Current instance state (e.g. 'running', 'stopped').\n * @returns Display info or undefined if state is not in STATE_LIST.\n */\nconst getStateDisplay = (current: string): { emoji: string; name: string } | undefined => {\n  const found = STATE_LIST.find((s) => s.state === current);\n  return found ? { emoji: found.emoji, name: found.name } : undefined;\n};\n\n/**\n * Processes one EC2 instance: describes state, issues start/stop when needed, then polls until\n * {@link isDesiredStableState} is satisfied (durable `step` / `wait` between attempts).\n *\n * @param ctx - Durable execution context (child context per instance recommended).\n * @param targetResource - EC2 instance ARN.\n * @param params - Scheduler params (`TagKey`, `TagValues`, `Mode`).\n * @param resourceIndex - Index used in durable step names for this resource.\n * @returns Final resource ARN, EC2 state name, parsed account, region, and instance id.\n * @throws {Error} If the state is neither actionable, transitioning, nor the desired stable state.\n */\nconst processOneResource = async (\n  ctx: DurableContext,\n  targetResource: string,\n  params: SchedulerEvent['Params'],\n  resourceIndex: number,\n): Promise<{ resource: string; status: string; account: string; region: string; identifier: string }> => {\n  const parts = targetResource.split('/');\n  const identifier = parts[parts.length - 1] ?? 'unknown';\n  const arnParts = targetResource.split(':');\n  const account = arnParts[4] ?? '';\n  const region = arnParts[3] ?? '';\n  const stepPrefix = `resource-${resourceIndex}-${identifier}`;\n\n  let loopCount = 0;\n  let currentState = '';\n  do {\n    currentState = await ctx.step(`${stepPrefix}-describe-${loopCount}`, async () => {\n      const ec2 = new EC2Client({});\n      const out = await ec2.send(new DescribeInstancesCommand({ InstanceIds: [identifier] }));\n      return out.Reservations?.[0]?.Instances?.[0]?.State?.Name ?? 'unknown';\n    });\n\n    const mode = params.Mode;\n\n    if (mode === 'Start' && currentState === 'stopped') {\n      await ctx.step(`${stepPrefix}-start-${loopCount}`, async () => {\n        const ec2 = new EC2Client({});\n        await ec2.send(new StartInstancesCommand({ InstanceIds: [identifier] }));\n      });\n      await ctx.wait({ seconds: STATUS_CHANGE_WAIT_SECONDS });\n      loopCount += 1;\n      continue;\n    }\n\n    if (mode === 'Stop' && currentState === 'running') {\n      await ctx.step(`${stepPrefix}-stop-${loopCount}`, async () => {\n        const ec2 = new EC2Client({});\n        await ec2.send(new StopInstancesCommand({ InstanceIds: [identifier] }));\n      });\n      await ctx.wait({ seconds: STATUS_CHANGE_WAIT_SECONDS });\n      loopCount += 1;\n      continue;\n    }\n\n    if (!isDesiredStableState(mode, currentState)) {\n      const transitioning =\n        (mode === 'Start' && currentState === 'pending') ||\n        (mode === 'Stop' && (currentState === 'stopping' || currentState === 'shutting-down'));\n\n      if (transitioning) {\n        await ctx.wait({ seconds: STATUS_CHANGE_WAIT_SECONDS });\n        loopCount += 1;\n        continue;\n      }\n\n      throw new Error(`instance status fail: mode=${mode} currentState=${currentState}`);\n    }\n  } while (!isDesiredStableState(params.Mode, currentState));\n\n  return {\n    identifier,\n    account,\n    region,\n    resource: targetResource,\n    status: currentState,\n  };\n};\n\n/**\n * Durable Lambda entry point for the EC2 running scheduler.\n *\n * Resolves instances via Resource Groups Tagging API, runs {@link processOneResource} for each ARN\n * in parallel (bounded concurrency), posts a parent Slack message and per-instance thread replies,\n * and uses durable `step` / `wait` / `map` so the run can resume across suspensions.\n *\n * @param event - Payload from EventBridge Scheduler; must include `Params.TagKey`, `Params.TagValues`, `Params.Mode`.\n * @param context - Root durable execution context.\n * @returns\n * - `{ status: 'TargetResourcesNotFound' }` when no instances match the tag filter.\n * - `{ status: 'Completed', processed, results }` when instances were handled (`results` entries match {@link processOneResource}).\n * @throws {Error} If `Params` is invalid, `SLACK_SECRET_NAME` is unset, the Slack secret is incomplete, or instance processing fails.\n */\nexport const handler = withDurableExecution(async (event: SchedulerEvent, context: DurableContext) => {\n\n  const params = event.Params;\n\n  if (!params?.TagKey || !params?.TagValues || !params?.Mode) {\n    throw new Error('Invalid event: Params.TagKey, Params.TagValues, Params.Mode are required.');\n  }\n  const slackSecretName = process.env.SLACK_SECRET_NAME;\n  if (!slackSecretName) {\n    throw new Error('missing environment variable SLACK_SECRET_NAME.');\n  }\n  const slackSecretValue = await context.step('fetch-slack-secret', async () => {\n    return secretFetcher.getSecretValue<SlackSecret>(slackSecretName);\n  });\n\n  if (!slackSecretValue?.token || !slackSecretValue?.channel) {\n    throw new Error('Slack secret must contain token and channel.');\n  }\n\n  const targetResources = await context.step('get-target-resources', async () => {\n    const client = new ResourceGroupsTaggingAPIClient({});\n    const result = await client.send(\n      new GetResourcesCommand({\n        ResourceTypeFilters: ['ec2:instance'],\n        TagFilters: [{ Key: params.TagKey, Values: params.TagValues }],\n      }),\n    );\n    return (result.ResourceTagMappingList ?? [])\n      .map((m: { ResourceARN?: string }) => m.ResourceARN)\n      .filter((arn: string | undefined): arn is string => arn != null);\n  });\n\n  if (targetResources.length === 0) {\n    return { status: 'TargetResourcesNotFound' as const };\n  }\n\n  const client = new WebClient(slackSecretValue.token);\n  const channel = slackSecretValue.channel;\n\n  // send slack message\n  const slackParentMessageResult = await context.step('post-slack-messages', async () => {\n    return client.chat.postMessage({\n      channel,\n      text: `${params.Mode === 'Start' ? '😆 Starts' : '🥱 Stops'} the scheduled EC2 Instance.`,\n    });\n  });\n\n  const results = await context.map(\n    targetResources,\n    // async (ctx: DurableContext, targetResource: string, index: number) =>\n    //   ctx.step(`process-resource-${index}`, async () =>\n    //     processOneResource(ctx, targetResource, params, index),\n    //   ),\n    async (ctx: DurableContext, targetResource: string, index: number) => {\n      return ctx.runInChildContext(`resource-${index}`, async (childCtx: DurableContext) => {\n        const result = await processOneResource(childCtx, targetResource, params, index);\n        // if (result.status === 'skipped') {\n        //   return result;\n        // }\n        // send slack thread message\n        await childCtx.step('post-slack-child-messages', async () => {\n          const display = getStateDisplay(result.status);\n\n          return client.chat.postMessage({\n            channel,\n            thread_ts: slackParentMessageResult?.ts,\n            attachments: [\n              {\n                color: '#36a64f',\n                pretext: `${display?.emoji} The status of the EC2 Instance ${result.identifier} changed to ${display?.name} due to the schedule.`,\n                fields: [\n                  { title: 'Account', value: result.account, short: true },\n                  { title: 'Region', value: result.region, short: true },\n                  { title: 'Identifier', value: result.identifier, short: true },\n                  { title: 'Status', value: (display?.name ?? 'Unknown'), short: true },\n                ],\n              },\n            ],\n          });\n        });\n        return result;\n      });\n    },\n    { maxConcurrency: 10 },\n  );\n\n  const resultList = Array.isArray(results) ? results : [];\n  return {\n    status: 'Completed' as const,\n    processed: targetResources.length,\n    results: resultList,\n  };\n});\n"]}
|
|
261
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"running-scheduler.lambda.js","sourceRoot":"","sources":["../../src/funcs/running-scheduler.lambda.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * EC2 Running Scheduler – Durable Functions implementation.\n *\n * Implements the running-control flow using AWS Lambda Durable Execution.\n * Step checkpoints, wait (no charge), and parallel map provide a flow equivalent to Step Functions.\n *\n * @see https://docs.aws.amazon.com/lambda/latest/dg/durable-execution-sdk.html\n */\n\nimport { withDurableExecution, type DurableContext } from '@aws/durable-execution-sdk-js';\nimport {\n  EC2Client,\n  DescribeInstancesCommand,\n  StartInstancesCommand,\n  StopInstancesCommand,\n} from '@aws-sdk/client-ec2';\nimport { GetResourcesCommand, ResourceGroupsTaggingAPIClient } from '@aws-sdk/client-resource-groups-tagging-api';\nimport { WebClient } from '@slack/web-api';\nimport { secretFetcher } from 'aws-lambda-secret-fetcher';\nimport { SafeEnvGetter } from 'safe-env-getter';\nimport { isDesiredStableState } from './running-scheduler-predicates';\n\n/** Mapping of EC2 instance state to display name and emoji for Slack. */\nconst STATE_LIST = [\n  { name: 'RUNNING', emoji: '😆', state: 'running' },\n  { name: 'STOPPED', emoji: '😴', state: 'stopped' },\n] as const;\n\n/** Seconds to wait between polling instance state after start/stop. */\nconst STATUS_CHANGE_WAIT_SECONDS = 20;\n\n/** Event payload from EventBridge Scheduler invoking this Lambda. */\nexport interface SchedulerEvent {\n  Params: {\n    /** Tag key used to select EC2 instances. */\n    TagKey: string;\n    /** Tag values to match. */\n    TagValues: string[];\n    /** Whether to start or stop instances. */\n    Mode: 'Start' | 'Stop';\n  };\n}\n\n/** Slack credentials and default channel loaded from Secrets Manager (`SLACK_SECRET_NAME`). */\ninterface SlackSecret {\n  /** Slack bot token for the Slack `WebClient`. */\n  token: string;\n  /** Channel ID or name passed to `chat.postMessage`. */\n  channel: string;\n}\n\n/**\n * Returns display name and emoji for an EC2 instance state.\n *\n * @param current - Current instance state (e.g. 'running', 'stopped').\n * @returns Display info or undefined if state is not in STATE_LIST.\n */\nconst getStateDisplay = (current: string): { emoji: string; name: string } | undefined => {\n  const found = STATE_LIST.find((s) => s.state === current);\n  return found ? { emoji: found.emoji, name: found.name } : undefined;\n};\n\n/**\n * Processes one EC2 instance: describes state, issues start/stop when needed, then polls until\n * {@link isDesiredStableState} is satisfied (durable `step` / `wait` between attempts).\n *\n * @param ctx - Durable execution context (child context per instance recommended).\n * @param targetResource - EC2 instance ARN.\n * @param params - Scheduler params (`TagKey`, `TagValues`, `Mode`).\n * @param resourceIndex - Index used in durable step names for this resource.\n * @returns Final resource ARN, EC2 state name, parsed account, region, and instance id.\n * @throws {Error} If the state is neither actionable, transitioning, nor the desired stable state.\n */\nconst processOneResource = async (\n  ctx: DurableContext,\n  targetResource: string,\n  params: SchedulerEvent['Params'],\n  resourceIndex: number,\n): Promise<{ resource: string; status: string; account: string; region: string; identifier: string }> => {\n  const parts = targetResource.split('/');\n  const identifier = parts[parts.length - 1] ?? 'unknown';\n  const arnParts = targetResource.split(':');\n  const account = arnParts[4] ?? '';\n  const region = arnParts[3] ?? '';\n  const stepPrefix = `resource-${resourceIndex}-${identifier}`;\n\n  ctx.logger.info('processOneResource: start', {\n    resourceIndex,\n    identifier,\n    region,\n    account,\n    mode: params.Mode,\n  });\n\n  let loopCount = 0;\n  let currentState = '';\n  do {\n    currentState = await ctx.step(`${stepPrefix}-describe-${loopCount}`, async () => {\n      const ec2 = new EC2Client({});\n      const out = await ec2.send(new DescribeInstancesCommand({ InstanceIds: [identifier] }));\n      return out.Reservations?.[0]?.Instances?.[0]?.State?.Name ?? 'unknown';\n    });\n\n    ctx.logger.info('processOneResource: described', {\n      identifier,\n      loopCount,\n      currentState,\n      mode: params.Mode,\n    });\n\n    const mode = params.Mode;\n\n    if (mode === 'Start' && currentState === 'stopped') {\n      ctx.logger.info('processOneResource: starting instance', { identifier, loopCount });\n      await ctx.step(`${stepPrefix}-start-${loopCount}`, async () => {\n        const ec2 = new EC2Client({});\n        await ec2.send(new StartInstancesCommand({ InstanceIds: [identifier] }));\n      });\n      ctx.logger.info('processOneResource: wait after start', {\n        identifier,\n        seconds: STATUS_CHANGE_WAIT_SECONDS,\n      });\n      await ctx.wait({ seconds: STATUS_CHANGE_WAIT_SECONDS });\n      loopCount += 1;\n      continue;\n    }\n\n    if (mode === 'Stop' && currentState === 'running') {\n      ctx.logger.info('processOneResource: stopping instance', { identifier, loopCount });\n      await ctx.step(`${stepPrefix}-stop-${loopCount}`, async () => {\n        const ec2 = new EC2Client({});\n        await ec2.send(new StopInstancesCommand({ InstanceIds: [identifier] }));\n      });\n      ctx.logger.info('processOneResource: wait after stop', {\n        identifier,\n        seconds: STATUS_CHANGE_WAIT_SECONDS,\n      });\n      await ctx.wait({ seconds: STATUS_CHANGE_WAIT_SECONDS });\n      loopCount += 1;\n      continue;\n    }\n\n    if (!isDesiredStableState(mode, currentState)) {\n      const transitioning =\n        (mode === 'Start' && currentState === 'pending') ||\n        (mode === 'Stop' && (currentState === 'stopping' || currentState === 'shutting-down'));\n\n      if (transitioning) {\n        ctx.logger.info('processOneResource: wait while transitioning', {\n          identifier,\n          loopCount,\n          currentState,\n          mode,\n          seconds: STATUS_CHANGE_WAIT_SECONDS,\n        });\n        await ctx.wait({ seconds: STATUS_CHANGE_WAIT_SECONDS });\n        loopCount += 1;\n        continue;\n      }\n\n      ctx.logger.error('processOneResource: unexpected state', {\n        identifier,\n        mode,\n        currentState,\n        loopCount,\n      });\n      throw new Error(`instance status fail: mode=${mode} currentState=${currentState}`);\n    }\n  } while (!isDesiredStableState(params.Mode, currentState));\n\n  ctx.logger.info('processOneResource: reached desired stable state', {\n    identifier,\n    finalState: currentState,\n    mode: params.Mode,\n    loopCount,\n  });\n\n  return {\n    identifier,\n    account,\n    region,\n    resource: targetResource,\n    status: currentState,\n  };\n};\n\n/**\n * Durable Lambda entry point for the EC2 running scheduler.\n *\n * Resolves instances via Resource Groups Tagging API, runs {@link processOneResource} for each ARN\n * in parallel (bounded concurrency), posts a parent Slack message and per-instance thread replies,\n * and uses durable `step` / `wait` / `map` so the run can resume across suspensions.\n *\n * @param event - Payload from EventBridge Scheduler; must include `Params.TagKey`, `Params.TagValues`, `Params.Mode`.\n * @param ctx - Root durable execution context.\n * @returns\n * - `{ status: 'TargetResourcesNotFound' }` when no instances match the tag filter.\n * - `{ status: 'Completed', processed, results }` when instances were handled (`results` entries match {@link processOneResource}).\n * @throws {Error} If `Params` is invalid, `SLACK_SECRET_NAME` is unset, the Slack secret is incomplete, or instance processing fails.\n */\nexport const handler = withDurableExecution(async (event: SchedulerEvent, ctx: DurableContext) => {\n\n  const params = event.Params;\n\n  ctx.logger.info('running-scheduler: invocation', {\n    mode: params?.Mode,\n    tagKey: params?.TagKey,\n    tagValueCount: params?.TagValues?.length ?? 0,\n  });\n\n  if (!params?.TagKey || !params?.TagValues || !params?.Mode) {\n    throw new Error('Invalid event: Params.TagKey, Params.TagValues, Params.Mode are required.');\n  }\n\n  // safe get Secrets name from environment variable\n  const slackSecretName = SafeEnvGetter.getEnv('SLACK_SECRET_NAME');\n\n  const slackSecretValue = await ctx.step('fetch-slack-secret', async () => {\n    ctx.logger.info('running-scheduler: fetching Slack secret', { secretName: slackSecretName });\n    return secretFetcher.getSecretValue<SlackSecret>(slackSecretName);\n  });\n\n  ctx.logger.info('running-scheduler: Slack secret loaded');\n\n  if (!slackSecretValue?.token || !slackSecretValue?.channel) {\n    throw new Error('Slack secret must contain token and channel.');\n  }\n\n  const targetResources = await ctx.step('get-target-resources', async () => {\n    const client = new ResourceGroupsTaggingAPIClient({});\n    const result = await client.send(\n      new GetResourcesCommand({\n        ResourceTypeFilters: ['ec2:instance'],\n        TagFilters: [{ Key: params.TagKey, Values: params.TagValues }],\n      }),\n    );\n    const arns = (result.ResourceTagMappingList ?? [])\n      .map((m: { ResourceARN?: string }) => m.ResourceARN)\n      .filter((arn: string | undefined): arn is string => arn != null);\n    ctx.logger.info('running-scheduler: get-target-resources done', { count: arns.length });\n    return arns;\n  });\n\n  if (targetResources.length === 0) {\n    ctx.logger.info('running-scheduler: no matching instances', { tagKey: params.TagKey });\n    return { status: 'TargetResourcesNotFound' as const };\n  }\n\n  const client = new WebClient(slackSecretValue.token);\n  const channel = slackSecretValue.channel;\n\n  ctx.logger.info('running-scheduler: posting parent Slack message', {\n    instanceCount: targetResources.length,\n  });\n\n  // send slack message\n  const slackParentMessageResult = await ctx.step('post-slack-messages', async () => {\n    return client.chat.postMessage({\n      channel,\n      text: `${params.Mode === 'Start' ? '😆 Starts' : '🥱 Stops'} the scheduled EC2 Instance.`,\n    });\n  });\n\n  ctx.logger.info('running-scheduler: parent Slack message posted', {\n    threadTs: slackParentMessageResult?.ts ?? null,\n  });\n\n  ctx.logger.info('running-scheduler: starting parallel instance processing', {\n    count: targetResources.length,\n    maxConcurrency: 10,\n  });\n\n  const results = await ctx.map(\n    targetResources,\n    // async (ctx: DurableContext, targetResource: string, index: number) =>\n    //   ctx.step(`process-resource-${index}`, async () =>\n    //     processOneResource(ctx, targetResource, params, index),\n    //   ),\n    async (mapCtx: DurableContext, targetResource: string, index: number) => {\n      return mapCtx.runInChildContext(`resource-${index}`, async (childCtx: DurableContext) => {\n        const result = await processOneResource(childCtx, targetResource, params, index);\n        // if (result.status === 'skipped') {\n        //   return result;\n        // }\n        childCtx.logger.info('running-scheduler: posting thread Slack message', {\n          index,\n          identifier: result.identifier,\n          status: result.status,\n        });\n        // send slack thread message\n        await childCtx.step('post-slack-child-messages', async () => {\n          const display = getStateDisplay(result.status);\n\n          return client.chat.postMessage({\n            channel,\n            thread_ts: slackParentMessageResult?.ts,\n            attachments: [\n              {\n                color: '#36a64f',\n                pretext: `${display?.emoji} The status of the EC2 Instance ${result.identifier} changed to ${display?.name} due to the schedule.`,\n                fields: [\n                  { title: 'Account', value: result.account, short: true },\n                  { title: 'Region', value: result.region, short: true },\n                  { title: 'Identifier', value: result.identifier, short: true },\n                  { title: 'Status', value: (display?.name ?? 'Unknown'), short: true },\n                ],\n              },\n            ],\n          });\n        });\n        return result;\n      });\n    },\n    { maxConcurrency: 10 },\n  );\n\n  const resultList = Array.isArray(results) ? results : [];\n  ctx.logger.info('running-scheduler: completed', {\n    processed: targetResources.length,\n    resultCount: resultList.length,\n  });\n  return {\n    status: 'Completed' as const,\n    processed: targetResources.length,\n    results: resultList,\n  };\n});\n"]}
|
|
@@ -29,5 +29,5 @@ class EC2InstanceRunningScheduleStack extends aws_cdk_lib_1.Stack {
|
|
|
29
29
|
}
|
|
30
30
|
exports.EC2InstanceRunningScheduleStack = EC2InstanceRunningScheduleStack;
|
|
31
31
|
_a = JSII_RTTI_SYMBOL_1;
|
|
32
|
-
EC2InstanceRunningScheduleStack[_a] = { fqn: "aws-ec2-instance-running-scheduler.EC2InstanceRunningScheduleStack", version: "3.
|
|
32
|
+
EC2InstanceRunningScheduleStack[_a] = { fqn: "aws-ec2-instance-running-scheduler.EC2InstanceRunningScheduleStack", version: "3.1.1" };
|
|
33
33
|
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZWMyLWluc3RhbmNlLXJ1bm5pbmctc2NoZWR1bGUtc3RhY2suanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvc3RhY2tzL2VjMi1pbnN0YW5jZS1ydW5uaW5nLXNjaGVkdWxlLXN0YWNrLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7Ozs7O0FBQUEsNkNBQWdEO0FBRWhELGlHQUE4SDtBQWtCOUg7O0dBRUc7QUFDSCxNQUFhLCtCQUFnQyxTQUFRLG1CQUFLO0lBQ3hEOzs7Ozs7T0FNRztJQUNILFlBQVksS0FBZ0IsRUFBRSxFQUFVLEVBQUUsS0FBMkM7UUFDbkYsS0FBSyxDQUFDLEtBQUssRUFBRSxFQUFFLEVBQUUsS0FBSyxDQUFDLENBQUM7UUFFeEIsSUFBSSw0REFBMkIsQ0FBQyxJQUFJLEVBQUUsNkJBQTZCLEVBQUU7WUFDbkUsY0FBYyxFQUFFLEtBQUssQ0FBQyxjQUFjO1lBQ3BDLGdCQUFnQixFQUFFLEtBQUssQ0FBQyxnQkFBZ0I7WUFDeEMsT0FBTyxFQUFFLEtBQUssQ0FBQyxPQUFPO1lBQ3RCLFlBQVksRUFBRSxLQUFLLENBQUMsWUFBWTtZQUNoQyxhQUFhLEVBQUUsS0FBSyxDQUFDLGFBQWE7U0FDbkMsQ0FBQyxDQUFDO0lBQ0wsQ0FBQzs7QUFsQkgsMEVBbUJDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgU3RhY2ssIFN0YWNrUHJvcHMgfSBmcm9tICdhd3MtY2RrLWxpYic7XG5pbXBvcnQgeyBDb25zdHJ1Y3QgfSBmcm9tICdjb25zdHJ1Y3RzJztcbmltcG9ydCB7IEVDMkluc3RhbmNlUnVubmluZ1NjaGVkdWxlciwgVGFyZ2V0UmVzb3VyY2UsIFNlY3JldHMsIFNjaGVkdWxlIH0gZnJvbSAnLi4vY29uc3RydWN0cy9lYzItaW5zdGFuY2UtcnVubmluZy1zY2hlZHVsZXInO1xuXG4vKipcbiAqIFByb3BzIGZvciB0aGUgRUMyIGluc3RhbmNlIHJ1bm5pbmcgc2NoZWR1bGUgQ0RLIHN0YWNrLlxuICovXG5leHBvcnQgaW50ZXJmYWNlIEVDMkluc3RhbmNlUnVubmluZ1NjaGVkdWxlU3RhY2tQcm9wcyBleHRlbmRzIFN0YWNrUHJvcHMge1xuICAvKiogVGFnLWJhc2VkIHRhcmdldCByZXNvdXJjZSBmb3IgRUMyIGluc3RhbmNlcyB0byBzdGFydC9zdG9wLiAqL1xuICByZWFkb25seSB0YXJnZXRSZXNvdXJjZTogVGFyZ2V0UmVzb3VyY2U7XG4gIC8qKiBXaGV0aGVyIHNjaGVkdWxpbmcgaXMgZW5hYmxlZC4gRGVmYXVsdHMgdG8gdHJ1ZSBpZiBvbWl0dGVkLiAqL1xuICByZWFkb25seSBlbmFibGVTY2hlZHVsaW5nPzogYm9vbGVhbjtcbiAgLyoqIFNlY3JldHMgKGUuZy4gU2xhY2spIGZvciB0aGUgc2NoZWR1bGVyLiAqL1xuICByZWFkb25seSBzZWNyZXRzOiBTZWNyZXRzO1xuICAvKiogQ3JvbiBzY2hlZHVsZSBmb3Igc3RvcHBpbmcgaW5zdGFuY2VzLiAqL1xuICByZWFkb25seSBzdG9wU2NoZWR1bGU/OiBTY2hlZHVsZTtcbiAgLyoqIENyb24gc2NoZWR1bGUgZm9yIHN0YXJ0aW5nIGluc3RhbmNlcy4gKi9cbiAgcmVhZG9ubHkgc3RhcnRTY2hlZHVsZT86IFNjaGVkdWxlO1xufVxuXG4vKipcbiAqIENESyBTdGFjayB0aGF0IGRlcGxveXMgdGhlIEVDMiBpbnN0YW5jZSBydW5uaW5nIHNjaGVkdWxlciAoRXZlbnRCcmlkZ2UgU2NoZWR1bGVyICsgRHVyYWJsZSBMYW1iZGEpLlxuICovXG5leHBvcnQgY2xhc3MgRUMySW5zdGFuY2VSdW5uaW5nU2NoZWR1bGVTdGFjayBleHRlbmRzIFN0YWNrIHtcbiAgLyoqXG4gICAqIENyZWF0ZXMgdGhlIHN0YWNrIGFuZCB0aGUgRUMySW5zdGFuY2VSdW5uaW5nU2NoZWR1bGVyIGNvbnN0cnVjdC5cbiAgICpcbiAgICogQHBhcmFtIHNjb3BlIC0gUGFyZW50IGNvbnN0cnVjdC5cbiAgICogQHBhcmFtIGlkIC0gU3RhY2sgaWQuXG4gICAqIEBwYXJhbSBwcm9wcyAtIFN0YWNrIHByb3BzICh0YXJnZXQgcmVzb3VyY2UsIHNjaGVkdWxlcywgc2VjcmV0cykuXG4gICAqL1xuICBjb25zdHJ1Y3RvcihzY29wZTogQ29uc3RydWN0LCBpZDogc3RyaW5nLCBwcm9wczogRUMySW5zdGFuY2VSdW5uaW5nU2NoZWR1bGVTdGFja1Byb3BzKSB7XG4gICAgc3VwZXIoc2NvcGUsIGlkLCBwcm9wcyk7XG5cbiAgICBuZXcgRUMySW5zdGFuY2VSdW5uaW5nU2NoZWR1bGVyKHRoaXMsICdFQzJJbnN0YW5jZVJ1bm5pbmdTY2hlZHVsZXInLCB7XG4gICAgICB0YXJnZXRSZXNvdXJjZTogcHJvcHMudGFyZ2V0UmVzb3VyY2UsXG4gICAgICBlbmFibGVTY2hlZHVsaW5nOiBwcm9wcy5lbmFibGVTY2hlZHVsaW5nLFxuICAgICAgc2VjcmV0czogcHJvcHMuc2VjcmV0cyxcbiAgICAgIHN0b3BTY2hlZHVsZTogcHJvcHMuc3RvcFNjaGVkdWxlLFxuICAgICAgc3RhcnRTY2hlZHVsZTogcHJvcHMuc3RhcnRTY2hlZHVsZSxcbiAgICB9KTtcbiAgfVxufSJdfQ==
|
package/package.json
CHANGED
|
@@ -6,36 +6,36 @@
|
|
|
6
6
|
"url": "https://github.com/gammarers-aws-cdk-constructs/aws-ec2-instance-running-scheduler.git"
|
|
7
7
|
},
|
|
8
8
|
"scripts": {
|
|
9
|
-
"build": "
|
|
10
|
-
"bump": "
|
|
11
|
-
"bundle": "
|
|
12
|
-
"bundle:funcs/running-scheduler.lambda": "
|
|
13
|
-
"bundle:funcs/running-scheduler.lambda:watch": "
|
|
14
|
-
"clobber": "
|
|
15
|
-
"compat": "
|
|
16
|
-
"compile": "
|
|
17
|
-
"default": "
|
|
18
|
-
"docgen": "
|
|
19
|
-
"eject": "
|
|
20
|
-
"eslint": "
|
|
21
|
-
"package": "
|
|
22
|
-
"package-all": "
|
|
23
|
-
"package:js": "
|
|
24
|
-
"post-compile": "
|
|
25
|
-
"post-upgrade": "
|
|
26
|
-
"pre-compile": "
|
|
27
|
-
"release": "
|
|
28
|
-
"test": "
|
|
29
|
-
"test:watch": "
|
|
30
|
-
"unbump": "
|
|
31
|
-
"upgrade": "
|
|
32
|
-
"watch": "
|
|
33
|
-
"projen": "
|
|
9
|
+
"build": "projen build",
|
|
10
|
+
"bump": "projen bump",
|
|
11
|
+
"bundle": "projen bundle",
|
|
12
|
+
"bundle:funcs/running-scheduler.lambda": "projen bundle:funcs/running-scheduler.lambda",
|
|
13
|
+
"bundle:funcs/running-scheduler.lambda:watch": "projen bundle:funcs/running-scheduler.lambda:watch",
|
|
14
|
+
"clobber": "projen clobber",
|
|
15
|
+
"compat": "projen compat",
|
|
16
|
+
"compile": "projen compile",
|
|
17
|
+
"default": "projen default",
|
|
18
|
+
"docgen": "projen docgen",
|
|
19
|
+
"eject": "projen eject",
|
|
20
|
+
"eslint": "projen eslint",
|
|
21
|
+
"package": "projen package",
|
|
22
|
+
"package-all": "projen package-all",
|
|
23
|
+
"package:js": "projen package:js",
|
|
24
|
+
"post-compile": "projen post-compile",
|
|
25
|
+
"post-upgrade": "projen post-upgrade",
|
|
26
|
+
"pre-compile": "projen pre-compile",
|
|
27
|
+
"release": "projen release",
|
|
28
|
+
"test": "projen test",
|
|
29
|
+
"test:watch": "projen test:watch",
|
|
30
|
+
"unbump": "projen unbump",
|
|
31
|
+
"upgrade": "projen upgrade",
|
|
32
|
+
"watch": "projen watch",
|
|
33
|
+
"projen": "projen"
|
|
34
34
|
},
|
|
35
35
|
"author": {
|
|
36
36
|
"name": "yicr",
|
|
37
37
|
"email": "yicr@users.noreply.github.com",
|
|
38
|
-
"organization":
|
|
38
|
+
"organization": false
|
|
39
39
|
},
|
|
40
40
|
"devDependencies": {
|
|
41
41
|
"@aws-sdk/client-ec2": "^3",
|
|
@@ -61,11 +61,12 @@
|
|
|
61
61
|
"jest": "^29.7.0",
|
|
62
62
|
"jest-junit": "^16",
|
|
63
63
|
"jsii": "5.9.x",
|
|
64
|
-
"jsii-diff": "^1.
|
|
64
|
+
"jsii-diff": "^1.128.0",
|
|
65
65
|
"jsii-docgen": "^10.5.0",
|
|
66
|
-
"jsii-pacmak": "^1.
|
|
66
|
+
"jsii-pacmak": "^1.128.0",
|
|
67
67
|
"jsii-rosetta": "5.9.x",
|
|
68
|
-
"projen": "^0.99.
|
|
68
|
+
"projen": "^0.99.47",
|
|
69
|
+
"safe-env-getter": "^0.2",
|
|
69
70
|
"ts-jest": "^29.4.9",
|
|
70
71
|
"ts-node": "^10.9.2",
|
|
71
72
|
"typescript": "5.9.x"
|
|
@@ -87,12 +88,19 @@
|
|
|
87
88
|
"engines": {
|
|
88
89
|
"node": ">= 20.0.0"
|
|
89
90
|
},
|
|
91
|
+
"devEngines": {
|
|
92
|
+
"packageManager": {
|
|
93
|
+
"name": "yarn",
|
|
94
|
+
"version": "1.22.22",
|
|
95
|
+
"onFail": "ignore"
|
|
96
|
+
}
|
|
97
|
+
},
|
|
90
98
|
"main": "lib/index.js",
|
|
91
99
|
"license": "Apache-2.0",
|
|
92
100
|
"publishConfig": {
|
|
93
101
|
"access": "public"
|
|
94
102
|
},
|
|
95
|
-
"version": "3.
|
|
103
|
+
"version": "3.1.1",
|
|
96
104
|
"jest": {
|
|
97
105
|
"coverageProvider": "v8",
|
|
98
106
|
"testMatch": [
|
|
@@ -148,5 +156,6 @@
|
|
|
148
156
|
"rootDir": "src"
|
|
149
157
|
}
|
|
150
158
|
},
|
|
159
|
+
"packageManager": "yarn@1.22.22",
|
|
151
160
|
"//": "~~ Generated by projen. To modify, edit .projenrc.ts and run \"npx projen\"."
|
|
152
161
|
}
|