@cdklabs/cdk-hyperledger-fabric-network
Version:
CDK construct to deploy a Hyperledger Fabric network running on Amazon Managed Blockchain
132 lines • 20.2 kB
JavaScript
;
// 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":"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","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}"]}