- Explore more advanced usages of the "run-commands" executor
- Go through an example of how to deploy an API to Heroku through Nx
-
For this workshop you'll need two CLI tools installed:
- Heroku CLI
- Verify installation via:
heroku --version
- Verify installation via:
- Docker - Verify via
docker --version
- Heroku CLI
-
Let's prepare Heroku to deploy our API:
# login first heroku login # create a new Heroku app where we'll deploy the API heroku create <choose some unique name for your api> # Get an authorization token so we don't have to login everytime heroku authorizations:create
⚠️ Make sure you remember and keep track of these values, as we'll use them later:- The authorization "Token"
- The exact app name you used
👉 This will determine the address where the API will be deployed to:https://<your-app-name>.herokuapp.com
-
Let's setup our ENV variables from the beginning now
apps/api/.local.env
HEROKU_API_KEY=<your-heroku-token>
-
Create a new file
apps/api/Dockerfile
# use a Node v12 based image FROM node:12-alpine # switch to the /app folder in the image WORKDIR /app # copy all files from the folder its in into the /app folder we switched to COPY ./ ./ # launch the main.js file CMD node main.js
❓ What's our plan here?
Heroku allows you to do container deployments. You define a Docker image that will run your server code. You then point the Heroku CLI to your image, and it will build it, deploy it, and run it at the remote address.
So the plan is:
- define a Docker image and have it sit idly in our app's source folder
- when we want to deploy, we'll build our app to
dist/apps/api
- we'll then copy this image over to
dist/apps/api
- because it will be in the same folder as our built assets, it will copy all of them into the container via the
COPY ./ ./
instruction - and then run the server via
CMD node main.js
-
If you
nx build api
right now- 👍 Then
cd dist/apps/api && node main.js
It should work. Because it has access tonode_modules
- 👎 If you copy your built sources to some other folder on your file system.
And then try to
node main.js
in that folder that doesn't hace access tonode_modules
- it will fail
💡 By default, dependencies of server projects are not bundled together, as opposed to your Angular apps. If curious why, you can read more here.
- 👍 Then
-
Let's fix the above - In
workspace.json
, under the production build options for the API (projects -> api -> targets -> build -> configurations -> production
) add this as an option:```json "externalDependencies": [ "@nestjs/microservices", "@nestjs/microservices/microservices-module", "@nestjs/websockets/socket-module", "class-transformer", "class-validator", "cache-manager" ], ``` <details> <summary>❓ What does this do?</summary> The above option tells webpack to bundle ALL the dependencies our API requires inside `main.js`, except the ones above (which fail the build if we tell webpack to include, because they're lazily loaded). Normally, it's not recommended to bundle any dependencies with your server bundles, but in this case it simplifies the deployment process. </details><br />
-
Use the
@nrwl/workspace:run-commands
generator to generate another "deploy" target:- This time for the
api
project - Use the
--cwd
option to ensure all commands execute in thedist/apps/api
folder - Leave the "command" blank for now
- This time for the
-
Let's customise the generated "deploy" config a bit
Go to
workspace.json
and add the commands that we'll need to run:"deploy": { "executor": "@nrwl/workspace:run-commands", "outputs": [], "options": { "commands": [ <--- ADD THESE COMMANDS "cp ../../../apps/api/Dockerfile .", "heroku container:login", "heroku container:push web -a <the name of your Heroku App>", "heroku container:release web -a <the name of your Heroku App>" ], "cwd": "dist/apps/api" } },
- By default, if you give a list of commands to
run-commands
, it will run them in parallel. In our case, we want them to run one after another. See if you can add a configuration option to make them run sequentially
-
Let's enable CORS on the server so our API can make requests to it (since they'll be deployed in separate places):
yarn add cors
ornpm i -S cors
- In
apps/api/src/main.ts
- Enable CORS:
import * as cors from 'cors'; app.use(cors());
⚠️ Normally, you want to restrict this to just a few origins. But to keep things simple in this workshop we'll enable it for all origins.
-
Heroku exposes a
PORT
env variable (capital letters), so we'll need to change our API to support this// in apps/api/src/main.ts, capitalize "port" const port = process.env.PORT || 3333;
-
Use Nx to build the API for production, and then deploy it!
⚠️ Note: On Windows, the deploy will fail. Create a separatedeploy-windows
target, as we did in the previous lab, but use this for the copy command:"commands": [ "xcopy \"..\\..\\..\\apps\\api\\Dockerfile\" .", ....
⚠️ Note: On Apple M1 commputers the deployement will succeed, but running Docker image on heroku app will fail due to unsupported ARM architecture. Don't worry you will still deploy your api viaGitHub
in the Lab 21. You can head over there and then come back to lab 20.
- Go to
https://<your-app-name>.herokuapp.com/api/games
- it should return you a list of games.
🎓 If you get stuck, check out the solution