This guide is to enable you to use CodePipeline to deploy CloudFrormation Stack (EC2 instance in a predefined VPC).
Step 1: Create CloudFormation Template
The following CloudFormation template will let you create a t2.micro EC2 instance in a public subnet inside a pre-created VPC.
- Save the following yaml configuration as a template DemoCloudFormationTemplate.yaml
- update VPC Id with your VPC ID (Go to VPC console and copy the VPC ID or create a new VPC )
- update Subnet Id with your Subnet ID (Go to VPC console, select Subnets from left panel and copy the Public Subnet Id or create a new Public Subnet in the VPC to use)
- Edit the SSHKeyname with your Key Pair. - Zip the file as DemoCloudFormation.zip (CodePipeline only takes the CloudFormation in .zip format from S3)
- Upload the zip file to the S3 bucket of your choice for example S3://DemoCloudFormation/DemoCloudFormation.zip
Make sure versioning is turned on for the Bucket.
AWSTemplateFormatVersion: '2010-09-09'
Description: Template to Create an EC2 instance in a VPC in us-east-1a region
Parameters:
ImageId:
Type: String
Description: 'Amazon Linux 2 AMI for us-east-1a region'
Default: 'ami-0742b4e673072066f'
VpcId:
Type: String
Description: VPC id
Default: vpc-xxxxxxxxxxxxxxx
PublicSubnetId:
Type: String
Description: Subnet in which to launch an EC2
Default: subnet-xxxxxxxxxxxxxx
AvailabilityZone:
Type: String
Description: Availability Zone into which instance will launch
Default: us-east-1a
InstanceType:
Type: String
Description: Choosing t2 micro to start with free tier
Default: t2.micro
SSHKeyName:
Description: SSH Keypair to login to the instance
Type: AWS::EC2::KeyPair::KeyName
Default: Demo-EC2-KP
Resources:
DemoInstance:
Type: 'AWS::EC2::Instance'
Properties:
ImageId: !Ref ImageId
InstanceType: !Ref InstanceType
AvailabilityZone: !Ref AvailabilityZone
KeyName: !Ref SSHKeyName
NetworkInterfaces:
- DeviceIndex: '0'
AssociatePublicIpAddress: true
DeleteOnTermination: true
SubnetId: !Ref PublicSubnetId
GroupSet:
- !Ref DemoSecurityGroup
DemoSecurityGroup:
Type: 'AWS::EC2::SecurityGroup'
Properties:
VpcId: !Ref VpcId
GroupDescription: SG to allow SSH access via port 22
SecurityGroupIngress:
- IpProtocol: tcp
FromPort: 22
ToPort: 22
CidrIp: '0.0.0.0/0'
Tags:
- Key: Name
Value: DemoStack
Outputs:
DemoInstanceId:
Description: Instance Id
Value: !Ref DemoInstance
Step 2: Create Service role for CodePipeline
Next step is to create a service role for the CodePipeline to access the S3 buck, deploy CloudFormation, Some OpsWork, Lambda Invocation etc.
- Go to IAM / Roles and click on create role.
- Here it gets tricky as there is no CodePipeline so select EC2 and we will change it to CodePipeline later.
data:image/s3,"s3://crabby-images/96f56/96f56d4021c5a71a9797ebea79b798064d3fe114" alt=""
- Click Next: Permissions. Do not add any permissions yet and click Next: Tags and create any tag if required
- Click Next: Review, Give a name to the role for eg. DemoCodePipeline-ServiceRole. Change the description to describe the role. Click on Create Role
data:image/s3,"s3://crabby-images/c57fe/c57fec25ddb23f948779f101d0f610ec5ba71feb" alt=""
- Now go back to Roles and search for newly created Role "DemoCodePipeline-ServiceRole". Click on it to edit the role.
- Add the following as an inline policy and save it.
{
"Statement": [
{
"Action": [
"s3:GetObject",
"s3:GetObjectVersion",
"s3:GetBucketVersioning",
"s3:PutObject"
],
"Resource": "arn:aws:s3:::codepipeline*",
"Effect": "Allow"
},
{
"Action": [
"codedeploy:CreateDeployment",
"codedeploy:GetApplicationRevision",
"codedeploy:GetDeployment",
"codedeploy:GetDeploymentConfig",
"codedeploy:RegisterApplicationRevision"
],
"Resource": "*",
"Effect": "Allow"
},
{
"Action": [
"ec2:*",
"cloudwatch:*",
"s3:*",
"sns:*",
"cloudformation:*",
"sqs:*",
"sqs:*",
"ecs:*",
"iam:PassRole"
],
"Resource": "*",
"Effect": "Allow"
},
{
"Action": [
"lambda:InvokeFunction",
"lambda:ListFunctions"
],
"Resource": "*",
"Effect": "Allow"
},
{
"Action": [
"cloudformation:CreateStack",
"cloudformation:DeleteStack",
"cloudformation:DescribeStacks",
"cloudformation:UpdateStack",
"cloudformation:CreateChangeSet",
"cloudformation:DeleteChangeSet",
"cloudformation:DescribeChangeSet",
"cloudformation:ExecuteChangeSet",
"cloudformation:SetStackPolicy",
"cloudformation:ValidateTemplate",
"iam:PassRole"
],
"Resource": "*",
"Effect": "Allow"
},
{
"Action": [
"codebuild:BatchGetBuilds",
"codebuild:StartBuild"
],
"Resource": "*",
"Effect": "Allow"
}
],
"Version": "2012-10-17"
}
- Go to Trusted Relationship tab and click on edit trust relationship to change EC2 to codepipeline. You are done with creating a custom CodePipeline Service Role.
data:image/s3,"s3://crabby-images/e3db5/e3db552cfa9f1da09c4e0df35324f7db105ed5c0" alt=""
data:image/s3,"s3://crabby-images/3f4bd/3f4bd44e5d38006a7c06532b0f2d3f20ee846181" alt=""
Step 3: Create Service Role for CloudFormation
Next step is to create a service role for the CloudFormationto to access the S3 bucket and deploy the EC2 instance in a VPC.
- Go to IAM / Roles and click on create role.
- Select CloudFormation and click Next: Permissions
- check AmazonEC2FullAccess and S3FullAccess. This is overly permissive so change it according to your restrictions
- Click Next:Tags, create tags if required, click Next:Review and give the Role a name "DemoCloudFormation_ServiceRole". and click Create Role. Now you are done with all the leg work.
data:image/s3,"s3://crabby-images/ce72f/ce72fb9b4e12d591311f4c43db035615315ed708" alt=""
Step 4: Create CodePipeline to deploy CloudFormation Stack
- Go to Codepipline Console and click on Create Pipeline
- Give the pipeline a name. Select Existing Service Role. From the dropdown Role ARN select the newly created custom code pipeline role "DemoCodePipeline-ServiceRole" and click Next
data:image/s3,"s3://crabby-images/0f5ca/0f5ca996f7e3e8157db8e25950fd10776d633ee2" alt=""
- In source provider dropdown, select Amazon S3
- Select the bucket name as defined in Step 1 - "DemoCloudFormation"
- For S3 object key type the .zip file name as described in Step 1 "DemoCloudFormation.zip". Click Next
- Skip the next build stage as we are not going to build any code
data:image/s3,"s3://crabby-images/ea70c/ea70c7a6b2d7344f1234480ba8b219b93398c8e4" alt=""
- In Deploy Provider drop down menu select AWS CloudFormation
- Action Mode - Create or Update a stack
- Since we are creating a new stack give it a name for eg. "DemoCloudFormationStack"
- Select SourceArtifact as Artifact name
- IMPORTANT: Write File Name as the template name which you have created in step 1 "DemoCloudFormationTemplate.yaml"
- Add role name as the role you have defined in Step 3 "DemoCloudFormation_ServiceRole"
data:image/s3,"s3://crabby-images/0d5e0/0d5e0f84a6c54ca5f00ded9572048b966806c800" alt=""
- Click Next and create pipeline
- Voila you are done. This will kick off two stage code pipeline.
Stage 1: Source - where it reads the CloudFormation Template from S3
Stage 2: Deploy: where it deploys the EC2 instance as defined in the CloudFormation template.
data:image/s3,"s3://crabby-images/088a1/088a1310c6a52bad411ac97c779cc70b82ca2e7f" alt=""
- Go to CloudFormation console and go to Events to check the status.
data:image/s3,"s3://crabby-images/f1758/f17587e4fbd80a098c96d2c746a43ab2b629568b" alt=""
data:image/s3,"s3://crabby-images/f81c7/f81c76aef49d86d74b0b436ea2e02e805589687f" alt=""
- Go to EC2 console to verify if the t2.micro EC2 instance is created
data:image/s3,"s3://crabby-images/d2e3d/d2e3debd30ca82fc0e81f0f03e9df861b32dbc50" alt=""