| 
									
										
										
										
											2021-01-31 19:22:28 +00:00
										 |  |  | /* eslint-disable no-plusplus */ | 
					
						
							|  |  |  | /* eslint-disable no-await-in-loop */ | 
					
						
							|  |  |  | import * as SDK from 'aws-sdk'; | 
					
						
							| 
									
										
										
										
											2021-02-07 00:45:33 +00:00
										 |  |  | import { nanoid } from 'nanoid'; | 
					
						
							| 
									
										
										
										
											2021-01-31 20:03:11 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-31 19:22:28 +00:00
										 |  |  | const fs = require('fs'); | 
					
						
							| 
									
										
										
										
											2021-02-07 20:25:23 +00:00
										 |  |  | const zlib = require('zlib'); | 
					
						
							| 
									
										
										
										
											2021-01-31 19:22:28 +00:00
										 |  |  | const core = require('@actions/core'); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class AWS { | 
					
						
							|  |  |  |   static async runBuildJob(buildParameters, baseImage) { | 
					
						
							| 
									
										
										
										
											2021-01-31 22:11:13 +00:00
										 |  |  |     await this.run( | 
					
						
							|  |  |  |       buildParameters.awsStackName, | 
					
						
							|  |  |  |       'alpine/git', | 
					
						
							| 
									
										
										
										
											2021-02-05 23:33:28 +00:00
										 |  |  |       ['clone', `https://github.com/${process.env.GITHUB_REPOSITORY}.git`, `repo`], | 
					
						
							| 
									
										
										
										
											2021-01-31 22:11:13 +00:00
										 |  |  |       [ | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |           name: 'GITHUB_SHA', | 
					
						
							|  |  |  |           value: process.env.GITHUB_SHA, | 
					
						
							|  |  |  |         }, | 
					
						
							|  |  |  |       ], | 
					
						
							|  |  |  |     ); | 
					
						
							|  |  |  |     await this.run( | 
					
						
							|  |  |  |       buildParameters.awsStackName, | 
					
						
							|  |  |  |       baseImage.toString(), | 
					
						
							|  |  |  |       ['bin/bash', '-c', 'echo "test"'], | 
					
						
							| 
									
										
										
										
											2021-02-07 00:37:34 +00:00
										 |  |  |       [], | 
					
						
							| 
									
										
										
										
											2021-01-31 22:11:13 +00:00
										 |  |  |     ); | 
					
						
							| 
									
										
										
										
											2021-01-31 19:22:28 +00:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-31 22:11:13 +00:00
										 |  |  |   static async run(stackName, image, commands, environment) { | 
					
						
							| 
									
										
										
										
											2021-01-31 19:22:28 +00:00
										 |  |  |     const ECS = new SDK.ECS(); | 
					
						
							|  |  |  |     const CF = new SDK.CloudFormation(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-02-07 02:14:58 +00:00
										 |  |  |     const taskDefStackName = `${stackName}-taskDef-${image}-${nanoid()}` | 
					
						
							|  |  |  |       .toString() | 
					
						
							|  |  |  |       .replace(/[^\da-z]/gi, ''); | 
					
						
							| 
									
										
										
										
											2021-02-07 16:33:47 +00:00
										 |  |  |     core.info('Creating build job resources'); | 
					
						
							| 
									
										
										
										
											2021-02-07 02:08:17 +00:00
										 |  |  |     const taskDefCloudFormation = fs.readFileSync(`${__dirname}/task-def-formation.yml`, 'utf8'); | 
					
						
							|  |  |  |     await CF.createStack({ | 
					
						
							|  |  |  |       StackName: taskDefStackName, | 
					
						
							|  |  |  |       TemplateBody: taskDefCloudFormation, | 
					
						
							|  |  |  |       Parameters: [ | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |           ParameterKey: 'ImageUrl', | 
					
						
							|  |  |  |           ParameterValue: image, | 
					
						
							|  |  |  |         }, | 
					
						
							| 
									
										
										
										
											2021-02-07 14:33:42 +00:00
										 |  |  |         { | 
					
						
							|  |  |  |           ParameterKey: 'ServiceName', | 
					
						
							|  |  |  |           ParameterValue: taskDefStackName, | 
					
						
							|  |  |  |         }, | 
					
						
							| 
									
										
										
										
											2021-02-07 02:08:17 +00:00
										 |  |  |       ], | 
					
						
							|  |  |  |     }).promise(); | 
					
						
							|  |  |  |     await CF.waitFor('stackCreateComplete', { StackName: taskDefStackName }).promise(); | 
					
						
							| 
									
										
										
										
											2021-01-31 19:22:28 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     const taskDefResources = await CF.describeStackResources({ | 
					
						
							|  |  |  |       StackName: taskDefStackName, | 
					
						
							|  |  |  |     }).promise(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const baseResources = await CF.describeStackResources({ StackName: stackName }).promise(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const clusterName = baseResources.StackResources.find( | 
					
						
							|  |  |  |       (x) => x.LogicalResourceId === 'ECSCluster', | 
					
						
							|  |  |  |     ).PhysicalResourceId; | 
					
						
							|  |  |  |     const task = await ECS.runTask({ | 
					
						
							|  |  |  |       cluster: clusterName, | 
					
						
							|  |  |  |       taskDefinition: taskDefResources.StackResources.find( | 
					
						
							|  |  |  |         (x) => x.LogicalResourceId === 'TaskDefinition', | 
					
						
							|  |  |  |       ).PhysicalResourceId, | 
					
						
							|  |  |  |       platformVersion: '1.4.0', | 
					
						
							|  |  |  |       launchType: 'FARGATE', | 
					
						
							|  |  |  |       overrides: { | 
					
						
							|  |  |  |         containerOverrides: [ | 
					
						
							|  |  |  |           { | 
					
						
							| 
									
										
										
										
											2021-02-07 14:37:59 +00:00
										 |  |  |             name: taskDefStackName, | 
					
						
							| 
									
										
										
										
											2021-01-31 22:11:13 +00:00
										 |  |  |             environment, | 
					
						
							| 
									
										
										
										
											2021-02-07 00:58:11 +00:00
										 |  |  |             command: commands, | 
					
						
							| 
									
										
										
										
											2021-01-31 19:22:28 +00:00
										 |  |  |           }, | 
					
						
							|  |  |  |         ], | 
					
						
							|  |  |  |       }, | 
					
						
							|  |  |  |       networkConfiguration: { | 
					
						
							|  |  |  |         awsvpcConfiguration: { | 
					
						
							|  |  |  |           subnets: [ | 
					
						
							|  |  |  |             baseResources.StackResources.find((x) => x.LogicalResourceId === 'PublicSubnetOne') | 
					
						
							|  |  |  |               .PhysicalResourceId, | 
					
						
							|  |  |  |             baseResources.StackResources.find((x) => x.LogicalResourceId === 'PublicSubnetTwo') | 
					
						
							|  |  |  |               .PhysicalResourceId, | 
					
						
							|  |  |  |           ], | 
					
						
							|  |  |  |           assignPublicIp: 'ENABLED', | 
					
						
							|  |  |  |           securityGroups: [ | 
					
						
							|  |  |  |             baseResources.StackResources.find( | 
					
						
							|  |  |  |               (x) => x.LogicalResourceId === 'ContainerSecurityGroup', | 
					
						
							|  |  |  |             ).PhysicalResourceId, | 
					
						
							|  |  |  |           ], | 
					
						
							|  |  |  |         }, | 
					
						
							|  |  |  |       }, | 
					
						
							|  |  |  |     }).promise(); | 
					
						
							| 
									
										
										
										
											2021-02-07 18:10:30 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-02-07 18:00:57 +00:00
										 |  |  |     core.info('Build job is starting'); | 
					
						
							| 
									
										
										
										
											2021-02-07 18:10:30 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-31 19:22:28 +00:00
										 |  |  |     await ECS.waitFor('tasksRunning', { | 
					
						
							|  |  |  |       cluster: clusterName, | 
					
						
							| 
									
										
										
										
											2021-01-31 20:43:20 +00:00
										 |  |  |       tasks: [task.tasks[0].taskArn], | 
					
						
							| 
									
										
										
										
											2021-01-31 19:22:28 +00:00
										 |  |  |     }).promise(); | 
					
						
							| 
									
										
										
										
											2021-02-07 18:10:30 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-02-07 18:00:57 +00:00
										 |  |  |     core.info(`Build job is running`); | 
					
						
							| 
									
										
										
										
											2021-01-31 19:22:28 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     // watching logs
 | 
					
						
							| 
									
										
										
										
											2021-02-07 15:42:15 +00:00
										 |  |  |     const kinesis = new SDK.Kinesis(); | 
					
						
							| 
									
										
										
										
											2021-02-07 15:27:24 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-02-07 15:48:22 +00:00
										 |  |  |     const getTaskStatus = async () => { | 
					
						
							|  |  |  |       const tasks = await ECS.describeTasks({ | 
					
						
							|  |  |  |         cluster: clusterName, | 
					
						
							|  |  |  |         tasks: [task.tasks[0].taskArn], | 
					
						
							|  |  |  |       }).promise(); | 
					
						
							|  |  |  |       return tasks.tasks[0].lastStatus; | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-02-07 15:53:59 +00:00
										 |  |  |     const stream = await kinesis | 
					
						
							|  |  |  |       .describeStream({ | 
					
						
							|  |  |  |         StreamName: taskDefResources.StackResources.find( | 
					
						
							|  |  |  |           (x) => x.LogicalResourceId === 'KinesisStream', | 
					
						
							|  |  |  |         ).PhysicalResourceId, | 
					
						
							|  |  |  |       }) | 
					
						
							|  |  |  |       .promise(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-02-07 15:59:57 +00:00
										 |  |  |     let iterator = ( | 
					
						
							|  |  |  |       await kinesis | 
					
						
							| 
									
										
										
										
											2021-02-07 15:48:22 +00:00
										 |  |  |         .getShardIterator({ | 
					
						
							|  |  |  |           ShardIteratorType: 'TRIM_HORIZON', | 
					
						
							| 
									
										
										
										
											2021-02-07 15:53:59 +00:00
										 |  |  |           StreamName: stream.StreamDescription.StreamName, | 
					
						
							|  |  |  |           ShardId: stream.StreamDescription.Shards[0].ShardId, | 
					
						
							| 
									
										
										
										
											2021-02-07 15:48:22 +00:00
										 |  |  |         }) | 
					
						
							| 
									
										
										
										
											2021-02-07 15:59:57 +00:00
										 |  |  |         .promise() | 
					
						
							|  |  |  |     ).ShardIterator; | 
					
						
							| 
									
										
										
										
											2021-02-07 16:36:53 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     core.info(`Task status is ${await getTaskStatus()}`); | 
					
						
							| 
									
										
										
										
											2021-02-07 15:59:57 +00:00
										 |  |  |     while ((await getTaskStatus()) === 'RUNNING') { | 
					
						
							| 
									
										
										
										
											2021-02-07 15:48:22 +00:00
										 |  |  |       const records = await kinesis | 
					
						
							|  |  |  |         .getRecords({ | 
					
						
							| 
									
										
										
										
											2021-02-07 16:03:25 +00:00
										 |  |  |           ShardIterator: iterator, | 
					
						
							| 
									
										
										
										
											2021-02-07 15:48:22 +00:00
										 |  |  |         }) | 
					
						
							|  |  |  |         .promise(); | 
					
						
							| 
									
										
										
										
											2021-02-07 15:59:57 +00:00
										 |  |  |       iterator = records.NextShardIterator; | 
					
						
							| 
									
										
										
										
											2021-02-07 16:56:12 +00:00
										 |  |  |       if (records.Records.length > 0) { | 
					
						
							| 
									
										
										
										
											2021-02-07 17:18:01 +00:00
										 |  |  |         for (let index = 0; index < records.Records.length; index++) { | 
					
						
							| 
									
										
										
										
											2021-02-07 20:25:23 +00:00
										 |  |  |           const result = await new Promise((resolve) => { | 
					
						
							|  |  |  |             zlib.gunzip( | 
					
						
							|  |  |  |               Buffer.from(records.Records[index].Data, 'base64').toString(), | 
					
						
							|  |  |  |               (error, results) => { | 
					
						
							|  |  |  |                 resolve(results); | 
					
						
							|  |  |  |               }, | 
					
						
							|  |  |  |             ); | 
					
						
							|  |  |  |           }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           core.info(result); | 
					
						
							| 
									
										
										
										
											2021-02-07 17:18:01 +00:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2021-02-07 16:24:06 +00:00
										 |  |  |       } | 
					
						
							| 
									
										
										
										
											2021-02-07 18:57:06 +00:00
										 |  |  |       await new Promise((resolve) => setTimeout(resolve, 3000)); | 
					
						
							| 
									
										
										
										
											2021-02-07 15:48:22 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-01-31 21:10:50 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-31 19:22:28 +00:00
										 |  |  |     await ECS.waitFor('tasksStopped', { | 
					
						
							|  |  |  |       cluster: clusterName, | 
					
						
							| 
									
										
										
										
											2021-01-31 20:43:20 +00:00
										 |  |  |       tasks: [task.tasks[0].taskArn], | 
					
						
							| 
									
										
										
										
											2021-01-31 19:22:28 +00:00
										 |  |  |     }).promise(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-02-07 17:53:37 +00:00
										 |  |  |     core.info('Build job has ended'); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-02-07 17:47:51 +00:00
										 |  |  |     await CF.deleteStack({ | 
					
						
							|  |  |  |       StackName: taskDefStackName, | 
					
						
							|  |  |  |     }).promise(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-02-07 17:53:37 +00:00
										 |  |  |     core.info('Cleanup complete'); | 
					
						
							| 
									
										
										
										
											2021-01-31 19:22:28 +00:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2021-02-05 23:14:45 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-02-05 23:24:01 +00:00
										 |  |  |   static onlog(batch) { | 
					
						
							|  |  |  |     batch.forEach((log) => { | 
					
						
							|  |  |  |       core.info(`log: ${log}`); | 
					
						
							|  |  |  |     }); | 
					
						
							| 
									
										
										
										
											2021-02-05 23:14:45 +00:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2021-01-31 19:22:28 +00:00
										 |  |  | } | 
					
						
							|  |  |  | export default AWS; |