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

[Microsoft.Extensions.AI.OpenAI] Add logging support to HandleToolCallsAsync #5715

Open
paulbatum opened this issue Dec 2, 2024 · 4 comments

Comments

@paulbatum
Copy link

When using OpenAIRealtimeExtensions.HandleToolCallsAsync there is no way to capture the error information from failed tool calls. The errors are reported to the model but then swallowed without any additional logging:

private static async Task<ConversationItem?> GetFunctionCallOutputAsync(
this ConversationItemStreamingFinishedUpdate update,
IReadOnlyList<AIFunction> tools,
bool? detailedErrors = false,
JsonSerializerOptions? jsonSerializerOptions = null,
CancellationToken cancellationToken = default)
{
if (!string.IsNullOrEmpty(update.FunctionName)
&& tools.FirstOrDefault(t => t.Metadata.Name == update.FunctionName) is AIFunction aiFunction)
{
var jsonOptions = jsonSerializerOptions ?? AIJsonUtilities.DefaultOptions;
var functionCallContent = FunctionCallContent.CreateFromParsedArguments(
update.FunctionCallArguments, update.FunctionCallId, update.FunctionName,
argumentParser: json => JsonSerializer.Deserialize(json,
(JsonTypeInfo<IDictionary<string, object>>)jsonOptions.GetTypeInfo(typeof(IDictionary<string, object>)))!);
try
{
var result = await aiFunction.InvokeAsync(functionCallContent.Arguments, cancellationToken).ConfigureAwait(false);
var resultJson = JsonSerializer.Serialize(result, jsonOptions.GetTypeInfo(typeof(object)));
return ConversationItem.CreateFunctionCallOutput(update.FunctionCallId, resultJson);
}
catch (JsonException)
{
return ConversationItem.CreateFunctionCallOutput(update.FunctionCallId, "Invalid JSON");
}
catch (Exception e) when (!cancellationToken.IsCancellationRequested)
{
var message = "Error calling tool";
if (detailedErrors == true)
{
message += $": {e.Message}";
}
return ConversationItem.CreateFunctionCallOutput(update.FunctionCallId, message);
}
}

The equivalent logging code for the non realtime case is in place:
extensions/src/Libraries/Microsoft.Extensions.AI/ChatCompletion/FunctionInvokingChatClient.cs at 5161cb90e1db3c3b6192ce40a3406dabc53db35a · dotnet/extensions.

@stephentoub
Copy link
Member

cc: @SteveSandersonMS

@SteveSandersonMS
Copy link
Member

@paulbatum and I discussed this earlier. I agree with the proposal. The realtime function invoker should be given similar logging semantics to FunctionInvokingChatClient where possible.

@stephentoub
Copy link
Member

The realtime function invoker should be given similar logging semantics to FunctionInvokingChatClient where possible.

Rather than doubling-down on these bespoke helpers, I wonder if at this point we should start working on the abstraction for the bidirectional streaming APIs. We know we're going to want something that encompasses OpenAI's realtime APIs, as well as the text-to-audio and audio-to-text scenarios, and wanting to layer in automatic function invocation highlights the need for a similar pipeline model.

@jeffhandley
Copy link
Member

@stephentoub & @SteveSandersonMS - Can the two of you revisit this and recommend a path forward please? This would be good to label as ai-ready-to-implement for assignment within the next couple weeks.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Projects
None yet
Development

No branches or pull requests

4 participants