Functions
Definitions
Section titled “Definitions”There are 3 kinds of function definitions:
- Topology functions
- Interned functions
- Standalone functions
Topology functions
Section titled “Topology functions”tc discovers functions in the current directory. A function is any directory that contains a
- handler.{py,rb,clj,js} file and/or
- function.yml file
At it’s simplest, a function directory (say foo) looks as follows:
foo/ - handler.{py, rb, js, clj}tc infers the kind of function, runtime and build instructions. However, we can be more specific as follows in a function.yml file
name: fooruntime: lang: python3.11 handler: handler.handlerInterned functions
Section titled “Interned functions”Interned functions are those that are explicitly defined in topology.yml
functions: fun1: uri: ../my-fun1 fun2: uri: ../my-fun2Standalone functions
Section titled “Standalone functions”A function that does belong to a topology or has no topology defined is a standalone function.
name: fooruntime: lang: python3.11 handler: handler.handlerComponents
Section titled “Components”| Key | Default | Optional? | Comments |
|---|---|---|---|
| lang | Inferred | yes | |
| handler | handler.handler | ||
| package_type | zip | possible values: zip, image | |
| uri | file:./lambda.zip | ||
| mount_fs | false | yes | |
| snapstart | false | yes | |
| memory | 128 | yes | |
| timeout | 30 | yes | |
| provisioned_concurrency | 0 | yes | |
| reserved_concurrency | 0 | yes | |
| layers | [] | yes | |
| extensions | [] | yes | |
| environment | {} | yes | Environment variables |
Permissions
Section titled “Permissions”By default, tc infers permissions and sets the right boundaries in the sandbox. However, you may need to override the permissions by specifying a custom roles file in $INFRA_ROOT (say $GIT_ROOT/infrastructure/tc/TOPOLOGY-NAME/roles/FUNCTION-NAME.json. The contents of the roles file is typically the IAM policy.
Environment variables
Section titled “Environment variables”Specify env variables in $GIT_ROOT/infrastructure/tc/TOPOLOGY-NAME/vars/FUNCTION-NAME.json
{ "default": { "timeout": 120, "memory_size": 128, "environment": { "API_KEY": "ssm:/path/to/api-key", "DB_HOST": "dev.db.net", "API_GATEWAY_URL": "{{API_GATEWAY_URL}}" } }, "prod": { "timeout": 60, "memory_size": 1024, "environment": { "API_KEY": "ssm:/path/to/api-key", "DB_HOST": "prod.db.net" } }, "SANDBOX-NAME": { "timeout": 60, "environment": { "API_KEY": "ssm:/path/to/api-key" } }}The vars or runtime file is map of default and sandbox-specific overrides. Environment variables in the runtime file can be either an URI or plain text. Supported URIs are ssm:/ , s3:/ and file:/. If an URI is specified, tc resolves the values and injects them as actual values when creating the lambda. Decryption using extensions is also available. See Extensions.
We can also discover Endpoints for routes and mutations that are sandbox-specific. tc does a topological sort and gets the URLs ahead of time before rendering the vars.json file.
Update components
Section titled “Update components”tc provides mechanisms to update specific component of entities or topology in a given sandbox. This is incredibly useful when developing your core topology.
tc update -s sandbox -e env -c functions/layerstc update -s sandbox -e env -c functions/varstc update -s sandbox -e env -c functions/concurrencytc update -s sandbox -e env -c functions/runtimetc update -s sandbox -e env -c functions/tagstc update -s sandbox -e env -c functions/rolestc update -s sandnox -e env -c functions/function-nameDependencies
Section titled “Dependencies”tc has a sophisticated function builder that can build different kinds of artifacts with various language runtimes (Clojure, Janet, Rust, Ruby, Python, Node)
In the simplest case, when there are no dependencies in a function, we can specify how the code is packed (zipped) as follows in function.yml:
name: simple-functionruntime: lang: python3.10 handler: handler.handlerThe above is a pretty trivial example and it gets complicated as we start adding more dependencies. We can specify how the function needs to be built. For example:
name: funciton-with-deps-exampleruntime: lang: python3.10 | python3.11 | python3.12 | ruby3.2 handler: handler.handlerbuild: kind: Code |Inline |Image | Layer pre: [String] post: [<String>] command: <String>| Attribute | Description |
|---|---|
| kind | Specifies how the dependencies are packaged Available options are Code, Inline, Image, Layer |
| pre | Array of commands to run before the dependencies are installed. Has shared build context, Host ssh-agent access. Typically useful to install system dependencies (yum) or private packages (ssh://github etc) |
| post | Array of commands to run after the dependencies are installed. Has shared build context, Host ssh-agent access and AWS access for given sandbox or centralized repo. Typically useful to pull models, CSV files etc from S3 or object stores and package them in the build artifact |
| command | Command to pack the code. Typically it is the zip command (zip -9 -q lambda.zip *) |
| share_context | Default: true. If true, copied current git repository for referencing any shared relative paths in the build container |
| skip_dev_deps | Default: false. Skips dev dependencies when building deps |
and then tc create -s <sandbox> -e <env> builds this function using the given command and creates it in the given sandbox and env.
Inline
Section titled “Inline”If the dependencies are reasonably small (< 50MB), we can inline those in the code’s artifact (lambda.zip).
name: python-inline-exampleruntime: lang: python3.12 package_type: zip handler: handler.handlerbuild: kind: Inline command: zip -9 -q lambda.zip *.pytc create -s <sandbox> -e <env> will implicitly build the artifact with inlined deps and create the function in the given sandbox and env. The dependencies are typically in lib/ including shared objects (.so files).
If inline build is heavy, we can try to layer the dependencies:
name: ppdruntime: lang: python3.10 handler: handler.handler layers: - ppd-layerbuild: kind: Layer pre: - yum install -y git - yum install -y gcc gcc-c++Note that we have specified the list of layers the function uses. The layer itself can be built independent of the function, unlike Inline build kind.
tc build --kind layertc publish --name ppd-layerWe can then create or update the function with this layer. At times, we may want to update just the layers in an existing sandboxed function
tc update -s <sandbox> -e <env> -c layersWhile Layer and Inline build kind should suffice to pack most dependencies, there are cases where 250MB is not good enough. Container Image kind is a good option. For example:
name: python-image-tree-exampleruntime: lang: python3.10 package_type: image handler: handler.handlerbuild: kind: Imagetc build --publishNote that the child image uses the parent’s version of the image as specified in the parent’s block.
Syncing base images
Section titled “Syncing base images”While we can docker pull the base and code images locally, it is cumbersome to do it for all functions recursively by resolving their versions. tc build --sync pulls the base and code images based on current function checksums. Having a copy the base or parent code images allows us to do incremental updates much faster.
Inspecting the images
Section titled “Inspecting the images”We can run tc build --shell in the function directory and access the bash shell. The shell is always on the code image of the current function checksum. Note that the code image using the Lambda Runtime Image as the source image.
Library
Section titled “Library”A library is a special kind of layer where there are no transitive dependencies packed into the layer artifact. This is useful if we have a directory of utilities.
lib/foo - bar - baztc build --kind library --name foo --publish -e <env>foo now can be used a regular layer in function.yml:runtime:layers
Filesystems
Section titled “Filesystems”We can mount a filesystem by specifying a runtime attribute mount_fs:
name: fn-with-fsruntime: lang: python3.11 package_type: image mount_fs: true handler: handler.handlerLayer lifecycle management
Section titled “Layer lifecycle management”On AWS, the layer versions are global and are not sandbox-aware. When associating layers with a function, we can pin the layers with monotonic versions as follows:
name: ppdruntime: lang: python3.10 package_type: zip handler: handler.handler layers: - ppd-layer:3However, this may not be practical when dealing with a large number of functions. Additionally, there is no way to tag layers and annotate if the version is stable or unusable. To solve this problem, tc provides a simple mechanism to differentiate between stable and dev layers. When creating a layer by default using tc build --layer <layer-name>, the layer’s name is suffixed with the string -dev. On updating the sandbox with the layers or when creating/updating functions, tc will bump the functions to the latest “dev” layer versions.
When a specific dev layer is ready to be promoted as stable, we do
tc build --promote --layer NAME [--version 123]If the version is not specified, tc will promote the latest dev layer to stable.
If a sandbox is named stable, it will use the stable layers. This is configurable in TC_CONFIG.
To use latest stable layers in your dev sandboxes, do:
TC_USE_STABLE_LAYERS=1 tc update -s SANDBOX -e PROFILE -c functions/layersProviders
Section titled “Providers”Default function provider is Lambda. We can make the same function code run in ECS Fargate with no change.
name: python-image-fargateruntime: handler: "python handler.py" package_type: image provider: Fargatebuild: kind: ImageTesting
Section titled “Testing”Invoke
Section titled “Invoke”By default, tc picks up a payload.json file in the current directory. You could optionally specify a payload file
tc invoke --sandbox main --env dev --payload payload.jsonor via stdin
cat payload.json | tc invoke --sandbox main --env devor as a param
tc invoke --sandbox main --env dev --payload '{"data": "foo"}'