@cloudsnorkel/cdk-github-runners 0.2.0 → 0.3.2
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/.gitattributes +8 -1
- package/.jsii +1371 -206
- package/API.md +1191 -93
- package/README.md +59 -49
- package/lib/index.d.ts +3 -1
- package/lib/index.js +7 -1
- package/lib/lambdas/build-image/index.js +121 -0
- package/lib/lambdas/delete-runner/index.js +5151 -2999
- package/lib/lambdas/setup/index.html +37 -0
- package/lib/lambdas/setup/index.js +140 -255
- package/lib/lambdas/status/index.js +5151 -2999
- package/lib/lambdas/token-retriever/index.js +5151 -2999
- package/lib/lambdas/update-lambda/index.js +55 -0
- package/lib/providers/codebuild.d.ts +31 -1
- package/lib/providers/codebuild.js +57 -13
- package/lib/providers/common.d.ts +87 -6
- package/lib/providers/common.js +64 -4
- package/lib/providers/docker-images/codebuild/linux-arm64/Dockerfile +63 -0
- package/lib/providers/docker-images/codebuild/{Dockerfile → linux-x64/Dockerfile} +14 -5
- package/lib/providers/docker-images/fargate/linux-arm64/Dockerfile +45 -0
- package/lib/providers/docker-images/fargate/{runner.sh → linux-arm64/runner.sh} +0 -0
- package/lib/providers/docker-images/fargate/{Dockerfile → linux-x64/Dockerfile} +14 -5
- package/lib/providers/docker-images/fargate/linux-x64/runner.sh +5 -0
- package/lib/providers/docker-images/lambda/linux-arm64/Dockerfile +36 -0
- package/lib/providers/docker-images/lambda/{runner.js → linux-arm64/runner.js} +0 -0
- package/lib/providers/docker-images/lambda/{runner.sh → linux-arm64/runner.sh} +0 -0
- package/lib/providers/docker-images/lambda/linux-x64/Dockerfile +35 -0
- package/lib/providers/docker-images/lambda/linux-x64/runner.js +29 -0
- package/lib/providers/docker-images/lambda/linux-x64/runner.sh +12 -0
- package/lib/providers/fargate.d.ts +33 -1
- package/lib/providers/fargate.js +39 -8
- package/lib/providers/image-builders/codebuild.d.ts +178 -0
- package/lib/providers/image-builders/codebuild.js +354 -0
- package/lib/providers/image-builders/static.d.ts +29 -0
- package/lib/providers/image-builders/static.js +58 -0
- package/lib/providers/lambda.d.ts +27 -1
- package/lib/providers/lambda.js +88 -9
- package/lib/runner.d.ts +56 -9
- package/lib/runner.js +37 -11
- package/lib/secrets.js +1 -1
- package/lib/utils.d.ts +2 -1
- package/lib/utils.js +14 -3
- package/lib/webhook.js +2 -1
- package/package.json +30 -12
- package/setup/index.html +12 -0
- package/setup/src/App.svelte +291 -0
- package/setup/src/app.scss +15 -0
- package/setup/src/main.ts +8 -0
- package/setup/src/vite-env.d.ts +2 -0
- package/setup/svelte.config.mjs +7 -0
- package/setup/tsconfig.json +21 -0
- package/setup/tsconfig.node.json +8 -0
- package/setup/vite.config.ts +15 -0
- package/lib/providers/docker-images/lambda/Dockerfile +0 -27
package/README.md
CHANGED
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
# GitHub Self-Hosted Runners CDK Constructs
|
|
2
2
|
|
|
3
|
-
[][
|
|
4
|
-
[][
|
|
3
|
+
[][7]
|
|
4
|
+
[][6]
|
|
5
5
|
[][8]
|
|
6
6
|
[][11]
|
|
7
7
|
[][12]
|
|
@@ -31,15 +31,16 @@ The best way to browse API documentation is on [Constructs Hub][13]. It is avail
|
|
|
31
31
|
|
|
32
32
|
A runner provider creates compute resources on-demand and uses [actions/runner][5] to start a runner.
|
|
33
33
|
|
|
34
|
-
|
|
|
35
|
-
|
|
36
|
-
| **Time limit**
|
|
37
|
-
| **vCPUs**
|
|
38
|
-
| **RAM**
|
|
39
|
-
| **Storage**
|
|
40
|
-
| **
|
|
41
|
-
| **
|
|
42
|
-
| **
|
|
34
|
+
| | CodeBuild | Fargate | Lambda |
|
|
35
|
+
|------------------|--------------------------|----------------|----------------|
|
|
36
|
+
| **Time limit** | 8 hours | Unlimited | 15 minutes |
|
|
37
|
+
| **vCPUs** | 2, 4, 8, or 72 | 0.25 to 4 | 1 to 6 |
|
|
38
|
+
| **RAM** | 3gb, 7gb, 15gb, or 145gb | 512mb to 30gb | 128mb to 10gb |
|
|
39
|
+
| **Storage** | 50gb to 824gb | 20gb to 200gb | Up to 10gb |
|
|
40
|
+
| **Architecture** | x86_64, ARM64 | x86_64, ARM64 | x86_64, ARM64 |
|
|
41
|
+
| **sudo** | ✔ | ✔ | ❌ |
|
|
42
|
+
| **Docker** | ✔ | ❌ | ❌ |
|
|
43
|
+
| **Spot pricing** | ❌ | ✔ | ❌ |
|
|
43
44
|
|
|
44
45
|
The best provider to use mostly depends on your current infrastructure. When in doubt, CodeBuild is always a good choice. Execution history and logs are easy to view, and it has no restrictive limits unless you need to run for more than 8 hours.
|
|
45
46
|
|
|
@@ -90,50 +91,59 @@ The default providers configured by `GitHubRunners` are useful for testing but p
|
|
|
90
91
|
For example:
|
|
91
92
|
|
|
92
93
|
```typescript
|
|
93
|
-
|
|
94
|
-
|
|
95
|
-
|
|
96
|
-
|
|
97
|
-
const app = new cdk.App();
|
|
98
|
-
const stack = new cdk.Stack(
|
|
99
|
-
app,
|
|
100
|
-
'github-runners-test',
|
|
101
|
-
{
|
|
102
|
-
env: {
|
|
103
|
-
account: process.env.CDK_DEFAULT_ACCOUNT,
|
|
104
|
-
region: process.env.CDK_DEFAULT_REGION,
|
|
105
|
-
},
|
|
106
|
-
},
|
|
107
|
-
);
|
|
108
|
-
|
|
109
|
-
const vpc = ec2.Vpc.fromLookup(stack, 'vpc', { vpcId: 'vpc-1234567' });
|
|
110
|
-
const runnerSg = new ec2.SecurityGroup(stack, 'runner security group', { vpc: vpc });
|
|
111
|
-
const dbSg = ec2.SecurityGroup.fromSecurityGroupId(stack, 'database security group', 'sg-1234567');
|
|
112
|
-
const bucket = new s3.Bucket(stack, 'runner bucket');
|
|
94
|
+
let vpc: ec2.Vpc;
|
|
95
|
+
let runnerSg: ec2.SecurityGroup;
|
|
96
|
+
let dbSg: ec2.SecurityGroup;
|
|
97
|
+
let bucket: s3.Bucket;
|
|
113
98
|
|
|
114
99
|
// create a custom CodeBuild provider
|
|
115
|
-
const myProvider = new CodeBuildRunner(
|
|
116
|
-
|
|
117
|
-
|
|
118
|
-
|
|
119
|
-
|
|
120
|
-
securityGroup: runnerSg,
|
|
121
|
-
},
|
|
122
|
-
);
|
|
100
|
+
const myProvider = new CodeBuildRunner(this, 'codebuild runner', {
|
|
101
|
+
label: 'my-codebuild',
|
|
102
|
+
vpc: vpc,
|
|
103
|
+
securityGroup: runnerSg,
|
|
104
|
+
});
|
|
123
105
|
// grant some permissions to the provider
|
|
124
106
|
bucket.grantReadWrite(myProvider);
|
|
125
107
|
dbSg.connections.allowFrom(runnerSg, ec2.Port.tcp(3306), 'allow runners to connect to MySQL database');
|
|
126
108
|
|
|
127
109
|
// create the runner infrastructure
|
|
128
|
-
new GitHubRunners(
|
|
129
|
-
stack,
|
|
130
|
-
'runners',
|
|
131
|
-
{
|
|
110
|
+
new GitHubRunners(this, 'runners', {
|
|
132
111
|
providers: [myProvider],
|
|
133
|
-
|
|
134
|
-
|
|
112
|
+
});
|
|
113
|
+
```
|
|
114
|
+
|
|
115
|
+
Another way to customize runners is by modifying the image used to spin them up. The image contains the [runner][5], any required dependencies, and integration code with the provider. You may choose to customize this image by adding more packages, for example.
|
|
116
|
+
|
|
117
|
+
```typescript
|
|
118
|
+
const myBuilder = new CodeBuildImageBuilder(this, 'image builder', {
|
|
119
|
+
dockerfilePath: FargateProvider.LINUX_X64_DOCKERFILE_PATH,
|
|
120
|
+
runnerVersion: RunnerVersion.specific('2.291.0'),
|
|
121
|
+
rebuildInterval: Duration.days(14),
|
|
122
|
+
});
|
|
123
|
+
myBuilder.setBuildArg('EXTRA_PACKAGES', 'nginx xz-utils');
|
|
124
|
+
|
|
125
|
+
const myProvider = new FargateProvider(this, 'fargate runner', {
|
|
126
|
+
label: 'customized-fargate',
|
|
127
|
+
vpc: vpc,
|
|
128
|
+
securityGroup: runnerSg,
|
|
129
|
+
});
|
|
130
|
+
|
|
131
|
+
// create the runner infrastructure
|
|
132
|
+
new GitHubRunners(stack, 'runners', {
|
|
133
|
+
providers: [myProvider],
|
|
134
|
+
});
|
|
135
|
+
```
|
|
136
|
+
|
|
137
|
+
Your workflow will then look like:
|
|
135
138
|
|
|
136
|
-
|
|
139
|
+
```yaml
|
|
140
|
+
name: self-hosted example
|
|
141
|
+
on: push
|
|
142
|
+
jobs:
|
|
143
|
+
self-hosted:
|
|
144
|
+
runs-on: [self-hosted, customized-fargate]
|
|
145
|
+
steps:
|
|
146
|
+
- run: echo hello world
|
|
137
147
|
```
|
|
138
148
|
|
|
139
149
|
## Architecture
|
|
@@ -162,12 +172,12 @@ app.synth();
|
|
|
162
172
|
[3]: https://github.com/philips-labs/terraform-aws-github-runner
|
|
163
173
|
[4]: https://github.com/actions-runner-controller/actions-runner-controller
|
|
164
174
|
[5]: https://github.com/actions/runner
|
|
165
|
-
[6]: https://
|
|
166
|
-
[7]: https://
|
|
175
|
+
[6]: https://pypi.org/project/cloudsnorkel.cdk-github-runners
|
|
176
|
+
[7]: https://www.npmjs.com/package/@cloudsnorkel/cdk-github-runners
|
|
167
177
|
[8]: https://search.maven.org/search?q=g:%22com.cloudsnorkel%22%20AND%20a:%22cdk.github.runners%22
|
|
168
178
|
[9]: https://docs.github.com/en/developers/apps/getting-started-with-apps/about-apps
|
|
169
179
|
[10]: https://docs.github.com/en/authentication/keeping-your-account-and-data-secure/creating-a-personal-access-token
|
|
170
180
|
[11]: https://pkg.go.dev/github.com/CloudSnorkel/cdk-github-runners-go/cloudsnorkelcdkgithubrunners
|
|
171
181
|
[12]: https://www.nuget.org/packages/CloudSnorkel.Cdk.Github.Runners/
|
|
172
182
|
[13]: https://constructs.dev/packages/@cloudsnorkel/cdk-github-runners/
|
|
173
|
-
[14]: https://docs.github.com/en/actions/hosting-your-own-runners/autoscaling-with-self-hosted-runners#using-ephemeral-runners-for-autoscaling
|
|
183
|
+
[14]: https://docs.github.com/en/actions/hosting-your-own-runners/autoscaling-with-self-hosted-runners#using-ephemeral-runners-for-autoscaling
|
package/lib/index.d.ts
CHANGED
|
@@ -3,4 +3,6 @@ export { GitHubRunners, GitHubRunnersProps } from './runner';
|
|
|
3
3
|
export { CodeBuildRunner, CodeBuildRunnerProps } from './providers/codebuild';
|
|
4
4
|
export { LambdaRunner, LambdaRunnerProps } from './providers/lambda';
|
|
5
5
|
export { FargateRunner, FargateRunnerProps } from './providers/fargate';
|
|
6
|
-
export { IRunnerProvider, RunnerProviderProps, RunnerVersion, RunnerRuntimeParameters } from './providers/common';
|
|
6
|
+
export { IRunnerProvider, RunnerProviderProps, RunnerVersion, RunnerRuntimeParameters, RunnerImage, IImageBuilder, Architecture, Os } from './providers/common';
|
|
7
|
+
export { CodeBuildImageBuilder, CodeBuildImageBuilderProps } from './providers/image-builders/codebuild';
|
|
8
|
+
export { StaticRunnerImage } from './providers/image-builders/static';
|
package/lib/index.js
CHANGED
|
@@ -12,4 +12,10 @@ var fargate_1 = require("./providers/fargate");
|
|
|
12
12
|
Object.defineProperty(exports, "FargateRunner", { enumerable: true, get: function () { return fargate_1.FargateRunner; } });
|
|
13
13
|
var common_1 = require("./providers/common");
|
|
14
14
|
Object.defineProperty(exports, "RunnerVersion", { enumerable: true, get: function () { return common_1.RunnerVersion; } });
|
|
15
|
-
|
|
15
|
+
Object.defineProperty(exports, "Architecture", { enumerable: true, get: function () { return common_1.Architecture; } });
|
|
16
|
+
Object.defineProperty(exports, "Os", { enumerable: true, get: function () { return common_1.Os; } });
|
|
17
|
+
var codebuild_2 = require("./providers/image-builders/codebuild");
|
|
18
|
+
Object.defineProperty(exports, "CodeBuildImageBuilder", { enumerable: true, get: function () { return codebuild_2.CodeBuildImageBuilder; } });
|
|
19
|
+
var static_1 = require("./providers/image-builders/static");
|
|
20
|
+
Object.defineProperty(exports, "StaticRunnerImage", { enumerable: true, get: function () { return static_1.StaticRunnerImage; } });
|
|
21
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi9zcmMvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7QUFBQSxxQ0FBb0M7QUFBM0Isa0dBQUEsT0FBTyxPQUFBO0FBQ2hCLG1DQUE2RDtBQUFwRCx1R0FBQSxhQUFhLE9BQUE7QUFDdEIsbURBQThFO0FBQXJFLDRHQUFBLGVBQWUsT0FBQTtBQUN4Qiw2Q0FBcUU7QUFBNUQsc0dBQUEsWUFBWSxPQUFBO0FBQ3JCLCtDQUF3RTtBQUEvRCx3R0FBQSxhQUFhLE9BQUE7QUFDdEIsNkNBQWdLO0FBQWpILHVHQUFBLGFBQWEsT0FBQTtBQUF1RCxzR0FBQSxZQUFZLE9BQUE7QUFBRSw0RkFBQSxFQUFFLE9BQUE7QUFDbkksa0VBQXlHO0FBQWhHLGtIQUFBLHFCQUFxQixPQUFBO0FBQzlCLDREQUFzRTtBQUE3RCwyR0FBQSxpQkFBaUIsT0FBQSIsInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCB7IFNlY3JldHMgfSBmcm9tICcuL3NlY3JldHMnO1xuZXhwb3J0IHsgR2l0SHViUnVubmVycywgR2l0SHViUnVubmVyc1Byb3BzIH0gZnJvbSAnLi9ydW5uZXInO1xuZXhwb3J0IHsgQ29kZUJ1aWxkUnVubmVyLCBDb2RlQnVpbGRSdW5uZXJQcm9wcyB9IGZyb20gJy4vcHJvdmlkZXJzL2NvZGVidWlsZCc7XG5leHBvcnQgeyBMYW1iZGFSdW5uZXIsIExhbWJkYVJ1bm5lclByb3BzIH0gZnJvbSAnLi9wcm92aWRlcnMvbGFtYmRhJztcbmV4cG9ydCB7IEZhcmdhdGVSdW5uZXIsIEZhcmdhdGVSdW5uZXJQcm9wcyB9IGZyb20gJy4vcHJvdmlkZXJzL2ZhcmdhdGUnO1xuZXhwb3J0IHsgSVJ1bm5lclByb3ZpZGVyLCBSdW5uZXJQcm92aWRlclByb3BzLCBSdW5uZXJWZXJzaW9uLCBSdW5uZXJSdW50aW1lUGFyYW1ldGVycywgUnVubmVySW1hZ2UsIElJbWFnZUJ1aWxkZXIsIEFyY2hpdGVjdHVyZSwgT3MgfSBmcm9tICcuL3Byb3ZpZGVycy9jb21tb24nO1xuZXhwb3J0IHsgQ29kZUJ1aWxkSW1hZ2VCdWlsZGVyLCBDb2RlQnVpbGRJbWFnZUJ1aWxkZXJQcm9wcyB9IGZyb20gJy4vcHJvdmlkZXJzL2ltYWdlLWJ1aWxkZXJzL2NvZGVidWlsZCc7XG5leHBvcnQgeyBTdGF0aWNSdW5uZXJJbWFnZSB9IGZyb20gJy4vcHJvdmlkZXJzL2ltYWdlLWJ1aWxkZXJzL3N0YXRpYyc7XG4iXX0=
|
|
@@ -0,0 +1,121 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
var __create = Object.create;
|
|
3
|
+
var __defProp = Object.defineProperty;
|
|
4
|
+
var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
|
|
5
|
+
var __getOwnPropNames = Object.getOwnPropertyNames;
|
|
6
|
+
var __getProtoOf = Object.getPrototypeOf;
|
|
7
|
+
var __hasOwnProp = Object.prototype.hasOwnProperty;
|
|
8
|
+
var __export = (target, all) => {
|
|
9
|
+
for (var name in all)
|
|
10
|
+
__defProp(target, name, { get: all[name], enumerable: true });
|
|
11
|
+
};
|
|
12
|
+
var __copyProps = (to, from, except, desc) => {
|
|
13
|
+
if (from && typeof from === "object" || typeof from === "function") {
|
|
14
|
+
for (let key of __getOwnPropNames(from))
|
|
15
|
+
if (!__hasOwnProp.call(to, key) && key !== except)
|
|
16
|
+
__defProp(to, key, { get: () => from[key], enumerable: !(desc = __getOwnPropDesc(from, key)) || desc.enumerable });
|
|
17
|
+
}
|
|
18
|
+
return to;
|
|
19
|
+
};
|
|
20
|
+
var __toESM = (mod, isNodeMode, target) => (target = mod != null ? __create(__getProtoOf(mod)) : {}, __copyProps(isNodeMode || !mod || !mod.__esModule ? __defProp(target, "default", { value: mod, enumerable: true }) : target, mod));
|
|
21
|
+
var __toCommonJS = (mod) => __copyProps(__defProp({}, "__esModule", { value: true }), mod);
|
|
22
|
+
|
|
23
|
+
// src/lambdas/build-image/index.ts
|
|
24
|
+
var build_image_exports = {};
|
|
25
|
+
__export(build_image_exports, {
|
|
26
|
+
handler: () => handler
|
|
27
|
+
});
|
|
28
|
+
module.exports = __toCommonJS(build_image_exports);
|
|
29
|
+
var AWS = __toESM(require("aws-sdk"));
|
|
30
|
+
var codebuild = new AWS.CodeBuild();
|
|
31
|
+
var ecr = new AWS.ECR();
|
|
32
|
+
async function handler(event, context) {
|
|
33
|
+
try {
|
|
34
|
+
console.log(JSON.stringify(event));
|
|
35
|
+
const repoName = event.ResourceProperties.RepoName;
|
|
36
|
+
const projectName = event.ResourceProperties.ProjectName;
|
|
37
|
+
switch (event.RequestType) {
|
|
38
|
+
case "Create":
|
|
39
|
+
case "Update":
|
|
40
|
+
console.log(`Starting CodeBuild project ${projectName}`);
|
|
41
|
+
await codebuild.startBuild({
|
|
42
|
+
projectName,
|
|
43
|
+
environmentVariablesOverride: [
|
|
44
|
+
{
|
|
45
|
+
type: "PLAINTEXT",
|
|
46
|
+
name: "STACK_ID",
|
|
47
|
+
value: event.StackId
|
|
48
|
+
},
|
|
49
|
+
{
|
|
50
|
+
type: "PLAINTEXT",
|
|
51
|
+
name: "REQUEST_ID",
|
|
52
|
+
value: event.RequestId
|
|
53
|
+
},
|
|
54
|
+
{
|
|
55
|
+
type: "PLAINTEXT",
|
|
56
|
+
name: "LOGICAL_RESOURCE_ID",
|
|
57
|
+
value: event.LogicalResourceId
|
|
58
|
+
},
|
|
59
|
+
{
|
|
60
|
+
type: "PLAINTEXT",
|
|
61
|
+
name: "RESPONSE_URL",
|
|
62
|
+
value: event.ResponseURL
|
|
63
|
+
}
|
|
64
|
+
]
|
|
65
|
+
}).promise();
|
|
66
|
+
break;
|
|
67
|
+
case "Delete":
|
|
68
|
+
const images = await ecr.listImages({ repositoryName: repoName, maxResults: 100 }).promise();
|
|
69
|
+
if (images.imageIds && images.imageIds.length > 0) {
|
|
70
|
+
await ecr.batchDeleteImage({
|
|
71
|
+
imageIds: images.imageIds.map((i) => {
|
|
72
|
+
return { imageDigest: i.imageDigest };
|
|
73
|
+
}),
|
|
74
|
+
repositoryName: repoName
|
|
75
|
+
}).promise();
|
|
76
|
+
}
|
|
77
|
+
await respond("SUCCESS", "OK", event.PhysicalResourceId, {});
|
|
78
|
+
break;
|
|
79
|
+
}
|
|
80
|
+
} catch (e) {
|
|
81
|
+
console.log(e);
|
|
82
|
+
await respond("FAILED", e.message || "Internal Error", context.logStreamName, {});
|
|
83
|
+
}
|
|
84
|
+
function respond(responseStatus, reason, physicalResourceId, data) {
|
|
85
|
+
const responseBody = JSON.stringify({
|
|
86
|
+
Status: responseStatus,
|
|
87
|
+
Reason: reason,
|
|
88
|
+
PhysicalResourceId: physicalResourceId,
|
|
89
|
+
StackId: event.StackId,
|
|
90
|
+
RequestId: event.RequestId,
|
|
91
|
+
LogicalResourceId: event.LogicalResourceId,
|
|
92
|
+
NoEcho: false,
|
|
93
|
+
Data: data
|
|
94
|
+
});
|
|
95
|
+
console.log("Responding", responseBody);
|
|
96
|
+
const parsedUrl = require("url").parse(event.ResponseURL);
|
|
97
|
+
const requestOptions = {
|
|
98
|
+
hostname: parsedUrl.hostname,
|
|
99
|
+
path: parsedUrl.path,
|
|
100
|
+
method: "PUT",
|
|
101
|
+
headers: {
|
|
102
|
+
"content-type": "",
|
|
103
|
+
"content-length": responseBody.length
|
|
104
|
+
}
|
|
105
|
+
};
|
|
106
|
+
return new Promise((resolve, reject) => {
|
|
107
|
+
try {
|
|
108
|
+
const request = require("https").request(requestOptions, resolve);
|
|
109
|
+
request.on("error", reject);
|
|
110
|
+
request.write(responseBody);
|
|
111
|
+
request.end();
|
|
112
|
+
} catch (e) {
|
|
113
|
+
reject(e);
|
|
114
|
+
}
|
|
115
|
+
});
|
|
116
|
+
}
|
|
117
|
+
}
|
|
118
|
+
// Annotate the CommonJS export names for ESM import in node:
|
|
119
|
+
0 && (module.exports = {
|
|
120
|
+
handler
|
|
121
|
+
});
|