@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.
@@ -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
@@ -0,0 +1,5 @@
1
+ export * from './network';
2
+ export * from './node';
3
+ export * from './client';
4
+ export * from './user';
5
+ export { SUPPORTED_REGIONS, SUPPORTED_AVAILABILITY_ZONES } from './utilities';
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=
@@ -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"]}