phoenix2pytest

June 22, 2026 ยท View on GitHub

CI CodeQL OpenSSF Best Practices Tessl codecov Python License: MIT Last commit

Turn production LLM failures into regression tests. Automatically.

phoenix2pytest demo

Built for: Google Cloud Rapid Agent Hackathon, Arize track

Live demo: phoenix2pytest-etm7pvfo3a-nw.a.run.app - the example is pre-filled, just click Generate. Single trace at /, batch at /batch.

Stack: Google Cloud, Vertex AI, Gemini 2.5, Agent Builder, Arize Phoenix MCP, FastAPI, pytest

Status: Stable (v1.0.0). Originally built for the Devpost submission cycle of June 2026, now stable within the documented scope (see Limits).


The problem

You ship an LLM feature. Three weeks later, a Slack thread mentions a customer got a weird response. You dig. The prompt has been edited twice since release. The model has been quietly re-quantised by the provider. Nobody added a test that would have caught it.

Existing eval frameworks ask you to predict failures up front. You write evals against your LLM, run them, get scores. That works for known failure modes you can imagine. It does not work for the failure that just got escalated to your phone.

The idea

phoenix2pytest goes the other direction. It reads traces from your Arize Phoenix project, picks the ones flagged as failures, and synthesises pytest cases that would have caught them. Production traffic feeds your regression suite without manual translation.

Existing toolsphoenix2pytest
Direction: spec to eval to runDirection: trace to failure to test
You predict what to testYou react to what broke
Eval scoresConcrete pytest assertions
Catches what you imaginedCatches what actually happened

How it works

  1. Your LLM application emits traces to Arize Phoenix (standard OpenInference instrumentation).
  2. You annotate failed traces in the Phoenix UI (manual review, or via Phoenix evals).
  3. phoenix2pytest reads annotated traces via the Phoenix MCP server.
  4. A Gemini agent extracts evidence and assertion strategy per failure.
  5. A second pass synthesises a runnable pytest file.
  6. You drop the test into your repo and your CI catches the regression next time.

The pipeline runs end-to-end on a single trace (/) or on many annotated traces in one request (/batch), both on the web UI and Cloud Run. Batch mode groups traces by failure mode and folds shared modes into one parametrised test.

Architecture

flowchart LR
    A[Phoenix project<br/>annotated traces] -->|MCP| B[Agent Builder<br/>orchestrator]
    B --> C[Gemini Flash<br/>extractor]
    B --> D[Gemini Pro<br/>synthesiser]
    C --> D
    D --> E[Generated<br/>pytest file]
    E --> F[CI / dev<br/>runs pytest]

The orchestrator runs on Cloud Run, fetches traces through the Arize Phoenix MCP server, calls Gemini twice per trace (Flash for evidence extraction, Pro for code generation), and writes the synthesised test file.

Quickstart

The web UI is the primary entry point. A console-script CLI is planned (see roadmap).

Local web UI:

git clone https://github.com/golikovichev/phoenix2pytest
cd phoenix2pytest
pip install -e ".[dev]"

Create a .env file in the repo root:

PHOENIX_BASE_URL=https://app.phoenix.arize.com/s/your-space
PHOENIX_API_KEY=your-phoenix-api-key
GOOGLE_CLOUD_PROJECT=your-gcp-project
GOOGLE_CLOUD_LOCATION=us-central1
GOOGLE_GENAI_USE_VERTEXAI=True

Application Default Credentials are picked up automatically for Vertex AI, so no API key is needed if you have run gcloud auth application-default login. If you prefer the direct Gemini API, set GEMINI_API_KEY instead of the Vertex variables.

Run the FastAPI web UI and open it in a browser:

uvicorn phoenix2pytest.web:app --reload --port 8000
# http://localhost:8000

Cloud Run deploy (see cloudbuild.yaml for the full pipeline):

gcloud builds submit --config cloudbuild.yaml

Demo

A 3-minute walkthrough video accompanies the Devpost submission. The demo shows a real Phoenix trace with a hallucination, phoenix2pytest extracting the failure, generating a pytest file, the run showing red, a prompt fix, and the run showing green.

How is this different from DeepEval, Opik, pytest-evals, Langfuse?

Short answer: those tools are about running evals you wrote. phoenix2pytest is about generating tests from failures you saw. Different direction, different mental model.

ToolWhat it isWhen to use
DeepEvalpytest-style framework for writing LLM evalsYou know the failure modes you care about and want to define metrics
OpikLLM observability with pytest integrationYou want eval scores in CI
pytest-evalsMinimal pytest plugin for running evals at scaleYou want parametrised eval runs
LangfuseLLM tracing platform with evalsYou want production tracing plus scoring
phoenix2pytestGenerates pytest tests from observed failuresYou want your regression suite to keep up with production reality

You can use phoenix2pytest alongside the others. It does not compete with eval frameworks; it feeds them. The output of phoenix2pytest is a pytest file you can run via DeepEval, Opik, pytest-evals, or plain pytest. Your choice.

What it catches and does not catch

Catches:

  • Hallucinations of specific facts when those facts appear as identifiable strings in the bad output
  • Format breaks: JSON wrapped in markdown when pure JSON was demanded, missing fields, wrong types
  • Refusals where the model should have answered
  • Wrong reasoning when correct answer or clarification was reachable
  • Stale-data claims when the model invented current information

Does not catch yet:

  • Semantic-level paraphrased failures where the model fabricates the same fact in different words
  • Failures that only appear in long context or multi-turn flows
  • Subtle quality degradations without a clear bad-string pattern

The roadmap covers paraphrase tolerance via embedding-similarity assertions.

Limits

  • Loads matched traces in memory. For projects with hundreds of thousands of failed traces you will want a streaming variant.
  • Generated tests assume the same model the failure was observed against. Cross-model regression suites need explicit configuration.
  • Test quality depends on Gemini synthesis quality. For unusual failure modes, manual review of the generated test is prudent.

Honesty notes

  • The synthesised pytest is a starting point, not a final test. Engineers review it before commit.
  • The agent does not auto-classify random traces. It works on traces already labeled as failures (manual annotation, eval framework, or heuristic). This is intentional. Automatic failure detection on raw traces is not reliable for hallucinations of facts the classifier itself cannot verify.
  • Built and maintained solo. The scope is deliberately narrow (see Limits) and stable within it; the roadmap covers where it grows next.

Roadmap

  • v1.0 (June 2026): one-trace and many-trace generation; hallucination, format_break, refusal_bug, stale_data, wrong_reasoning, and off_topic_drift coverage; Cloud Run hosting; web UI; generated code is validated as parseable Python before it is returned.
  • v1.1: paraphrase-tolerant assertions via embedding similarity.
  • v1.2: multi-turn trace handling.
  • v1.3: phoenix2pytest console-script CLI and broader documentation.

Contributing

Contributions are welcome. See CONTRIBUTING.md for setup and the workflow.

License

MIT. See LICENSE.

Acknowledgements

Built on Arize Phoenix, Google Cloud Vertex AI, OpenTelemetry, and OpenInference semantic conventions. Thanks to the maintainers of all four projects.