Declarative slot-based onboarding schemas for LLM-driven conversations.
Project description
slotflow
Drive multi-turn LLM conversations from a Pydantic schema. Declare what to collect with Slot(), choose a flow mode (sequential, freeform, steps), and let slotflow handle question generation, extraction, response judging, and follow-ups — with immutable state that serializes to Redis out of the box.
- Composition over modification — slots do not know about flows
- Pydantic does all validation, including dynamic per-call wrapper models
- Immutable state — every turn returns a new
FlowState - LLM is injected, never constructed internally (provider-agnostic)
- Core install is LLM-free; LangChain and OpenAI extras are optional
Installation
# Schema layer only (no LLM dependency)
pip install llm-slotflow
# With LangChain backend
pip install "slotflow[langchain]"
# With OpenAI backend (no LangChain)
pip install "slotflow[openai]"
Requires Python 3.10+.
Quickstart
1. Declare a schema
from datetime import date
from enum import Enum
from typing import Optional
from slotflow import OnboardingSchema, Slot
class DocumentType(str, Enum):
DNI = "DNI"
CE = "CE"
PASSPORT = "Passport"
class UserOnboarding(OnboardingSchema):
full_name: str = Slot(description="User's full name")
document_type: DocumentType = Slot(description="Type of identity document")
birth_date: date = Slot(description="Date of birth")
phone: Optional[str] = Slot(default=None, description="Phone number (optional)")
Slots without a default are required; slots with one are optional.
2. Extract slots from free-form text
import asyncio
from langchain_openai import ChatOpenAI
from slotflow import extract_slots
llm = ChatOpenAI(model="gpt-4o-mini", temperature=0)
result = asyncio.run(extract_slots(
schema=UserOnboarding,
field_names=["full_name", "document_type", "birth_date"],
text="I'm John Smith, passport, born on 15/05/1990.",
llm=llm,
))
print(result.value)
# {'full_name': 'John Smith', 'document_type': <DocumentType.PASSPORT>, 'birth_date': date(1990, 5, 15)}
extract_slot / extract_slots retry up to max_attempts (default 3),
feeding the Pydantic ValidationError back to the LLM as feedback.
3. Drive a full conversation
import asyncio
from langchain_openai import ChatOpenAI
from slotflow import (
FlowMode, OnboardingFlow, SlotPrompt,
initial_state, next_message, process_response,
)
flow = OnboardingFlow(
schema=UserOnboarding,
mode=FlowMode.SEQUENTIAL,
prompts={
"phone": SlotPrompt(
follow_up_hint="If the user hesitates, remind them it's optional."
),
},
)
llm = ChatOpenAI(model="gpt-4o-mini", temperature=0)
async def main() -> None:
state = initial_state(flow)
state, turn = await next_message(flow=flow, state=state, llm=llm)
while not turn.done:
print("Bot >", turn.message)
user_text = input("You > ")
state, turn = await process_response(
flow=flow, state=state, user_text=user_text, llm=llm
)
print("Captured:", dict(state.filled))
print("Skipped:", list(state.skipped))
asyncio.run(main())
Flow modes
| Mode | What it does |
|---|---|
SEQUENTIAL |
Asks one slot at a time in schema declaration order. |
UNORDERED |
Asks one slot at a time but accepts answers for any pending slot in the same turn. |
FREEFORM |
Opens with everything that is missing; extracts many slots from one response. |
STEPS |
Groups slots into ordered Step(...)s; finishes a step before moving on. |
Per-slot prompt overrides
Override question wording per slot, or hint the follow-up generator:
OnboardingFlow(
schema=UserOnboarding,
prompts={
"full_name": SlotPrompt(question="What's your full name?"),
"birth_date": SlotPrompt(
follow_up_hint="Ask for an explicit day/month/year.",
),
},
)
When SlotPrompt.question is set, the LLM question-generation call is skipped
entirely — useful for tightly controlled wording or to save tokens.
How a turn works
- Extract —
extract_slot/extract_slotsis always called first. - Judge —
judge_responseclassifies the response as one ofCOMPLETE / PARTIAL / INSUFFICIENT / SKIP_INTENT / REFUSED, given the user's raw text and the extracted value. - Decide — the runner fills the slot, marks an optional slot as skipped, asks a follow-up, or generates a nudge for a refused required slot.
Stateless runner
FlowState, Turn, and FlowTurn are all frozen dataclasses. Every call to
next_message / process_response returns a new FlowState; the input is
never mutated. This makes it trivial to:
- run many conversations concurrently
- pickle the state between turns (Redis, DB, queue)
- snapshot/restore for testing or replay
import pickle
serialized = pickle.dumps(state) # works out of the box
Examples
Runnable scripts live in examples/:
01_extract_single_slot.py— minimal extraction example02_sequential_flow.py—SEQUENTIALmode end to end03_freeform_flow.py—FREEFORMmode end to end04_steps_flow.py—STEPSmode with grouped slots
All read OPENAI_API_KEY from .env (see .env.example).
Development
python3 -m venv .venv
.venv/bin/pip install -e ".[dev]"
# Run tests
.venv/bin/pytest tests/
# Lint + format
.venv/bin/ruff check .
.venv/bin/ruff format .
# Type check
.venv/bin/mypy
# Tests with coverage
.venv/bin/pytest --cov
License
MIT — see LICENSE.
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 llm_slotflow-0.2.0.tar.gz.
File metadata
- Download URL: llm_slotflow-0.2.0.tar.gz
- Upload date:
- Size: 26.3 kB
- Tags: Source
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.12
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
3172c5770b0962b411cafcca7b8d2923fcbf70e836ebb374ca6a514a20a01a5e
|
|
| MD5 |
5f6a854b7a60d32ad7f175888ccf8c68
|
|
| BLAKE2b-256 |
3061ff0f1a890a00c340a0909ae1cb94da8b8af7db081279945939ac28bfaa1e
|
Provenance
The following attestation bundles were made for llm_slotflow-0.2.0.tar.gz:
Publisher:
publish.yml on m0rgAn115/slotflow
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
llm_slotflow-0.2.0.tar.gz -
Subject digest:
3172c5770b0962b411cafcca7b8d2923fcbf70e836ebb374ca6a514a20a01a5e - Sigstore transparency entry: 1695721045
- Sigstore integration time:
-
Permalink:
m0rgAn115/slotflow@df1caef122d77665e2bb9ddbb8fe0a4961589b89 -
Branch / Tag:
refs/tags/v0.2.0 - Owner: https://github.com/m0rgAn115
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
publish.yml@df1caef122d77665e2bb9ddbb8fe0a4961589b89 -
Trigger Event:
push
-
Statement type:
File details
Details for the file llm_slotflow-0.2.0-py3-none-any.whl.
File metadata
- Download URL: llm_slotflow-0.2.0-py3-none-any.whl
- Upload date:
- Size: 25.0 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 |
46d79e2333f8c053197dc0fb8574310cb441c53bff2b42d3aa180488ccfe4425
|
|
| MD5 |
327e204e88e8d43cea516ca1260ad6bf
|
|
| BLAKE2b-256 |
74f8865a22918090abffa96a2eaf1fc41569ffed66534022a5b3cda68dbe71c2
|
Provenance
The following attestation bundles were made for llm_slotflow-0.2.0-py3-none-any.whl:
Publisher:
publish.yml on m0rgAn115/slotflow
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
llm_slotflow-0.2.0-py3-none-any.whl -
Subject digest:
46d79e2333f8c053197dc0fb8574310cb441c53bff2b42d3aa180488ccfe4425 - Sigstore transparency entry: 1695721161
- Sigstore integration time:
-
Permalink:
m0rgAn115/slotflow@df1caef122d77665e2bb9ddbb8fe0a4961589b89 -
Branch / Tag:
refs/tags/v0.2.0 - Owner: https://github.com/m0rgAn115
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
publish.yml@df1caef122d77665e2bb9ddbb8fe0a4961589b89 -
Trigger Event:
push
-
Statement type: