Core runtime for OpenUI — define component libraries, generate model prompts, and render structured UI from streaming LLM output.
npm install @openuidev/react-lang
# or
pnpm add @openuidev/react-langPeer dependencies: react >=19.0.0
@openuidev/react-lang provides three core capabilities:
- Define components — Use
defineComponentandcreateLibraryto declare what the model is allowed to generate, with typed props via Zod schemas. - Generate prompts — Call
library.prompt()to produce a system prompt that instructs the model how to emit OpenUI Lang output. - Render output — Use
<Renderer>to parse and progressively render streamed OpenUI Lang into React components.
import { defineComponent } from "@openuidev/react-lang";
import { z } from "zod";
const Greeting = defineComponent({
name: "Greeting",
description: "Displays a greeting message",
props: z.object({
name: z.string().describe("The person's name"),
mood: z.enum(["happy", "excited"]).optional().describe("Tone of the greeting"),
}),
component: ({ name, mood }) => (
<div className={mood === "excited" ? "text-xl font-bold" : ""}>
Hello, {name}!
</div>
),
});import { createLibrary } from "@openuidev/react-lang";
const library = createLibrary({
components: [Greeting, Card, Table /* ... */],
root: "Card", // optional default root component
});const systemPrompt = library.prompt({
preamble: "You are a helpful assistant.",
additionalRules: ["Always greet the user by name."],
examples: ["<Greeting name='Alice' mood='happy' />"],
});import { Renderer } from "@openuidev/react-lang";
function AssistantMessage({ response, isStreaming }) {
return (
<Renderer
response={response}
library={library}
isStreaming={isStreaming}
onAction={(event) => console.log("Action:", event)}
/>
);
}| Export | Description |
|---|---|
defineComponent(config) |
Define a single component with a name, Zod props schema, description, and React renderer |
createLibrary(definition) |
Create a library from an array of defined components |
| Export | Description |
|---|---|
Renderer |
React component that parses and renders OpenUI Lang output |
RendererProps:
| Prop | Type | Description |
|---|---|---|
response |
string | null |
Raw OpenUI Lang text from the model |
library |
Library |
Component library from createLibrary() |
isStreaming |
boolean |
Whether the model is still streaming (disables form interactions) |
onAction |
(event: ActionEvent) => void |
Callback when a component triggers an action |
onStateUpdate |
(state: Record<string, any>) => void |
Callback when form field values change |
initialState |
Record<string, any> |
Initial form state for hydration |
onParseResult |
(result: ParseResult | null) => void |
Callback when the parse result changes |
| Export | Description |
|---|---|
createParser(library) |
Create a one-shot parser for complete OpenUI Lang text |
createStreamingParser(library) |
Create an incremental parser for streaming input |
The streaming parser exposes two methods:
| Method | Description |
|---|---|
push(chunk) |
Feed the next chunk; returns the latest ParseResult |
getResult() |
Get the latest result without consuming new data |
After the stream ends, check meta.unresolved for any identifiers that were referenced but never defined. During streaming these are expected (forward refs) and are not treated as errors.
ParseResult.meta.errors contains structured OpenUIError objects. Each error has a type discriminant (currently always "validation") and a code for consumer-side filtering:
| Code | Meaning |
|---|---|
missing-required |
Required prop absent with no default |
null-required |
Required prop explicitly null with no default |
unknown-component |
Component name not found in the library schema |
excess-args |
More positional args passed than the schema defines |
Errors do not affect rendering — the parser stays permissive and renders what it can. Use code to decide how to surface or log errors:
const result = parser.parse(output);
const critical = result.meta.errors.filter(
(e) => e.code === "unknown-component"
);To check for unresolved references after streaming, inspect meta.unresolved:
if (result.meta.unresolved.length > 0) {
console.warn("Unresolved refs:", result.meta.unresolved);
}Use these inside component renderers to interact with the rendering context:
| Hook | Description |
|---|---|
useIsStreaming() |
Whether the model is still streaming |
useRenderNode() |
Render child element nodes |
useTriggerAction() |
Trigger an action event |
useGetFieldValue() |
Get a form field's current value |
useSetFieldValue() |
Set a form field's value |
useSetDefaultValue() |
Set a field's default value |
useFormName() |
Get the current form's name |
| Export | Description |
|---|---|
useFormValidation() |
Access form validation state |
useCreateFormValidation() |
Create a form validation context |
validate(value, rules) |
Run validation rules against a value |
builtInValidators |
Built-in validators (required, email, min, max, etc.) |
import type {
Library,
LibraryDefinition,
DefinedComponent,
ComponentRenderer,
ComponentRenderProps,
ComponentGroup,
PromptOptions,
RendererProps,
ActionEvent,
ElementNode,
ParseResult,
OpenUIError,
ValidationErrorCode,
LibraryJSONSchema,
} from "@openuidev/react-lang";Libraries can also produce a JSON Schema representation of their components:
const schema = library.toJSONSchema();
// schema.$defs["Card"] → { properties: {...}, required: [...] }
// schema.$defs["Greeting"] → { properties: {...}, required: [...] }Full documentation, guides, and the language specification are available at openui.com.