Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat: add apiserver support to Python SDK #327

Merged
merged 8 commits into from
Oct 24, 2024
Merged

Conversation

masci
Copy link
Member

@masci masci commented Oct 18, 2024

Introducing apiserver support through a new Python SDK.

fixes #290

Basic usage

from llama_deploy.client import Client

# Use the same client for async and sync operations
c = Client()

async def an_async_function():
    status = await client.apiserver.status()

def normal_function():
    status = client.sync.apiserver.status()

Streaming events

with open("deployment.yml") as f:
    deployments = await client.apiserver.deployments()
    d = await deployments.create(f)

tasks = await d.tasks()
task = await tasks.create(TaskDefinition(input='{"a": "b"}'))
async for ev in task.events():
    print(ev)

@coveralls
Copy link

coveralls commented Oct 21, 2024

Coverage Status

coverage: 67.038% (+1.4%) from 65.613%
when pulling 05ff40d on massi/apiserver-client
into 3c8410b on main.

@masci masci force-pushed the massi/apiserver-client branch 2 times, most recently from 4ff9346 to 2539adb Compare October 23, 2024 09:46
try

checkpoint

add asgiref to support async-to-sync:

fix unit tests

remove pydantic warning

add unit tests for client

fix mock path

test model

added tests and fix discovered bugs

fix connection error handling

fix bugs surfaced in end-to-end

use explicity properties

fix awaitable checks

add instance method

revert to return sync class

extract base model

use instance() method on models

add e2e tests

working state

fix unit tests

more fixes
@masci masci force-pushed the massi/apiserver-client branch from 2539adb to e04269e Compare October 23, 2024 09:52
@masci masci marked this pull request as ready for review October 23, 2024 09:57
session = await deployment.client.get_session(session_id=session_def.session_id)
for task_def in await session.get_tasks():
tasks.append(task_def)
return JSONResponse(tasks)
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Should this be a dict of session_id -> list[task] ? (Just thinking about building a UI that would use this function)

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Good point, many return payloads of the apiserver API are not well thought, we can start fixing it from here

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I had a look and I realised if we change this we need to update the CLI as well and the PR would grow considerably. I propose we do that in a follow up.

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Tracked in #337

Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Sounds good to me!


Using the class constructor is not possible because we want to alter the class method to
accommodate sync/async usage before creating an instance, and __init__ would be too late.
"""
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Ha, this docstring answered so many questions. Thank you!

if asyncio.iscoroutinefunction(method) and not name.startswith("_"):
setattr(Wrapper, name, async_to_sync(method))
# Static type checkers can't assess Wrapper is indeed a type[T], let's promise it is.
return cast(type[T], Wrapper)
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

So just confirming how this works

  • we define a class with all async methods
  • if a flag is set, we apply this wrapper
  • this wrapped only converts public methods
  • type checkers and IDEs will see this as a sync method instead of async?

🤯

Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Hmm, seems like IDE/type-checker assumes its always async (which tbh is fine)

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Ah correct, didn't think about that so to clarify, the static checker would be happy with await client.sync.apiserver.status() because the actual wrapping happens at runtime. I agree this would be fine, but I'll think about possible implications

@masci masci merged commit 2accf9d into main Oct 24, 2024
10 checks passed
@masci masci deleted the massi/apiserver-client branch October 24, 2024 16:39
@masci masci added the enhancement New feature or request label Nov 15, 2024
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
enhancement New feature or request
Projects
None yet
Development

Successfully merging this pull request may close these issues.

Python SDK should support calling the APIServer
3 participants