diff --git a/.github/workflows/clean-cicd.yml b/.github/workflows/clean-cicd.yml new file mode 100644 index 000000000..7e1be00b5 --- /dev/null +++ b/.github/workflows/clean-cicd.yml @@ -0,0 +1,68 @@ + +name: Clean CI/CD Pipeline + +on: + push: + branches: + - clean + paths-ignore: + - .github/workflows/** + +concurrency: + group: ${{ github.ref }} + cancel-in-progress: true + +jobs: + build: + runs-on: ubuntu-latest + + steps: + - name: Checkout code + uses: actions/checkout@v3 + + - name: Set up Node.js + uses: actions/setup-node@v3 + with: + node-version: "20" + + - name: Cache pnpm modules + uses: actions/cache@v3 + with: + path: ~/.pnpm-store + key: ${{ runner.os }}-pnpm-${{ hashFiles('**/pnpm-lock.yaml') }} + restore-keys: | + ${{ runner.os }}-pnpm- + + - name: Install pnpm + uses: pnpm/action-setup@v4 + with: + version: 9 + + - name: Install dependencies + run: pnpm install + + - name: Lint code + run: pnpm lint + + - name: Build email + run: pnpm email:build + + - name: Build project + run: pnpm build + + - name: Run tests + run: pnpm run test:ci + + deploy: + runs-on: ubuntu-latest + needs: build + + steps: + - name: Deploy to dev environment + uses: appleboy/ssh-action@v1.0.3 + with: + host: ${{ secrets.HOST }} + username: ${{ secrets.USERNAME }} + password: ${{ secrets.PASSWORD }} + script: | + ./deploy_dev.sh next \ No newline at end of file diff --git a/.github/workflows/dev-cicd.yml b/.github/workflows/dev-cicd.yml new file mode 100644 index 000000000..bc02c8e0d --- /dev/null +++ b/.github/workflows/dev-cicd.yml @@ -0,0 +1,68 @@ + +name: Development CI/CD Pipeline + +on: + push: + branches: + - dev + paths-ignore: + - .github/workflows/** + +concurrency: + group: ${{ github.ref }} + cancel-in-progress: true + +jobs: + build: + runs-on: ubuntu-latest + + steps: + - name: Checkout code + uses: actions/checkout@v3 + + - name: Set up Node.js + uses: actions/setup-node@v3 + with: + node-version: "20" + + - name: Cache pnpm modules + uses: actions/cache@v3 + with: + path: ~/.pnpm-store + key: ${{ runner.os }}-pnpm-${{ hashFiles('**/pnpm-lock.yaml') }} + restore-keys: | + ${{ runner.os }}-pnpm- + + - name: Install pnpm + uses: pnpm/action-setup@v4 + with: + version: 9 + + - name: Install dependencies + run: pnpm install + + - name: Lint code + run: pnpm lint + + - name: Build email + run: pnpm email:build + + - name: Build project + run: pnpm build + + - name: Run tests + run: pnpm run test:ci + + deploy: + runs-on: ubuntu-latest + needs: build + + steps: + - name: Deploy to dev environment + uses: appleboy/ssh-action@v1.0.3 + with: + host: ${{ secrets.HOST }} + username: ${{ secrets.USERNAME }} + password: ${{ secrets.PASSWORD }} + script: | + ./deploy_dev.sh next diff --git a/.github/workflows/prod-cicd.yml b/.github/workflows/prod-cicd.yml new file mode 100644 index 000000000..43feefc62 --- /dev/null +++ b/.github/workflows/prod-cicd.yml @@ -0,0 +1,67 @@ +name: Production CI/CD Pipeline + +on: + push: + branches: + - main + paths-ignore: + - .github/workflows/** + +concurrency: + group: ${{ github.ref }} + cancel-in-progress: true + +jobs: + build: + runs-on: ubuntu-latest + + steps: + - name: Checkout code + uses: actions/checkout@v3 + + - name: Set up Node.js + uses: actions/setup-node@v3 + with: + node-version: "20" + + - name: Cache pnpm modules + uses: actions/cache@v3 + with: + path: ~/.pnpm-store + key: ${{ runner.os }}-pnpm-${{ hashFiles('**/pnpm-lock.yaml') }} + restore-keys: | + ${{ runner.os }}-pnpm- + + - name: Install pnpm + uses: pnpm/action-setup@v4 + with: + version: 9 + + - name: Install dependencies + run: pnpm install + + - name: Lint code + run: pnpm lint + + - name: Build email + run: pnpm email:build + + - name: Build project + run: pnpm build + + - name: Run tests + run: pnpm run test:ci + + deploy: + runs-on: ubuntu-latest + needs: build + + steps: + - name: Deploy to prod environment + uses: appleboy/ssh-action@v1.0.3 + with: + host: ${{ secrets.HOST }} + username: ${{ secrets.USERNAME }} + password: ${{ secrets.PASSWORD }} + script: | + ./deploy_prod.sh next \ No newline at end of file diff --git a/.github/workflows/staging-cicd.yml b/.github/workflows/staging-cicd.yml new file mode 100644 index 000000000..5745d2ab2 --- /dev/null +++ b/.github/workflows/staging-cicd.yml @@ -0,0 +1,67 @@ +name: Staging CI/CD Pipeline + +on: + push: + branches: + - staging + paths-ignore: + - .github/workflows/** + +concurrency: + group: ${{ github.ref }} + cancel-in-progress: true + +jobs: + build: + runs-on: ubuntu-latest + + steps: + - name: Checkout code + uses: actions/checkout@v3 + + - name: Set up Node.js + uses: actions/setup-node@v3 + with: + node-version: "20" + + - name: Cache pnpm modules + uses: actions/cache@v3 + with: + path: ~/.pnpm-store + key: ${{ runner.os }}-pnpm-${{ hashFiles('**/pnpm-lock.yaml') }} + restore-keys: | + ${{ runner.os }}-pnpm- + + - name: Install pnpm + uses: pnpm/action-setup@v4 + with: + version: 9 + + - name: Install dependencies + run: pnpm install + + - name: Lint code + run: pnpm lint + + - name: Build email + run: pnpm email:build + + - name: Build project + run: pnpm build + + - name: Run tests + run: pnpm run test:ci + + deploy: + runs-on: ubuntu-latest + needs: build + + steps: + - name: Deploy to staging environment + uses: appleboy/ssh-action@v1.0.3 + with: + host: ${{ secrets.HOST }} + username: ${{ secrets.USERNAME }} + password: ${{ secrets.PASSWORD }} + script: | + ./deploy_staging.sh next \ No newline at end of file diff --git a/docker/development/Dockerfile b/docker/development/Dockerfile new file mode 100644 index 000000000..2c54c36c3 --- /dev/null +++ b/docker/development/Dockerfile @@ -0,0 +1,56 @@ +# Use the official Node.js image based on Alpine Linux for a smaller image size +FROM node:20-alpine AS base + +FROM base AS deps + +# Install build dependencies +RUN apk add --no-cache libc6-compat + +# Set the working directory +WORKDIR /app + +# Copy package.json and pnpm-lock.yaml +COPY package.json pnpm-lock.yaml ./ + +# Install dependencies +RUN corepack enable pnpm && pnpm install --frozen-lockfile + +# Rebuild the source code only when needed +FROM base AS builder +WORKDIR /app +COPY --from=deps /app/node_modules ./node_modules +COPY . . + +# Build the application +RUN corepack enable pnpm && pnpm run build + + +# Production image, copy all the files and run next +FROM base AS runner +WORKDIR /app + +ENV NODE_ENV production +# Uncomment the following line in case you want to disable telemetry during runtime. +# ENV NEXT_TELEMETRY_DISABLED 1 + +RUN addgroup --system --gid 1001 nodejs +RUN adduser --system --uid 1001 nextjs + +# Copy necessary files for production +COPY --from=builder /app/public ./public +COPY --from=builder /app/.next/standalone ./ +COPY --from=builder /app/.next/static ./.next/static + +# Set the correct permission for prerender cache +RUN mkdir -p .next +RUN chown nextjs:nodejs .next + +USER nextjs + +EXPOSE 3000 + +ENV PORT 3000 + +# server.js is created by next build from the standalone output +# https://nextjs.org/docs/pages/api-reference/next-config-js/output +CMD HOSTNAME="0.0.0.0" node server.js \ No newline at end of file diff --git a/docker/development/docker-compose.yml b/docker/development/docker-compose.yml new file mode 100644 index 000000000..f9cbfb510 --- /dev/null +++ b/docker/development/docker-compose.yml @@ -0,0 +1,17 @@ +services: + + next-dev-frontend-1: + build: + context: ../../ + dockerfile: docker/development/Dockerfile + ports: + - 3100:3000 + restart: always + + next-dev-frontend-2: + build: + context: ../../ + dockerfile: docker/development/Dockerfile + ports: + - 3130:3000 + restart: always diff --git a/docker/prod/Dockerfile b/docker/prod/Dockerfile new file mode 100644 index 000000000..a0d1e4139 --- /dev/null +++ b/docker/prod/Dockerfile @@ -0,0 +1,55 @@ +Use the official Node.js image based on Alpine Linux for a smaller image size +FROM node:20-alpine AS base + +FROM base AS deps + +# Install build dependencies +RUN apk add --no-cache libc6-compat + +# Set the working directory +WORKDIR /app + +# Copy package.json and pnpm-lock.yaml +COPY package.json pnpm-lock.yaml ./ + +# Install dependencies +RUN corepack enable pnpm && pnpm install --frozen-lockfile + +# Rebuild the source code only when needed +FROM base AS builder +WORKDIR /app +COPY --from=deps /app/node_modules ./node_modules +COPY . . + +# Build the application +RUN corepack enable pnpm && pnpm run build + +# Production image, copy all the files and run next +FROM base AS runner +WORKDIR /app + +ENV NODE_ENV development +# Uncomment the following line in case you want to disable telemetry during runtime. +# ENV NEXT_TELEMETRY_DISABLED 1 + +RUN addgroup --system --gid 1001 nodejs +RUN adduser --system --uid 1001 nextjs + +# Copy necessary files for production +COPY --from=builder /app/public ./public +COPY --from=builder /app/.next/standalone ./ +COPY --from=builder /app/.next/static ./.next/static + +# Set the correct permission for prerender cache +RUN mkdir -p .next +RUN chown nextjs:nodejs .next + +USER nextjs + +EXPOSE 3000 + +ENV PORT 3000 + +# server.js is created by next build from the standalone output +# https://nextjs.org/docs/pages/api-reference/next-config-js/output +CMD HOSTNAME="0.0.0.0" node server.js \ No newline at end of file diff --git a/docker/prod/docker-compose.yml b/docker/prod/docker-compose.yml new file mode 100644 index 000000000..b3bd96b55 --- /dev/null +++ b/docker/prod/docker-compose.yml @@ -0,0 +1,17 @@ +services: + + next-prod-frontend-1: + build: + context: ../../ + dockerfile: docker/prod/Dockerfile + ports: + - 3000:3000 + restart: always + + next-prod-frontend-2: + build: + context: ../../ + dockerfile: docker/prod/Dockerfile + ports: + - 3030:3000 + restart: always diff --git a/docker/staging/Dockerfile b/docker/staging/Dockerfile new file mode 100644 index 000000000..a0d1e4139 --- /dev/null +++ b/docker/staging/Dockerfile @@ -0,0 +1,55 @@ +Use the official Node.js image based on Alpine Linux for a smaller image size +FROM node:20-alpine AS base + +FROM base AS deps + +# Install build dependencies +RUN apk add --no-cache libc6-compat + +# Set the working directory +WORKDIR /app + +# Copy package.json and pnpm-lock.yaml +COPY package.json pnpm-lock.yaml ./ + +# Install dependencies +RUN corepack enable pnpm && pnpm install --frozen-lockfile + +# Rebuild the source code only when needed +FROM base AS builder +WORKDIR /app +COPY --from=deps /app/node_modules ./node_modules +COPY . . + +# Build the application +RUN corepack enable pnpm && pnpm run build + +# Production image, copy all the files and run next +FROM base AS runner +WORKDIR /app + +ENV NODE_ENV development +# Uncomment the following line in case you want to disable telemetry during runtime. +# ENV NEXT_TELEMETRY_DISABLED 1 + +RUN addgroup --system --gid 1001 nodejs +RUN adduser --system --uid 1001 nextjs + +# Copy necessary files for production +COPY --from=builder /app/public ./public +COPY --from=builder /app/.next/standalone ./ +COPY --from=builder /app/.next/static ./.next/static + +# Set the correct permission for prerender cache +RUN mkdir -p .next +RUN chown nextjs:nodejs .next + +USER nextjs + +EXPOSE 3000 + +ENV PORT 3000 + +# server.js is created by next build from the standalone output +# https://nextjs.org/docs/pages/api-reference/next-config-js/output +CMD HOSTNAME="0.0.0.0" node server.js \ No newline at end of file diff --git a/docker/staging/docker-compose.yml b/docker/staging/docker-compose.yml new file mode 100644 index 000000000..96ed1f242 --- /dev/null +++ b/docker/staging/docker-compose.yml @@ -0,0 +1,17 @@ +services: + + next-staging-frontend-1: + build: + context: ../../ + dockerfile: docker/staging/Dockerfile + ports: + - 3200:3000 + restart: always + + # next-staging-frontend-2: + # build: + # context: ../../ + # dockerfile: docker/staging/Dockerfile + # ports: + # - 3230:3000 + # restart: alwaysraven@frontend:/home/nextjs/staging/hng_boilerplate_nextjs/docker/staging$