The Serverless framework provides a powerful variable system to give your serverless.yml
configuration file extra flexibility. With Serverless Variables, you'll be able to do the following:
- Reference & load variables from environment variables
- Reference & load variables from CLI options
- Recursively reference properties of any type from the same
serverless.yml
file - Recursively reference properties of any type from other YAML/JSON files
- Recursively nest variable references within each other for ultimate flexibility
- Combine multiple variable references to overwrite each other
To reference environment variables, you'll need to use the ${env:SOME_VAR}
syntax in your serverless.yml
configuration file. Here's an example:
service: new-service
provider: aws
functions:
hello:
name: ${env:FUNC_PREFIX}-hello
handler: handler.hello
world:
name: ${env:FUNC_PREFIX}-world
handler: handler.world
In the previous example you're dynamically adding a prefix to the function names by referencing the FUNC_PREFIX
env var. So you can easily change that prefix for all functions by changing the FUNC_PREFIX
env var.
To reference CLI options that you passed, you'll need to use the ${opt:some_option}
syntax in your serverless.yml
configuration file. Here's an example:
service: new-service
provider: aws
functions:
hello:
name: ${opt:stage}-hello
handler: handler.hello
world:
name: ${opt:stage}-world
handler: handler.world
In the previous example you're dynamically adding a prefix to the function names by referencing the stage
option that you pass in the CLI when you run serverless deploy --stage dev
. So when you deploy, the function name will always include the stage you're deploying to.
To self-reference properties in serverless.yml
, you'll need to use the ${self:someProperty}
syntax in your serverless.yml
configuration file. This functionality is recursive, so you can go as deep in the object tree as you want. Here's an example:
service: new-service
provider: aws
custom:
globalSchedule: rate(10 minutes)
functions:
hello:
handler: handler.hello
events:
- schedule: ${self:custom.globalSchedule}
world:
handler: handler.world
events:
- schedule: ${self:custom.globalSchedule}
In the previous example you're setting a global schedule for all functions by referencing the globalSchedule
property in the same serverless.yml
file. This way, you can easily change the schedule for all functions whenever you like.
To reference variables in other YAML or JSON files, you'll need to use the ${file(./myFile.yml):someProperty}
syntax in your serverless.yml
configuration file. This functionality is recursive, so you can go as deep in that file as you want. Here's an example:
# myCustomFile.yml
globalSchedule: rate(10 minutes)
# serverless.yml
service: new-service
provider: aws
custom: ${file(./myCustomFile.yml)} # You can reference the entire file
functions:
hello:
handler: handler.hello
events:
- schedule: ${file(./myCustomFile.yml):globalSchedule} # Or you can reference a specific property
world:
handler: handler.world
events:
- schedule: ${self:custom.globalSchedule} # This would also work in this case
In the previous example you're referencing the entire myCustomFile.yml
file in the custom
property. You just need to pass the path relative to your service directory. You can also request specific properties in that file as shown in the schedule
property. It's completely recursive and you can go as deep as you want.
Adding many custom resources to your serverless.yml
file could bloat the whole file, so you can use the Serverless Variable syntax to split this up.
resources:
Resources: ${file(cloudformation-resources.json)}
The corresponding resources which are defined inside the cloudformation-resources.json
file will be resolved and loaded
into the Resources
section.
The Serverless variable system allows you to nest variable references within each other for ultimate flexibility. So you can reference certain variables based on other variables. Here's an example:
service: new-service
provider: aws
custom:
myFlexibleArn: ${env:${opt:stage}_arn}
functions:
hello:
handler: handler.hello
In the previous example, if you pass dev
as a stage option, the framework will look for the dev_arn
environment variable. If you pass production
, the framework will look for production_arn
, and so on. This allows you to creatively use multiple variables by using a certain naming pattern without having to update the values of these variables constantly. You can go as deep as you want in your nesting, and can reference variables at any level of nesting from any source (env, opt, self or file).
The Serverless framework gives you an intuitive way to reference multiple variables as a fallback strategy in case one of the variables is missing. This way you'll be able to use a default value from a certain source, if the variable from another source is missing.
For example, if you want to reference the stage you're deploying to, but you don't want to keep on providing the stage
option in the CLI. What you can do in serverless.yml
is:
service: new-service
provider:
name: aws
stage: dev
custom:
myStage: ${opt:stage, self:provider.stage}
functions:
hello:
handler: handler.hello
What this says is to use the stage
CLI option if it exists, if not, use the default stage (which lives in provider.stage
). So during development you can safely deploy with serverless deploy
, but during production you can do serverless deploy --stage production
and the stage will be picked up for you without having to make any changes to serverless.yml
.
This overwrite functionality is super powerful. You can have as many variable references as you want, from any source you want, and each of them can be of different type and different name.
Previously we used the serverless.env.yml
file to track Serverless Variables. It was a completely different system with different concepts. To migrate your variables from serverless.env.yml
, you'll need to decide where you want to store your variables.
- Using a config file: You can still use
serverless.env.yml
, but the difference now is that you can structure the file however you want, and you'll need to reference each variable/property correctly inserverless.yml
. For more info, you can check the file reference section above. - Using the same
serverless.yml
file: You can store your variables inserverless.yml
if they don't contain sensitive data, and then reference them elsewhere in the file usingself:someProperty
. For more info, you can check the self reference section above. - Using environment variables: You can instead store your variables in environment variables and reference them with
env.someEnvVar
. For more info, you can check the environment variable reference section above. - Making your variables stage/region specific:
serverless.env.yml
allowed you to have different values for the same variable based on the stage/region you're deploying to. You can achieve the same result by using the nesting functionality of the new variable system. For example, if you have two different ARNs, one fordev
stage and the other forprod
stage, you can do the following:${env:${opt:stage}_arn}
. This will make sure the correct env var is referenced based on the stage provided as an option. Of course you'll need to export bothdev_arn
andprod_arn
env vars on your local system.
Now you don't need serverless.env.yml
at all, but you can still use it if you want. It's just not required anymore. Migrating to the new variable system is easy and you just need to know how the new system works and make small adjustments to how you store & reference your variables.