llm-connector 0.4.20

Next-generation Rust library for LLM protocol abstraction. Configuration-driven architecture with unified output format. Supports 9+ providers (OpenAI, Anthropic, Aliyun, Zhipu, Ollama, Tencent, Volcengine, LongCat) with clean Protocol/Provider separation, type-safe interface, and universal streaming.
Documentation
//! LongCat API 服务提供商实现
//!
//! LongCat 支持两种 API 格式:
//! 1. OpenAI 格式 - 使用 OpenAI 兼容接口
//! 2. Anthropic 格式 - 使用 Anthropic 兼容接口,但认证方式为 Bearer token
//!
//! 注意:LongCat 的 Anthropic 格式使用 `Authorization: Bearer` 认证,
//! 而不是标准 Anthropic 的 `x-api-key` 认证。

use crate::core::{ConfigurableProtocol, ProviderBuilder, ProtocolConfig, EndpointConfig, AuthConfig};
use crate::protocols::AnthropicProtocol;
use crate::error::LlmConnectorError;

/// LongCat Anthropic 格式协议适配器
///
/// 使用 ConfigurableProtocol 包装 Anthropic protocol,使用 Bearer 认证
pub type LongCatAnthropicProtocol = ConfigurableProtocol<AnthropicProtocol>;

/// LongCat Anthropic 格式服务提供商类型
pub type LongCatAnthropicProvider = crate::core::GenericProvider<LongCatAnthropicProtocol>;

/// 创建 LongCat Anthropic 格式服务提供商
///
/// # 参数
/// - `api_key`: LongCat API 密钥 (格式: ak_...)
///
/// # 示例
/// ```rust,no_run
/// use llm_connector::providers::longcat_anthropic;
///
/// let provider = longcat_anthropic("ak_...").unwrap();
/// ```
pub fn longcat_anthropic(api_key: &str) -> Result<LongCatAnthropicProvider, LlmConnectorError> {
    longcat_anthropic_with_config(api_key, None, None, None)
}

/// 创建带有自定义配置的 LongCat Anthropic 服务提供商
///
/// # 参数
/// - `api_key`: API 密钥
/// - `base_url`: 自定义基础 URL (可选,默认为 LongCat Anthropic 端点)
/// - `timeout_secs`: 超时时间(秒) (可选)
/// - `proxy`: 代理 URL (可选)
///
/// # 示例
/// ```rust,no_run
/// use llm_connector::providers::longcat_anthropic_with_config;
///
/// let provider = longcat_anthropic_with_config(
///     "ak_...",
///     None, // 使用默认 URL
///     Some(60), // 60秒超时
///     None
/// ).unwrap();
/// ```
pub fn longcat_anthropic_with_config(
    api_key: &str,
    base_url: Option<&str>,
    timeout_secs: Option<u64>,
    proxy: Option<&str>,
) -> Result<LongCatAnthropicProvider, LlmConnectorError> {
    // 创建配置驱动的协议(使用 Bearer 认证 + 额外头部)
    let protocol = ConfigurableProtocol::new(
        AnthropicProtocol::new(api_key),
        ProtocolConfig {
            name: "longcat-anthropic".to_string(),
            endpoints: EndpointConfig {
                chat_template: "{base_url}/v1/messages".to_string(),
                models_template: None,
            },
            auth: AuthConfig::Bearer,  // 使用 Bearer 而不是 x-api-key
            extra_headers: vec![
                ("anthropic-version".to_string(), "2023-06-01".to_string()),
            ],
        }
    );

    // 使用 Builder 构建
    let mut builder = ProviderBuilder::new(
        protocol,
        base_url.unwrap_or("https://api.longcat.chat/anthropic")
    );

    if let Some(timeout) = timeout_secs {
        builder = builder.timeout(timeout);
    }

    if let Some(proxy_url) = proxy {
        builder = builder.proxy(proxy_url);
    }

    builder.build()
}

#[cfg(test)]
mod tests {
    use super::*;
    use crate::core::Protocol;
    
    #[test]
    fn test_longcat_anthropic() {
        let provider = longcat_anthropic("ak_test");
        assert!(provider.is_ok());
    }
    
    #[test]
    fn test_longcat_anthropic_with_config() {
        let provider = longcat_anthropic_with_config(
            "ak_test",
            Some("https://custom.url"),
            Some(60),
            None
        );
        assert!(provider.is_ok());
    }
    
    #[test]
    fn test_longcat_anthropic_protocol_auth_headers() {
        let protocol = ConfigurableProtocol::new(
            AnthropicProtocol::new("ak_test123"),
            ProtocolConfig {
                name: "longcat-anthropic".to_string(),
                endpoints: EndpointConfig {
                    chat_template: "{base_url}/v1/messages".to_string(),
                    models_template: None,
                },
                auth: AuthConfig::Bearer,
                extra_headers: vec![
                    ("anthropic-version".to_string(), "2023-06-01".to_string()),
                ],
            }
        );
        let headers = protocol.auth_headers();

        // 应该使用 Bearer 认证
        assert!(headers.iter().any(|(k, v)| k == "Authorization" && v == "Bearer ak_test123"));

        // 应该包含 anthropic-version
        assert!(headers.iter().any(|(k, v)| k == "anthropic-version" && v == "2023-06-01"));

        // 不应该包含 x-api-key
        assert!(!headers.iter().any(|(k, _)| k == "x-api-key"));
    }
}