Barebones Rails, Nuxt and Fly.io Tutorial
With RSpec, Vitest and Playwright Tests on CircleCI
01/29/2025
read time:17 mins
Here we’ll make a MVP (“minimum viable product”) of a rails API-only backend with rspec tests that talks to a nuxt frontend with vitest component tests and playwright end-to-end tests. This will have close to no functionality and no styling, but everything will be wired together correctly and all three test suites will pass when run locally or on CircleCI.
This tutorial is available in markdown on github.
Local Preliminary App Setup
- Create app directory:
mkdir app
cd app
- Initialize git repo:
git init
(hit enter for all prompts)
- Let’s create some blank files for app secrets (like the PostgreSQL connection details) and for non-secret app info.
touch .gitignore .secrets .appinfo
- Let’s add
.secrets
and.appinfo
to our.gitignore
:.gitignore
.secrets .appinfo
- Choose what your unique front and backend app names will be. If someone else on Fly.io already has an app with the same name, they will ask you to choose a new name. So the more unique, the better. Something like
myuniqueapp001-backend
andmyuniqueapp001-frontend
should do the trick. - In
.appinfo
add (replacing the<...>
parts with the names you chose above):backend app name: <your backend app name> frontend app name: <your frontend app name> backend app url: https://<your backend app name>.fly.dev frontend app url: https://<your frontend app name>.fly.dev
Local Frontend Setup (Nuxt)
- Set up a Nuxt 3 project (from the app’s main folder):
npx nuxi@latest init frontend
- choose
npm
for package manager - choose
no
for initialize git repository
- choose
cd frontend
- Create a simple component and use it in
app.vue
:mkdir components
touch components/Hello.vue
<!--- frontend/components/Hello.vue --> <template> <p data-testid="frontend-message">Hello from Nuxt!</p> </template>
app.vue
<!--- frontend/app.vue --> <template> <Hello /> </template>
- Run the Nuxt app locally:
npm run dev
(athttp://localhost:3000
you should see “Hello from Nuxt!“)
- Stop the server by pressing
command + c
.
Vitest (Local)
We’ll setup Vitest here for component tests.
- Install Vitest and testing dependencies:
npm i --save-dev @nuxt/test-utils vitest @vue/test-utils happy-dom playwright-core
- Change your
nuxt.config.ts
to this:// frontend/nuxt.config.ts export default defineNuxtConfig({ modules: ['@nuxt/test-utils/module'] })
- Create a
vitest.config.ts
:touch vitest.config.ts
// frontend/vitest.config.ts import { defineVitestConfig } from '@nuxt/test-utils/config' export default defineVitestConfig({ })
- Let’s create our test directory and test file:
mkdir -p spec/components
touch spec/components/hello.nuxt.spec.ts
// frontend/spec/components/hello.nuxt.spec.ts import { it, expect } from 'vitest' import { mountSuspended } from '@nuxt/test-utils/runtime' import { Hello } from '#components' it('can mount some component', async () => { const component = await mountSuspended(Hello) expect(component.text()).toMatchInlineSnapshot( '"Hello from Nuxt!"' ) })
- Run Vitest:
npx vitest spec/components
(1 test should pass)cd ..
Vitest Docker
- From the root directory of our app, let’s create a
docker-compse.yml
file:touch docker-compose.yml
# docker-compose.yml services: vitest: image: cimg/node:18.18 working_dir: /app/frontend command: bash -c "npm ci && npx nuxi prepare && npx vitest spec/components" volumes: - .:/app
- Let’s run Vitest on our local Docker setup:
docker-compose run --rm vitest
Vitest CircleCI
Now from the root directory of our app, let’s setup Vitest for CircleCI.
- Let’s create a
.circleci/config.yml
, the config file for CircleCI:mkdir .circleci
touch .circleci/config.yml
jobs: vitest: docker: - image: cimg/node:18.18 steps: - checkout - run: name: Install dependencies command: cd frontend && npm ci - run: name: Generate Nuxt files command: cd frontend && npx nuxi prepare - run: name: Run Vitest command: cd frontend && npx vitest spec/components workflows: version: 2 test_workflow: jobs: - vitest
- You’ll need to deploy the whole app to github.
git add .
git commit -m "Add Vitest"
- Create a new public repo in the github UI
git branch -M main
- From the github UI, get the repo’s “web url” (the url that ends in
.git
, likehttps://github.com/mark-mcdermott/testingtestinghaaay.git
) git remote add origin <repo web url>
git push -u origin main
- Configure CircleCI for the new repo
- Login to CircleCI
- Click the “Go to application” button
- Click “Projects” in the left sidebar
- Find your new repo in the Project list
- In your repo’s project row, click “Set up Project” towards the right.
- The “Select your config.yml file” modal shows and “Fastest” is already the selected radio option
- In the “From which branch” field under “Fastest”, type
main
- Click the “Set up Project” button on the modal.
- This will take you to your new repo’s “Pipeline” and a run will have started
- You can watch the run and when it’s finished, the Vitest step should have passed and everything should be green.
Tweak Frontend To Talk To Backend
- Let’s configure our
nuxt.config.ts
to talk to the backend:cd frontend
frontend/nuxt.config.ts
(make sure to replace<...>
with your app backend name)
// frontend/nuxt.config.ts export default defineNuxtConfig({ server: { port: 3001, host: '0.0.0.0' }, runtimeConfig: { public: { apiURL: 'http://localhost:3000/api/v1'}}, $production: { runtimeConfig: { public: { apiURL: 'https://<app backend name>.fly.dev/api/v1' }}}, modules: ['@nuxt/test-utils/module'] })
- Let’s make our component talk to the backend as well:
<!--- frontend/components/Hello.vue --> <template> <p data-testid="backend-message">{{ message }}</p> <p data-testid="frontend-message">Hello from Nuxt!</p> </template> <script setup> import { ref, onMounted } from 'vue'; const runtimeConfig = useRuntimeConfig() const message = ref(''); onMounted(async () => { const response = await fetch(`${runtimeConfig.public.apiURL}/hello`); const data = await response.json(); message.value = data.message; }); </script>
- Let’s update our component test to match the new html:
// frontend/spec/components/hello.nuxt.spec.ts import { it, expect, vi } from 'vitest' import { mountSuspended } from '@nuxt/test-utils/runtime' import { Hello } from '#components' global.fetch = vi.fn(() => Promise.resolve({ json: () => Promise.resolve({ message: 'Hello from Rails!' }), }) ) it('can mount some component', async () => { const component = await mountSuspended(Hello) expect(component.html()).toMatchInlineSnapshot(` "<p data-testid="backend-message">Hello from Rails!</p> <p data-testid="frontend-message">Hello from Nuxt!</p>" `) })
- Before we setup the backend, let’s double check Vitest is still passing locally, on local Docker and on CircleCI.
- locally:
npx vitest spec/components
(1 test should pass)
- local docker:
cd ..
docker-compose build
docker-compose up
(1 test should pass)
- CircleCI:
git add .
git commit -m "Setup frontend to talk to backend"
git push
- Check the CircleCI build (vitest section should be green and passing)
- locally:
Local Backend Setup (Rails)
- Set up Rails API-only backend with RSpec:
rails new backend --api -T -d postgresql
cd backend
rm -rf .git
bundle add rack-cors
- in
backend/Gemfile
change line 3 (the ruby version line) toruby "~> 3.3"
- Create a simple controller:
mkdir -p app/controllers/api/v1
touch app/controllers/api/v1/hello_controller.rb
# backend/controllers/api/v1/hello_controller.rb module Api module V1 class HelloController < ApplicationController def index render json: { message: 'Hello from Rails!' } end end end end
- Set the routes:
# backend/config/routes.rb Rails.application.routes.draw do namespace :api do namespace :v1 do get 'hello', to: 'hello#index' end end end
- Set our CORS configuration the backend doesn’t block calls from the frontend:
config/initializers/cors.rb
(make sure to replace<...>
with your app frontend name)# backend/config/initializers/cors.rb Rails.application.config.middleware.insert_before 0, Rack::Cors do allow do origins "http://localhost:3001", "https://<app frontend name>.fly.dev" resource "*", headers: :any, methods: [:get, :post, :put, :patch, :delete, :options, :head] end end
- Database setup:
- Start postgreSQL on your computer (I use the desktop PostgreSQL app) and run it in the background
- run the database setup:
rails db:create db:migrate
- or if have a databases called
backend_development
andbackend_test
(like if you’ve already run through this tutorial before):
rails db:drop db:create db:migrate
- Change directory to the app’s root directory:
cd ..
Run App Locally
- Run the backend server in terminal 1:
cd backend
rails server
- Run the Nuxt app in terminal 2:
cd frontend
npm run dev
(athttp://localhost:3001
you should see “Hello from Rails!” and “Hello from Nuxt!“)- If you only see “Hello from Nuxt!” and not “Hello from Rails!”, don’t panic. In my experience, sometimes the backend starts up properly right away and other times it can take about 5 minutes.
- Stop the frontend and the backend servers by pressing
command + c
in their respective terminals.
Deploy To Prod (Fly.io)
- Here we’ll deploy our backend and frontend apps to Fly.io.
- In Fly.io, these will be two totally separate apps. We’ll have them setup to talk to each other, but they’ll be two separate apps listed in your Fly.io apps section.
- When you deploy the backend, there will also be a third app automatically created—your backend PostgreSQL database. This is what we want.
- Install Fly CLI and log in:
brew install flyctl
(only ifflyctl
is not installed yet)fly auth login
Deploy Frontend
- In your
frontend/package.json
in thescripts
section, add this line:"start": "nuxt start",
- Let’s add our Fly.io configuration file, mostly, so it won’t ask us what the app name and region are when we
fly launch
—it’s a little faster this way. Make sure to replace<...>
with your app frontend name. Also replacedfw
in theprimary_region
with your region code.touch fly.toml
# frontend/fly.toml app = '<app frontend name>' primary_region = 'dfw' [build] [http_service] internal_port = 3001 force_https = true auto_stop_machines = 'stop' auto_start_machines = true min_machines_running = 0 processes = ['app'] [[vm]] memory = '1gb' cpu_kind = 'shared' cpus = 1
fly launch
- When it asks, “A fly.toml file was found. Would you like to copy its configuration to the new app?”, press
y
- When it asks, “Do you want to tweak these settings before proceeding?”, press
N
or enter - When
fly launch
if finished, look at the output for “Visit your newly deployed app at<url>
” and copy/paste the url into the “frontend app url” line of your.appinfo
- When it asks, “A fly.toml file was found. Would you like to copy its configuration to the new app?”, press
- Go to your frontend app url in a browser.
- You should see “Hello from Nuxt!”, but not “Hello from Rails!”
- In the console, there will be a failed request to the backend (becuase they backend’s not deployed yet)
cd ..
into the app’s root directory
Deploy Backend
cd backend
- We’ll start by deleting the Docker files that Rails creates with
rails new
:rm Dockerfile
rm .dockerignore
- Let’s create our backend
fly.toml
file, our Fly.io configuration file for our backend, mostly, so it won’t ask us what the app name and region are when wefly launch
—it’s a little faster this way. Make sure to replace<...>
with your app backend name. Also replacedfw
in theprimary_region
with your region code.touch fly.toml
# backend/fly.toml app = '<app backend>' primary_region = 'dfw' console_command = '/rails/bin/rails console' [build] [deploy] release_command = './bin/rails db:prepare' [http_service] internal_port = 3000 force_https = true auto_stop_machines = 'stop' auto_start_machines = true min_machines_running = 0 processes = ['app'] [[vm]] memory = '1gb' cpu_kind = 'shared' cpus = 1 [[statics]] guest_path = '/rails/public' url_prefix = '/'
- Deploy the backend:
fly launch
- It’ll ask you some questions:
- “An existing fly.toml file was found. Would you like to copy its configuration to the new app?” Answer
y
- ”Do you want to tweak these settings before proceeding?” Answer
N
or hit enter - After this, the deployment starts. It can take a few minutes to finish and a lot of output will scroll down your screen like the Matrix. Watch this output—some of it’s important.
- Watch the output and look for the “Postgres cluster details”, which end with the line, “Save your credentials in a secure place — you won’t be able to see them again!” When you see it, copy and paste this section to your
.secrets
file. - When it asks, “Overwrite entrypoint?”, press ‘n’
- When it asks, “Overwrite fly.toml?”, press ‘n’
- “An existing fly.toml file was found. Would you like to copy its configuration to the new app?” Answer
- It’ll ask you some questions:
- Go to your frontend app url in a browser.
- You should see “Hello from Nuxt!” and “Hello from Rails!”
- In the console, there should be no 404 errors
cd ..
into the app’s root directory
Playwright (Local)
- First we’ll install Playwright:
cd frontend
npm install -D @playwright/test
npx playwright install
- Then we’ll configure Playwright:
touch playwright.config.ts
// frontend/playwright.config.ts import { defineConfig } from '@playwright/test'; export default defineConfig({ testDir: './spec/e2e', use: { baseURL: 'http://localhost:3001', browserName: 'chromium', headless: true, // Change to `false` to see the browser in action }, webServer: { command: 'npm run dev', port: 3001, reuseExistingServer: true, }, });
- Now we’ll create our Playwright test:
mkdir spec/e2e
touch spec/e2e/hello.spec.ts
// frontend/e2e/hello.spec.ts import { test, expect } from '@playwright/test'; test('frontend and backend are working', async ({ page }) => { await page.goto('/'); await expect(page.locator('[data-testid="frontend-message"]')).toHaveText('Hello from Nuxt!'); await expect(page.locator('[data-testid="backend-message"]')).toHaveText('Hello from Rails!'); });
- Now run Playwright locally:
- In a terminal pane in the backend directory, run
rails server
- In a terminal pane in the frontend directory, run
npm run dev
- In another terminal pane in the frontend directory, run
npx playwright test spec/e2e
(1 test should pass) - Stop the first two terminal panes with control + c.
- In a terminal pane in the backend directory, run
Playwright Docker
I was unable to get playwright working on docker on my computer. I ran into issues with ARM64 incompatability with the appropriate Docker images and wasn’t able to figure out a way around them. So just skip this part.
But if you’re feeling intrepid, chaning the docker-compose.yml
to something like this might get you partway there:
# docker-compose.yml
services:
playwright:
image: cimg/ruby:3.3-node
working_dir: /app/frontend
command: |
apt-get update && apt-get install -y \
libnss3 libnspr4 libatk1.0-0 libatk-bridge2.0-0 libatspi2.0-0 \
libxcomposite1 libxdamage1 libgbm1 libpango-1.0-0 libxrandr2 \
libcups2 libdrm2 libxshmfence1 libasound2 && \
npm ci && npx playwright install && npx playwright test spec/e2e
depends_on:
- nuxt
volumes:
- .:/app
db:
image: postgres:13
environment:
POSTGRES_USER: postgres
POSTGRES_PASSWORD: postgres
ports:
- "5432:5432"
rails:
image: cimg/ruby:3.3-node
working_dir: /app/backend
environment:
RAILS_ENV: test
command: bundle install && rails server -e test -p 3000
ports:
- "3000:3000"
volumes:
- .:/app
depends_on:
- db
nuxt:
image: cimg/node:18.18
working_dir: /app/frontend
command: npm ci && npx nuxi dev -p 3001
ports:
- "3001:3001"
volumes:
- .:/app
depends_on:
- rails
vitest:
image: cimg/node:18.18
working_dir: /app/frontend
command: bash -c "npm ci && npx nuxi prepare && npx vitest spec/components"
volumes:
- .:/app
Playwright On CircleCI
- From the root directory of our app, let’s change our
.circleci/config.yml
to include a Playwright section:# .circleci/config.yml jobs: vitest: docker: - image: cimg/node:18.18 steps: - checkout - run: name: Install dependencies command: cd frontend && npm ci - run: name: Generate Nuxt files command: cd frontend && npx nuxi prepare - run: name: Run Vitest command: cd frontend && npx vitest spec/components playwright: docker: - image: cimg/ruby:3.3-node environment: RAILS_ENV: test steps: - checkout - run: name: Install System Dependencies for Playwright command: | sudo apt-get update && sudo apt-get install -y \ libnss3 libnspr4 libatk1.0-0 libatk-bridge2.0-0 libatspi2.0-0 \ libxcomposite1 libxdamage1 libgbm1 libpango-1.0-0 libxrandr2 \ libcups2 libdrm2 libxshmfence1 libasound2 - run: name: Install Frontend Dependencies command: cd frontend && npm ci - run: name: Install Playwright Browsers command: cd frontend && npx playwright install - run: name: Install Bundler command: gem install bundler - run: name: Install Rails Gems command: cd backend && bundle install - run: name: Start Rails Backend command: cd backend && rails server -e test -p 3000 background: true - run: name: Start Nuxt Frontend command: cd frontend && npx nuxi dev -p 3001 background: true - run: name: Wait for Services to Start command: sleep 10 - run: name: Run Playwright Tests command: cd frontend && npx playwright test spec/e2e workflows: version: 2 test_workflow: jobs: - vitest - playwright
- Let’s commit these changes, push them up and run Playwright on CircleCI:
git add .
git commit -m "Add Playwright"
git push
- That should start the tests running and now Vitest and Playwright should pass and show green.
RSpec (Local)
Here we’ll create a simple RSpec test for the Rails HelloController. Then we’ll verify it passes when running locally.
- Install RSpec:
cd backend
bundle add rspec-rails --group "development, test"
rails generate rspec:install
- Create a HelloController test:
mkdir -p spec/requests/api/v1
touch spec/requests/api/v1/hello_spec.rb
# backend/spec/requests/api/v1/hello_spec.rb # frozen_string_literal: true require 'rails_helper' RSpec.describe "Api::V1::Hello", type: :request do describe "GET /api/v1/hello" do it "returns a JSON message from the hello controller" do get "/api/v1/hello" expect(response).to have_http_status(:ok) json_body = JSON.parse(response.body) expect(json_body["message"]).to eq("Hello from Rails!") end end end
- Run RSpec locally:
rspec
(1 test should run and pass)
cd ..
RSpec On Local Docker
- In the root directory of our app, let’s add
db
andrspec
services todocker-compose.yml
:# docker-compose.yml services: vitest: image: cimg/node:18.18 working_dir: /app/frontend command: bash -c "npm ci && npx nuxi prepare && npx vitest spec/components" volumes: - .:/app db: image: postgres:13 environment: POSTGRES_USER: postgres POSTGRES_PASSWORD: postgres ports: - "5432:5432" healthcheck: test: ["CMD", "pg_isready", "-U", "postgres"] interval: 5s timeout: 10s retries: 5 rspec: image: cimg/ruby:3.3-node working_dir: /app/backend environment: RAILS_ENV: test DB_HOST: db DB_USERNAME: postgres DB_PASSWORD: postgres DB_PORT: 5432 command: bash -c "./wait-for-it.sh db:5432 -- bundle install && bin/rails db:create db:migrate && rspec" volumes: - .:/app - ./wait-for-it.sh:/app/backend/wait-for-it.sh depends_on: db: condition: service_healthy
- Let’s download a
wait-for-it.sh
script to our app’s root directory and set the file permissions. This lets us wait for the database to be ready before running RSpec:curl -o wait-for-it.sh https://raw.githubusercontent.com/vishnubob/wait-for-it/master/wait-for-it.sh
chmod +x wait-for-it.sh
- In
backend/config/database.yml
, change thetest:
section to:test: <<: *default database: backend_test username: <%= ENV.fetch("DB_USERNAME", "postgres") %> password: <%= ENV.fetch("DB_PASSWORD", "postgres") %> host: <%= ENV.fetch("DB_HOST", "localhost") %> port: <%= ENV.fetch("DB_PORT", "5432") %>
- Run RSpec locally on Docker:
docker-compose down --volumes --remove-orphans
docker-compose run --rm rspec
(you should see in green1 example, 0 failures
)
- Make sure RSpec still works locally:
cd backend
rspec
(you should see in green1 example, 0 failures
here, too)cd ..
RSpec On CircleCI
- Let’s now add RSpec to our
.circleci/config.yml
:# .circleci/config.yml jobs: vitest: docker: - image: cimg/node:18.18 steps: - checkout - run: name: Install dependencies command: cd frontend && npm ci - run: name: Generate Nuxt files command: cd frontend && npx nuxi prepare - run: name: Run Vitest command: cd frontend && npx vitest spec/components playwright: docker: - image: cimg/ruby:3.3-node environment: RAILS_ENV: test steps: - checkout - run: name: Install System Dependencies for Playwright command: | sudo apt-get update && sudo apt-get install -y \ libnss3 libnspr4 libatk1.0-0 libatk-bridge2.0-0 libatspi2.0-0 \ libxcomposite1 libxdamage1 libgbm1 libpango-1.0-0 libxrandr2 \ libcups2 libdrm2 libxshmfence1 libasound2 - run: name: Install Frontend Dependencies command: cd frontend && npm ci - run: name: Install Playwright Browsers command: cd frontend && npx playwright install - run: name: Install Bundler command: gem install bundler - run: name: Install Rails Gems command: cd backend && bundle install - run: name: Start Rails Backend command: cd backend && rails server -e test -p 3000 background: true - run: name: Start Nuxt Frontend command: cd frontend && npx nuxi dev -p 3001 background: true - run: name: Wait for Services to Start command: sleep 10 - run: name: Run Playwright Tests command: cd frontend && npx playwright test spec/e2e rspec: docker: - image: cimg/ruby:3.3-node environment: RAILS_ENV: test - image: postgres:13 environment: POSTGRES_USER: postgres POSTGRES_PASSWORD: postgres POSTGRES_DB: backend_test steps: - checkout - run: name: Wait for Database command: | for i in {1..30}; do pg_isready -h db -p 5432 -U postgres && break || sleep 2; done - run: name: Install Dependencies command: | cd backend && gem install bundler && bundle install - run: name: Run Database Setup command: | cd backend && bin/rails db:create db:migrate - run: name: Run RSpec command: cd backend && rspec workflows: version: 2 test_workflow: jobs: - rspec - vitest - playwright
- Just commit your changes and push:
git add .
git commit -m "Add RSpec"
git push
(rspec, vitest and playwright should all be green and passing on CircleCI)
Redeploy To Fly.io
It’s time to redeploy our app and make sure all the changes we’ve been making haven’t broken prod. It’s ok to do these two steps at the same time if you want.
- Let’s redeploy our backend:
cd backend
fly deploy
- Let’s redeploy our frontend
cd .../frontend
fly deploy
- Go to your frontend app url in a browser.
- You should see “Hello from Nuxt!” and “Hello from Rails!”