perf: optimize CI pipeline — eliminate double dependency installs, use local registry cache

- Frontend Dockerfile: split into deps/test/builder/nginx stages so npm ci
  runs once (cached when package-lock.json unchanged), tests run in build
- Backend Dockerfile: add test stage that runs pytest inside the build,
  eliminating separate test image build
- .drone.yml: remove separate test steps (now inside Dockerfile builds),
  point cache_from/cache_repo at local registry (10.0.20.10:5000) instead
  of Docker Hub for faster layer cache pulls
This commit is contained in:
Viktor Barzin 2026-02-21 15:10:55 +00:00
parent 68859ae577
commit b1be4d4170
No known key found for this signature in database
GPG key ID: 0EB088298288D958
3 changed files with 68 additions and 49 deletions

View file

@ -30,17 +30,7 @@ steps:
exit 1
- git checkout "$DRONE_COMMIT"
- name: Run frontend tests
image: node:24-alpine
resources:
limits:
memory: 2048MiB
commands:
- cd frontend
- npm ci
- npx vitest run
- name: Build frontend image
- name: Build and test frontend image
image: plugins/kaniko
resources:
limits:
@ -53,10 +43,30 @@ steps:
dockerfile: frontend/Dockerfile
context: frontend
enable_cache: true
cache_repo: viktorbarzin/immoweb-cache
cache_repo: 10.0.20.10:5000/immoweb-cache
registry: 10.0.20.10:5000
insecure: true
tags:
- latest
- ${DRONE_BUILD_NUMBER}
- "${DRONE_BUILD_NUMBER}"
- name: Push to Docker Hub
image: plugins/kaniko
resources:
limits:
memory: 512MiB
settings:
username: viktorbarzin
password:
from_secret: dockerhub-token
repo: viktorbarzin/immoweb
dockerfile: frontend/Dockerfile
context: frontend
enable_cache: true
cache_repo: 10.0.20.10:5000/immoweb-cache
tags:
- latest
- "${DRONE_BUILD_NUMBER}"
- name: Update deployment
image: alpine
@ -131,42 +141,38 @@ steps:
exit 1
- git checkout "$DRONE_COMMIT"
- name: Cache test image
- name: Build and test API image
image: plugins/docker
settings:
username: viktorbarzin
password:
from_secret: dockerhub-token
repo: viktorbarzin/realestatecrawler
repo: 10.0.20.10:5000/viktorbarzin/realestatecrawler
dockerfile: Dockerfile
context: .
target: test
insecure: true
cache_from:
- viktorbarzin/realestatecrawler:test
- viktorbarzin/realestatecrawler:builder
tags:
- test
- name: Run backend tests
image: viktorbarzin/realestatecrawler:test
commands:
- pytest tests/ -x -q
- name: Build API image
image: plugins/docker
settings:
username: viktorbarzin
password:
from_secret: dockerhub-token
repo: viktorbarzin/realestatecrawler
dockerfile: Dockerfile
context: .
cache_from:
- viktorbarzin/realestatecrawler:builder
- viktorbarzin/realestatecrawler:latest
- 10.0.20.10:5000/viktorbarzin/realestatecrawler:latest
- 10.0.20.10:5000/viktorbarzin/realestatecrawler:builder
tags:
- latest
- ${DRONE_BUILD_NUMBER}
- builder
- "${DRONE_BUILD_NUMBER}"
- name: Push to Docker Hub
image: plugins/docker
settings:
username: viktorbarzin
password:
from_secret: dockerhub-token
repo: viktorbarzin/realestatecrawler
dockerfile: Dockerfile
context: .
cache_from:
- 10.0.20.10:5000/viktorbarzin/realestatecrawler:latest
tags:
- latest
- "${DRONE_BUILD_NUMBER}"
- name: Update deployment
image: alpine

View file

@ -28,7 +28,7 @@ RUN apt-get update && apt-get install -y --no-install-recommends \
libmariadb3 \
&& rm -rf /var/lib/apt/lists/*
# Stage 3: Test image — runtime deps + venv + test dependencies
# Stage 3: Test — runtime deps + venv + test dependencies + run tests
FROM runtime-base AS test
WORKDIR /app
@ -38,6 +38,10 @@ ENV PATH="/app/.venv/bin:$PATH"
RUN pip install --no-cache-dir pytest pytest-asyncio pytest-xdist httpx aioresponses fakeredis
COPY . .
RUN pytest tests/ -x -q
# Stage 4: Final image — combine venv from builder + runtime base
FROM runtime-base

View file

@ -1,22 +1,32 @@
# Stage 1: Build the React app
FROM node:24-alpine AS builder
# Stage 1: Install dependencies (cached if package-lock.json unchanged)
FROM node:24-alpine AS deps
WORKDIR /app
# Limit Node.js heap to avoid OOM in constrained CI environments
ENV NODE_OPTIONS="--max-old-space-size=1024"
# Copy package files first for better caching
# Copy package files first for better layer caching
COPY package.json package-lock.json* ./
RUN npm ci
# Copy all files and build
# Stage 2: Run tests (fails the build if tests fail)
FROM deps AS test
COPY . .
# Skip tsc type-checking (already done in test step); Vite transpiles via SWC
RUN npx vite build
RUN npx vitest run
# Stage 3: Build production bundle
FROM deps AS builder
COPY . .
# Skip tsc type-checking (vitest already validated); Vite transpiles via SWC
RUN npx vite build
# Stage 4: Serve with nginx
FROM nginx:alpine
# Remove default nginx static files
@ -24,9 +34,8 @@ RUN rm -rf /usr/share/nginx/html/*
WORKDIR /app
# Copy only necessary files from the builder stage
COPY --from=builder /app/dist /usr/share/nginx/html
COPY --from=builder /app/nginx.conf /etc/nginx/conf.d/default.conf
COPY --from=builder /app/dist /usr/share/nginx/html
COPY --from=builder /app/nginx.conf /etc/nginx/conf.d/default.conf
EXPOSE 80
CMD ["nginx", "-g", "daemon off;"]