tools.rs 2.83 KB
Newer Older
1
2
3
// SPDX-FileCopyrightText: Copyright (c) 2024-2025 NVIDIA CORPORATION & AFFILIATES. All rights reserved.
// SPDX-License-Identifier: Apache-2.0

4
5
pub use super::config::ToolCallConfig;
pub use super::parsers::detect_and_parse_tool_call;
6
7
8
9

/// Try parsing a string as a structured tool call, for aggregation usage.
///
/// If successful, returns a `ChatCompletionMessageToolCall`.
10
pub async fn try_tool_call_parse_aggregate(
11
    message: &str,
12
    parser_str: Option<&str>,
13
    tools: Option<&[super::ToolDefinition]>,
14
15
16
17
) -> anyhow::Result<(
    Vec<dynamo_async_openai::types::ChatCompletionMessageToolCall>,
    Option<String>,
)> {
18
19
20
21
22
    if parser_str.is_none() {
        tracing::info!("No tool parser provided. Trying parsing with default parser.");
    } else {
        tracing::info!("Using tool parser: {:?}", parser_str);
    }
23
    let (parsed, content) = detect_and_parse_tool_call(message, parser_str, tools).await?;
24
    if parsed.is_empty() {
25
        return Ok((vec![], content));
26
    }
27
28
29
30
31
32
33
34
35
36
37
    Ok((
        parsed
            .into_iter()
            .map(
                |parsed| dynamo_async_openai::types::ChatCompletionMessageToolCall {
                    id: parsed.id,
                    r#type: dynamo_async_openai::types::ChatCompletionToolType::Function,
                    function: dynamo_async_openai::types::FunctionCall {
                        name: parsed.function.name,
                        arguments: parsed.function.arguments,
                    },
38
                },
39
40
41
42
            )
            .collect(),
        content,
    ))
43
44
45
46
47
}

/// Try parsing a string as a structured tool call, for streaming (delta) usage.
///
/// If successful, returns a `ChatCompletionMessageToolCallChunk`.
48
pub async fn try_tool_call_parse_stream(
49
    message: &str,
50
    parser_str: Option<&str>,
51
    tools: Option<&[super::ToolDefinition]>,
52
53
54
55
) -> anyhow::Result<(
    Vec<dynamo_async_openai::types::ChatCompletionMessageToolCallChunk>,
    Option<String>,
)> {
56
    let (parsed, content) = detect_and_parse_tool_call(message, parser_str, tools).await?;
57
    if parsed.is_empty() {
58
        return Ok((vec![], content));
59
    }
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
    Ok((
        parsed
            .into_iter()
            .enumerate()
            .map(
                |(idx, parsed)| dynamo_async_openai::types::ChatCompletionMessageToolCallChunk {
                    index: idx as u32,
                    id: Some(parsed.id),
                    r#type: Some(dynamo_async_openai::types::ChatCompletionToolType::Function),
                    function: Some(dynamo_async_openai::types::FunctionCallStream {
                        name: Some(parsed.function.name),
                        arguments: Some(parsed.function.arguments),
                    }),
                    // Add other fields as needed if required by the struct definition
                },
            )
            .collect(),
        content,
    ))
79
}