Skip to content

Full Stack Web Pipeline

Now let's create a Full Stack Web App a little more complex than the previous designed to leverage a combination of technologies and not only just a mini API.

The project integrates a Firebase database with a Python FastAPI backend (we can leverage the config of the previous section) and served to users through a Vue.js frontend.

This architecture is chosen to facilitate the CRUD (Create, Read, Update, Delete) operations on a Story Firebase Document object within a Firebase collection, enabling dynamic content management and real-time data interaction.

Here our document are called Story with the following fileds :

class Story(BaseModel):
    audioURL: Optional[HttpUrl] = None
    promptThemes: Optional[List[PromptTheme]] = []
    timesSeen: Optional[int] = 0
    timesListen: Optional[int] = 0
    imageURL: Optional[List[HttpUrl]] = []
    id: Optional[str] = None
    story: Optional[str] = None
    isPrinted: Optional[bool] = False
    language: Optional[str] = None
    title: str 
    isFavorite: Optional[bool] = False

Project architecture

  • Vue.js (Frontend) : JavaScript framework used for building user interfaces and single-page applications in a fast and clean way. Vue is a component-based architecture and support the routing and state management. We will be using NuxtUI in order to speed up our work 😎

  • Python FastAPI (Backend) : Modern and fast python web framework for building APIs with Python 3.7+. The key features are its speed and ease of use, with automatic OpenAPI documentation generation.

  • Firebase (Database) : NoSQL database provided by Google, known for real-time data synchronization and automatic scaling. Flexible and schemaless design for the storage of complex nested objects.
  • Docker compose : In order to the network communication and expose our app through HTTPS with a simple nginx.conf file.
  • Cloudron : Minimalist web server with awersome applications and a docker ready deployment kit for reals 👨‍🍳
.
├── README.md
├── back
├── certs 
├── docker-compose.yml
├── front
├── my.secrets
└── nginx.conf

Firebase Backend

Let's begin with the python FastAPI backend and Firebase ! Our goal as you may know is to connect Firebase to our Python FastAPI backend and test it.

First go the official documentation, download and install the Firebase SDK to your project.

Do not forget to add the Firebase SDK to your requirements 🤓

Then add the Firebase initialisation code into your python FastAPI application like this :

from firebase_admin import credentials, firestore, initialize_app
# other import 

# your init code like env variable, init FastAPI app ... 
app = FastAPI()

# initialize Firebase 
cred = credentials.Certificate('./firebase-key.json')
initialize_app(cred)
db = firestore.client()
COLLECTION_NAME = "your-collection-name"

Now we want to test if the connection is good between firebase and our mini python API, let's create two endpoints for that :

  • @app.get("/") for testing purpose
  • @app.get("/stories/") in order to fetch all the story documents and return it in json format.

First Endpoints

Let's begin by adding some typing to our request with the Pydantic's BaseModel object like in the official documentation for our Firebase documents.

from fastapi import FastAPI, HTTPException, Response
from fastapi.middleware.cors import CORSMiddleware
from firebase_admin import credentials, firestore, initialize_app
import json, os
from pydantic import BaseModel, HttpUrl, ValidationError
from typing import List, Optional, Dict, Any
import logging

# Initialize FastAPI & log
app = FastAPI()
logger = logging.getLogger(__name__)  

# Configure CORS
app.add_middleware(
    CORSMiddleware,
    allow_origins=["*"],  # Allows all origins
    allow_credentials=True,
    allow_methods=["*"],  # Allows all methods
    allow_headers=["*"],  # Allows all headers
)

# Initialize Firebase 
cred = credentials.Certificate('./firebase-key.json')
initialize_app(cred)
db = firestore.client()
COLLECTION_NAME = "private-stories"


#
#   🧙🏼‍♂️ Helpers 🧙🏼‍♂️
#

# edit this class according to your firebase documents/collections
class Story(BaseModel):
    audioURL: Optional[HttpUrl] = None
    promptThemes: Optional[List[PromptTheme]] = []
    timesSeen: Optional[int] = 0
    timesListen: Optional[int] = 0
    imageURL: Optional[List[HttpUrl]] = []
    id: Optional[str] = None
    story: Optional[str] = None
    isPrinted: Optional[bool] = False
    language: Optional[str] = None
    title: str 
    isFavorite: Optional[bool] = False


#test http
@app.get("/")
async def test():
    return {"message": "OK"}

Now that we have a clean set up we can create the other endpoint @app.get("/stories/") for fetching all documents in the Firebase database.

If you are not comfortable with Cross-Origin Resource Sharing (CORS) configuration just take a look to the fastAPI documentation on this subject here

@app.get("/stories/") Endpoint

This endpoint look like a little more complexe than the other one 😅

No worries let's focus on the role of the FastAPI backend. It's all about receiving requests from the frontend, processes them, and performs the necessary CRUD operations on the Firebase database. It acts as an intermediary, ensuring data validation, authentication (if applicable), and business logic execution.

Processing data from Firebase

Processing data in our case is essential especially when dealing with data retrieved from a database like Firestore in a FastAPI application for: - Data Normalization: When you retrieve documents from a database, the data might not always be in the exact format you need for your application or API response (this is our case for the Null/None value). - Data Consistency: By processing the data before returning it to the client (our vue front), you ensure that the API's response is consistent, regardless of how the data is stored in the database. - Error Handling and Validation: Processing data allows you to validate the data and handle errors gracefully before sending it to the client. - Adaptation to Client Needs: Sometimes, data stored in a database contains more information than the client needs. Processing data allows you to reshape it, removing unnecessary fields or adding derived ones, to better match the client's requirements. - Security: Processing data before sending it out can also be a security measure. For example, if certain information should not be exposed to all clients, processing steps can filter out sensitive data based on the client's permissions.

For our case let's write a mini python functions called process_doc_data() for our particular case :

#
#   🧙🏼‍♂️ Helpers 🧙🏼‍♂️
#

def process_doc_data(doc_data: Dict[str, Any]) -> Dict[str, Any]:
    # Replace 'Null' string with None for audioURL
    if doc_data.get("audioURL") == "Null":
        doc_data["audioURL"] = None
    return doc_data

This process_doc_data() function normalizes the audioURL field by converting the string Null to Python's None. This is important because Null as a string is not the same as None in Python.

Now let's code our @app.get("/stories/") endpoint here :

@app.get("/stories/")
async def read_stories():
    try:
        docs = db.collection(COLLECTION_NAME).stream()
        stories = []
        for doc in docs:
            # Process each document data
            processed_data = process_doc_data(doc.to_dict())
            # Create a Story object and convert to a dictionary
            story = Story(**processed_data).dict()
            # Ensure URLs are serialized as strings
            if story.get('audioURL'):
                story['audioURL'] = str(story['audioURL'])
            for idx, url in enumerate(story.get('imageURL', [])):
                story['imageURL'][idx] = str(url)

            stories.append(story)

        return Response(content=json.dumps(stories, indent=4), media_type="application/json")

    except Exception as e:
        raise HTTPException(status_code=500, detail=str(e))

As you have seen, this method, read_stories is designed to retrieve and return a list of stories from a database. You can read the comment on the function and test it part by part in your own side for your data.

CRUD Endpoints

Now it's time to code (and test) our CRUD endpoints, we will not go into details into all the CRUD methods, it's pretty much the same methodology. Just be carful to your data, like if their are very specific you will use some pre-processing functions like we have seen with the process_doc_data() function.

# read
@app.get("/story/{story_id}",)
def read_story_by_id(story_id: str):
    try:
        doc_ref = db.collection(COLLECTION_NAME).document(story_id)
        doc = doc_ref.get()  
        # Check if the document exists
        if not doc.exists:
            return {"message": "Story not found"}
        return Response(content=json.dumps(doc.to_dict(), indent=4), media_type="application/json")
    except Exception as e:
        raise HTTPException(status_code=500, detail=str(e))

#create
@app.post("/story")
async def create_story(story: CreateStory):
    try:
        # Add a new document with an auto-generated ID
        doc_ref = db.collection(COLLECTION_NAME).document()
        story_data = story.dict(exclude_none=True)
        doc_ref.set(story_data)
        return {"id": doc_ref.id}  # Return the generated ID
    except Exception as e:
        raise HTTPException(status_code=500, detail=str(e))


#delete
@app.delete("/story/{story_id}", status_code=204)
async def delete_story(story_id: str):
    try:
        doc_ref = db.collection(COLLECTION_NAME).document(story_id)
        doc = doc_ref.get()
        if not doc.exists:
            raise HTTPException(status_code=404, detail="Story not found")

        doc_ref.delete()
        return {"detail": f"Story {doc_ref.id} deleted successfully"}
    except Exception as e:
        raise HTTPException(status_code=500, detail=str(e))

#upload file method --> bonus 
@app.post("/uploadfiles/")
async def create_upload_files(files: List[UploadFile] = File(...)):
    if BUCKET_NAME is not None:
        uploaded_file_urls = upload_to_gcp(files, BUCKET_NAME, "TestUploatFromWebSite")
        return {"filenames": uploaded_file_urls}
    return {"message": "Bucket not assign"}

 # update
@app.put("/story/{story_id}")
async def update_story(story_id: str, updated_story: Story):
    try:
        doc_ref = db.collection(COLLECTION_NAME).document(story_id)
        doc = doc_ref.get()
        story_data = doc.to_dict()
        if not doc.exists:
            raise HTTPException(status_code=404, detail="Story not found")
        # my specific conditions before updating 
        if (updated_story.title is not None) and (updated_story.story is not None):
            doc_ref.update(updated_story)
        else:
            raise HTTPException(status_code=400, detail="No updates provided")
        return {"id": doc_ref.id}

    except Exception as e:
        print(f"Error: {e}")
        raise HTTPException(status_code=500, detail=f"Server error: {str(e)}")

Docker Packaging

In this step we want to write the Dockerfile for the backend server and the docker-compose.yml file in order to add an nginx service then serve our backend through HTTPS as we said before.

Dockerfile

Let's start with the FastAPI base image here :

FROM tiangolo/uvicorn-gunicorn-fastapi:python3.9
WORKDIR /app
COPY . /app
RUN pip install --no-cache-dir firebase-admin pydantic

EXPOSE 8000
#COPY .env /app
# Set environment variables from .env file
ENV ENV_FILE_LOCATION=/app/.env

ENTRYPOINT [ "/app/start.sh" ]

Do not forget to add a .env file into your ./back folder

Docker-compose file

Here the docker-compose.yml file to expose our services (backend + frontend) and a nginx container for the SSL part.

version: '3.8'

services:
  frontend:
    build:
     context: ./front 
     dockerfile: Dockerfile 
   ports:
     - "3000:3000" 
   volumes:
     - ./front:/src 
   environment:
     - NODE_ENV=production

   healthcheck:
     test: [ "CMD", "curl", "-f", "http://0.0.0.0:3000/" ]
     interval: 50s
     timeout: 5s
     retries: 3

  backend:
    build:
      context: ./back 
      dockerfile: Dockerfile 
    ports:
      - "8000:8000" 
    volumes:
      - ./back:/app #path for live-reload
    environment:
      - ENV_FILE_LOCATION=/app/.env

    healthcheck:
      test: [ "CMD", "curl", "-f", "http://0.0.0.0:8000/" ]
      interval: 30s
      timeout: 10s
      retries: 3

  nginx:
    image: nginx:latest
    ports:
      - "80:80"
      - "443:443"
    volumes:
      - ./nginx.conf:/etc/nginx/nginx.conf
      - ./certs:/etc/ssl/certs
    depends_on:
      - backend
    restart: always
Add nginx proxy

Then we can generate our SSL certificate in order to serve our backendf application through HTTPS, with editing a simple nginx.conf file like this :

events {}

http {
  upstream backend {
    server backend:8000;
  }

  server {
    listen 80;
    listen [::]:80;
    server_name yourdomain.com;
    return 301 https://$host$request_uri;
  }

  server {
    listen 443 ssl http2;
    listen [::]:443 ssl http2;
    server_name yourdomain.com;

    ssl_certificate /etc/ssl/certs/cert.pem;
    ssl_certificate_key /etc/ssl/certs/key.pem;

    location / {
      proxy_pass http://backend;
      proxy_set_header Host $host;
      proxy_set_header X-Real-IP $remote_addr;
      proxy_set_header X-Forwarded-Proto https;
      proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for;
    }
  }
}

Do not forget to generate a free certificate ! if you are not comfortable with letsencrypt go to the Docker for app deployment section of this course then go to the Docker and HTTPS article you will find all the necessary instructions to handle your SSL key generation.

Tests with pytest

We can test our backend and the firebase connection with pytest by testing one by one each endpoints like we did before in the previous part. Here is a simple example of endpoint testing with pytest. Like in the github ci-cd previous part we can rely on the documentation

from fastapi.testclient import TestClient
from .main import app

client = TestClient(app)

def test_read_main():
    response = client.get("/")
    assert response.status_code == 200
    assert response.json() == {"message": "OK"}

In this project we will not be testing all endpointsour backend directly with the pytest library. We will be using cypress End to End Test in order to discover an other approach for testing. We will see this into the front testing part 🤓

Frontend

For this front part, we will be using vue.js and the Nuxt UI framework in order to speed up our devlopement and leverage the Nuxt components.

Frontend architecture

.
├── front
   ├── CloudronManifest.json   #our deployment file for Cloudron  
   ├── Dockerfile
   ├── api.js
   ├── app.config.ts
   ├── app.vue
   ├── assets
   ├── components
   ├── composables
   ├── cypress #our tests will go there 
   ├── cypress.config.ts
   ├── icon.png        #our favicon 
   ├── my.secrets  #our secrets for testing git workflow in local
   ├── node_modules
   ├── nuxt.config.ts
   ├── package-lock.json
   ├── package.json
   ├── pages
   ├── plugins
   ├── public
   ├── server
   ├── tailwind.config.ts
   ├── tsconfig.json
└───└── types

Quickstart Nuxt UI

You can check at the Nuxt UI official guide here and install it with your favourite package manager npm or yarn for example.

Then the good thing to do is to install some modules like tailwindcss in order tot deal with css and other needed plugins. For example in my case, I have several plugings :

  • Algolia to put a search bar in my website
  • VueFire to link our Firebase database to our front
  • Icons and flags to add more style to our page 😎

For this you need to edit your nuxt.config.ts file at the project root like this :

// https://nuxt.com/docs/api/configuration/nuxt-config
export default defineNuxtConfig({
  devtools: { enabled: true },
  modules: [
    '@nuxt/ui',
    '@nuxtjs/algolia',
    '@vueuse/nuxt',
    'nuxt-vuefire',
  ],
  runtimeConfig: {
    firebase: {
      bucketName: 'BUCKET_NAME',
      pageSizeMax: 100,
      pageSizeDefault: 10,
    },
    deepL: {
      apiKey: 'DEEPL_API_KEY',
    },
  },
  css: [
    '@/assets/css/scrollbars.css',
  ],
  algolia: {
    apiKey: process.env.ALGOLIA_SEARCH_API_KEY,
    applicationId: process.env.ALGOLIA_APPLICATION_ID,
    instantSearch: {
      theme: 'algolia',
    },
  },
  ui: {
    icons: ['mdi', 'flag'],
  },
  vuefire: {
    config: {
    },
  },
})

As uyou can see I used a .env file to not expose the secrets on github, I recommend you to do the same 😇

NuxtUI consists of the following concepts :

  • Base components - simple components to build on top, see documentation
  • Plugings - package to simplify usage and customization of third-party, see documentation
  • Themes - a configuration file to keep your styling consistent across the app, see documentation
  • Composables - functions to help you build an interactive experience for the users, see documentation

We will not go into details on how to code the full front NuxtUI website, I will just show you the final result here :

As you can see I used cards to display my firebase data informations. Thanks to my friend and mentor Christophe Vilas Boas who help me a lot with this project 😇

Continuous Integration

For this continuous integration part, we will use Cypress for End to End testing and github action with the cypress plugin in order to speed up our tests and deployment 😎

This is the cypress.yaml test file I used in order to test our application on each pull request like good devs 😇

name: Cypress Tests

on:
  pull_request:
    branches: [ "main" ]
  workflow_call: # allow this workflow to be called from other workflows
    secrets:
      GOOGLE_APPLICATION_CREDENTIALS:
        required: true
      NUXT_FIREBASE_BUCKET_NAME:
        required: true
      ALGOLIA_API_KEY:
        required: true
      ALGOLIA_APPLICATION_ID:
        required: true

jobs:
  cypress_tests:
    name: Cypress Tests
    runs-on: ubuntu-latest

    steps:
      - name: Checkout
        uses: actions/checkout@v4

      - name: Setup Node
        uses: actions/setup-node@v4
        with: 
          node-version: '20'

      - name: Install Dependencies
        run: yarn install --frozen-lockfile

      - name: Start Server
        env:
          GOOGLE_APPLICATION_CREDENTIALS: ${{ secrets.GOOGLE_APPLICATION_CREDENTIALS }}
          NUXT_FIREBASE_BUCKET_NAME: ${{ secrets.NUXT_FIREBASE_BUCKET_NAME }}
          ALGOLIA_API_KEY: ${{secrets.ALGOLIA_API_KEY}}
          ALGOLIA_APPLICATION_ID: ${{secrets.ALGOLIA_APPLICATION_ID}}
        run: yarn dev --host & sleep 30

      - name: Run Cypress Tests
        run: npx cypress run

I will not go into all the tests because it will be very long. I will just drop here some test template I used for e2e testing of the frontend app part and the /api backend part here :

// cypress/e2e/spec.cy.js
describe('App', () => {
  it('home-page', () => {
    cy.visit('/', {
      failOnStatusCode: true,
    })

    // A StoryCard should exist
    cy.get('[data-test="StoryCard"]').should('exist')
  })
  it('api', () => {
    cy.request({
      method: 'GET',
      url: '/api',
    }).then((response) => {
      // Expecting the response status code to be 200
      expect(response.status).to.eq(200)
      expect(response.body.message).to.eq('OK')
    })
  })
})

and for the backend part. This is a short version because there is a lot to test like the all CRUD operations :

// cypress/e2e/spec.cy.js
describe('Server API', () => {
  it('Up', () => {
    cy.request({
      method: 'GET',
      url: '/api',
    }).then((response) => {
      // Expecting the response status code to be 200
      expect(response.status, 'response.status').to.eq(200)
      expect(response.body.message, 'response.body.message').to.eq('OK')
    })
  })

I think you see the point here, it is not very long to write but it take some time to identify all the test cases scenarios and implement it without bug 🤓

Cloudron auto deployment

We will use github action to auto deploy our solution on Cloudron with this yml bellow here :

Cloudron Auto Deploy
name: CloudRon Auto Deployment

on:
  push:
    branches: [ "main" ]

jobs:
  cypress_tests:
    name: Cypress Tests
    uses: ./.github/workflows/cypress.yml
    secrets:
      GOOGLE_APPLICATION_CREDENTIALS: ${{ secrets.GOOGLE_APPLICATION_CREDENTIALS }}
      NUXT_FIREBASE_BUCKET_NAME: ${{ secrets.NUXT_FIREBASE_BUCKET_NAME }}
      ALGOLIA_API_KEY: ${{ secrets.ALGOLIA_API_KEY }}
      ALGOLIA_APPLICATION_ID: ${{ secrets.ALGOLIA_APPLICATION_ID }}

  build_push_to_registry:
    name: Build & Push Docker image to Docker Hub
    runs-on: ubuntu-latest
    needs: cypress_tests
    steps:
      - uses: actions/checkout@v4

      - name: Build docker image
        run: docker build . --file Dockerfile --tag docker.io/fairycorp/nuxt-fabla:gh-${{github.run_number}}

      - name: Login to Docker Hub
        uses: docker/login-action@v3
        with:
          username: ${{ secrets.DOCKER_USERNAME }}
          password: ${{ secrets.DOCKER_PASSWORD }}

      - name: Push to Docker Hub
        run: docker push docker.io/fairycorp/nuxt-fabla:gh-${{github.run_number}}

  deploy:
    name: Deploy to Cloudron
    runs-on: ubuntu-latest
    needs: build_push_to_registry
    steps:
      - uses: actions/checkout@v4

      - name: Environment Setup
        uses: actions/setup-node@v4
        with:
          node-version: '20'

      - name: Deploy setup
        run: npm install -g cloudron

      - name: Update App
        run: |
          update="cloudron update \
            --server ${{ secrets.CLOUDRON_SERVER }} \
            --token ${{ secrets.CLOUDRON_TOKEN }} \
            --app ${{ secrets.CLOUDRON_APP }} \
            --image docker.io/fairycorp/nuxt-fabla:gh-${{github.run_number}}"
          # Retry up to 5 times (with linear backoff)
          NEXT_WAIT_TIME=0
          until [ $NEXT_WAIT_TIME -eq 5 ] || $update; do
              sleep $(( NEXT_WAIT_TIME++ ))
          done
          [ $NEXT_WAIT_TIME -lt 5 ]

This should looks like this diagram here :

Wrap it up

In this article we have seen how to set up a full stack web project and how we can auto deploy it with cloudron and github action. Like I told you earlier, I have not went into details about the frontend

And that's it you have the complete full stack web pipeline, hope you have learn a thing or two in this article 🥳