From messy analysis folders to traceable, publication-oriented figures.
FigOps is a small research-ops toolkit for figure work: it reads a project config, checks the declared data contract, runs analysis and plotting scripts, applies journal/presentation styling, and records enough provenance to make the figure auditable later.
It is intentionally boring in the best way: one config, one command, clear inputs, repeatable outputs.
python -m pip install figops
figops --help
figops-mcp --smokeIf those commands work, the CLI is installed and the MCP surface is alive.
Research figures often start as a few scripts and a folder of exported data. That works until the figure changes, a collaborator asks where a value came from, or a manuscript revision needs the same plot in a different journal style.
FigOps keeps that workflow lightweight while making the important parts explicit:
- Data is the API — inputs are declared, checked, and traceable.
- Figures are rebuildable — analysis, plotting, diagrams, and assembly live behind the same project contract.
- Style is reusable — journal and presentation targets are selected through config instead of one-off plotting edits.
- Agents can inspect safely — the MCP server exposes read, render, and smoke surfaces for tool-assisted figure workflows.
| Item | Status |
|---|---|
| Source checkout | 0.17.11 release metadata (pyproject.toml) |
| Published package | figops==0.17.11 is the latest locally documented PyPI release |
| TestPyPI dry run | figops==0.17.11 was published and install-smoke verified |
| Python | 3.12+ |
| License | Apache-2.0 for public package distribution |
| Commands | figops, figops-mcp |
| Compatibility aliases | graphhub, graphhub-mcp |
| GitHub Release | v0.17.11 is the latest locally documented release asset |
The source checkout, public PyPI package, and GitHub Release asset are aligned
at 0.17.11.
For normal use:
python -m pip install figopsFor a pinned, reproducible install:
python -m pip install figops==0.17.11If you need the exact GitHub Release asset:
gh release download v0.17.11 --repo Moonweave-Research/figops --pattern "*.whl" --dir dist-release
python -m pip install dist-release/figops-0.17.11-py3-none-any.whl
figops-mcp --smokeFor an installed package, verify the published CLI surface first:
figops --help
figops-mcp --smoke
figops-mcp doctorFor a source checkout, use hub_uv.py instead of bare uv run so the managed
environment stays outside the repository:
python hub_uv.py sync
python hub_uv.py --print-env
python hub_uv.py run python figops_mcp_server.py --hub-path . --research-root . --runtime-root .omo/evidence/task-6-runtime doctor --json
python hub_uv.py run python figops_mcp_server.py doctor
python hub_uv.py run python -m pytest tests/test_doctor.py -qIf uv is not installed, the source-checkout path cannot bootstrap itself. Use
an already prepared Python 3.12+ environment with the FigOps runtime/dev
dependencies, or install uv outside the repository and rerun the wrapper
commands. The JSON doctor output reports the active Python, uv on PATH, the
resolved runtime root, and the planned external uv environment/cache paths. R is
only required for projects that declare lang: R.
Create a new figure project:
figops --init --project my_figure_project
cd my_figure_projectThat creates a scaffold with:
my_figure_project/
├── project_config.yaml
├── raw/
│ └── example_input.csv
└── hub_scripts/
├── analyze.R
├── plot.py
└── project_context.py
Run the project:
figops --project . --step allFor a first sanity check, list the available CLI options:
figops --helpA FigOps run coordinates the work around a research figure:
- read
project_config.yaml, - resolve declared input files,
- validate data contracts and research-ops rules,
- run analysis scripts when needed,
- render figures, diagrams, or assembled panels,
- apply the selected style profile,
- write provenance and runtime metadata for auditability.
A minimal config looks like this:
project:
name: "Example Study"
visual_style:
target_format: nature
font_scale: 1.0
profile: baseline
pipeline:
analysis:
- script: "hub_scripts/analyze.R"
lang: R
cache: true
figures:
- id: Fig1
script: "hub_scripts/plot.py"
output: "results/figures/Fig1.png"
cache: trueR is only required when your configured analysis scripts use R. Python plotting and package/MCP smoke checks run from the Python package install.
# Choose a configured project interactively
figops
# List configured projects
figops --list-projects
# Run the full pipeline for one project
figops --project "ProjectName" --step all
# Re-render figures only
figops --project "ProjectName" --step plot
# Re-render diagrams only
figops --project "ProjectName" --step diagrams
# Force a clean rerun and bypass cache
figops --project "ProjectName" --step all --forceFrom a source checkout, use the repo-local runtime wrapper:
python hub_uv.py sync
python hub_uv.py run python orchestrator.py --list-projects
python hub_uv.py run python -m pytest tests/test_runtime_paths.py -qhub_uv.py keeps the Python runtime outside the repository so the working tree
does not get polluted with local virtualenv state.
FigOps includes a Model Context Protocol server entry point:
figops-mcp --smokeA healthy smoke response looks like:
{"status": "ok", "health_status": "ok", "tool_surface": "figops_mcp"}Use this before wiring FigOps into Claude, Codex, or another MCP-capable client.
For compatibility with earlier local setups, graphhub-mcp remains available as
an alias.
| Symptom | What to try |
|---|---|
project_config.yaml not found |
Run figops --init --project "<project>" or move into a configured project. |
Project directory not found |
Run figops --list-projects and copy the exact configured name. |
| Strict lockfile errors | --strict-lock is for reproducibility checks. For quick local rendering, rerun without strict mode. |
| Google Drive files feel stuck | Let Drive finish syncing, then rerun. The prefetch layer can help with declared inputs, but it cannot repair a broken Drive login/session. |
Source-checkout tests fail with No module named pytest |
Run python hub_uv.py sync --group dev, then rerun tests through python hub_uv.py run python -m pytest .... |
Doctor reports missing pandas, matplotlib, or yaml |
Run python hub_uv.py sync from the checkout, or use a Python 3.12+ environment with FigOps runtime dependencies installed. |
| R script fails immediately | Confirm Rscript is available if your project config uses lang: R. |
Release candidates are checked with the packaging and test gates below:
uv build
python scripts/package_metadata_smoke.py
python scripts/public_package_surface.py
python scripts/consumer_install_smoke.py
uv run --with twine python -m twine check dist/*
python hub_uv.py run python -m pytest -q
python hub_uv.py run ruff check .After release assets are uploaded, verify both the GitHub artifact and the public install path:
python scripts/github_release_asset_smoke.py
python -m pip install figops==0.17.11
figops-mcp --smokePublishing uses the manual Trusted Publishing workflow in
.github/workflows/publish.yml: TestPyPI
first, install smoke, then PyPI. See
docs/packaging/trusted-publishing.md
for the exact runbook.
| Path | Purpose |
|---|---|
orchestrator.py |
CLI entry point and top-level pipeline coordinator |
hub_core/ |
Config loading, validation, cache, provenance, process execution, MCP logic |
hub_core/mcp/ |
MCP server, schemas, transport, resources, prompts, and tool handlers |
plotting/ |
Reusable plotting helpers and figure assembly utilities |
themes/ |
Journal and presentation style presets |
examples/ |
Small synthetic projects and package-facing examples |
scripts/ |
Release, packaging, and distribution checks |
tests/ |
Regression and contract tests |
docs/packaging/ |
PyPI readiness, clearance checklist, and Trusted Publishing runbook |
For the next release, keep the same conservative path:
- bump the version,
- rebuild wheel/sdist from a clean tree,
- confirm package artifacts exclude private docs/tests/research markers,
- publish to TestPyPI through
.github/workflows/publish.yml, - install-check from TestPyPI,
- promote the same version to PyPI,
- install-check from public PyPI.
FigOps is distributed under the Apache-2.0 license. See LICENSE and NOTICE.