We've been using AWS Codepipeline for some time now and for the most part it's a great managed service. Easy to get started with and pretty simple to use.
That being said, it does lack some features out of the box that most CICD systems have ready for you. The one I'll be tackling today is alerting on a stage failure.
Out of the box Codepipeline won't alert you when there's a failure at a stage. Unless you go in and literally look at it in the console, you won't know that anything is broken. For example when I started working on this blog entry, I checked one of the pipelines that delivers to our test environment, and found it in a failed state.
In this case the failure is because our Opsworks stacks are set to turn off test instances when not during business hours, but for almost any other failure I would want to alert the team responsible for making the change that failed.
For a solution, we'll use these resources
That being said, it does lack some features out of the box that most CICD systems have ready for you. The one I'll be tackling today is alerting on a stage failure.
Out of the box Codepipeline won't alert you when there's a failure at a stage. Unless you go in and literally look at it in the console, you won't know that anything is broken. For example when I started working on this blog entry, I checked one of the pipelines that delivers to our test environment, and found it in a failed state.
In this case the failure is because our Opsworks stacks are set to turn off test instances when not during business hours, but for almost any other failure I would want to alert the team responsible for making the change that failed.
For a solution, we'll use these resources
- AWS Lambda
- Boto3
- AWS SNS Topics
- Cloudformation
First we'll need a Lambda function that can get a list of pipelines that are in our account, scan their stages, detect failures, and produce alerts. Below is a basic example of what we're using. I'm far from a python expert, so I understand that there are improvements that could be made for error handling.
import boto3 import logging import os def lambda_handler(event, context): # Get a cloudwatch logger logger = logging.getLogger('mvp-alert-on-cp-failure') logger.setLevel(logging.DEBUG) sns_topic_arn = os.environ['TOPIC_ARN'] # Obtain boto3 resources logger.info('Getting boto 3 resources') code_pipeline_client = boto3.client('codepipeline') sns_client = boto3.client('sns') logger.debug('Getting pipelines') for pipeline in code_pipeline_client.list_pipelines()['pipelines']: logger.debug('Checking pipeline ' + pipeline['name'] + ' for failures') for stage in code_pipeline_client.get_pipeline_state(name=pipeline['name'])['stageStates']: logger.debug('Checking stage ' + stage['stageName'] + ' for failures') if 'latestExecution' in stage and stage['latestExecution']['status'] == 'Failed': logger.debug('Stage failed! Sending SNS notification to ' + sns_topic_arn) failed_actions = '' for action in stage['actionStates']: logger.debug(action) logger.debug('Checking action ' + action['actionName'] + ' for failures') if 'latestExecution' in action and action['latestExecution']['status'] == 'Failed': logger.debug('Action failed!') failed_actions += action['actionName'] logger.debug('Publishing failure alert: ' + pipeline['name'] + '|' + stage['stageName'] + '|' + action['actionName']) logger.debug('Publishing failure alert: ' + pipeline['name'] + '|' + stage['stageName'] + '|' + failed_actions) alert_subject = 'Codepipeline failure in ' + pipeline['name'] + ' at stage ' + stage['stageName'] alert_message = 'Codepipeline failure in ' + pipeline['name'] + ' at stage ' + stage['stageName'] + '. Failed actions: ' + failed_actions logger.debug('Sending SNS notification') sns_client.publish(TopicArn=sns_topic_arn,Subject=alert_subject,Message=alert_message) return "And we're done!"
If you're looking closely, you're probably wondering what the environment variable named "TOPIC_ARN" is, which leads us to the next piece: A cloudformation template to create this lambda function.
The Cloudformation template needs to do a few things.
- Create the Lambda function. I've chosen to do this using AWS Serverless Application Model.
- Create an IAM Role for the Lambda function to execute under
- Create IAM policies that will give the IAM role read access to your pipelines, and publish access to your SNS topic
- Create an SNS topic with a list of individuals you want to get the email
The only really new fangled Cloudformation feature I'm using here is AWS SAM, the rest of these have existed for quite a while. In my opinion one of the main ideas behind AWS SAM is to package your entire Serverless Function in a single Cloudformation template, so the example below does all four of these steps.
############################################# ### Lambda function to alert on pipeline failures ############################################# LambdaAlertCPTestFail: Type: AWS::Serverless::Function Properties: Handler: mvp-alert-on-cp-failure.lambda_handler Role: !GetAtt IAMRoleAlertOnCPTestFailure.Arn Runtime: python2.7 Timeout: 300 Events: CheckEvery30Minutes: Type: Schedule Properties: Schedule: cron(0/30 12-23 ? * MON-FRI *) Environment: Variables: STAGE_NAME: Test TOPIC_ARN: !Ref CodePipelineTestStageFailureTopic CodePipelineTestStageFailureTopic: Type: "AWS::SNS::Topic" Properties: DisplayName: MvpPipelineFailure Subscription: - Endpoint: 'pipelineCurator@example.com' Protocol: 'email' TopicName: MvpPipelineFailure IAMPolicyPublishToTestFailureTopic: Type: "AWS::IAM::Policy" DependsOn: MoveToPHIIAMRole Properties: PolicyName: !Sub "Role=AlertOnCPTestFailure,Env=${AccountParameter},Service=SNS,Rights=Publish" PolicyDocument: Version: "2012-10-17" Statement: - Effect: "Allow" Action: - "sns:Publish" Resource: - !Ref CodePipelineTestStageFailureTopic Roles: - !Ref IAMRoleAlertOnCPTestFailure IAMPolicyGetPipelineStatus: Type: "AWS::IAM::Policy" DependsOn: MoveToPHIIAMRole Properties: PolicyName: !Sub "Role=AlertOnCPTestFailure,Env=${AccountParameter},Service=CodePipeline,Rights=R" PolicyDocument: Version: "2012-10-17" Statement: - Effect: "Allow" Action: - "codepipeline:GetPipeline" - "codepipeline:GetPipelineState" - "codepipeline:ListPipelines" Resource: - "*" Roles: - !Ref IAMRoleAlertOnCPTestFailure IAMRoleAlertOnCPTestFailure: Type: "AWS::IAM::Role" Properties: RoleName: !Sub "Role=AlertOnCPTestFailure,Env=${AccountParameter},Service=Lambda" AssumeRolePolicyDocument: Version: "2012-10-17" Statement: - Effect: "Allow" Principal: Service: - "lambda.amazonaws.com" Action: - "sts:AssumeRole" Path: "/" ############################################# ### End of pipeline failure alerting lambda function #############################################
And that's about it. A couple notes on the Cloudformation template:
Alert Frequency
I'm using a cron expression as my schedule, currently set to go off every half hour during business hours, because we don't have over night staff that would be able to look at Pipeline failures. You can easily up the frequency with something like
cron(0/5 12-23 ? * MON-FRI *)
Lambda Environment Variables
One of the announcements from reInvent I was most excited about was AWS Lambda environment variables. This is a pretty magical feature that lets you pass in values to your Lambda functions. In this case, I'm using it to pass an SNS topic ARN that's being created in the same Cloudformation template into the Lambda function.Long story short, that means we can create resources in AWS and pass references to them into code without having to have a way to search for them or put their values into source code.
Environment: Variables: STAGE_NAME: Test TOPIC_ARN: !Ref CodePipelineTestStageFailureTopic
Flowerboxes
The CFT this example comes from contains multiple pipeline management functions, so the flower boxes ("###############") at the beginning and end of the Lambda Function definition are our way of keeping resources for each lambda function separated.
SNS Notifications
When you create an SNS topic with an email, the user will have to register with the topic. They'll get an email and have to click the link to allow the notifications.
Snippets
These are snippets I pulled out of our pipeline management Cloudformation stack. Obviously you'll have to put them into a Cloudformation template that references the SAM Cloudformation Transform and has a valid header like the one below:
AWSTemplateFormatVersion: '2010-09-09' Transform: AWS::Serverless-2016-10-31 Resources:......
Happy alerting!
No comments:
Post a Comment