feat(affinescript-deno-test): v0.2.0 multi-test-per-file #285
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
| # SPDX-License-Identifier: PMPL-1.0-or-later | |
| # Hypatia Neurosymbolic CI/CD Security Scan | |
| name: Hypatia Security Scan | |
| on: | |
| push: | |
| branches: [ main, master, develop ] | |
| pull_request: | |
| branches: [ main, master ] | |
| schedule: | |
| - cron: '0 0 * * 0' # Weekly on Sunday | |
| workflow_dispatch: | |
| permissions: | |
| contents: read | |
| jobs: | |
| scan: | |
| name: Hypatia Neurosymbolic Analysis | |
| runs-on: ubuntu-latest | |
| steps: | |
| - name: Checkout repository | |
| uses: actions/checkout@de0fac2e4500dabe0009e67214ff5f5447ce83dd # v4 | |
| with: | |
| fetch-depth: 0 # Full history for better pattern analysis | |
| - name: Clone Hypatia | |
| run: | | |
| if [ ! -d "$HOME/hypatia" ]; then | |
| git clone https://github.com/hyperpolymath/hypatia.git "$HOME/hypatia" | |
| fi | |
| chmod +x "$HOME/hypatia/hypatia-cli.sh" | |
| - name: Run Hypatia scan | |
| id: scan | |
| run: | | |
| echo "Scanning repository: ${{ github.repository }}" | |
| # Run scanner (exits non-zero when findings exist, which is expected) | |
| HYPATIA_FORMAT=json "$HOME/hypatia/hypatia-cli.sh" scan . > hypatia-findings.json || true | |
| # Count findings (handle both flat array and nested structures) | |
| FINDING_COUNT=$(jq 'if type == "array" then length else 0 end' hypatia-findings.json 2>/dev/null || echo 0) | |
| echo "findings_count=$FINDING_COUNT" >> $GITHUB_OUTPUT | |
| # Extract severity counts (flatten if nested, filter by .severity) | |
| CRITICAL=$(jq '[.. | objects | select(.severity? == "critical")] | length' hypatia-findings.json 2>/dev/null || echo 0) | |
| HIGH=$(jq '[.. | objects | select(.severity? == "high")] | length' hypatia-findings.json 2>/dev/null || echo 0) | |
| MEDIUM=$(jq '[.. | objects | select(.severity? == "medium")] | length' hypatia-findings.json 2>/dev/null || echo 0) | |
| echo "critical=$CRITICAL" >> $GITHUB_OUTPUT | |
| echo "high=$HIGH" >> $GITHUB_OUTPUT | |
| echo "medium=$MEDIUM" >> $GITHUB_OUTPUT | |
| echo "## Hypatia Scan Results" >> $GITHUB_STEP_SUMMARY | |
| echo "- Total findings: $FINDING_COUNT" >> $GITHUB_STEP_SUMMARY | |
| echo "- Critical: $CRITICAL" >> $GITHUB_STEP_SUMMARY | |
| echo "- High: $HIGH" >> $GITHUB_STEP_SUMMARY | |
| echo "- Medium: $MEDIUM" >> $GITHUB_STEP_SUMMARY | |
| - name: Upload findings artifact | |
| uses: actions/upload-artifact@bbbca2ddaa5d8feaa63e36b76fdaad77386f024f # v7.0.0 | |
| with: | |
| name: hypatia-findings | |
| path: hypatia-findings.json | |
| retention-days: 90 | |
| - name: Submit findings to gitbot-fleet (Phase 2) | |
| if: steps.scan.outputs.findings_count > 0 | |
| env: | |
| GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} | |
| GITHUB_REPOSITORY: ${{ github.repository }} | |
| GITHUB_SHA: ${{ github.sha }} | |
| run: | | |
| echo "📤 Submitting ${{ steps.scan.outputs.findings_count }} findings to gitbot-fleet..." | |
| # Clone gitbot-fleet to temp directory | |
| FLEET_DIR="/tmp/gitbot-fleet-$$" | |
| git clone https://github.com/hyperpolymath/gitbot-fleet.git "$FLEET_DIR" | |
| # Run submission script | |
| bash "$FLEET_DIR/scripts/submit-finding.sh" hypatia-findings.json | |
| # Cleanup | |
| rm -rf "$FLEET_DIR" | |
| echo "✅ Finding submission complete" | |
| - name: Check for critical issues | |
| if: steps.scan.outputs.critical > 0 | |
| run: | | |
| echo "⚠️ Critical security issues found!" | |
| echo "Review hypatia-findings.json for details" | |
| # Don't fail the build yet - just warn | |
| # exit 1 | |
| - name: Generate scan report | |
| run: | | |
| cat << EOF > hypatia-report.md | |
| # Hypatia Security Scan Report | |
| **Repository:** ${{ github.repository }} | |
| **Scan Date:** $(date -u +"%Y-%m-%d %H:%M:%S UTC") | |
| **Commit:** ${{ github.sha }} | |
| ## Summary | |
| | Severity | Count | | |
| |----------|-------| | |
| | Critical | ${{ steps.scan.outputs.critical }} | | |
| | High | ${{ steps.scan.outputs.high }} | | |
| | Medium | ${{ steps.scan.outputs.medium }} | | |
| | **Total**| ${{ steps.scan.outputs.findings_count }} | | |
| ## Next Steps | |
| 1. Review findings in the artifact: hypatia-findings.json | |
| 2. Auto-fixable issues will be addressed by robot-repo-automaton (Phase 3) | |
| 3. Manual review required for complex issues | |
| ## Learning | |
| These findings feed Hypatia's learning engine to improve future rules. | |
| --- | |
| *Powered by [Hypatia](https://github.com/hyperpolymath/hypatia) - Neurosymbolic CI/CD Intelligence* | |
| EOF | |
| cat hypatia-report.md >> $GITHUB_STEP_SUMMARY | |
| - name: Comment on PR with findings | |
| if: github.event_name == 'pull_request' && steps.scan.outputs.findings_count > 0 | |
| uses: actions/github-script@ed597411d8f924073f98dfc5c65a23a2325f34cd # v8.0.0 | |
| with: | |
| script: | | |
| const fs = require('fs'); | |
| const findings = JSON.parse(fs.readFileSync('hypatia-findings.json', 'utf8')); | |
| const critical = findings.filter(f => f.severity === 'critical').length; | |
| const high = findings.filter(f => f.severity === 'high').length; | |
| let comment = `## 🔍 Hypatia Security Scan\n\n`; | |
| comment += `**Findings:** ${findings.length} issues detected\n\n`; | |
| comment += `| Severity | Count |\n|----------|-------|\n`; | |
| comment += `| 🔴 Critical | ${critical} |\n`; | |
| comment += `| 🟠 High | ${high} |\n`; | |
| comment += `| 🟡 Medium | ${findings.length - critical - high} |\n\n`; | |
| if (critical > 0) { | |
| comment += `⚠️ **Action Required:** Critical security issues found!\n\n`; | |
| } | |
| comment += `<details><summary>View findings</summary>\n\n`; | |
| comment += `\`\`\`json\n${JSON.stringify(findings.slice(0, 10), null, 2)}\n\`\`\`\n`; | |
| comment += `</details>\n\n`; | |
| comment += `*Powered by Hypatia Neurosymbolic CI/CD Intelligence*`; | |
| github.rest.issues.createComment({ | |
| owner: context.repo.owner, | |
| repo: context.repo.repo, | |
| issue_number: context.issue.number, | |
| body: comment | |
| }); |