@cdklabs/cdk-hyperledger-fabric-network 0.0.20
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 +4508 -0
- package/LICENSE +15 -0
- package/README.md +150 -0
- package/examples/python/hyperledger.py +90 -0
- package/examples/typescript/hyperledger.ts +106 -0
- package/lambdas/fabric/.eslintrc.js +17 -0
- package/lambdas/fabric/enroll-admin.js +33 -0
- package/lambdas/fabric/package.json +14 -0
- package/lambdas/fabric/register-user.js +62 -0
- package/lambdas/fabric/utilities.js +54 -0
- package/lib/client.d.ts +31 -0
- package/lib/client.js +40 -0
- package/lib/identity.d.ts +25 -0
- package/lib/identity.js +107 -0
- package/lib/index.d.ts +5 -0
- package/lib/index.js +22 -0
- package/lib/network.d.ts +183 -0
- package/lib/network.js +225 -0
- package/lib/node.d.ts +88 -0
- package/lib/node.js +161 -0
- package/lib/user.d.ts +40 -0
- package/lib/user.js +56 -0
- package/lib/utilities.d.ts +12 -0
- package/lib/utilities.js +116 -0
- package/package.json +154 -0
package/lib/identity.js
ADDED
|
@@ -0,0 +1,107 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
// Copyright Amazon.com, Inc. or its affiliates. All Rights Reserved.
|
|
3
|
+
// SPDX-License-Identifier: MIT-0
|
|
4
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
5
|
+
exports.HyperledgerFabricIdentity = void 0;
|
|
6
|
+
const path = require("path");
|
|
7
|
+
const cdk = require("aws-cdk-lib");
|
|
8
|
+
const ec2 = require("aws-cdk-lib/aws-ec2");
|
|
9
|
+
const iam = require("aws-cdk-lib/aws-iam");
|
|
10
|
+
const lambda = require("aws-cdk-lib/aws-lambda");
|
|
11
|
+
const logs = require("aws-cdk-lib/aws-logs");
|
|
12
|
+
const customresources = require("aws-cdk-lib/custom-resources");
|
|
13
|
+
const constructs = require("constructs");
|
|
14
|
+
const utilities = require("./utilities");
|
|
15
|
+
/**
|
|
16
|
+
* Creates custom resources to enroll admin and register user
|
|
17
|
+
* identities with the CA using the fabric-ca-client SDK.
|
|
18
|
+
* Admin identity is enrolled by default. User identities are
|
|
19
|
+
* registered and enrolled, if provided.
|
|
20
|
+
*/
|
|
21
|
+
class HyperledgerFabricIdentity extends constructs.Construct {
|
|
22
|
+
constructor(scope, id) {
|
|
23
|
+
super(scope, id);
|
|
24
|
+
// Collect metadata on the stack
|
|
25
|
+
const partition = cdk.Stack.of(this).partition;
|
|
26
|
+
const region = cdk.Stack.of(this).region;
|
|
27
|
+
// Retrieve the S3 Bucket and key that contains the TLS cert file
|
|
28
|
+
const tlsBucketData = utilities.getTlsBucket(region);
|
|
29
|
+
const adminPasswordArn = scope.adminPasswordSecret.secretArn;
|
|
30
|
+
const adminPrivateKeyArn = scope.adminPrivateKeySecret.secretArn;
|
|
31
|
+
const adminSignedCertArn = scope.adminSignedCertSecret.secretArn;
|
|
32
|
+
const caEndpoint = scope.caEndpoint;
|
|
33
|
+
const client = scope.client;
|
|
34
|
+
const memberName = scope.memberName;
|
|
35
|
+
// Role for the custom resource lambda functions
|
|
36
|
+
const customResourceRole = new iam.Role(this, 'CustomResourceRole', { assumedBy: new iam.ServicePrincipal('lambda.amazonaws.com') });
|
|
37
|
+
// Policies for the custom resource lambda to enroll and register users
|
|
38
|
+
customResourceRole.addManagedPolicy(iam.ManagedPolicy.fromAwsManagedPolicyName('service-role/AWSLambdaBasicExecutionRole'));
|
|
39
|
+
customResourceRole.addManagedPolicy(iam.ManagedPolicy.fromAwsManagedPolicyName('service-role/AWSLambdaVPCAccessExecutionRole'));
|
|
40
|
+
customResourceRole.addToPolicy(new iam.PolicyStatement({
|
|
41
|
+
actions: ['s3:GetObject', 'secretsmanager:CreateSecret', 'secretsmanager:GetSecretValue', 'secretsmanager:PutSecretValue'],
|
|
42
|
+
resources: [
|
|
43
|
+
`arn:${partition}:s3:::${tlsBucketData.bucketName}/*`,
|
|
44
|
+
adminPasswordArn,
|
|
45
|
+
adminPrivateKeyArn,
|
|
46
|
+
adminSignedCertArn,
|
|
47
|
+
],
|
|
48
|
+
}));
|
|
49
|
+
// Lambda function to enroll the admin and import credentials to secrets manager
|
|
50
|
+
const adminFunction = new lambda.Function(this, 'AdminFunction', {
|
|
51
|
+
runtime: lambda.Runtime.NODEJS_14_X,
|
|
52
|
+
handler: 'enroll-admin.handler',
|
|
53
|
+
code: lambda.Code.fromAsset(path.join(__dirname, '../lambdas/fabric')),
|
|
54
|
+
environment: {
|
|
55
|
+
ADMIN_PASSWORD_ARN: adminPasswordArn,
|
|
56
|
+
CA_ENDPOINT: caEndpoint,
|
|
57
|
+
PRIVATE_KEY_ARN: adminPrivateKeyArn,
|
|
58
|
+
SIGNED_CERT_ARN: adminSignedCertArn,
|
|
59
|
+
TLS_CERT_BUCKET: tlsBucketData.bucketName,
|
|
60
|
+
TLS_CERT_KEY: tlsBucketData.key,
|
|
61
|
+
},
|
|
62
|
+
role: customResourceRole,
|
|
63
|
+
vpc: client.vpc,
|
|
64
|
+
vpcSubnets: client.vpc.selectSubnets(),
|
|
65
|
+
timeout: cdk.Duration.minutes(1),
|
|
66
|
+
});
|
|
67
|
+
// Port range to access the Network
|
|
68
|
+
const ledgerPortRange = ec2.Port.tcpRange(utilities.STARTING_PORT, utilities.ENDING_PORT);
|
|
69
|
+
// Add access to the lambda for the Network ports
|
|
70
|
+
client.vpcEndpoint.connections.allowFrom(adminFunction, ledgerPortRange);
|
|
71
|
+
// Custom Resource provider
|
|
72
|
+
this.adminProvider = new customresources.Provider(this, 'AdminProvider', {
|
|
73
|
+
onEventHandler: adminFunction,
|
|
74
|
+
logRetention: logs.RetentionDays.ONE_DAY,
|
|
75
|
+
});
|
|
76
|
+
// Lambda function to register and enroll users and
|
|
77
|
+
// import credentials to secrets manager
|
|
78
|
+
const userFunction = new lambda.Function(scope, 'UserFunction', {
|
|
79
|
+
runtime: lambda.Runtime.NODEJS_14_X,
|
|
80
|
+
handler: 'register-user.handler',
|
|
81
|
+
code: lambda.Code.fromAsset(path.join(__dirname, '../lambdas/fabric')),
|
|
82
|
+
environment: {
|
|
83
|
+
CA_ENDPOINT: caEndpoint,
|
|
84
|
+
MEMBER_NAME: memberName,
|
|
85
|
+
PRIVATE_KEY_ARN: adminPrivateKeyArn,
|
|
86
|
+
SIGNED_CERT_ARN: adminSignedCertArn,
|
|
87
|
+
TLS_CERT_BUCKET: tlsBucketData.bucketName,
|
|
88
|
+
TLS_CERT_KEY: tlsBucketData.key,
|
|
89
|
+
},
|
|
90
|
+
role: customResourceRole,
|
|
91
|
+
vpc: client.vpc,
|
|
92
|
+
vpcSubnets: client.vpc.selectSubnets(),
|
|
93
|
+
timeout: cdk.Duration.minutes(1),
|
|
94
|
+
});
|
|
95
|
+
// Add access to the lambda for the Network ports
|
|
96
|
+
scope.client.vpcEndpoint.connections.allowFrom(userFunction, ledgerPortRange);
|
|
97
|
+
// Custom Resource provider
|
|
98
|
+
HyperledgerFabricIdentity.userProvider = new customresources.Provider(scope, 'UserProvider', {
|
|
99
|
+
onEventHandler: userFunction,
|
|
100
|
+
logRetention: logs.RetentionDays.ONE_DAY,
|
|
101
|
+
});
|
|
102
|
+
// Populate the custom role static variable
|
|
103
|
+
HyperledgerFabricIdentity.customRole = customResourceRole;
|
|
104
|
+
}
|
|
105
|
+
}
|
|
106
|
+
exports.HyperledgerFabricIdentity = HyperledgerFabricIdentity;
|
|
107
|
+
//# 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 * 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    // 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, '../lambdas/fabric')),\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, '../lambdas/fabric')),\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}"]}
|
package/lib/index.d.ts
ADDED
package/lib/index.js
ADDED
|
@@ -0,0 +1,22 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
// Copyright Amazon.com, Inc. or its affiliates. All Rights Reserved.
|
|
3
|
+
// SPDX-License-Identifier: MIT-0
|
|
4
|
+
var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
|
|
5
|
+
if (k2 === undefined) k2 = k;
|
|
6
|
+
Object.defineProperty(o, k2, { enumerable: true, get: function() { return m[k]; } });
|
|
7
|
+
}) : (function(o, m, k, k2) {
|
|
8
|
+
if (k2 === undefined) k2 = k;
|
|
9
|
+
o[k2] = m[k];
|
|
10
|
+
}));
|
|
11
|
+
var __exportStar = (this && this.__exportStar) || function(m, exports) {
|
|
12
|
+
for (var p in m) if (p !== "default" && !exports.hasOwnProperty(p)) __createBinding(exports, m, p);
|
|
13
|
+
};
|
|
14
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
15
|
+
__exportStar(require("./network"), exports);
|
|
16
|
+
__exportStar(require("./node"), exports);
|
|
17
|
+
__exportStar(require("./client"), exports);
|
|
18
|
+
__exportStar(require("./user"), exports);
|
|
19
|
+
var utilities_1 = require("./utilities");
|
|
20
|
+
Object.defineProperty(exports, "SUPPORTED_REGIONS", { enumerable: true, get: function () { return utilities_1.SUPPORTED_REGIONS; } });
|
|
21
|
+
Object.defineProperty(exports, "SUPPORTED_AVAILABILITY_ZONES", { enumerable: true, get: function () { return utilities_1.SUPPORTED_AVAILABILITY_ZONES; } });
|
|
22
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi9zcmMvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IjtBQUFBLHFFQUFxRTtBQUNyRSxpQ0FBaUM7Ozs7Ozs7Ozs7OztBQUdqQyw0Q0FBMEI7QUFDMUIseUNBQXVCO0FBQ3ZCLDJDQUF5QjtBQUN6Qix5Q0FBdUI7QUFFdkIseUNBQThFO0FBQXJFLDhHQUFBLGlCQUFpQixPQUFBO0FBQUUseUhBQUEsNEJBQTRCLE9BQUEiLCJzb3VyY2VzQ29udGVudCI6WyIvLyBDb3B5cmlnaHQgQW1hem9uLmNvbSwgSW5jLiBvciBpdHMgYWZmaWxpYXRlcy4gQWxsIFJpZ2h0cyBSZXNlcnZlZC5cbi8vIFNQRFgtTGljZW5zZS1JZGVudGlmaWVyOiBNSVQtMFxuXG5cbmV4cG9ydCAqIGZyb20gJy4vbmV0d29yayc7XG5leHBvcnQgKiBmcm9tICcuL25vZGUnO1xuZXhwb3J0ICogZnJvbSAnLi9jbGllbnQnO1xuZXhwb3J0ICogZnJvbSAnLi91c2VyJztcblxuZXhwb3J0IHsgU1VQUE9SVEVEX1JFR0lPTlMsIFNVUFBPUlRFRF9BVkFJTEFCSUxJVFlfWk9ORVMgfSBmcm9tICcuL3V0aWxpdGllcyc7XG4iXX0=
|
package/lib/network.d.ts
ADDED
|
@@ -0,0 +1,183 @@
|
|
|
1
|
+
import * as secretsmanager from 'aws-cdk-lib/aws-secretsmanager';
|
|
2
|
+
import * as constructs from 'constructs';
|
|
3
|
+
import * as client from './client';
|
|
4
|
+
import * as node from './node';
|
|
5
|
+
import * as user from './user';
|
|
6
|
+
export declare enum FrameworkVersion {
|
|
7
|
+
VERSION_1_2 = "1.2",
|
|
8
|
+
VERSION_1_4 = "1.4",
|
|
9
|
+
VERSION_2_2 = "2.2"
|
|
10
|
+
}
|
|
11
|
+
export declare enum NetworkEdition {
|
|
12
|
+
STARTER = "STARTER",
|
|
13
|
+
STANDARD = "STANDARD"
|
|
14
|
+
}
|
|
15
|
+
export declare enum ThresholdComparator {
|
|
16
|
+
GREATER_THAN = "GREATER_THAN",
|
|
17
|
+
GREATER_THAN_OR_EQUAL_TO = "GREATER_THAN_OR_EQUAL_TO"
|
|
18
|
+
}
|
|
19
|
+
/**
|
|
20
|
+
* Construct properties for `HyperledgerFabricNetwork`
|
|
21
|
+
*/
|
|
22
|
+
export interface HyperledgerFabricNetworkProps {
|
|
23
|
+
/**
|
|
24
|
+
* Managed Blockchain network name
|
|
25
|
+
*/
|
|
26
|
+
readonly networkName: string;
|
|
27
|
+
/**
|
|
28
|
+
* Managed Blockchain network description
|
|
29
|
+
*
|
|
30
|
+
* @default - Set to match network name
|
|
31
|
+
*/
|
|
32
|
+
readonly networkDescription?: string;
|
|
33
|
+
/**
|
|
34
|
+
* Managed Blockchain member name
|
|
35
|
+
*/
|
|
36
|
+
readonly memberName: string;
|
|
37
|
+
/**
|
|
38
|
+
* Managed Blockchain member description
|
|
39
|
+
*
|
|
40
|
+
* @default - Set to match member name
|
|
41
|
+
*/
|
|
42
|
+
readonly memberDescription?: string;
|
|
43
|
+
/**
|
|
44
|
+
* Hyperledger Fabric framework version
|
|
45
|
+
*
|
|
46
|
+
* @default - FrameworkVersion.VERSION_1_4
|
|
47
|
+
*/
|
|
48
|
+
readonly frameworkVersion?: FrameworkVersion;
|
|
49
|
+
/**
|
|
50
|
+
* Managed Blockchain network edition
|
|
51
|
+
*
|
|
52
|
+
* @default - NetworkEdition.STANDARD
|
|
53
|
+
*/
|
|
54
|
+
readonly networkEdition?: NetworkEdition;
|
|
55
|
+
/**
|
|
56
|
+
* The duration from the time that a proposal is created until it expires
|
|
57
|
+
* @default - 24 hours
|
|
58
|
+
*/
|
|
59
|
+
readonly proposalDurationInHours?: number;
|
|
60
|
+
/**
|
|
61
|
+
* The percentage of votes among all members that must be yes for a proposal to be approved
|
|
62
|
+
* @default - 50 percent
|
|
63
|
+
*/
|
|
64
|
+
readonly thresholdPercentage?: number;
|
|
65
|
+
/**
|
|
66
|
+
* Determines whether the yes votes must be greater than the threshold percentage
|
|
67
|
+
* or must be greater than or equal to the threhold percentage to be approved
|
|
68
|
+
* @default - GREATER_THAN
|
|
69
|
+
*/
|
|
70
|
+
readonly thresholdComparator?: ThresholdComparator;
|
|
71
|
+
/**
|
|
72
|
+
* The configuration to enable or disable certificate authority logging
|
|
73
|
+
* @default - true
|
|
74
|
+
*/
|
|
75
|
+
readonly enableCaLogging?: boolean;
|
|
76
|
+
/**
|
|
77
|
+
* List of nodes to create on the network
|
|
78
|
+
*
|
|
79
|
+
* @default - One node with default configuration
|
|
80
|
+
*/
|
|
81
|
+
readonly nodes?: Array<node.HyperledgerFabricNodeProps>;
|
|
82
|
+
/**
|
|
83
|
+
* The Client network to interact with the Hyperledger Fabric network
|
|
84
|
+
* @default - Client network with Default properties
|
|
85
|
+
* (CIDR-`10.0.0.0/16` and subnets of type `PRIVATE_ISOLATED`)
|
|
86
|
+
*/
|
|
87
|
+
readonly client?: client.HyperledgerFabricClientProps;
|
|
88
|
+
/**
|
|
89
|
+
* List of users to register with Fabric CA
|
|
90
|
+
*/
|
|
91
|
+
readonly users?: Array<user.HyperledgerFabricUserProps>;
|
|
92
|
+
}
|
|
93
|
+
/**
|
|
94
|
+
* Creates a Hyperledger Fabric network on Amazon Managed Blockchain
|
|
95
|
+
*/
|
|
96
|
+
export declare class HyperledgerFabricNetwork extends constructs.Construct {
|
|
97
|
+
/**
|
|
98
|
+
* Managed Blockchain network name
|
|
99
|
+
*/
|
|
100
|
+
readonly networkName: string;
|
|
101
|
+
/**
|
|
102
|
+
* Managed Blockchain network description
|
|
103
|
+
*/
|
|
104
|
+
readonly networkDescription: string;
|
|
105
|
+
/**
|
|
106
|
+
* Managed Blockchain network identifier generated on construction
|
|
107
|
+
*/
|
|
108
|
+
readonly networkId: string;
|
|
109
|
+
/**
|
|
110
|
+
* Managed Blockchain member name
|
|
111
|
+
*/
|
|
112
|
+
readonly memberName: string;
|
|
113
|
+
/**
|
|
114
|
+
* Managed Blockchain member description
|
|
115
|
+
*/
|
|
116
|
+
readonly memberDescription: string;
|
|
117
|
+
/**
|
|
118
|
+
* Managed Blockchain member identifier generated on construction
|
|
119
|
+
*/
|
|
120
|
+
readonly memberId: string;
|
|
121
|
+
/**
|
|
122
|
+
* Hyperledger Fabric framework version
|
|
123
|
+
*/
|
|
124
|
+
readonly frameworkVersion: FrameworkVersion;
|
|
125
|
+
/**
|
|
126
|
+
* Managed Blockchain network edition
|
|
127
|
+
*/
|
|
128
|
+
readonly networkEdition: NetworkEdition;
|
|
129
|
+
/**
|
|
130
|
+
* The duration from the time that a proposal is created until it expires
|
|
131
|
+
*/
|
|
132
|
+
readonly proposalDurationInHours: number;
|
|
133
|
+
/**
|
|
134
|
+
* The percentage of votes among all members that must be yes for a proposal to be approved
|
|
135
|
+
*/
|
|
136
|
+
readonly thresholdPercentage: number;
|
|
137
|
+
/**
|
|
138
|
+
* Determines whether the yes votes must be greater than the threshold percentage
|
|
139
|
+
* or must be greater than or equal to the threhold percentage to be approved
|
|
140
|
+
*/
|
|
141
|
+
readonly thresholdComparator: ThresholdComparator;
|
|
142
|
+
/**
|
|
143
|
+
* The configuration to enable or disable certificate authority logging
|
|
144
|
+
*/
|
|
145
|
+
readonly enableCaLogging: boolean;
|
|
146
|
+
/**
|
|
147
|
+
* Managed Blockchain network VPC endpoint service name
|
|
148
|
+
*/
|
|
149
|
+
readonly vpcEndpointServiceName: string;
|
|
150
|
+
/**
|
|
151
|
+
* Managed Blockchain network ordering service endpoint
|
|
152
|
+
*/
|
|
153
|
+
readonly ordererEndpoint: string;
|
|
154
|
+
/**
|
|
155
|
+
* Managed Blockchain member CA endpoint
|
|
156
|
+
*/
|
|
157
|
+
readonly caEndpoint: string;
|
|
158
|
+
/**
|
|
159
|
+
* Secret ARN for the Hyperledger Fabric admin password
|
|
160
|
+
*/
|
|
161
|
+
readonly adminPasswordSecret: secretsmanager.Secret;
|
|
162
|
+
/**
|
|
163
|
+
* Secret for Hyperledger Fabric admin private key
|
|
164
|
+
*/
|
|
165
|
+
readonly adminPrivateKeySecret: secretsmanager.Secret;
|
|
166
|
+
/**
|
|
167
|
+
* Secret for Hyperledger Fabric admin signed certificate
|
|
168
|
+
*/
|
|
169
|
+
readonly adminSignedCertSecret: secretsmanager.Secret;
|
|
170
|
+
/**
|
|
171
|
+
* List of nodes created in the network
|
|
172
|
+
*/
|
|
173
|
+
readonly nodes: Array<node.HyperledgerFabricNode>;
|
|
174
|
+
/**
|
|
175
|
+
* The client network to interact with the Hyperledger Fabric network
|
|
176
|
+
*/
|
|
177
|
+
readonly client: client.HyperledgerFabricClient;
|
|
178
|
+
/**
|
|
179
|
+
* List of users registered with CA
|
|
180
|
+
*/
|
|
181
|
+
readonly users: Array<user.HyperledgerFabricUser>;
|
|
182
|
+
constructor(scope: constructs.Construct, id: string, props: HyperledgerFabricNetworkProps);
|
|
183
|
+
}
|
package/lib/network.js
ADDED
|
@@ -0,0 +1,225 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
var _a;
|
|
3
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
4
|
+
exports.HyperledgerFabricNetwork = exports.ThresholdComparator = exports.NetworkEdition = exports.FrameworkVersion = void 0;
|
|
5
|
+
const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
|
|
6
|
+
// Copyright Amazon.com, Inc. or its affiliates. All Rights Reserved.
|
|
7
|
+
// SPDX-License-Identifier: MIT-0
|
|
8
|
+
const cdk = require("aws-cdk-lib");
|
|
9
|
+
const managedblockchain = require("aws-cdk-lib/aws-managedblockchain");
|
|
10
|
+
const secretsmanager = require("aws-cdk-lib/aws-secretsmanager");
|
|
11
|
+
const customresources = require("aws-cdk-lib/custom-resources");
|
|
12
|
+
const constructs = require("constructs");
|
|
13
|
+
const client = require("./client");
|
|
14
|
+
const identity = require("./identity");
|
|
15
|
+
const node = require("./node");
|
|
16
|
+
const user = require("./user");
|
|
17
|
+
const utilities = require("./utilities");
|
|
18
|
+
/*
|
|
19
|
+
* Define which Hyperledger Fabric framework to use
|
|
20
|
+
*/
|
|
21
|
+
var FrameworkVersion;
|
|
22
|
+
(function (FrameworkVersion) {
|
|
23
|
+
FrameworkVersion["VERSION_1_2"] = "1.2";
|
|
24
|
+
FrameworkVersion["VERSION_1_4"] = "1.4";
|
|
25
|
+
FrameworkVersion["VERSION_2_2"] = "2.2";
|
|
26
|
+
})(FrameworkVersion = exports.FrameworkVersion || (exports.FrameworkVersion = {}));
|
|
27
|
+
/*
|
|
28
|
+
* Starter networks are cheaper, but are limited to 2 nodes that
|
|
29
|
+
* can only be from a subset of types (see node.ts for the list)
|
|
30
|
+
*/
|
|
31
|
+
var NetworkEdition;
|
|
32
|
+
(function (NetworkEdition) {
|
|
33
|
+
NetworkEdition["STARTER"] = "STARTER";
|
|
34
|
+
NetworkEdition["STANDARD"] = "STANDARD";
|
|
35
|
+
})(NetworkEdition = exports.NetworkEdition || (exports.NetworkEdition = {}));
|
|
36
|
+
/*
|
|
37
|
+
* Constants to define ties in voting for new members
|
|
38
|
+
*/
|
|
39
|
+
var ThresholdComparator;
|
|
40
|
+
(function (ThresholdComparator) {
|
|
41
|
+
ThresholdComparator["GREATER_THAN"] = "GREATER_THAN";
|
|
42
|
+
ThresholdComparator["GREATER_THAN_OR_EQUAL_TO"] = "GREATER_THAN_OR_EQUAL_TO";
|
|
43
|
+
})(ThresholdComparator = exports.ThresholdComparator || (exports.ThresholdComparator = {}));
|
|
44
|
+
/**
|
|
45
|
+
* Creates a Hyperledger Fabric network on Amazon Managed Blockchain
|
|
46
|
+
*/
|
|
47
|
+
class HyperledgerFabricNetwork extends constructs.Construct {
|
|
48
|
+
constructor(scope, id, props) {
|
|
49
|
+
super(scope, id);
|
|
50
|
+
// Collect metadata on the stack
|
|
51
|
+
const partition = cdk.Stack.of(this).partition;
|
|
52
|
+
const region = cdk.Stack.of(this).region;
|
|
53
|
+
const account = cdk.Stack.of(this).account;
|
|
54
|
+
// Populate instance variables from input properties, using defaults if values not provided
|
|
55
|
+
this.networkName = props.networkName;
|
|
56
|
+
this.networkDescription = props.networkDescription ?? props.networkName;
|
|
57
|
+
this.memberName = props.memberName;
|
|
58
|
+
this.memberDescription = props.memberDescription ?? props.memberName;
|
|
59
|
+
this.frameworkVersion = props.frameworkVersion ?? FrameworkVersion.VERSION_1_4;
|
|
60
|
+
this.networkEdition = props.networkEdition ?? NetworkEdition.STANDARD;
|
|
61
|
+
this.proposalDurationInHours = props.proposalDurationInHours ?? 24;
|
|
62
|
+
this.thresholdPercentage = props.thresholdPercentage ?? 50;
|
|
63
|
+
this.thresholdComparator = props.thresholdComparator ?? ThresholdComparator.GREATER_THAN;
|
|
64
|
+
this.enableCaLogging = props.enableCaLogging ?? true;
|
|
65
|
+
this.users = [];
|
|
66
|
+
// Ensure the parameters captured above are valid, so we don't
|
|
67
|
+
// need to wait until deployment time to discover an error
|
|
68
|
+
utilities.validateRegion(region);
|
|
69
|
+
if (!utilities.validateString(this.networkName, 1, 64)) {
|
|
70
|
+
throw new Error('Network name is invalid or not provided. It can be up to 64 characters long.');
|
|
71
|
+
}
|
|
72
|
+
if (!utilities.validateString(this.networkDescription, 0, 128)) {
|
|
73
|
+
throw new Error('Network description is invalid. It can be up to 128 characters long.');
|
|
74
|
+
}
|
|
75
|
+
if (!utilities.validateString(this.memberName, 1, 64, /^(?!-|[0-9])(?!.*-$)(?!.*?--)[a-zA-Z0-9-]+$/)) {
|
|
76
|
+
throw new Error('Member name is invalid or not provided. It can be up to 64 characters long, and can have alphanumeric characters and hyphen(s). It cannot start with a number, or start and end with a hyphen (-), or have two consecutive hyphens. The member name must also be unique across the network.');
|
|
77
|
+
}
|
|
78
|
+
if (!utilities.validateString(this.memberDescription, 0, 128)) {
|
|
79
|
+
throw new Error('Member description is invalid. It can be up to 128 characters long.');
|
|
80
|
+
}
|
|
81
|
+
if (!utilities.validateInteger(this.proposalDurationInHours, 1, 168)) {
|
|
82
|
+
throw new Error('Voting policy proposal duration must be between 1 and 168 hours.');
|
|
83
|
+
}
|
|
84
|
+
if (!utilities.validateInteger(this.thresholdPercentage, 0, 100)) {
|
|
85
|
+
throw new Error('Voting policy threshold percentage must be between 0 and 100.');
|
|
86
|
+
}
|
|
87
|
+
// Ensure the user affiliation includes the member name,
|
|
88
|
+
// if the user list for registration is provided
|
|
89
|
+
if (props.users) {
|
|
90
|
+
props.users.forEach(e => {
|
|
91
|
+
if (!e.affilitation.startsWith(this.memberName))
|
|
92
|
+
throw new Error('User affiliation is invalid. Affiliation should start with Member name');
|
|
93
|
+
});
|
|
94
|
+
}
|
|
95
|
+
// Per the Managed Blockchain documentation, the admin password must be at least eight
|
|
96
|
+
// characters long and no more than 32 characters. It must contain at least one uppercase
|
|
97
|
+
// letter, one lowercase letter, and one digit. It cannot have a single quotation mark (‘),
|
|
98
|
+
// a double quotation marks (“), a forward slash(/), a backward slash(\), @, or a space;
|
|
99
|
+
// several other characters are exluded here to make the password easier to use in scripts
|
|
100
|
+
const passwordRequirements = {
|
|
101
|
+
passwordLength: 32,
|
|
102
|
+
requireEachIncludedType: true,
|
|
103
|
+
excludeCharacters: '\'"/\\@ &{}<>*|',
|
|
104
|
+
};
|
|
105
|
+
this.adminPasswordSecret = new secretsmanager.Secret(this, 'AdminPassword', { generateSecretString: passwordRequirements });
|
|
106
|
+
// The initially enrolled admin user credentials will be stored in these secrets
|
|
107
|
+
this.adminPrivateKeySecret = new secretsmanager.Secret(this, 'AdminPrivateKey');
|
|
108
|
+
this.adminSignedCertSecret = new secretsmanager.Secret(this, 'AdminSignedCert');
|
|
109
|
+
// Build out the Cloudformation construct for the network/member
|
|
110
|
+
const networkConfiguration = {
|
|
111
|
+
name: this.networkName,
|
|
112
|
+
description: this.networkDescription,
|
|
113
|
+
framework: 'HYPERLEDGER_FABRIC',
|
|
114
|
+
frameworkVersion: this.frameworkVersion,
|
|
115
|
+
networkFrameworkConfiguration: {
|
|
116
|
+
networkFabricConfiguration: {
|
|
117
|
+
edition: this.networkEdition,
|
|
118
|
+
},
|
|
119
|
+
},
|
|
120
|
+
votingPolicy: {
|
|
121
|
+
approvalThresholdPolicy: {
|
|
122
|
+
proposalDurationInHours: this.proposalDurationInHours,
|
|
123
|
+
thresholdPercentage: this.thresholdPercentage,
|
|
124
|
+
thresholdComparator: this.thresholdComparator,
|
|
125
|
+
},
|
|
126
|
+
},
|
|
127
|
+
};
|
|
128
|
+
// Note the use of the unwrap below is the only possible way to get
|
|
129
|
+
// the secret value into the CloudFormation; it will still not directly
|
|
130
|
+
// be included in the synthesized template so usage here is still safe
|
|
131
|
+
const memberConfiguration = {
|
|
132
|
+
name: this.memberName,
|
|
133
|
+
description: this.memberDescription,
|
|
134
|
+
memberFrameworkConfiguration: {
|
|
135
|
+
memberFabricConfiguration: {
|
|
136
|
+
adminUsername: 'admin',
|
|
137
|
+
adminPassword: this.adminPasswordSecret.secretValue.unsafeUnwrap(),
|
|
138
|
+
},
|
|
139
|
+
},
|
|
140
|
+
};
|
|
141
|
+
const network = new managedblockchain.CfnMember(this, 'Network', { networkConfiguration, memberConfiguration });
|
|
142
|
+
// Capture data included in the Cloudformation output in instance variables
|
|
143
|
+
this.networkId = network.getAtt('NetworkId').toString();
|
|
144
|
+
this.memberId = network.getAtt('MemberId').toString();
|
|
145
|
+
// Build out the associated node constructs
|
|
146
|
+
this.nodes = node.HyperledgerFabricNode.constructNodes(this, props.nodes);
|
|
147
|
+
// Due to a race condition in CDK custom resources (https://github.com/aws/aws-cdk/issues/18237),
|
|
148
|
+
// the necessary permissions for all SDK calls in the stack need to be added here, even though
|
|
149
|
+
// the calls in this construct don't need access to the nodes; this also means node constructs
|
|
150
|
+
// can't populate their outputs fully until later, which is annoying
|
|
151
|
+
const nodeIds = this.nodes.map(n => n.nodeId);
|
|
152
|
+
const nodeArns = nodeIds.map(i => `arn:${partition}:managedblockchain:${region}:${account}:nodes/${i}`);
|
|
153
|
+
const sdkCallPolicy = customresources.AwsCustomResourcePolicy.fromSdkCalls({
|
|
154
|
+
resources: [
|
|
155
|
+
`arn:${partition}:managedblockchain:${region}::networks/${this.networkId}`,
|
|
156
|
+
`arn:${partition}:managedblockchain:${region}:${account}:members/${this.memberId}`,
|
|
157
|
+
...nodeArns,
|
|
158
|
+
],
|
|
159
|
+
});
|
|
160
|
+
// Cloudformation doesn't include all the network and member attributes
|
|
161
|
+
// needed to use Hyperledger Fabric, so use SDK calls to fetch said data
|
|
162
|
+
const networkDataSdkCall = {
|
|
163
|
+
service: 'ManagedBlockchain',
|
|
164
|
+
action: 'getNetwork',
|
|
165
|
+
parameters: { NetworkId: this.networkId },
|
|
166
|
+
physicalResourceId: customresources.PhysicalResourceId.of('Id'),
|
|
167
|
+
};
|
|
168
|
+
const memberDataSdkCall = {
|
|
169
|
+
service: 'ManagedBlockchain',
|
|
170
|
+
action: 'getMember',
|
|
171
|
+
parameters: { NetworkId: this.networkId, MemberId: this.memberId },
|
|
172
|
+
physicalResourceId: customresources.PhysicalResourceId.of('Id'),
|
|
173
|
+
};
|
|
174
|
+
// Data items need fetching on creation and updating; nothing needs doing on deletion
|
|
175
|
+
const networkData = new customresources.AwsCustomResource(this, 'NetworkDataResource', {
|
|
176
|
+
policy: sdkCallPolicy,
|
|
177
|
+
onCreate: networkDataSdkCall,
|
|
178
|
+
onUpdate: networkDataSdkCall,
|
|
179
|
+
});
|
|
180
|
+
const memberData = new customresources.AwsCustomResource(this, 'MemberDataResource', {
|
|
181
|
+
policy: sdkCallPolicy,
|
|
182
|
+
onCreate: memberDataSdkCall,
|
|
183
|
+
onUpdate: memberDataSdkCall,
|
|
184
|
+
});
|
|
185
|
+
// Cloudformation doesn't include logging configuration so use SDK call to do so
|
|
186
|
+
const logConfiguration = {
|
|
187
|
+
Fabric: { CaLogs: { Cloudwatch: { Enabled: this.enableCaLogging } } },
|
|
188
|
+
};
|
|
189
|
+
const configureCaLogSdkCall = {
|
|
190
|
+
service: 'ManagedBlockchain',
|
|
191
|
+
action: 'updateMember',
|
|
192
|
+
parameters: { NetworkId: this.networkId, MemberId: this.memberId, LogPublishingConfiguration: logConfiguration },
|
|
193
|
+
physicalResourceId: customresources.PhysicalResourceId.of('Id'),
|
|
194
|
+
};
|
|
195
|
+
new customresources.AwsCustomResource(this, 'ConfigureCaLogResource', {
|
|
196
|
+
policy: sdkCallPolicy,
|
|
197
|
+
onCreate: configureCaLogSdkCall,
|
|
198
|
+
onUpdate: configureCaLogSdkCall,
|
|
199
|
+
});
|
|
200
|
+
// Grab items out of the above return values and stick them in output properties
|
|
201
|
+
this.vpcEndpointServiceName = networkData.getResponseField('Network.VpcEndpointServiceName');
|
|
202
|
+
this.ordererEndpoint = networkData.getResponseField('Network.FrameworkAttributes.Fabric.OrderingServiceEndpoint');
|
|
203
|
+
this.caEndpoint = memberData.getResponseField('Member.FrameworkAttributes.Fabric.CaEndpoint');
|
|
204
|
+
// As stated earlier, node constructs can't populate all their properties
|
|
205
|
+
// until after the above network and member SDK calls succeed; thus the
|
|
206
|
+
// function calls below where fetches are split out and logging is configured
|
|
207
|
+
for (const n of this.nodes) {
|
|
208
|
+
n.configureLogging(sdkCallPolicy);
|
|
209
|
+
n.fetchData(sdkCallPolicy);
|
|
210
|
+
}
|
|
211
|
+
// Build out the client VPC construct
|
|
212
|
+
this.client = new client.HyperledgerFabricClient(this, 'NetworkClient', props.client);
|
|
213
|
+
// Build out all the custom resources to register and enroll identities to CA
|
|
214
|
+
const identityResources = new identity.HyperledgerFabricIdentity(this, 'Identity');
|
|
215
|
+
// Enroll the administrator and store its credentials on Secrets Manager
|
|
216
|
+
new cdk.CustomResource(this, 'AdminCustomResource', { serviceToken: identityResources.adminProvider.serviceToken });
|
|
217
|
+
// Register and enroll users, if provided
|
|
218
|
+
if (props.users)
|
|
219
|
+
this.users = Array.from(props.users.entries()).map(e => new user.HyperledgerFabricUser(this, `User${e[0]}`, e[1]));
|
|
220
|
+
}
|
|
221
|
+
}
|
|
222
|
+
exports.HyperledgerFabricNetwork = HyperledgerFabricNetwork;
|
|
223
|
+
_a = JSII_RTTI_SYMBOL_1;
|
|
224
|
+
HyperledgerFabricNetwork[_a] = { fqn: "@cdklabs/cdk-hyperledger-fabric-network.HyperledgerFabricNetwork", version: "0.0.20" };
|
|
225
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"network.js","sourceRoot":"","sources":["../src/network.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\n\nimport * as cdk from 'aws-cdk-lib';\nimport * as managedblockchain from 'aws-cdk-lib/aws-managedblockchain';\nimport * as secretsmanager from 'aws-cdk-lib/aws-secretsmanager';\nimport * as customresources from 'aws-cdk-lib/custom-resources';\nimport * as constructs from 'constructs';\n\nimport * as client from './client';\nimport * as identity from './identity';\nimport * as node from './node';\nimport * as user from './user';\nimport * as utilities from './utilities';\n\n\n/*\n * Define which Hyperledger Fabric framework to use\n */\nexport enum FrameworkVersion {\n  VERSION_1_2 = '1.2',\n  VERSION_1_4 = '1.4',\n  VERSION_2_2 = '2.2',\n}\n\n/*\n * Starter networks are cheaper, but are limited to 2 nodes that\n * can only be from a subset of types (see node.ts for the list)\n */\nexport enum NetworkEdition {\n  STARTER = 'STARTER',\n  STANDARD = 'STANDARD',\n}\n\n/*\n * Constants to define ties in voting for new members\n */\nexport enum ThresholdComparator {\n  GREATER_THAN = 'GREATER_THAN',\n  GREATER_THAN_OR_EQUAL_TO = 'GREATER_THAN_OR_EQUAL_TO',\n}\n\n\n/**\n * Construct properties for `HyperledgerFabricNetwork`\n */\nexport interface HyperledgerFabricNetworkProps {\n\n  /**\n   * Managed Blockchain network name\n   */\n  readonly networkName: string;\n\n  /**\n   * Managed Blockchain network description\n   *\n   * @default - Set to match network name\n   */\n  readonly networkDescription?: string;\n\n  /**\n   * Managed Blockchain member name\n   */\n  readonly memberName: string;\n\n  /**\n   * Managed Blockchain member description\n   *\n   * @default - Set to match member name\n   */\n  readonly memberDescription?: string;\n\n  /**\n   * Hyperledger Fabric framework version\n   *\n   * @default - FrameworkVersion.VERSION_1_4\n   */\n  readonly frameworkVersion?: FrameworkVersion;\n\n  /**\n   * Managed Blockchain network edition\n   *\n   * @default - NetworkEdition.STANDARD\n   */\n  readonly networkEdition?: NetworkEdition;\n\n  /**\n   * The duration from the time that a proposal is created until it expires\n   * @default - 24 hours\n   */\n  readonly proposalDurationInHours?: number;\n\n  /**\n   * The percentage of votes among all members that must be yes for a proposal to be approved\n   * @default - 50 percent\n   */\n  readonly thresholdPercentage?: number;\n\n  /**\n   * Determines whether the yes votes must be greater than the threshold percentage\n   * or must be greater than or equal to the threhold percentage to be approved\n   * @default - GREATER_THAN\n   */\n  readonly thresholdComparator?: ThresholdComparator;\n\n  /**\n   * The configuration to enable or disable certificate authority logging\n   * @default - true\n   */\n  readonly enableCaLogging?: boolean;\n\n  /**\n   * List of nodes to create on the network\n   *\n   * @default - One node with default configuration\n   */\n  readonly nodes?: Array<node.HyperledgerFabricNodeProps>;\n\n  /**\n   * The Client network to interact with the Hyperledger Fabric network\n   * @default - Client network with Default properties\n   * (CIDR-`10.0.0.0/16` and subnets of type `PRIVATE_ISOLATED`)\n   */\n  readonly client?: client.HyperledgerFabricClientProps;\n\n  /**\n   * List of users to register with Fabric CA\n   */\n  readonly users?: Array<user.HyperledgerFabricUserProps>;\n\n}\n\n\n/**\n * Creates a Hyperledger Fabric network on Amazon Managed Blockchain\n */\nexport class HyperledgerFabricNetwork extends constructs.Construct {\n\n  /**\n   * Managed Blockchain network name\n   */\n  public readonly networkName: string;\n\n  /**\n   * Managed Blockchain network description\n   */\n  public readonly networkDescription: string;\n\n  /**\n   * Managed Blockchain network identifier generated on construction\n   */\n  public readonly networkId: string;\n\n  /**\n   * Managed Blockchain member name\n   */\n  public readonly memberName: string;\n\n  /**\n   * Managed Blockchain member description\n   */\n  public readonly memberDescription: string;\n\n  /**\n   * Managed Blockchain member identifier generated on construction\n   */\n  public readonly memberId: string;\n\n  /**\n   * Hyperledger Fabric framework version\n   */\n  public readonly frameworkVersion: FrameworkVersion;\n\n  /**\n   * Managed Blockchain network edition\n   */\n  public readonly networkEdition: NetworkEdition;\n\n  /**\n   * The duration from the time that a proposal is created until it expires\n   */\n  public readonly proposalDurationInHours: number;\n\n  /**\n   * The percentage of votes among all members that must be yes for a proposal to be approved\n   */\n  public readonly thresholdPercentage: number;\n\n  /**\n   * Determines whether the yes votes must be greater than the threshold percentage\n   * or must be greater than or equal to the threhold percentage to be approved\n   */\n  public readonly thresholdComparator: ThresholdComparator;\n\n  /**\n   * The configuration to enable or disable certificate authority logging\n   */\n  public readonly enableCaLogging: boolean;\n\n  /**\n   * Managed Blockchain network VPC endpoint service name\n   */\n  public readonly vpcEndpointServiceName: string;\n\n  /**\n   * Managed Blockchain network ordering service endpoint\n   */\n  public readonly ordererEndpoint: string;\n\n  /**\n   * Managed Blockchain member CA endpoint\n   */\n  public readonly caEndpoint: string;\n\n  /**\n   * Secret ARN for the Hyperledger Fabric admin password\n   */\n  public readonly adminPasswordSecret: secretsmanager.Secret;\n\n  /**\n   * Secret for Hyperledger Fabric admin private key\n   */\n  public readonly adminPrivateKeySecret: secretsmanager.Secret;\n\n  /**\n   * Secret for Hyperledger Fabric admin signed certificate\n   */\n  public readonly adminSignedCertSecret: secretsmanager.Secret;\n\n  /**\n   * List of nodes created in the network\n   */\n  public readonly nodes: Array<node.HyperledgerFabricNode>;\n\n  /**\n   * The client network to interact with the Hyperledger Fabric network\n   */\n  public readonly client: client.HyperledgerFabricClient;\n\n  /**\n   * List of users registered with CA\n   */\n  public readonly users: Array<user.HyperledgerFabricUser>;\n\n\n  constructor(scope: constructs.Construct, id: string, props: HyperledgerFabricNetworkProps) {\n\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    const account = cdk.Stack.of(this).account;\n\n    // Populate instance variables from input properties, using defaults if values not provided\n    this.networkName = props.networkName;\n    this.networkDescription = props.networkDescription ?? props.networkName;\n    this.memberName = props.memberName;\n    this.memberDescription = props.memberDescription ?? props.memberName;\n    this.frameworkVersion = props.frameworkVersion ?? FrameworkVersion.VERSION_1_4;\n    this.networkEdition = props.networkEdition ?? NetworkEdition.STANDARD;\n    this.proposalDurationInHours = props.proposalDurationInHours ?? 24;\n    this.thresholdPercentage = props.thresholdPercentage ?? 50;\n    this.thresholdComparator = props.thresholdComparator ?? ThresholdComparator.GREATER_THAN;\n    this.enableCaLogging = props.enableCaLogging ?? true;\n    this.users = [];\n\n    // Ensure the parameters captured above are valid, so we don't\n    // need to wait until deployment time to discover an error\n    utilities.validateRegion(region);\n    if (!utilities.validateString(this.networkName, 1, 64)) {\n      throw new Error('Network name is invalid or not provided. It can be up to 64 characters long.');\n    }\n    if (!utilities.validateString(this.networkDescription, 0, 128)) {\n      throw new Error('Network description is invalid. It can be up to 128 characters long.');\n    }\n    if (!utilities.validateString(this.memberName, 1, 64, /^(?!-|[0-9])(?!.*-$)(?!.*?--)[a-zA-Z0-9-]+$/)) {\n      throw new Error('Member name is invalid or not provided. It can be up to 64 characters long, and can have alphanumeric characters and hyphen(s). It cannot start with a number, or start and end with a hyphen (-), or have two consecutive hyphens. The member name must also be unique across the network.');\n    }\n    if (!utilities.validateString(this.memberDescription, 0, 128)) {\n      throw new Error('Member description is invalid. It can be up to 128 characters long.');\n    }\n    if (!utilities.validateInteger(this.proposalDurationInHours, 1, 168)) {\n      throw new Error('Voting policy proposal duration must be between 1 and 168 hours.');\n    }\n    if (!utilities.validateInteger(this.thresholdPercentage, 0, 100)) {\n      throw new Error('Voting policy threshold percentage must be between 0 and 100.');\n    }\n\n    // Ensure the user affiliation includes the member name,\n    // if the user list for registration is provided\n    if (props.users) {\n      props.users.forEach(e => {\n        if (!e.affilitation.startsWith(this.memberName)) throw new Error('User affiliation is invalid. Affiliation should start with Member name');\n      });\n    }\n\n    // Per the Managed Blockchain documentation, the admin password must be at least eight\n    // characters long and no more than 32 characters. It must contain at least one uppercase\n    // letter, one lowercase letter, and one digit. It cannot have a single quotation mark (‘),\n    // a double quotation marks (“), a forward slash(/), a backward slash(\\), @, or a space;\n    // several other characters are exluded here to make the password easier to use in scripts\n    const passwordRequirements = {\n      passwordLength: 32,\n      requireEachIncludedType: true,\n      excludeCharacters: '\\'\"/\\\\@ &{}<>*|',\n    };\n    this.adminPasswordSecret = new secretsmanager.Secret(this, 'AdminPassword', { generateSecretString: passwordRequirements });\n\n    // The initially enrolled admin user credentials will be stored in these secrets\n    this.adminPrivateKeySecret = new secretsmanager.Secret(this, 'AdminPrivateKey');\n    this.adminSignedCertSecret = new secretsmanager.Secret(this, 'AdminSignedCert');\n\n    // Build out the Cloudformation construct for the network/member\n    const networkConfiguration = {\n      name: this.networkName,\n      description: this.networkDescription,\n      framework: 'HYPERLEDGER_FABRIC',\n      frameworkVersion: this.frameworkVersion,\n      networkFrameworkConfiguration: {\n        networkFabricConfiguration: {\n          edition: this.networkEdition,\n        },\n      },\n      votingPolicy: {\n        approvalThresholdPolicy: {\n          proposalDurationInHours: this.proposalDurationInHours,\n          thresholdPercentage: this.thresholdPercentage,\n          thresholdComparator: this.thresholdComparator,\n        },\n      },\n    };\n\n    // Note the use of the unwrap below is the only possible way to get\n    // the secret value into the CloudFormation; it will still not directly\n    // be included in the synthesized template so usage here is still safe\n    const memberConfiguration = {\n      name: this.memberName,\n      description: this.memberDescription,\n      memberFrameworkConfiguration: {\n        memberFabricConfiguration: {\n          adminUsername: 'admin',\n          adminPassword: this.adminPasswordSecret.secretValue.unsafeUnwrap(),\n        },\n      },\n    };\n    const network = new managedblockchain.CfnMember(this, 'Network', { networkConfiguration, memberConfiguration });\n\n    // Capture data included in the Cloudformation output in instance variables\n    this.networkId = network.getAtt('NetworkId').toString();\n    this.memberId = network.getAtt('MemberId').toString();\n\n    // Build out the associated node constructs\n    this.nodes = node.HyperledgerFabricNode.constructNodes(this, props.nodes);\n\n    // Due to a race condition in CDK custom resources (https://github.com/aws/aws-cdk/issues/18237),\n    // the necessary permissions for all SDK calls in the stack need to be added here, even though\n    // the calls in this construct don't need access to the nodes; this also means node constructs\n    // can't populate their outputs fully until later, which is annoying\n    const nodeIds = this.nodes.map(n => n.nodeId);\n    const nodeArns = nodeIds.map(i => `arn:${partition}:managedblockchain:${region}:${account}:nodes/${i}`);\n    const sdkCallPolicy = customresources.AwsCustomResourcePolicy.fromSdkCalls({\n      resources: [\n        `arn:${partition}:managedblockchain:${region}::networks/${this.networkId}`,\n        `arn:${partition}:managedblockchain:${region}:${account}:members/${this.memberId}`,\n        ...nodeArns,\n      ],\n    });\n\n    // Cloudformation doesn't include all the network and member attributes\n    // needed to use Hyperledger Fabric, so use SDK calls to fetch said data\n    const networkDataSdkCall = {\n      service: 'ManagedBlockchain',\n      action: 'getNetwork',\n      parameters: { NetworkId: this.networkId },\n      physicalResourceId: customresources.PhysicalResourceId.of('Id'),\n    };\n    const memberDataSdkCall = {\n      service: 'ManagedBlockchain',\n      action: 'getMember',\n      parameters: { NetworkId: this.networkId, MemberId: this.memberId },\n      physicalResourceId: customresources.PhysicalResourceId.of('Id'),\n    };\n\n    // Data items need fetching on creation and updating; nothing needs doing on deletion\n    const networkData = new customresources.AwsCustomResource(this, 'NetworkDataResource', {\n      policy: sdkCallPolicy,\n      onCreate: networkDataSdkCall,\n      onUpdate: networkDataSdkCall,\n    });\n    const memberData = new customresources.AwsCustomResource(this, 'MemberDataResource', {\n      policy: sdkCallPolicy,\n      onCreate: memberDataSdkCall,\n      onUpdate: memberDataSdkCall,\n    });\n\n    // Cloudformation doesn't include logging configuration so use SDK call to do so\n    const logConfiguration = {\n      Fabric: { CaLogs: { Cloudwatch: { Enabled: this.enableCaLogging } } },\n    };\n    const configureCaLogSdkCall = {\n      service: 'ManagedBlockchain',\n      action: 'updateMember',\n      parameters: { NetworkId: this.networkId, MemberId: this.memberId, LogPublishingConfiguration: logConfiguration },\n      physicalResourceId: customresources.PhysicalResourceId.of('Id'),\n    };\n    new customresources.AwsCustomResource(this, 'ConfigureCaLogResource', {\n      policy: sdkCallPolicy,\n      onCreate: configureCaLogSdkCall,\n      onUpdate: configureCaLogSdkCall,\n    });\n\n    // Grab items out of the above return values and stick them in output properties\n    this.vpcEndpointServiceName = networkData.getResponseField('Network.VpcEndpointServiceName');\n    this.ordererEndpoint = networkData.getResponseField('Network.FrameworkAttributes.Fabric.OrderingServiceEndpoint');\n    this.caEndpoint = memberData.getResponseField('Member.FrameworkAttributes.Fabric.CaEndpoint');\n\n    // As stated earlier, node constructs can't populate all their properties\n    // until after the above network and member SDK calls succeed; thus the\n    // function calls below where fetches are split out and logging is configured\n    for (const n of this.nodes) {\n      n.configureLogging(sdkCallPolicy);\n      n.fetchData(sdkCallPolicy);\n    }\n\n    // Build out the client VPC construct\n    this.client = new client.HyperledgerFabricClient(this, 'NetworkClient', props.client);\n\n    // Build out all the custom resources to register and enroll identities to CA\n    const identityResources = new identity.HyperledgerFabricIdentity(this, 'Identity');\n\n    // Enroll the administrator and store its credentials on Secrets Manager\n    new cdk.CustomResource(this, 'AdminCustomResource', { serviceToken: identityResources.adminProvider.serviceToken });\n\n    // Register and enroll users, if provided\n    if (props.users) this.users = Array.from(props.users.entries()).map(e => new user.HyperledgerFabricUser(this, `User${e[0]}`, e[1]));\n  }\n\n}\n"]}
|