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/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
- // TODO: Support proxy download
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.0" };
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.0" };
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.0" };
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,8 @@
1
+ version: 2
2
+ updates:
3
+ - package-ecosystem: npm
4
+ directory: "/"
5
+ schedule:
6
+ interval: daily
7
+ time: "04:00"
8
+ open-pull-requests-limit: 10
@@ -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
+ [![js-standard-style](https://img.shields.io/badge/code%20style-standard-brightgreen.svg?style=flat)](http://standardjs.com/) ![build](https://github.com/delvedor/hpagent/workflows/build/badge.svg)
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,5 @@
1
+ import mod from './index.js'
2
+
3
+ export default mod
4
+ export const HttpProxyAgent = mod.HttpProxyAgent
5
+ export const HttpsProxyAgent = mod.HttpsProxyAgent
@@ -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
+ })