mirror of
https://github.com/sipeed/picoclaw.git
synced 2026-06-12 18:08:54 +00:00
Deduplicate further functions
This commit is contained in:
@@ -0,0 +1,27 @@
|
||||
package common
|
||||
|
||||
import "strings"
|
||||
|
||||
// NormalizeBaseURL ensures the Anthropic base URL is properly formatted.
|
||||
// It removes a trailing /v1 suffix if present (to avoid duplication), then
|
||||
// re-appends /v1 when appendV1Suffix is true. An empty apiBase falls back to
|
||||
// defaultBaseURL.
|
||||
func NormalizeBaseURL(apiBase, defaultBaseURL string, appendV1Suffix bool) string {
|
||||
base := strings.TrimSpace(apiBase)
|
||||
if base == "" {
|
||||
return defaultBaseURL
|
||||
}
|
||||
|
||||
base = strings.TrimRight(base, "/")
|
||||
if before, ok := strings.CutSuffix(base, "/v1"); ok {
|
||||
base = before
|
||||
}
|
||||
if base == "" {
|
||||
return defaultBaseURL
|
||||
}
|
||||
|
||||
if appendV1Suffix {
|
||||
return base + "/v1"
|
||||
}
|
||||
return base
|
||||
}
|
||||
@@ -0,0 +1,59 @@
|
||||
package common
|
||||
|
||||
import "testing"
|
||||
|
||||
func TestNormalizeAnthropicBaseURL(t *testing.T) {
|
||||
const defaultURL = "https://api.anthropic.com"
|
||||
const defaultURLWithV1 = "https://api.anthropic.com/v1"
|
||||
|
||||
tests := []struct {
|
||||
name string
|
||||
apiBase string
|
||||
defaultBase string
|
||||
appendV1Suffix bool
|
||||
expected string
|
||||
}{
|
||||
{"empty with v1", "", defaultURLWithV1, true, defaultURLWithV1},
|
||||
{"empty without v1", "", defaultURL, false, defaultURL},
|
||||
{
|
||||
"URL without v1 gets it appended",
|
||||
"https://api.example.com/anthropic", defaultURLWithV1,
|
||||
true, "https://api.example.com/anthropic/v1",
|
||||
},
|
||||
{
|
||||
"URL without v1 stays as-is",
|
||||
"https://api.example.com/anthropic", defaultURL,
|
||||
false, "https://api.example.com/anthropic",
|
||||
},
|
||||
{
|
||||
"URL with v1 remains unchanged when appending",
|
||||
"https://api.example.com/v1", defaultURLWithV1,
|
||||
true, "https://api.example.com/v1",
|
||||
},
|
||||
{
|
||||
"URL with v1 gets it stripped when not appending",
|
||||
"https://api.example.com/v1", defaultURL,
|
||||
false, "https://api.example.com",
|
||||
},
|
||||
{
|
||||
"trailing slash cleaned with v1",
|
||||
"https://api.example.com/anthropic/", defaultURLWithV1,
|
||||
true, "https://api.example.com/anthropic/v1",
|
||||
},
|
||||
{
|
||||
"trailing slash cleaned without v1",
|
||||
"https://api.example.com/anthropic/", defaultURL,
|
||||
false, "https://api.example.com/anthropic",
|
||||
},
|
||||
}
|
||||
|
||||
for _, tt := range tests {
|
||||
t.Run(tt.name, func(t *testing.T) {
|
||||
got := NormalizeBaseURL(tt.apiBase, tt.defaultBase, tt.appendV1Suffix)
|
||||
if got != tt.expected {
|
||||
t.Errorf("NormalizeAnthropicBaseURL(%q, %q, %v) = %q, want %q",
|
||||
tt.apiBase, tt.defaultBase, tt.appendV1Suffix, got, tt.expected)
|
||||
}
|
||||
})
|
||||
}
|
||||
}
|
||||
@@ -722,64 +722,6 @@ func TestParseDataAudioURL(t *testing.T) {
|
||||
}
|
||||
}
|
||||
|
||||
// --- NormalizeAnthropicBaseURL tests ---
|
||||
|
||||
func TestNormalizeAnthropicBaseURL(t *testing.T) {
|
||||
const defaultURL = "https://api.anthropic.com"
|
||||
const defaultURLWithV1 = "https://api.anthropic.com/v1"
|
||||
|
||||
tests := []struct {
|
||||
name string
|
||||
apiBase string
|
||||
defaultBase string
|
||||
appendV1Suffix bool
|
||||
expected string
|
||||
}{
|
||||
{"empty with v1", "", defaultURLWithV1, true, defaultURLWithV1},
|
||||
{"empty without v1", "", defaultURL, false, defaultURL},
|
||||
{
|
||||
"URL without v1 gets it appended",
|
||||
"https://api.example.com/anthropic", defaultURLWithV1,
|
||||
true, "https://api.example.com/anthropic/v1",
|
||||
},
|
||||
{
|
||||
"URL without v1 stays as-is",
|
||||
"https://api.example.com/anthropic", defaultURL,
|
||||
false, "https://api.example.com/anthropic",
|
||||
},
|
||||
{
|
||||
"URL with v1 remains unchanged when appending",
|
||||
"https://api.example.com/v1", defaultURLWithV1,
|
||||
true, "https://api.example.com/v1",
|
||||
},
|
||||
{
|
||||
"URL with v1 gets it stripped when not appending",
|
||||
"https://api.example.com/v1", defaultURL,
|
||||
false, "https://api.example.com",
|
||||
},
|
||||
{
|
||||
"trailing slash cleaned with v1",
|
||||
"https://api.example.com/anthropic/", defaultURLWithV1,
|
||||
true, "https://api.example.com/anthropic/v1",
|
||||
},
|
||||
{
|
||||
"trailing slash cleaned without v1",
|
||||
"https://api.example.com/anthropic/", defaultURL,
|
||||
false, "https://api.example.com/anthropic",
|
||||
},
|
||||
}
|
||||
|
||||
for _, tt := range tests {
|
||||
t.Run(tt.name, func(t *testing.T) {
|
||||
got := NormalizeAnthropicBaseURL(tt.apiBase, tt.defaultBase, tt.appendV1Suffix)
|
||||
if got != tt.expected {
|
||||
t.Errorf("NormalizeAnthropicBaseURL(%q, %q, %v) = %q, want %q",
|
||||
tt.apiBase, tt.defaultBase, tt.appendV1Suffix, got, tt.expected)
|
||||
}
|
||||
})
|
||||
}
|
||||
}
|
||||
|
||||
// --- WrapHTMLResponseError tests ---
|
||||
|
||||
func TestWrapHTMLResponseError(t *testing.T) {
|
||||
|
||||
@@ -0,0 +1,70 @@
|
||||
package common
|
||||
|
||||
import (
|
||||
"encoding/json"
|
||||
"strings"
|
||||
|
||||
"github.com/sipeed/picoclaw/pkg/providers/protocoltypes"
|
||||
)
|
||||
|
||||
// NormalizeStoredToolCall extracts the tool name, arguments, and thought signature
|
||||
// from a stored ToolCall. It handles both the top-level fields and the nested
|
||||
// Function struct used by different API formats.
|
||||
func NormalizeStoredToolCall(tc protocoltypes.ToolCall) (string, map[string]any, string) {
|
||||
name := tc.Name
|
||||
args := tc.Arguments
|
||||
thoughtSignature := ""
|
||||
|
||||
if name == "" && tc.Function != nil {
|
||||
name = tc.Function.Name
|
||||
thoughtSignature = tc.Function.ThoughtSignature
|
||||
} else if tc.Function != nil {
|
||||
thoughtSignature = tc.Function.ThoughtSignature
|
||||
}
|
||||
|
||||
if args == nil {
|
||||
args = map[string]any{}
|
||||
}
|
||||
|
||||
if len(args) == 0 && tc.Function != nil && tc.Function.Arguments != "" {
|
||||
var parsed map[string]any
|
||||
if err := json.Unmarshal([]byte(tc.Function.Arguments), &parsed); err == nil && parsed != nil {
|
||||
args = parsed
|
||||
}
|
||||
}
|
||||
|
||||
return name, args, thoughtSignature
|
||||
}
|
||||
|
||||
// ResolveToolResponseName returns the tool name for a given tool call ID.
|
||||
// It first checks the provided name map, then falls back to inferring the
|
||||
// name from the call ID format.
|
||||
func ResolveToolResponseName(toolCallID string, toolCallNames map[string]string) string {
|
||||
if toolCallID == "" {
|
||||
return ""
|
||||
}
|
||||
|
||||
if name, ok := toolCallNames[toolCallID]; ok && name != "" {
|
||||
return name
|
||||
}
|
||||
|
||||
return InferToolNameFromCallID(toolCallID)
|
||||
}
|
||||
|
||||
// InferToolNameFromCallID extracts a tool name from a call ID in the format
|
||||
// "call_<name>_<suffix>". Returns the original ID if it doesn't match.
|
||||
func InferToolNameFromCallID(toolCallID string) string {
|
||||
if !strings.HasPrefix(toolCallID, "call_") {
|
||||
return toolCallID
|
||||
}
|
||||
|
||||
rest := strings.TrimPrefix(toolCallID, "call_")
|
||||
if idx := strings.LastIndex(rest, "_"); idx > 0 {
|
||||
candidate := rest[:idx]
|
||||
if candidate != "" {
|
||||
return candidate
|
||||
}
|
||||
}
|
||||
|
||||
return toolCallID
|
||||
}
|
||||
@@ -0,0 +1,146 @@
|
||||
package common
|
||||
|
||||
import (
|
||||
"testing"
|
||||
|
||||
"github.com/sipeed/picoclaw/pkg/providers/protocoltypes"
|
||||
)
|
||||
|
||||
func TestNormalizeStoredToolCall_TopLevelFields(t *testing.T) {
|
||||
tc := protocoltypes.ToolCall{
|
||||
Name: "search",
|
||||
Arguments: map[string]any{"q": "hello"},
|
||||
}
|
||||
name, args, sig := NormalizeStoredToolCall(tc)
|
||||
if name != "search" {
|
||||
t.Errorf("name = %q, want %q", name, "search")
|
||||
}
|
||||
if args["q"] != "hello" {
|
||||
t.Errorf("args[q] = %v, want %q", args["q"], "hello")
|
||||
}
|
||||
if sig != "" {
|
||||
t.Errorf("thoughtSignature = %q, want empty", sig)
|
||||
}
|
||||
}
|
||||
|
||||
func TestNormalizeStoredToolCall_FallsBackToFunction(t *testing.T) {
|
||||
tc := protocoltypes.ToolCall{
|
||||
Function: &protocoltypes.FunctionCall{
|
||||
Name: "read_file",
|
||||
Arguments: `{"path":"/tmp"}`,
|
||||
ThoughtSignature: "sig123",
|
||||
},
|
||||
}
|
||||
name, args, sig := NormalizeStoredToolCall(tc)
|
||||
if name != "read_file" {
|
||||
t.Errorf("name = %q, want %q", name, "read_file")
|
||||
}
|
||||
if args["path"] != "/tmp" {
|
||||
t.Errorf("args[path] = %v, want %q", args["path"], "/tmp")
|
||||
}
|
||||
if sig != "sig123" {
|
||||
t.Errorf("thoughtSignature = %q, want %q", sig, "sig123")
|
||||
}
|
||||
}
|
||||
|
||||
func TestNormalizeStoredToolCall_TopLevelNameWithFunctionSig(t *testing.T) {
|
||||
tc := protocoltypes.ToolCall{
|
||||
Name: "search",
|
||||
Arguments: map[string]any{"q": "hi"},
|
||||
Function: &protocoltypes.FunctionCall{
|
||||
ThoughtSignature: "thought1",
|
||||
},
|
||||
}
|
||||
name, _, sig := NormalizeStoredToolCall(tc)
|
||||
if name != "search" {
|
||||
t.Errorf("name = %q, want %q", name, "search")
|
||||
}
|
||||
if sig != "thought1" {
|
||||
t.Errorf("thoughtSignature = %q, want %q", sig, "thought1")
|
||||
}
|
||||
}
|
||||
|
||||
func TestNormalizeStoredToolCall_NilArgs(t *testing.T) {
|
||||
tc := protocoltypes.ToolCall{Name: "test"}
|
||||
_, args, _ := NormalizeStoredToolCall(tc)
|
||||
if args == nil {
|
||||
t.Fatal("args should not be nil")
|
||||
}
|
||||
if len(args) != 0 {
|
||||
t.Errorf("args should be empty, got %v", args)
|
||||
}
|
||||
}
|
||||
|
||||
func TestNormalizeStoredToolCall_EmptyArgsParseFromFunction(t *testing.T) {
|
||||
tc := protocoltypes.ToolCall{
|
||||
Name: "tool",
|
||||
Arguments: map[string]any{},
|
||||
Function: &protocoltypes.FunctionCall{
|
||||
Arguments: `{"key":"val"}`,
|
||||
},
|
||||
}
|
||||
_, args, _ := NormalizeStoredToolCall(tc)
|
||||
if args["key"] != "val" {
|
||||
t.Errorf("args[key] = %v, want %q", args["key"], "val")
|
||||
}
|
||||
}
|
||||
|
||||
func TestNormalizeStoredToolCall_InvalidFunctionJSON(t *testing.T) {
|
||||
tc := protocoltypes.ToolCall{
|
||||
Name: "tool",
|
||||
Function: &protocoltypes.FunctionCall{
|
||||
Arguments: `not-json`,
|
||||
},
|
||||
}
|
||||
_, args, _ := NormalizeStoredToolCall(tc)
|
||||
if len(args) != 0 {
|
||||
t.Errorf("args should be empty for invalid JSON, got %v", args)
|
||||
}
|
||||
}
|
||||
|
||||
func TestResolveToolResponseName_FromMap(t *testing.T) {
|
||||
names := map[string]string{"call_1": "search"}
|
||||
got := ResolveToolResponseName("call_1", names)
|
||||
if got != "search" {
|
||||
t.Errorf("got %q, want %q", got, "search")
|
||||
}
|
||||
}
|
||||
|
||||
func TestResolveToolResponseName_EmptyID(t *testing.T) {
|
||||
got := ResolveToolResponseName("", map[string]string{"x": "y"})
|
||||
if got != "" {
|
||||
t.Errorf("got %q, want empty", got)
|
||||
}
|
||||
}
|
||||
|
||||
func TestResolveToolResponseName_FallsBackToInfer(t *testing.T) {
|
||||
got := ResolveToolResponseName("call_search_docs_999", map[string]string{})
|
||||
if got != "search_docs" {
|
||||
t.Errorf("got %q, want %q", got, "search_docs")
|
||||
}
|
||||
}
|
||||
|
||||
func TestInferToolNameFromCallID(t *testing.T) {
|
||||
tests := []struct {
|
||||
name string
|
||||
id string
|
||||
want string
|
||||
}{
|
||||
{"standard format", "call_search_docs_999", "search_docs"},
|
||||
{"single name", "call_read_123", "read"},
|
||||
{"no call prefix", "some_id", "some_id"},
|
||||
{"call prefix no underscore suffix", "call_onlyname", "call_onlyname"},
|
||||
{"empty string", "", ""},
|
||||
}
|
||||
for _, tt := range tests {
|
||||
t.Run(tt.name, func(t *testing.T) {
|
||||
got := InferToolNameFromCallID(tt.id)
|
||||
if got != tt.want {
|
||||
t.Errorf(
|
||||
"InferToolNameFromCallID(%q) = %q, want %q",
|
||||
tt.id, got, tt.want,
|
||||
)
|
||||
}
|
||||
})
|
||||
}
|
||||
}
|
||||
Reference in New Issue
Block a user