How to Build a Production-Ready CI/CD Pipeline in GitHub Actions
Building a production-ready CI/CD pipeline is crucial for delivering software reliably and securely. In this guide, I'll walk you through creating a comprehensive CI/CD pipeline using GitHub Actions that follows industry best practices.
Pipeline Architecture
A production-ready CI/CD pipeline should include:
- Code Quality Checks - Linting, type checking, formatting
- Security Scanning - Dependency scanning, secret detection
- Testing - Unit tests, integration tests, e2e tests
- Building - Compile and package your application
- Deployment - Staging and production deployments
- Monitoring - Post-deployment verification
Step 1: Basic Pipeline Structure
Let's start with a basic workflow file:
name: CI/CD Pipeline
on:
push:
branches: [main, develop]
pull_request:
branches: [main]
jobs:
quality:
runs-on: ubuntu-latest
steps:
- uses: actions/checkout@v4
- uses: actions/setup-node@v4
with:
node-version: '20'
- run: npm ci
- run: npm run lint
- run: npm run type-check
- run: npm run format:check
Step 2: Security Scanning
Add security scanning to catch vulnerabilities early:
security:
runs-on: ubuntu-latest
steps:
- uses: actions/checkout@v4
- name: Run Trivy vulnerability scanner
uses: aquasecurity/trivy-action@master
with:
scan-type: 'fs'
scan-ref: '.'
- name: Check for secrets
uses: trufflesecurity/trufflehog@main
with:
path: ./
base: ${{ github.event.repository.default_branch }}
Step 3: Testing
Comprehensive testing ensures code quality:
test:
runs-on: ubuntu-latest
strategy:
matrix:
node-version: [18, 20]
steps:
- uses: actions/checkout@v4
- uses: actions/setup-node@v4
with:
node-version: ${{ matrix.node-version }}
- run: npm ci
- run: npm run test:unit
- run: npm run test:integration
- name: Upload coverage
uses: codecov/codecov-action@v3
with:
files: ./coverage/lcov.info
Step 4: Building
Build your application with proper caching:
build:
needs: [quality, security, test]
runs-on: ubuntu-latest
steps:
- uses: actions/checkout@v4
- uses: actions/setup-node@v4
with:
node-version: '20'
- uses: actions/cache@v3
with:
path: ~/.npm
key: ${{ runner.os }}-node-${{ hashFiles('**/package-lock.json') }}
- run: npm ci
- run: npm run build
- name: Build Docker image
run: docker build -t myapp:${{ github.sha }} .
Step 5: Deployment
Deploy with proper environment separation:
deploy-staging:
needs: build
if: github.ref == 'refs/heads/develop'
runs-on: ubuntu-latest
environment: staging
steps:
- uses: actions/checkout@v4
- name: Deploy to staging
run: |
# Your deployment commands here
echo "Deploying to staging..."
deploy-production:
needs: build
if: github.ref == 'refs/heads/main'
runs-on: ubuntu-latest
environment: production
steps:
- uses: actions/checkout@v4
- name: Deploy to production
run: |
# Your deployment commands here
echo "Deploying to production..."
Best Practices
1. Use Matrix Strategies
Test across multiple versions and platforms:
strategy:
matrix:
os: [ubuntu-latest, windows-latest, macos-latest]
node-version: [18, 20]
2. Cache Dependencies
Speed up builds with caching:
- uses: actions/cache@v3
with:
path: node_modules
key: ${{ runner.os }}-node-${{ hashFiles('**/package-lock.json') }}
3. Use Secrets Properly
Never hardcode secrets:
- name: Deploy
env:
AWS_ACCESS_KEY_ID: ${{ secrets.AWS_ACCESS_KEY_ID }}
AWS_SECRET_ACCESS_KEY: ${{ secrets.AWS_SECRET_ACCESS_KEY }}
4. Add Approval Gates
Require manual approval for production:
environment:
name: production
url: https://myapp.com
Configure in GitHub repository settings to require reviewers.
5. Parallel Execution
Run independent jobs in parallel:
jobs:
quality:
# ...
security:
# ...
test:
# ...
Advanced Features
Conditional Deployments
Only deploy when tests pass:
deploy:
needs: [test, build]
if: success()
Rollback Strategy
Implement automatic rollback on failure:
- name: Health check
run: |
if ! curl -f https://myapp.com/health; then
echo "Deployment failed, rolling back..."
# Rollback logic
exit 1
fi
Notifications
Notify team on deployment:
- name: Notify Slack
uses: 8398a7/action-slack@v3
with:
status: ${{ job.status }}
text: 'Deployment completed!'
Conclusion
A production-ready CI/CD pipeline is more than just running tests and deploying code. It should include:
- ✅ Comprehensive quality checks
- ✅ Security scanning
- ✅ Proper testing strategy
- ✅ Efficient builds with caching
- ✅ Safe deployment practices
- ✅ Monitoring and rollback capabilities
By following these practices, you'll have a robust pipeline that delivers software reliably and securely.