awscdk-construct-live-channel-from-mp4-file
Version:
CDK Construct for setting up a simple live channel for testing
163 lines • 24.9 kB
JavaScript
;
var _a;
Object.defineProperty(exports, "__esModule", { value: true });
exports.HarvestJobLambda = void 0;
const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
const crypto = require("crypto");
const fs = require("fs");
const path = require("path");
const aws_cdk_lib_1 = require("aws-cdk-lib");
const cloudfront = require("aws-cdk-lib/aws-cloudfront");
const cloudfront_origins = require("aws-cdk-lib/aws-cloudfront-origins");
const iam = require("aws-cdk-lib/aws-iam");
const aws_lambda_1 = require("aws-cdk-lib/aws-lambda");
const aws_lambda_nodejs_1 = require("aws-cdk-lib/aws-lambda-nodejs");
const s3 = require("aws-cdk-lib/aws-s3");
const custom_resources_1 = require("aws-cdk-lib/custom-resources");
const constructs_1 = require("constructs");
class HarvestJobLambda extends constructs_1.Construct {
constructor(scope, id, props) {
super(scope, id);
const { channelId, endpointId, startTime, endTime, destination, publish = false, retain = false, } = props;
let bucket;
if (destination) {
bucket = s3.Bucket.fromBucketName(this, 'Bucket', destination.bucketName);
this.destination = destination;
}
else {
// Create S3 bucket
const bucketName = `${crypto.randomUUID()}`;
bucket = new s3.Bucket(this, bucketName, {
bucketName,
removalPolicy: retain ? aws_cdk_lib_1.RemovalPolicy.RETAIN : aws_cdk_lib_1.RemovalPolicy.DESTROY,
autoDeleteObjects: retain ? false : true,
});
bucket.addToResourcePolicy(new iam.PolicyStatement({
effect: iam.Effect.ALLOW,
actions: ['s3:*'],
resources: [bucket.bucketArn, `${bucket.bucketArn}/*`],
principals: [new iam.ServicePrincipal('mediapackage.amazonaws.com')],
}));
this.destination = {
bucketName,
manifestKey: 'index.m3u8',
};
}
//Create an IAM Role for MediaPackage to access S3
const role = new iam.Role(this, 'IamRoleForMediaPackage', {
inlinePolicies: {
policy: new iam.PolicyDocument({
statements: [
new iam.PolicyStatement({
effect: iam.Effect.ALLOW,
actions: ['s3:*'],
resources: [bucket.bucketArn, `${bucket.bucketArn}/*`],
}),
],
}),
},
assumedBy: new iam.ServicePrincipal('mediapackage.amazonaws.com'),
});
const roleArn = role.roleArn;
const TS_ENTRY = path.resolve(__dirname, 'code', 'index.ts');
const JS_ENTRY = path.resolve(__dirname, 'code', 'index.js');
this.func = new aws_lambda_nodejs_1.NodejsFunction(scope, 'HarvestJobFunction', {
runtime: aws_lambda_1.Runtime.NODEJS_18_X,
entry: fs.existsSync(TS_ENTRY) ? TS_ENTRY : JS_ENTRY,
// projectRoot: path.resolve(__dirname, '..'),
// depsLockFilePath: path.resolve(__dirname, '..', 'yarn.lock'),
handler: 'handler',
timeout: aws_cdk_lib_1.Duration.seconds(30),
environment: {
NODE_ENV: process.env.NODE_ENV,
REGION: process.env.CDK_DEFAULT_REGION,
CHANNEL_ID: channelId,
ORIGIN_ENDPOINT_ID: endpointId,
START_TIME: startTime.toISOString(),
END_TIME: endTime ? endTime.toISOString() : '',
S3_BUCKET_NAME: this.destination.bucketName,
MANIFEST_KEY: this.destination.manifestKey,
S3_ROLE_ARN: roleArn,
},
logRetention: aws_cdk_lib_1.aws_logs.RetentionDays.TWO_WEEKS,
});
// Add a statement to call MediaLive schedule API
this.func.addToRolePolicy(iam.PolicyStatement.fromJson({
Effect: 'Allow',
Action: 'medialive:*',
Resource: '*',
}));
// Add a statement to call MediaPackage harvest job API
this.func.addToRolePolicy(iam.PolicyStatement.fromJson({
Effect: 'Allow',
Action: 'mediapackage:*',
Resource: '*',
}));
// Add a statement to pass the IAM role to MediaPackage
this.func.addToRolePolicy(iam.PolicyStatement.fromJson({
Effect: 'Allow',
Action: 'iam:PassRole',
Resource: roleArn,
}));
if (publish) {
// Create an Origin Access Identity (OAI)
const oai = new cloudfront.OriginAccessIdentity(this, 'OAI', {
comment: 'OAI for CloudFront to access private S3 bucket',
});
// Create a CloudFront distribution
const distribution = new cloudfront.Distribution(this, 'Distribution', {
defaultBehavior: {
origin: new cloudfront_origins.S3Origin(bucket, { originAccessIdentity: oai }),
viewerProtocolPolicy: cloudfront.ViewerProtocolPolicy.ALLOW_ALL,
allowedMethods: cloudfront.AllowedMethods.ALLOW_GET_HEAD_OPTIONS,
cachedMethods: cloudfront.CachedMethods.CACHE_GET_HEAD_OPTIONS,
cachePolicy: cloudfront.CachePolicy.ELEMENTAL_MEDIA_PACKAGE,
},
enabled: true,
});
// Grant the OAI access to the private S3 bucket
const statement = new iam.PolicyStatement({
effect: iam.Effect.ALLOW,
actions: ['s3:GetObject'],
resources: [`${bucket.bucketArn}/*`],
principals: [new iam.CanonicalUserPrincipal(oai.cloudFrontOriginAccessIdentityS3CanonicalUserId)],
});
bucket.addToResourcePolicy(statement);
if (retain) {
oai.applyRemovalPolicy(aws_cdk_lib_1.RemovalPolicy.RETAIN);
distribution.applyRemovalPolicy(aws_cdk_lib_1.RemovalPolicy.RETAIN);
// Need to manually retain the resource policy due to the known issue:
// https://github.com/aws/aws-cdk/issues/27125
const documentString = JSON.stringify(new iam.PolicyDocument({
assignSids: true,
statements: [statement],
}));
new aws_cdk_lib_1.CfnOutput(this, 'PutBucketPolicyCLI', {
value: `aws s3api put-bucket-policy --bucket ${this.destination.bucketName} --policy '${documentString}'`,
exportName: `${aws_cdk_lib_1.Aws.STACK_NAME}-PutBucketPolicyCLI`,
description: 'Run this command when cdk destroy shows error.',
});
new custom_resources_1.AwsCustomResource(this, 'PutBucketPolicy', {
onDelete: {
service: 'S3',
action: 'PutBucketPolicy',
parameters: {
Bucket: this.destination.bucketName,
PolicyDocument: documentString,
},
physicalResourceId: custom_resources_1.PhysicalResourceId.of(`${crypto.randomUUID()}`),
},
//Will ignore any resource and use the assumedRoleArn as resource and 'sts:AssumeRole' for service:action
policy: custom_resources_1.AwsCustomResourcePolicy.fromSdkCalls({
resources: custom_resources_1.AwsCustomResourcePolicy.ANY_RESOURCE,
}),
});
}
this.publishedUrl = `https://${distribution.distributionDomainName}/${this.destination.manifestKey}`;
}
}
}
exports.HarvestJobLambda = HarvestJobLambda;
_a = JSII_RTTI_SYMBOL_1;
HarvestJobLambda[_a] = { fqn: "awscdk-construct-live-channel-from-mp4-file.HarvestJobLambda", version: "1.2.27" };
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"HarvestJobLambda.js","sourceRoot":"","sources":["../src/HarvestJobLambda.ts"],"names":[],"mappings":"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","sourcesContent":["import * as crypto from 'crypto';\nimport * as fs from 'fs';\nimport * as path from 'path';\nimport { Duration, RemovalPolicy, Aws, CfnOutput, aws_logs as logs } from 'aws-cdk-lib';\nimport * as cloudfront from 'aws-cdk-lib/aws-cloudfront';\nimport * as cloudfront_origins from 'aws-cdk-lib/aws-cloudfront-origins';\nimport * as iam from 'aws-cdk-lib/aws-iam';\nimport { Runtime } from 'aws-cdk-lib/aws-lambda';\nimport { NodejsFunction } from 'aws-cdk-lib/aws-lambda-nodejs';\nimport * as s3 from 'aws-cdk-lib/aws-s3';\nimport { AwsCustomResource, AwsCustomResourcePolicy, PhysicalResourceId } from 'aws-cdk-lib/custom-resources';\nimport { Construct } from 'constructs';\n\nexport interface HarvestJobLambdaProps {\n  readonly channelId: string;\n  readonly endpointId: string;\n  readonly startTime: Date;\n  readonly endTime?: Date;\n  readonly destination?: HarvestJobDestinationProps;\n  readonly publish?: boolean;\n  readonly retain?: boolean;\n}\n\nexport interface HarvestJobDestinationProps {\n  readonly bucketName: string; // The name of the S3 bucket.\n  readonly manifestKey: string; // The S3 bucket key of the manifest file.\n}\n\nexport class HarvestJobLambda extends Construct {\n  public readonly func: NodejsFunction;\n  public readonly destination: HarvestJobDestinationProps;\n  public readonly publishedUrl?: string;\n\n  constructor(scope: Construct, id: string, props: HarvestJobLambdaProps) {\n    super(scope, id);\n\n    const {\n      channelId,\n      endpointId,\n      startTime,\n      endTime,\n      destination,\n      publish = false,\n      retain = false,\n    } = props;\n\n    let bucket: s3.IBucket;\n    if (destination) {\n      bucket = s3.Bucket.fromBucketName(this, 'Bucket', destination.bucketName);\n      this.destination = destination;\n    } else {\n      // Create S3 bucket\n      const bucketName = `${crypto.randomUUID()}`;\n      bucket = new s3.Bucket(this, bucketName, {\n        bucketName,\n        removalPolicy: retain ? RemovalPolicy.RETAIN : RemovalPolicy.DESTROY,\n        autoDeleteObjects: retain ? false : true,\n      });\n      bucket.addToResourcePolicy(new iam.PolicyStatement({\n        effect: iam.Effect.ALLOW,\n        actions: ['s3:*'],\n        resources: [bucket.bucketArn, `${bucket.bucketArn}/*`],\n        principals: [new iam.ServicePrincipal('mediapackage.amazonaws.com')],\n      }));\n      this.destination = {\n        bucketName,\n        manifestKey: 'index.m3u8',\n      };\n    }\n    //Create an IAM Role for MediaPackage to access S3\n    const role = new iam.Role(this, 'IamRoleForMediaPackage', {\n      inlinePolicies: {\n        policy: new iam.PolicyDocument({\n          statements: [\n            new iam.PolicyStatement({\n              effect: iam.Effect.ALLOW,\n              actions: ['s3:*'],\n              resources: [bucket.bucketArn, `${bucket.bucketArn}/*`],\n            }),\n          ],\n        }),\n      },\n      assumedBy: new iam.ServicePrincipal('mediapackage.amazonaws.com'),\n    });\n    const roleArn = role.roleArn;\n\n    const TS_ENTRY = path.resolve(__dirname, 'code', 'index.ts');\n    const JS_ENTRY = path.resolve(__dirname, 'code', 'index.js');\n\n    this.func = new NodejsFunction(scope, 'HarvestJobFunction', {\n      runtime: Runtime.NODEJS_18_X,\n      entry: fs.existsSync(TS_ENTRY) ? TS_ENTRY : JS_ENTRY,\n      // projectRoot: path.resolve(__dirname, '..'),\n      // depsLockFilePath: path.resolve(__dirname, '..', 'yarn.lock'),\n      handler: 'handler',\n      timeout: Duration.seconds(30),\n      environment: {\n        NODE_ENV: process.env.NODE_ENV as string,\n        REGION: process.env.CDK_DEFAULT_REGION as string,\n        CHANNEL_ID: channelId,\n        ORIGIN_ENDPOINT_ID: endpointId,\n        START_TIME: startTime.toISOString(),\n        END_TIME: endTime ? endTime.toISOString() : '',\n        S3_BUCKET_NAME: this.destination.bucketName,\n        MANIFEST_KEY: this.destination.manifestKey,\n        S3_ROLE_ARN: roleArn,\n      },\n      logRetention: logs.RetentionDays.TWO_WEEKS,\n    });\n    // Add a statement to call MediaLive schedule API\n    this.func.addToRolePolicy(\n      iam.PolicyStatement.fromJson({\n        Effect: 'Allow',\n        Action: 'medialive:*',\n        Resource: '*',\n      }),\n    );\n    // Add a statement to call MediaPackage harvest job API\n    this.func.addToRolePolicy(\n      iam.PolicyStatement.fromJson({\n        Effect: 'Allow',\n        Action: 'mediapackage:*',\n        Resource: '*',\n      }),\n    );\n    // Add a statement to pass the IAM role to MediaPackage\n    this.func.addToRolePolicy(\n      iam.PolicyStatement.fromJson({\n        Effect: 'Allow',\n        Action: 'iam:PassRole',\n        Resource: roleArn,\n      }),\n    );\n\n    if (publish) {\n      // Create an Origin Access Identity (OAI)\n      const oai = new cloudfront.OriginAccessIdentity(this, 'OAI', {\n        comment: 'OAI for CloudFront to access private S3 bucket',\n      });\n\n      // Create a CloudFront distribution\n      const distribution = new cloudfront.Distribution(this, 'Distribution', {\n        defaultBehavior: {\n          origin: new cloudfront_origins.S3Origin(bucket, { originAccessIdentity: oai }),\n          viewerProtocolPolicy: cloudfront.ViewerProtocolPolicy.ALLOW_ALL,\n          allowedMethods: cloudfront.AllowedMethods.ALLOW_GET_HEAD_OPTIONS,\n          cachedMethods: cloudfront.CachedMethods.CACHE_GET_HEAD_OPTIONS,\n          cachePolicy: cloudfront.CachePolicy.ELEMENTAL_MEDIA_PACKAGE,\n        },\n        enabled: true,\n      });\n\n      // Grant the OAI access to the private S3 bucket\n      const statement = new iam.PolicyStatement({\n        effect: iam.Effect.ALLOW,\n        actions: ['s3:GetObject'],\n        resources: [`${bucket.bucketArn}/*`],\n        principals: [new iam.CanonicalUserPrincipal(oai.cloudFrontOriginAccessIdentityS3CanonicalUserId)],\n      });\n      bucket.addToResourcePolicy(statement);\n\n      if (retain) {\n        oai.applyRemovalPolicy(RemovalPolicy.RETAIN);\n        distribution.applyRemovalPolicy(RemovalPolicy.RETAIN);\n        // Need to manually retain the resource policy due to the known issue:\n        // https://github.com/aws/aws-cdk/issues/27125\n        const documentString = JSON.stringify(new iam.PolicyDocument({\n          assignSids: true,\n          statements: [statement],\n        }));\n        new CfnOutput(this, 'PutBucketPolicyCLI', {\n          value: `aws s3api put-bucket-policy --bucket ${this.destination.bucketName} --policy '${documentString}'`,\n          exportName: `${Aws.STACK_NAME}-PutBucketPolicyCLI`,\n          description: 'Run this command when cdk destroy shows error.',\n        });\n        new AwsCustomResource(this, 'PutBucketPolicy', {\n          onDelete: {\n            service: 'S3',\n            action: 'PutBucketPolicy',\n            parameters: {\n              Bucket: this.destination.bucketName,\n              PolicyDocument: documentString,\n            },\n            physicalResourceId: PhysicalResourceId.of(`${crypto.randomUUID()}`),\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      }\n\n      this.publishedUrl = `https://${distribution.distributionDomainName}/${this.destination.manifestKey}`;\n    }\n  }\n}"]}