Google GenAI wrapper with mandatory generate → format → validate pipeline
Project description
safe-genai
A Python library that wraps Google Gemini with a mandatory generate → format → validate pipeline, so every response is structured and semantically checked before you receive it.
How it works
Every call to SafeAgent.run() executes three steps in sequence:
- Generate — calls Gemini (with Google Search grounding) using your system prompt
- Format — coerces the raw text into your Pydantic schema via a second LLM call
- Validate — verifies the output fulfills the original prompt requirements, rejecting evasive non-answers (skippable via
skip_validation=True)
If validation fails, the pipeline retries from the top (configurable via max_retries). On final failure it raises SafeGenAIValidationError.
Installation
pip install safe-genai
Requires a Google Gemini API key or Vertex AI project:
export GEMINI_API_KEY="your-key-here"
Quickstart
from pydantic import BaseModel
from safe_genai import SafeAgent
class LatestRelease(BaseModel):
version: str
release_date: str
notable_changes: list[str]
agent = SafeAgent(
output_schema=LatestRelease,
system_prompt=(
"You are a software research assistant. "
"Use Google Search to find the most recent release information. "
"Always return the latest published version, never guess."
),
)
result = agent.run("What is the latest stable release of Python?")
print(result.output.version)
print(result.output.release_date)
Async usage
result = await agent.run_async("What is 2 + 2?")
Configuration
from safe_genai import SafeAgent, AgentConfig
agent = SafeAgent(
output_schema=Answer,
system_prompt="You are a helpful assistant.",
additional_formatter_instructions="Always express numbers as digits, not words.",
additional_validator_instructions="Reject answers with confidence below 0.8.",
max_retries=2,
config=AgentConfig(
generator_model="gemini-3.1-pro-preview",
formatter_model="gemini-3-flash-preview",
validator_model="gemini-3-flash-preview",
api_key="your-key", # or set GEMINI_API_KEY env var
),
)
Vertex AI (gcloud ADC)
If you're already authenticated with gcloud auth application-default login, you can use Vertex AI instead of an API key:
from safe_genai import SafeAgent, AgentConfig
agent = SafeAgent(
output_schema=Answer,
system_prompt="You are a helpful assistant.",
config=AgentConfig(
vertexai=True,
vertex_project="my-gcp-project", # or set GOOGLE_CLOUD_PROJECT env var
vertex_location="us-central1", # default
),
)
No GEMINI_API_KEY is needed — the client uses your ambient gcloud credentials.
Error handling
from safe_genai import SafeGenAIValidationError, SafeGenAIFormatterError
try:
result = agent.run("some prompt")
except SafeGenAIValidationError as e:
print(f"Validation failed after {e.attempts} attempt(s): {e.feedback}")
print(e.output) # last Pydantic model produced
print(e.raw_response) # raw text from the generator
except SafeGenAIFormatterError as e:
print(f"Could not coerce output into {e.schema_name}: {e.cause}")
Token usage
Every result includes a breakdown of token usage by pipeline step, accumulated across all retry attempts:
result = agent.run("What is the latest stable release of Python?")
print(result.usage.generator) # Usage(input_tokens=..., output_tokens=...)
print(result.usage.formatter) # Usage(input_tokens=..., output_tokens=...)
print(result.usage.validator) # Usage(input_tokens=..., output_tokens=...)
print(result.usage.total) # Usage(input_tokens=..., output_tokens=...) ← sum of all three
Usage is a plain dataclass with two fields: input_tokens and output_tokens.
Development
uv sync --extra dev
uv run pytest
uv build
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 safe_genai-0.2.1.tar.gz.
File metadata
- Download URL: safe_genai-0.2.1.tar.gz
- Upload date:
- Size: 61.4 kB
- Tags: Source
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.7
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
a257f4b551b3c502e2b5858f5967f0414e091fbccab9fb0d79fa54fb0b8ad9e7
|
|
| MD5 |
2dbda8e3fe32fb756dc790fa9acc1b35
|
|
| BLAKE2b-256 |
96ed9e2c62f4d8c174bfdca187ec5f9bf6da61eb00be2d139574db8c1352c2bc
|
Provenance
The following attestation bundles were made for safe_genai-0.2.1.tar.gz:
Publisher:
publish.yml on GuilleQP/safe-genai
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
safe_genai-0.2.1.tar.gz -
Subject digest:
a257f4b551b3c502e2b5858f5967f0414e091fbccab9fb0d79fa54fb0b8ad9e7 - Sigstore transparency entry: 1133113888
- Sigstore integration time:
-
Permalink:
GuilleQP/safe-genai@b504d13351a2211838f3380c4138a89a3388631c -
Branch / Tag:
refs/tags/v0.2.1 - Owner: https://github.com/GuilleQP
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
publish.yml@b504d13351a2211838f3380c4138a89a3388631c -
Trigger Event:
push
-
Statement type:
File details
Details for the file safe_genai-0.2.1-py3-none-any.whl.
File metadata
- Download URL: safe_genai-0.2.1-py3-none-any.whl
- Upload date:
- Size: 12.5 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.7
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
c47d1db310b9ec7d4e95c95ba874d6dffa6609b08640a3436c456639c42101d4
|
|
| MD5 |
e12279f64ed0c150520fd4255a2311ac
|
|
| BLAKE2b-256 |
d65d3b9fdd0a0dedd63eb2af1ea868407e7793742d80a45b71a8d7c56c4a695b
|
Provenance
The following attestation bundles were made for safe_genai-0.2.1-py3-none-any.whl:
Publisher:
publish.yml on GuilleQP/safe-genai
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
safe_genai-0.2.1-py3-none-any.whl -
Subject digest:
c47d1db310b9ec7d4e95c95ba874d6dffa6609b08640a3436c456639c42101d4 - Sigstore transparency entry: 1133113931
- Sigstore integration time:
-
Permalink:
GuilleQP/safe-genai@b504d13351a2211838f3380c4138a89a3388631c -
Branch / Tag:
refs/tags/v0.2.1 - Owner: https://github.com/GuilleQP
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
publish.yml@b504d13351a2211838f3380c4138a89a3388631c -
Trigger Event:
push
-
Statement type: