mirror of
https://github.com/ruvnet/RuView
synced 2026-06-19 11:53:19 +00:00
df617145d6
* feat(ADR-262 P3): live RuField surface — RuView sensing speaks RuField on /api/field + /ws/field Wire the P1 `wifi-densepose-rufield` bridge into the live `wifi-densepose-sensing-server` so the governed sensing cycle emits real signed RuField `FieldEvent`s on two additive endpoints. - Cargo: add the `wifi-densepose-rufield` path dep (the single coupling point, ADR-262 §5.4 — no new RuView-internal coupling). - New `src/rufield_surface.rs` (kept out of the 8k-line main.rs): `FieldSurface` holds a dedicated ed25519 `Signer` + a bounded ring of recent events + the `/ws/field` broadcast topic; `GET /api/field` and `GET /ws/field` handlers; a standalone `router()` for isolated testing. - Signer (defers the P2 key decision, ADR-262 §8 Q1): a STANDALONE dev/sensing key from `WDP_RUFIELD_SIGNING_SEED`, else a deterministic dev default with a logged WARN. Reusing the `cog-ha-matter` Ed25519 key is the deferred P2 call — P3 does not pre-empt it. - Tap: at the ESP32 governed-trust cycle (`main.rs` ~5886 observe_cycle / ~5938 SensingUpdate build), `emit_rufield_event` joins the cycle's features/classification/signal_field with the engine's effective_class/demoted trust state into a `SensingSnapshot` and surfaces it via the bridge. Existing endpoints (`/ws/sensing` etc.) are unchanged — purely additive. - Privacy egress: `network_egress_allowed` is fail-closed for an unattended live surface — only P1/P2 leave the box; P0 raw and P3/P4/P5 (identity/biometric/aggregate) are held edge-local. A `Derived` cycle maps to P4/P5 and never surfaces. - No-phantom: `emit` drops no-presence cycles (no fabricated events). Gates (tests/rufield_surface_test.rs, tower::oneshot, 4/0): well-formed signed event (WifiCsi, P2 not P1, is_fusable, real timestamp); empty cycle → no phantom; Derived trust never surfaces; mixed stream surfaces only egress-safe events. Honesty (ADR-262 §0/§6): real plumbing on a live endpoint, NOT accuracy. Single-link CSI with its existing caveats (no validated room-coordinate accuracy); dedicated dev signing key pending the P2 ownership decision; no accuracy claim. Co-Authored-By: claude-flow <ruv@ruv.net> * docs(ADR-262 P3): mark P1+P3 implemented; document /api/field + /ws/field; CHANGELOG - ADR-262 Status → "P1 + P3 implemented"; add a P3 implementation-status block (tap site, endpoints, dedicated dev signer deferring the §8 Q1 key decision, fail-closed egress, gates). Keep the honesty framing: real plumbing on a live endpoint, not accuracy. - CHANGELOG [Unreleased]: add the ADR-262 P3 entry. - user-guide: add `/api/field` to the REST table + a "RuField surface (ADR-262 P3)" section covering `/api/field` + `/ws/field`, the fail-closed P1/P2-only egress, the WDP_RUFIELD_SIGNING_SEED dev key, and the no-accuracy honesty note. Co-Authored-By: claude-flow <ruv@ruv.net> * ci: checkout submodules everywhere + Dockerfile copies vendor/rufield Making wifi-densepose-rufield (ADR-262 bridge) a v2 workspace member means EVERY cargo-on-workspace context must have the vendor/rufield submodule present (cargo loads all member manifests). P1 only fixed the rust-tests job; this adds `submodules: recursive` to all workflow checkouts that run cargo (mqtt-integration was failing on the missing submodule manifest), and makes Dockerfile.rust COPY vendor/rufield/ to /vendor/rufield (matches the bridge's ../../../vendor/rufield path-dep under the collapsed Docker layout). update-submodules.yml left alone (it manages submodules itself). Co-Authored-By: claude-flow <ruv@ruv.net> --------- Co-authored-by: ruv <ruvnet@gmail.com>
293 lines
12 KiB
YAML
293 lines
12 KiB
YAML
# ADR-117 P5 — cibuildwheel + PyPI publish workflow for `wifi-densepose`
|
|
#
|
|
# This workflow is **explicitly NOT** triggered on every push. It runs only on:
|
|
# - a maintainer-dispatched `workflow_dispatch`
|
|
# - a pushed tag matching `v*-pip` (e.g. `v2.0.0-pip`)
|
|
#
|
|
# The reason for the `-pip` tag suffix is that the repo already cuts
|
|
# `v0.X.Y-esp32` tags for firmware releases (see CLAUDE.md). The `-pip`
|
|
# suffix keeps the pip release schedule independent of the firmware
|
|
# release schedule.
|
|
#
|
|
# Sequencing on release day (per ADR-117 §7.3):
|
|
# 1. cut tag `v1.99.0-pip` → publishes the tombstone wheel first
|
|
# 2. cut tag `v2.0.0-pip` → publishes the PyO3 v2 wheel matrix
|
|
#
|
|
# Publishes via the `PYPI_API_TOKEN` GitHub Actions secret. The
|
|
# token-refresh runbook (GCP Secret Manager → gh secret set) lives in
|
|
# docs/integrations/pypi-release.md so KICS does not flag the
|
|
# secret name as a generic-secret literal in the workflow.
|
|
#
|
|
# Q3 (witness hash v2 — open in ADR-117 §11.3) MUST be resolved
|
|
# before the first v2.0.0 publish. When v2 lands, add a parallel
|
|
# step that verifies the v2 hash against the Rust pipeline.
|
|
|
|
name: pip-release
|
|
|
|
on:
|
|
workflow_dispatch:
|
|
inputs:
|
|
target:
|
|
description: "Which package to release"
|
|
required: true
|
|
type: choice
|
|
options:
|
|
- v2-wheels
|
|
- v1-99-tombstone
|
|
publish_to:
|
|
description: "Where to publish"
|
|
required: true
|
|
default: testpypi
|
|
type: choice
|
|
options:
|
|
- testpypi # dry-run target
|
|
- pypi # production
|
|
push:
|
|
tags:
|
|
- "v*-pip"
|
|
|
|
permissions:
|
|
contents: read
|
|
|
|
jobs:
|
|
# ────────────────────────────────────────────────────────────────
|
|
# v2.0.0 — cibuildwheel matrix (5 wheels + sdist)
|
|
# ────────────────────────────────────────────────────────────────
|
|
|
|
build-wheels:
|
|
name: Build ${{ matrix.os }} ${{ matrix.arch }}
|
|
if: |
|
|
github.event_name == 'workflow_dispatch' && inputs.target == 'v2-wheels' ||
|
|
startsWith(github.ref, 'refs/tags/v2.')
|
|
strategy:
|
|
fail-fast: false
|
|
matrix:
|
|
include:
|
|
- os: ubuntu-latest
|
|
arch: x86_64
|
|
- os: ubuntu-latest
|
|
arch: aarch64
|
|
- os: macos-13 # x86_64 runner
|
|
arch: x86_64
|
|
- os: macos-14 # arm64 runner
|
|
arch: arm64
|
|
- os: windows-latest
|
|
arch: AMD64
|
|
runs-on: ${{ matrix.os }}
|
|
steps:
|
|
- uses: actions/checkout@v4
|
|
with:
|
|
submodules: recursive
|
|
|
|
# Linux aarch64 needs QEMU for cross-build on x86_64 runners.
|
|
- name: Set up QEMU
|
|
if: matrix.os == 'ubuntu-latest' && matrix.arch == 'aarch64'
|
|
uses: docker/setup-qemu-action@v3
|
|
|
|
# ADR-117 §5.4: abi3-py310 — one binary per OS/arch covers all
|
|
# Python minor versions ≥ 3.10. Build only cp310 wheels.
|
|
- name: Build wheels (cibuildwheel)
|
|
uses: pypa/cibuildwheel@v2.21
|
|
env:
|
|
CIBW_BUILD: "cp310-*"
|
|
CIBW_ARCHS_LINUX: ${{ matrix.arch }}
|
|
CIBW_ARCHS_MACOS: ${{ matrix.arch }}
|
|
CIBW_ARCHS_WINDOWS: ${{ matrix.arch }}
|
|
CIBW_BUILD_FRONTEND: "build"
|
|
CIBW_BEFORE_BUILD: "pip install maturin>=1.7"
|
|
# The PyO3 sdist landing depends on the cargo/Rust toolchain
|
|
# being present. cibuildwheel images carry rustup on Linux
|
|
# but we also pin a known-good version for reproducibility.
|
|
CIBW_BEFORE_ALL_LINUX: "curl --proto '=https' --tlsv1.2 -sSf https://sh.rustup.rs | sh -s -- -y --default-toolchain 1.82"
|
|
CIBW_ENVIRONMENT_LINUX: 'PATH="$HOME/.cargo/bin:$PATH"'
|
|
# Smoke-test every built wheel before accepting it. Catches
|
|
# the case where the wheel imports but the compiled symbols
|
|
# are missing.
|
|
CIBW_TEST_REQUIRES: "pytest>=8.0"
|
|
CIBW_TEST_COMMAND: 'python -c "import wifi_densepose; assert wifi_densepose.hello() == \"ok\"; print(wifi_densepose.__build_features__)"'
|
|
with:
|
|
package-dir: python
|
|
output-dir: wheelhouse
|
|
|
|
- uses: actions/upload-artifact@v4
|
|
with:
|
|
name: wheels-${{ matrix.os }}-${{ matrix.arch }}
|
|
path: wheelhouse/*.whl
|
|
if-no-files-found: error
|
|
|
|
build-sdist:
|
|
name: Build v2 sdist
|
|
if: |
|
|
github.event_name == 'workflow_dispatch' && inputs.target == 'v2-wheels' ||
|
|
startsWith(github.ref, 'refs/tags/v2.')
|
|
runs-on: ubuntu-latest
|
|
steps:
|
|
- uses: actions/checkout@v4
|
|
with:
|
|
submodules: recursive
|
|
- name: Install maturin
|
|
run: pip install maturin>=1.7
|
|
- name: Build sdist
|
|
working-directory: python
|
|
run: maturin sdist --out ../sdist
|
|
- uses: actions/upload-artifact@v4
|
|
with:
|
|
name: sdist
|
|
path: sdist/*.tar.gz
|
|
if-no-files-found: error
|
|
|
|
# ────────────────────────────────────────────────────────────────
|
|
# v1.99.0 — tombstone wheel (pure Python, single sdist + wheel)
|
|
# ────────────────────────────────────────────────────────────────
|
|
|
|
build-tombstone:
|
|
name: Build v1.99.0 tombstone
|
|
if: |
|
|
github.event_name == 'workflow_dispatch' && inputs.target == 'v1-99-tombstone' ||
|
|
startsWith(github.ref, 'refs/tags/v1.99')
|
|
runs-on: ubuntu-latest
|
|
steps:
|
|
- uses: actions/checkout@v4
|
|
with:
|
|
submodules: recursive
|
|
- uses: actions/setup-python@v5
|
|
with:
|
|
python-version: '3.12'
|
|
- name: Install build backend
|
|
run: python -m pip install --upgrade pip build>=1.2
|
|
- name: Build sdist + wheel
|
|
working-directory: python/tombstone
|
|
run: python -m build --outdir ../../tombstone-dist
|
|
# Inspect what was actually built — the previous v1.99.0-pip run
|
|
# showed an `import wifi_densepose` that returned cleanly instead
|
|
# of raising, even though build logs said `adding 'wifi_densepose/__init__.py'`.
|
|
# Print the wheel manifest + the __init__.py content so any
|
|
# future regression is debuggable from the run log alone.
|
|
- name: Inspect wheel contents
|
|
run: |
|
|
set -e
|
|
WHL=tombstone-dist/wifi_densepose-1.99.0-py3-none-any.whl
|
|
echo "--- wheel listing ---"
|
|
python -m zipfile -l "$WHL"
|
|
echo "--- wifi_densepose/__init__.py inside the wheel ---"
|
|
python -m zipfile -e "$WHL" /tmp/tomb-inspect
|
|
cat /tmp/tomb-inspect/wifi_densepose/__init__.py
|
|
echo "--- size in bytes ---"
|
|
wc -c /tmp/tomb-inspect/wifi_densepose/__init__.py
|
|
# Smoke-test in an ISOLATED venv. The previous run's failure
|
|
# mode was that the ubuntu-latest runner's system `python` had
|
|
# site-packages picking up something other than the user-installed
|
|
# wheel, so the import resolved to a different module. A clean
|
|
# venv removes any ambiguity about which wifi_densepose is loaded.
|
|
- name: Smoke-test tombstone in isolated venv
|
|
run: |
|
|
set -e
|
|
# Copy the wheel to /tmp BEFORE entering the venv — we must
|
|
# cd OUT of the repo root because the repo contains a
|
|
# `wifi_densepose/` directory left over from the legacy v1
|
|
# source. Python puts cwd at sys.path[0], so an import from
|
|
# the repo root would resolve to the legacy directory and
|
|
# bypass the freshly-installed wheel entirely (this was the
|
|
# silent failure mode of the previous two run attempts).
|
|
cp tombstone-dist/wifi_densepose-1.99.0-py3-none-any.whl /tmp/
|
|
python -m venv /tmp/smoke-venv
|
|
/tmp/smoke-venv/bin/python -m pip install --upgrade pip
|
|
/tmp/smoke-venv/bin/python -m pip install /tmp/wifi_densepose-1.99.0-py3-none-any.whl
|
|
cd /tmp # away from the repo root's stray wifi_densepose/
|
|
/tmp/smoke-venv/bin/python -c "import importlib.util as u; s = u.find_spec('wifi_densepose'); print('Resolved to:', s.origin); print('--- file content ---'); print(open(s.origin).read())"
|
|
set +e
|
|
/tmp/smoke-venv/bin/python -c "import wifi_densepose" 2> import-output.txt
|
|
rc=$?
|
|
set -e
|
|
if [ "$rc" -eq 0 ]; then
|
|
echo "ERROR: tombstone import succeeded — should have raised ImportError"
|
|
exit 1
|
|
fi
|
|
if ! grep -q "github.com/ruvnet/RuView" import-output.txt; then
|
|
echo "ERROR: tombstone ImportError missing migration URL"
|
|
cat import-output.txt
|
|
exit 1
|
|
fi
|
|
echo "Tombstone wheel correctly raises ImportError with migration URL."
|
|
- uses: actions/upload-artifact@v4
|
|
with:
|
|
name: tombstone
|
|
path: tombstone-dist/*
|
|
if-no-files-found: error
|
|
|
|
# ────────────────────────────────────────────────────────────────
|
|
# Publish — gated by manual dispatch OR by the tag form
|
|
# ────────────────────────────────────────────────────────────────
|
|
|
|
publish-v2:
|
|
name: Publish v2 wheels
|
|
needs: [build-wheels, build-sdist]
|
|
if: |
|
|
always() &&
|
|
needs.build-wheels.result == 'success' &&
|
|
needs.build-sdist.result == 'success' &&
|
|
(
|
|
github.event_name == 'workflow_dispatch' && inputs.target == 'v2-wheels' ||
|
|
startsWith(github.ref, 'refs/tags/v2.')
|
|
)
|
|
runs-on: ubuntu-latest
|
|
steps:
|
|
- name: Gather all artifacts into dist/
|
|
uses: actions/download-artifact@v4
|
|
with:
|
|
path: dist-staging
|
|
- name: Flatten artifacts
|
|
run: |
|
|
mkdir -p dist
|
|
find dist-staging -type f \( -name '*.whl' -o -name '*.tar.gz' \) -exec cp -v {} dist/ \;
|
|
ls -lh dist/
|
|
- name: Publish to TestPyPI (dry-run target)
|
|
if: github.event_name == 'workflow_dispatch' && inputs.publish_to == 'testpypi'
|
|
uses: pypa/gh-action-pypi-publish@release/v1
|
|
with:
|
|
repository-url: https://test.pypi.org/legacy/
|
|
password: ${{ secrets.PYPI_API_TOKEN }}
|
|
packages-dir: dist
|
|
skip-existing: true
|
|
- name: Publish to PyPI
|
|
if: |
|
|
startsWith(github.ref, 'refs/tags/v2.') ||
|
|
(github.event_name == 'workflow_dispatch' && inputs.publish_to == 'pypi')
|
|
uses: pypa/gh-action-pypi-publish@release/v1
|
|
with:
|
|
password: ${{ secrets.PYPI_API_TOKEN }}
|
|
packages-dir: dist
|
|
|
|
publish-tombstone:
|
|
name: Publish v1.99 tombstone
|
|
needs: [build-tombstone]
|
|
if: |
|
|
always() &&
|
|
needs.build-tombstone.result == 'success' &&
|
|
(
|
|
github.event_name == 'workflow_dispatch' && inputs.target == 'v1-99-tombstone' ||
|
|
startsWith(github.ref, 'refs/tags/v1.99')
|
|
)
|
|
runs-on: ubuntu-latest
|
|
steps:
|
|
- uses: actions/download-artifact@v4
|
|
with:
|
|
name: tombstone
|
|
path: dist
|
|
- name: Publish to TestPyPI (dry-run target)
|
|
if: github.event_name == 'workflow_dispatch' && inputs.publish_to == 'testpypi'
|
|
uses: pypa/gh-action-pypi-publish@release/v1
|
|
with:
|
|
repository-url: https://test.pypi.org/legacy/
|
|
password: ${{ secrets.PYPI_API_TOKEN }}
|
|
packages-dir: dist
|
|
skip-existing: true
|
|
- name: Publish to PyPI
|
|
if: |
|
|
startsWith(github.ref, 'refs/tags/v1.99') ||
|
|
(github.event_name == 'workflow_dispatch' && inputs.publish_to == 'pypi')
|
|
uses: pypa/gh-action-pypi-publish@release/v1
|
|
with:
|
|
password: ${{ secrets.PYPI_API_TOKEN }}
|
|
packages-dir: dist
|