Python SDK for authoring Kodelet extensions
Project description
kodelet-sdk
Python SDK for authoring Kodelet extensions.
The SDK speaks Kodelet's JSON-RPC extension protocol over stdio and provides an asyncio-first API for registering tools, commands, and event handlers.
Quick start
from kodelet_sdk import BaseModel, Extension, ToolContext, ToolExecutionResult
ext = Extension(name="weather", version="0.1.0")
class WeatherInput(BaseModel):
location: str
@ext.tool("get_weather", description="Get weather", input_schema=WeatherInput)
async def get_weather(input: WeatherInput, ctx: ToolContext) -> ToolExecutionResult:
return {"content": f"Weather for {input.location}"}
@ext.on("session.start")
async def session_start(event, ctx):
ctx.log.info("extension started")
if __name__ == "__main__":
ext.run_sync()
Public API
Agent sessions
Use Client to launch Kodelet and drive an agent session from Python. The
client speaks to kodelet acp over stdio JSON-RPC, so normal profile
resolution, conversation persistence, tools, skills, MCP, and extensions still
come from the Kodelet executable.
from kodelet_sdk import Client
client = Client()
session = await client.create_session()
response = await session.run_and_wait(message="what is the meaning of life?")
print(response.content)
await client.close()
Pass a named or inline Profile when creating a session, and listen for typed
stream events while a run is active:
from kodelet_sdk import Client, Profile
client = Client(command="kodelet")
session = await client.create_session(
profile=Profile(
{
"provider": "openai",
"model": "gpt-5.5",
"openai": {"api_mode": "responses", "service_tier": "fast"},
}
),
max_turns=4,
streaming=True,
)
session.on(
"assistant.message_delta",
lambda event: print(event.data.deltaContent, end="", flush=True),
)
response = await session.run_and_wait(message="help me choose an approach")
print("\nfinal:", response.content)
await client.close()
Agent sessions can expose in-process Python extensions for that session. Inline extensions are served through a temporary JSON-RPC bridge and are removed when the session closes.
from kodelet_sdk import BaseModel, Client, Extension, define_extension
def workspace(ext: Extension) -> None:
ext.set_metadata(name="workspace")
class AskInput(BaseModel):
question: str
options: list[str]
@ext.tool("ask_user_question", description="Ask the user", input_schema=AskInput)
async def ask_user_question(input: AskInput, ctx):
choice = await ctx.ui.select({"title": input.question, "options": input.options})
return choice or "dismissed"
client = Client()
session = await client.create_session(
extensions=[define_extension(workspace)],
ui={"select": lambda request: request["options"][0]},
)
response = await session.run_and_wait(message="ask me to choose")
await client.close()
Inline extension bridges use Unix domain sockets by default. If your environment blocks Unix sockets, use a loopback TCP bridge instead:
session = await client.create_session(
extensions=[define_extension(workspace)],
extension_transport="tcp", # binds an ephemeral 127.0.0.1 port
)
Extension registration
Extension(name=None, version=None)creates an extension host.@ext.tool(name=None, description=None, input_schema=None, timeout_in_sec=None)registers a tool.@ext.command(name=None, description=None, input_schema=None, aliases=None, kind=None, timeout_in_sec=None)registers a command.@ext.on(event, priority=0, timeout_in_sec=None)registers an event handler such assession.start,tool.call, oragent.end.await ext.run()starts the async stdio runtime;ext.run_sync()is a synchronous entrypoint convenience.
Handlers may be synchronous or asynchronous. Tool handlers may return a string, which is converted to { "content": ... }, or a protocol-shaped mapping. Command handlers return { "action": "pass" }, { "action": "respond", "response": ... }, or { "action": "runAgent", "prompt": ... }.
The decorators preserve concrete function signatures for type checkers, so handlers can annotate their inputs and contexts directly:
from kodelet_sdk import CommandContext, CommandResult, EventContext, ToolCallEvent
@ext.command("doctor", description="Check health", input_schema=WeatherInput)
async def doctor(input: WeatherInput, ctx: CommandContext) -> CommandResult:
return {"action": "respond", "response": ctx.input["commandName"]}
@ext.on("tool.call")
def approve(event: ToolCallEvent, ctx: EventContext):
return {"message": event.tool.name}
Pydantic and Jinja2 bridge dependencies
kodelet-sdk depends on Pydantic and Jinja2 and re-exports common entry points so extensions can be self-contained:
from kodelet_sdk import BaseModel, Field, Jinja2, Pydantic, render_template
class ReviewInput(BaseModel):
target: str = Field(min_length=1)
assert render_template("Review {{ target }}", {"target": "main"}) == "Review main"
assert Jinja2.Template("Hello {{ name }}").render(name="Kodelet") == "Hello Kodelet"
assert Pydantic.TypeAdapter(int).validate_python("1") == 1
Pydantic input schemas are converted to JSON Schema during initialization and validate incoming tool/command inputs before handlers run. Commands with validation failures return {"action": "pass"} so another command route can handle the invocation.
Context helpers
Handlers receive ctx with Kodelet call metadata and helper namespaces:
ctx.storage.read_text/write_text/read_json/write_json(...)for extension data files.ctx.path.resolve_workspace_path(...)andctx.path.relative_to_workspace(...).ctx.fs.exists/read_text/write_text/list(...)for workspace file access.ctx.process.exec(...)andctx.process.spawn(...)for async process execution.ctx.env.get(...)for environment access.ctx.log.debug/info/warn/error(...)for JSON logs to stderr.ctx.ui.input/confirm/select/notify(...)for host UI reverse-RPC calls.
UI helpers accept protocol-shaped typed requests: UIInputRequest, UIConfirmRequest, UISelectRequest, and UINotifyRequest.
from kodelet_sdk import UIInputRequest, UISelectRequest
input_request: UIInputRequest = {"title": "Branch name", "required": True}
select_request: UISelectRequest = {"title": "Mode", "options": ["fast", "thorough"]}
branch = await ctx.ui.input(input_request)
mode = await ctx.ui.select(select_request)
Testing extensions
Use create_test_harness to exercise registrations without spawning a subprocess:
from kodelet_sdk import Extension, create_test_harness
async def test_tool():
ext = Extension(name="example")
@ext.tool("echo", description="Echo", input_schema={"type": "object"})
async def echo(input, ctx):
return {"content": input["text"]}
harness = await create_test_harness(ext)
result = await harness.execute_tool({"name": "echo", "input": {"text": "hi"}})
assert result == {"content": "hi"}
Examples
Runnable example extensions live in examples/:
examples/review/kodelet-extension-reviewis a review command extension.examples/workspace/kodelet-extension-workspaceis a workspace helper/policy extension.
From a checked-out SDK repository, run an example with:
uv run -s examples/review/kodelet-extension-review
The kodelet-extension-* files are executable wrappers so Kodelet can discover and launch them directly.
Releases
Package versions are read from VERSION.txt. To publish a release, configure PyPI Trusted Publishing for the Release workflow, then update and commit VERSION.txt manually:
git add VERSION.txt pyproject.toml uv.lock
git commit -m "chore: release v0.1.0"
make release
Pushing the vX.Y.Z tag runs the GitHub Actions release workflow, builds the package, and publishes to PyPI using OIDC trusted publishing.
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 kodelet_sdk-0.1.2.tar.gz.
File metadata
- Download URL: kodelet_sdk-0.1.2.tar.gz
- Upload date:
- Size: 78.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 |
2eef432c94466bb411eaff77923dbfedfba146055d11d489fa2da61547cd4347
|
|
| MD5 |
106a7c2928e5b15ef685674e778077de
|
|
| BLAKE2b-256 |
ca589f919d6ed34fa1f35fc16448ada06ea4df0a5373baa2994912718376eb4b
|
Provenance
The following attestation bundles were made for kodelet_sdk-0.1.2.tar.gz:
Publisher:
release.yml on jingkaihe/kodelet-python-sdk
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
kodelet_sdk-0.1.2.tar.gz -
Subject digest:
2eef432c94466bb411eaff77923dbfedfba146055d11d489fa2da61547cd4347 - Sigstore transparency entry: 1735976189
- Sigstore integration time:
-
Permalink:
jingkaihe/kodelet-python-sdk@efa90a6ebe731253ea0b7ead7f3a00a7f65161f4 -
Branch / Tag:
refs/tags/v0.1.2 - Owner: https://github.com/jingkaihe
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
release.yml@efa90a6ebe731253ea0b7ead7f3a00a7f65161f4 -
Trigger Event:
push
-
Statement type:
File details
Details for the file kodelet_sdk-0.1.2-py3-none-any.whl.
File metadata
- Download URL: kodelet_sdk-0.1.2-py3-none-any.whl
- Upload date:
- Size: 40.8 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 |
8b6f581a228bdf0b0531667074a36704d6b1b2d196a98656c83262e300d87636
|
|
| MD5 |
4e8ab3d59e242d069a9159653a092055
|
|
| BLAKE2b-256 |
415e82795337bd295bf410aad6e9e3eff4583030e9b1e57e882d2a7ad7401043
|
Provenance
The following attestation bundles were made for kodelet_sdk-0.1.2-py3-none-any.whl:
Publisher:
release.yml on jingkaihe/kodelet-python-sdk
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
kodelet_sdk-0.1.2-py3-none-any.whl -
Subject digest:
8b6f581a228bdf0b0531667074a36704d6b1b2d196a98656c83262e300d87636 - Sigstore transparency entry: 1735976234
- Sigstore integration time:
-
Permalink:
jingkaihe/kodelet-python-sdk@efa90a6ebe731253ea0b7ead7f3a00a7f65161f4 -
Branch / Tag:
refs/tags/v0.1.2 - Owner: https://github.com/jingkaihe
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
release.yml@efa90a6ebe731253ea0b7ead7f3a00a7f65161f4 -
Trigger Event:
push
-
Statement type: