@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/lib/node.d.ts ADDED
@@ -0,0 +1,88 @@
1
+ import * as customresources from 'aws-cdk-lib/custom-resources';
2
+ import * as constructs from 'constructs';
3
+ import * as network from './network';
4
+ /**
5
+ * Supported instance types for Managed Blockchain nodes
6
+ */
7
+ export declare enum InstanceType {
8
+ BURSTABLE3_SMALL = "bc.t3.small",
9
+ BURSTABLE3_MEDIUM = "bc.t3.medium",
10
+ BURSTABLE3_LARGE = "bc.t3.large",
11
+ BURSTABLE3_XLARGE = "bc.t3.xlarge",
12
+ STANDARD5_LARGE = "bc.m5.large",
13
+ STANDARD5_XLARGE = "bc.m5.xlarge",
14
+ STANDARD5_XLARGE2 = "bc.m5.2xlarge",
15
+ STANDARD5_XLARGE4 = "bc.m5.4xlarge",
16
+ COMPUTE5_LARGE = "bc.c5.large",
17
+ COMPUTE5_XLARGE = "bc.c5.xlarge",
18
+ COMPUTE5_XLARGE2 = "bc.c5.2xlarge",
19
+ COMPUTE5_XLARGE4 = "bc.c5.4xlarge"
20
+ }
21
+ /**
22
+ * Valid instance types for starter networks
23
+ */
24
+ export declare const STARTER_INSTANCE_TYPES: InstanceType[];
25
+ /**
26
+ * Construct properties for `HyperledgerFabricNode`
27
+ */
28
+ export interface HyperledgerFabricNodeProps {
29
+ /**
30
+ * The Availability Zone in which the node will be created
31
+ * @default - The first AZ in the region
32
+ */
33
+ readonly availabilityZone?: string;
34
+ /**
35
+ * The Amazon Managed Blockchain instance type for the node
36
+ * @default - BURSTABLE3_SMALL
37
+ */
38
+ readonly instanceType?: InstanceType;
39
+ /**
40
+ * The configuration to enable or disable chaincode logging
41
+ * @default - true
42
+ */
43
+ readonly enableChaincodeLogging?: boolean;
44
+ /**
45
+ * The configuration to enable or disable node logging
46
+ * @default - true
47
+ */
48
+ readonly enableNodeLogging?: boolean;
49
+ }
50
+ /**
51
+ * Creates a Hyperledger Fabric node on an Amazon Managed Blockchain network
52
+ */
53
+ export declare class HyperledgerFabricNode extends constructs.Construct {
54
+ static constructNodes(scope: network.HyperledgerFabricNetwork, nodeProps?: Array<HyperledgerFabricNodeProps>): HyperledgerFabricNode[];
55
+ /**
56
+ * Managed Blockchain network identifier
57
+ */
58
+ readonly networkId: string;
59
+ /**
60
+ * Managed Blockchain member identifier
61
+ */
62
+ readonly memberId: string;
63
+ /**
64
+ * Managed Blockchain node identifier generated on construction
65
+ */
66
+ readonly nodeId: string;
67
+ /**
68
+ * The Availability Zone in which the node exists
69
+ */
70
+ readonly availabilityZone: string;
71
+ /**
72
+ * The Amazon Managed Blockchain instance type for the node
73
+ */
74
+ readonly instanceType: InstanceType;
75
+ /**
76
+ * The configuration to enable or disable chaincode logging
77
+ */
78
+ readonly enableChaincodeLogging: boolean;
79
+ /**
80
+ * The configuration to enable or disable node logging
81
+ */
82
+ readonly enableNodeLogging: boolean;
83
+ endpoint: string;
84
+ eventEndpoint: string;
85
+ constructor(scope: network.HyperledgerFabricNetwork, id: string, props?: HyperledgerFabricNodeProps);
86
+ configureLogging(sdkCallPolicy: customresources.AwsCustomResourcePolicy): void;
87
+ fetchData(dataSdkCallPolicy: customresources.AwsCustomResourcePolicy): void;
88
+ }
package/lib/node.js ADDED
@@ -0,0 +1,161 @@
1
+ "use strict";
2
+ var _a;
3
+ Object.defineProperty(exports, "__esModule", { value: true });
4
+ exports.HyperledgerFabricNode = exports.STARTER_INSTANCE_TYPES = exports.InstanceType = 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 customresources = require("aws-cdk-lib/custom-resources");
11
+ const constructs = require("constructs");
12
+ const network = require("./network");
13
+ const utilities = require("./utilities");
14
+ /**
15
+ * Supported instance types for Managed Blockchain nodes
16
+ */
17
+ var InstanceType;
18
+ (function (InstanceType) {
19
+ InstanceType["BURSTABLE3_SMALL"] = "bc.t3.small";
20
+ InstanceType["BURSTABLE3_MEDIUM"] = "bc.t3.medium";
21
+ InstanceType["BURSTABLE3_LARGE"] = "bc.t3.large";
22
+ InstanceType["BURSTABLE3_XLARGE"] = "bc.t3.xlarge";
23
+ InstanceType["STANDARD5_LARGE"] = "bc.m5.large";
24
+ InstanceType["STANDARD5_XLARGE"] = "bc.m5.xlarge";
25
+ InstanceType["STANDARD5_XLARGE2"] = "bc.m5.2xlarge";
26
+ InstanceType["STANDARD5_XLARGE4"] = "bc.m5.4xlarge";
27
+ InstanceType["COMPUTE5_LARGE"] = "bc.c5.large";
28
+ InstanceType["COMPUTE5_XLARGE"] = "bc.c5.xlarge";
29
+ InstanceType["COMPUTE5_XLARGE2"] = "bc.c5.2xlarge";
30
+ InstanceType["COMPUTE5_XLARGE4"] = "bc.c5.4xlarge";
31
+ })(InstanceType = exports.InstanceType || (exports.InstanceType = {}));
32
+ /**
33
+ * Valid instance types for starter networks
34
+ */
35
+ exports.STARTER_INSTANCE_TYPES = [
36
+ InstanceType.BURSTABLE3_SMALL,
37
+ InstanceType.BURSTABLE3_MEDIUM,
38
+ ];
39
+ /**
40
+ * Creates a Hyperledger Fabric node on an Amazon Managed Blockchain network
41
+ */
42
+ class HyperledgerFabricNode extends constructs.Construct {
43
+ constructor(scope, id, props) {
44
+ super(scope, id);
45
+ // These cannot be readonly since they have to be set after construction
46
+ // due the race condition documented in https://github.com/aws/aws-cdk/issues/18237.
47
+ this.endpoint = '';
48
+ this.eventEndpoint = '';
49
+ // Collect metadata on the stack
50
+ const region = cdk.Stack.of(this).region;
51
+ // Populate instance variables from input properties, using defaults if values not provided
52
+ if (typeof props === 'undefined')
53
+ props = {};
54
+ this.availabilityZone = props.availabilityZone ?? `${region}a`;
55
+ this.instanceType = props.instanceType ?? InstanceType.BURSTABLE3_SMALL;
56
+ this.enableChaincodeLogging = props.enableChaincodeLogging ?? true;
57
+ this.enableNodeLogging = props.enableNodeLogging ?? true;
58
+ this.networkId = scope.networkId;
59
+ this.memberId = scope.memberId;
60
+ // Ensure the parameters captured above are valid, so we don't
61
+ // need to wait until deployment time to discover an error
62
+ utilities.validateRegion(region);
63
+ utilities.validateAvailabilityZone(region, this.availabilityZone);
64
+ if (scope.networkEdition === network.NetworkEdition.STARTER && !exports.STARTER_INSTANCE_TYPES.includes(this.instanceType)) {
65
+ const starterInstanceTypeList = exports.STARTER_INSTANCE_TYPES.join(', ');
66
+ throw new Error(`Instance type in a starter network must be one of the following: ${starterInstanceTypeList}.`);
67
+ }
68
+ // Build out the Cloudformation construct for the network/member
69
+ const node = new managedblockchain.CfnNode(this, 'Node', {
70
+ networkId: this.networkId,
71
+ memberId: this.memberId,
72
+ nodeConfiguration: {
73
+ availabilityZone: this.availabilityZone,
74
+ instanceType: this.instanceType,
75
+ },
76
+ });
77
+ // Capture data included in the Cloudformation output in instance variables
78
+ this.nodeId = node.getAtt('NodeId').toString();
79
+ }
80
+ /*
81
+ * Build out a list of HyperledgerFabricNode constructs given a list of input property
82
+ * objects; additionally checks to ensure node count is supported given the network type
83
+ */
84
+ static constructNodes(scope, nodeProps) {
85
+ // If no node configurations are provided, create one; the empty object
86
+ // will be populated with defaults when passed to the node constructor
87
+ if (typeof nodeProps === 'undefined')
88
+ nodeProps = [{}];
89
+ const starter = scope.networkEdition === network.NetworkEdition.STARTER;
90
+ if (starter && nodeProps.length > 2) {
91
+ throw new Error('A starter network can have at most 2 nodes per member.');
92
+ }
93
+ if (!starter && nodeProps.length > 3) {
94
+ throw new Error('A standard network can have at most 3 nodes per member.');
95
+ }
96
+ // Construct the node list, using an index value in the identifier
97
+ return Array.from(nodeProps.entries()).map(e => new HyperledgerFabricNode(scope, `Node${e[0]}`, e[1]));
98
+ }
99
+ /*
100
+ * Configure logging for the node via SDK call; this function
101
+ * should be merged back into the constructor once the race condition is solved
102
+ */
103
+ configureLogging(sdkCallPolicy) {
104
+ // This call doesn't really need all the permissions its using in the
105
+ // provided policy, but since the policy must be constructed all at once
106
+ // this is the only way to do it effectively
107
+ const logPublishingConfiguration = {
108
+ Fabric: {
109
+ ChaincodeLogs: {
110
+ Cloudwatch: { Enabled: this.enableChaincodeLogging },
111
+ },
112
+ PeerLogs: {
113
+ Cloudwatch: { Enabled: this.enableNodeLogging },
114
+ },
115
+ },
116
+ };
117
+ const configureNodeLogSdkCall = {
118
+ service: 'ManagedBlockchain',
119
+ action: 'updateNode',
120
+ parameters: {
121
+ NetworkId: this.networkId,
122
+ MemberId: this.memberId,
123
+ NodeId: this.nodeId,
124
+ LogPublishingConfiguration: logPublishingConfiguration,
125
+ },
126
+ physicalResourceId: customresources.PhysicalResourceId.of('Id'),
127
+ };
128
+ new customresources.AwsCustomResource(this, 'ConfigureNodeLogResource', {
129
+ policy: sdkCallPolicy,
130
+ onCreate: configureNodeLogSdkCall,
131
+ onUpdate: configureNodeLogSdkCall,
132
+ });
133
+ }
134
+ /*
135
+ * Populate the output properties that must be fetched via SDK call; this function
136
+ * should be merged back into the constructor once the race condition is solved
137
+ */
138
+ fetchData(dataSdkCallPolicy) {
139
+ // This call doesn't really need all the permissions its using in the
140
+ // provided policy, but since the policy must be constructed all at once
141
+ // this is the only way to do it effectively
142
+ const nodeDataSdkCall = {
143
+ service: 'ManagedBlockchain',
144
+ action: 'getNode',
145
+ parameters: { NetworkId: this.networkId, MemberId: this.memberId, NodeId: this.nodeId },
146
+ physicalResourceId: customresources.PhysicalResourceId.of('Id'),
147
+ };
148
+ const nodeData = new customresources.AwsCustomResource(this, 'NodeDataResource', {
149
+ policy: dataSdkCallPolicy,
150
+ onCreate: nodeDataSdkCall,
151
+ onUpdate: nodeDataSdkCall,
152
+ });
153
+ // Grab items out of the above return values and stick them in output properties
154
+ this.endpoint = nodeData.getResponseField('Node.FrameworkAttributes.Fabric.PeerEndpoint');
155
+ this.eventEndpoint = nodeData.getResponseField('Node.FrameworkAttributes.Fabric.PeerEventEndpoint');
156
+ }
157
+ }
158
+ exports.HyperledgerFabricNode = HyperledgerFabricNode;
159
+ _a = JSII_RTTI_SYMBOL_1;
160
+ HyperledgerFabricNode[_a] = { fqn: "@cdklabs/cdk-hyperledger-fabric-network.HyperledgerFabricNode", version: "0.0.20" };
161
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"node.js","sourceRoot":"","sources":["../src/node.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 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/**\n * Supported instance types for Managed Blockchain nodes\n */\nexport enum InstanceType {\n  BURSTABLE3_SMALL = 'bc.t3.small',\n  BURSTABLE3_MEDIUM = 'bc.t3.medium',\n  BURSTABLE3_LARGE = 'bc.t3.large',\n  BURSTABLE3_XLARGE = 'bc.t3.xlarge',\n  STANDARD5_LARGE = 'bc.m5.large',\n  STANDARD5_XLARGE = 'bc.m5.xlarge',\n  STANDARD5_XLARGE2 = 'bc.m5.2xlarge',\n  STANDARD5_XLARGE4 = 'bc.m5.4xlarge',\n  COMPUTE5_LARGE = 'bc.c5.large',\n  COMPUTE5_XLARGE = 'bc.c5.xlarge',\n  COMPUTE5_XLARGE2 = 'bc.c5.2xlarge',\n  COMPUTE5_XLARGE4 = 'bc.c5.4xlarge',\n}\n\n\n/**\n * Valid instance types for starter networks\n */\nexport const STARTER_INSTANCE_TYPES = [\n  InstanceType.BURSTABLE3_SMALL,\n  InstanceType.BURSTABLE3_MEDIUM,\n];\n\n\n/**\n * Construct properties for `HyperledgerFabricNode`\n */\nexport interface HyperledgerFabricNodeProps {\n\n  /**\n   * The Availability Zone in which the node will be created\n   * @default - The first AZ in the region\n   */\n  readonly availabilityZone?: string;\n\n  /**\n   * The Amazon Managed Blockchain instance type for the node\n   * @default - BURSTABLE3_SMALL\n   */\n  readonly instanceType?: InstanceType;\n\n  /**\n   * The configuration to enable or disable chaincode logging\n   * @default - true\n   */\n  readonly enableChaincodeLogging?: boolean;\n\n  /**\n   * The configuration to enable or disable node logging\n   * @default - true\n   */\n  readonly enableNodeLogging?: boolean;\n\n}\n\n\n/**\n * Creates a Hyperledger Fabric node on an Amazon Managed Blockchain network\n */\nexport class HyperledgerFabricNode extends constructs.Construct {\n\n  /*\n   * Build out a list of HyperledgerFabricNode constructs given a list of input property\n   * objects; additionally checks to ensure node count is supported given the network type\n   */\n  public static constructNodes(scope: network.HyperledgerFabricNetwork, nodeProps?: Array<HyperledgerFabricNodeProps>) {\n    // If no node configurations are provided, create one; the empty object\n    // will be populated with defaults when passed to the node constructor\n    if (typeof nodeProps === 'undefined') nodeProps = [{}];\n    const starter = scope.networkEdition === network.NetworkEdition.STARTER;\n    if (starter && nodeProps.length > 2) {\n      throw new Error('A starter network can have at most 2 nodes per member.');\n    }\n    if (!starter && nodeProps.length > 3) {\n      throw new Error('A standard network can have at most 3 nodes per member.');\n    }\n    // Construct the node list, using an index value in the identifier\n    return Array.from(nodeProps.entries()).map(e => new HyperledgerFabricNode(scope, `Node${e[0]}`, e[1]));\n  }\n\n  /**\n   * Managed Blockchain network identifier\n   */\n  public readonly networkId: string;\n\n  /**\n   * Managed Blockchain member identifier\n   */\n  public readonly memberId: string;\n\n  /**\n   * Managed Blockchain node identifier generated on construction\n   */\n  public readonly nodeId: string;\n\n  /**\n   * The Availability Zone in which the node exists\n   */\n  public readonly availabilityZone: string;\n\n  /**\n   * The Amazon Managed Blockchain instance type for the node\n   */\n  public readonly instanceType: InstanceType;\n\n  /**\n   * The configuration to enable or disable chaincode logging\n   */\n  public readonly enableChaincodeLogging: boolean;\n\n  /**\n   * The configuration to enable or disable node logging\n   */\n  public readonly enableNodeLogging: boolean;\n\n  // These cannot be readonly since they have to be set after construction\n  // due the race condition documented in https://github.com/aws/aws-cdk/issues/18237.\n  public endpoint: string = '';\n  public eventEndpoint: string = '';\n\n  constructor(scope: network.HyperledgerFabricNetwork, id: string, props?: HyperledgerFabricNodeProps) {\n\n    super(scope, id);\n\n    // Collect metadata on the stack\n    const region = cdk.Stack.of(this).region;\n\n    // Populate instance variables from input properties, using defaults if values not provided\n    if (typeof props === 'undefined') props = {};\n    this.availabilityZone = props.availabilityZone ?? `${region}a`;\n    this.instanceType = props.instanceType ?? InstanceType.BURSTABLE3_SMALL;\n    this.enableChaincodeLogging = props.enableChaincodeLogging ?? true;\n    this.enableNodeLogging = props.enableNodeLogging ?? true;\n    this.networkId = scope.networkId;\n    this.memberId = scope.memberId;\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    utilities.validateAvailabilityZone(region, this.availabilityZone);\n    if (scope.networkEdition === network.NetworkEdition.STARTER && !STARTER_INSTANCE_TYPES.includes(this.instanceType)) {\n      const starterInstanceTypeList = STARTER_INSTANCE_TYPES.join(', ');\n      throw new Error(`Instance type in a starter network must be one of the following: ${starterInstanceTypeList}.`);\n    }\n\n    // Build out the Cloudformation construct for the network/member\n    const node = new managedblockchain.CfnNode(this, 'Node', {\n      networkId: this.networkId,\n      memberId: this.memberId,\n      nodeConfiguration: {\n        availabilityZone: this.availabilityZone,\n        instanceType: this.instanceType,\n      },\n    });\n\n    // Capture data included in the Cloudformation output in instance variables\n    this.nodeId = node.getAtt('NodeId').toString();\n\n  }\n\n  /*\n   * Configure logging for the node via SDK call; this function\n   * should be merged back into the constructor once the race condition is solved\n   */\n  public configureLogging(sdkCallPolicy: customresources.AwsCustomResourcePolicy) {\n\n    // This call doesn't really need all the permissions its using in the\n    // provided policy, but since the policy must be constructed all at once\n    // this is the only way to do it effectively\n    const logPublishingConfiguration = {\n      Fabric: {\n        ChaincodeLogs: {\n          Cloudwatch: { Enabled: this.enableChaincodeLogging },\n        },\n        PeerLogs: {\n          Cloudwatch: { Enabled: this.enableNodeLogging },\n        },\n      },\n    };\n    const configureNodeLogSdkCall = {\n      service: 'ManagedBlockchain',\n      action: 'updateNode',\n      parameters: {\n        NetworkId: this.networkId,\n        MemberId: this.memberId,\n        NodeId: this.nodeId,\n        LogPublishingConfiguration: logPublishingConfiguration,\n      },\n      physicalResourceId: customresources.PhysicalResourceId.of('Id'),\n    };\n    new customresources.AwsCustomResource(this, 'ConfigureNodeLogResource', {\n      policy: sdkCallPolicy,\n      onCreate: configureNodeLogSdkCall,\n      onUpdate: configureNodeLogSdkCall,\n    });\n  }\n\n  /*\n   * Populate the output properties that must be fetched via SDK call; this function\n   * should be merged back into the constructor once the race condition is solved\n   */\n  public fetchData(dataSdkCallPolicy: customresources.AwsCustomResourcePolicy) {\n\n    // This call doesn't really need all the permissions its using in the\n    // provided policy, but since the policy must be constructed all at once\n    // this is the only way to do it effectively\n    const nodeDataSdkCall = {\n      service: 'ManagedBlockchain',\n      action: 'getNode',\n      parameters: { NetworkId: this.networkId, MemberId: this.memberId, NodeId: this.nodeId },\n      physicalResourceId: customresources.PhysicalResourceId.of('Id'),\n    };\n    const nodeData = new customresources.AwsCustomResource(this, 'NodeDataResource', {\n      policy: dataSdkCallPolicy,\n      onCreate: nodeDataSdkCall,\n      onUpdate: nodeDataSdkCall,\n    });\n\n    // Grab items out of the above return values and stick them in output properties\n    this.endpoint = nodeData.getResponseField('Node.FrameworkAttributes.Fabric.PeerEndpoint');\n    this.eventEndpoint = nodeData.getResponseField('Node.FrameworkAttributes.Fabric.PeerEventEndpoint');\n\n  }\n\n}\n"]}
package/lib/user.d.ts ADDED
@@ -0,0 +1,40 @@
1
+ import * as secretsmanager from 'aws-cdk-lib/aws-secretsmanager';
2
+ import * as constructs from 'constructs';
3
+ import * as network from './network';
4
+ /**
5
+ * Construct properties for `HyperledgerFabricUser`
6
+ */
7
+ export interface HyperledgerFabricUserProps {
8
+ /**
9
+ * User's affiliation to the member.
10
+ * Should be hierarchical with member name as root(`MemberName.Dept1`).
11
+ */
12
+ readonly affilitation: string;
13
+ /**
14
+ * User ID to register with CA
15
+ */
16
+ readonly userId: string;
17
+ }
18
+ /**
19
+ * Creates custom resources to register and enroll users identities
20
+ * with the CA using the fabric-ca-client SDK
21
+ */
22
+ export declare class HyperledgerFabricUser extends constructs.Construct {
23
+ /**
24
+ * User ID registered with CA
25
+ */
26
+ readonly userId: string;
27
+ /**
28
+ * User's affiliation to the member
29
+ */
30
+ readonly affiliation: string;
31
+ /**
32
+ * Secret for user signed certificate
33
+ */
34
+ readonly userSignedCertSecret: secretsmanager.Secret;
35
+ /**
36
+ * Secret for user private key
37
+ */
38
+ readonly userPrivateKeySecret: secretsmanager.Secret;
39
+ constructor(scope: network.HyperledgerFabricNetwork, id: string, props: HyperledgerFabricUserProps);
40
+ }
package/lib/user.js ADDED
@@ -0,0 +1,56 @@
1
+ "use strict";
2
+ var _a;
3
+ Object.defineProperty(exports, "__esModule", { value: true });
4
+ exports.HyperledgerFabricUser = 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 iam = require("aws-cdk-lib/aws-iam");
10
+ const secretsmanager = require("aws-cdk-lib/aws-secretsmanager");
11
+ const constructs = require("constructs");
12
+ const identity = require("./identity");
13
+ /**
14
+ * Creates custom resources to register and enroll users identities
15
+ * with the CA using the fabric-ca-client SDK
16
+ */
17
+ class HyperledgerFabricUser extends constructs.Construct {
18
+ constructor(scope, id, props) {
19
+ super(scope, id);
20
+ // Populate instance variables from input properties
21
+ this.userId = props.userId;
22
+ this.affiliation = props.affilitation;
23
+ // Get the custom resources from Identity Resources class
24
+ const customResourceRole = identity.HyperledgerFabricIdentity.customRole;
25
+ const registerProvider = identity.HyperledgerFabricIdentity.userProvider;
26
+ // Ensure the user affiliation includes the member name
27
+ if (!this.affiliation.startsWith(scope.memberName))
28
+ throw new Error('User affiliation is invalid. Affiliation should start with Member name');
29
+ // The user credentials will be stored in these secrets
30
+ const userPasswordSecret = new secretsmanager.Secret(this, `${this.userId}-Password`);
31
+ this.userPrivateKeySecret = new secretsmanager.Secret(this, `${this.userId}-PrivateKey`);
32
+ this.userSignedCertSecret = new secretsmanager.Secret(this, `${this.userId}-SignedCert`);
33
+ // Add secrets arn to the custom role
34
+ customResourceRole.addToPolicy(new iam.PolicyStatement({
35
+ actions: ['secretsmanager:PutSecretValue'],
36
+ resources: [
37
+ userPasswordSecret.secretArn,
38
+ this.userPrivateKeySecret.secretArn,
39
+ this.userSignedCertSecret.secretArn,
40
+ ],
41
+ }));
42
+ const userProp = {
43
+ userId: this.userId,
44
+ affiliation: this.affiliation,
45
+ passwordArn: userPasswordSecret.secretArn,
46
+ privateKeyArn: this.userPrivateKeySecret.secretArn,
47
+ signedCertArn: this.userSignedCertSecret.secretArn,
48
+ };
49
+ // Custom resource to enroll the admin identity using fabric client SDK
50
+ new cdk.CustomResource(this, `${this.userId}Resource`, { serviceToken: registerProvider.serviceToken, properties: userProp });
51
+ }
52
+ }
53
+ exports.HyperledgerFabricUser = HyperledgerFabricUser;
54
+ _a = JSII_RTTI_SYMBOL_1;
55
+ HyperledgerFabricUser[_a] = { fqn: "@cdklabs/cdk-hyperledger-fabric-network.HyperledgerFabricUser", version: "0.0.20" };
56
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidXNlci5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uL3NyYy91c2VyLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7Ozs7O0FBQUEscUVBQXFFO0FBQ3JFLGlDQUFpQztBQUVqQyxtQ0FBbUM7QUFDbkMsMkNBQTJDO0FBQzNDLGlFQUFpRTtBQUNqRSx5Q0FBeUM7QUFFekMsdUNBQXVDO0FBcUJ2Qzs7O0dBR0c7QUFDSCxNQUFhLHFCQUFzQixTQUFRLFVBQVUsQ0FBQyxTQUFTO0lBdUI3RCxZQUFZLEtBQXVDLEVBQUUsRUFBVSxFQUFFLEtBQWlDO1FBQ2hHLEtBQUssQ0FBQyxLQUFLLEVBQUUsRUFBRSxDQUFDLENBQUM7UUFFakIsb0RBQW9EO1FBQ3BELElBQUksQ0FBQyxNQUFNLEdBQUcsS0FBSyxDQUFDLE1BQU0sQ0FBQztRQUMzQixJQUFJLENBQUMsV0FBVyxHQUFHLEtBQUssQ0FBQyxZQUFZLENBQUM7UUFFdEMseURBQXlEO1FBQ3pELE1BQU0sa0JBQWtCLEdBQUcsUUFBUSxDQUFDLHlCQUF5QixDQUFDLFVBQVUsQ0FBQztRQUN6RSxNQUFNLGdCQUFnQixHQUFHLFFBQVEsQ0FBQyx5QkFBeUIsQ0FBQyxZQUFZLENBQUM7UUFFekUsdURBQXVEO1FBQ3ZELElBQUksQ0FBQyxJQUFJLENBQUMsV0FBVyxDQUFDLFVBQVUsQ0FBQyxLQUFLLENBQUMsVUFBVSxDQUFDO1lBQUUsTUFBTSxJQUFJLEtBQUssQ0FBQyx3RUFBd0UsQ0FBQyxDQUFDO1FBRTlJLHVEQUF1RDtRQUN2RCxNQUFNLGtCQUFrQixHQUFHLElBQUksY0FBYyxDQUFDLE1BQU0sQ0FBQyxJQUFJLEVBQUUsR0FBRyxJQUFJLENBQUMsTUFBTSxXQUFXLENBQUMsQ0FBQztRQUN0RixJQUFJLENBQUMsb0JBQW9CLEdBQUcsSUFBSSxjQUFjLENBQUMsTUFBTSxDQUFDLElBQUksRUFBRSxHQUFHLElBQUksQ0FBQyxNQUFNLGFBQWEsQ0FBQyxDQUFDO1FBQ3pGLElBQUksQ0FBQyxvQkFBb0IsR0FBRyxJQUFJLGNBQWMsQ0FBQyxNQUFNLENBQUMsSUFBSSxFQUFFLEdBQUcsSUFBSSxDQUFDLE1BQU0sYUFBYSxDQUFDLENBQUM7UUFFekYscUNBQXFDO1FBQ3JDLGtCQUFrQixDQUFDLFdBQVcsQ0FBRSxJQUFJLEdBQUcsQ0FBQyxlQUFlLENBQUM7WUFDdEQsT0FBTyxFQUFFLENBQUMsK0JBQStCLENBQUM7WUFDMUMsU0FBUyxFQUFFO2dCQUNULGtCQUFrQixDQUFDLFNBQVM7Z0JBQzVCLElBQUksQ0FBQyxvQkFBb0IsQ0FBQyxTQUFTO2dCQUNuQyxJQUFJLENBQUMsb0JBQW9CLENBQUMsU0FBUzthQUNwQztTQUNGLENBQUMsQ0FBQyxDQUFDO1FBRUosTUFBTSxRQUFRLEdBQUc7WUFDZixNQUFNLEVBQUUsSUFBSSxDQUFDLE1BQU07WUFDbkIsV0FBVyxFQUFFLElBQUksQ0FBQyxXQUFXO1lBQzdCLFdBQVcsRUFBRSxrQkFBa0IsQ0FBQyxTQUFTO1lBQ3pDLGFBQWEsRUFBRSxJQUFJLENBQUMsb0JBQW9CLENBQUMsU0FBUztZQUNsRCxhQUFhLEVBQUUsSUFBSSxDQUFDLG9CQUFvQixDQUFDLFNBQVM7U0FDbkQsQ0FBQztRQUVGLHVFQUF1RTtRQUN2RSxJQUFJLEdBQUcsQ0FBQyxjQUFjLENBQUMsSUFBSSxFQUFFLEdBQUcsSUFBSSxDQUFDLE1BQU0sVUFBVSxFQUFFLEVBQUUsWUFBWSxFQUFFLGdCQUFnQixDQUFDLFlBQVksRUFBRSxVQUFVLEVBQUUsUUFBUSxFQUFFLENBQUMsQ0FBQztJQUVoSSxDQUFDOztBQS9ESCxzREFpRUMiLCJzb3VyY2VzQ29udGVudCI6WyIvLyBDb3B5cmlnaHQgQW1hem9uLmNvbSwgSW5jLiBvciBpdHMgYWZmaWxpYXRlcy4gQWxsIFJpZ2h0cyBSZXNlcnZlZC5cbi8vIFNQRFgtTGljZW5zZS1JZGVudGlmaWVyOiBNSVQtMFxuXG5pbXBvcnQgKiBhcyBjZGsgZnJvbSAnYXdzLWNkay1saWInO1xuaW1wb3J0ICogYXMgaWFtIGZyb20gJ2F3cy1jZGstbGliL2F3cy1pYW0nO1xuaW1wb3J0ICogYXMgc2VjcmV0c21hbmFnZXIgZnJvbSAnYXdzLWNkay1saWIvYXdzLXNlY3JldHNtYW5hZ2VyJztcbmltcG9ydCAqIGFzIGNvbnN0cnVjdHMgZnJvbSAnY29uc3RydWN0cyc7XG5cbmltcG9ydCAqIGFzIGlkZW50aXR5IGZyb20gJy4vaWRlbnRpdHknO1xuaW1wb3J0ICogYXMgbmV0d29yayBmcm9tICcuL25ldHdvcmsnO1xuXG4vKipcbiAqIENvbnN0cnVjdCBwcm9wZXJ0aWVzIGZvciBgSHlwZXJsZWRnZXJGYWJyaWNVc2VyYFxuICovXG5leHBvcnQgaW50ZXJmYWNlIEh5cGVybGVkZ2VyRmFicmljVXNlclByb3BzIHtcblxuICAvKipcbiAgICogVXNlcidzIGFmZmlsaWF0aW9uIHRvIHRoZSBtZW1iZXIuXG4gICAqIFNob3VsZCBiZSBoaWVyYXJjaGljYWwgd2l0aCBtZW1iZXIgbmFtZSBhcyByb290KGBNZW1iZXJOYW1lLkRlcHQxYCkuXG4gICAqL1xuICByZWFkb25seSBhZmZpbGl0YXRpb246IHN0cmluZztcblxuICAvKipcbiAgICogVXNlciBJRCB0byByZWdpc3RlciB3aXRoIENBXG4gICAqL1xuICByZWFkb25seSB1c2VySWQ6IHN0cmluZztcblxufVxuXG4vKipcbiAqIENyZWF0ZXMgY3VzdG9tIHJlc291cmNlcyB0byByZWdpc3RlciBhbmQgZW5yb2xsIHVzZXJzIGlkZW50aXRpZXNcbiAqIHdpdGggdGhlIENBIHVzaW5nIHRoZSBmYWJyaWMtY2EtY2xpZW50IFNES1xuICovXG5leHBvcnQgY2xhc3MgSHlwZXJsZWRnZXJGYWJyaWNVc2VyIGV4dGVuZHMgY29uc3RydWN0cy5Db25zdHJ1Y3Qge1xuXG4gIC8qKlxuICAgKiBVc2VyIElEIHJlZ2lzdGVyZWQgd2l0aCBDQVxuICAgKi9cbiAgcHVibGljIHJlYWRvbmx5IHVzZXJJZDogc3RyaW5nO1xuXG4gIC8qKlxuICAgKiBVc2VyJ3MgYWZmaWxpYXRpb24gdG8gdGhlIG1lbWJlclxuICAgKi9cbiAgcHVibGljIHJlYWRvbmx5IGFmZmlsaWF0aW9uOiBzdHJpbmc7XG5cbiAgLyoqXG4gICAqIFNlY3JldCBmb3IgdXNlciBzaWduZWQgY2VydGlmaWNhdGVcbiAgICovXG4gIHB1YmxpYyByZWFkb25seSB1c2VyU2lnbmVkQ2VydFNlY3JldDogc2VjcmV0c21hbmFnZXIuU2VjcmV0O1xuXG4gIC8qKlxuICAgKiBTZWNyZXQgZm9yIHVzZXIgcHJpdmF0ZSBrZXlcbiAgICovXG4gIHB1YmxpYyByZWFkb25seSB1c2VyUHJpdmF0ZUtleVNlY3JldDogc2VjcmV0c21hbmFnZXIuU2VjcmV0O1xuXG5cbiAgY29uc3RydWN0b3Ioc2NvcGU6IG5ldHdvcmsuSHlwZXJsZWRnZXJGYWJyaWNOZXR3b3JrLCBpZDogc3RyaW5nLCBwcm9wczogSHlwZXJsZWRnZXJGYWJyaWNVc2VyUHJvcHMpIHtcbiAgICBzdXBlcihzY29wZSwgaWQpO1xuXG4gICAgLy8gUG9wdWxhdGUgaW5zdGFuY2UgdmFyaWFibGVzIGZyb20gaW5wdXQgcHJvcGVydGllc1xuICAgIHRoaXMudXNlcklkID0gcHJvcHMudXNlcklkO1xuICAgIHRoaXMuYWZmaWxpYXRpb24gPSBwcm9wcy5hZmZpbGl0YXRpb247XG5cbiAgICAvLyBHZXQgdGhlIGN1c3RvbSByZXNvdXJjZXMgZnJvbSBJZGVudGl0eSBSZXNvdXJjZXMgY2xhc3NcbiAgICBjb25zdCBjdXN0b21SZXNvdXJjZVJvbGUgPSBpZGVudGl0eS5IeXBlcmxlZGdlckZhYnJpY0lkZW50aXR5LmN1c3RvbVJvbGU7XG4gICAgY29uc3QgcmVnaXN0ZXJQcm92aWRlciA9IGlkZW50aXR5Lkh5cGVybGVkZ2VyRmFicmljSWRlbnRpdHkudXNlclByb3ZpZGVyO1xuXG4gICAgLy8gRW5zdXJlIHRoZSB1c2VyIGFmZmlsaWF0aW9uIGluY2x1ZGVzIHRoZSBtZW1iZXIgbmFtZVxuICAgIGlmICghdGhpcy5hZmZpbGlhdGlvbi5zdGFydHNXaXRoKHNjb3BlLm1lbWJlck5hbWUpKSB0aHJvdyBuZXcgRXJyb3IoJ1VzZXIgYWZmaWxpYXRpb24gaXMgaW52YWxpZC4gQWZmaWxpYXRpb24gc2hvdWxkIHN0YXJ0IHdpdGggTWVtYmVyIG5hbWUnKTtcblxuICAgIC8vIFRoZSB1c2VyIGNyZWRlbnRpYWxzIHdpbGwgYmUgc3RvcmVkIGluIHRoZXNlIHNlY3JldHNcbiAgICBjb25zdCB1c2VyUGFzc3dvcmRTZWNyZXQgPSBuZXcgc2VjcmV0c21hbmFnZXIuU2VjcmV0KHRoaXMsIGAke3RoaXMudXNlcklkfS1QYXNzd29yZGApO1xuICAgIHRoaXMudXNlclByaXZhdGVLZXlTZWNyZXQgPSBuZXcgc2VjcmV0c21hbmFnZXIuU2VjcmV0KHRoaXMsIGAke3RoaXMudXNlcklkfS1Qcml2YXRlS2V5YCk7XG4gICAgdGhpcy51c2VyU2lnbmVkQ2VydFNlY3JldCA9IG5ldyBzZWNyZXRzbWFuYWdlci5TZWNyZXQodGhpcywgYCR7dGhpcy51c2VySWR9LVNpZ25lZENlcnRgKTtcblxuICAgIC8vIEFkZCBzZWNyZXRzIGFybiB0byB0aGUgY3VzdG9tIHJvbGVcbiAgICBjdXN0b21SZXNvdXJjZVJvbGUuYWRkVG9Qb2xpY3koIG5ldyBpYW0uUG9saWN5U3RhdGVtZW50KHtcbiAgICAgIGFjdGlvbnM6IFsnc2VjcmV0c21hbmFnZXI6UHV0U2VjcmV0VmFsdWUnXSxcbiAgICAgIHJlc291cmNlczogW1xuICAgICAgICB1c2VyUGFzc3dvcmRTZWNyZXQuc2VjcmV0QXJuLFxuICAgICAgICB0aGlzLnVzZXJQcml2YXRlS2V5U2VjcmV0LnNlY3JldEFybixcbiAgICAgICAgdGhpcy51c2VyU2lnbmVkQ2VydFNlY3JldC5zZWNyZXRBcm4sXG4gICAgICBdLFxuICAgIH0pKTtcblxuICAgIGNvbnN0IHVzZXJQcm9wID0ge1xuICAgICAgdXNlcklkOiB0aGlzLnVzZXJJZCxcbiAgICAgIGFmZmlsaWF0aW9uOiB0aGlzLmFmZmlsaWF0aW9uLFxuICAgICAgcGFzc3dvcmRBcm46IHVzZXJQYXNzd29yZFNlY3JldC5zZWNyZXRBcm4sXG4gICAgICBwcml2YXRlS2V5QXJuOiB0aGlzLnVzZXJQcml2YXRlS2V5U2VjcmV0LnNlY3JldEFybixcbiAgICAgIHNpZ25lZENlcnRBcm46IHRoaXMudXNlclNpZ25lZENlcnRTZWNyZXQuc2VjcmV0QXJuLFxuICAgIH07XG5cbiAgICAvLyBDdXN0b20gcmVzb3VyY2UgdG8gZW5yb2xsIHRoZSBhZG1pbiBpZGVudGl0eSB1c2luZyBmYWJyaWMgY2xpZW50IFNES1xuICAgIG5ldyBjZGsuQ3VzdG9tUmVzb3VyY2UodGhpcywgYCR7dGhpcy51c2VySWR9UmVzb3VyY2VgLCB7IHNlcnZpY2VUb2tlbjogcmVnaXN0ZXJQcm92aWRlci5zZXJ2aWNlVG9rZW4sIHByb3BlcnRpZXM6IHVzZXJQcm9wIH0pO1xuXG4gIH1cblxufSJdfQ==
@@ -0,0 +1,12 @@
1
+ export declare const SUPPORTED_REGIONS: string[];
2
+ export declare const SUPPORTED_AVAILABILITY_ZONES: Record<string, Array<string>>;
3
+ export declare const STARTING_PORT = 30001;
4
+ export declare const ENDING_PORT = 30004;
5
+ export declare function getTlsBucket(region: string): {
6
+ bucketName: string;
7
+ key: string;
8
+ };
9
+ export declare function validateRegion(region: string): void;
10
+ export declare function validateAvailabilityZone(region: string, availabilityZone?: string): void;
11
+ export declare function validateInteger(value: number, min: number, max: number): boolean;
12
+ export declare function validateString(value: string, min: number, max: number, regexp?: RegExp | undefined): boolean;
@@ -0,0 +1,116 @@
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.validateString = exports.validateInteger = exports.validateAvailabilityZone = exports.validateRegion = exports.getTlsBucket = exports.ENDING_PORT = exports.STARTING_PORT = exports.SUPPORTED_AVAILABILITY_ZONES = exports.SUPPORTED_REGIONS = void 0;
6
+ /*
7
+ * Regions where Managed Blockchain is supported, for details see
8
+ * https://aws.amazon.com/managed-blockchain/pricing/hyperledger/
9
+ */
10
+ exports.SUPPORTED_REGIONS = [
11
+ 'ap-northeast-1',
12
+ 'ap-northeast-2',
13
+ 'ap-southeast-1',
14
+ 'eu-west-1',
15
+ 'eu-west-2',
16
+ 'us-east-1',
17
+ ];
18
+ /*
19
+ * Map of supported regions to their availability zones
20
+ */
21
+ exports.SUPPORTED_AVAILABILITY_ZONES = {
22
+ 'ap-northeast-1': [
23
+ 'ap-northeast-1a',
24
+ 'ap-northeast-1b',
25
+ 'ap-northeast-1c',
26
+ ],
27
+ 'ap-northeast-2': [
28
+ 'ap-northeast-2a',
29
+ 'ap-northeast-2b',
30
+ 'ap-northeast-2c',
31
+ 'ap-northeast-2d',
32
+ ],
33
+ 'ap-southeast-1': [
34
+ 'ap-southeast-1a',
35
+ 'ap-southeast-1b',
36
+ 'ap-southeast-1c',
37
+ ],
38
+ 'eu-west-1': [
39
+ 'eu-west-1a',
40
+ 'eu-west-1b',
41
+ 'eu-west-1c',
42
+ ],
43
+ 'eu-west-2': [
44
+ 'eu-west-2a',
45
+ 'eu-west-2b',
46
+ 'eu-west-2c',
47
+ ],
48
+ 'us-east-1': [
49
+ 'us-east-1a',
50
+ 'us-east-1b',
51
+ 'us-east-1c',
52
+ 'us-east-1d',
53
+ 'us-east-1e',
54
+ 'us-east-1f',
55
+ ],
56
+ };
57
+ /*
58
+ * Starting port of the Network port range
59
+ */
60
+ exports.STARTING_PORT = 30001;
61
+ /*
62
+ * Ending port of the Network port range
63
+ */
64
+ exports.ENDING_PORT = 30004;
65
+ /*
66
+ * Returns the S3 Bucket and key that contains the TLS cert file
67
+ */
68
+ function getTlsBucket(region) {
69
+ return { bucketName: `${region}.managedblockchain`, key: 'etc/managedblockchain-tls-chain.pem' };
70
+ }
71
+ exports.getTlsBucket = getTlsBucket;
72
+ /*
73
+ * Throw an error if provided region is not in the supported list
74
+ */
75
+ function validateRegion(region) {
76
+ if (!exports.SUPPORTED_REGIONS.includes(region)) {
77
+ const regionList = exports.SUPPORTED_REGIONS.join(', ');
78
+ throw new Error(`Managed Blockchain is only available in the following regions: ${regionList}.`);
79
+ }
80
+ }
81
+ exports.validateRegion = validateRegion;
82
+ /*
83
+ * Throw an error if provided availability is not in the supported list
84
+ */
85
+ function validateAvailabilityZone(region, availabilityZone) {
86
+ const availabililtyZonesForRegion = exports.SUPPORTED_AVAILABILITY_ZONES[region];
87
+ if (typeof availabilityZone === 'undefined' || !availabililtyZonesForRegion.includes(availabilityZone)) {
88
+ const availabilityZoneList = availabililtyZonesForRegion.join(', ');
89
+ throw new Error(`Managed Blockchain in ${region} is only available in the following availability zones: ${availabilityZoneList}.`);
90
+ }
91
+ }
92
+ exports.validateAvailabilityZone = validateAvailabilityZone;
93
+ /*
94
+ * Throw an error if provided number is not an integer, or not with the given range (inclusive)
95
+ */
96
+ function validateInteger(value, min, max) {
97
+ if (!Number.isInteger(value))
98
+ return false;
99
+ if (value < min || value > max)
100
+ return false;
101
+ return true;
102
+ }
103
+ exports.validateInteger = validateInteger;
104
+ /*
105
+ * Throw an error if provided string has length with a given range (inclusive),
106
+ * and optionally matches a provided regular expression pattern
107
+ */
108
+ function validateString(value, min, max, regexp) {
109
+ if (value.length < min || value.length > max)
110
+ return false;
111
+ if (typeof regexp !== 'undefined' && !value.match(regexp))
112
+ return false;
113
+ return true;
114
+ }
115
+ exports.validateString = validateString;
116
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidXRpbGl0aWVzLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vc3JjL3V0aWxpdGllcy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiO0FBQUEscUVBQXFFO0FBQ3JFLGlDQUFpQzs7O0FBR2pDOzs7R0FHRztBQUNVLFFBQUEsaUJBQWlCLEdBQUc7SUFDL0IsZ0JBQWdCO0lBQ2hCLGdCQUFnQjtJQUNoQixnQkFBZ0I7SUFDaEIsV0FBVztJQUNYLFdBQVc7SUFDWCxXQUFXO0NBQ1osQ0FBQztBQUVGOztHQUVHO0FBQ1UsUUFBQSw0QkFBNEIsR0FBa0M7SUFDekUsZ0JBQWdCLEVBQUU7UUFDaEIsaUJBQWlCO1FBQ2pCLGlCQUFpQjtRQUNqQixpQkFBaUI7S0FDbEI7SUFDRCxnQkFBZ0IsRUFBRTtRQUNoQixpQkFBaUI7UUFDakIsaUJBQWlCO1FBQ2pCLGlCQUFpQjtRQUNqQixpQkFBaUI7S0FDbEI7SUFDRCxnQkFBZ0IsRUFBRTtRQUNoQixpQkFBaUI7UUFDakIsaUJBQWlCO1FBQ2pCLGlCQUFpQjtLQUNsQjtJQUNELFdBQVcsRUFBRTtRQUNYLFlBQVk7UUFDWixZQUFZO1FBQ1osWUFBWTtLQUNiO0lBQ0QsV0FBVyxFQUFFO1FBQ1gsWUFBWTtRQUNaLFlBQVk7UUFDWixZQUFZO0tBQ2I7SUFDRCxXQUFXLEVBQUU7UUFDWCxZQUFZO1FBQ1osWUFBWTtRQUNaLFlBQVk7UUFDWixZQUFZO1FBQ1osWUFBWTtRQUNaLFlBQVk7S0FDYjtDQUNGLENBQUM7QUFFRjs7R0FFRztBQUNVLFFBQUEsYUFBYSxHQUFHLEtBQUssQ0FBQztBQUVuQzs7R0FFRztBQUNVLFFBQUEsV0FBVyxHQUFHLEtBQUssQ0FBQztBQUVqQzs7R0FFRztBQUNILFNBQWdCLFlBQVksQ0FBQyxNQUFjO0lBQ3pDLE9BQU8sRUFBRSxVQUFVLEVBQUUsR0FBRyxNQUFNLG9CQUFvQixFQUFFLEdBQUcsRUFBRSxxQ0FBcUMsRUFBRSxDQUFDO0FBQ25HLENBQUM7QUFGRCxvQ0FFQztBQUVEOztHQUVHO0FBQ0gsU0FBZ0IsY0FBYyxDQUFDLE1BQWM7SUFDM0MsSUFBSSxDQUFDLHlCQUFpQixDQUFDLFFBQVEsQ0FBQyxNQUFNLENBQUMsRUFBRTtRQUN2QyxNQUFNLFVBQVUsR0FBRyx5QkFBaUIsQ0FBQyxJQUFJLENBQUMsSUFBSSxDQUFDLENBQUM7UUFDaEQsTUFBTSxJQUFJLEtBQUssQ0FBQyxrRUFBa0UsVUFBVSxHQUFHLENBQUMsQ0FBQztLQUNsRztBQUNILENBQUM7QUFMRCx3Q0FLQztBQUVEOztHQUVHO0FBQ0gsU0FBZ0Isd0JBQXdCLENBQUMsTUFBYyxFQUFFLGdCQUF5QjtJQUNoRixNQUFNLDJCQUEyQixHQUFHLG9DQUE0QixDQUFDLE1BQU0sQ0FBQyxDQUFDO0lBQ3pFLElBQUksT0FBTyxnQkFBZ0IsS0FBSyxXQUFXLElBQUksQ0FBQywyQkFBMkIsQ0FBQyxRQUFRLENBQUMsZ0JBQWdCLENBQUMsRUFBRTtRQUN0RyxNQUFNLG9CQUFvQixHQUFHLDJCQUEyQixDQUFDLElBQUksQ0FBQyxJQUFJLENBQUMsQ0FBQztRQUNwRSxNQUFNLElBQUksS0FBSyxDQUFDLHlCQUF5QixNQUFNLDJEQUEyRCxvQkFBb0IsR0FBRyxDQUFDLENBQUM7S0FDcEk7QUFDSCxDQUFDO0FBTkQsNERBTUM7QUFFRDs7R0FFRztBQUNILFNBQWdCLGVBQWUsQ0FBQyxLQUFhLEVBQUUsR0FBVyxFQUFFLEdBQVc7SUFDckUsSUFBSSxDQUFDLE1BQU0sQ0FBQyxTQUFTLENBQUMsS0FBSyxDQUFDO1FBQUUsT0FBTyxLQUFLLENBQUM7SUFDM0MsSUFBSSxLQUFLLEdBQUcsR0FBRyxJQUFJLEtBQUssR0FBRyxHQUFHO1FBQUUsT0FBTyxLQUFLLENBQUM7SUFDN0MsT0FBTyxJQUFJLENBQUM7QUFDZCxDQUFDO0FBSkQsMENBSUM7QUFFRDs7O0dBR0c7QUFDSCxTQUFnQixjQUFjLENBQUMsS0FBYSxFQUFFLEdBQVcsRUFBRSxHQUFXLEVBQUUsTUFBMkI7SUFDakcsSUFBSSxLQUFLLENBQUMsTUFBTSxHQUFHLEdBQUcsSUFBSSxLQUFLLENBQUMsTUFBTSxHQUFHLEdBQUc7UUFBRSxPQUFPLEtBQUssQ0FBQztJQUMzRCxJQUFJLE9BQU8sTUFBTSxLQUFLLFdBQVcsSUFBSSxDQUFDLEtBQUssQ0FBQyxLQUFLLENBQUMsTUFBTSxDQUFDO1FBQUUsT0FBTyxLQUFLLENBQUM7SUFDeEUsT0FBTyxJQUFJLENBQUM7QUFDZCxDQUFDO0FBSkQsd0NBSUMiLCJzb3VyY2VzQ29udGVudCI6WyIvLyBDb3B5cmlnaHQgQW1hem9uLmNvbSwgSW5jLiBvciBpdHMgYWZmaWxpYXRlcy4gQWxsIFJpZ2h0cyBSZXNlcnZlZC5cbi8vIFNQRFgtTGljZW5zZS1JZGVudGlmaWVyOiBNSVQtMFxuXG5cbi8qXG4gKiBSZWdpb25zIHdoZXJlIE1hbmFnZWQgQmxvY2tjaGFpbiBpcyBzdXBwb3J0ZWQsIGZvciBkZXRhaWxzIHNlZVxuICogaHR0cHM6Ly9hd3MuYW1hem9uLmNvbS9tYW5hZ2VkLWJsb2NrY2hhaW4vcHJpY2luZy9oeXBlcmxlZGdlci9cbiAqL1xuZXhwb3J0IGNvbnN0IFNVUFBPUlRFRF9SRUdJT05TID0gW1xuICAnYXAtbm9ydGhlYXN0LTEnLFxuICAnYXAtbm9ydGhlYXN0LTInLFxuICAnYXAtc291dGhlYXN0LTEnLFxuICAnZXUtd2VzdC0xJyxcbiAgJ2V1LXdlc3QtMicsXG4gICd1cy1lYXN0LTEnLFxuXTtcblxuLypcbiAqIE1hcCBvZiBzdXBwb3J0ZWQgcmVnaW9ucyB0byB0aGVpciBhdmFpbGFiaWxpdHkgem9uZXNcbiAqL1xuZXhwb3J0IGNvbnN0IFNVUFBPUlRFRF9BVkFJTEFCSUxJVFlfWk9ORVM6IFJlY29yZDxzdHJpbmcsIEFycmF5PHN0cmluZz4+ID0ge1xuICAnYXAtbm9ydGhlYXN0LTEnOiBbXG4gICAgJ2FwLW5vcnRoZWFzdC0xYScsXG4gICAgJ2FwLW5vcnRoZWFzdC0xYicsXG4gICAgJ2FwLW5vcnRoZWFzdC0xYycsXG4gIF0sXG4gICdhcC1ub3J0aGVhc3QtMic6IFtcbiAgICAnYXAtbm9ydGhlYXN0LTJhJyxcbiAgICAnYXAtbm9ydGhlYXN0LTJiJyxcbiAgICAnYXAtbm9ydGhlYXN0LTJjJyxcbiAgICAnYXAtbm9ydGhlYXN0LTJkJyxcbiAgXSxcbiAgJ2FwLXNvdXRoZWFzdC0xJzogW1xuICAgICdhcC1zb3V0aGVhc3QtMWEnLFxuICAgICdhcC1zb3V0aGVhc3QtMWInLFxuICAgICdhcC1zb3V0aGVhc3QtMWMnLFxuICBdLFxuICAnZXUtd2VzdC0xJzogW1xuICAgICdldS13ZXN0LTFhJyxcbiAgICAnZXUtd2VzdC0xYicsXG4gICAgJ2V1LXdlc3QtMWMnLFxuICBdLFxuICAnZXUtd2VzdC0yJzogW1xuICAgICdldS13ZXN0LTJhJyxcbiAgICAnZXUtd2VzdC0yYicsXG4gICAgJ2V1LXdlc3QtMmMnLFxuICBdLFxuICAndXMtZWFzdC0xJzogW1xuICAgICd1cy1lYXN0LTFhJyxcbiAgICAndXMtZWFzdC0xYicsXG4gICAgJ3VzLWVhc3QtMWMnLFxuICAgICd1cy1lYXN0LTFkJyxcbiAgICAndXMtZWFzdC0xZScsXG4gICAgJ3VzLWVhc3QtMWYnLFxuICBdLFxufTtcblxuLypcbiAqIFN0YXJ0aW5nIHBvcnQgb2YgdGhlIE5ldHdvcmsgcG9ydCByYW5nZVxuICovXG5leHBvcnQgY29uc3QgU1RBUlRJTkdfUE9SVCA9IDMwMDAxO1xuXG4vKlxuICogRW5kaW5nIHBvcnQgb2YgdGhlIE5ldHdvcmsgcG9ydCByYW5nZVxuICovXG5leHBvcnQgY29uc3QgRU5ESU5HX1BPUlQgPSAzMDAwNDtcblxuLypcbiAqIFJldHVybnMgdGhlIFMzIEJ1Y2tldCBhbmQga2V5IHRoYXQgY29udGFpbnMgdGhlIFRMUyBjZXJ0IGZpbGVcbiAqL1xuZXhwb3J0IGZ1bmN0aW9uIGdldFRsc0J1Y2tldChyZWdpb246IHN0cmluZykge1xuICByZXR1cm4geyBidWNrZXROYW1lOiBgJHtyZWdpb259Lm1hbmFnZWRibG9ja2NoYWluYCwga2V5OiAnZXRjL21hbmFnZWRibG9ja2NoYWluLXRscy1jaGFpbi5wZW0nIH07XG59XG5cbi8qXG4gKiBUaHJvdyBhbiBlcnJvciBpZiBwcm92aWRlZCByZWdpb24gaXMgbm90IGluIHRoZSBzdXBwb3J0ZWQgbGlzdFxuICovXG5leHBvcnQgZnVuY3Rpb24gdmFsaWRhdGVSZWdpb24ocmVnaW9uOiBzdHJpbmcpIHtcbiAgaWYgKCFTVVBQT1JURURfUkVHSU9OUy5pbmNsdWRlcyhyZWdpb24pKSB7XG4gICAgY29uc3QgcmVnaW9uTGlzdCA9IFNVUFBPUlRFRF9SRUdJT05TLmpvaW4oJywgJyk7XG4gICAgdGhyb3cgbmV3IEVycm9yKGBNYW5hZ2VkIEJsb2NrY2hhaW4gaXMgb25seSBhdmFpbGFibGUgaW4gdGhlIGZvbGxvd2luZyByZWdpb25zOiAke3JlZ2lvbkxpc3R9LmApO1xuICB9XG59XG5cbi8qXG4gKiBUaHJvdyBhbiBlcnJvciBpZiBwcm92aWRlZCBhdmFpbGFiaWxpdHkgaXMgbm90IGluIHRoZSBzdXBwb3J0ZWQgbGlzdFxuICovXG5leHBvcnQgZnVuY3Rpb24gdmFsaWRhdGVBdmFpbGFiaWxpdHlab25lKHJlZ2lvbjogc3RyaW5nLCBhdmFpbGFiaWxpdHlab25lPzogc3RyaW5nKSB7XG4gIGNvbnN0IGF2YWlsYWJpbGlsdHlab25lc0ZvclJlZ2lvbiA9IFNVUFBPUlRFRF9BVkFJTEFCSUxJVFlfWk9ORVNbcmVnaW9uXTtcbiAgaWYgKHR5cGVvZiBhdmFpbGFiaWxpdHlab25lID09PSAndW5kZWZpbmVkJyB8fCAhYXZhaWxhYmlsaWx0eVpvbmVzRm9yUmVnaW9uLmluY2x1ZGVzKGF2YWlsYWJpbGl0eVpvbmUpKSB7XG4gICAgY29uc3QgYXZhaWxhYmlsaXR5Wm9uZUxpc3QgPSBhdmFpbGFiaWxpbHR5Wm9uZXNGb3JSZWdpb24uam9pbignLCAnKTtcbiAgICB0aHJvdyBuZXcgRXJyb3IoYE1hbmFnZWQgQmxvY2tjaGFpbiBpbiAke3JlZ2lvbn0gaXMgb25seSBhdmFpbGFibGUgaW4gdGhlIGZvbGxvd2luZyBhdmFpbGFiaWxpdHkgem9uZXM6ICR7YXZhaWxhYmlsaXR5Wm9uZUxpc3R9LmApO1xuICB9XG59XG5cbi8qXG4gKiBUaHJvdyBhbiBlcnJvciBpZiBwcm92aWRlZCBudW1iZXIgaXMgbm90IGFuIGludGVnZXIsIG9yIG5vdCB3aXRoIHRoZSBnaXZlbiByYW5nZSAoaW5jbHVzaXZlKVxuICovXG5leHBvcnQgZnVuY3Rpb24gdmFsaWRhdGVJbnRlZ2VyKHZhbHVlOiBudW1iZXIsIG1pbjogbnVtYmVyLCBtYXg6IG51bWJlcikge1xuICBpZiAoIU51bWJlci5pc0ludGVnZXIodmFsdWUpKSByZXR1cm4gZmFsc2U7XG4gIGlmICh2YWx1ZSA8IG1pbiB8fCB2YWx1ZSA+IG1heCkgcmV0dXJuIGZhbHNlO1xuICByZXR1cm4gdHJ1ZTtcbn1cblxuLypcbiAqIFRocm93IGFuIGVycm9yIGlmIHByb3ZpZGVkIHN0cmluZyBoYXMgbGVuZ3RoIHdpdGggYSBnaXZlbiByYW5nZSAoaW5jbHVzaXZlKSxcbiAqIGFuZCBvcHRpb25hbGx5IG1hdGNoZXMgYSBwcm92aWRlZCByZWd1bGFyIGV4cHJlc3Npb24gcGF0dGVyblxuICovXG5leHBvcnQgZnVuY3Rpb24gdmFsaWRhdGVTdHJpbmcodmFsdWU6IHN0cmluZywgbWluOiBudW1iZXIsIG1heDogbnVtYmVyLCByZWdleHA/OiBSZWdFeHAgfCB1bmRlZmluZWQpIHtcbiAgaWYgKHZhbHVlLmxlbmd0aCA8IG1pbiB8fCB2YWx1ZS5sZW5ndGggPiBtYXgpIHJldHVybiBmYWxzZTtcbiAgaWYgKHR5cGVvZiByZWdleHAgIT09ICd1bmRlZmluZWQnICYmICF2YWx1ZS5tYXRjaChyZWdleHApKSByZXR1cm4gZmFsc2U7XG4gIHJldHVybiB0cnVlO1xufVxuIl19