Originally posted at Serverless on Sep 18, 2017
- New to Step Functions? Check out this Ultimate Guide to AWS Step Functions
When diving into the Functions as a Service (FaaS) world, a question that often pops up is:
If serverless functions are stateless, how do I manage state?
There are a number of ways to manage state with backend data stores, tmp directories & building this logic into your existing lambda functions but there is a simpler alternative provided by AWS: Step Functions.
Step Functions allows you to control complex workflows using Lambda functions without the underlying application managing and orchestrating the state. In essence, it’s a state machine to help with complex workflows and aims at keeping your lambda functions free of this additional logic.
Serverless + Step Functions
A couple months ago, I created the Serverless Step Functions plugin to deploy and manage Step Functions and a bunch of composed Lambda functions via the Serverless Framework.
In this post, I will share the functionality and usage of the plugin, and a workflow for your development.
So let’s get down to business!
Install
Before getting started, you need to install the plugin. This is hosted on the Serverless Plugins registry, so you can install this via the plugin install command which is introduced since v1.22.0.
Please run the following command in your service, then the plugin will be added automatically in plugins array in your serverless.yml file.
$ serverless plugin install --name serverless-step-functions |
If you run serverless --help command and you can see an explanation of subcommands for the plugin like `serverless invoke stepf, installing is successful.
Getting Started
Define AWS state language
To define a workflow with Step Functions, you need write a structured language called Amazon States Language, which can be defined within definition section with yaml format in your serverless.yml.
I recommend using in combination with Serverless AWS Pseudo Parameters since it makes it easy to set up in Resource section in serverless.yml.
The following is an example which is a simplest state machine definition, which is composed of a single lambda function.
stepFunctions: | |
stateMachines: | |
hellostepfunc1: | |
definition: | |
Comment: "A Hello World example of the Amazon States Language using an AWS Lambda Function" | |
StartAt: HelloWorld1 | |
States: | |
HelloWorld1: | |
Type: Task | |
Resource: "arn:aws:lambda:#{AWS::Region}:#{AWS::AccountId}:function:${self:service}-${opt:stage}-foobar-baz" | |
End: true | |
plugins: | |
- serverless-step-functions | |
- serverless-pseudo-parameters |
Event
You can define events to invoke your Step Functions. Currently, http and scheduled events have been supported. The configuration syntax is similar to the Lambda events provided by the framework core.
Here’s how to define those events:
stepFunctions: | |
stateMachines: | |
hello: | |
events: | |
- http: | |
path: hello | |
method: GET | |
- schedule: rate(2 hours) | |
definition: |
Use triggered Lambda events
If you want to use events other than http and scheduled, you can create a Lambda function which only run your statemachine
Using the AWS SDK, you can trigger your step functions like:
'use strict'; | |
const AWS = require('aws-sdk'); | |
const stepfunctions = new AWS.StepFunctions(); | |
module.exports.start = (event, context, callback) => { | |
const stateMachineArn = process.env.statemachine_arn; | |
const params = { | |
stateMachineArn | |
} | |
return stepfunctions.startExecution(params).promise().then(() => { | |
callback(null, `Your statemachine ${stateMachineArn} executed successfully`); | |
}).catch(error => { | |
callback(error.message); | |
}); | |
}; |
Then, you set up the Lambda will be triggered by events what you want. startExecution API requires a statemachine ARN so you can pass that via environment variables system.
Here’s serverless.yml sample which a triggered statemachine by S3 event.
service: example-stepf-nodejs | |
provider: | |
name: aws | |
runtime: nodejs6.10 | |
iamRoleStatements: | |
- Effect: "Allow" | |
Action: | |
- "states:StartExecution" | |
Resource: | |
- "*" | |
functions: | |
startExecution: | |
handler: handler.start | |
events: | |
- s3: photos | |
environment: | |
statemachine_arn: ${self:resources.Outputs.MyStateMachine.Value} | |
stepFunctions: | |
stateMachines: | |
hellostepfunc1: | |
name: MyStateMachine | |
definition: | |
Comment: "A Hello World example of the Amazon States Language using an AWS Lambda Function" | |
StartAt: HelloWorld1 | |
States: | |
HelloWorld1: | |
Type: Task | |
Resource: "arn:aws:lambda:#{AWS::Region}:#{AWS::AccountId}:function:${self:service}-${opt:stage}-hello" | |
End: true | |
resources: | |
Outputs: | |
MyStateMachine: | |
Description: The ARN of the example state machine | |
Value: | |
Ref: MyStateMachine | |
plugins: | |
- serverless-step-functions | |
- serverless-pseudo-parameters |
Create a sample application
Let’s consider a small 2 step application that starts EC2 and write the result on S3 bucket.
First, we will create a Lambda function that only starts an EC2 instance, to which will be passed instanceId via API Body request parameter.
'use strict'; | |
const AWS = require('aws-sdk'); | |
module.exports.startEC2 = (event, context, callback) => { | |
const ec2 = new AWS.EC2(); | |
const params = { | |
InstanceIds: [ | |
event.instanceId | |
] | |
} | |
return ec2.startInstances(params).promise().then(() => { | |
callback(null, `Your ${event.instanceId} instance started successfully`); | |
}).catch(error => { | |
callback(error.message); | |
}); | |
}; |
Then, here is another Lambda function which writes a log to S3 Bucket.
'use strict'; | |
const AWS = require('aws-sdk'); | |
module.exports.writeS3 = (event, context, callback) => { | |
const s3 = new AWS.S3(); | |
const params = { | |
Bucket: 'sls-logs-bukect', | |
Key: 'success!!' | |
} | |
return s3.putObject(params).promise().then(() => { | |
callback(null, `a log writed successfully`); | |
}).catch(error => { | |
callback(error.message); | |
}); | |
}; |
In the end, describe your serverless.yml looks like, and deploy with serverless deploy.
service: example-stepf-nodejs | |
provider: | |
name: aws | |
runtime: nodejs6.10 | |
iamRoleStatements: | |
- Effect: "Allow" | |
Action: | |
- "ec2:*" | |
- "s3:*" | |
Resource: | |
- "*" | |
functions: | |
startEC2: | |
handler: handler.startEC2 | |
writeS3: | |
handler: handler.writeS3 | |
stepFunctions: | |
stateMachines: | |
hellostepfunc1: | |
events: | |
- http: | |
path: startEC2 | |
method: post | |
definition: | |
Comment: "A sample application" | |
StartAt: StartEC2 | |
States: | |
StartEC2: | |
Type: Task | |
Resource: "arn:aws:lambda:#{AWS::Region}:#{AWS::AccountId}:function:${self:service}-${opt:stage}-startEC2" | |
Next: WriteS3 | |
WriteS3: | |
Type: Task | |
Resource: "arn:aws:lambda:#{AWS::Region}:#{AWS::AccountId}:function:${self:service}-${opt:stage}-writeS3" | |
End: true | |
plugins: | |
- serverless-step-functions | |
- serverless-pseudo-parameters |
If you can see the API Gateway endpoint on your console, it means to deploy successfully。
Serverless StepFunctions OutPuts | |
endpoints: | |
POST - https://ae0dyh8676.execute-api.us-east-1.amazonaws.com/dev/startEC2 |
Send a CURL request to your live endpoint:
curl -XPOST https://ae0dyh8676.execute-api.us-east-1.amazonaws.com/dev/startEC2 -d '{"instanceId":"<your instance ID>"}' |
You should see that specified EC2 will be started and a log will be written to S3 Bucket.
Summary
The Serverless Step Functions plugin makes it easier to manage and deploy your Step Functions.
If you have any comments or feedback, please create a new issue or send a Pull Request. I always welcome them!!
One more thing, tutorial on how to use the plugin has been coverd on FOOBAR youtube channel. You can also learn it there. Thanks @mavi888uy for making the great video!
Originally published at https://www.serverless.com.
Top comments (0)