UNPKG

@cdklabs/cdk-hyperledger-fabric-network

Version:

CDK construct to deploy a Hyperledger Fabric network running on Amazon Managed Blockchain

132 lines 20.2 kB
"use strict"; // Copyright Amazon.com, Inc. or its affiliates. All Rights Reserved. // SPDX-License-Identifier: MIT-0 Object.defineProperty(exports, "__esModule", { value: true }); exports.HyperledgerFabricIdentity = void 0; const child_process_1 = require("child_process"); const path = require("path"); const cdk = require("aws-cdk-lib"); const ec2 = require("aws-cdk-lib/aws-ec2"); const iam = require("aws-cdk-lib/aws-iam"); const lambda = require("aws-cdk-lib/aws-lambda"); const logs = require("aws-cdk-lib/aws-logs"); const customresources = require("aws-cdk-lib/custom-resources"); const constructs = require("constructs"); const utilities = require("./utilities"); /** * Creates custom resources to enroll admin and register user * identities with the CA using the fabric-ca-client SDK. * Admin identity is enrolled by default. User identities are * registered and enrolled, if provided. */ class HyperledgerFabricIdentity extends constructs.Construct { constructor(scope, id) { super(scope, id); // Collect metadata on the stack const partition = cdk.Stack.of(this).partition; const region = cdk.Stack.of(this).region; // Retrieve the S3 Bucket and key that contains the TLS cert file const tlsBucketData = utilities.getTlsBucket(region); const adminPasswordArn = scope.adminPasswordSecret.secretArn; const adminPrivateKeyArn = scope.adminPrivateKeySecret.secretArn; const adminSignedCertArn = scope.adminSignedCertSecret.secretArn; const caEndpoint = scope.caEndpoint; const client = scope.client; const memberName = scope.memberName; // Role for the custom resource lambda functions const customResourceRole = new iam.Role(this, 'CustomResourceRole', { assumedBy: new iam.ServicePrincipal('lambda.amazonaws.com') }); // Policies for the custom resource lambda to enroll and register users customResourceRole.addManagedPolicy(iam.ManagedPolicy.fromAwsManagedPolicyName('service-role/AWSLambdaBasicExecutionRole')); customResourceRole.addManagedPolicy(iam.ManagedPolicy.fromAwsManagedPolicyName('service-role/AWSLambdaVPCAccessExecutionRole')); customResourceRole.addToPolicy(new iam.PolicyStatement({ actions: ['s3:GetObject', 'secretsmanager:CreateSecret', 'secretsmanager:GetSecretValue', 'secretsmanager:PutSecretValue'], resources: [ `arn:${partition}:s3:::${tlsBucketData.bucketName}/*`, adminPasswordArn, adminPrivateKeyArn, adminSignedCertArn, ], })); const lambdaDirectory = '../lambdas/fabric'; // Have to use docker local bundling as esbuild doesn't resolve the path on the fabric-proto package const codeProp = { bundling: { image: lambda.Runtime.NODEJS_14_X.bundlingImage, command: [ 'bash', '-c', 'cp -a . /asset-output', 'npm install --prefix /asset-output', ], local: { tryBundle(outputDir) { try { (0, child_process_1.execSync)('npm --version'); (0, child_process_1.execSync)(`cp -a ${path.join(__dirname, lambdaDirectory)}/. ${outputDir}`); (0, child_process_1.execSync)(`npm install --prefix ${outputDir}`); } catch { return false; } return true; }, }, }, }; // Lambda function to enroll the admin and import credentials to secrets manager. const adminFunction = new lambda.Function(this, 'AdminFunction', { runtime: lambda.Runtime.NODEJS_14_X, handler: 'enroll-admin.handler', code: lambda.Code.fromAsset(path.join(__dirname, lambdaDirectory), codeProp), environment: { ADMIN_PASSWORD_ARN: adminPasswordArn, CA_ENDPOINT: caEndpoint, PRIVATE_KEY_ARN: adminPrivateKeyArn, SIGNED_CERT_ARN: adminSignedCertArn, TLS_CERT_BUCKET: tlsBucketData.bucketName, TLS_CERT_KEY: tlsBucketData.key, }, role: customResourceRole, vpc: client.vpc, vpcSubnets: client.vpc.selectSubnets(), timeout: cdk.Duration.minutes(1), }); // Port range to access the Network const ledgerPortRange = ec2.Port.tcpRange(utilities.STARTING_PORT, utilities.ENDING_PORT); // Add access to the lambda for the Network ports client.vpcEndpoint.connections.allowFrom(adminFunction, ledgerPortRange); // Custom Resource provider this.adminProvider = new customresources.Provider(this, 'AdminProvider', { onEventHandler: adminFunction, logRetention: logs.RetentionDays.ONE_DAY, }); // Lambda function to register and enroll users and // import credentials to secrets manager. const userFunction = new lambda.Function(scope, 'UserFunction', { runtime: lambda.Runtime.NODEJS_14_X, handler: 'register-user.handler', code: lambda.Code.fromAsset(path.join(__dirname, lambdaDirectory), codeProp), environment: { CA_ENDPOINT: caEndpoint, MEMBER_NAME: memberName, PRIVATE_KEY_ARN: adminPrivateKeyArn, SIGNED_CERT_ARN: adminSignedCertArn, TLS_CERT_BUCKET: tlsBucketData.bucketName, TLS_CERT_KEY: tlsBucketData.key, }, role: customResourceRole, vpc: client.vpc, vpcSubnets: client.vpc.selectSubnets(), timeout: cdk.Duration.minutes(1), }); // Add access to the lambda for the Network ports scope.client.vpcEndpoint.connections.allowFrom(userFunction, ledgerPortRange); // Custom Resource provider HyperledgerFabricIdentity.userProvider = new customresources.Provider(scope, 'UserProvider', { onEventHandler: userFunction, logRetention: logs.RetentionDays.ONE_DAY, }); // Populate the custom role static variable HyperledgerFabricIdentity.customRole = customResourceRole; } } exports.HyperledgerFabricIdentity = HyperledgerFabricIdentity; //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"identity.js","sourceRoot":"","sources":["../src/identity.ts"],"names":[],"mappings":";AAAA,qEAAqE;AACrE,iCAAiC;;;AAEjC,iDAAyC;AACzC,6BAA6B;AAC7B,mCAAmC;AACnC,2CAA2C;AAC3C,2CAA2C;AAC3C,iDAAiD;AACjD,6CAA6C;AAC7C,gEAAgE;AAChE,yCAAyC;AAGzC,yCAAyC;AAEzC;;;;;GAKG;AACH,MAAa,yBAA0B,SAAQ,UAAU,CAAC,SAAS;IAkBjE,YAAY,KAAuC,EAAE,EAAU;QAC7D,KAAK,CAAC,KAAK,EAAE,EAAE,CAAC,CAAC;QAEjB,gCAAgC;QAChC,MAAM,SAAS,GAAG,GAAG,CAAC,KAAK,CAAC,EAAE,CAAC,IAAI,CAAC,CAAC,SAAS,CAAC;QAC/C,MAAM,MAAM,GAAG,GAAG,CAAC,KAAK,CAAC,EAAE,CAAC,IAAI,CAAC,CAAC,MAAM,CAAC;QAEzC,iEAAiE;QACjE,MAAM,aAAa,GAAG,SAAS,CAAC,YAAY,CAAC,MAAM,CAAC,CAAC;QAErD,MAAM,gBAAgB,GAAG,KAAK,CAAC,mBAAmB,CAAC,SAAS,CAAC;QAC7D,MAAM,kBAAkB,GAAG,KAAK,CAAC,qBAAqB,CAAC,SAAS,CAAC;QACjE,MAAM,kBAAkB,GAAG,KAAK,CAAC,qBAAqB,CAAC,SAAS,CAAC;QACjE,MAAM,UAAU,GAAG,KAAK,CAAC,UAAU,CAAC;QACpC,MAAM,MAAM,GAAG,KAAK,CAAC,MAAM,CAAC;QAC5B,MAAM,UAAU,GAAG,KAAK,CAAC,UAAU,CAAC;QAEpC,gDAAgD;QAChD,MAAM,kBAAkB,GAAG,IAAI,GAAG,CAAC,IAAI,CAAC,IAAI,EAAE,oBAAoB,EAAE,EAAE,SAAS,EAAE,IAAI,GAAG,CAAC,gBAAgB,CAAC,sBAAsB,CAAC,EAAE,CAAC,CAAC;QAErI,uEAAuE;QACvE,kBAAkB,CAAC,gBAAgB,CAAC,GAAG,CAAC,aAAa,CAAC,wBAAwB,CAAC,0CAA0C,CAAC,CAAC,CAAC;QAC5H,kBAAkB,CAAC,gBAAgB,CAAC,GAAG,CAAC,aAAa,CAAC,wBAAwB,CAAC,8CAA8C,CAAC,CAAC,CAAC;QAChI,kBAAkB,CAAC,WAAW,CAAE,IAAI,GAAG,CAAC,eAAe,CAAC;YACtD,OAAO,EAAE,CAAC,cAAc,EAAE,6BAA6B,EAAE,+BAA+B,EAAE,+BAA+B,CAAC;YAC1H,SAAS,EAAE;gBACT,OAAO,SAAS,SAAS,aAAa,CAAC,UAAU,IAAI;gBACrD,gBAAgB;gBAChB,kBAAkB;gBAClB,kBAAkB;aACnB;SACF,CAAC,CAAC,CAAC;QAEJ,MAAM,eAAe,GAAG,mBAAmB,CAAC;QAE5C,oGAAoG;QACpG,MAAM,QAAQ,GAAG;YACf,QAAQ,EAAE;gBACR,KAAK,EAAE,MAAM,CAAC,OAAO,CAAC,WAAW,CAAC,aAAa;gBAC/C,OAAO,EAAE;oBACP,MAAM,EAAE,IAAI,EAAE,uBAAuB;oBACrC,oCAAoC;iBACrC;gBACD,KAAK,EAAE;oBACL,SAAS,CAAC,SAAiB;wBACzB,IAAI,CAAC;4BACH,IAAA,wBAAQ,EAAC,eAAe,CAAC,CAAC;4BAC1B,IAAA,wBAAQ,EAAC,SAAS,IAAI,CAAC,IAAI,CAAC,SAAS,EAAE,eAAe,CAAC,MAAM,SAAS,EAAE,CAAC,CAAC;4BAC1E,IAAA,wBAAQ,EAAC,wBAAwB,SAAS,EAAE,CAAC,CAAC;wBAChD,CAAC;wBAAC,MAAM,CAAC;4BACP,OAAO,KAAK,CAAC;wBACf,CAAC;wBACD,OAAO,IAAI,CAAC;oBACd,CAAC;iBACF;aACF;SACF,CAAC;QAEF,iFAAiF;QACjF,MAAM,aAAa,GAAG,IAAI,MAAM,CAAC,QAAQ,CAAC,IAAI,EAAE,eAAe,EAAE;YAC/D,OAAO,EAAE,MAAM,CAAC,OAAO,CAAC,WAAW;YACnC,OAAO,EAAE,sBAAsB;YAC/B,IAAI,EAAE,MAAM,CAAC,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,IAAI,CAAC,SAAS,EAAE,eAAe,CAAC,EAAE,QAAQ,CAAC;YAC5E,WAAW,EAAE;gBACX,kBAAkB,EAAE,gBAAgB;gBACpC,WAAW,EAAE,UAAU;gBACvB,eAAe,EAAE,kBAAkB;gBACnC,eAAe,EAAE,kBAAkB;gBACnC,eAAe,EAAE,aAAa,CAAC,UAAU;gBACzC,YAAY,EAAE,aAAa,CAAC,GAAG;aAChC;YACD,IAAI,EAAE,kBAAkB;YACxB,GAAG,EAAE,MAAM,CAAC,GAAG;YACf,UAAU,EAAE,MAAM,CAAC,GAAG,CAAC,aAAa,EAAE;YACtC,OAAO,EAAE,GAAG,CAAC,QAAQ,CAAC,OAAO,CAAC,CAAC,CAAC;SACjC,CAAC,CAAC;QAEH,mCAAmC;QACnC,MAAM,eAAe,GAAG,GAAG,CAAC,IAAI,CAAC,QAAQ,CAAC,SAAS,CAAC,aAAa,EAAE,SAAS,CAAC,WAAW,CAAC,CAAC;QAE1F,iDAAiD;QACjD,MAAM,CAAC,WAAW,CAAC,WAAW,CAAC,SAAS,CAAC,aAAa,EAAE,eAAe,CAAC,CAAC;QAEzE,2BAA2B;QAC3B,IAAI,CAAC,aAAa,GAAG,IAAI,eAAe,CAAC,QAAQ,CAAC,IAAI,EAAE,eAAe,EAAE;YACvE,cAAc,EAAE,aAAa;YAC7B,YAAY,EAAE,IAAI,CAAC,aAAa,CAAC,OAAO;SACzC,CAAC,CAAC;QAEH,mDAAmD;QACnD,yCAAyC;QACzC,MAAM,YAAY,GAAG,IAAI,MAAM,CAAC,QAAQ,CAAC,KAAK,EAAE,cAAc,EAAE;YAC9D,OAAO,EAAE,MAAM,CAAC,OAAO,CAAC,WAAW;YACnC,OAAO,EAAE,uBAAuB;YAChC,IAAI,EAAE,MAAM,CAAC,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,IAAI,CAAC,SAAS,EAAE,eAAe,CAAC,EAAE,QAAQ,CAAC;YAC5E,WAAW,EAAE;gBACX,WAAW,EAAE,UAAU;gBACvB,WAAW,EAAE,UAAU;gBACvB,eAAe,EAAE,kBAAkB;gBACnC,eAAe,EAAE,kBAAkB;gBACnC,eAAe,EAAE,aAAa,CAAC,UAAU;gBACzC,YAAY,EAAE,aAAa,CAAC,GAAG;aAChC;YACD,IAAI,EAAE,kBAAkB;YACxB,GAAG,EAAE,MAAM,CAAC,GAAG;YACf,UAAU,EAAE,MAAM,CAAC,GAAG,CAAC,aAAa,EAAE;YACtC,OAAO,EAAE,GAAG,CAAC,QAAQ,CAAC,OAAO,CAAC,CAAC,CAAC;SACjC,CAAC,CAAC;QAEH,iDAAiD;QACjD,KAAK,CAAC,MAAM,CAAC,WAAW,CAAC,WAAW,CAAC,SAAS,CAAC,YAAY,EAAE,eAAe,CAAC,CAAC;QAE9E,2BAA2B;QAC3B,yBAAyB,CAAC,YAAY,GAAG,IAAI,eAAe,CAAC,QAAQ,CAAC,KAAK,EAAE,cAAc,EAAE;YAC3F,cAAc,EAAE,YAAY;YAC5B,YAAY,EAAE,IAAI,CAAC,aAAa,CAAC,OAAO;SACzC,CAAC,CAAC;QAEH,2CAA2C;QAC3C,yBAAyB,CAAC,UAAU,GAAG,kBAAkB,CAAC;IAE5D,CAAC;CAEF;AA7ID,8DA6IC","sourcesContent":["// Copyright Amazon.com, Inc. or its affiliates. All Rights Reserved.\n// SPDX-License-Identifier: MIT-0\n\nimport { execSync } from 'child_process';\nimport * as path from 'path';\nimport * as cdk from 'aws-cdk-lib';\nimport * as ec2 from 'aws-cdk-lib/aws-ec2';\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 customresources from 'aws-cdk-lib/custom-resources';\nimport * as constructs from 'constructs';\n\nimport * as network from './network';\nimport * as utilities from './utilities';\n\n/**\n * Creates custom resources to enroll admin and register user\n * identities with the CA using the fabric-ca-client SDK.\n * Admin identity is enrolled by default. User identities are\n * registered and enrolled, if provided.\n */\nexport class HyperledgerFabricIdentity extends constructs.Construct {\n\n  /**\n   * Role for custom resource lambda to assume\n   */\n  public static customRole: iam.Role;\n\n  /**\n   * Custom provider to register user identity\n   */\n  public static userProvider: customresources.Provider;\n\n  /**\n   * Custom provider to enroll admin identity\n   */\n  public readonly adminProvider: customresources.Provider;\n\n\n  constructor(scope: network.HyperledgerFabricNetwork, id: string) {\n    super(scope, id);\n\n    // Collect metadata on the stack\n    const partition = cdk.Stack.of(this).partition;\n    const region = cdk.Stack.of(this).region;\n\n    // Retrieve the S3 Bucket and key that contains the TLS cert file\n    const tlsBucketData = utilities.getTlsBucket(region);\n\n    const adminPasswordArn = scope.adminPasswordSecret.secretArn;\n    const adminPrivateKeyArn = scope.adminPrivateKeySecret.secretArn;\n    const adminSignedCertArn = scope.adminSignedCertSecret.secretArn;\n    const caEndpoint = scope.caEndpoint;\n    const client = scope.client;\n    const memberName = scope.memberName;\n\n    // Role for the custom resource lambda functions\n    const customResourceRole = new iam.Role(this, 'CustomResourceRole', { assumedBy: new iam.ServicePrincipal('lambda.amazonaws.com') });\n\n    // Policies for the custom resource lambda to enroll and register users\n    customResourceRole.addManagedPolicy(iam.ManagedPolicy.fromAwsManagedPolicyName('service-role/AWSLambdaBasicExecutionRole'));\n    customResourceRole.addManagedPolicy(iam.ManagedPolicy.fromAwsManagedPolicyName('service-role/AWSLambdaVPCAccessExecutionRole'));\n    customResourceRole.addToPolicy( new iam.PolicyStatement({\n      actions: ['s3:GetObject', 'secretsmanager:CreateSecret', 'secretsmanager:GetSecretValue', 'secretsmanager:PutSecretValue'],\n      resources: [\n        `arn:${partition}:s3:::${tlsBucketData.bucketName}/*`,\n        adminPasswordArn,\n        adminPrivateKeyArn,\n        adminSignedCertArn,\n      ],\n    }));\n\n    const lambdaDirectory = '../lambdas/fabric';\n\n    // Have to use docker local bundling as esbuild doesn't resolve the path on the fabric-proto package\n    const codeProp = {\n      bundling: {\n        image: lambda.Runtime.NODEJS_14_X.bundlingImage,\n        command: [\n          'bash', '-c', 'cp -a . /asset-output',\n          'npm install --prefix /asset-output',\n        ],\n        local: {\n          tryBundle(outputDir: string) {\n            try {\n              execSync('npm --version');\n              execSync(`cp -a ${path.join(__dirname, lambdaDirectory)}/. ${outputDir}`);\n              execSync(`npm install --prefix ${outputDir}`);\n            } catch {\n              return false;\n            }\n            return true;\n          },\n        },\n      },\n    };\n\n    // Lambda function to enroll the admin and import credentials to secrets manager.\n    const adminFunction = new lambda.Function(this, 'AdminFunction', {\n      runtime: lambda.Runtime.NODEJS_14_X,\n      handler: 'enroll-admin.handler',\n      code: lambda.Code.fromAsset(path.join(__dirname, lambdaDirectory), codeProp),\n      environment: {\n        ADMIN_PASSWORD_ARN: adminPasswordArn,\n        CA_ENDPOINT: caEndpoint,\n        PRIVATE_KEY_ARN: adminPrivateKeyArn,\n        SIGNED_CERT_ARN: adminSignedCertArn,\n        TLS_CERT_BUCKET: tlsBucketData.bucketName,\n        TLS_CERT_KEY: tlsBucketData.key,\n      },\n      role: customResourceRole,\n      vpc: client.vpc,\n      vpcSubnets: client.vpc.selectSubnets(),\n      timeout: cdk.Duration.minutes(1),\n    });\n\n    // Port range to access the Network\n    const ledgerPortRange = ec2.Port.tcpRange(utilities.STARTING_PORT, utilities.ENDING_PORT);\n\n    // Add access to the lambda for the Network ports\n    client.vpcEndpoint.connections.allowFrom(adminFunction, ledgerPortRange);\n\n    // Custom Resource provider\n    this.adminProvider = new customresources.Provider(this, 'AdminProvider', {\n      onEventHandler: adminFunction,\n      logRetention: logs.RetentionDays.ONE_DAY,\n    });\n\n    // Lambda function to register and enroll users and\n    // import credentials to secrets manager.\n    const userFunction = new lambda.Function(scope, 'UserFunction', {\n      runtime: lambda.Runtime.NODEJS_14_X,\n      handler: 'register-user.handler',\n      code: lambda.Code.fromAsset(path.join(__dirname, lambdaDirectory), codeProp),\n      environment: {\n        CA_ENDPOINT: caEndpoint,\n        MEMBER_NAME: memberName,\n        PRIVATE_KEY_ARN: adminPrivateKeyArn,\n        SIGNED_CERT_ARN: adminSignedCertArn,\n        TLS_CERT_BUCKET: tlsBucketData.bucketName,\n        TLS_CERT_KEY: tlsBucketData.key,\n      },\n      role: customResourceRole,\n      vpc: client.vpc,\n      vpcSubnets: client.vpc.selectSubnets(),\n      timeout: cdk.Duration.minutes(1),\n    });\n\n    // Add access to the lambda for the Network ports\n    scope.client.vpcEndpoint.connections.allowFrom(userFunction, ledgerPortRange);\n\n    // Custom Resource provider\n    HyperledgerFabricIdentity.userProvider = new customresources.Provider(scope, 'UserProvider', {\n      onEventHandler: userFunction,\n      logRetention: logs.RetentionDays.ONE_DAY,\n    });\n\n    // Populate the custom role static variable\n    HyperledgerFabricIdentity.customRole = customResourceRole;\n\n  }\n\n}"]}