The wheel's pyproject.toml has declared `molecule-runtime = "molecule_runtime.main:main_sync"` since the publish pipeline was created on 2026-04-26, but the function itself was never present in workspace/main.py — it lived in the pre-monorepo molecule-ai-workspace-runtime repo and was lost during the consolidation that made workspace/ the source of truth. The 0.1.15 wheel still had main_sync from a leftover snapshot, so the regression went unnoticed until 0.1.16 (the first wheel built from the new source-of-truth) shipped. Symptom: every workspace container restart loops with ImportError: cannot import name 'main_sync' from 'molecule_runtime.main' — the molecule-runtime CLI script's first line tries to import the missing symbol. Workspaces stay in `provisioning` until the 10-min sweep marks them failed. Caught by .github/workflows/runtime-pin-compat.yml, which already imports the symbol by name as its smoke test. (That check kept failing red on every recent merge_group run; this PR fixes the underlying symbol-not-found instead of the smoke step.) Also strengthens publish-runtime.yml's wheel smoke from `import molecule_runtime.main` (loads the module — passes even when entry-point target is missing) to `from molecule_runtime.main import main_sync` (the actual contract the CLI script needs). Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
231 lines
11 KiB
YAML
231 lines
11 KiB
YAML
name: publish-runtime
|
|
|
|
# Publishes molecule-ai-workspace-runtime to PyPI from monorepo workspace/.
|
|
# Monorepo workspace/ is the only source-of-truth for runtime code; this
|
|
# workflow is the bridge from monorepo edits to the PyPI artifact that
|
|
# the 8 workspace-template-* repos depend on.
|
|
#
|
|
# Triggered by:
|
|
# - Pushing a tag matching `runtime-vX.Y.Z` (the version is derived from
|
|
# the tag — `runtime-v0.1.6` publishes `0.1.6`).
|
|
# - Manual workflow_dispatch with an explicit `version` input (useful for
|
|
# dev/test releases without tagging the repo).
|
|
# - Auto: any push to `staging` that touches `workspace/**`. The version
|
|
# is derived by querying PyPI for the current latest and bumping the
|
|
# patch component. This closes the human-in-loop gap that caused the
|
|
# 2026-04-27 RuntimeCapabilities ImportError outage — adapter symbol
|
|
# additions in workspace/adapters/base.py used to require an operator
|
|
# to remember to publish; now the merge itself triggers the publish.
|
|
#
|
|
# The workflow:
|
|
# 1. Runs scripts/build_runtime_package.py to copy workspace/ →
|
|
# build/molecule_runtime/ with imports rewritten (`a2a_client` →
|
|
# `molecule_runtime.a2a_client`).
|
|
# 2. Builds wheel + sdist with `python -m build`.
|
|
# 3. Publishes to PyPI via the PyPA Trusted Publisher action (OIDC).
|
|
# No static API token is stored — PyPI verifies the workflow's
|
|
# OIDC claim against the trusted-publisher config registered for
|
|
# molecule-ai-workspace-runtime (Molecule-AI/molecule-core,
|
|
# publish-runtime.yml, environment pypi-publish).
|
|
#
|
|
# After publish: the 8 template repos pick up the new version on their
|
|
# next image rebuild (their requirements.txt pin
|
|
# `molecule-ai-workspace-runtime>=0.1.0`, so any new release is eligible).
|
|
# To force-pull immediately, bump the pin in each template repo's
|
|
# requirements.txt and merge — that triggers their own publish-image.yml.
|
|
|
|
on:
|
|
push:
|
|
tags:
|
|
- "runtime-v*"
|
|
branches:
|
|
- staging
|
|
paths:
|
|
# Auto-publish when staging gets workspace/ changes. Path filter
|
|
# ONLY applies to branch pushes — tag pushes still fire regardless.
|
|
- "workspace/**"
|
|
workflow_dispatch:
|
|
inputs:
|
|
version:
|
|
description: "Version to publish (e.g. 0.1.6). Required for manual dispatch."
|
|
required: true
|
|
type: string
|
|
|
|
permissions:
|
|
contents: read
|
|
|
|
# Serialize publishes so two staging merges landing seconds apart don't
|
|
# both compute "latest+1" and race on PyPI upload. The second one waits.
|
|
concurrency:
|
|
group: publish-runtime
|
|
cancel-in-progress: false
|
|
|
|
jobs:
|
|
publish:
|
|
runs-on: ubuntu-latest
|
|
environment: pypi-publish
|
|
permissions:
|
|
contents: read
|
|
id-token: write # PyPI Trusted Publisher (OIDC) — no PYPI_TOKEN needed
|
|
outputs:
|
|
version: ${{ steps.version.outputs.version }}
|
|
steps:
|
|
- uses: actions/checkout@v4
|
|
|
|
- uses: actions/setup-python@v5
|
|
with:
|
|
python-version: "3.11"
|
|
cache: pip
|
|
|
|
- name: Derive version (tag, manual input, or PyPI auto-bump)
|
|
id: version
|
|
run: |
|
|
if [ "${{ github.event_name }}" = "workflow_dispatch" ]; then
|
|
VERSION="${{ inputs.version }}"
|
|
elif echo "$GITHUB_REF_NAME" | grep -q "^runtime-v"; then
|
|
# Tag is `runtime-vX.Y.Z` — strip the prefix.
|
|
VERSION="${GITHUB_REF_NAME#runtime-v}"
|
|
else
|
|
# Auto-publish from staging push. Query PyPI for the current
|
|
# latest and bump the patch component. concurrency: group above
|
|
# serializes parallel staging merges so we don't race on the
|
|
# bump. If PyPI is unreachable, fail loud — better to skip a
|
|
# publish than to overwrite an existing version.
|
|
LATEST=$(curl -fsS --retry 3 https://pypi.org/pypi/molecule-ai-workspace-runtime/json \
|
|
| python -c "import sys,json; print(json.load(sys.stdin)['info']['version'])")
|
|
MAJOR=$(echo "$LATEST" | cut -d. -f1)
|
|
MINOR=$(echo "$LATEST" | cut -d. -f2)
|
|
PATCH=$(echo "$LATEST" | cut -d. -f3)
|
|
VERSION="${MAJOR}.${MINOR}.$((PATCH+1))"
|
|
echo "Auto-bumped from PyPI latest $LATEST -> $VERSION"
|
|
fi
|
|
if ! echo "$VERSION" | grep -qE '^[0-9]+\.[0-9]+\.[0-9]+(\.dev[0-9]+|rc[0-9]+|a[0-9]+|b[0-9]+|\.post[0-9]+)?$'; then
|
|
echo "::error::version $VERSION does not match PEP 440"
|
|
exit 1
|
|
fi
|
|
echo "version=$VERSION" >> "$GITHUB_OUTPUT"
|
|
echo "Publishing molecule-ai-workspace-runtime $VERSION"
|
|
|
|
- name: Install build tooling
|
|
run: pip install build twine
|
|
|
|
- name: Build package from workspace/
|
|
run: |
|
|
python scripts/build_runtime_package.py \
|
|
--version "${{ steps.version.outputs.version }}" \
|
|
--out "${{ runner.temp }}/runtime-build"
|
|
|
|
- name: Build wheel + sdist
|
|
working-directory: ${{ runner.temp }}/runtime-build
|
|
run: python -m build
|
|
|
|
- name: Verify package contents (sanity)
|
|
working-directory: ${{ runner.temp }}/runtime-build
|
|
run: |
|
|
python -m twine check dist/*
|
|
# Smoke-import the built wheel to catch import-rewrite mistakes
|
|
# before they hit PyPI. Asserts on STABLE INVARIANTS only —
|
|
# symbols + classes that are part of the package's public
|
|
# contract (BaseAdapter interface, the canonical a2a sentinel,
|
|
# core submodules). Don't add feature-flag-style assertions
|
|
# here — they fire false-positive every time staging is mid-
|
|
# release of that feature.
|
|
python -m venv /tmp/smoke
|
|
/tmp/smoke/bin/pip install --quiet dist/*.whl
|
|
WORKSPACE_ID=00000000-0000-0000-0000-000000000000 \
|
|
PLATFORM_URL=http://localhost:8080 \
|
|
/tmp/smoke/bin/python -c "
|
|
# Importing main is the strongest smoke test we can do here:
|
|
# main.py is the entry point and pulls every other module
|
|
# transitively. If the build script missed an import rewrite
|
|
# (e.g. left a bare \`from transcript_auth import ...\` instead
|
|
# of \`from molecule_runtime.transcript_auth import ...\` — the
|
|
# 0.1.16 incident), this fails with ModuleNotFoundError instead
|
|
# of shipping to PyPI and breaking every workspace startup.
|
|
# Import the entry-point target by NAME — not just the module.
|
|
# The wheel's pyproject.toml declares
|
|
# `molecule-runtime = molecule_runtime.main:main_sync` so if
|
|
# main_sync goes missing (it did in 0.1.16-0.1.18), every
|
|
# workspace startup fails with `ImportError: cannot import name
|
|
# 'main_sync'`. Plain `import molecule_runtime.main` doesn't
|
|
# catch that because the module loads fine.
|
|
from molecule_runtime.main import main_sync # noqa: F401
|
|
from molecule_runtime import a2a_client, a2a_tools
|
|
from molecule_runtime.builtin_tools import memory
|
|
from molecule_runtime.adapters import get_adapter, BaseAdapter, AdapterConfig
|
|
# Stable invariants: package exports + BaseAdapter shape.
|
|
assert a2a_client._A2A_ERROR_PREFIX, 'a2a_client missing error sentinel'
|
|
assert callable(get_adapter), 'adapters.get_adapter must be callable'
|
|
assert hasattr(BaseAdapter, 'name'), 'BaseAdapter interface broken'
|
|
assert hasattr(AdapterConfig, '__init__'), 'AdapterConfig dataclass missing'
|
|
print('✓ smoke import passed')
|
|
"
|
|
|
|
- name: Publish to PyPI (Trusted Publisher / OIDC)
|
|
# PyPI side is configured: project molecule-ai-workspace-runtime →
|
|
# publisher Molecule-AI/molecule-core, workflow publish-runtime.yml,
|
|
# environment pypi-publish. The action mints a short-lived OIDC
|
|
# token and exchanges it for a PyPI upload credential — no static
|
|
# API token in this repo's secrets.
|
|
uses: pypa/gh-action-pypi-publish@release/v1
|
|
with:
|
|
packages-dir: ${{ runner.temp }}/runtime-build/dist/
|
|
|
|
cascade:
|
|
# After PyPI accepts the upload, fan out a repository_dispatch to each
|
|
# template repo so they rebuild their image against the new runtime.
|
|
# Each template's `runtime-published.yml` receiver picks up the event,
|
|
# pulls the new PyPI version (their requirements.txt pin is `>=`), and
|
|
# republishes ghcr.io/molecule-ai/workspace-template-<runtime>:latest.
|
|
#
|
|
# Soft-fail per repo: if one template's dispatch fails (perms missing,
|
|
# repo archived, etc.) we still try the others and surface the failures
|
|
# in the workflow summary instead of aborting the whole cascade.
|
|
needs: publish
|
|
runs-on: ubuntu-latest
|
|
steps:
|
|
- name: Fan out repository_dispatch
|
|
env:
|
|
# Fine-grained PAT with `actions:write` on the 8 template repos.
|
|
# GITHUB_TOKEN can't fire dispatches across repos — needs an explicit
|
|
# token. Stored as a repo secret; rotate per the standard schedule.
|
|
DISPATCH_TOKEN: ${{ secrets.TEMPLATE_DISPATCH_TOKEN }}
|
|
# Single source of truth: the publish job's output, which handles
|
|
# tag/manual-input/auto-bump uniformly. The previous fallback
|
|
# (`steps.version.outputs.version` from inside the cascade job)
|
|
# was a dead reference — different job, no shared step scope.
|
|
RUNTIME_VERSION: ${{ needs.publish.outputs.version }}
|
|
run: |
|
|
set +e # don't abort on a single repo failure — collect them all
|
|
if [ -z "$DISPATCH_TOKEN" ]; then
|
|
echo "::warning::TEMPLATE_DISPATCH_TOKEN secret not set — skipping cascade. PyPI was published; templates will pick up the new version on their own next rebuild."
|
|
exit 0
|
|
fi
|
|
VERSION="$RUNTIME_VERSION"
|
|
if [ -z "$VERSION" ]; then
|
|
echo "::error::publish job did not expose a version output — cascade cannot fan out"
|
|
exit 1
|
|
fi
|
|
TEMPLATES="claude-code langgraph crewai autogen deepagents hermes gemini-cli openclaw"
|
|
FAILED=""
|
|
for tpl in $TEMPLATES; do
|
|
REPO="Molecule-AI/molecule-ai-workspace-template-$tpl"
|
|
STATUS=$(curl -sS -o /tmp/dispatch.out -w "%{http_code}" \
|
|
-X POST "https://api.github.com/repos/$REPO/dispatches" \
|
|
-H "Authorization: Bearer $DISPATCH_TOKEN" \
|
|
-H "Accept: application/vnd.github+json" \
|
|
-H "X-GitHub-Api-Version: 2022-11-28" \
|
|
-d "{\"event_type\":\"runtime-published\",\"client_payload\":{\"runtime_version\":\"$VERSION\"}}")
|
|
if [ "$STATUS" = "204" ]; then
|
|
echo "✓ dispatched $tpl ($VERSION)"
|
|
else
|
|
echo "::warning::✗ failed to dispatch $tpl: HTTP $STATUS — $(cat /tmp/dispatch.out)"
|
|
FAILED="$FAILED $tpl"
|
|
fi
|
|
done
|
|
if [ -n "$FAILED" ]; then
|
|
echo "::warning::Cascade incomplete. Failed templates:$FAILED"
|
|
# Don't fail the whole job — PyPI publish already succeeded;
|
|
# operators can retry the failed templates manually.
|
|
fi
|