Sometimes you want to add custom, provider related resources to your service to use provider specific functionality
which is not yet available through events or plugins. Serverless has you covered and enables you a convenient way to add
those resources with the help of resources
section in the serverless.yml
file.
Serverless uses the services resources
object as a place to store all the provider specific resources like compiled
functions or events.
After initialization, Serverless will try to load the resources
object from the
serverless.yml
file into memory.
It will create an own, empty one if it doesn't exist.
You can use this place to add custom provider resources by writing the resource definition in YAML syntax inside the
resources
object. You can also use your variables from serverless.env.yml
in the Values
# serverless.yml
resources:
Resources:
CustomProviderResource:
Type: ResourceType
Properties:
Key: Value
Sometimes you need an extra S3 bucket to store some data in (say, thumbnails). This works by adding an extra S3 Bucket Resource to your serverless.yml
:
service: lambda-screenshots
provider: aws
functions:
...
resources:
Resources:
ThumbnailsBucket:
Type: AWS::S3::Bucket
Properties:
# You can also set properties for the resource, based on the CloudFormation properties
BucketName: my-awesome-thumbnails
# Or you could use a variable from your serverless.env.yml
# BucketName: ${bucketname}
As a practical example for adding a custom resource to your service, we're going to demonstrate how you can create an API Gateway HTTP proxy using CloudFormation templates/resources.
To set up an HTTP proxy, you'll need two CloudFormation templates, one for the endpoint (known as resource in CF), and
one for method. These two templates will work together to construct your proxy. So if you want to set your-app.com/serverless
as a proxy for serverless.com
, you'll need the following two templates in your serverless.yml
:
# serverless.yml
service: service-name
provider: aws
functions:
...
resources:
Resources:
ProxyResource:
Type: AWS::ApiGateway::Resource
Properties:
ParentId:
Fn::GetAtt:
- RestApiApigEvent # our default Rest API logical ID
- RootResourceId
PathPart: serverless # the endpoint in your API that is set as proxy
RestApiId:
Ref: RestApiApigEvent
ProxyMethod:
ResourceId:
Ref: ProxyResource
RestApiId:
Ref: RestApiApigEvent
Type: AWS::ApiGateway::Method
Properties:
HttpMethod: GET # the method of your proxy. Is it GET or POST or ... ?
MethodResponses:
- StatusCode: 200
Integration:
IntegrationHttpMethod: POST
Type: HTTP
Uri: http://serverless.com # the URL you want to set a proxy to
IntegrationResponses:
- StatusCode: 200
There's a lot going on in these two templates, but all you need to know to set up a simple proxy is setting the method & endpoint of your proxy, and the URI you want to set a proxy to.
Now that you have these two CloudFormation templates defined in your serverless.yml
file, you can simply run
serverless deploy
and that will deploy these custom resources for you along with your service and set up a proxy on your Rest API.
Sometimes it's hard to translate the provider specific resources into valid YAML syntax. Furthermore the resource code
might be verbose and make the serverless.yml
file bloated.
You can always use JSON-Ref to reference external .json
files. This way you can organize your resource related code
into an own .json
file and reference it from within serverless.yml
like this:
resources:
Resources:
$ref: ./cloudformation-resources.json
The corresponding resources which are defined inside the cloudformation-resources.json
file will be resolved and loaded
into the Resources
section.
If you want to give permission to your functions to access certain resources on your AWS account, you can add custom IAM role statements to your service by adding the statements in the iamRoleStatements
array in the provider
object. As those statements will be merged into the CloudFormation template you can use Join, Ref or any other CloudFormation method or feature. You're also able to either use YAML for defining the statement (including the methods) or use embedded JSON if you prefer it. Here's an example that uses all of the above:
# serverless.yml
service: new-service
provider:
name: aws
iamRoleStatements:
- Effect: "Allow"
Action:
- "s3:ListBucket"
Resource: { "Fn::Join" : ["", ["arn:aws:s3:::", { "Ref" : "ServerlessDeploymentBucket"} ] ] }
- Effect: "Allow"
Action:
- "s3:PutObject"
Resource:
Fn::Join:
- ""
- - "arn:aws:s3:::"
- "Ref" : "ServerlessDeploymentBucket"
On deployment, all these statements will be added to the IAM role that is assumed by your lambda functions.
On deployment Serverless will load the base stack template and merge the custom resources you've defined in the resources
section of the service alongside the compiled function and corresponding event resources into it.
After that the template (with all merged resources) will be deployed on the providers infrastructure.
The resources
section inside the serverless.yml
file is a place
where you can add custom, provider specific resource definitions which should be created on service deployment.
It gives you access to the whole feature set your provider offers and makes Serverless even more extensible.
The last thing we need to learn is how we can remove our service. Let's take a look at this now.