Que Claude genere interfaces de usuario con A2UI en sus respuestas.
Project description
___ _ _ _____
/\ |__ \ | | | | |_ _|
/ \ ) | | | | | | |
/ /\ \ / / | | | | | |
/ ____ \ / /_ | |__| | _| |_
/_/ \_\ |____| \____/ |_____|
anthropic-a2ui
Let Claude generate user interfaces with A2UI in its responses.
anthropic-a2ui bridges the A2UI protocol with the Anthropic SDK, enabling
Claude to emit declarative, interactive user interfaces as part of its
natural responses. Instead of returning plain text, Claude can now build
forms, dashboards, product cards, surveys, calendars, configuration panels,
modals, and any other UI composed from the A2UI component catalog — all
validated, repaired, and ready to render in any A2UI-compatible renderer
(Lit, React, Angular, Flutter, SwiftUI, and more).
The package does not wrap or replace the Anthropic SDK. It composes with
it: you use anthropic directly and plug in the A2UI pieces where they add
value. The system prompt, the tool definition, the stream parser, the
validator, the automatic repair layer, and the retry-with-feedback loop are
all designed to be transparent and composable.
How it works
When a user asks Claude for an interface — "make me a contact form", "show me a shopping cart", "build me a weather panel" — the following happens behind the scenes:
-
Prompt injection:
ClaudeA2uiPromptBuilderconstructs a system prompt containing the A2UI JSON Schema, few-shot examples, the list of 59 valid icon names, and the 14 catalog functions. Claude receives this as its system context and knows how to speak A2UI. -
Tool definition:
create_a2ui_toolproduces a tool definition (send_a2ui_json_to_client) that Claude can invoke to deliver UI. The schema is wrapped to comply with Anthropic's constraint againstoneOf/allOf/anyOfat the top level ofinput_schema. -
Streaming parse: As Claude streams its response,
ClaudeStreamParserintercepts tool-use events (or<a2ui-json>text blocks) and emitsResponsePartobjects containing either conversational text or a complete A2UI payload. -
Automatic repair: Before validation, the payload passes through five repair functions that fix known issues silently — invalid icon names, non-existent catalog functions, orphaned components, ambiguous DateTimeInput formats, and dynamic child lists in Row/Column.
-
Validation: The repaired payload is validated against the A2UI schema. If it fails, the error is fed back to Claude as a tool result, and Claude corrects itself on the next attempt (up to
max_retries). -
Delivery: The validated payload is wrapped in an
A2uiPartwith MIMEapplication/a2ui+json, ready to be sent to any A2UI renderer.
The user never mentions A2UI, components, functions, or schemas. They just talk naturally, and Claude responds with a working interface.
Installation
uv add anthropic-a2ui
# or
pip install anthropic-a2ui
Requires ANTHROPIC_API_KEY in the environment.
Quick start
The simplest way to use the package is generate_a2ui. One function call
handles everything: prompt construction, Claude invocation, streaming,
parsing, repair, validation, and retry-on-failure.
import anthropic
from anthropic_a2ui import generate_a2ui
client = anthropic.Anthropic()
result = generate_a2ui(
client,
"make me a registration form with name, email, and a submit button",
model="claude-haiku-4-5-20251001",
log_repairs=True,
)
if result.success:
# result.a2ui_json is a validated list of A2UI messages
# ready to be rendered by any A2UI-compatible renderer
render(result.a2ui_json)
for repair in result.repairs:
print(f"Repaired: {repair}")
else:
print(f"Failed after {result.attempts} attempts: {result.error}")
An async version is also available for integration with async web frameworks like FastAPI or Starlette:
import anthropic
from anthropic_a2ui import generate_a2ui_async
client = anthropic.AsyncAnthropic()
result = await generate_a2ui_async(
client,
"make me a registration form",
model="claude-haiku-4-5-20251001",
)
Usage modes
The package offers four complementary ways to generate A2UI, depending on the level of control and the use case.
Mode 1: Single call with generate_a2ui
Best for one-shot UI generation. The user asks for something, Claude
responds with a complete A2UI payload. If the payload is invalid, the
package automatically feeds the validation error back to Claude and
retries (up to max_retries times, default 2). Claude corrects itself
and resubmits.
result = generate_a2ui(
client,
"make me a contact form with name, email, subject, and message",
model="claude-opus-4-8",
max_retries=3,
)
if result.success:
render(result.a2ui_json)
Key parameters:
builder: customClaudeA2uiPromptBuilder(defaults to v0.9 Basic Catalog).model: any Claude model that supports tool use.max_tokens: per-attempt token budget (default 8192).max_retries: how many times to retry on validation failure (default 2).use_cache: enable prompt caching to reduce cost ~80% on repeated calls (defaultTrue).log_repairs: record what was repaired for debugging (defaultFalse).
Mode 2: Multi-turn conversation with A2uiConversation
Best for iterative UI design. The user starts with a request, then refines it across multiple turns: "make me a form" -> "add a phone field" -> "change the button color to blue". Claude maintains the full conversation context and generates a new A2UI payload with all changes applied in each turn.
import anthropic
from anthropic_a2ui import A2uiConversation
client = anthropic.Anthropic()
conv = A2uiConversation(client, model="claude-haiku-4-5-20251001")
# Turn 1: create the initial form
r1 = conv.send("make me a contact form")
# r1.a2ui_json contains the form
# r1.success is True
# Turn 2: add a field
r2 = conv.send("add a phone number field")
# r2.a2ui_json contains the form + phone field
# Turn 3: change the styling
r3 = conv.send("make the submit button red")
# r3.a2ui_json contains the form + phone field + red button
# Access the full conversation history
print(f"Turns: {len(conv.turns)}")
print(f"Last valid A2UI: {conv.last_a2ui_json is not None}")
# Reset to start a new conversation
conv.reset()
The async version works identically:
import anthropic
from anthropic_a2ui import A2uiConversationAsync
client = anthropic.AsyncAnthropic()
conv = A2uiConversationAsync(client)
r1 = await conv.send("make me a form")
r2 = await conv.send("add an email field")
Mode 3: Forced JSON with structured outputs
Best when you want pure UI output with no conversational text. This mode
uses Anthropic's output_config.format structured output parameter with
json_schema to guarantee that the response is parseable JSON. No tools,
no tags — just a structured payload.
import anthropic
from anthropic_a2ui import (
ClaudeA2uiPromptBuilder,
create_a2ui_output_config,
parse_json_response,
)
builder = ClaudeA2uiPromptBuilder()
output_config = create_a2ui_output_config(builder.get_catalog())
client = anthropic.Anthropic()
response = client.messages.create(
model="claude-sonnet-4-6",
system=builder.build(role_description="You create user interfaces."),
output_config=output_config,
max_tokens=8192,
messages=[{"role": "user", "content": "make me a form"}],
)
# Extract, unwrap, validate, and repair the A2UI payload
a2ui_json = parse_json_response(response, builder.get_catalog())
render(a2ui_json)
Mode 4: Manual streaming
Best when you need full control over the stream, token by token. You construct the system prompt, define the tool, and process the stream yourself. The parser handles unwrapping, repair, and validation.
import anthropic
from anthropic_a2ui import (
ClaudeA2uiPromptBuilder,
create_a2ui_tool,
ClaudeStreamParser,
)
builder = ClaudeA2uiPromptBuilder()
tool = create_a2ui_tool(builder.get_catalog())
parser = ClaudeStreamParser(catalog=builder.get_catalog())
client = anthropic.Anthropic()
with client.messages.stream(
model="claude-sonnet-4-6",
system=builder.build(role_description="You create user interfaces."),
tools=[tool],
max_tokens=8192,
messages=[{"role": "user", "content": "make me a form"}],
) as stream:
for event in stream:
for part in parser.process_event(event):
if part.a2ui_json:
# A complete, validated, repaired A2UI payload
render(part.a2ui_json)
if part.text:
# Conversational text from Claude
print(part.text, end="")
The parser also supports the <a2ui-json> tag mode (no tool use), where
Claude embeds the JSON directly in its text response between
<a2ui-json> and </a2ui-json> tags.
Automatic repairs
Claude is remarkably good at generating valid A2UI, but it occasionally makes mistakes — especially the smaller models. The package includes five repair functions that fix these issues transparently before validation, so the caller never sees a false rejection:
| Repair | What it fixes | Example |
|---|---|---|
patch_catalog_schema |
DateTimeInput.min/max uses oneOf of three date formats, which jsonschema rejects as ambiguous |
Changes oneOf to anyOf in the schema |
repair_orphans |
Components created by Claude but not connected to the root tree |
Reconnects orphans as children of the root container |
repair_icons |
Icon names that don't exist in the catalog enum (cloud, sunny, trash, ...) |
Maps 100+ common aliases to the 59 valid icons; unknown names become info |
repair_functions |
FunctionCall with functions that don't exist in the catalog (ternary, if, switch) |
Replaces with a literal value (False for boolean, "" for string) or removes the check |
repair_childlists |
Dynamic child lists ({componentId, path}) used in Row or Column, which only accept static string arrays |
Converts to a static array of component IDs |
All repairs are enabled by default in generate_a2ui, A2uiConversation,
and ClaudeStreamParser. They can be disabled with repair=False for
strict validation.
When log_repairs=True, the RetryResult.repairs list contains
human-readable descriptions of what was repaired:
result = generate_a2ui(client, "make me a weather panel", log_repairs=True)
for r in result.repairs:
print(f"Repaired: {r}")
# Repaired: Icon 'sun_icon' substituted with 'info'
# Repaired: Text 'star1' emptied (possible function removed)
Prompt caching
The A2UI system prompt is large: it includes the full JSON Schema, few-shot examples, the list of 59 valid icon names, and the 14 catalog functions. Sending this on every call would be expensive and slow.
generate_a2ui and A2uiConversation enable prompt caching by default
(use_cache=True). This adds cache_control: {type: "ephemeral"} to the
system prompt block, allowing Anthropic to cache it and reduce the cost by
approximately 80% on subsequent calls within the cache window.
To disable caching (for debugging or testing):
result = generate_a2ui(client, "make me a form", use_cache=False)
Type safety
The package ships with a py.typed marker, so IDEs like VS Code (with
Pylance) and PyCharm provide full autocompletion, type checking, and
inline documentation for all public APIs.
Supported versions
A2UI is an evolving protocol. The package supports three versions:
| Version | Catalog | Components | Functions |
|---|---|---|---|
| v0.8 | standard (legacy) | 18 (includes MultipleChoice) |
0 |
| v0.9 | basic | 18 (includes ChoicePicker) |
14 |
| v0.9 | minimal | 5 (Text, Row, Column, Button, TextField) | 1 (capitalize) |
| v0.9.1 | basic | 18 | 14 |
The default is v0.9 with the Basic Catalog. To use a different version or catalog:
from anthropic_a2ui import ClaudeA2uiPromptBuilder
# v0.8 legacy
builder = ClaudeA2uiPromptBuilder(version="0.8")
# Pruned to specific components (saves tokens)
builder = ClaudeA2uiPromptBuilder()
prompt = builder.build(
role_description="You create simple forms.",
allowed_components=["Text", "TextField", "Button"],
)
Tested models
The package has been tested with natural conversational prompts (no mention of A2UI, components, or functions) across four Claude models. Each model received 10 different prompts and was asked to generate a complete A2UI interface:
| Model | Success rate | Avg. time |
|---|---|---|
| claude-haiku-4-5 | 10/10 (100%) | ~8s |
| claude-opus-4-7 | 10/10 (100%) | ~17s |
| claude-opus-4-8 | 10/10 (100%) | ~15s |
| claude-sonnet-4-5 | 8/8 (100%) | ~12s |
All 30 test cases (10 prompts x 3 models) passed validation after automatic repair. The test prompts cover forms, dashboards, surveys, calendars, profiles, configuration panels, task lists, product galleries, and modal dialogs.
API reference
High-level functions
generate_a2ui(client, prompt, **kwargs) -> RetryResult: one-shot generation with retries and caching.generate_a2ui_async(client, prompt, **kwargs) -> RetryResult: async version.A2uiConversation(client, **kwargs): multi-turn conversation..send(prompt) -> ConversationTurn,.reset().A2uiConversationAsync(client, **kwargs): async version.create_a2ui_output_config(catalog) -> dict:output_configfor forced JSON mode. Pair withparse_json_response(message, catalog).create_a2ui_response_format(catalog) -> dict: low-leveloutput_config.formatobject for forced JSON mode.
Building blocks
ClaudeA2uiPromptBuilder(version, catalogs): constructs the system prompt..build(role_description, ...)returns the prompt string..get_catalog()returns theA2uiCatalog.create_a2ui_tool(catalog, **kwargs) -> dict: tool definition for Anthropic.ClaudeStreamParser(catalog, **kwargs): stream parser..process_event(event) -> list[ResponsePart]..parse_stream(stream)is an iterator shortcut.validate_tool_input(catalog, input_json, *, repair, strict_integrity): validates a payload against the schema.to_a2ui_part(payload) -> A2uiPart: wraps a payload for transport with MIMEapplication/a2ui+json.
Repair functions
repair_orphans(payload): reconnects orphaned components.repair_icons(payload): fixes invalid icon names.repair_functions(payload): fixes non-existent function calls.repair_childlists(payload): fixes dynamic child lists in Row/Column.patch_catalog_schema(schema): patches the DateTimeInput schema.find_orphans(payload) -> list[str]: diagnostic, returns orphan IDs.
License
Apache-2.0.
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 anthropic_a2ui-0.1.1.tar.gz.
File metadata
- Download URL: anthropic_a2ui-0.1.1.tar.gz
- Upload date:
- Size: 28.5 kB
- Tags: Source
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.12
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
ba8f4b39f59c8dcee699a925c50f95d2d77e8c2113479264b41075dbd6d7bad3
|
|
| MD5 |
b492868b54fcdb6fa830e1a1fcbeff69
|
|
| BLAKE2b-256 |
b7395fa40a31baf8b4bfb33fbbe212bc93b7fc3c2bd0a67e3b5dbf00c2a49ce3
|
Provenance
The following attestation bundles were made for anthropic_a2ui-0.1.1.tar.gz:
Publisher:
publish.yml on 686f6c61/anthropic-a2ui
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
anthropic_a2ui-0.1.1.tar.gz -
Subject digest:
ba8f4b39f59c8dcee699a925c50f95d2d77e8c2113479264b41075dbd6d7bad3 - Sigstore transparency entry: 1900291219
- Sigstore integration time:
-
Permalink:
686f6c61/anthropic-a2ui@d54bebe124302f6b9f81d34b2feb91271cc6042c -
Branch / Tag:
refs/tags/v0.1.1 - Owner: https://github.com/686f6c61
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
publish.yml@d54bebe124302f6b9f81d34b2feb91271cc6042c -
Trigger Event:
push
-
Statement type:
File details
Details for the file anthropic_a2ui-0.1.1-py3-none-any.whl.
File metadata
- Download URL: anthropic_a2ui-0.1.1-py3-none-any.whl
- Upload date:
- Size: 31.5 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.12
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
52a0b503de78bcea573af060f4ccc20b2838f032dd08c76d441637bda357dc10
|
|
| MD5 |
a76808a7b172ea9f7a38f03425b807ba
|
|
| BLAKE2b-256 |
9a117c6b337f93f551f85dcc5df9fdd46ecd2d74c795adadd7bc719f404a442a
|
Provenance
The following attestation bundles were made for anthropic_a2ui-0.1.1-py3-none-any.whl:
Publisher:
publish.yml on 686f6c61/anthropic-a2ui
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
anthropic_a2ui-0.1.1-py3-none-any.whl -
Subject digest:
52a0b503de78bcea573af060f4ccc20b2838f032dd08c76d441637bda357dc10 - Sigstore transparency entry: 1900291313
- Sigstore integration time:
-
Permalink:
686f6c61/anthropic-a2ui@d54bebe124302f6b9f81d34b2feb91271cc6042c -
Branch / Tag:
refs/tags/v0.1.1 - Owner: https://github.com/686f6c61
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
publish.yml@d54bebe124302f6b9f81d34b2feb91271cc6042c -
Trigger Event:
push
-
Statement type: