cdk-ecr-deployment 2.2.0 → 2.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/.jsii +44 -24
- package/API.md +2 -0
- package/README.md +6 -5
- package/changelog.md +3 -3
- package/lambda/Dockerfile +2 -2
- package/lambda/go.mod +19 -2
- package/lambda/go.sum +189 -27
- package/lambda/install.js +5 -3
- package/lib/index.d.ts +11 -0
- package/lib/index.js +12 -8
- package/node_modules/hpagent/.github/dependabot.yml +8 -0
- package/node_modules/hpagent/.github/workflows/build.yml +29 -0
- package/node_modules/hpagent/LICENSE +21 -0
- package/node_modules/hpagent/README.md +138 -0
- package/node_modules/hpagent/index.d.ts +26 -0
- package/node_modules/hpagent/index.js +101 -0
- package/node_modules/hpagent/index.mjs +5 -0
- package/node_modules/hpagent/package.json +49 -0
- package/node_modules/hpagent/test/got.test.js +111 -0
- package/node_modules/hpagent/test/http-http.test.js +351 -0
- package/node_modules/hpagent/test/http-https.test.js +313 -0
- package/node_modules/hpagent/test/https-http.test.js +313 -0
- package/node_modules/hpagent/test/https-https.test.js +313 -0
- package/node_modules/hpagent/test/index.test-d.ts +45 -0
- package/node_modules/hpagent/test/needle.test.js +103 -0
- package/node_modules/hpagent/test/node-fetch.test.js +103 -0
- package/node_modules/hpagent/test/simple-get.test.js +139 -0
- package/node_modules/hpagent/test/ssl.cert +19 -0
- package/node_modules/hpagent/test/ssl.key +27 -0
- package/node_modules/hpagent/test/utils.js +59 -0
- package/package.json +5 -3
- package/releasetag.txt +1 -1
- package/version.txt +1 -1
package/lambda/install.js
CHANGED
|
@@ -3,7 +3,7 @@ const got = require('got');
|
|
|
3
3
|
const path = require('path');
|
|
4
4
|
const stream = require('stream');
|
|
5
5
|
const crypto = require('crypto');
|
|
6
|
-
|
|
6
|
+
const { HttpProxyAgent, HttpsProxyAgent } = require('hpagent');
|
|
7
7
|
const { promisify } = require('util');
|
|
8
8
|
const pipeline = promisify(stream.pipeline);
|
|
9
9
|
|
|
@@ -29,10 +29,12 @@ function sha256sum(p) {
|
|
|
29
29
|
}
|
|
30
30
|
|
|
31
31
|
async function download(url, dest) {
|
|
32
|
-
|
|
32
|
+
const agent = {};
|
|
33
|
+
agent.https = process.env.HTTPS_PROXY ? new HttpsProxyAgent({proxy: process.env.HTTPS_PROXY}): undefined;
|
|
34
|
+
agent.http = process.env.HTTP_PROXY ? new HttpProxyAgent({proxy: process.env.HTTP_PROXY}): undefined;
|
|
33
35
|
console.log(`download ${url}`);
|
|
34
36
|
await pipeline(
|
|
35
|
-
got.stream(url),
|
|
37
|
+
got.stream(url, { agent }),
|
|
36
38
|
fs.createWriteStream(dest)
|
|
37
39
|
);
|
|
38
40
|
}
|
package/lib/index.d.ts
CHANGED
|
@@ -4,6 +4,17 @@ import { Construct } from 'constructs';
|
|
|
4
4
|
* @stability stable
|
|
5
5
|
*/
|
|
6
6
|
export interface ECRDeploymentProps {
|
|
7
|
+
/**
|
|
8
|
+
* Image to use to build Golang lambda for custom resource, if download fails or is not wanted.
|
|
9
|
+
*
|
|
10
|
+
* Might be needed for local build if all images need to come from own registry.
|
|
11
|
+
*
|
|
12
|
+
* Note that image should use yum as a package manager and have golang available.
|
|
13
|
+
*
|
|
14
|
+
* @default public.ecr.aws/sam/build-go1.x:latest
|
|
15
|
+
* @stability stable
|
|
16
|
+
*/
|
|
17
|
+
readonly buildImage?: string;
|
|
7
18
|
/**
|
|
8
19
|
* The source of the docker image.
|
|
9
20
|
*
|
package/lib/index.js
CHANGED
|
@@ -9,7 +9,7 @@ const child_process = require("child_process");
|
|
|
9
9
|
const path = require("path");
|
|
10
10
|
const aws_cdk_lib_1 = require("aws-cdk-lib");
|
|
11
11
|
const constructs_1 = require("constructs");
|
|
12
|
-
function getCode() {
|
|
12
|
+
function getCode(buildImage) {
|
|
13
13
|
const { CI, NO_PREBUILT_LAMBDA } = process.env;
|
|
14
14
|
if (!(CI && ['true', true, 1, '1'].includes(CI)) || (NO_PREBUILT_LAMBDA && ['true', true, 1, '1'].includes(NO_PREBUILT_LAMBDA))) {
|
|
15
15
|
try {
|
|
@@ -24,7 +24,11 @@ function getCode() {
|
|
|
24
24
|
}
|
|
25
25
|
}
|
|
26
26
|
console.log('Build lambda from scratch');
|
|
27
|
-
return aws_cdk_lib_1.aws_lambda.Code.fromDockerBuild(path.join(__dirname, '../lambda')
|
|
27
|
+
return aws_cdk_lib_1.aws_lambda.Code.fromDockerBuild(path.join(__dirname, '../lambda'), {
|
|
28
|
+
buildArgs: {
|
|
29
|
+
buildImage,
|
|
30
|
+
},
|
|
31
|
+
});
|
|
28
32
|
}
|
|
29
33
|
/**
|
|
30
34
|
* @stability stable
|
|
@@ -48,7 +52,7 @@ class DockerImageName {
|
|
|
48
52
|
}
|
|
49
53
|
exports.DockerImageName = DockerImageName;
|
|
50
54
|
_a = JSII_RTTI_SYMBOL_1;
|
|
51
|
-
DockerImageName[_a] = { fqn: "cdk-ecr-deployment.DockerImageName", version: "2.2
|
|
55
|
+
DockerImageName[_a] = { fqn: "cdk-ecr-deployment.DockerImageName", version: "2.3.2" };
|
|
52
56
|
/**
|
|
53
57
|
* @stability stable
|
|
54
58
|
*/
|
|
@@ -74,7 +78,7 @@ class S3ArchiveName {
|
|
|
74
78
|
}
|
|
75
79
|
exports.S3ArchiveName = S3ArchiveName;
|
|
76
80
|
_b = JSII_RTTI_SYMBOL_1;
|
|
77
|
-
S3ArchiveName[_b] = { fqn: "cdk-ecr-deployment.S3ArchiveName", version: "2.2
|
|
81
|
+
S3ArchiveName[_b] = { fqn: "cdk-ecr-deployment.S3ArchiveName", version: "2.3.2" };
|
|
78
82
|
/**
|
|
79
83
|
* @stability stable
|
|
80
84
|
*/
|
|
@@ -83,12 +87,12 @@ class ECRDeployment extends constructs_1.Construct {
|
|
|
83
87
|
* @stability stable
|
|
84
88
|
*/
|
|
85
89
|
constructor(scope, id, props) {
|
|
86
|
-
var _d;
|
|
90
|
+
var _d, _e;
|
|
87
91
|
super(scope, id);
|
|
88
92
|
const memoryLimit = (_d = props.memoryLimit) !== null && _d !== void 0 ? _d : 512;
|
|
89
93
|
const handler = new aws_cdk_lib_1.aws_lambda.SingletonFunction(this, 'CustomResourceHandler', {
|
|
90
94
|
uuid: this.renderSingletonUuid(memoryLimit),
|
|
91
|
-
code: getCode(),
|
|
95
|
+
code: getCode((_e = props.buildImage) !== null && _e !== void 0 ? _e : 'public.ecr.aws/sam/build-go1.x:latest'),
|
|
92
96
|
runtime: aws_cdk_lib_1.aws_lambda.Runtime.GO_1_X,
|
|
93
97
|
handler: 'main',
|
|
94
98
|
environment: props.environment,
|
|
@@ -157,5 +161,5 @@ class ECRDeployment extends constructs_1.Construct {
|
|
|
157
161
|
}
|
|
158
162
|
exports.ECRDeployment = ECRDeployment;
|
|
159
163
|
_c = JSII_RTTI_SYMBOL_1;
|
|
160
|
-
ECRDeployment[_c] = { fqn: "cdk-ecr-deployment.ECRDeployment", version: "2.2
|
|
161
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"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","sourcesContent":["// Copyright Amazon.com, Inc. or its affiliates. All Rights Reserved.\n// SPDX-License-Identifier: Apache-2.0\n\n\nimport * as child_process from 'child_process';\nimport * as path from 'path';\nimport { aws_ec2 as ec2, aws_iam as iam, aws_lambda as lambda, Duration, CustomResource, Token } from 'aws-cdk-lib';\nimport { Construct } from 'constructs';\n\nexport interface ECRDeploymentProps {\n                                                \n  readonly src: IImageName;\n\n                                                     \n  readonly dest: IImageName;\n\n                                                                                                                                                                                                                                                                                               \n  readonly memoryLimit?: number;\n\n                                                                                                                  \n  readonly role?: iam.IRole;\n\n                                                                                                   \n  readonly vpc?: ec2.IVpc;\n\n                                                                                                                                                                             \n  readonly vpcSubnets?: ec2.SubnetSelection;\n\n                                                \n  readonly environment?: { [key: string]: string };\n}\n\nexport interface IImageName {\n                                                                                                                   \n  readonly uri: string;\n\n                                                                              \n  creds?: string;\n}\n\nfunction getCode(): lambda.AssetCode {\n  const { CI, NO_PREBUILT_LAMBDA } = process.env;\n  if (!(CI && ['true', true, 1, '1'].includes(CI)) || (NO_PREBUILT_LAMBDA && ['true', true, 1, '1'].includes(NO_PREBUILT_LAMBDA))) {\n    try {\n      console.log('Try to get prebuilt lambda');\n\n      const installScript = path.join(__dirname, '../lambda/install.js');\n      const prebuiltPath = path.join(__dirname, '../lambda/out');\n      child_process.execSync(`${process.argv0} ${installScript} ${prebuiltPath}`);\n\n      return lambda.Code.fromAsset(prebuiltPath);\n    } catch (err) {\n      console.warn(`Can not get prebuilt lambda: ${err}`);\n    }\n  }\n\n  console.log('Build lambda from scratch');\n\n  return lambda.Code.fromDockerBuild(path.join(__dirname, '../lambda'));\n}\n\nexport class DockerImageName implements IImageName {\n  public constructor(private name: string, public creds?: string) { }\n  public get uri(): string { return `docker://${this.name}`; }\n}\n\nexport class S3ArchiveName implements IImageName {\n  private name: string;\n  public constructor(p: string, ref?: string, public creds?: string) {\n    this.name = p;\n    if (ref) {\n      this.name += ':' + ref;\n    }\n  }\n  public get uri(): string { return `s3://${this.name}`; }\n}\n\nexport class ECRDeployment extends Construct {\n  constructor(scope: Construct, id: string, props: ECRDeploymentProps) {\n    super(scope, id);\n    const memoryLimit = props.memoryLimit ?? 512;\n    const handler = new lambda.SingletonFunction(this, 'CustomResourceHandler', {\n      uuid: this.renderSingletonUuid(memoryLimit),\n      code: getCode(),\n      runtime: lambda.Runtime.GO_1_X,\n      handler: 'main',\n      environment: props.environment,\n      lambdaPurpose: 'Custom::CDKECRDeployment',\n      timeout: Duration.minutes(15),\n      role: props.role,\n      memorySize: memoryLimit,\n      vpc: props.vpc,\n      vpcSubnets: props.vpcSubnets,\n    });\n\n    const handlerRole = handler.role;\n    if (!handlerRole) { throw new Error('lambda.SingletonFunction should have created a Role'); }\n\n    handlerRole.addToPrincipalPolicy(\n      new iam.PolicyStatement({\n        effect: iam.Effect.ALLOW,\n        actions: [\n          'ecr:GetAuthorizationToken',\n          'ecr:BatchCheckLayerAvailability',\n          'ecr:GetDownloadUrlForLayer',\n          'ecr:GetRepositoryPolicy',\n          'ecr:DescribeRepositories',\n          'ecr:ListImages',\n          'ecr:DescribeImages',\n          'ecr:BatchGetImage',\n          'ecr:ListTagsForResource',\n          'ecr:DescribeImageScanFindings',\n          'ecr:InitiateLayerUpload',\n          'ecr:UploadLayerPart',\n          'ecr:CompleteLayerUpload',\n          'ecr:PutImage',\n        ],\n        resources: ['*'],\n      }));\n    handlerRole.addToPrincipalPolicy(new iam.PolicyStatement({\n      effect: iam.Effect.ALLOW,\n      actions: [\n        's3:GetObject',\n      ],\n      resources: ['*'],\n    }));\n\n    new CustomResource(this, 'CustomResource', {\n      serviceToken: handler.functionArn,\n      resourceType: 'Custom::CDKBucketDeployment',\n      properties: {\n        SrcImage: props.src.uri,\n        SrcCreds: props.src.creds,\n        DestImage: props.dest.uri,\n        DestCreds: props.dest.creds,\n      },\n    });\n  }\n\n  private renderSingletonUuid(memoryLimit?: number) {\n    let uuid = 'bd07c930-edb9-4112-a20f-03f096f53666';\n\n    // if user specify a custom memory limit, define another singleton handler\n    // with this configuration. otherwise, it won't be possible to use multiple\n    // configurations since we have a singleton.\n    if (memoryLimit) {\n      if (Token.isUnresolved(memoryLimit)) {\n        throw new Error('Can\\'t use tokens when specifying \"memoryLimit\" since we use it to identify the singleton custom resource handler');\n      }\n\n      uuid += `-${memoryLimit.toString()}MiB`;\n    }\n\n    return uuid;\n  }\n}\n"]}
|
|
164
|
+
ECRDeployment[_c] = { fqn: "cdk-ecr-deployment.ECRDeployment", version: "2.3.2" };
|
|
165
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"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","sourcesContent":["// Copyright Amazon.com, Inc. or its affiliates. All Rights Reserved.\n// SPDX-License-Identifier: Apache-2.0\n\n\nimport * as child_process from 'child_process';\nimport * as path from 'path';\nimport { aws_ec2 as ec2, aws_iam as iam, aws_lambda as lambda, Duration, CustomResource, Token } from 'aws-cdk-lib';\nimport { Construct } from 'constructs';\n\nexport interface ECRDeploymentProps {\n\n                                                                                                                                                                                                                                                                                                                                                       \n  readonly buildImage?: string;\n                                                \n  readonly src: IImageName;\n\n                                                     \n  readonly dest: IImageName;\n\n                                                                                                                                                                                                                                                                                               \n  readonly memoryLimit?: number;\n\n                                                                                                                  \n  readonly role?: iam.IRole;\n\n                                                                                                   \n  readonly vpc?: ec2.IVpc;\n\n                                                                                                                                                                             \n  readonly vpcSubnets?: ec2.SubnetSelection;\n\n                                                \n  readonly environment?: { [key: string]: string };\n}\n\nexport interface IImageName {\n                                                                                                                   \n  readonly uri: string;\n\n                                                                              \n  creds?: string;\n}\n\nfunction getCode(buildImage: string): lambda.AssetCode {\n  const { CI, NO_PREBUILT_LAMBDA } = process.env;\n  if (!(CI && ['true', true, 1, '1'].includes(CI)) || (NO_PREBUILT_LAMBDA && ['true', true, 1, '1'].includes(NO_PREBUILT_LAMBDA))) {\n    try {\n      console.log('Try to get prebuilt lambda');\n\n      const installScript = path.join(__dirname, '../lambda/install.js');\n      const prebuiltPath = path.join(__dirname, '../lambda/out');\n      child_process.execSync(`${process.argv0} ${installScript} ${prebuiltPath}`);\n\n      return lambda.Code.fromAsset(prebuiltPath);\n    } catch (err) {\n      console.warn(`Can not get prebuilt lambda: ${err}`);\n    }\n  }\n\n  console.log('Build lambda from scratch');\n\n  return lambda.Code.fromDockerBuild(path.join(__dirname, '../lambda'), {\n    buildArgs: {\n      buildImage,\n    },\n  });\n}\n\nexport class DockerImageName implements IImageName {\n  public constructor(private name: string, public creds?: string) { }\n  public get uri(): string { return `docker://${this.name}`; }\n}\n\nexport class S3ArchiveName implements IImageName {\n  private name: string;\n  public constructor(p: string, ref?: string, public creds?: string) {\n    this.name = p;\n    if (ref) {\n      this.name += ':' + ref;\n    }\n  }\n  public get uri(): string { return `s3://${this.name}`; }\n}\n\nexport class ECRDeployment extends Construct {\n  constructor(scope: Construct, id: string, props: ECRDeploymentProps) {\n    super(scope, id);\n    const memoryLimit = props.memoryLimit ?? 512;\n    const handler = new lambda.SingletonFunction(this, 'CustomResourceHandler', {\n      uuid: this.renderSingletonUuid(memoryLimit),\n      code: getCode(props.buildImage ?? 'public.ecr.aws/sam/build-go1.x:latest'),\n      runtime: lambda.Runtime.GO_1_X,\n      handler: 'main',\n      environment: props.environment,\n      lambdaPurpose: 'Custom::CDKECRDeployment',\n      timeout: Duration.minutes(15),\n      role: props.role,\n      memorySize: memoryLimit,\n      vpc: props.vpc,\n      vpcSubnets: props.vpcSubnets,\n    });\n\n    const handlerRole = handler.role;\n    if (!handlerRole) { throw new Error('lambda.SingletonFunction should have created a Role'); }\n\n    handlerRole.addToPrincipalPolicy(\n      new iam.PolicyStatement({\n        effect: iam.Effect.ALLOW,\n        actions: [\n          'ecr:GetAuthorizationToken',\n          'ecr:BatchCheckLayerAvailability',\n          'ecr:GetDownloadUrlForLayer',\n          'ecr:GetRepositoryPolicy',\n          'ecr:DescribeRepositories',\n          'ecr:ListImages',\n          'ecr:DescribeImages',\n          'ecr:BatchGetImage',\n          'ecr:ListTagsForResource',\n          'ecr:DescribeImageScanFindings',\n          'ecr:InitiateLayerUpload',\n          'ecr:UploadLayerPart',\n          'ecr:CompleteLayerUpload',\n          'ecr:PutImage',\n        ],\n        resources: ['*'],\n      }));\n    handlerRole.addToPrincipalPolicy(new iam.PolicyStatement({\n      effect: iam.Effect.ALLOW,\n      actions: [\n        's3:GetObject',\n      ],\n      resources: ['*'],\n    }));\n\n    new CustomResource(this, 'CustomResource', {\n      serviceToken: handler.functionArn,\n      resourceType: 'Custom::CDKBucketDeployment',\n      properties: {\n        SrcImage: props.src.uri,\n        SrcCreds: props.src.creds,\n        DestImage: props.dest.uri,\n        DestCreds: props.dest.creds,\n      },\n    });\n  }\n\n  private renderSingletonUuid(memoryLimit?: number) {\n    let uuid = 'bd07c930-edb9-4112-a20f-03f096f53666';\n\n    // if user specify a custom memory limit, define another singleton handler\n    // with this configuration. otherwise, it won't be possible to use multiple\n    // configurations since we have a singleton.\n    if (memoryLimit) {\n      if (Token.isUnresolved(memoryLimit)) {\n        throw new Error('Can\\'t use tokens when specifying \"memoryLimit\" since we use it to identify the singleton custom resource handler');\n      }\n\n      uuid += `-${memoryLimit.toString()}MiB`;\n    }\n\n    return uuid;\n  }\n}\n"]}
|
|
@@ -0,0 +1,29 @@
|
|
|
1
|
+
name: build
|
|
2
|
+
|
|
3
|
+
on: [push, pull_request]
|
|
4
|
+
|
|
5
|
+
jobs:
|
|
6
|
+
test:
|
|
7
|
+
name: Test
|
|
8
|
+
runs-on: ${{ matrix.os }}
|
|
9
|
+
|
|
10
|
+
strategy:
|
|
11
|
+
matrix:
|
|
12
|
+
node-version: [10.x, 12.x, 14.x]
|
|
13
|
+
os: [ubuntu-latest, windows-latest, macOS-latest]
|
|
14
|
+
|
|
15
|
+
steps:
|
|
16
|
+
- uses: actions/checkout@v2
|
|
17
|
+
|
|
18
|
+
- name: Use Node.js ${{ matrix.node-version }}
|
|
19
|
+
uses: actions/setup-node@v1
|
|
20
|
+
with:
|
|
21
|
+
node-version: ${{ matrix.node-version }}
|
|
22
|
+
|
|
23
|
+
- name: Install
|
|
24
|
+
run: |
|
|
25
|
+
npm install
|
|
26
|
+
|
|
27
|
+
- name: Test
|
|
28
|
+
run: |
|
|
29
|
+
npm test
|
|
@@ -0,0 +1,21 @@
|
|
|
1
|
+
MIT License
|
|
2
|
+
|
|
3
|
+
Copyright (c) 2020 Tomas Della Vedova
|
|
4
|
+
|
|
5
|
+
Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
6
|
+
of this software and associated documentation files (the "Software"), to deal
|
|
7
|
+
in the Software without restriction, including without limitation the rights
|
|
8
|
+
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
9
|
+
copies of the Software, and to permit persons to whom the Software is
|
|
10
|
+
furnished to do so, subject to the following conditions:
|
|
11
|
+
|
|
12
|
+
The above copyright notice and this permission notice shall be included in all
|
|
13
|
+
copies or substantial portions of the Software.
|
|
14
|
+
|
|
15
|
+
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
16
|
+
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
17
|
+
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
18
|
+
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
19
|
+
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
20
|
+
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|
21
|
+
SOFTWARE.
|
|
@@ -0,0 +1,138 @@
|
|
|
1
|
+
# hpagent
|
|
2
|
+
|
|
3
|
+
[](http://standardjs.com/) 
|
|
4
|
+
|
|
5
|
+
A ready to use http and https agent for working with proxies that keeps connections alive!
|
|
6
|
+
|
|
7
|
+
## Install
|
|
8
|
+
|
|
9
|
+
```
|
|
10
|
+
npm install hpagent
|
|
11
|
+
```
|
|
12
|
+
|
|
13
|
+
## Usage
|
|
14
|
+
|
|
15
|
+
Based on your infrastructure, you should use the http agent or the https agent.
|
|
16
|
+
The following table will help you picking the right one.
|
|
17
|
+
|
|
18
|
+
| Type | Proxy | Server |
|
|
19
|
+
|-------------------|--------|--------|
|
|
20
|
+
| `HttpProxyAgent` | HTTP | HTTP |
|
|
21
|
+
| `HttpProxyAgent` | HTTPS | HTTP |
|
|
22
|
+
| `HttpsProxyAgent` | HTTP | HTTPS |
|
|
23
|
+
| `HttpsProxyAgent` | HTTPS | HTTPS |
|
|
24
|
+
|
|
25
|
+
```js
|
|
26
|
+
const { HttpProxyAgent, HttpsProxyAgent } = require('hpagent')
|
|
27
|
+
```
|
|
28
|
+
|
|
29
|
+
Once you have understood the right agent for your use case, you can instance it. It takes the same parameter of the Node.js core's http(s) agent and an additional `proxy` option, which is the url of your proxy.
|
|
30
|
+
|
|
31
|
+
```js
|
|
32
|
+
const http = require('http')
|
|
33
|
+
const { HttpProxyAgent } = require('hpagent')
|
|
34
|
+
|
|
35
|
+
const agent = new HttpProxyAgent({
|
|
36
|
+
keepAlive: true,
|
|
37
|
+
keepAliveMsecs: 1000,
|
|
38
|
+
maxSockets: 256,
|
|
39
|
+
maxFreeSockets: 256,
|
|
40
|
+
proxy: 'http://localhost:8080'
|
|
41
|
+
})
|
|
42
|
+
|
|
43
|
+
http.get('http://localhost:9200', { agent })
|
|
44
|
+
.on('response', console.log)
|
|
45
|
+
.end()
|
|
46
|
+
```
|
|
47
|
+
|
|
48
|
+
If your proxy requires basic authentication, you can configure it in the proxy url:
|
|
49
|
+
|
|
50
|
+
```js
|
|
51
|
+
const http = require('http')
|
|
52
|
+
const { HttpProxyAgent } = require('hpagent')
|
|
53
|
+
|
|
54
|
+
const agent = new HttpProxyAgent({
|
|
55
|
+
keepAlive: true,
|
|
56
|
+
keepAliveMsecs: 1000,
|
|
57
|
+
maxSockets: 256,
|
|
58
|
+
maxFreeSockets: 256,
|
|
59
|
+
proxy: 'http://user:pwd@localhost:8080'
|
|
60
|
+
})
|
|
61
|
+
|
|
62
|
+
http.get('http://localhost:9200', { agent })
|
|
63
|
+
.on('response', console.log)
|
|
64
|
+
.end()
|
|
65
|
+
```
|
|
66
|
+
|
|
67
|
+
## Integrations
|
|
68
|
+
|
|
69
|
+
Following you can find the list of userland http libraries that are tested with this agent.
|
|
70
|
+
|
|
71
|
+
### [got](https://github.com/sindresorhus/got)
|
|
72
|
+
|
|
73
|
+
```js
|
|
74
|
+
got('http://localhost:9200', {
|
|
75
|
+
agent: {
|
|
76
|
+
http: new HttpProxyAgent({
|
|
77
|
+
keepAlive: true,
|
|
78
|
+
keepAliveMsecs: 1000,
|
|
79
|
+
maxSockets: 256,
|
|
80
|
+
maxFreeSockets: 256,
|
|
81
|
+
scheduling: 'lifo',
|
|
82
|
+
proxy: 'http://localhost:8080'
|
|
83
|
+
})
|
|
84
|
+
}
|
|
85
|
+
})
|
|
86
|
+
```
|
|
87
|
+
|
|
88
|
+
### [needle](https://github.com/tomas/needle)
|
|
89
|
+
|
|
90
|
+
```js
|
|
91
|
+
needle('get', 'http://localhost:9200', {
|
|
92
|
+
agent: new HttpProxyAgent({
|
|
93
|
+
keepAlive: true,
|
|
94
|
+
keepAliveMsecs: 1000,
|
|
95
|
+
maxSockets: 256,
|
|
96
|
+
maxFreeSockets: 256,
|
|
97
|
+
scheduling: 'lifo',
|
|
98
|
+
proxy: 'http://localhost:8080'
|
|
99
|
+
})
|
|
100
|
+
})
|
|
101
|
+
```
|
|
102
|
+
|
|
103
|
+
### [node-fetch](https://github.com/node-fetch/node-fetch)
|
|
104
|
+
|
|
105
|
+
```js
|
|
106
|
+
fetch('http://localhost:9200', {
|
|
107
|
+
agent: new HttpProxyAgent({
|
|
108
|
+
keepAlive: true,
|
|
109
|
+
keepAliveMsecs: 1000,
|
|
110
|
+
maxSockets: 256,
|
|
111
|
+
maxFreeSockets: 256,
|
|
112
|
+
scheduling: 'lifo',
|
|
113
|
+
proxy: 'http://localhost:8080'
|
|
114
|
+
})
|
|
115
|
+
})
|
|
116
|
+
```
|
|
117
|
+
|
|
118
|
+
### [simple-get](https://github.com/feross/simple-get)
|
|
119
|
+
|
|
120
|
+
```js
|
|
121
|
+
sget.concat({
|
|
122
|
+
url: `http://${server.address().address}:${server.address().port}`,
|
|
123
|
+
agent: new HttpProxyAgent({
|
|
124
|
+
keepAlive: true,
|
|
125
|
+
keepAliveMsecs: 1000,
|
|
126
|
+
maxSockets: 256,
|
|
127
|
+
maxFreeSockets: 256,
|
|
128
|
+
scheduling: 'lifo',
|
|
129
|
+
proxy: `https://${proxy.address().address}:${proxy.address().port}`
|
|
130
|
+
})
|
|
131
|
+
}, function (err, response, data) {
|
|
132
|
+
// handle the response
|
|
133
|
+
})
|
|
134
|
+
```
|
|
135
|
+
|
|
136
|
+
## License
|
|
137
|
+
|
|
138
|
+
This software is licensed under the [MIT](./LICENSE).
|
|
@@ -0,0 +1,26 @@
|
|
|
1
|
+
import * as http from 'http'
|
|
2
|
+
import * as https from 'https'
|
|
3
|
+
import { URL } from 'url'
|
|
4
|
+
|
|
5
|
+
declare class HttpProxyAgent extends http.Agent {
|
|
6
|
+
constructor(options: HttpProxyAgentOptions)
|
|
7
|
+
}
|
|
8
|
+
|
|
9
|
+
interface HttpProxyAgentOptions extends http.AgentOptions {
|
|
10
|
+
proxy: string | URL
|
|
11
|
+
}
|
|
12
|
+
|
|
13
|
+
declare class HttpsProxyAgent extends https.Agent {
|
|
14
|
+
constructor(options: HttpsProxyAgentOptions)
|
|
15
|
+
}
|
|
16
|
+
|
|
17
|
+
interface HttpsProxyAgentOptions extends https.AgentOptions {
|
|
18
|
+
proxy: string | URL
|
|
19
|
+
}
|
|
20
|
+
|
|
21
|
+
export {
|
|
22
|
+
HttpProxyAgent,
|
|
23
|
+
HttpProxyAgentOptions,
|
|
24
|
+
HttpsProxyAgent,
|
|
25
|
+
HttpsProxyAgentOptions
|
|
26
|
+
}
|
|
@@ -0,0 +1,101 @@
|
|
|
1
|
+
'use strict'
|
|
2
|
+
|
|
3
|
+
const https = require('https')
|
|
4
|
+
const http = require('http')
|
|
5
|
+
const { URL } = require('url')
|
|
6
|
+
|
|
7
|
+
class HttpProxyAgent extends http.Agent {
|
|
8
|
+
constructor (options) {
|
|
9
|
+
const { proxy, ...opts } = options
|
|
10
|
+
super(opts)
|
|
11
|
+
this.proxy = typeof proxy === 'string'
|
|
12
|
+
? new URL(proxy)
|
|
13
|
+
: proxy
|
|
14
|
+
}
|
|
15
|
+
|
|
16
|
+
createConnection (options, callback) {
|
|
17
|
+
const requestOptions = {
|
|
18
|
+
method: 'CONNECT',
|
|
19
|
+
host: this.proxy.hostname,
|
|
20
|
+
port: this.proxy.port,
|
|
21
|
+
path: `${options.host}:${options.port}`,
|
|
22
|
+
setHost: false,
|
|
23
|
+
headers: { connection: this.keepAlive ? 'keep-alive' : 'close', host: `${options.host}:${options.port}` },
|
|
24
|
+
agent: false
|
|
25
|
+
}
|
|
26
|
+
|
|
27
|
+
if (this.proxy.username || this.proxy.password) {
|
|
28
|
+
const base64 = Buffer.from(`${this.proxy.username || ''}:${this.proxy.password || ''}`).toString('base64')
|
|
29
|
+
requestOptions.headers['proxy-authorization'] = `Basic ${base64}`
|
|
30
|
+
}
|
|
31
|
+
|
|
32
|
+
const request = (this.proxy.protocol === 'http:' ? http : https).request(requestOptions)
|
|
33
|
+
request.once('connect', (response, socket, head) => {
|
|
34
|
+
request.removeAllListeners()
|
|
35
|
+
socket.removeAllListeners()
|
|
36
|
+
if (response.statusCode === 200) {
|
|
37
|
+
callback(null, socket)
|
|
38
|
+
} else {
|
|
39
|
+
callback(new Error(`Bad response: ${response.statusCode}`), null)
|
|
40
|
+
}
|
|
41
|
+
})
|
|
42
|
+
|
|
43
|
+
request.once('error', err => {
|
|
44
|
+
request.removeAllListeners()
|
|
45
|
+
callback(err, null)
|
|
46
|
+
})
|
|
47
|
+
|
|
48
|
+
request.end()
|
|
49
|
+
}
|
|
50
|
+
}
|
|
51
|
+
|
|
52
|
+
class HttpsProxyAgent extends https.Agent {
|
|
53
|
+
constructor (options) {
|
|
54
|
+
const { proxy, ...opts } = options
|
|
55
|
+
super(opts)
|
|
56
|
+
this.proxy = typeof proxy === 'string'
|
|
57
|
+
? new URL(proxy)
|
|
58
|
+
: proxy
|
|
59
|
+
}
|
|
60
|
+
|
|
61
|
+
createConnection (options, callback) {
|
|
62
|
+
const requestOptions = {
|
|
63
|
+
method: 'CONNECT',
|
|
64
|
+
host: this.proxy.hostname,
|
|
65
|
+
port: this.proxy.port,
|
|
66
|
+
path: `${options.host}:${options.port}`,
|
|
67
|
+
setHost: false,
|
|
68
|
+
headers: { connection: this.keepAlive ? 'keep-alive' : 'close', host: `${options.host}:${options.port}` },
|
|
69
|
+
agent: false
|
|
70
|
+
}
|
|
71
|
+
|
|
72
|
+
if (this.proxy.username || this.proxy.password) {
|
|
73
|
+
const base64 = Buffer.from(`${this.proxy.username || ''}:${this.proxy.password || ''}`).toString('base64')
|
|
74
|
+
requestOptions.headers['proxy-authorization'] = `Basic ${base64}`
|
|
75
|
+
}
|
|
76
|
+
|
|
77
|
+
const request = (this.proxy.protocol === 'http:' ? http : https).request(requestOptions)
|
|
78
|
+
request.once('connect', (response, socket, head) => {
|
|
79
|
+
request.removeAllListeners()
|
|
80
|
+
socket.removeAllListeners()
|
|
81
|
+
if (response.statusCode === 200) {
|
|
82
|
+
const secureSocket = super.createConnection({ ...options, socket })
|
|
83
|
+
callback(null, secureSocket)
|
|
84
|
+
} else {
|
|
85
|
+
callback(new Error(`Bad response: ${response.statusCode}`), null)
|
|
86
|
+
}
|
|
87
|
+
})
|
|
88
|
+
|
|
89
|
+
request.once('error', err => {
|
|
90
|
+
request.removeAllListeners()
|
|
91
|
+
callback(err, null)
|
|
92
|
+
})
|
|
93
|
+
|
|
94
|
+
request.end()
|
|
95
|
+
}
|
|
96
|
+
}
|
|
97
|
+
|
|
98
|
+
module.exports = {
|
|
99
|
+
HttpProxyAgent,
|
|
100
|
+
HttpsProxyAgent
|
|
101
|
+
}
|
|
@@ -0,0 +1,49 @@
|
|
|
1
|
+
{
|
|
2
|
+
"name": "hpagent",
|
|
3
|
+
"version": "0.1.2",
|
|
4
|
+
"description": "A ready to use http and https agent for working with proxies that keeps connections alive!",
|
|
5
|
+
"main": "index.js",
|
|
6
|
+
"types": "index.d.ts",
|
|
7
|
+
"exports": {
|
|
8
|
+
".": {
|
|
9
|
+
"require": "./index.js",
|
|
10
|
+
"import": "./index.mjs"
|
|
11
|
+
},
|
|
12
|
+
"./": "./"
|
|
13
|
+
},
|
|
14
|
+
"scripts": {
|
|
15
|
+
"test": "standard && ava -v test/*.test.js && tsd"
|
|
16
|
+
},
|
|
17
|
+
"repository": {
|
|
18
|
+
"type": "git",
|
|
19
|
+
"url": "git+https://github.com/delvedor/hpagent.git"
|
|
20
|
+
},
|
|
21
|
+
"keywords": [
|
|
22
|
+
"agent",
|
|
23
|
+
"http",
|
|
24
|
+
"https",
|
|
25
|
+
"secure",
|
|
26
|
+
"proxy",
|
|
27
|
+
"alive",
|
|
28
|
+
"keep-alive"
|
|
29
|
+
],
|
|
30
|
+
"author": "Tomas Della Vedova",
|
|
31
|
+
"license": "MIT",
|
|
32
|
+
"bugs": {
|
|
33
|
+
"url": "https://github.com/delvedor/hpagent/issues"
|
|
34
|
+
},
|
|
35
|
+
"homepage": "https://github.com/delvedor/hpagent#readme",
|
|
36
|
+
"tsd": {
|
|
37
|
+
"directory": "test"
|
|
38
|
+
},
|
|
39
|
+
"devDependencies": {
|
|
40
|
+
"ava": "^3.10.1",
|
|
41
|
+
"got": "^11.5.1",
|
|
42
|
+
"needle": "^2.5.0",
|
|
43
|
+
"node-fetch": "^2.6.0",
|
|
44
|
+
"proxy": "^1.0.2",
|
|
45
|
+
"simple-get": "^4.0.0",
|
|
46
|
+
"standard": "^16.0.1",
|
|
47
|
+
"tsd": "^0.13.1"
|
|
48
|
+
}
|
|
49
|
+
}
|
|
@@ -0,0 +1,111 @@
|
|
|
1
|
+
'use strict'
|
|
2
|
+
|
|
3
|
+
const got = require('got')
|
|
4
|
+
const test = require('ava')
|
|
5
|
+
const {
|
|
6
|
+
createServer,
|
|
7
|
+
createSecureServer,
|
|
8
|
+
createProxy,
|
|
9
|
+
createSecureProxy
|
|
10
|
+
} = require('./utils')
|
|
11
|
+
const { HttpProxyAgent, HttpsProxyAgent } = require('../')
|
|
12
|
+
|
|
13
|
+
test('http to http', async t => {
|
|
14
|
+
const server = await createServer()
|
|
15
|
+
const proxy = await createProxy()
|
|
16
|
+
server.on('request', (req, res) => res.end('ok'))
|
|
17
|
+
|
|
18
|
+
const response = await got(`http://${server.address().address}:${server.address().port}`, {
|
|
19
|
+
agent: {
|
|
20
|
+
http: new HttpProxyAgent({
|
|
21
|
+
keepAlive: true,
|
|
22
|
+
keepAliveMsecs: 1000,
|
|
23
|
+
maxSockets: 256,
|
|
24
|
+
maxFreeSockets: 256,
|
|
25
|
+
scheduling: 'lifo',
|
|
26
|
+
proxy: `http://${proxy.address().address}:${proxy.address().port}`
|
|
27
|
+
})
|
|
28
|
+
}
|
|
29
|
+
})
|
|
30
|
+
|
|
31
|
+
t.is(response.body, 'ok')
|
|
32
|
+
t.is(response.statusCode, 200)
|
|
33
|
+
|
|
34
|
+
server.close()
|
|
35
|
+
proxy.close()
|
|
36
|
+
})
|
|
37
|
+
|
|
38
|
+
test('https to http', async t => {
|
|
39
|
+
const server = await createServer()
|
|
40
|
+
const proxy = await createSecureProxy()
|
|
41
|
+
server.on('request', (req, res) => res.end('ok'))
|
|
42
|
+
|
|
43
|
+
const response = await got(`http://${server.address().address}:${server.address().port}`, {
|
|
44
|
+
agent: {
|
|
45
|
+
http: new HttpProxyAgent({
|
|
46
|
+
keepAlive: true,
|
|
47
|
+
keepAliveMsecs: 1000,
|
|
48
|
+
maxSockets: 256,
|
|
49
|
+
maxFreeSockets: 256,
|
|
50
|
+
scheduling: 'lifo',
|
|
51
|
+
proxy: `https://${proxy.address().address}:${proxy.address().port}`
|
|
52
|
+
})
|
|
53
|
+
}
|
|
54
|
+
})
|
|
55
|
+
|
|
56
|
+
t.is(response.body, 'ok')
|
|
57
|
+
t.is(response.statusCode, 200)
|
|
58
|
+
|
|
59
|
+
server.close()
|
|
60
|
+
proxy.close()
|
|
61
|
+
})
|
|
62
|
+
|
|
63
|
+
test('http to https', async t => {
|
|
64
|
+
const server = await createSecureServer()
|
|
65
|
+
const proxy = await createProxy()
|
|
66
|
+
server.on('request', (req, res) => res.end('ok'))
|
|
67
|
+
|
|
68
|
+
const response = await got(`https://${server.address().address}:${server.address().port}`, {
|
|
69
|
+
agent: {
|
|
70
|
+
http: new HttpsProxyAgent({
|
|
71
|
+
keepAlive: true,
|
|
72
|
+
keepAliveMsecs: 1000,
|
|
73
|
+
maxSockets: 256,
|
|
74
|
+
maxFreeSockets: 256,
|
|
75
|
+
scheduling: 'lifo',
|
|
76
|
+
proxy: `http://${proxy.address().address}:${proxy.address().port}`
|
|
77
|
+
})
|
|
78
|
+
}
|
|
79
|
+
})
|
|
80
|
+
|
|
81
|
+
t.is(response.body, 'ok')
|
|
82
|
+
t.is(response.statusCode, 200)
|
|
83
|
+
|
|
84
|
+
server.close()
|
|
85
|
+
proxy.close()
|
|
86
|
+
})
|
|
87
|
+
|
|
88
|
+
test('https to https', async t => {
|
|
89
|
+
const server = await createSecureServer()
|
|
90
|
+
const proxy = await createSecureProxy()
|
|
91
|
+
server.on('request', (req, res) => res.end('ok'))
|
|
92
|
+
|
|
93
|
+
const response = await got(`https://${server.address().address}:${server.address().port}`, {
|
|
94
|
+
agent: {
|
|
95
|
+
http: new HttpsProxyAgent({
|
|
96
|
+
keepAlive: true,
|
|
97
|
+
keepAliveMsecs: 1000,
|
|
98
|
+
maxSockets: 256,
|
|
99
|
+
maxFreeSockets: 256,
|
|
100
|
+
scheduling: 'lifo',
|
|
101
|
+
proxy: `https://${proxy.address().address}:${proxy.address().port}`
|
|
102
|
+
})
|
|
103
|
+
}
|
|
104
|
+
})
|
|
105
|
+
|
|
106
|
+
t.is(response.body, 'ok')
|
|
107
|
+
t.is(response.statusCode, 200)
|
|
108
|
+
|
|
109
|
+
server.close()
|
|
110
|
+
proxy.close()
|
|
111
|
+
})
|