mirror of
https://github.com/router-for-me/CLIProxyAPI.git
synced 2026-02-03 13:00:52 +08:00
fix(translator): address PR feedback for tool name sanitization
- Pre-compile sanitization regex for better performance. - Optimize SanitizeFunctionName for conciseness and correctness. - Handle 64-char edge cases by truncating before prepending underscore. - Fix bug in Antigravity translator (incorrect join index). - Refactor Gemini translators to avoid redundant sanitization calls. - Add comprehensive unit tests including 64-char edge cases. Co-authored-by: factory-droid[bot] <138933559+factory-droid[bot]@users.noreply.github.com>
This commit is contained in:
@@ -225,11 +225,12 @@ func ConvertClaudeRequestToAntigravity(modelName string, inputRawJSON []byte, _
|
|||||||
} else if contentTypeResult.Type == gjson.String && contentTypeResult.String() == "tool_result" {
|
} else if contentTypeResult.Type == gjson.String && contentTypeResult.String() == "tool_result" {
|
||||||
toolCallID := contentResult.Get("tool_use_id").String()
|
toolCallID := contentResult.Get("tool_use_id").String()
|
||||||
if toolCallID != "" {
|
if toolCallID != "" {
|
||||||
funcName := util.SanitizeFunctionName(toolCallID)
|
rawFuncName := toolCallID
|
||||||
toolCallIDs := strings.Split(toolCallID, "-")
|
toolCallIDs := strings.Split(toolCallID, "-")
|
||||||
if len(toolCallIDs) > 1 {
|
if len(toolCallIDs) > 1 {
|
||||||
funcName = util.SanitizeFunctionName(strings.Join(toolCallIDs[0:len(toolCallIDs)-2], "-"))
|
rawFuncName = strings.Join(toolCallIDs[0:len(toolCallIDs)-1], "-")
|
||||||
}
|
}
|
||||||
|
funcName := util.SanitizeFunctionName(rawFuncName)
|
||||||
functionResponseResult := contentResult.Get("content")
|
functionResponseResult := contentResult.Get("content")
|
||||||
|
|
||||||
functionResponseJSON := `{}`
|
functionResponseJSON := `{}`
|
||||||
|
|||||||
@@ -107,11 +107,12 @@ func ConvertClaudeRequestToCLI(modelName string, inputRawJSON []byte, _ bool) []
|
|||||||
if toolCallID == "" {
|
if toolCallID == "" {
|
||||||
return true
|
return true
|
||||||
}
|
}
|
||||||
funcName := util.SanitizeFunctionName(toolCallID)
|
rawFuncName := toolCallID
|
||||||
toolCallIDs := strings.Split(toolCallID, "-")
|
toolCallIDs := strings.Split(toolCallID, "-")
|
||||||
if len(toolCallIDs) > 1 {
|
if len(toolCallIDs) > 1 {
|
||||||
funcName = util.SanitizeFunctionName(strings.Join(toolCallIDs[0:len(toolCallIDs)-1], "-"))
|
rawFuncName = strings.Join(toolCallIDs[0:len(toolCallIDs)-1], "-")
|
||||||
}
|
}
|
||||||
|
funcName := util.SanitizeFunctionName(rawFuncName)
|
||||||
responseData := contentResult.Get("content").Raw
|
responseData := contentResult.Get("content").Raw
|
||||||
part := `{"functionResponse":{"name":"","response":{"result":""}}}`
|
part := `{"functionResponse":{"name":"","response":{"result":""}}}`
|
||||||
part, _ = sjson.Set(part, "functionResponse.name", funcName)
|
part, _ = sjson.Set(part, "functionResponse.name", funcName)
|
||||||
|
|||||||
@@ -100,11 +100,12 @@ func ConvertClaudeRequestToGemini(modelName string, inputRawJSON []byte, _ bool)
|
|||||||
if toolCallID == "" {
|
if toolCallID == "" {
|
||||||
return true
|
return true
|
||||||
}
|
}
|
||||||
funcName := util.SanitizeFunctionName(toolCallID)
|
rawFuncName := toolCallID
|
||||||
toolCallIDs := strings.Split(toolCallID, "-")
|
toolCallIDs := strings.Split(toolCallID, "-")
|
||||||
if len(toolCallIDs) > 1 {
|
if len(toolCallIDs) > 1 {
|
||||||
funcName = util.SanitizeFunctionName(strings.Join(toolCallIDs[0:len(toolCallIDs)-1], "-"))
|
rawFuncName = strings.Join(toolCallIDs[0:len(toolCallIDs)-1], "-")
|
||||||
}
|
}
|
||||||
|
funcName := util.SanitizeFunctionName(rawFuncName)
|
||||||
responseData := contentResult.Get("content").Raw
|
responseData := contentResult.Get("content").Raw
|
||||||
part := `{"functionResponse":{"name":"","response":{"result":""}}}`
|
part := `{"functionResponse":{"name":"","response":{"result":""}}}`
|
||||||
part, _ = sjson.Set(part, "functionResponse.name", funcName)
|
part, _ = sjson.Set(part, "functionResponse.name", funcName)
|
||||||
|
|||||||
@@ -26,6 +26,7 @@ func TestSanitizeFunctionName(t *testing.T) {
|
|||||||
{"Exactly 64 chars", "this_is_a_very_long_name_that_exactly_reaches_sixty_four_charact", "this_is_a_very_long_name_that_exactly_reaches_sixty_four_charact"},
|
{"Exactly 64 chars", "this_is_a_very_long_name_that_exactly_reaches_sixty_four_charact", "this_is_a_very_long_name_that_exactly_reaches_sixty_four_charact"},
|
||||||
{"Too long (65 chars)", "this_is_a_very_long_name_that_exactly_reaches_sixty_four_charactX", "this_is_a_very_long_name_that_exactly_reaches_sixty_four_charact"},
|
{"Too long (65 chars)", "this_is_a_very_long_name_that_exactly_reaches_sixty_four_charactX", "this_is_a_very_long_name_that_exactly_reaches_sixty_four_charact"},
|
||||||
{"Very long", "this_is_a_very_long_name_that_exceeds_the_sixty_four_character_limit_for_function_names", "this_is_a_very_long_name_that_exceeds_the_sixty_four_character_l"},
|
{"Very long", "this_is_a_very_long_name_that_exceeds_the_sixty_four_character_limit_for_function_names", "this_is_a_very_long_name_that_exceeds_the_sixty_four_character_l"},
|
||||||
|
{"Starts with digit (64 chars total)", "1234567890123456789012345678901234567890123456789012345678901234", "_123456789012345678901234567890123456789012345678901234567890123"},
|
||||||
{"Empty", "", ""},
|
{"Empty", "", ""},
|
||||||
{"Single character invalid", "@", "_"},
|
{"Single character invalid", "@", "_"},
|
||||||
{"Single character valid", "a", "a"},
|
{"Single character valid", "a", "a"},
|
||||||
|
|||||||
@@ -15,28 +15,29 @@ import (
|
|||||||
log "github.com/sirupsen/logrus"
|
log "github.com/sirupsen/logrus"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
var functionNameSanitizer = regexp.MustCompile(`[^a-zA-Z0-9_.:-]`)
|
||||||
|
|
||||||
// SanitizeFunctionName ensures a function name matches the requirements for Gemini/Vertex AI.
|
// SanitizeFunctionName ensures a function name matches the requirements for Gemini/Vertex AI.
|
||||||
// It replaces invalid characters with underscores, ensures it starts with a letter or underscore,
|
// It replaces invalid characters with underscores, ensures it starts with a letter or underscore,
|
||||||
// and truncates it to 64 characters if necessary.
|
// and truncates it to 64 characters if necessary.
|
||||||
// Regex Rule: [^a-zA-Z0-9_.:-] replaced with _.
|
// Regex Rule: [^a-zA-Z0-9_.:-] replaced with _.
|
||||||
func SanitizeFunctionName(name string) string {
|
func SanitizeFunctionName(name string) string {
|
||||||
if name == "" {
|
if name == "" {
|
||||||
return name
|
return ""
|
||||||
}
|
}
|
||||||
// Replace invalid characters with underscore
|
// Replace invalid characters with underscore
|
||||||
re := regexp.MustCompile(`[^a-zA-Z0-9_.:-]`)
|
sanitized := functionNameSanitizer.ReplaceAllString(name, "_")
|
||||||
sanitized := re.ReplaceAllString(name, "_")
|
|
||||||
|
|
||||||
// Ensure it starts with a letter or underscore
|
// Ensure it starts with a letter or underscore
|
||||||
if len(sanitized) > 0 {
|
first := sanitized[0]
|
||||||
first := sanitized[0]
|
if !((first >= 'a' && first <= 'z') || (first >= 'A' && first <= 'Z') || first == '_') {
|
||||||
if !((first >= 'a' && first <= 'z') || (first >= 'A' && first <= 'Z') || first == '_') {
|
// If it starts with an allowed character but not allowed at the beginning,
|
||||||
// If it starts with an allowed character but not allowed at the beginning,
|
// we must prepend an underscore.
|
||||||
// we must prepend an underscore.
|
// To stay within the 64-character limit while prepending, we may need to truncate first.
|
||||||
sanitized = "_" + sanitized
|
if len(sanitized) >= 64 {
|
||||||
|
sanitized = sanitized[:63]
|
||||||
}
|
}
|
||||||
} else {
|
sanitized = "_" + sanitized
|
||||||
sanitized = "_"
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Truncate to 64 characters
|
// Truncate to 64 characters
|
||||||
|
|||||||
Reference in New Issue
Block a user