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.
- package/.jsii +4351 -0
- package/API.md +343 -0
- package/LICENSE +19 -0
- package/README.md +75 -0
- package/lib/LiveFeedFromFile.d.ts +29 -0
- package/lib/LiveFeedFromFile.js +300 -0
- package/lib/index.d.ts +1 -0
- package/lib/index.js +18 -0
- package/package.json +134 -0
|
@@ -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
|
+
}
|