name: "DevSecOps Enterprise Pipeline" on: push: branches: [ main ] jobs: security-gate-and-deploy: runs-on: ubuntu-latest steps: - name: Checkout Code uses: actions/checkout@v3 with: fetch-depth: 0 # ========================================== # ETAPA 1: TESTES ESTÁTICOS DE SEGURANÇA # ========================================== # 1.1. SECRET SCANNING (Protege contra novas fugas de chaves) - name: Gitleaks Scan run: | curl -sL https://github.com/gitleaks/gitleaks/releases/download/v8.18.2/gitleaks_8.18.2_linux_x64.tar.gz | tar -xz -C /tmp /tmp/gitleaks protect --source . --verbose --redact --staged --exit-code 1 # 1.2. SCA - Verifica vulnerabilidades conhecidas no Nginx - name: Scan Docker Image Vulnerabilities (Trivy) run: | curl -sfL https://raw.githubusercontent.com/aquasecurity/trivy/main/contrib/install.sh | sh -s -- -b /usr/local/bin trivy image --severity HIGH,CRITICAL nginx:alpine # 1.3. SAST - Análise de Código Fonte - name: SonarQube Analysis run: | curl -sL https://binaries.sonarsource.com/Distribution/sonar-scanner-cli/sonar-scanner-cli-5.0.1.3006-linux.zip -o sonar-scanner.zip unzip -q sonar-scanner.zip ./sonar-scanner-5.0.1.3006-linux/bin/sonar-scanner \ -Dsonar.projectKey=website-test \ -Dsonar.sources=. \ -Dsonar.host.url=http://51.89.40.2:9000 \ -Dsonar.token=${{ secrets.SONAR_TOKEN }} \ -Dsonar.qualitygate.wait=true # ========================================== # ETAPA 2: SANDBOX (AMBIENTE DE TESTE DINÂMICO) # ========================================== - name: Criar Sandbox Temporária run: | # Remover qualquer resíduo de sandbox anterior docker rm -f website-test-sandbox || true # Criamos a Sandbox. Como o ZAP vai aceder pela rede interna do Docker, # já não precisamos de expor portas para o exterior (muito mais seguro)! docker run -d --name website-test-sandbox nginx:alpine # Copiar o index.html atual para a Sandbox docker cp index.html website-test-sandbox:/usr/share/nginx/html/index.html # Aguardar 5 segundos para o servidor Nginx iniciar sleep 5 # ========================================== # ETAPA 3: DAST - TESTE DINÂMICO (OWASP ZAP) # ========================================== - name: OWASP ZAP Baseline Scan run: | # Criamos as pastas necessárias no runner mkdir -p qatests mkdir -p /tmp/zap-share chmod 777 /tmp/zap-share # LIMPEZA PREVENTIVA: Remove o container zap-scanner de execuções passadas, se existir docker rm -f zap-scanner || true # Corremos o ZAP apontando o volume para /tmp/zap-share docker run --name zap-scanner \ --link website-test-sandbox:website-test-sandbox \ -v /tmp/zap-share:/zap/wrk/:rw \ -t ghcr.io/zaproxy/zaproxy:stable zap-baseline.py \ -t http://website-test-sandbox \ -r report.html \ -I || true # Copiamos o relatório gerado na pasta partilhada para a pasta qatests do runner cp /tmp/zap-share/report.html qatests/report.html # Limpamos o container e os ficheiros temporários no final docker rm -f zap-scanner || true rm -rf /tmp/zap-share # ========================================== # ETAPA 4: DEPLOY EM PRODUÇÃO (SÓ SE TUDO PASSAR) # ========================================== - name: Hardened Deploy (Produção - Porta 3000) run: | # Backup do ficheiro anterior em Produção docker exec website-test-backend tar -czf /tmp/index_backup.tar.gz -C /usr/share/nginx/html index.html || true # Limpar a pasta de produção e copiar o novo código aprovado docker exec website-test-backend sh -c "rm -rf /usr/share/nginx/html/*" docker cp index.html website-test-backend:/usr/share/nginx/html/index.html # Aplicar Hardening de permissões docker exec website-test-backend chown root:root /usr/share/nginx/html/index.html docker exec website-test-backend chmod 444 /usr/share/nginx/html/index.html # Testar se o site responde localmente em produção docker exec website-test-backend curl --silent --show-error --fail http://localhost:80 || exit 1 # ========================================== # ETAPA 5: ARTEFACTOS E NOTIFICAÇÕES # ========================================== # Guarda o relatório interativo gerado pelo OWASP ZAP para poderes descarregar no Gitea - name: Guardar Relatorio ZAP if: always() uses: actions/upload-artifact@v3 with: name: owasp-zap-report path: qatests/report.html - name: Slack/Discord Notification if: always() run: | echo "Pipeline finalizada com status: ${{ job.status }}"