LLM plugin for Apple Foundation Models (Apple Intelligence)
Project description
llm-apple
LLM plugin for Apple Foundation Models (Apple Intelligence)
This plugin exposes Apple's on-device Foundation Models through the llm CLI tool.
Requirements
- macOS 26.0 or later
- Apple Intelligence enabled
- Python 3.9 or later
- apple-foundation-models >= 0.2.0 installed
Installation
pip install llm # if llm is not already installed
llm install llm-apple
Usage
Basic usage (streaming is enabled by default):
llm -m apple "What is the capital of France?"
Without streaming:
llm -m apple "Tell me a story" --no-stream
With options:
llm -m apple "Write a poem" -o temperature 1.5 -o max_tokens 500
With system instructions:
llm -m apple "What is Python?" --system "You are a helpful programming tutor"
Conversations
The plugin supports conversations, maintaining context across multiple prompts:
# Start a conversation
llm -m apple "My name is Alice" --save conversation1
# Continue the conversation
llm -m apple "What is my name?" --continue conversation1
Tool Calling
The plugin supports tool calling, allowing the model to call Python functions to access real-time data, perform actions, or integrate with external systems.
CLI Tool Usage
You can use tools from the command line using the --functions option:
# Define a function inline
llm -m apple "What's the weather in Paris?" \
--functions 'def get_weather(location: str) -> str:
"""Get the current weather for a location."""
return f"Weather in {location}: 72°F, sunny"'
Or load functions from a Python file:
# Create a tools.py file
cat > tools.py << 'EOF'
def get_current_time() -> str:
"""Get the current time."""
from datetime import datetime
return datetime.now().strftime("%I:%M %p")
def get_weather(location: str) -> str:
"""Get weather for a location."""
return f"Weather in {location}: 72°F, sunny"
EOF
# Use the functions from the file
llm -m apple "What time is it and what's the weather in Tokyo?" --functions tools.py
You can also use registered tool plugins with the -T or --tool flag (see llm tool documentation for more details).
Python API Tool Usage
import llm
def get_weather(location: str) -> str:
"""Get the current weather for a location."""
# In a real implementation, this would call a weather API
return f"Weather in {location}: 72°F, sunny"
model = llm.get_model("apple")
response = model.prompt(
"What's the weather in San Francisco?",
tools=[llm.Tool(
name="get_weather",
description="Get current weather for a location",
implementation=get_weather
)]
)
print(response.text())
Tool Types Supported
Tools can have various parameter signatures:
No parameters:
def get_current_time() -> str:
"""Get the current time."""
return "2:30 PM"
Single parameter:
def search_docs(query: str) -> str:
"""Search documentation."""
return f"Results for: {query}"
Multiple parameters with mixed types:
def calculate(operation: str, x: int, y: int) -> str:
"""Perform a calculation."""
ops = {"add": x + y, "multiply": x * y}
return str(ops.get(operation, 0))
Optional parameters:
def get_temperature(city: str, units: str = "celsius") -> str:
"""Get temperature for a city."""
return f"Temperature in {city}: 20°{units[0].upper()}"
Multiple Tools
You can register multiple tools in a single call:
def get_time() -> str:
"""Get the current time."""
return "2:30 PM"
def get_date() -> str:
"""Get the current date."""
return "November 7, 2024"
def get_weather(location: str) -> str:
"""Get weather for a location."""
return f"Weather in {location}: 72°F, sunny"
tools = [
llm.Tool(name="get_time", description="Get current time", implementation=get_time),
llm.Tool(name="get_date", description="Get current date", implementation=get_date),
llm.Tool(name="get_weather", description="Get weather", implementation=get_weather),
]
response = model.prompt(
"What's the date, time, and weather in Paris?",
tools=tools
)
The model will automatically select and call the appropriate tools based on the prompt.
Available Options
temperature(float, 0.0-2.0, default: 1.0): Controls randomness in generation- 0.0 = deterministic
- 2.0 = very random
max_tokens(int, default: 1024): Maximum tokens to generate
System prompts can be provided using llm's built-in --system or -s flag.
Availability
The plugin checks Apple Intelligence availability on startup. If Apple Intelligence is not available, you'll see an error message with details on why.
Common reasons:
- Device not eligible (requires Apple Silicon)
- Apple Intelligence not enabled in Settings
- Model not ready (downloading or initializing)
Examples
Creative writing with higher temperature:
llm -m apple "Write a creative story about a robot" -o temperature 1.8
Factual query with lower temperature:
llm -m apple "Explain quantum computing" -o temperature 0.3
With system prompt for career guidance:
llm -m apple "Should I learn Python or JavaScript?" \
--system "You are a career counselor specializing in tech"
Development
Running Tests
# Run all tests (unit tests with mocks)
uv run pytest
# Run tests with coverage
uv run pytest --cov=llm_apple --cov-report=html --cov-report=term
# Run integration tests (requires Apple Intelligence)
uv run pytest tests/test_integration_tools.py -v -s
Most tests use mocks to simulate the Apple Foundation Models API, so they can run on any platform without requiring actual Apple Intelligence hardware.
Integration tests in tests/test_integration_tools.py require Apple Intelligence to be available and will be automatically skipped if not present.
Project details
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_apple-0.2.0.tar.gz.
File metadata
- Download URL: llm_apple-0.2.0.tar.gz
- Upload date:
- Size: 17.6 kB
- Tags: Source
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.7
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
e09ea1ece986a057a34a8cd8b277c7d73e1ebb1756fd079c0d9c58341474d93f
|
|
| MD5 |
c57abe69e2ad0baaa9e1300cba186a9e
|
|
| BLAKE2b-256 |
16293c9414b8645b3efa773a6937f13f71b89cd05dee170626f04176dc1d7e9e
|
Provenance
The following attestation bundles were made for llm_apple-0.2.0.tar.gz:
Publisher:
publish-to-pypi.yml on btucker/llm-apple
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
llm_apple-0.2.0.tar.gz -
Subject digest:
e09ea1ece986a057a34a8cd8b277c7d73e1ebb1756fd079c0d9c58341474d93f - Sigstore transparency entry: 698039814
- Sigstore integration time:
-
Permalink:
btucker/llm-apple@1de7ad55aa9dad9784dd18324899b5e67133978d -
Branch / Tag:
refs/tags/v0.2.0 - Owner: https://github.com/btucker
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
publish-to-pypi.yml@1de7ad55aa9dad9784dd18324899b5e67133978d -
Trigger Event:
release
-
Statement type:
File details
Details for the file llm_apple-0.2.0-py3-none-any.whl.
File metadata
- Download URL: llm_apple-0.2.0-py3-none-any.whl
- Upload date:
- Size: 7.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 |
626f212d3ff4b009cc35ce93b0dd096b31e513fb5b26285d9339fd22da3719be
|
|
| MD5 |
eb797d29cde7f89c308c67b53d78f41a
|
|
| BLAKE2b-256 |
253675a0d083abb2f33efe1de711cbfb37af372c33dea9d3b4c7af2afbcf0796
|
Provenance
The following attestation bundles were made for llm_apple-0.2.0-py3-none-any.whl:
Publisher:
publish-to-pypi.yml on btucker/llm-apple
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
llm_apple-0.2.0-py3-none-any.whl -
Subject digest:
626f212d3ff4b009cc35ce93b0dd096b31e513fb5b26285d9339fd22da3719be - Sigstore transparency entry: 698039821
- Sigstore integration time:
-
Permalink:
btucker/llm-apple@1de7ad55aa9dad9784dd18324899b5e67133978d -
Branch / Tag:
refs/tags/v0.2.0 - Owner: https://github.com/btucker
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
publish-to-pypi.yml@1de7ad55aa9dad9784dd18324899b5e67133978d -
Trigger Event:
release
-
Statement type: