Skip to main content

Graph-based agent framework powered by oauth-codex

Project description

Fabrix

Language: English | 한국어
API Guides: English | 한국어

Overview

Fabrix is a graph-based agent framework built on top of oauth-codex>=2.3.0. It provides a structured execution graph with streaming events for tool-driven workflows.

Key Features

  • Graph-based 3-state execution: reasoning, tool_call, response
  • Structured state outputs powered by Pydantic models
  • Sequential tool execution with strict payload validation
  • Async streaming event API for step-by-step observability
  • Multimodal input with explicit message models: TextMessage, ImageMessage

Installation

pip install fabrix-ai

Quickstart

import asyncio

from pydantic import BaseModel

from fabrix import Agent
from fabrix.events import (
    ReasoningEvent,
    ResponseEvent,
    TaskFailedEvent,
    ToolEvent,
)
from fabrix.messages import TextMessage
from fabrix.tools import ToolOutput


class AddInput(BaseModel):
    a: int
    b: int


def add_numbers(payload: AddInput) -> ToolOutput:
    return ToolOutput.json({"sum": payload.a + payload.b})


async def main() -> None:
    agent = Agent(
        instructions="You are a precise assistant.",
        model="gpt-5.3-codex",
        tools=[add_numbers],
    )

    messages = [TextMessage(text="Use add_numbers to compute 3 + 9")]
    async for event in agent.run_stream(messages=messages):
        print(f"[step={event.step}] {event.event_type}")

        if isinstance(event, ReasoningEvent):
            print("reasoning:", event.reasoning)
            print("focus:", event.focus)
        elif isinstance(event, ToolEvent):
            if event.phase == "start":
                print("tool call:", event.tool_name, event.arguments)
            elif event.result is not None:
                print("tool result:", event.result.model_dump())
        elif isinstance(event, ResponseEvent):
            if event.response is not None:
                print("response:", event.response)
            if event.parts is not None:
                print("parts:", [part.model_dump(mode="json") for part in event.parts])
            if event.response is None and event.parts is None:
                print("response: <empty>")
        elif isinstance(event, TaskFailedEvent):
            print("failed:", event.error_code, event.message)


asyncio.run(main())

Message Models

Fabrix input is now list[TextMessage | ImageMessage].

  • TextMessage(role: str = "user", text: str)
  • ImageMessage(role: str = "user", image: str | Path | bytes, text: str | None = None)
  • Unknown message fields are rejected at construction time.

ImageMessage.image accepts:

  • remote URL (https://...)
  • local path (Path or string path)
  • raw bytes (bytes), normalized to a data URL for model calls

Multimodal Input

from fabrix.messages import ImageMessage, TextMessage

messages = [
    TextMessage(text="Describe this screenshot"),
    ImageMessage(image="https://example.com/screenshot.png"),
    TextMessage(text="Focus on errors"),
]

async for event in agent.run_stream(messages=messages):
    ...

Tool Contract

Fabrix accepts tools in this shape:

def tool(payload: BaseModel) -> ToolOutput: ...
  • The tool must accept exactly one parameter.
  • The parameter type must be a Pydantic BaseModel.
  • The return type must be ToolOutput (breaking in v1.2.0).
  • Runtime arguments must be a JSON object matching payload fields.
  • Extra argument keys are rejected.
  • Both sync and async tools are supported.
  • ToolOutput.image(...) keeps http(s)/data: values as-is.
  • ToolOutput.image(...) normalizes file://, local paths, and bytes to local absolute file references.
  • Tool-call argument strictness is enforced by model output_schema with strict_output=True.
  • Prompt policy and runtime context are no longer duplicated; runtime control context is appended as a final control message.
  • During LLM history serialization, reasoning/tool_call/response (and legacy tool_result) records are preserved, and local image references are re-normalized to data URLs.

Event Stream

run_stream(...) yields these event types:

  • reasoning
  • tool (phase="start" / phase="finish")
  • response
  • task_failed

reasoning is a step-level decision trace / plan summary, not raw internal chain-of-thought. response events now support both response: str | None and parts (structured text/image/json parts); both fields may be None for an empty response event. Terminate by setting next_state=null in response state.

Migration (Breaking)

run_task_stream(task, images, context) has been removed.

  • Before: agent.run_task_stream(task=..., images=..., context=...)
  • After: agent.run_stream(messages=[...])

Mapping:

  • task text -> TextMessage(text="...")
  • images -> ImageMessage(image="..." | Path(...) | b"...")
  • context -> include serialized context in TextMessage.text

Tool migration:

  • Before: tool returns str / dict / scalar / arbitrary JSON-like objects
  • After: tool must return ToolOutput (for example ToolOutput.text(...), ToolOutput.json(...), ToolOutput.image(...))

Documentation

Examples

Notes

  • Public runtime entry point is fabrix.Agent.
  • Execution defaults are fixed internally: max_steps=128 and no public per-tool timeout option.
  • On successful completion, the stream ends right after the final response event (next_state=null in response state).
  • If max_steps is reached, the stream ends without emitting an additional terminal event.

Project details


Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distribution

fabrix_ai-1.5.1.tar.gz (39.3 kB view details)

Uploaded Source

Built Distribution

If you're not sure about the file name format, learn more about wheel file names.

fabrix_ai-1.5.1-py3-none-any.whl (24.7 kB view details)

Uploaded Python 3

File details

Details for the file fabrix_ai-1.5.1.tar.gz.

File metadata

  • Download URL: fabrix_ai-1.5.1.tar.gz
  • Upload date:
  • Size: 39.3 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.13.7

File hashes

Hashes for fabrix_ai-1.5.1.tar.gz
Algorithm Hash digest
SHA256 fcbc1d9643c355613a2bbcc446a7daaad011fa19500033a07d6b2e8c90c7d68e
MD5 e4d554aef3148bc5ea157d7838aa9e3f
BLAKE2b-256 35fef88317bbf6a1058d1d96b7dd9e8565224abd8eaf6fa82dd6b67f0198a2de

See more details on using hashes here.

Provenance

The following attestation bundles were made for fabrix_ai-1.5.1.tar.gz:

Publisher: publish-pypi.yml on smturtle2/fabrix

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

File details

Details for the file fabrix_ai-1.5.1-py3-none-any.whl.

File metadata

  • Download URL: fabrix_ai-1.5.1-py3-none-any.whl
  • Upload date:
  • Size: 24.7 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.13.7

File hashes

Hashes for fabrix_ai-1.5.1-py3-none-any.whl
Algorithm Hash digest
SHA256 5b4e2860fd070587e90995069cd8d85db488c25712b65d22f3a615a33208ef6c
MD5 1af04b2e518214646b8b5b337ab8bf7f
BLAKE2b-256 f3017ae66321e47e79315d73b41811e23b3a9e2ef739d4e1a2075c9e07bf34cf

See more details on using hashes here.

Provenance

The following attestation bundles were made for fabrix_ai-1.5.1-py3-none-any.whl:

Publisher: publish-pypi.yml on smturtle2/fabrix

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

Supported by

AWS Cloud computing and Security Sponsor Datadog Monitoring Depot Continuous Integration Fastly CDN Google Download Analytics Pingdom Monitoring Sentry Error logging StatusPage Status page