Deploying Node.js apps to Fly.io

November 17, 20224 min read

This post covers main notes from setting up the Fly.io CLI tool to CI/CD pipeline with Github actions


  • Run the following command to install the CLI tool and follow the instructions to add the necessary commands to the appropriate shell configuration file
curl -L https://fly.io/install.sh | sh
  • Create an account or log in if you already have an account
fly auth signup
fly auth login

Node server

  • Run the following commands for packages setup
npm init -y
npm i express
  • Write basic server
// server.js
const express = require('express');
const app = express();
const PORT = process.env.PORT || 8080;
app.get('/', (_, res) => res.send('Hello world'));
app.listen(PORT, () => {
console.log(`Server is running on port ${PORT}`);
  • Configure start script
// package.json
"scripts": {
"start": "node server.js"

Project setup

Choose the project name and region for deployment. The following command will create a project and generate a configuration file. Project URL will be <PROJECT_NAME>.fly.dev

fly launch

Environment variables

Configuration variables

Generated fly.toml file contains env section for environment variables which can be publicly available like URLs, HTTP timeout values, etc.

// ...
PORT = "8080"
NEW_VARIABLE = "value"
// ...


The following commands are used for setting up the secrets like API keys, credentials, etc.

fly secrets set KEY=value
fly secrets list
fly secrets unset KEY


Fly.io provides the following databases


Run the following commands to provision the Redis database and connect with it, eviction is disabled by default. URL for it can be found with the status command, set it as a secret.

fly redis create
fly redis connect
fly redis list
fly redis status <database-name>


Run the following command to bootstrap a Postgres database cluster. Connection string will be prompted only once, set it as a secret.

fly postgres create
fly postgres connect -a <postgress-app-name>

Fly.io performs daily snapshots of the provisioned volumes, snapshots are volume-specific. Postgres database can be restored out of the provided snapshot.

fly volumes list -a <postgres-app-name>
fly volumes snapshots list <volume-id>
fly postgres create --snapshot-id <snapshot-id>


Every app runs inside a virtual machine, the following command shows VM size and how many app instances are running.

fly scale show

Vertical scaling

  • The following command shows the list of VM types and their memory allocations
fly platform vm-sizes
  • Run the following command to upgrade VM to one of the types from the previous command, memory allocation in MBs can be specified as well
fly scale vm <type> --memory 1024
  • Run the following command to increase VM memory, 1024MB in this case
fly scale memory 1024

Horizontal scaling

Run the following command to increase the number of app instances, 3 in this case

fly scale count 3

Autoscaling is disabled by default, run the following commands to configure it

fly autoscale show
fly autoscale set min=3 max=6


  • Get the real-time logs
fly logs
  • Get the info about the app, services, and IP addresses
fly info
  • Get the state of the project
fly status
  • Get the app IP addresses
fly ips list
  • Restart the app
fly restart <app-name>


Manual deployment

The following commands are used to deploy and open the deployed version

fly deploy
fly open

Continuous deployment with Github actions

A project auth token is needed, run the following command to generate it. Add the generated key as a Github action secret in the Settings Secrets Actions page

fly auth token

Add .github/workflows/config.yml file with the following configuration

name: CI/CD pipeline
- main
runs-on: ubuntu-latest
container: node:18.12.1-alpine3.15
- name: Github checkout
uses: actions/checkout@v3
- uses: actions/setup-node@v3
node-version: 18
- run: npm ci
- run: npm run lint
- run: npm test
- run: npm audit
- name: Setup Fly.io config
uses: superfly/flyctl-actions/setup-flyctl@master
- name: Deploy to Fly.io
run: flyctl deploy --remote-only


© 2022