Makes working with LLMs like OpenAI GPT, Anthropic Claude, Google Gemini and Open source models super easy
Project description
JustAI
Package to make working with Large Language Models in Python super easy. Supports OpenAI, Anthropic Claude, Google Gemini, X Grok, DeepSeek, Perplexity, Reve, OpenRouter and local GGUF models.
Author: Hans-Peter Harmsen (hp@harmsen.nl)
Current version: 5.5.3
Installation
- Install the package:
pip install justai
-
Create an API key for the provider(s) you intend to use:
- OpenAI: platform.openai.com
- Anthropic: console.anthropic.com
- Google: aistudio.google.com
- X AI: console.x.ai
- DeepSeek: platform.deepseek.com
-
Create a
.envfile with the relevant keys:
OPENAI_API_KEY=your-openai-api-key
ANTHROPIC_API_KEY=your-anthropic-api-key
GOOGLE_API_KEY=your-google-api-key
X_API_KEY=your-x-ai-api-key
DEEPSEEK_API_KEY=your-deepseek-api-key
PERPLEXITY_API_KEY=your-perplexity-api-key
Basic usage
from justai import Model
model = Model('gpt-5-mini')
model.system = """You are a movie critic. I feed you with movie
titles and you give me a review in 50 words."""
response = model.chat("Forrest Gump", cached=True)
print(response)
The cached=True parameter tells justai to cache the prompt and response locally.
Models
The provider is chosen automatically based on the model name prefix:
| Prefix | Provider |
|---|---|
gpt*, o1*, o3* |
OpenAI |
claude* |
Anthropic |
gemini* |
|
grok* |
X AI |
deepseek* |
DeepSeek |
sonar* |
Perplexity |
reve* |
Reve |
openrouter/* |
OpenRouter |
*.gguf |
Local GGUF |
Features
JSON and structured output
model = Model('gemini-2.5-flash')
prompt = 'Give me the main characters from Seinfeld. Return json with keys name, profession and weirdness'
data = model.chat(prompt, return_json=True)
For typed structured output, pass a Pydantic model or Python type as response_format:
from pydantic import BaseModel as PydanticModel
class Character(PydanticModel):
name: str
profession: str
weirdness: str
result = model.chat(prompt, response_format=list[Character])
Images
Pass images as URLs, raw bytes or PIL images:
model = Model('gpt-5-nano')
url = 'https://upload.wikimedia.org/wikipedia/commons/9/94/Common_dolphin.jpg'
message = model.chat("What is in this image", images=url)
Image generation
model = Model('gpt-5')
pil_image = model.generate_image("A dolphin reading a book")
Input images can be passed for editing or style transfer:
model = Model('gemini-2.5-flash-image-preview')
pil_image = model.generate_image("Convert to Van Gogh style", images=source_image)
Async streaming
import asyncio
async def stream(model_name, prompt):
model = Model(model_name)
async for word in model.chat_async(prompt):
print(word, end='')
asyncio.run(stream('sonar-pro', 'Give me 5 names for a juice bar'))
Prompt caching (Anthropic)
model = Model('claude-sonnet-4-6')
model.system_message = 'You are an experienced book analyzer'
model.cached_prompt = SOME_LONG_TEXT
response = model.chat('Who is the main character?', cached=False)
Agent
JustAI includes an Agent class for autonomous, tool-using agent execution. The agent runs in a loop: it reads a task file, calls tools as needed, and returns a final answer.
Basic agent usage
import asyncio
from justai import Agent, FileSystemTool
agent = Agent(
model='claude-sonnet-4-6',
role='Code reviewer',
goal='Review Python files and report issues',
tools=[FileSystemTool(read=['/path/to/src'])],
max_iterations=10,
)
async def main():
async for event in agent.run('tasks.md'):
if event.type == 'response':
print(event.content, end='')
elif event.type == 'done':
print(f'\nAnswer: {event.result.answer}')
asyncio.run(main())
Built-in tools
FileSystemTool — read/write files with path traversal protection:
FileSystemTool(read=['/allowed/read/dir'], write=['/allowed/write/dir'])
ShellTool — run shell commands with allowlist-based security:
ShellTool(allowlist=['echo', 'ls', 'python'])
WebFetchTool — fetch URLs with SSRF protection:
WebFetchTool()
Custom tools
@agent.tool
def search_database(ctx, query: str) -> str:
"""Search the database for matching records."""
return db.search(query)
Dynamic instructions
@agent.instructions
def inject_context(ctx) -> str:
return f'Current user: {ctx.deps["username"]}'
Skills
Load .md skill files to extend the agent's system prompt:
agent = Agent(
model='claude-sonnet-4-6',
role='Assistant',
goal='Help with tasks',
skills_dir='./skills',
)
Agent events
The agent.run() async generator yields AgentEvent objects with these types:
status— status messagesresponse— streamed text from the modeltool_call— tool invocation (withname,arguments,tool_result)error— error messagesdone— final result withAgentResult(answer, audit trail, token usage, iterations)
License
MIT
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 justai-5.5.3.tar.gz.
File metadata
- Download URL: justai-5.5.3.tar.gz
- Upload date:
- Size: 48.8 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.2.0 CPython/3.11.3
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
e8c8a696e4fdbe574d0e801bec2d9498d92e71a09f7bf4c6261d68105a1cd378
|
|
| MD5 |
0415f1a92c68ef2effa6cad9c9a17eb7
|
|
| BLAKE2b-256 |
d76fc24365e98f30df409c9f7c8f70caae47da112486bc705259b9e9cec82143
|
File details
Details for the file justai-5.5.3-py3-none-any.whl.
File metadata
- Download URL: justai-5.5.3-py3-none-any.whl
- Upload date:
- Size: 54.6 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.2.0 CPython/3.11.3
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
c47c73b4a51b3fb9c16fcd2ad670ed66bc6cbc2aa5f1ff6ca615cc8d8c29c617
|
|
| MD5 |
871a5bf8617869f79c0cadb7f378a4a0
|
|
| BLAKE2b-256 |
6b61361b0222cf0c4e9daebc59e1e1bdc06de4e58af0b388b40f33ad711301ca
|