basemind 0.2.2

Full AI context layer over MCP — tree-sitter code-map, document RAG (PDF/Office/HTML/email + OCR + reranker), shared agent memory, on-demand web crawl, git history + blame + per-symbol diff. 300+ languages, 8 coding-agent harnesses, content-addressed Fjall + LanceDB.
name: ci

on:
  push:
    branches: [main]
  pull_request:
    branches: [main]
  # Nightly run of the hardening harness — gated by the `hardening` job's `if:` below.
  schedule:
    - cron: "0 4 * * *"
  workflow_dispatch: {}

# Cancel superseded runs on the same ref — fresher pushes win.
concurrency:
  group: ci-${{ github.workflow }}-${{ github.ref }}
  cancel-in-progress: ${{ github.ref != 'refs/heads/main' }}

env:
  CARGO_TERM_COLOR: always
  RUST_BACKTRACE: 1

jobs:
  test:
    name: test / ${{ matrix.os }} / ${{ matrix.features }}
    runs-on: ${{ matrix.os }}
    strategy:
      fail-fast: false
      matrix:
        os: [ubuntu-latest, macos-latest]
        # Default-features only on per-PR CI. The opt-in intelligence features
        # (`memory`, `documents`, `crawl`, `full`) pull in kreuzberg's heavy
        # system-dep stack (libheif source build, Tesseract, ONNX runtime,
        # downloadable embedding models) — too expensive for fast PR feedback.
        # The nightly `hardening` job runs the full feature surface against
        # real OSS repos and is the authoritative signal for those builds.
        features: ['']
    steps:
      - uses: actions/checkout@v4

      - uses: dtolnay/rust-toolchain@stable
        with:
          components: rustfmt, clippy

      # protoc is required by lance-encoding's build script (transitive via the
      # `memory`, `documents`, `crawl`, and `full` features). Install on every
      # matrix point — the default-features build doesn't need it but the cost
      # is negligible and keeps the steps uniform across matrix entries.
      - name: Install protoc (Linux)
        if: runner.os == 'Linux'
        run: sudo apt-get update && sudo apt-get install -y protobuf-compiler
      - name: Install protoc (macOS)
        if: runner.os == 'macOS'
        run: brew install protobuf

      - uses: Swatinem/rust-cache@v2
        with:
          # Differentiate caches per-OS and features so the cache key is stable across runs.
          key: ${{ matrix.os }}-${{ matrix.features }}

      - name: cargo fmt
        run: cargo fmt --all --check

      - name: cargo clippy
        run: cargo clippy --workspace --all-targets --tests --features "${{ matrix.features }}" -- -D warnings

      - name: cargo test
        run: cargo test --workspace --features "${{ matrix.features }}" --quiet

      - name: cargo build --release
        run: cargo build --release --quiet --bin basemind --features "${{ matrix.features }}"

  deny:
    name: cargo-deny
    runs-on: ubuntu-latest
    steps:
      - uses: actions/checkout@v4
      - uses: dtolnay/rust-toolchain@stable
      - uses: Swatinem/rust-cache@v2
      - uses: EmbarkStudios/cargo-deny-action@v2
        with:
          # Uses deny.toml at the repo root.
          command: check

  # Real-OSS hardening harness. Clones large upstream repos and exercises every MCP tool
  # against them — too heavy for per-PR but ideal as a nightly canary. Runs on manual
  # dispatch and once a day.
  hardening:
    name: hardening harness (nightly)
    runs-on: ubuntu-latest
    if: github.event_name == 'workflow_dispatch' || github.event_name == 'schedule'
    steps:
      - uses: actions/checkout@v4
      - uses: dtolnay/rust-toolchain@stable
      # protoc is required by lance-encoding's build script (transitive via the
      # `full` feature surface that harden.sh builds). Mirror the `test` job.
      - name: Install protoc
        run: sudo apt-get update && sudo apt-get install -y protobuf-compiler
      - uses: Swatinem/rust-cache@v2
      - name: run harden.sh
        run: ./scripts/harden.sh
      - name: upload results
        if: always()
        uses: actions/upload-artifact@v4
        with:
          name: harden-results
          path: /tmp/basemind-harden/results.ndjson