feat(gateway): add web search emulation for Anthropic API Key accounts
Inject web search capability for Claude Console (API Key) accounts that
don't natively support Anthropic's web_search tool. When a pure
web_search request is detected, the gateway calls Brave Search or Tavily
API directly and constructs an Anthropic-protocol-compliant SSE/JSON
response without forwarding to upstream.
Backend:
- New `pkg/websearch/` SDK: Brave and Tavily provider implementations
with io.LimitReader, proxy support, and Redis-based quota tracking
(Lua atomic INCR + TTL, DECR rollback on failure)
- Global config via `settings.web_search_emulation_config` (JSON) with
in-process cache + singleflight, input validation, API key merge on
save, and sanitized API responses
- Channel-level toggle via `channels.features_config` JSONB column
(DB migration 101)
- Account-level toggle via `accounts.extra.web_search_emulation`
- Request interception in `Forward()` with SSE streaming response
construction using json.Marshal (no manual string concatenation)
- Manager hot-reload: `RebuildWebSearchManager()` called on config save
and startup via `SetWebSearchRedisClient()`
- 70 unit tests covering providers, manager, config validation,
sanitization, tool detection, query extraction, and response building
Frontend:
- Settings → Gateway tab: Web Search Emulation config card with global
toggle, provider list (add/remove, API key, priority, quota, proxy)
- Channels → Anthropic tab: web search emulation toggle with global
state linkage (disabled when global off)
- Account Create/Edit modals: web search emulation toggle for API Key
type with Toggle component
- Full i18n coverage (zh + en)
2026-04-12 00:02:26 +08:00
|
|
|
package websearch
|
|
|
|
|
|
|
|
|
|
import (
|
|
|
|
|
"bytes"
|
|
|
|
|
"context"
|
|
|
|
|
"encoding/json"
|
|
|
|
|
"fmt"
|
|
|
|
|
"io"
|
|
|
|
|
"net/http"
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
const (
|
|
|
|
|
tavilySearchEndpoint = "https://api.tavily.com/search"
|
|
|
|
|
tavilyProviderName = "tavily"
|
|
|
|
|
tavilySearchDepthBasic = "basic"
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
// TavilyProvider implements web search via the Tavily Search API.
|
|
|
|
|
type TavilyProvider struct {
|
|
|
|
|
apiKey string
|
|
|
|
|
httpClient *http.Client
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// NewTavilyProvider creates a Tavily Search provider.
|
|
|
|
|
// The caller is responsible for configuring the http.Client with proxy/timeouts.
|
|
|
|
|
func NewTavilyProvider(apiKey string, httpClient *http.Client) *TavilyProvider {
|
|
|
|
|
if httpClient == nil {
|
|
|
|
|
httpClient = http.DefaultClient
|
|
|
|
|
}
|
|
|
|
|
return &TavilyProvider{apiKey: apiKey, httpClient: httpClient}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func (t *TavilyProvider) Name() string { return tavilyProviderName }
|
|
|
|
|
|
|
|
|
|
func (t *TavilyProvider) Search(ctx context.Context, req SearchRequest) (*SearchResponse, error) {
|
|
|
|
|
maxResults := req.MaxResults
|
|
|
|
|
if maxResults <= 0 {
|
|
|
|
|
maxResults = defaultMaxResults
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
payload := tavilyRequest{
|
|
|
|
|
APIKey: t.apiKey,
|
|
|
|
|
Query: req.Query,
|
|
|
|
|
MaxResults: maxResults,
|
|
|
|
|
SearchDepth: tavilySearchDepthBasic,
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bodyBytes, err := json.Marshal(payload)
|
|
|
|
|
if err != nil {
|
|
|
|
|
return nil, fmt.Errorf("tavily: encode request: %w", err)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
httpReq, err := http.NewRequestWithContext(ctx, http.MethodPost, tavilySearchEndpoint, bytes.NewReader(bodyBytes))
|
|
|
|
|
if err != nil {
|
|
|
|
|
return nil, fmt.Errorf("tavily: build request: %w", err)
|
|
|
|
|
}
|
|
|
|
|
httpReq.Header.Set("Content-Type", "application/json")
|
|
|
|
|
|
|
|
|
|
resp, err := t.httpClient.Do(httpReq)
|
|
|
|
|
if err != nil {
|
|
|
|
|
return nil, fmt.Errorf("tavily: request failed: %w", err)
|
|
|
|
|
}
|
2026-04-14 12:11:08 +08:00
|
|
|
defer func() { _ = resp.Body.Close() }()
|
feat(gateway): add web search emulation for Anthropic API Key accounts
Inject web search capability for Claude Console (API Key) accounts that
don't natively support Anthropic's web_search tool. When a pure
web_search request is detected, the gateway calls Brave Search or Tavily
API directly and constructs an Anthropic-protocol-compliant SSE/JSON
response without forwarding to upstream.
Backend:
- New `pkg/websearch/` SDK: Brave and Tavily provider implementations
with io.LimitReader, proxy support, and Redis-based quota tracking
(Lua atomic INCR + TTL, DECR rollback on failure)
- Global config via `settings.web_search_emulation_config` (JSON) with
in-process cache + singleflight, input validation, API key merge on
save, and sanitized API responses
- Channel-level toggle via `channels.features_config` JSONB column
(DB migration 101)
- Account-level toggle via `accounts.extra.web_search_emulation`
- Request interception in `Forward()` with SSE streaming response
construction using json.Marshal (no manual string concatenation)
- Manager hot-reload: `RebuildWebSearchManager()` called on config save
and startup via `SetWebSearchRedisClient()`
- 70 unit tests covering providers, manager, config validation,
sanitization, tool detection, query extraction, and response building
Frontend:
- Settings → Gateway tab: Web Search Emulation config card with global
toggle, provider list (add/remove, API key, priority, quota, proxy)
- Channels → Anthropic tab: web search emulation toggle with global
state linkage (disabled when global off)
- Account Create/Edit modals: web search emulation toggle for API Key
type with Toggle component
- Full i18n coverage (zh + en)
2026-04-12 00:02:26 +08:00
|
|
|
|
|
|
|
|
body, err := io.ReadAll(io.LimitReader(resp.Body, maxResponseSize))
|
|
|
|
|
if err != nil {
|
|
|
|
|
return nil, fmt.Errorf("tavily: read body: %w", err)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if resp.StatusCode != http.StatusOK {
|
|
|
|
|
return nil, fmt.Errorf("tavily: status %d: %s", resp.StatusCode, truncateBody(body))
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
var raw tavilyResponse
|
|
|
|
|
if err := json.Unmarshal(body, &raw); err != nil {
|
|
|
|
|
return nil, fmt.Errorf("tavily: decode response: %w", err)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
results := make([]SearchResult, 0, len(raw.Results))
|
|
|
|
|
for _, r := range raw.Results {
|
|
|
|
|
results = append(results, SearchResult{
|
|
|
|
|
URL: r.URL,
|
|
|
|
|
Title: r.Title,
|
|
|
|
|
Snippet: r.Content,
|
|
|
|
|
})
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return &SearchResponse{Results: results, Query: req.Query}, nil
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
type tavilyRequest struct {
|
|
|
|
|
APIKey string `json:"api_key"`
|
|
|
|
|
Query string `json:"query"`
|
|
|
|
|
MaxResults int `json:"max_results"`
|
|
|
|
|
SearchDepth string `json:"search_depth"`
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
type tavilyResponse struct {
|
|
|
|
|
Results []tavilyResult `json:"results"`
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
type tavilyResult struct {
|
|
|
|
|
URL string `json:"url"`
|
|
|
|
|
Title string `json:"title"`
|
|
|
|
|
Content string `json:"content"`
|
|
|
|
|
Score float64 `json:"score"`
|
|
|
|
|
}
|