Skip to content

Deployment

Let's deploy the todo app to railway.app.

Prepare for Production

In this tutorial, we'll deploy both the Angular app and the API server as one server-side app, and redirect all non-API requests to return the Angular app.

In addition, to follow a few basic production best practices, we'll use compression middleware to improve performance and helmet middleware for security

  1. Install compression and helmet.
sh
npm i compression helmet
npm i @types/compression --save-dev
  1. Add the highlighted code lines to src/server/index.ts, and modify the app.listen function's port argument to prefer a port number provided by the production host's PORT environment variable.
ts
// src/server/index.ts

import express from "express"
import { api } from "./api"
import session from "cookie-session"
import { auth } from "./auth"
import helmet from "helmet"
import compression from "compression"
import path from "path"

const app = express()
app.use(
  session({
    secret: process.env["SESSION_SECRET"] || "my secret"
  })
)
app.use(
  helmet({
    contentSecurityPolicy: {
      directives: {
        'script-src-attr': ["'unsafe-inline'"],
      },
    },
  })
)
app.use(compression())
app.use(auth)
app.use(api)

app.use(express.static(path.join(__dirname, '../remult-angular-todo/browser')));
app.get('/*', (req, res) => {
  res.sendFile(
    path.join(__dirname, '../remult-angular-todo/browser', 'index.html')
  );
});

app.listen(process.env["PORT"] || 3002, () => console.log("Server started"))

Angular versions <17

If you're using angular version 16 or less, the result path is: '../remult-angular-todo - adjust both lines in the src/server/index.ts accordingly

  1. Modify the highlighted code in the api server module to prefer a connectionString provided by the production host's DATABASE_URL environment variable.

    ts
    // src/server/api.ts
    
    //...
    const DATABASE_URL = process.env["DATABASE_URL"];
    
    export const api = remultExpress({
     dataProvider: DATABASE_URL
       ? createPostgresDataProvider({ connectionString: DATABASE_URL })
       : undefined,
       //...
     })

Note

In order to connect to a local PostgresDB, add DATABASE_URL to an .env file, or simply replace process.env["DATABASE_URL"] with your connectionString.

If no DATABASE_URL has found, it'll fallback to our local JSON files.

  1. In the root folder, create a TypeScript configuration file tsconfig.server.json for the build of the server project using TypeScript.
json
// tsconfig.server.json

{
  "extends": "./tsconfig.json",
  "compilerOptions": {
    "module": "commonjs",
    "esModuleInterop": true,
    "noEmit": false,
    "outDir": "dist",
    "skipLibCheck": true,
    "rootDir": "src"
  },
  "include": ["src/server/index.ts"]
}
  1. Modify the project's build npm script to additionally transpile the API server's TypeScript code to JavaScript (using tsc).
json
// package.json

"build": "ng build && tsc -p tsconfig.server.json"
  1. Modify the project's start npm script to start the production Node.js server.
json
// package.json

"start": "node dist/server/"

The todo app is now ready for deployment to production.

Test Locally

To test the application locally run

sh
npm run build
npm run start

Now navigate to http://localhost:3002 and test the application locally

Deploy to Railway

In order to deploy the todo app to railway you'll need a railway account. You'll also need Railway CLI installed, and you'll need to login to railway from the cli, using railway login.

Click enter multiple times to answer all its questions with the default answer

  1. Create a Railway project.

    From the terminal in your project folder run:

    sh
    railway init
  2. Select Empty Project

  3. Set a project name.

  4. Once it's done add a database by running the following command:

    sh
    railway add
  5. Select postgressql as the database.

  6. Once that's done run the following command to upload the project to railway:

    sh
    railway up
  7. got to the railway project's site and click on the project

  8. Switch to the variables tab

  9. Click on + New Variable, and in the VARIABLE_NAME click Add Reference and select DATABASE_URL

  10. Add another variable called SESSION_SECRET and set it to a random string, you can use an online UUID generator

  11. Switch to the settings tab

  12. Under Environment click on Generate Domain

  13. Click on the newly generated url to open the app in the browser and you'll see the app live in production. (it may take a few minutes to go live)

Note

If you run into trouble deploying the app to Railway, try using Railway's documentation.

That's it - our application is deployed to production, play with it and enjoy.

Love Remult?  Give our repo a star.⭐

MIT Licensed | Made by the Remult team with ❤️