Serverless with Typescript in a monorepo? 4 ways to make it work.
New journey
Several months ago our team has embarked on a new project. An exciting opportunity to choose a modern technology stack that suits you! So we did. The detail is that you need to make this new stack working smoothly, and your choices don’t backfire on you. Sometimes it is not trivial.
For example, you decided to manage all your projects in a single repository. Which also means managing their dependencies. At the same time, you opted for Serverless architecture. Your Lambdas need to be packaged with just enough libraries: they can run but remain slim. You’d think all should work out of the box. Not necessarily.
With a bit of experimentation, you can get it right. We’d like to save you some time, as we have found several alternative solutions to the Lambda packaging issue.
Stack
We opted for technologies that allow you to develop and scale your system quickly.
- AWS as a Cloud Provider
- Serverless framework to manage the backend services and infrastructure
- NodeJS on the backend
- React on the frontend (no mobile yet)
- Typescript as a primary language everywhere
- Yarn as a package manager
- Monorepo as a way to manage the codebase.
It appeared that a combination of these technologies requires some extra care.
Show me code
A real code example makes it easier to picture the point. Here is a simple app to create polls, vote, and see the results (it’s not production-grade).
The app consists of 4 parts:
- Infrastructure — a DynamoDB table that stores polls
- Service — business logic and API implemented as AWS Lambda and API Gateway
- App — a react application
- Model — shared code between service and app (In the real world you rather go with OpenAPI/JSON schema)
Around 100 lines of YAML, and Serverless can create the infrastructure and deploy the backend. Marvelous!
And… it doesn’t work.
You realize why by exploring a packaged Lambda function: it misses all the runtime dependencies. The service depends on the application’s model and the shortid
library:
package.json # All code examples assume poll/service directory"dependencies": {
"@poll/model": "1.0.0",
"shortid": "^2.2.15"
}
A deployed Lambda does not have any dependency:
Let’s find out why!
Looking for a missing piece
A standard practice in a monorepo is to use Yarn workspaces. In this case, the dependencies get hoisted, so there is only one big fat node_modules
folder, at the project root. In our case, the service
workspace has an empty node_modules
. Maybe it confuses Serverless?
Attempt 1: Disable hoisting
Yarn Workspaces offer the nohoist option: you can specify the dependencies you want to reside in a local node_modules
. It should help:
package.json"private": true,
+ "workspaces": {
+ "nohoist": [
+ "@poll/model",
+ "shortid"
+ ]
+ },
Good news: now the Lambda package includes the shortid
library. Bad news: it still misses the model. It’s weird because the model is present in the node_modules
directory of the service:
ls node_modules/
@poll shortidll node_modules/@poll
model -> ../../../model
Aha, it’s a symlink! And it doesn’t get packaged.
One more issue, by the way, is that shortid
depends on another library nanoid
that is also missing.
Clearly, disabling hoisting does not work by itself.
We got it to work, but…
Disabling hoisting completely
package.json
"private": true,
+ "workspaces": {
+ "nohoist": [
+ "**"
+ ]
+ },
and employing Yalc to copy dependencies into node_modules
instead of using symlinks does the trick. You can see the result in this branch.
This is quite suboptimal. Having multiple workspaces with a big fat node_modules
makes your builds slower. And you need to build tooling that copies around shared modules every time they get changed (later we found the way to avoid copying and make symlinks work— see Attempt 2). Is there a more nifty solution?
Fortunately, Serverless has plenty of useful plugins. Let’s have a look.
Attempt 2: serverless-plugin-monorepo
It’s easy to google this plugin, and it looks promising:
A Serverless plugin design to make it possible to use Serverless in a Javascript mono repo with hoisted dependencies, e.g. when using Yarn Workspaces.
This plugin alleviates the need to use nohoist functionality by creating symlinks to all declared dependencies.
The plugin has created symlinks to all the necessary libraries, but these libraries didn’t make it to Lambda’s bundle. Same problem as in the previous case.
After some digging, it appears that the culprit is serverless-plugin-typescript
that transpiles Typescript to Javascript before packaging Lambdas. When the plugin is active, Serverless ignores the symlinked dependencies. The workaround would be to not use the plugin and call tsc
manually. In a nutshell:
serverless.yml
plugins:
- - serverless-plugin-typescript
+ - serverless-plugin-monorepo# Exclude Typescript source and other dev-related files
+package:
+ exclude:
+ - src/*.ts
+ - node_modules/.bin/**
+ - '*.json'package.json:
+ "scripts": {
+ "package": "tsc && serverless package",
+ "deploy": "tsc && serverless deploy",
+ "deploy:fn": "tsc && serverless deploy function"
+ }tsconfig.json:
+ "compile": "tsc",
Then:
cd poll/service
sls package
ls -lh .serverless40K poll.zip
Looks OK, still can be better. Please also note that the package is the same for all functions. It means that for each Lambda function it will deploy the other Lambdas and their dependencies as well. In this case, every Lambda would weight only 40KB. In real life it can grow much bigger (just add axios
as a dependency and see happens). It would be nice to find a way to package only code that a function needs.
Let’s keep digging!
Attempt 3: serverless-webpack-plugin
Another popular way to package Lambdas is by using the serverless-webpack plugin. It transpiles Typescript with Babel, no need in serverless-plugin-typescript
.
Additionally, the plugin can package each function only with the dependencies it uses, resolving the size issue that we recently mentioned.
Let’s add a couple of lines to serverless.yml:
serverless.yml
+ package:
+ individually: true
And build it:
sls packagels -lh .serverless/poll.zip1.2M create.zip
1.2M getPoll.zip
1.2M getSummary.zip
1,2K vote.zip
Oops, 1.2 megabytes per function doesn’t look good. Apparently, aws-sdk
creeps in, even if you explicitly exclude it from the bundle. Fortunately, there is a workaround.
A bit of sorcery in webpack.config.js
(the full version is here), and the result is quite good:
sls package
ls -lh .serverless/13K create.zip
5,8K getPoll.zip
5,8K getSummary.zip
5,8K vote.zip
This is essentially what we want! And it can be even better.
Attempt 4: serverless-plugin-optimize
Here comesserverless-plugin-optimize
Bundle with Browserify, transpile and minify with Babel automatically to your NodeJS runtime compatible JavaScript.
It doesn’t require any additional configuration and works seamlessly with serverless-plugin-typescript
:
serverless.json
plugins:
- serverless-plugin-typescript
+ - serverless-plugin-optimize
+package:
+ individually: true # should optimize each function
and the result:
sls package
cd poll/service
sls package
ls -lh .serverless6,8K create.zip
3,0K getPoll.zip
3,0K getSummary.zip
3,0K vote.zip
The total size of the functions is only 16K. You can see the effort in this pull request. Only 4 lines of code resolve the packaging issue and ensure our Lambdas are tiny. Neat!
Conclusion
When you meet a technical issue, Google is your big friend. Sometimes though you find an approach works for others, and not for you because of a little detail that you need to discover. You can also find a solution quickly, but is it the best one?
In this post we explored several ways of packaging Lambda functions when you deal with Serverless, NodeJS and Typescript in a monorepo. 3 out of 4 required a workaround to start working! Clearly, some are more efficient than others.
Here they are, ranked by our preference:
serverless-plugin-optimize
nails it! 👍serverless-plugin-webpack
a strong contender. Requires more configuration, and the package sizes are a bit bigger. 👍serverless-plugin-monorepo
needs a workaround to transpile Typescript, and the package size is not optimal. 😐- Disable hoisting and use
yalc
. Headache with dependencies, package size is not optimal. 👎
If you have another solution that works for you, please let us know!