ZyVOP Logo
Content That Connects
SeriesCategoriesTags
ZyVOP Logo
Content That Connects

Empowering developers and creators with cutting-edge insights, comprehensive tutorials, and innovative solutions for the digital future.

Content

  • Tags
  • Write Article

Company

  • About Us
  • Contact

Connect

  • Privacy Policy
  • Terms of Service
  • Cookie Policy
  • DMCA Policy
  • Code of Conduct

© 2026 ZyVOP. Crafted with care for the developer community.

Made with ❤️ by the ZyVOP team
All systems operational
HomeDitch Vercel: A Complete Guide to Auto-Deploying Next.js to a VPS via GitHub Actions
👍1

Ditch Vercel: A Complete Guide to Auto-Deploying Next.js to a VPS via GitHub Actions

Build a production-grade CI/CD pipeline for Next.js using GitHub Actions, Docker, PM2, Nginx, and a self-hosted VPS — without platform lock-in or unpredictable hosting costs.

#Vercel Alternative#Next.js#VPS Hosting#GitHub Actions#CI/CD#Deployment Automation
Z
ZyVOP

Senior Developer

May 22, 2026
5 min read
12 views
Ditch Vercel: A Complete Guide to Auto-Deploying Next.js to a VPS via GitHub Actions

Vercel provides an unmatched developer experience. You push to main, and your site is live. But as your startup grows, Vercel's pricing limits—especially regarding serverless function execution time, background jobs, and bandwidth—can quickly force you into hundreds of dollars a month.

Moving to a cheap Virtual Private Server (VPS) on DigitalOcean, Hetzner, or AWS EC2 is the logical next step. A $10/month VPS can handle millions of requests a month. But giving up the "git push to deploy" magic is a tough pill to swallow.

In this guide, we are going to recreate that magic. We will configure GitHub Actions to securely SSH into your server, pull the code, utilize Docker build caching for blazing-fast builds, and securely inject your .env variables.


Prerequisites

  1. A Linux VPS with Docker and Docker Compose installed.

  2. Your application codebase on the server, running via docker-compose.yml.

  3. Your codebase hosted on GitHub.

Step 1: Secure Server Access via SSH Keys

GitHub Actions needs permission to log into your server. We will not use a password. We will generate a dedicated SSH key pair exclusively for GitHub.

SSH into your VPS and generate a new key:

ssh-keygen -t ed25519 -f ~/.ssh/github_actions_key -C "github_actions"

When prompted for a passphrase, leave it empty (press Enter).

Now, authorize this key to access your server by adding the public key to your authorized_keys file:

cat ~/.ssh/github_actions_key.pub >> ~/.ssh/authorized_keys

Display the private key and copy the entire output (including the BEGIN and END lines):

cat ~/.ssh/github_actions_key

Step 2: Configure GitHub Secrets

We must never hardcode server IPs or SSH keys into our codebase.

Go to your repository on GitHub. Navigate to Settings -> Secrets and variables -> Actions -> New repository secret.

Add the following three secrets:

  • SERVER_HOST: Your server's public IP address (e.g., 203.0.113.50).

  • SERVER_USERNAME: The user that runs your app (e.g., ubuntu or root).

  • SSH_PRIVATE_KEY: Paste the entire private key you copied in Step 1.

Handling .env Variables Securely

You likely have API keys or database URLs that your app needs. Do not commit a .env file to GitHub. Instead, create a fourth secret:

  • ENV_FILE: Paste the literal contents of your production .env file here.

Step 3: The GitHub Actions Workflow

In your local repository, create a directory path: .github/workflows/. Create a file named production-deploy.yml.

Paste the following configuration. I have commented on what each step does, including how we safely write the .env file and optimize Docker cache.

name: Deploy to Production VPS

# Trigger the workflow only when code is pushed to the main branch
on:
  push:
    branches:
      - main

jobs:
  deploy:
    runs-on: ubuntu-latest
    
    steps:
      - name: Execute Deployment Script via SSH
        uses: appleboy/ssh-action@v1.0.3
        with:
          host: ${{ secrets.SERVER_HOST }}
          username: ${{ secrets.SERVER_USERNAME }}
          key: ${{ secrets.SSH_PRIVATE_KEY }}
          # Increase timeout for large docker builds
          timeout: 10m
          # We pass the ENV_FILE secret as an environment variable to the SSH session
          envs: ENV_FILE
          script: |
            # 1. Navigate to the project directory
            cd /opt/myapp

            # 2. Pull the latest code
            git reset --hard HEAD
            git pull origin main

            # 3. Securely write the .env file from GitHub Secrets
            # We use 'printenv' to safely output the multiline secret without breaking bash syntax
            printenv ENV_FILE > .env
            chmod 600 .env # Restrict read permissions

            # 4. Build the Docker image
            # We don't use --no-cache so Docker can reuse previously built layers (like npm install)
            docker compose build web

            # 5. Restart the container
            # -d runs it in the background
            docker compose up -d web

            # 6. Clean up dangling images to prevent the server's disk from filling up
            docker image prune -af --filter "until=24h"

Step 4: Optimizing Docker Caching (Crucial for Node.js)

If your deployments take 5 minutes because npm install runs every single time, your Dockerfile is optimized incorrectly.

To take advantage of the caching we enabled in the workflow, ensure your Dockerfile copies package.json before the rest of your code:

FROM node:18-alpine
WORKDIR /app

# Step A: Copy ONLY package files first
COPY package.json package-lock.json ./

# Step B: Install dependencies
RUN npm ci

# Step C: Copy the rest of the application
COPY . .

RUN npm run build
CMD ["npm", "start"]
\

Because Docker builds in layers, if you only change a React component (Step C), Docker realizes package.json hasn't changed. It instantly grabs the cached node_modules from Step B, entirely skipping the 2-minute npm install step. Your deployments will drop from 5 minutes to 15 seconds.

Rollback Strategy

What happens if you push a bug that breaks production? Because your code is on GitHub, rolling back is as simple as reverting the commit:

  1. On your local machine: git revert HEAD

  2. git push origin main

GitHub Actions will instantly trigger, SSH into your server, pull the reverted code, and deploy the stable version.

Conclusion

By combining GitHub Actions, SSH, and Docker, you have built a professional CI/CD pipeline. Your environment variables are injected securely on the fly, your builds are cached and fast, and old Docker images are pruned automatically.

You now have PaaS-level deployment infrastructure on an unmanaged server, saving you hundreds of dollars as you scale.

Z

ZyVOP

Passionate developer sharing knowledge about modern web technologies and best practices.

Comments (0)

Login to post a comment.

Stay Updated

Get the latest articles delivered to your inbox.

We respect your privacy. Unsubscribe anytime.

Related Posts

Automate Your Code Quality with Git Hooks (And Never Argue in Code Review Again)

Most code review comments should never require a reviewer. This guide shows how to automate formatting, linting, staged-file checks, and commit message validation using Git hooks, Husky, lint-staged, and commitlint before bad code ever reaches your repository.

Read article

Stop Dropping Connections: The Engineer's Guide to Zero-Downtime Deployments with Docker Compose

Read article

Popular Tags

#.env.example Node.js#0x profiling#12-factor#AI agents#AI code security#AI coding tools 2026#AI-assisted development#AI-generated vulnerabilities#ALTER TABLE no lock#API Design