forjar 1.4.1

Rust-native Infrastructure as Code — bare-metal first, BLAKE3 state, provenance tracing
Documentation
# FJ-2607: Behavior-driven infrastructure specs
# Runs forjar behavior tests against spec YAML files
name: behavior
on:
  push:
    branches: [main]
    paths:
      - "src/**"
      - "tests/**/*.spec.yaml"
      - ".cargo/config.toml"
      - ".github/workflows/behavior.yml"
  pull_request:
    paths:
      - "src/**"
      - "tests/**/*.spec.yaml"
  workflow_dispatch:

concurrency:
  group: behavior-${{ github.ref }}
  cancel-in-progress: true

jobs:
  behavior:
    runs-on: ubuntu-latest
    timeout-minutes: 15
    steps:
      - uses: actions/checkout@v4

      - name: Checkout provable-contracts (path dep)
        uses: actions/checkout@11bd71901bbe5b1630ceea73d27597364c9af683  # v4.2.2
        with:
          repository: paiml/provable-contracts
          path: provable-contracts

      - name: Symlink provable-contracts for Cargo path deps
        run: ln -sf "$GITHUB_WORKSPACE/provable-contracts" "$GITHUB_WORKSPACE/../provable-contracts"

      - name: Install Rust
        uses: dtolnay/rust-toolchain@stable

      - uses: Swatinem/rust-cache@v2

      - name: Generate contract assertions — pv codegen
        run: |
          if ! command -v pv >/dev/null 2>&1; then
            cargo install provable-contracts-cli --locked || true
          fi
          PV=$(command -v pv 2>/dev/null || true)
          if [ -z "$PV" ]; then
            echo "::warning::pv not found — skipping contract generation"
          else
            PC_CONTRACTS="$GITHUB_WORKSPACE/../provable-contracts/contracts"
            if [ -f src/lib.rs ] && grep -q 'mod generated_contracts' src/lib.rs && [ ! -f src/generated_contracts.rs ]; then
              if [ -d "$PC_CONTRACTS" ]; then
                "$PV" codegen "$PC_CONTRACTS" -o src/generated_contracts.rs || true
              elif [ -d contracts ]; then
                "$PV" codegen contracts/ -o src/generated_contracts.rs || true
              fi
            fi
          fi

      - name: Build forjar
        run: cargo build --release

      - name: Run behavior specs
        run: |
          shopt -s nullglob
          specs=(tests/**/*.spec.yaml)
          if [ ${#specs[@]} -eq 0 ]; then
            echo "No spec files found — skipping behavior tests"
            exit 0
          fi
          for spec in "${specs[@]}"; do
            echo "--- Running: $spec"
            ./target/release/forjar test --file "$spec" || exit 1
          done

      - name: Upload test artifacts
        if: always()
        uses: actions/upload-artifact@v4
        with:
          name: behavior-results
          path: state/test-artifacts/behavior/
          retention-days: 14
          if-no-files-found: ignore