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

Provider-Agnostic Call Decorator #729

Open
willbakst opened this issue Dec 4, 2024 · 7 comments
Open

Provider-Agnostic Call Decorator #729

willbakst opened this issue Dec 4, 2024 · 7 comments
Assignees
Labels
enhancement New feature or request mirascope

Comments

@willbakst
Copy link
Contributor

willbakst commented Dec 4, 2024

Description

We currently support provider-agnostic flows through prompt templates combined with call decorators to produce multiple provider-specific calls. I think we can further improve this by providing a single decorator that supports calling any provider+model pairing.

My rough thought is as follows:

from mirascope import llm

@llm.call("openai:gpt-4o-mini")
def recommend_book(genre: str) -> str:
    return f"Recommend a {genre} book"

response = recommend_book("fantasy")  # response will have type `CallResponse[ChatCompletion]`
print(response.content)

The CallResponse[ResponseTypeT] return type would operate just like a provider-specific call response except that it would coerce certain fields (e.g. finish_reason into a more standardized format. I also think that messages such as response.message_param should return BaseMessageParam in this case so that the CallResponse class can operate more truly as a provider-agnostic call response.

We will also need to create additional provider-agnostic classes (e.g. Stream[...] and Tool[...]) so that we can return correctly typed objects down to the original response.

This should also make it possible to switch providers in the middle of a chat. For example, if you're maintaining a history of messages purely as BaseMessageParam instances, you could switch to Anthropic if you're getting rate limited by OpenAI without having to worry about conversion etc.

On the naming front, do we like from mirascope import llm and llm.call or should we instead do something like import mirascope and mirascope.call? For example:

import mirascope

@mirascope.call("openai:gpt-4o-mini")
def recommend_book(genre: str) -> str:
    return f"Recommend a {genre} book"

One note is that we'll need to be more vigilant around updating the accepted model strings since we'll have to use Literal[...] typing with overloads to get proper typing (meaning that new models won't have correct typing until we add them). This isn't difficult, just something to be mindful of. Also worth looking into if this is actually the case or if there's some way to do a form of more general matching on the string.

Lastly, it might be worth updating the documentation to use this general form everywhere. This would enable removing a level of tabs from the docs that are somewhat difficult to maintain. Of course, we should still document the provider-specific usage, but I think we should reverse the ratio once this is implemented (i.e. have a provider-specific section and then have everything else be the provider-agnostic form). This means we should also add a new section (either page or full tab) that clearly shows supported providers as well as what each provider supports across Mirascope features. A simple table would likely suffice. My current preference leans toward a tab rather than a page.

It's also likely worth continuing to write examples for every provider-specific form for everything we support as a means of testing (since we run pyright on the examples) even if we don't render the examples in the docs. Need to figure out if the maintenance cost is worthwhile or if there's a better approach that would be easier to maintain.

@willbakst
Copy link
Contributor Author

Another thing worth thinking about would be runtime overrides. For example:

from mirascope import llm

@llm.call("openai:gpt-4o-mini")
def recommend_book(genre: str) -> str:
    return f"Recommend a {genre} book"

response = recommend_book(
    "fantasy",
    model_override="anthropic:claude-3-5-sonnet-latest",
    call_params_override={"temperature": 0.7},
)
print(response.content)

Super easy to inject these with type hints I believe as keyword arguments, but I'm not 100% certain yet.

This would be a nice feature to include so that the decorated function is truly provider-agnostic (where the model provided in the decorator works as a default).

@Kigstn
Copy link

Kigstn commented Dec 20, 2024

Another thing worth thinking about would be runtime overrides. For example:

from mirascope import llm

@llm.call("openai:gpt-4o-mini")
def recommend_book(genre: str) -> str:
    return f"Recommend a {genre} book"

response = recommend_book(
    "fantasy",
    model_override="anthropic:claude-3-5-sonnet-latest",
    call_params_override={"temperature": 0.7},
)
print(response.content)

Super easy to inject these with type hints I believe as keyword arguments, but I'm not 100% certain yet.

This would be a nice feature to include so that the decorated function is truly provider-agnostic (where the model provided in the decorator works as a default).

To give me two cents about this: I would personally be in favour of splitting the provider and model name. IMO Parsing openai:gpt-4o-mini is always error prone (what if openai introduces a model with a : in the name?), and does not offer proper typing for the IDE.

I suggest something like this instead:

@llm.call(provider="openai", model="gpt-4o-mini")

@koxudaxi
Copy link
Collaborator

@Kigstn
I'm working on implementing this in the PR. While it's still in progress, our plan is to register all combinations of "{provider}:{model}" as Literals to enable IDE assistance.
However, the main advantage of this new decorator is that it returns a common Response instance that's independent of the input provider. In other words, regardless of which model you use, you only need to work with a common Response class (CallResponse).

@Kigstn
Copy link

Kigstn commented Dec 21, 2024

@Kigstn I'm working on implementing this in the PR. While it's still in progress, our plan is to register all combinations of "{provider}:{model}" as Literals to enable IDE assistance. However, the main advantage of this new decorator is that it returns a common Response instance that's independent of the input provider. In other words, regardless of which model you use, you only need to work with a common Response class (CallResponse).

Ah nice, that alleviates by biggest concern! I think I would personally still prefer separating the provider & model. In my eyes they are two different things and thus should not be connected :)

@koxudaxi
Copy link
Collaborator

While I've implemented the llm.call decorator, I'm having issues with overriding provider-specific parameters using override arguments.

I've created a PR for this issue. While it works as expected at runtime, we're finding it difficult to implement our intended type specifications in Python.

Specifically, we can't add keyword arguments like model_override to functions wrapped by the decorator. The only option available is to add them as positional arguments at the beginning using typing.Concatenate.
https://peps.python.org/pep-0612/#concatenating-keyword-parameters

After thoroughly reviewing the PEP for Unpack[TypedDict], it appears that merging or combining types isn't possible. I'm unsure whether we should address this issue since it consistently produces type checking errors when the specification can't be implemented.
https://peps.python.org/pep-0692/

As alternative solutions, I've considered:

response = recommend_book.override(
    model_override="anthropic:claude-3-5-sonnet-latest",
    call_params_override={"temperature": 0.7},
)("fantasy")

or:

response = llm.create_call(recommend_book,
    model_override="anthropic:claude-3-5-sonnet-latest",
    call_params_override={"temperature": 0.7},
)("fantasy")

However, the first approach feels unnatural as it adds methods to what should be a function, and the second approach might be somewhat less intuitive to use.

@willbakst willbakst added the mirascope label Jan 7, 2025 — with Linear
@willbakst
Copy link
Contributor Author

What if we did something like this?

response = llm.override(
    recommend_book,
    model_override="anthropic:claude-3-5-sonnet-latest",
    call_params_override={"temperature": 0.7},
)("fantasy")

@koxudaxi
Copy link
Collaborator

koxudaxi commented Jan 7, 2025

@willbakst
OK, I have updated the implementation.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
enhancement New feature or request mirascope
Projects
None yet
Development

No branches or pull requests

3 participants