Initial commit: Telegram Management System
Some checks failed
Deploy / deploy (push) Has been cancelled
Some checks failed
Deploy / deploy (push) Has been cancelled
Full-stack web application for Telegram management - Frontend: Vue 3 + Vben Admin - Backend: NestJS - Features: User management, group broadcast, statistics 🤖 Generated with [Claude Code](https://claude.com/claude-code) Co-Authored-By: Claude <noreply@anthropic.com>
This commit is contained in:
348
marketing-agent/.github/workflows/cd.yml
vendored
Normal file
348
marketing-agent/.github/workflows/cd.yml
vendored
Normal file
@@ -0,0 +1,348 @@
|
||||
name: CD Pipeline
|
||||
|
||||
on:
|
||||
push:
|
||||
branches: [ main ]
|
||||
tags: [ 'v*.*.*' ]
|
||||
workflow_dispatch:
|
||||
inputs:
|
||||
environment:
|
||||
description: 'Deployment environment'
|
||||
required: true
|
||||
default: 'staging'
|
||||
type: choice
|
||||
options:
|
||||
- staging
|
||||
- production
|
||||
version:
|
||||
description: 'Version to deploy (leave empty for latest)'
|
||||
required: false
|
||||
|
||||
env:
|
||||
REGISTRY: ghcr.io
|
||||
IMAGE_NAME: ${{ github.repository }}
|
||||
HELM_VERSION: '3.13.0'
|
||||
|
||||
jobs:
|
||||
# Prepare deployment
|
||||
prepare-deployment:
|
||||
name: Prepare Deployment
|
||||
runs-on: ubuntu-latest
|
||||
outputs:
|
||||
environment: ${{ steps.determine-env.outputs.environment }}
|
||||
version: ${{ steps.determine-version.outputs.version }}
|
||||
should-deploy: ${{ steps.check-deploy.outputs.should-deploy }}
|
||||
steps:
|
||||
- uses: actions/checkout@v4
|
||||
|
||||
- name: Determine environment
|
||||
id: determine-env
|
||||
run: |
|
||||
if [[ "${{ github.event_name }}" == "workflow_dispatch" ]]; then
|
||||
echo "environment=${{ github.event.inputs.environment }}" >> $GITHUB_OUTPUT
|
||||
elif [[ "${{ github.ref }}" == "refs/heads/main" ]]; then
|
||||
echo "environment=staging" >> $GITHUB_OUTPUT
|
||||
elif [[ "${{ github.ref }}" =~ ^refs/tags/v[0-9]+\.[0-9]+\.[0-9]+$ ]]; then
|
||||
echo "environment=production" >> $GITHUB_OUTPUT
|
||||
else
|
||||
echo "environment=development" >> $GITHUB_OUTPUT
|
||||
fi
|
||||
|
||||
- name: Determine version
|
||||
id: determine-version
|
||||
run: |
|
||||
if [[ "${{ github.event_name }}" == "workflow_dispatch" && -n "${{ github.event.inputs.version }}" ]]; then
|
||||
echo "version=${{ github.event.inputs.version }}" >> $GITHUB_OUTPUT
|
||||
elif [[ "${{ github.ref }}" =~ ^refs/tags/v[0-9]+\.[0-9]+\.[0-9]+$ ]]; then
|
||||
echo "version=${GITHUB_REF#refs/tags/}" >> $GITHUB_OUTPUT
|
||||
else
|
||||
echo "version=${{ github.sha }}" >> $GITHUB_OUTPUT
|
||||
fi
|
||||
|
||||
- name: Check deployment conditions
|
||||
id: check-deploy
|
||||
run: |
|
||||
ENV="${{ steps.determine-env.outputs.environment }}"
|
||||
if [[ "$ENV" == "production" && ! "${{ github.ref }}" =~ ^refs/tags/v[0-9]+\.[0-9]+\.[0-9]+$ ]]; then
|
||||
echo "should-deploy=false" >> $GITHUB_OUTPUT
|
||||
echo "::warning::Production deployment requires a semantic version tag"
|
||||
else
|
||||
echo "should-deploy=true" >> $GITHUB_OUTPUT
|
||||
fi
|
||||
|
||||
# Database Migration
|
||||
database-migration:
|
||||
name: Database Migration
|
||||
runs-on: ubuntu-latest
|
||||
needs: [prepare-deployment]
|
||||
if: needs.prepare-deployment.outputs.should-deploy == 'true'
|
||||
environment: ${{ needs.prepare-deployment.outputs.environment }}
|
||||
steps:
|
||||
- uses: actions/checkout@v4
|
||||
|
||||
- name: Setup Node.js
|
||||
uses: actions/setup-node@v4
|
||||
with:
|
||||
node-version: '18.x'
|
||||
|
||||
- name: Configure AWS credentials
|
||||
uses: aws-actions/configure-aws-credentials@v4
|
||||
with:
|
||||
aws-access-key-id: ${{ secrets.AWS_ACCESS_KEY_ID }}
|
||||
aws-secret-access-key: ${{ secrets.AWS_SECRET_ACCESS_KEY }}
|
||||
aws-region: ${{ vars.AWS_REGION }}
|
||||
|
||||
- name: Get database connection string
|
||||
id: get-db-connection
|
||||
run: |
|
||||
SECRET_ARN="${{ secrets.DB_SECRET_ARN }}"
|
||||
DB_SECRET=$(aws secretsmanager get-secret-value --secret-id $SECRET_ARN --query SecretString --output text)
|
||||
echo "::add-mask::$DB_SECRET"
|
||||
echo "DB_CONNECTION=$DB_SECRET" >> $GITHUB_ENV
|
||||
|
||||
- name: Run migrations
|
||||
run: |
|
||||
cd migrations
|
||||
npm ci
|
||||
npm run migrate:up
|
||||
env:
|
||||
MONGODB_URI: ${{ env.DB_CONNECTION }}
|
||||
MIGRATION_ENV: ${{ needs.prepare-deployment.outputs.environment }}
|
||||
|
||||
# Deploy to Kubernetes
|
||||
deploy-kubernetes:
|
||||
name: Deploy to Kubernetes
|
||||
runs-on: ubuntu-latest
|
||||
needs: [prepare-deployment, database-migration]
|
||||
if: needs.prepare-deployment.outputs.should-deploy == 'true'
|
||||
environment:
|
||||
name: ${{ needs.prepare-deployment.outputs.environment }}
|
||||
url: ${{ steps.deploy.outputs.app-url }}
|
||||
strategy:
|
||||
matrix:
|
||||
region: [us-east-1, eu-west-1, ap-southeast-1]
|
||||
steps:
|
||||
- uses: actions/checkout@v4
|
||||
|
||||
- name: Configure AWS credentials
|
||||
uses: aws-actions/configure-aws-credentials@v4
|
||||
with:
|
||||
aws-access-key-id: ${{ secrets.AWS_ACCESS_KEY_ID }}
|
||||
aws-secret-access-key: ${{ secrets.AWS_SECRET_ACCESS_KEY }}
|
||||
aws-region: ${{ matrix.region }}
|
||||
|
||||
- name: Login to Amazon ECR
|
||||
id: login-ecr
|
||||
uses: aws-actions/amazon-ecr-login@v2
|
||||
|
||||
- name: Setup kubectl
|
||||
uses: azure/setup-kubectl@v3
|
||||
with:
|
||||
version: 'v1.28.0'
|
||||
|
||||
- name: Setup Helm
|
||||
uses: azure/setup-helm@v3
|
||||
with:
|
||||
version: ${{ env.HELM_VERSION }}
|
||||
|
||||
- name: Update kubeconfig
|
||||
run: |
|
||||
aws eks update-kubeconfig --name marketing-agent-${{ needs.prepare-deployment.outputs.environment }}-${{ matrix.region }}
|
||||
|
||||
- name: Create namespace
|
||||
run: |
|
||||
kubectl create namespace marketing-agent-${{ needs.prepare-deployment.outputs.environment }} --dry-run=client -o yaml | kubectl apply -f -
|
||||
|
||||
- name: Deploy with Helm
|
||||
id: deploy
|
||||
run: |
|
||||
NAMESPACE="marketing-agent-${{ needs.prepare-deployment.outputs.environment }}"
|
||||
RELEASE_NAME="marketing-agent"
|
||||
|
||||
helm upgrade --install $RELEASE_NAME ./helm/marketing-agent \
|
||||
--namespace $NAMESPACE \
|
||||
--values ./helm/marketing-agent/values.yaml \
|
||||
--values ./helm/marketing-agent/values.${{ needs.prepare-deployment.outputs.environment }}.yaml \
|
||||
--set global.image.tag=${{ needs.prepare-deployment.outputs.version }} \
|
||||
--set global.image.registry=${{ steps.login-ecr.outputs.registry }} \
|
||||
--set global.region=${{ matrix.region }} \
|
||||
--set-string global.environment=${{ needs.prepare-deployment.outputs.environment }} \
|
||||
--wait \
|
||||
--timeout 10m
|
||||
|
||||
# Get the application URL
|
||||
APP_URL=$(kubectl get ingress -n $NAMESPACE -o jsonpath='{.items[0].spec.rules[0].host}')
|
||||
echo "app-url=https://$APP_URL" >> $GITHUB_OUTPUT
|
||||
|
||||
- name: Verify deployment
|
||||
run: |
|
||||
NAMESPACE="marketing-agent-${{ needs.prepare-deployment.outputs.environment }}"
|
||||
kubectl rollout status deployment -n $NAMESPACE --timeout=5m
|
||||
kubectl get pods -n $NAMESPACE
|
||||
|
||||
- name: Run smoke tests
|
||||
run: |
|
||||
APP_URL="${{ steps.deploy.outputs.app-url }}"
|
||||
./scripts/smoke-tests.sh $APP_URL
|
||||
|
||||
# Deploy Static Assets to CDN
|
||||
deploy-cdn:
|
||||
name: Deploy Static Assets to CDN
|
||||
runs-on: ubuntu-latest
|
||||
needs: [prepare-deployment, database-migration]
|
||||
if: needs.prepare-deployment.outputs.should-deploy == 'true'
|
||||
steps:
|
||||
- uses: actions/checkout@v4
|
||||
|
||||
- name: Setup Node.js
|
||||
uses: actions/setup-node@v4
|
||||
with:
|
||||
node-version: '18.x'
|
||||
cache: 'npm'
|
||||
|
||||
- name: Build frontend
|
||||
working-directory: ./frontend
|
||||
run: |
|
||||
npm ci
|
||||
npm run build
|
||||
env:
|
||||
VITE_API_URL: ${{ vars.API_URL }}
|
||||
VITE_ENVIRONMENT: ${{ needs.prepare-deployment.outputs.environment }}
|
||||
|
||||
- name: Configure AWS credentials
|
||||
uses: aws-actions/configure-aws-credentials@v4
|
||||
with:
|
||||
aws-access-key-id: ${{ secrets.AWS_ACCESS_KEY_ID }}
|
||||
aws-secret-access-key: ${{ secrets.AWS_SECRET_ACCESS_KEY }}
|
||||
aws-region: us-east-1
|
||||
|
||||
- name: Deploy to S3
|
||||
run: |
|
||||
BUCKET_NAME="marketing-agent-frontend-${{ needs.prepare-deployment.outputs.environment }}"
|
||||
aws s3 sync ./frontend/dist s3://$BUCKET_NAME \
|
||||
--delete \
|
||||
--cache-control "public, max-age=31536000" \
|
||||
--exclude "index.html" \
|
||||
--exclude "*.map"
|
||||
|
||||
# Upload index.html with no-cache
|
||||
aws s3 cp ./frontend/dist/index.html s3://$BUCKET_NAME/index.html \
|
||||
--cache-control "no-cache, no-store, must-revalidate"
|
||||
|
||||
- name: Invalidate CloudFront
|
||||
run: |
|
||||
DISTRIBUTION_ID="${{ secrets.CLOUDFRONT_DISTRIBUTION_ID }}"
|
||||
aws cloudfront create-invalidation \
|
||||
--distribution-id $DISTRIBUTION_ID \
|
||||
--paths "/*"
|
||||
|
||||
# Post-deployment tasks
|
||||
post-deployment:
|
||||
name: Post Deployment Tasks
|
||||
runs-on: ubuntu-latest
|
||||
needs: [prepare-deployment, deploy-kubernetes, deploy-cdn]
|
||||
if: always() && needs.prepare-deployment.outputs.should-deploy == 'true'
|
||||
steps:
|
||||
- uses: actions/checkout@v4
|
||||
|
||||
- name: Update deployment status
|
||||
uses: actions/github-script@v7
|
||||
with:
|
||||
script: |
|
||||
const environment = '${{ needs.prepare-deployment.outputs.environment }}';
|
||||
const version = '${{ needs.prepare-deployment.outputs.version }}';
|
||||
const status = '${{ needs.deploy-kubernetes.result }}';
|
||||
|
||||
// Create deployment status
|
||||
await github.rest.repos.createDeploymentStatus({
|
||||
owner: context.repo.owner,
|
||||
repo: context.repo.repo,
|
||||
deployment_id: context.payload.deployment.id,
|
||||
state: status === 'success' ? 'success' : 'failure',
|
||||
environment_url: status === 'success' ? '${{ needs.deploy-kubernetes.outputs.app-url }}' : '',
|
||||
description: `Deployed version ${version} to ${environment}`
|
||||
});
|
||||
|
||||
- name: Send deployment notification
|
||||
if: success()
|
||||
uses: 8398a7/action-slack@v3
|
||||
with:
|
||||
status: custom
|
||||
custom_payload: |
|
||||
{
|
||||
"text": "Deployment Successful! :rocket:",
|
||||
"attachments": [{
|
||||
"color": "good",
|
||||
"fields": [
|
||||
{ "title": "Environment", "value": "${{ needs.prepare-deployment.outputs.environment }}", "short": true },
|
||||
{ "title": "Version", "value": "${{ needs.prepare-deployment.outputs.version }}", "short": true },
|
||||
{ "title": "Deployed By", "value": "${{ github.actor }}", "short": true },
|
||||
{ "title": "Repository", "value": "${{ github.repository }}", "short": true }
|
||||
]
|
||||
}]
|
||||
}
|
||||
webhook_url: ${{ secrets.SLACK_WEBHOOK }}
|
||||
|
||||
- name: Create release notes
|
||||
if: startsWith(github.ref, 'refs/tags/')
|
||||
uses: actions/github-script@v7
|
||||
with:
|
||||
script: |
|
||||
const tag = context.ref.replace('refs/tags/', '');
|
||||
const { data: commits } = await github.rest.repos.compareCommits({
|
||||
owner: context.repo.owner,
|
||||
repo: context.repo.repo,
|
||||
base: 'v1.0.0', // Previous release tag
|
||||
head: tag
|
||||
});
|
||||
|
||||
const releaseNotes = commits.commits
|
||||
.map(commit => `- ${commit.commit.message}`)
|
||||
.join('\n');
|
||||
|
||||
await github.rest.repos.createRelease({
|
||||
owner: context.repo.owner,
|
||||
repo: context.repo.repo,
|
||||
tag_name: tag,
|
||||
name: `Release ${tag}`,
|
||||
body: releaseNotes,
|
||||
draft: false,
|
||||
prerelease: false
|
||||
});
|
||||
|
||||
# Rollback on failure
|
||||
rollback:
|
||||
name: Rollback Deployment
|
||||
runs-on: ubuntu-latest
|
||||
needs: [prepare-deployment, deploy-kubernetes]
|
||||
if: failure() && needs.prepare-deployment.outputs.environment == 'production'
|
||||
steps:
|
||||
- name: Configure AWS credentials
|
||||
uses: aws-actions/configure-aws-credentials@v4
|
||||
with:
|
||||
aws-access-key-id: ${{ secrets.AWS_ACCESS_KEY_ID }}
|
||||
aws-secret-access-key: ${{ secrets.AWS_SECRET_ACCESS_KEY }}
|
||||
aws-region: ${{ vars.AWS_REGION }}
|
||||
|
||||
- name: Rollback Kubernetes deployment
|
||||
run: |
|
||||
aws eks update-kubeconfig --name marketing-agent-production
|
||||
NAMESPACE="marketing-agent-production"
|
||||
helm rollback marketing-agent -n $NAMESPACE
|
||||
|
||||
- name: Send rollback notification
|
||||
uses: 8398a7/action-slack@v3
|
||||
with:
|
||||
status: custom
|
||||
custom_payload: |
|
||||
{
|
||||
"text": "Production Deployment Failed - Rollback Initiated! :warning:",
|
||||
"attachments": [{
|
||||
"color": "danger",
|
||||
"fields": [
|
||||
{ "title": "Environment", "value": "production", "short": true },
|
||||
{ "title": "Failed Version", "value": "${{ needs.prepare-deployment.outputs.version }}", "short": true }
|
||||
]
|
||||
}]
|
||||
}
|
||||
webhook_url: ${{ secrets.SLACK_WEBHOOK }}
|
||||
Reference in New Issue
Block a user