Composable function transformations for text-space programs
Project description
autoform
Trace once. Transform freely.
Composable function transformations for text-space programs[^spaces].
JAX-like, but for text-space programs: trace a Python function into an IR, then apply program transforms around it.
Quickstart - Composition - Concurrency - Reference - GitHub - Documentation
[^spaces]: A text-space program is a traced program whose active values and feedback live in text-like leaves such as strings and structured LM outputs. The same machinery can be extended to other spaces by registering traceable values, avals, zeros, cotangent accumulators, and operator dispatch. See the array extension recipe for a concrete NumPy-backed example.
pip install git+https://github.com/ASEM000/autoform.git
Set provider credentials for the active LM client. For OpenAI through LiteLLM:
export OPENAI_API_KEY=...
Quickstart
The quickstart writes one function, traces it once, then reuses the same IR in a few ways.
import autoform as af
def explain(topic: str) -> str:
prompt = af.format("Explain {} in one paragraph.", topic)
msg = dict(role="user", content=prompt)
return af.lm_call([msg], model="gpt-5.5")
# trace with a representative input; this records structure
ir = af.trace(explain)("placeholder topic")
# execute the same ir with real input
answer = ir.call("recursion")
print(answer)
Expected result: one paragraph about recursion.
Batch the same program without rewriting explain:
# batch vectorizes the original ir over the input leaf
topics = ["recursion", "gravity", "memoization"]
answers = af.batch(ir).call(topics)
assert len(answers) == len(topics)
The result is one answer per topic.
Send output feedback backward to the original input:
# pullback returns the output and feedback for the original inputs
pb_ir = af.pullback(ir)
answer, (topic_hint,) = pb_ir.call(("recursion",), "too abstract")
print(topic_hint)
Expected result: text feedback for the input topic.
Compose both:
# one pullback per topic, batched by the transform
topics = ["recursion", "gravity", "memoization"]
critiques = ["too abstract", "too terse", "needs an example"]
composed = af.batch(af.pullback(ir))
answers, (topic_hints,) = composed.call((topics,), critiques)
assert len(topic_hints) == len(topics)
That last line is the core design: pullback(ir) returns an IR, and batch
accepts an IR.
Why
A text-space program written as ordinary Python tends to grow a second implementation for each new execution concern: batching, feedback, concurrency, debugging, or provider routing.
autoform keeps those concerns outside the function. It records the function
once as an IR, then applies transforms and execution contexts around that
recorded program. The quickstart shows the split: write normal Python, trace it
once, then decide how to transform or run it.
Composition
The pieces do different jobs:
| Job | API | For |
|---|---|---|
| Transform an IR | batch, pullback, pushforward, sched, dce |
Build another IR from an existing IR. |
| Customize a boundary | @af.custom |
Give a traceable Python function transform-specific rules. |
| Wrap tracing or execution | memoize, lm_client, collect, inject, tag, fold |
Change behavior inside a with block. |
| Choose execution mode | .call(...), .acall(...) |
Run the same IR synchronously or asynchronously. |
Concurrency
Write the function sequentially. Schedule the IR afterward.
import asyncio
import autoform as af
def compare(topic: str) -> str:
explain_prompt = af.format("Explain {} in one sentence.", topic)
example_prompt = af.format("Give one concrete example of {}.", topic)
explain_msg = dict(role="user", content=explain_prompt)
example_msg = dict(role="user", content=example_prompt)
explanation = af.lm_call([explain_msg], model="gpt-5.5")
example = af.lm_call([example_msg], model="gpt-5.5")
combine_prompt = af.format("Combine these:\n{}\n{}", explanation, example)
combine_msg = dict(role="user", content=combine_prompt)
return af.lm_call([combine_msg], model="gpt-5.5")
ir = af.trace(compare)("placeholder topic")
scheduled = af.sched(ir)
answer = asyncio.run(scheduled.acall("recursion"))
flowchart TD
topic["topic"] --> explain["LM: explain"]
topic --> example["LM: example"]
explain --> combine["LM: combine"]
example --> combine
There is no async def in compare. Use .call(...) for a sync run and
.acall(...) for an async run.
Debugging
checkpoint labels an intermediate. collect and inject wrap execution.
def pipeline(topic: str) -> str:
draft_prompt = af.format("Draft one sentence about {}.", topic)
draft_msg = dict(role="user", content=draft_prompt)
draft = af.lm_call([draft_msg], model="gpt-5.5")
draft = af.checkpoint(draft, key="draft", collection="debug")
final_prompt = af.format("Tighten this answer:\n{}", draft)
final_msg = dict(role="user", content=final_prompt)
return af.lm_call([final_msg], model="gpt-5.5")
ir = af.trace(pipeline)("placeholder topic")
with af.collect(collection="debug") as captured:
result = ir.call("recursion")
with af.inject(collection="debug", values={"draft": ["Recursion calls itself."]}):
result = ir.call("recursion")
The original function and IR stay the same. The context around execution changes what happens at checkpointed values.
Agents
Tool-use agents are just traced programs with structured outputs, switch
branches, and bounded while_loop state.
flowchart TD
question["question"] --> state["state"]
state --> condition{"continue?"}
condition -- "yes" --> decision{"tool?"}
decision -- "search" --> tool["search branch"]
tool --> state
decision -- "done" --> result["result"]
condition -- "no" --> result
Because the agent is one IR, the same transforms still apply:
agent_ir = af.trace(agent)("question")
batched_feedback = af.batch(af.pullback(agent_ir))
See the Tool-Use Agent recipe for the full version.
Reference
Early development: API Reference may change before a stable release.
Project details
Release history Release notifications | RSS feed
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distribution
Built Distribution
Filter files by name, interpreter, ABI, and platform.
If you're not sure about the file name format, learn more about wheel file names.
Copy a direct link to the current filters
File details
Details for the file autoform-0.3.0.tar.gz.
File metadata
- Download URL: autoform-0.3.0.tar.gz
- Upload date:
- Size: 103.4 kB
- Tags: Source
- Uploaded using Trusted Publishing? Yes
- Uploaded via: uv/0.11.17 {"installer":{"name":"uv","version":"0.11.17","subcommand":["publish"]},"python":null,"implementation":{"name":null,"version":null},"distro":{"name":"Ubuntu","version":"24.04","id":"noble","libc":null},"system":{"name":null,"release":null},"cpu":null,"openssl_version":null,"setuptools_version":null,"rustc_version":null,"ci":true}
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
a04c8106458274787ae350743e6744abc75fd910fce68c977d7ff32e3a28ab59
|
|
| MD5 |
b85c3c8dce3968887574e85f63c413ee
|
|
| BLAKE2b-256 |
8050f2813e63e7db98a9ef613c5412ce2c5e691f9407dd13f238c40be3df0baa
|
File details
Details for the file autoform-0.3.0-py3-none-any.whl.
File metadata
- Download URL: autoform-0.3.0-py3-none-any.whl
- Upload date:
- Size: 71.0 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? Yes
- Uploaded via: uv/0.11.17 {"installer":{"name":"uv","version":"0.11.17","subcommand":["publish"]},"python":null,"implementation":{"name":null,"version":null},"distro":{"name":"Ubuntu","version":"24.04","id":"noble","libc":null},"system":{"name":null,"release":null},"cpu":null,"openssl_version":null,"setuptools_version":null,"rustc_version":null,"ci":true}
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
8324602db07e02a4400d0dd5c78cd298ae4d7bd06a4c38515e683cf8ffc58f92
|
|
| MD5 |
100f9c537fc2ae7ae56b1af1319472e7
|
|
| BLAKE2b-256 |
a8eaae89c33d57b72866cd88b8af81455d5d946b0a4b255b4bdf4549ef65f5f5
|