awscdk-construct-mediaconnect-flow 0.0.0

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,300 @@
1
+ "use strict";
2
+ var _a;
3
+ Object.defineProperty(exports, "__esModule", { value: true });
4
+ exports.LiveFeedFromFile = void 0;
5
+ exports.startFlow = startFlow;
6
+ const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
7
+ const crypto = require("crypto");
8
+ const cdk = require("aws-cdk-lib");
9
+ const ec2 = require("aws-cdk-lib/aws-ec2");
10
+ const iam = require("aws-cdk-lib/aws-iam");
11
+ const aws_mediaconnect_1 = require("aws-cdk-lib/aws-mediaconnect");
12
+ const asm = require("aws-cdk-lib/aws-secretsmanager");
13
+ const custom_resources_1 = require("aws-cdk-lib/custom-resources");
14
+ const awscdk_construct_medialive_channel_1 = require("awscdk-construct-medialive-channel");
15
+ const constructs_1 = require("constructs");
16
+ const sourceIngestPort = 5000;
17
+ const discoveryServerPort = 5959;
18
+ const VPC_INTERFACE_NAME = 'vpcInterfaceName';
19
+ class LiveFeedFromFile extends constructs_1.Construct {
20
+ constructor(scope, id, props) {
21
+ super(scope, id);
22
+ const { file, source = {
23
+ protocol: 'SRT',
24
+ type: 'STANDARD-SOURCE',
25
+ }, vpcConfig, autoStart = true, } = props;
26
+ const uuid = `${crypto.randomUUID()}`;
27
+ const protocol = (() => {
28
+ switch (source.protocol) {
29
+ case 'RTP':
30
+ return 'rtp';
31
+ case 'RTP-FEC':
32
+ return 'rtp-fec';
33
+ case 'SRT':
34
+ default:
35
+ return 'srt-listener';
36
+ }
37
+ })();
38
+ // Create a VPC
39
+ const vpc = vpcConfig ? new ec2.Vpc(this, 'VPC', vpcConfig.props) : undefined;
40
+ vpc && vpc.applyRemovalPolicy(cdk.RemovalPolicy.DESTROY);
41
+ // Allocate an Elastic IP for the VPC interface
42
+ const eip = vpc ? new ec2.CfnEIP(this, 'EIP', {
43
+ domain: 'vpc', // Ensure the EIP is allocated in the VPC
44
+ }) : undefined;
45
+ // Create a security group to allow push input
46
+ const description = 'Allow Push input from MediaLive';
47
+ const sg = vpc ? new ec2.SecurityGroup(this, 'SecurityGroup', {
48
+ vpc,
49
+ description,
50
+ allowAllOutbound: true,
51
+ }) : undefined;
52
+ sg && sg.applyRemovalPolicy(cdk.RemovalPolicy.DESTROY);
53
+ sg && sg.addIngressRule(ec2.Peer.anyIpv4(), ec2.Port.udp(sourceIngestPort), description);
54
+ // Create an NDI discovery server
55
+ let ndiConfig;
56
+ if (vpc && vpcConfig?.enableNDI) {
57
+ const instance = createNdiDiscoveryServer(this, vpc);
58
+ ndiConfig = {
59
+ ndiDiscoveryServers: [{
60
+ discoveryServerAddress: instance.instancePrivateIp, // Use the private IP of the NDI Discovery Server
61
+ vpcInterfaceAdapter: VPC_INTERFACE_NAME,
62
+ discoveryServerPort,
63
+ }],
64
+ ndiState: 'ENABLED',
65
+ };
66
+ }
67
+ // Create a secret
68
+ const randomstring = Math.random().toString(36).slice(-8);
69
+ const sourcePassword = new asm.Secret(this, 'SourcePassword', {
70
+ secretName: `secret-${uuid}`,
71
+ generateSecretString: {
72
+ secretStringTemplate: JSON.stringify({ password: randomstring }),
73
+ generateStringKey: 'password',
74
+ excludePunctuation: true,
75
+ },
76
+ removalPolicy: cdk.RemovalPolicy.DESTROY,
77
+ });
78
+ // Create an IAM role for MediaConnect to access the VPC
79
+ const role = new iam.Role(this, 'MediaConnectRole', {
80
+ assumedBy: new iam.ServicePrincipal('mediaconnect.amazonaws.com'),
81
+ inlinePolicies: {
82
+ policy: new iam.PolicyDocument({
83
+ statements: [
84
+ new iam.PolicyStatement({
85
+ resources: [sourcePassword.secretArn],
86
+ actions: [
87
+ 'secretsmanager:GetResourcePolicy',
88
+ 'secretsmanager:GetSecretValue',
89
+ 'secretsmanager:DescribeSecret',
90
+ 'secretsmanager:ListSecretVersionIds',
91
+ ],
92
+ }),
93
+ ],
94
+ }),
95
+ },
96
+ managedPolicies: [
97
+ iam.ManagedPolicy.fromAwsManagedPolicyName('AmazonVPCFullAccess'),
98
+ ],
99
+ });
100
+ // Create a MediaConnect flow
101
+ const flow = new aws_mediaconnect_1.CfnFlow(this, 'MyCfnFlow', {
102
+ name: `lcp-demo-${uuid}`,
103
+ source: {
104
+ name: `lcp-demo-source-${uuid}`,
105
+ protocol,
106
+ // maxLatency: 2000,
107
+ // minLatency: 1000,
108
+ // vpcInterfaceName: VPC_INTERFACE_NAME,
109
+ whitelistCidr: source.type === 'STANDARD-SOURCE' ? '0.0.0.0/0' : undefined,
110
+ decryption: {
111
+ // algorithm: 'aes128',
112
+ roleArn: role.roleArn,
113
+ secretArn: sourcePassword.secretArn,
114
+ },
115
+ // sourceIngestPort: `${sourceIngestPort}`,
116
+ vpcInterfaceName: source.type === 'VPC-SOURCE' ? VPC_INTERFACE_NAME : undefined,
117
+ },
118
+ availabilityZone: vpcConfig?.availabilityZone ?? vpc?.availabilityZones[0],
119
+ flowSize: vpcConfig?.enableNDI ? 'LARGE' : 'MEDIUM',
120
+ ndiConfig,
121
+ sourceMonitoringConfig: {
122
+ thumbnailState: 'ENABLED',
123
+ contentQualityAnalysisState: 'ENABLED',
124
+ },
125
+ vpcInterfaces: vpc ? [{
126
+ name: VPC_INTERFACE_NAME,
127
+ roleArn: role.roleArn,
128
+ securityGroupIds: [sg.securityGroupId],
129
+ subnetId: vpcConfig?.subnetId ?? vpc.privateSubnets[0].subnetId,
130
+ }] : [],
131
+ });
132
+ flow.applyRemovalPolicy(cdk.RemovalPolicy.DESTROY);
133
+ // Start the MediaConnect Flow
134
+ autoStart && startFlow(this, 'StartMediaConnectFlow', flow.attrFlowArn);
135
+ // Create MediaLive channel
136
+ const eml = new awscdk_construct_medialive_channel_1.MediaLive(this, 'MediaLive', {
137
+ sources: [{ url: file.url.replace('s3://', 's3ssl://'), type: file.type ?? 'MP4_FILE' }],
138
+ destinations: [{
139
+ id: 'SRT',
140
+ srtSettings: [{
141
+ url: `srt://${flow.attrSourceIngestIp}:${flow.attrSourceSourceIngestPort}`, // Use the MediaConnect Flow URL
142
+ encryptionPassphraseSecretArn: sourcePassword.secretArn,
143
+ }],
144
+ }],
145
+ channelClass: 'SINGLE_PIPELINE',
146
+ encoderSpec: {
147
+ outputGroupSettingsList: [{
148
+ srtGroupSettings: {
149
+ inputLossAction: 'DROP_TS',
150
+ },
151
+ }],
152
+ outputSettingsList: [{
153
+ // Add valid OutputSettingsProperty fields here if needed
154
+ srtOutputSettings: {
155
+ latency: 2000, // Latency in milliseconds
156
+ destination: {
157
+ destinationRefId: 'SRT',
158
+ },
159
+ encryptionType: 'AES128', // Encryption type
160
+ containerSettings: {
161
+ m2TsSettings: {},
162
+ },
163
+ },
164
+ }],
165
+ gopLengthInSeconds: 2, // The length of the GOP in seconds.
166
+ timecodeBurninPrefix: 'Ch', // The prefix for the timecode burn-in.
167
+ },
168
+ vpc: source.type === 'VPC-SOURCE' && vpc ? {
169
+ publicAddressAllocationIds: [eip.attrAllocationId],
170
+ subnetIds: [vpc.privateSubnets[0].subnetId],
171
+ securityGroupIds: [sg.securityGroupId],
172
+ } : undefined,
173
+ secret: sourcePassword,
174
+ });
175
+ // Start the MediaLive channel
176
+ autoStart && (0, awscdk_construct_medialive_channel_1.startChannel)(this, 'StartMediaLiveChannel', eml.channel.ref);
177
+ this.flow = flow;
178
+ this.vpc = vpc;
179
+ }
180
+ }
181
+ exports.LiveFeedFromFile = LiveFeedFromFile;
182
+ _a = JSII_RTTI_SYMBOL_1;
183
+ LiveFeedFromFile[_a] = { fqn: "awscdk-construct-mediaconnect-flow.LiveFeedFromFile", version: "0.0.0" };
184
+ function createNdiDiscoveryServer(scope, vpc) {
185
+ const description = 'Allow NDI Discovery Service';
186
+ const sg = new ec2.SecurityGroup(scope, 'NDISecurityGroup', {
187
+ vpc,
188
+ description,
189
+ allowAllOutbound: true,
190
+ });
191
+ sg.applyRemovalPolicy(cdk.RemovalPolicy.DESTROY);
192
+ sg.addIngressRule(ec2.Peer.anyIpv4(), ec2.Port.tcp(discoveryServerPort), description);
193
+ const instance = new ec2.Instance(scope, 'Instance', {
194
+ vpc,
195
+ instanceType: ec2.InstanceType.of(ec2.InstanceClass.BURSTABLE3, ec2.InstanceSize.MICRO),
196
+ machineImage: ec2.MachineImage.latestAmazonLinux2023(),
197
+ vpcSubnets: { subnetType: ec2.SubnetType.PRIVATE_WITH_EGRESS },
198
+ securityGroup: sg,
199
+ });
200
+ instance.applyRemovalPolicy(cdk.RemovalPolicy.DESTROY);
201
+ // Get the CloudFormation logical ID for the instance
202
+ const cfnInstance = instance.node.defaultChild;
203
+ const logicalId = cfnInstance.logicalId;
204
+ instance.applyCloudFormationInit(ec2.CloudFormationInit.fromConfigSets({
205
+ configSets: {
206
+ // Applies the configs below in this order
207
+ default: ['configInstance', 'finalize'],
208
+ },
209
+ configs: {
210
+ configInstance: new ec2.InitConfig([
211
+ // NDI Discovery Service script
212
+ ec2.InitFile.fromString('/etc/systemd/ndi-discovery.service', `[Unit]
213
+ Description=NDI Discovery Service
214
+
215
+ [Service]
216
+ ExecStartPre=/bin/sleep 30
217
+ User=ec2-user
218
+ WorkingDirectory=/home/ec2-user/bin/x86_64-linux-gnu
219
+ ExecStart=/home/ec2-user/bin/x86_64-linux-gnu/ndi-discovery-server
220
+ Restart=always
221
+
222
+ [Install]
223
+ WantedBy=multi-user.target
224
+ `),
225
+ // NDI Discovery Server Installation script
226
+ ec2.InitFile.fromString('/tmp/install-ndi-discovery.sh', `#!/bin/bash -xe
227
+ cd /home/ec2-user
228
+
229
+ #Install updates
230
+ yum update -y
231
+
232
+ #Download the NDI Linux SDK and extract it and run the install script
233
+ wget https://downloads.ndi.tv/SDK/NDI_SDK_Linux/Install_NDI_SDK_v6_Linux.tar.gz
234
+ tar -xzf Install_NDI_SDK_v6_Linux.tar.gz
235
+ yes | sudo ./Install_NDI_SDK_v6_Linux.sh
236
+
237
+ #Clean up and prepare directories
238
+ rm -f Install_NDI_SDK_v6_Linux.tar.gz
239
+ rm -f Install_NDI_SDK_v6_Linux.sh
240
+ cp -r 'NDI SDK for Linux'/* ./
241
+ rm -r 'NDI SDK for Linux'/
242
+ `),
243
+ // Install the NDI Discovery Service
244
+ ec2.InitCommand.shellCommand('sudo bash /tmp/install-ndi-discovery.sh'),
245
+ // Run the NDI Discovery Service
246
+ ec2.InitCommand.shellCommand('sudo systemctl enable /etc/systemd/ndi-discovery.service'),
247
+ ]),
248
+ finalize: new ec2.InitConfig([
249
+ // Start the NDI Discovery Service
250
+ ec2.InitCommand.shellCommand(cdk.Fn.sub('/opt/aws/bin/cfn-signal -e $? --stack ${StackName} --resource ${Resource} --region ${Region}', {
251
+ StackName: cdk.Aws.STACK_NAME,
252
+ Resource: logicalId,
253
+ Region: cdk.Aws.REGION,
254
+ })),
255
+ ]),
256
+ },
257
+ }));
258
+ instance.addUserData(cdk.Fn.sub(`
259
+ #!/bin/bash -xe
260
+ yum install -y aws-cfn-bootstrap
261
+ sudo /opt/aws/bin/cfn-init --configsets default -v --stack \${StackName} --resource \${Resource} --region \${Region}
262
+ sudo reboot
263
+ `, {
264
+ StackName: cdk.Aws.STACK_NAME,
265
+ Resource: logicalId,
266
+ Region: cdk.Aws.REGION,
267
+ }));
268
+ return instance;
269
+ }
270
+ function startFlow(scope, id, flowArn) {
271
+ // Start channel
272
+ new custom_resources_1.AwsCustomResource(scope, id, {
273
+ onCreate: {
274
+ service: 'MediaConnect',
275
+ action: 'StartFlow',
276
+ parameters: {
277
+ FlowArn: flowArn,
278
+ },
279
+ physicalResourceId: custom_resources_1.PhysicalResourceId.of(`${crypto.randomUUID()}`),
280
+ // ignoreErrorCodesMatching: '*',
281
+ outputPaths: ['FlowArn', 'Status'],
282
+ },
283
+ onDelete: {
284
+ service: 'MediaConnect',
285
+ action: 'StopFlow',
286
+ parameters: {
287
+ FlowArn: flowArn,
288
+ },
289
+ physicalResourceId: custom_resources_1.PhysicalResourceId.of(`${crypto.randomUUID()}`),
290
+ // ignoreErrorCodesMatching: '*',
291
+ outputPaths: ['FlowArn', 'Status'],
292
+ },
293
+ //Will ignore any resource and use the assumedRoleArn as resource and 'sts:AssumeRole' for service:action
294
+ policy: custom_resources_1.AwsCustomResourcePolicy.fromSdkCalls({
295
+ resources: custom_resources_1.AwsCustomResourcePolicy.ANY_RESOURCE,
296
+ }),
297
+ });
298
+ return new Date();
299
+ }
300
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"LiveFeedFromFile.js","sourceRoot":"","sources":["../src/LiveFeedFromFile.ts"],"names":[],"mappings":"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","sourcesContent":["import * as crypto from 'crypto';\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 { CfnFlow } from 'aws-cdk-lib/aws-mediaconnect';\nimport * as asm from 'aws-cdk-lib/aws-secretsmanager';\nimport { AwsCustomResource, AwsCustomResourcePolicy, PhysicalResourceId } from 'aws-cdk-lib/custom-resources';\nimport { MediaLive, startChannel } from 'awscdk-construct-medialive-channel';\nimport { Construct } from 'constructs';\n\nexport interface FileSpec {\n  readonly type?: 'MP4_FILE' | 'TS_FILE'; // Type of the input file\n  readonly url: string; // S3 URL of the input file\n}\n\nexport interface LiveSourceSpec {\n  readonly protocol: 'RTP' | 'RTP-FEC' | 'SRT'; // Protocol of the live source\n  readonly type: 'STANDARD-SOURCE' | 'VPC-SOURCE'; // Type of the live source\n}\n\nexport interface VpcConfig {\n  readonly props: ec2.VpcProps;\n  readonly availabilityZone?: string;\n  readonly subnetId?: string;\n  readonly enableNDI?: boolean; // Settings for NDI output\n}\n\nexport interface LiveFeedFromFileProps {\n  readonly file: FileSpec; // File specification\n  readonly source?: LiveSourceSpec; // Optional live source specification\n  readonly vpcConfig?: VpcConfig; // Settings for VPC. Required when the source type is VPC-SOURCE and/or VPC outputs will be added to this flow.\n  readonly autoStart?: boolean; // Whether to automatically start the MediaLive channel and MediaConnect flow\n}\n\nconst sourceIngestPort = 5000;\nconst discoveryServerPort = 5959;\nconst VPC_INTERFACE_NAME = 'vpcInterfaceName';\n\nexport class LiveFeedFromFile extends Construct {\n  public readonly flow: CfnFlow;\n  public readonly vpc?: ec2.IVpc;\n\n  constructor(scope: Construct, id: string, props: LiveFeedFromFileProps) {\n    super(scope, id);\n\n    const {\n      file,\n      source = {\n        protocol: 'SRT',\n        type: 'STANDARD-SOURCE',\n      },\n      vpcConfig,\n      autoStart = true,\n    } = props;\n\n    const uuid = `${crypto.randomUUID()}`;\n    const protocol = (() => {\n      switch (source.protocol) {\n        case 'RTP':\n          return 'rtp';\n        case 'RTP-FEC':\n          return 'rtp-fec';\n        case 'SRT':\n        default:\n          return 'srt-listener';\n      }\n    })();\n\n    // Create a VPC\n    const vpc = vpcConfig ? new ec2.Vpc(this, 'VPC', vpcConfig.props) : undefined;\n    vpc && vpc.applyRemovalPolicy(cdk.RemovalPolicy.DESTROY);\n\n    // Allocate an Elastic IP for the VPC interface\n    const eip = vpc ? new ec2.CfnEIP(this, 'EIP', {\n      domain: 'vpc', // Ensure the EIP is allocated in the VPC\n    }) : undefined;\n\n    // Create a security group to allow push input\n    const description = 'Allow Push input from MediaLive';\n    const sg = vpc ? new ec2.SecurityGroup(this, 'SecurityGroup', {\n      vpc,\n      description,\n      allowAllOutbound: true,\n    }): undefined;\n    sg && sg.applyRemovalPolicy(cdk.RemovalPolicy.DESTROY);\n    sg && sg.addIngressRule(\n      ec2.Peer.anyIpv4(),\n      ec2.Port.udp(sourceIngestPort),\n      description,\n    );\n\n    // Create an NDI discovery server\n    let ndiConfig: CfnFlow.NdiConfigProperty | undefined;\n    if (vpc && vpcConfig?.enableNDI) {\n      const instance = createNdiDiscoveryServer(this, vpc);\n      ndiConfig = {\n        ndiDiscoveryServers: [{\n          discoveryServerAddress: instance.instancePrivateIp, // Use the private IP of the NDI Discovery Server\n          vpcInterfaceAdapter: VPC_INTERFACE_NAME,\n          discoveryServerPort,\n        }],\n        ndiState: 'ENABLED',\n      };\n    }\n\n    // Create a secret\n    const randomstring = Math.random().toString(36).slice(-8);\n    const sourcePassword = new asm.Secret(this, 'SourcePassword', {\n      secretName: `secret-${uuid}`,\n      generateSecretString: {\n        secretStringTemplate: JSON.stringify({ password: randomstring }),\n        generateStringKey: 'password',\n        excludePunctuation: true,\n      },\n      removalPolicy: cdk.RemovalPolicy.DESTROY,\n    });\n\n    // Create an IAM role for MediaConnect to access the VPC\n    const role = new iam.Role(this, 'MediaConnectRole', {\n      assumedBy: new iam.ServicePrincipal('mediaconnect.amazonaws.com'),\n      inlinePolicies: {\n        policy: new iam.PolicyDocument({\n          statements: [\n            new iam.PolicyStatement({\n              resources: [sourcePassword.secretArn],\n              actions: [\n                'secretsmanager:GetResourcePolicy',\n                'secretsmanager:GetSecretValue',\n                'secretsmanager:DescribeSecret',\n                'secretsmanager:ListSecretVersionIds',\n              ],\n            }),\n          ],\n        }),\n      },\n      managedPolicies: [\n        iam.ManagedPolicy.fromAwsManagedPolicyName('AmazonVPCFullAccess'),\n      ],\n    });\n\n    // Create a MediaConnect flow\n    const flow = new CfnFlow(this, 'MyCfnFlow', {\n      name: `lcp-demo-${uuid}`,\n      source: {\n        name: `lcp-demo-source-${uuid}`,\n        protocol,\n        // maxLatency: 2000,\n        // minLatency: 1000,\n        // vpcInterfaceName: VPC_INTERFACE_NAME,\n        whitelistCidr: source.type === 'STANDARD-SOURCE' ? '0.0.0.0/0' : undefined,\n        decryption: {\n          // algorithm: 'aes128',\n          roleArn: role.roleArn,\n          secretArn: sourcePassword.secretArn,\n        },\n        // sourceIngestPort: `${sourceIngestPort}`,\n        vpcInterfaceName: source.type === 'VPC-SOURCE' ? VPC_INTERFACE_NAME : undefined,\n      },\n      availabilityZone: vpcConfig?.availabilityZone ?? vpc?.availabilityZones[0],\n      flowSize: vpcConfig?.enableNDI ? 'LARGE' : 'MEDIUM',\n      ndiConfig,\n      sourceMonitoringConfig: {\n        thumbnailState: 'ENABLED',\n        contentQualityAnalysisState: 'ENABLED',\n      },\n      vpcInterfaces: vpc ? [{\n        name: VPC_INTERFACE_NAME,\n        roleArn: role.roleArn,\n        securityGroupIds: [sg!.securityGroupId],\n        subnetId: vpcConfig?.subnetId ?? vpc.privateSubnets[0].subnetId,\n      }] : [],\n    });\n    flow.applyRemovalPolicy(cdk.RemovalPolicy.DESTROY);\n\n    // Start the MediaConnect Flow\n    autoStart && startFlow(this, 'StartMediaConnectFlow', flow.attrFlowArn);\n\n    // Create MediaLive channel\n    const eml = new MediaLive(this, 'MediaLive', {\n      sources: [{ url: file.url.replace('s3://', 's3ssl://'), type: file.type ?? 'MP4_FILE' }],\n      destinations: [{\n        id: 'SRT',\n        srtSettings: [{\n          url: `srt://${flow.attrSourceIngestIp}:${flow.attrSourceSourceIngestPort}`, // Use the MediaConnect Flow URL\n          encryptionPassphraseSecretArn: sourcePassword.secretArn,\n        }],\n      }],\n      channelClass: 'SINGLE_PIPELINE',\n      encoderSpec: {\n        outputGroupSettingsList: [{\n          srtGroupSettings: {\n            inputLossAction: 'DROP_TS',\n          },\n        }],\n        outputSettingsList: [{\n          // Add valid OutputSettingsProperty fields here if needed\n          srtOutputSettings: {\n            latency: 2000, // Latency in milliseconds\n            destination: {\n              destinationRefId: 'SRT',\n            },\n            encryptionType: 'AES128', // Encryption type\n            containerSettings: {\n              m2TsSettings: {},\n            },\n          },\n        }],\n        gopLengthInSeconds: 2, // The length of the GOP in seconds.\n        timecodeBurninPrefix: 'Ch', // The prefix for the timecode burn-in.\n      },\n      vpc: source.type === 'VPC-SOURCE' && vpc ? {\n        publicAddressAllocationIds: [eip!.attrAllocationId],\n        subnetIds: [vpc.privateSubnets[0].subnetId],\n        securityGroupIds: [sg!.securityGroupId],\n      } : undefined,\n      secret: sourcePassword,\n    });\n\n    // Start the MediaLive channel\n    autoStart && startChannel(this, 'StartMediaLiveChannel', eml.channel.ref);\n\n    this.flow = flow;\n    this.vpc = vpc;\n  }\n}\n\nfunction createNdiDiscoveryServer(scope: Construct, vpc: ec2.IVpc): ec2.Instance {\n  const description = 'Allow NDI Discovery Service';\n  const sg = new ec2.SecurityGroup(scope, 'NDISecurityGroup', {\n    vpc,\n    description,\n    allowAllOutbound: true,\n  });\n  sg.applyRemovalPolicy(cdk.RemovalPolicy.DESTROY);\n  sg.addIngressRule(\n    ec2.Peer.anyIpv4(),\n    ec2.Port.tcp(discoveryServerPort),\n    description,\n  );\n  const instance = new ec2.Instance(scope, 'Instance', {\n    vpc,\n    instanceType: ec2.InstanceType.of(ec2.InstanceClass.BURSTABLE3, ec2.InstanceSize.MICRO),\n    machineImage: ec2.MachineImage.latestAmazonLinux2023(),\n    vpcSubnets: { subnetType: ec2.SubnetType.PRIVATE_WITH_EGRESS },\n    securityGroup: sg,\n  });\n  instance.applyRemovalPolicy(cdk.RemovalPolicy.DESTROY);\n  // Get the CloudFormation logical ID for the instance\n  const cfnInstance = instance.node.defaultChild as cdk.CfnResource;\n  const logicalId = cfnInstance.logicalId;\n  instance.applyCloudFormationInit(ec2.CloudFormationInit.fromConfigSets({\n    configSets: {\n      // Applies the configs below in this order\n      default: ['configInstance', 'finalize'],\n    },\n    configs: {\n      configInstance: new ec2.InitConfig([\n        // NDI Discovery Service script\n        ec2.InitFile.fromString(\n          '/etc/systemd/ndi-discovery.service',\n          `[Unit]\n          Description=NDI Discovery Service\n          \n          [Service]\n          ExecStartPre=/bin/sleep 30\n          User=ec2-user\n          WorkingDirectory=/home/ec2-user/bin/x86_64-linux-gnu\n          ExecStart=/home/ec2-user/bin/x86_64-linux-gnu/ndi-discovery-server\n          Restart=always\n\n          [Install]\n          WantedBy=multi-user.target\n        `),\n        // NDI Discovery Server Installation script\n        ec2.InitFile.fromString(\n          '/tmp/install-ndi-discovery.sh',\n          `#!/bin/bash -xe\n          cd /home/ec2-user\n\n          #Install updates\n          yum update -y\n\n          #Download the NDI Linux SDK and extract it and run the install script\n          wget https://downloads.ndi.tv/SDK/NDI_SDK_Linux/Install_NDI_SDK_v6_Linux.tar.gz\n          tar -xzf Install_NDI_SDK_v6_Linux.tar.gz\n          yes | sudo ./Install_NDI_SDK_v6_Linux.sh\n\n          #Clean up and prepare directories\n          rm -f Install_NDI_SDK_v6_Linux.tar.gz\n          rm -f Install_NDI_SDK_v6_Linux.sh\n          cp -r 'NDI SDK for Linux'/* ./\n          rm -r 'NDI SDK for Linux'/\n        `),\n        // Install the NDI Discovery Service\n        ec2.InitCommand.shellCommand('sudo bash /tmp/install-ndi-discovery.sh'),\n        // Run the NDI Discovery Service\n        ec2.InitCommand.shellCommand('sudo systemctl enable /etc/systemd/ndi-discovery.service'),\n      ]),\n      finalize: new ec2.InitConfig([\n        // Start the NDI Discovery Service\n        ec2.InitCommand.shellCommand(\n          cdk.Fn.sub('/opt/aws/bin/cfn-signal -e $? --stack ${StackName} --resource ${Resource} --region ${Region}',\n            {\n              StackName: cdk.Aws.STACK_NAME,\n              Resource: logicalId,\n              Region: cdk.Aws.REGION,\n            }),\n        ),\n      ]),\n    },\n  }));\n  instance.addUserData(cdk.Fn.sub(`\n    #!/bin/bash -xe\n    yum install -y aws-cfn-bootstrap\n    sudo /opt/aws/bin/cfn-init --configsets default -v --stack \\${StackName} --resource \\${Resource} --region \\${Region}\n    sudo reboot\n  `, {\n    StackName: cdk.Aws.STACK_NAME,\n    Resource: logicalId,\n    Region: cdk.Aws.REGION,\n  }));\n  return instance;\n}\n\nexport function startFlow(scope: Construct, id: string, flowArn: string): Date {\n  // Start channel\n  new AwsCustomResource(scope, id, {\n    onCreate: {\n      service: 'MediaConnect',\n      action: 'StartFlow',\n      parameters: {\n        FlowArn: flowArn,\n      },\n      physicalResourceId: PhysicalResourceId.of(`${crypto.randomUUID()}`),\n      // ignoreErrorCodesMatching: '*',\n      outputPaths: ['FlowArn', 'Status'],\n    },\n    onDelete: {\n      service: 'MediaConnect',\n      action: 'StopFlow',\n      parameters: {\n        FlowArn: flowArn,\n      },\n      physicalResourceId: PhysicalResourceId.of(`${crypto.randomUUID()}`),\n      // ignoreErrorCodesMatching: '*',\n      outputPaths: ['FlowArn', 'Status'],\n    },\n    //Will ignore any resource and use the assumedRoleArn as resource and 'sts:AssumeRole' for service:action\n    policy: AwsCustomResourcePolicy.fromSdkCalls({\n      resources: AwsCustomResourcePolicy.ANY_RESOURCE,\n    }),\n  });\n  return new Date();\n}\n"]}
package/lib/index.d.ts ADDED
@@ -0,0 +1 @@
1
+ export * from './LiveFeedFromFile';
package/lib/index.js ADDED
@@ -0,0 +1,18 @@
1
+ "use strict";
2
+ var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
3
+ if (k2 === undefined) k2 = k;
4
+ var desc = Object.getOwnPropertyDescriptor(m, k);
5
+ if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
6
+ desc = { enumerable: true, get: function() { return m[k]; } };
7
+ }
8
+ Object.defineProperty(o, k2, desc);
9
+ }) : (function(o, m, k, k2) {
10
+ if (k2 === undefined) k2 = k;
11
+ o[k2] = m[k];
12
+ }));
13
+ var __exportStar = (this && this.__exportStar) || function(m, exports) {
14
+ for (var p in m) if (p !== "default" && !Object.prototype.hasOwnProperty.call(exports, p)) __createBinding(exports, m, p);
15
+ };
16
+ Object.defineProperty(exports, "__esModule", { value: true });
17
+ __exportStar(require("./LiveFeedFromFile"), exports);
18
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi9zcmMvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7Ozs7Ozs7Ozs7Ozs7OztBQUFBLHFEQUFtQyIsInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCAqIGZyb20gJy4vTGl2ZUZlZWRGcm9tRmlsZSc7Il19
package/package.json ADDED
@@ -0,0 +1,134 @@
1
+ {
2
+ "name": "awscdk-construct-mediaconnect-flow",
3
+ "repository": {
4
+ "type": "git",
5
+ "url": "https://github.com/kuu/awscdk-construct-mediaconnect-flow.git"
6
+ },
7
+ "scripts": {
8
+ "build": "npx projen build",
9
+ "bump": "npx projen bump",
10
+ "clobber": "npx projen clobber",
11
+ "compat": "npx projen compat",
12
+ "compile": "npx projen compile",
13
+ "default": "npx projen default",
14
+ "docgen": "npx projen docgen",
15
+ "eject": "npx projen eject",
16
+ "eslint": "npx projen eslint",
17
+ "package": "npx projen package",
18
+ "package-all": "npx projen package-all",
19
+ "package:js": "npx projen package:js",
20
+ "post-compile": "npx projen post-compile",
21
+ "post-upgrade": "npx projen post-upgrade",
22
+ "pre-compile": "npx projen pre-compile",
23
+ "release": "npx projen release",
24
+ "test": "npx projen test",
25
+ "test:watch": "npx projen test:watch",
26
+ "unbump": "npx projen unbump",
27
+ "upgrade": "npx projen upgrade",
28
+ "watch": "npx projen watch",
29
+ "projen": "npx projen"
30
+ },
31
+ "author": {
32
+ "name": "Kuu Miyazaki",
33
+ "email": "miyazaqui@gmail.com",
34
+ "organization": false
35
+ },
36
+ "devDependencies": {
37
+ "@stylistic/eslint-plugin": "^2",
38
+ "@types/jest": "^30.0.0",
39
+ "@types/node": "^24.0.7",
40
+ "@typescript-eslint/eslint-plugin": "^8",
41
+ "@typescript-eslint/parser": "^8",
42
+ "commit-and-tag-version": "^12",
43
+ "eslint": "^9",
44
+ "eslint-import-resolver-typescript": "^4.4.4",
45
+ "eslint-plugin-import": "^2.32.0",
46
+ "jest": "^30.0.3",
47
+ "jest-junit": "^16",
48
+ "jsii": "~5.8.9",
49
+ "jsii-diff": "^1.112.0",
50
+ "jsii-docgen": "^10.5.0",
51
+ "jsii-pacmak": "^1.112.0",
52
+ "jsii-rosetta": "~5.8.9",
53
+ "projen": "^0.94.0",
54
+ "ts-jest": "^29.4.0",
55
+ "ts-node": "^10.9.2",
56
+ "typescript": "^5.8.3"
57
+ },
58
+ "peerDependencies": {
59
+ "aws-cdk-lib": "^2.202.0",
60
+ "constructs": "^10.0.5"
61
+ },
62
+ "dependencies": {
63
+ "aws-cdk-lib": "^2.202.0",
64
+ "awscdk-construct-medialive-channel": "^0.0.6",
65
+ "constructs": "^10.4.2"
66
+ },
67
+ "keywords": [
68
+ "MediaConnect",
69
+ "cdk",
70
+ "cdk-construct"
71
+ ],
72
+ "main": "lib/index.js",
73
+ "license": "MIT",
74
+ "publishConfig": {
75
+ "access": "public"
76
+ },
77
+ "version": "0.0.0",
78
+ "jest": {
79
+ "coverageProvider": "v8",
80
+ "testMatch": [
81
+ "<rootDir>/@(src|test)/**/*(*.)@(spec|test).ts?(x)",
82
+ "<rootDir>/@(src|test)/**/__tests__/**/*.ts?(x)",
83
+ "<rootDir>/@(projenrc)/**/*(*.)@(spec|test).ts?(x)",
84
+ "<rootDir>/@(projenrc)/**/__tests__/**/*.ts?(x)"
85
+ ],
86
+ "clearMocks": true,
87
+ "collectCoverage": true,
88
+ "coverageReporters": [
89
+ "json",
90
+ "lcov",
91
+ "clover",
92
+ "cobertura",
93
+ "text"
94
+ ],
95
+ "coverageDirectory": "coverage",
96
+ "coveragePathIgnorePatterns": [
97
+ "/node_modules/"
98
+ ],
99
+ "testPathIgnorePatterns": [
100
+ "/node_modules/"
101
+ ],
102
+ "watchPathIgnorePatterns": [
103
+ "/node_modules/"
104
+ ],
105
+ "reporters": [
106
+ "default",
107
+ [
108
+ "jest-junit",
109
+ {
110
+ "outputDirectory": "test-reports"
111
+ }
112
+ ]
113
+ ],
114
+ "transform": {
115
+ "^.+\\.[t]sx?$": [
116
+ "ts-jest",
117
+ {
118
+ "tsconfig": "tsconfig.dev.json"
119
+ }
120
+ ]
121
+ }
122
+ },
123
+ "types": "lib/index.d.ts",
124
+ "stability": "stable",
125
+ "jsii": {
126
+ "outdir": "dist",
127
+ "targets": {},
128
+ "tsc": {
129
+ "outDir": "lib",
130
+ "rootDir": "src"
131
+ }
132
+ },
133
+ "//": "~~ Generated by projen. To modify, edit .projenrc.ts and run \"npx projen\"."
134
+ }