Skip to main content

vtcode_core/llm/providers/
ollama.rs

1use crate::config::TimeoutsConfig;
2use crate::config::constants::{env_vars, models, urls};
3use crate::config::core::{AnthropicConfig, ModelConfig, PromptCachingConfig};
4use crate::llm::client::LLMClient;
5use crate::llm::provider::{
6    ContentPart, FinishReason, LLMError, LLMProvider, LLMRequest, LLMResponse, LLMStream,
7    LLMStreamEvent, Message, MessageContent, MessageRole, ToolCall, ToolChoice, ToolDefinition,
8    Usage,
9};
10use crate::utils::http_client;
11use anyhow::Result;
12use async_stream::try_stream;
13use async_trait::async_trait;
14use futures::StreamExt;
15use hashbrown::HashMap;
16use reqwest::Client as HttpClient;
17use serde::{Deserialize, Serialize};
18use serde_json::{Map, Value};
19
20pub mod client;
21pub mod parser;
22pub mod pull;
23pub mod url;
24
25pub use client::OllamaClient;
26pub use parser::pull_events_from_value;
27pub use pull::{
28    CliPullProgressReporter, OllamaPullEvent, OllamaPullProgressReporter, TuiPullProgressReporter,
29};
30pub use url::{base_url_to_host_root, is_openai_compatible_base_url};
31
32use semver::{Version, VersionReq};
33
34use super::common::{
35    assistant_interleaved_history_text, collect_history_system_directives,
36    extract_reasoning_text_from_detail_values, extract_reasoning_text_from_serialized_details,
37    is_minimax_m2_model, merge_system_prompt_with_history_directives, override_base_url,
38    parse_client_prompt_common, resolve_model, serialize_reasoning_detail_values,
39};
40use super::error_handling::{format_network_error, format_parse_error};
41
42// ============================================================================
43// Wire API Detection (adapted from OpenAI Codex's codex-ollama/src/lib.rs)
44// ============================================================================
45
46/// Wire protocol that the Ollama server supports.
47/// Based on OpenAI Codex's WireApi enum.
48#[derive(Debug, Clone, Copy, PartialEq, Eq)]
49pub enum OllamaWireApi {
50    /// The Responses API (OpenAI-compatible at `/v1/responses`).
51    Responses,
52    /// Regular Chat Completions compatible with `/v1/chat/completions`.
53    Chat,
54}
55
56/// Result of detecting which wire API the Ollama server supports.
57pub struct WireApiDetection {
58    pub wire_api: OllamaWireApi,
59    pub version: Option<Version>,
60}
61
62/// Version requirement for Ollama servers that support the Responses API.
63/// Release versions >= 0.13.3 support the Responses API.
64static RESPONSES_API_VERSION_REQ: std::sync::LazyLock<VersionReq> =
65    std::sync::LazyLock::new(|| {
66        VersionReq::parse(">=0.13.3").expect("valid version requirement literal")
67    });
68
69/// Determine which wire API to use based on the Ollama server version.
70///
71/// Version 0.0.0 is used for development builds, which typically support the
72/// latest features.
73fn wire_api_for_version(version: &Version) -> OllamaWireApi {
74    if *version == Version::new(0, 0, 0) || RESPONSES_API_VERSION_REQ.matches(version) {
75        OllamaWireApi::Responses
76    } else {
77        OllamaWireApi::Chat
78    }
79}
80
81/// Detect which wire API the running Ollama server supports based on its version.
82/// Returns `Ok(None)` when the version endpoint is missing or unparsable; callers
83/// should keep the configured default in that case.
84///
85/// Adapted from OpenAI Codex's codex-ollama/src/lib.rs
86pub async fn detect_wire_api(
87    base_url: Option<String>,
88) -> std::io::Result<Option<WireApiDetection>> {
89    let resolved_base_url = override_base_url(
90        urls::OLLAMA_API_BASE,
91        base_url,
92        Some(env_vars::OLLAMA_BASE_URL),
93    );
94
95    let client = match OllamaClient::try_from_base_url(&resolved_base_url).await {
96        Ok(c) => c,
97        Err(e) => {
98            tracing::debug!("Failed to connect to Ollama server for version detection: {e}");
99            return Ok(None);
100        }
101    };
102
103    let Some(version) = client.fetch_version().await? else {
104        return Ok(None);
105    };
106
107    let wire_api = wire_api_for_version(&version);
108
109    Ok(Some(WireApiDetection {
110        wire_api,
111        version: Some(version),
112    }))
113}
114
115/// Prepare the local OSS environment when using Ollama.
116///
117/// - Ensures a local Ollama server is reachable.
118/// - Checks if the model exists locally and pulls it if missing.
119///
120/// Adapted from OpenAI Codex's codex-ollama/src/lib.rs
121pub async fn ensure_oss_ready(
122    model: Option<&str>,
123    base_url: Option<String>,
124) -> std::io::Result<()> {
125    let target_model = model.unwrap_or(models::ollama::DEFAULT_MODEL);
126
127    let resolved_base_url = override_base_url(
128        urls::OLLAMA_API_BASE,
129        base_url,
130        Some(env_vars::OLLAMA_BASE_URL),
131    );
132
133    // Verify local Ollama is reachable
134    let ollama_client = OllamaClient::try_from_base_url(&resolved_base_url).await?;
135
136    // If the model is not present locally, pull it
137    match ollama_client.fetch_models().await {
138        Ok(existing_models) => {
139            if !existing_models.iter().any(|m| m == target_model) {
140                tracing::info!("Model '{target_model}' not found locally, pulling...");
141                let mut reporter = CliPullProgressReporter::new();
142                ollama_client
143                    .pull_with_reporter(target_model, &mut reporter)
144                    .await?;
145            }
146        }
147        Err(e) => {
148            tracing::warn!("Failed to list Ollama models: {e}");
149            // Continue anyway; model might exist but listing failed
150        }
151    }
152
153    Ok(())
154}
155
156#[derive(Debug, Deserialize, Serialize)]
157struct OllamaTagsResponse {
158    models: Vec<OllamaTag>,
159}
160
161#[derive(Debug, Deserialize, Serialize)]
162struct OllamaTag {
163    name: Option<String>,
164    model: Option<String>,
165    modified_at: Option<String>,
166    size: Option<u64>,
167    digest: Option<String>,
168    details: Option<OllamaModelDetails>,
169}
170
171#[derive(Debug, Deserialize, Serialize)]
172struct OllamaModelDetails {
173    format: Option<String>,
174    family: Option<String>,
175    families: Option<Vec<String>>,
176    parameter_size: Option<String>,
177    quantization_level: Option<String>,
178}
179
180pub(super) fn ollama_model_name_from_fields<'a>(
181    name: Option<&'a str>,
182    model: Option<&'a str>,
183) -> Option<&'a str> {
184    name.or(model)
185        .map(str::trim)
186        .filter(|value| !value.is_empty())
187}
188
189pub(super) const OLLAMA_CONNECTION_ERROR: &str = "No running Ollama server detected. Start it with: `ollama serve` (after installing)\n\
190     Install instructions: https://github.com/ollama/ollama?tab=readme-ov-file";
191
192/// Fetches available local Ollama models from the Ollama API endpoint
193pub async fn fetch_ollama_models(base_url: Option<String>) -> Result<Vec<String>, anyhow::Error> {
194    use crate::config::constants::{env_vars, urls};
195
196    let resolved_base_url = override_base_url(
197        urls::OLLAMA_API_BASE,
198        base_url,
199        Some(env_vars::OLLAMA_BASE_URL),
200    );
201
202    // Construct the tags endpoint URL
203    let tags_url = format!("{}/api/tags", resolved_base_url);
204
205    // Create HTTP client with connection timeout
206    let client = http_client::create_client_with_timeout(std::time::Duration::from_secs(5));
207
208    // Make GET request to fetch models
209    let response = client
210        .get(&tags_url)
211        .header("Content-Type", "application/json")
212        .send()
213        .await
214        .map_err(|e| {
215            tracing::warn!("Failed to connect to Ollama server: {e:?}");
216            anyhow::anyhow!(OLLAMA_CONNECTION_ERROR)
217        })?;
218
219    if !response.status().is_success() {
220        return Err(anyhow::anyhow!(
221            "Failed to fetch Ollama models: HTTP {}. {}",
222            response.status(),
223            if response.status() == reqwest::StatusCode::NOT_FOUND {
224                "Ensure Ollama server is running."
225            } else {
226                ""
227            }
228        ));
229    }
230
231    // Parse the response
232    let tags_response: OllamaTagsResponse = response
233        .json()
234        .await
235        .map_err(|e| anyhow::anyhow!("Failed to parse Ollama models response: {}", e))?;
236
237    // Extract model names
238    let model_names: Vec<String> = tags_response
239        .models
240        .into_iter()
241        .filter_map(|model| {
242            ollama_model_name_from_fields(model.name.as_deref(), model.model.as_deref())
243                .map(str::to_string)
244        })
245        .collect();
246
247    Ok(model_names)
248}
249
250pub struct OllamaProvider {
251    http_client: HttpClient,
252    base_url: String,
253    model: String,
254    api_key: Option<String>,
255    model_behavior: Option<ModelConfig>,
256}
257
258impl OllamaProvider {
259    fn merged_system_prompt(request: &LLMRequest) -> Option<String> {
260        const HISTORY_DIRECTIVES_SECTION_HEADER: &str = "[History Directives]";
261        let directives = collect_history_system_directives(request);
262        merge_system_prompt_with_history_directives(
263            request.system_prompt.as_ref().map(|prompt| prompt.as_str()),
264            &directives,
265            HISTORY_DIRECTIVES_SECTION_HEADER,
266        )
267    }
268
269    pub fn new(api_key: String) -> Self {
270        Self::with_model(api_key, models::ollama::DEFAULT_MODEL.to_string())
271    }
272
273    pub fn with_model(api_key: String, model: String) -> Self {
274        Self::with_model_internal(model, None, Some(api_key), None)
275    }
276
277    pub fn new_with_client(
278        api_key: String,
279        model: String,
280        http_client: reqwest::Client,
281        base_url: String,
282        _timeouts: TimeoutsConfig,
283    ) -> Self {
284        Self {
285            http_client,
286            base_url,
287            model,
288            api_key: Some(api_key),
289            model_behavior: None,
290        }
291    }
292
293    pub fn from_config(
294        api_key: Option<String>,
295        model: Option<String>,
296        base_url: Option<String>,
297        _prompt_cache: Option<PromptCachingConfig>,
298        _timeouts: Option<TimeoutsConfig>,
299        _anthropic: Option<AnthropicConfig>,
300        model_behavior: Option<ModelConfig>,
301    ) -> Self {
302        let resolved_model = resolve_model(model, models::ollama::DEFAULT_MODEL);
303        Self::with_model_internal(resolved_model, base_url, api_key, model_behavior)
304    }
305
306    fn normalize_api_key(api_key: Option<String>) -> Option<String> {
307        api_key.and_then(|value| {
308            let trimmed = value.trim();
309            if trimmed.is_empty() {
310                None
311            } else {
312                Some(trimmed.to_string())
313            }
314        })
315    }
316
317    fn is_local_base_url(base_url: &str) -> bool {
318        let lowered = base_url.trim().to_ascii_lowercase();
319        const LOCAL_PREFIXES: &[&str] = &[
320            "http://localhost",
321            "https://localhost",
322            "http://127.",
323            "https://127.",
324            "http://0.0.0.0",
325            "https://0.0.0.0",
326            "http://[::1]",
327            "https://[::1]",
328        ];
329
330        LOCAL_PREFIXES
331            .iter()
332            .any(|prefix| lowered.starts_with(prefix))
333    }
334
335    fn with_model_internal(
336        model: String,
337        base_url: Option<String>,
338        api_key: Option<String>,
339        model_behavior: Option<ModelConfig>,
340    ) -> Self {
341        let normalized_api_key = Self::normalize_api_key(api_key);
342        let is_cloud_model = model.contains(":cloud") || model.contains("-cloud");
343
344        let default_base = if is_cloud_model {
345            urls::OLLAMA_CLOUD_API_BASE
346        } else {
347            urls::OLLAMA_API_BASE
348        };
349
350        let resolved_base =
351            override_base_url(default_base, base_url, Some(env_vars::OLLAMA_BASE_URL));
352        let target_is_local = Self::is_local_base_url(&resolved_base);
353
354        // Never send API keys to local endpoints; keep keys for cloud/remote targets
355        let effective_api_key = if target_is_local {
356            None
357        } else {
358            normalized_api_key
359        };
360
361        Self {
362            http_client: http_client::create_default_client(),
363            base_url: resolved_base,
364            model,
365            api_key: effective_api_key,
366            model_behavior,
367        }
368    }
369
370    fn chat_url(&self) -> String {
371        format!("{}/api/chat", self.base_url.trim_end_matches('/'))
372    }
373
374    fn authorized_post(&self, url: String) -> reqwest::RequestBuilder {
375        let builder = self.http_client.post(url);
376        if let Some(api_key) = &self.api_key {
377            builder.bearer_auth(api_key)
378        } else {
379            builder
380        }
381    }
382
383    fn parse_client_prompt(&self, prompt: &str) -> LLMRequest {
384        parse_client_prompt_common(prompt, &self.model, |value| self.parse_chat_request(value))
385    }
386
387    fn parse_chat_request(&self, value: &Value) -> Option<LLMRequest> {
388        let messages_value = value.get("messages")?.as_array()?;
389        let mut system_prompt = value
390            .get("system")
391            .and_then(|entry| entry.as_str())
392            .filter(|text| !text.trim().is_empty())
393            .map(|text| text.to_string());
394        let mut messages = Vec::new();
395
396        for entry in messages_value {
397            let role = entry
398                .get("role")
399                .and_then(|r| r.as_str())
400                .unwrap_or(crate::config::constants::message_roles::USER);
401            let content = entry
402                .get("content")
403                .map(|c| match c {
404                    Value::String(text) => text.to_string(),
405                    other => other.to_string(),
406                })
407                .unwrap_or_default();
408
409            if content.trim().is_empty() {
410                continue;
411            }
412
413            match role {
414                "system" => {
415                    if system_prompt.is_none() {
416                        system_prompt = Some(content);
417                    }
418                }
419                "assistant" => messages.push(Message::assistant(content)),
420                "user" => messages.push(Message::user(content)),
421                _ => {}
422            }
423        }
424
425        if messages.is_empty() {
426            return None;
427        }
428
429        let tools = value
430            .get("tools")
431            .and_then(|entry| serde_json::from_value::<Vec<ToolDefinition>>(entry.clone()).ok());
432
433        Some(LLMRequest {
434            messages,
435            system_prompt: system_prompt.map(std::sync::Arc::new),
436            tools: tools.map(std::sync::Arc::new),
437            model: value
438                .get("model")
439                .and_then(|m| m.as_str())
440                .filter(|m| !m.trim().is_empty())
441                .map(|m| m.to_string())
442                .unwrap_or_else(|| self.model.clone()),
443            max_tokens: value
444                .get("max_tokens")
445                .and_then(|entry| entry.as_u64())
446                .map(|value| value as u32),
447            temperature: value
448                .get("temperature")
449                .and_then(|entry| entry.as_f64())
450                .map(|value| value as f32),
451            stream: value
452                .get("stream")
453                .and_then(|entry| entry.as_bool())
454                .unwrap_or(false),
455            ..Default::default()
456        })
457    }
458
459    fn build_payload(
460        &self,
461        request: &LLMRequest,
462        stream: bool,
463    ) -> Result<OllamaChatRequest, LLMError> {
464        let mut messages = Vec::new();
465        let mut tool_names: HashMap<String, String> = HashMap::new();
466        let minimax_tool_followup_compat = Self::minimax_tool_followup_compat_mode(request);
467
468        if let Some(system) = Self::merged_system_prompt(request) {
469            messages.push(OllamaChatMessage {
470                role: "system".to_string(),
471                content: Some(system),
472                thinking: None,
473                tool_calls: None,
474                tool_call_id: None,
475                tool_name: None,
476                images: None,
477            });
478        }
479
480        for message in &request.messages {
481            let interleaved_content = assistant_interleaved_history_text(message, &request.model);
482            let used_interleaved_content = interleaved_content.is_some();
483            let (content_text, images) = if let Some(interleaved_content) = interleaved_content {
484                (interleaved_content, None)
485            } else {
486                Self::extract_content_and_images(&message.content)
487            };
488            match message.role {
489                MessageRole::System => continue,
490                MessageRole::Tool => {
491                    let tool_name = message
492                        .tool_call_id
493                        .as_ref()
494                        .and_then(|id| tool_names.get(id).cloned());
495                    let tool_name = tool_name.or_else(|| message.origin_tool.clone());
496                    let tool_call_id = if minimax_tool_followup_compat && tool_name.is_some() {
497                        None
498                    } else {
499                        message.tool_call_id.clone()
500                    };
501                    messages.push(OllamaChatMessage {
502                        role: "tool".to_string(),
503                        content: Some(content_text),
504                        thinking: None,
505                        tool_calls: None,
506                        tool_call_id,
507                        tool_name,
508                        images: None,
509                    });
510                }
511                _ => {
512                    let thinking = if used_interleaved_content {
513                        None
514                    } else {
515                        Self::assistant_thinking_history_text(message)
516                    };
517                    let mut payload_message = OllamaChatMessage {
518                        role: message.role.as_generic_str().to_string(),
519                        content: Some(content_text),
520                        thinking,
521                        tool_calls: None,
522                        tool_call_id: None,
523                        tool_name: None,
524                        images,
525                    };
526
527                    if let Some(tool_calls) = message.get_tool_calls() {
528                        let mut converted = Vec::new();
529                        for (index, tool_call) in tool_calls.iter().enumerate() {
530                            if let Some(ref func) = tool_call.function {
531                                if !tool_call.id.is_empty() {
532                                    tool_names
533                                        .entry(tool_call.id.clone())
534                                        .or_insert_with(|| func.name.clone());
535                                }
536
537                                let arguments = tool_call.execution_arguments().map_err(|err| {
538                                    LLMError::InvalidRequest {
539                                        message: format!(
540                                            "Failed to parse tool arguments for Ollama: {err}"
541                                        ),
542                                        metadata: None,
543                                    }
544                                })?;
545                                converted.push(OllamaToolCall {
546                                    call_type: tool_call.call_type.clone(),
547                                    function: OllamaToolFunctionCall {
548                                        name: func.name.clone(),
549                                        arguments: Some(arguments),
550                                        index: Some(index as u32),
551                                    },
552                                });
553                            }
554                        }
555
556                        if !converted.is_empty() {
557                            payload_message.tool_calls = Some(converted);
558                            if payload_message.content.is_none() {
559                                payload_message.content = Some(String::new());
560                            }
561                        }
562                    }
563
564                    messages.push(payload_message);
565                }
566            }
567        }
568
569        let options = if request.temperature.is_some() || request.max_tokens.is_some() {
570            Some(OllamaChatOptions {
571                temperature: request.temperature,
572                num_predict: request.max_tokens,
573            })
574        } else {
575            None
576        };
577
578        let tools = match request.tool_choice {
579            Some(ToolChoice::None) => None,
580            _ => request.tools.as_ref().map(|tools| {
581                tools
582                    .iter()
583                    .filter_map(|tool| {
584                        // Normalize all tools to function type for Ollama compatibility
585                        tool.function.as_ref().map(|func| {
586                            ToolDefinition::function(
587                                func.name.clone(),
588                                func.description.clone(),
589                                func.parameters.clone(),
590                            )
591                        })
592                    })
593                    .collect()
594            }),
595        };
596
597        Ok(OllamaChatRequest {
598            model: request.model.clone(),
599            messages,
600            stream,
601            format: request.output_format.clone(),
602            options,
603            tools,
604            think: Self::think_value(request),
605        })
606    }
607
608    fn assistant_thinking_history_text(message: &Message) -> Option<String> {
609        if message.role != MessageRole::Assistant {
610            return None;
611        }
612
613        message
614            .reasoning
615            .as_deref()
616            .map(str::trim)
617            .filter(|value| !value.is_empty())
618            .map(str::to_owned)
619            .or_else(|| {
620                message
621                    .reasoning_details
622                    .as_deref()
623                    .and_then(extract_reasoning_text_from_detail_values)
624            })
625    }
626
627    fn extract_content_and_images(content: &MessageContent) -> (String, Option<Vec<String>>) {
628        let mut images = Vec::new();
629        if let MessageContent::Parts(parts) = content {
630            for part in parts {
631                if let ContentPart::Image { data, .. } = part {
632                    images.push(data.clone());
633                }
634            }
635        }
636
637        let text = content.as_text().into_owned();
638        let images = if images.is_empty() {
639            None
640        } else {
641            Some(images)
642        };
643        (text, images)
644    }
645
646    fn think_value(request: &LLMRequest) -> Option<Value> {
647        let model_id = request.model.as_str();
648        if Self::minimax_tool_followup_compat_mode(request) {
649            return None;
650        }
651        if !models::ollama::REASONING_MODELS.contains(&model_id) {
652            return None;
653        }
654
655        if models::ollama::REASONING_LEVEL_MODELS.contains(&model_id) {
656            request
657                .reasoning_effort
658                .map(|effort| Value::String(effort.to_string()))
659        } else {
660            Some(Value::Bool(true))
661        }
662    }
663
664    fn minimax_tool_followup_compat_mode(request: &LLMRequest) -> bool {
665        is_minimax_m2_model(&request.model)
666            && request
667                .messages
668                .iter()
669                .any(|message| message.role == MessageRole::Tool || message.has_tool_calls())
670    }
671
672    fn convert_tool_calls(
673        tool_calls: Option<Vec<OllamaResponseToolCall>>,
674    ) -> Result<Option<Vec<ToolCall>>, LLMError> {
675        let Some(tool_calls) = tool_calls else {
676            return Ok(None);
677        };
678
679        if tool_calls.is_empty() {
680            return Ok(None);
681        }
682
683        let mut converted = Vec::new();
684        for (index, call) in tool_calls.into_iter().enumerate() {
685            let function = call.function.ok_or_else(|| LLMError::Provider {
686                message: "Ollama response missing function details for tool call".to_string(),
687                metadata: None,
688            })?;
689
690            let name = function.name.ok_or_else(|| LLMError::Provider {
691                message: "Ollama response missing tool function name".to_string(),
692                metadata: None,
693            })?;
694
695            let arguments_value = function
696                .arguments
697                .unwrap_or_else(|| Value::Object(Map::new()));
698            let arguments = match arguments_value {
699                Value::String(raw) => raw,
700                other => serde_json::to_string(&other).map_err(|err| LLMError::Provider {
701                    message: format!("Failed to serialize Ollama tool arguments: {err}"),
702                    metadata: None,
703                })?,
704            };
705
706            let id = function
707                .index
708                .map(|value| format!("tool_call_{value}"))
709                .unwrap_or_else(|| format!("tool_call_{index}"));
710
711            converted.push(ToolCall::function(id, name, arguments));
712        }
713
714        Ok(Some(converted))
715    }
716
717    fn usage_from_counts(
718        prompt_tokens: Option<u32>,
719        completion_tokens: Option<u32>,
720    ) -> Option<Usage> {
721        if prompt_tokens.is_none() && completion_tokens.is_none() {
722            return None;
723        }
724
725        let prompt = prompt_tokens.unwrap_or_default();
726        let completion = completion_tokens.unwrap_or_default();
727        Some(Usage {
728            prompt_tokens: prompt,
729            completion_tokens: completion,
730            total_tokens: prompt + completion,
731            cached_prompt_tokens: None,
732            cache_creation_tokens: None,
733            cache_read_tokens: None,
734        })
735    }
736
737    fn finish_reason_from(reason: Option<&str>) -> FinishReason {
738        match reason {
739            Some("stop") | None => FinishReason::Stop,
740            Some("length") => FinishReason::Length,
741            Some("tool_calls") => FinishReason::ToolCalls,
742            Some(other) => FinishReason::Error(other.to_string()),
743        }
744    }
745
746    fn build_response(
747        content: Option<String>,
748        tool_calls: Option<Vec<ToolCall>>,
749        reasoning: Option<String>,
750        reasoning_details: Option<Vec<String>>,
751        model: String,
752        finish_reason: Option<&str>,
753        prompt_tokens: Option<u32>,
754        completion_tokens: Option<u32>,
755    ) -> LLMResponse {
756        let mut finish = Self::finish_reason_from(finish_reason);
757        if tool_calls.as_ref().is_some_and(|calls| !calls.is_empty()) {
758            finish = FinishReason::ToolCalls;
759        }
760
761        LLMResponse {
762            content,
763            tool_calls,
764            model,
765            usage: Self::usage_from_counts(prompt_tokens, completion_tokens),
766            finish_reason: finish,
767            reasoning,
768            reasoning_details,
769            tool_references: Vec::new(),
770            request_id: None,
771            organization_id: None,
772            compaction: None,
773        }
774    }
775
776    fn response_from_chat_payload(
777        model: String,
778        parsed: OllamaChatResponse,
779    ) -> Result<LLMResponse, LLMError> {
780        if let Some(error) = parsed.error {
781            return Err(LLMError::Provider {
782                message: error,
783                metadata: None,
784            });
785        }
786
787        let (content, reasoning, tool_calls, native_reasoning_details) =
788            if let Some(message) = parsed.message {
789                let content = message
790                    .content
791                    .and_then(|value| (!value.is_empty()).then_some(value));
792                let reasoning = message
793                    .thinking
794                    .and_then(|value| (!value.is_empty()).then_some(value));
795                let tool_calls = Self::convert_tool_calls(message.tool_calls)?;
796                let native_reasoning_details = message.reasoning_details.filter(|d| !d.is_empty());
797                (content, reasoning, tool_calls, native_reasoning_details)
798            } else {
799                (None, None, None, None)
800            };
801
802        let reasoning = reasoning.or_else(|| {
803            native_reasoning_details
804                .as_deref()
805                .and_then(extract_reasoning_text_from_detail_values)
806        });
807        let mut reasoning_details = native_reasoning_details
808            .as_deref()
809            .and_then(serialize_reasoning_detail_values);
810
811        // Fallback: Extract reasoning from content if not provided natively
812        // This handles MiniMax-M2.5 cloud models that use <think></think> tags
813        let (final_reasoning, final_content) = if reasoning.is_none() {
814            if let Some(ref content_str) = content {
815                let (reasoning_parts, cleaned_content) =
816                    crate::llm::utils::extract_reasoning_content(content_str);
817                if reasoning_parts.is_empty() {
818                    (None, content)
819                } else {
820                    super::common::preserve_interleaved_content_in_reasoning_details(
821                        &mut reasoning_details,
822                        content_str,
823                    );
824                    (
825                        Some(reasoning_parts.join("\n\n")),
826                        cleaned_content.or(content),
827                    )
828                }
829            } else {
830                (None, content)
831            }
832        } else {
833            (reasoning, content)
834        };
835
836        Ok(Self::build_response(
837            final_content,
838            tool_calls,
839            final_reasoning,
840            reasoning_details,
841            model,
842            parsed.done_reason.as_deref(),
843            parsed.prompt_eval_count,
844            parsed.eval_count,
845        ))
846    }
847
848    fn authorized_post_with_key(
849        http_client: &HttpClient,
850        url: &str,
851        api_key: Option<&str>,
852    ) -> reqwest::RequestBuilder {
853        let builder = http_client.post(url.to_string());
854        if let Some(value) = api_key {
855            builder.bearer_auth(value)
856        } else {
857            builder
858        }
859    }
860
861    async fn request_non_stream_response(
862        http_client: &HttpClient,
863        url: &str,
864        api_key: Option<&str>,
865        payload: &OllamaChatRequest,
866        model: String,
867    ) -> Result<LLMResponse, LLMError> {
868        let response = Self::authorized_post_with_key(http_client, url, api_key)
869            .json(payload)
870            .send()
871            .await
872            .map_err(|e| format_network_error("Ollama", &e))?;
873
874        if !response.status().is_success() {
875            let status = response.status();
876            let body = response.text().await.unwrap_or_default();
877            let error_message = Self::extract_error(&body)
878                .unwrap_or_else(|| format!("Ollama request failed ({status}): {body}"));
879            return Err(LLMError::Provider {
880                message: error_message,
881                metadata: None,
882            });
883        }
884
885        let parsed = response
886            .json::<OllamaChatResponse>()
887            .await
888            .map_err(|e| format_parse_error("Ollama", &e))?;
889        Self::response_from_chat_payload(model, parsed)
890    }
891
892    fn extract_error(body: &str) -> Option<String> {
893        serde_json::from_str::<OllamaErrorResponse>(body)
894            .ok()
895            .and_then(|resp| resp.error)
896    }
897}
898
899#[derive(Debug, Serialize)]
900struct OllamaChatRequest {
901    model: String,
902    messages: Vec<OllamaChatMessage>,
903    stream: bool,
904    #[serde(skip_serializing_if = "Option::is_none")]
905    format: Option<Value>,
906    #[serde(skip_serializing_if = "Option::is_none")]
907    options: Option<OllamaChatOptions>,
908    #[serde(skip_serializing_if = "Option::is_none")]
909    tools: Option<Vec<ToolDefinition>>,
910    #[serde(skip_serializing_if = "Option::is_none")]
911    think: Option<Value>,
912}
913
914#[derive(Debug, Serialize)]
915struct OllamaChatMessage {
916    role: String,
917    #[serde(skip_serializing_if = "Option::is_none")]
918    content: Option<String>,
919    #[serde(skip_serializing_if = "Option::is_none")]
920    thinking: Option<String>,
921    #[serde(skip_serializing_if = "Option::is_none")]
922    images: Option<Vec<String>>,
923    #[serde(skip_serializing_if = "Option::is_none")]
924    tool_calls: Option<Vec<OllamaToolCall>>,
925    #[serde(skip_serializing_if = "Option::is_none")]
926    tool_call_id: Option<String>,
927    #[serde(skip_serializing_if = "Option::is_none")]
928    tool_name: Option<String>,
929}
930
931#[derive(Debug, Serialize)]
932struct OllamaChatOptions {
933    #[serde(skip_serializing_if = "Option::is_none")]
934    temperature: Option<f32>,
935    #[serde(skip_serializing_if = "Option::is_none")]
936    num_predict: Option<u32>,
937}
938
939#[derive(Debug, Serialize)]
940struct OllamaToolCall {
941    #[serde(rename = "type")]
942    call_type: String,
943    function: OllamaToolFunctionCall,
944}
945
946#[derive(Debug, Serialize)]
947struct OllamaToolFunctionCall {
948    name: String,
949    #[serde(skip_serializing_if = "Option::is_none")]
950    arguments: Option<Value>,
951    #[serde(skip_serializing_if = "Option::is_none")]
952    index: Option<u32>,
953}
954
955#[derive(Debug, Deserialize)]
956struct OllamaChatResponse {
957    message: Option<OllamaResponseMessage>,
958    #[serde(default)]
959    done: bool,
960    #[serde(default)]
961    done_reason: Option<String>,
962    #[serde(default)]
963    prompt_eval_count: Option<u32>,
964    #[serde(default)]
965    eval_count: Option<u32>,
966    #[serde(default)]
967    error: Option<String>,
968}
969
970#[derive(Debug, Deserialize)]
971struct OllamaResponseMessage {
972    #[serde(default)]
973    #[expect(dead_code)]
974    role: Option<String>,
975    #[serde(default)]
976    content: Option<String>,
977    #[serde(default)]
978    thinking: Option<String>,
979    #[serde(default)]
980    reasoning_details: Option<Vec<Value>>,
981    #[serde(default)]
982    tool_calls: Option<Vec<OllamaResponseToolCall>>,
983}
984
985#[derive(Debug, Deserialize, Serialize, Clone)]
986struct OllamaResponseToolCall {
987    #[serde(default)]
988    #[serde(rename = "type")]
989    call_type: Option<String>,
990    #[serde(default)]
991    function: Option<OllamaResponseFunctionCall>,
992}
993
994#[derive(Debug, Deserialize, Serialize, Clone)]
995struct OllamaResponseFunctionCall {
996    #[serde(default)]
997    name: Option<String>,
998    #[serde(default)]
999    arguments: Option<Value>,
1000    #[serde(default)]
1001    index: Option<u32>,
1002}
1003
1004#[derive(Debug, Deserialize)]
1005struct OllamaErrorResponse {
1006    error: Option<String>,
1007}
1008
1009fn parse_stream_chunk(line: &str) -> Result<OllamaChatResponse, LLMError> {
1010    serde_json::from_str::<OllamaChatResponse>(line).map_err(|err| LLMError::Provider {
1011        message: format!("Failed to parse Ollama stream chunk: {err}"),
1012        metadata: None,
1013    })
1014}
1015
1016#[async_trait]
1017impl LLMProvider for OllamaProvider {
1018    fn name(&self) -> &str {
1019        "ollama"
1020    }
1021
1022    fn supports_streaming(&self) -> bool {
1023        true
1024    }
1025
1026    fn supports_tools(&self, _model: &str) -> bool {
1027        true
1028    }
1029
1030    fn supports_reasoning(&self, model: &str) -> bool {
1031        // Codex-inspired robustness: Setting model_supports_reasoning to false
1032        // does NOT disable it for known reasoning models.
1033        models::ollama::REASONING_MODELS.contains(&model)
1034            || self
1035                .model_behavior
1036                .as_ref()
1037                .and_then(|b| b.model_supports_reasoning)
1038                .unwrap_or(false)
1039    }
1040
1041    fn supports_reasoning_effort(&self, model: &str) -> bool {
1042        // Same robustness logic for reasoning effort
1043        models::ollama::REASONING_LEVEL_MODELS.contains(&model)
1044            || self
1045                .model_behavior
1046                .as_ref()
1047                .and_then(|b| b.model_supports_reasoning_effort)
1048                .unwrap_or(false)
1049    }
1050
1051    async fn generate(&self, mut request: LLMRequest) -> Result<LLMResponse, LLMError> {
1052        self.validate_request(&request)?;
1053        if request.model.is_empty() {
1054            request.model = self.model.clone();
1055        }
1056        let model = request.model.clone();
1057        let payload = self.build_payload(&request, false)?;
1058        let url = self.chat_url();
1059        Self::request_non_stream_response(
1060            &self.http_client,
1061            &url,
1062            self.api_key.as_deref(),
1063            &payload,
1064            model,
1065        )
1066        .await
1067    }
1068
1069    async fn stream(&self, mut request: LLMRequest) -> Result<LLMStream, LLMError> {
1070        self.validate_request(&request)?;
1071        if request.model.is_empty() {
1072            request.model = self.model.clone();
1073        }
1074        let model = request.model.clone();
1075        let payload = self.build_payload(&request, true)?;
1076        let fallback_payload = self.build_payload(&request, false)?;
1077        let url = self.chat_url();
1078
1079        let response = self
1080            .authorized_post(url.clone())
1081            .header(reqwest::header::ACCEPT_ENCODING, "identity")
1082            .json(&payload)
1083            .send()
1084            .await
1085            .map_err(|e| format_network_error("Ollama", &e))?;
1086
1087        if !response.status().is_success() {
1088            let status = response.status();
1089            let body = response.text().await.unwrap_or_default();
1090            let error_message = Self::extract_error(&body)
1091                .unwrap_or_else(|| format!("Ollama streaming request failed ({status}): {body}"));
1092            return Err(LLMError::Provider {
1093                message: error_message,
1094                metadata: None,
1095            });
1096        }
1097
1098        let byte_stream = response.bytes_stream();
1099        let mut buffer: Vec<u8> = Vec::new();
1100        let mut aggregator = crate::llm::providers::shared::StreamAggregator::new(model.clone());
1101        let fallback_http_client = self.http_client.clone();
1102        let fallback_api_key = self.api_key.clone();
1103        let fallback_model = model.clone();
1104        let fallback_url = url.clone();
1105        let any_interleaved = request
1106            .messages
1107            .iter()
1108            .any(|msg| assistant_interleaved_history_text(msg, &request.model).is_some());
1109        let stream = try_stream! {
1110            let mut prompt_tokens: Option<u32> = None;
1111            let mut completion_tokens: Option<u32> = None;
1112            let mut finish_reason: Option<String> = None;
1113            let mut completed = false;
1114            let mut saw_stream_chunk = false;
1115
1116            futures::pin_mut!(byte_stream);
1117            while let Some(chunk_result) = byte_stream.next().await {
1118                let chunk = match chunk_result {
1119                    Ok(chunk) => {
1120                        saw_stream_chunk = true;
1121                        chunk
1122                    }
1123                    Err(err) if !saw_stream_chunk => {
1124                        tracing::warn!(
1125                            model = %fallback_model,
1126                            url = %fallback_url,
1127                            error = %err,
1128                            "Ollama stream failed before first chunk; retrying once as non-stream response"
1129                        );
1130                        let fallback_response = Self::request_non_stream_response(
1131                            &fallback_http_client,
1132                            &fallback_url,
1133                            fallback_api_key.as_deref(),
1134                            &fallback_payload,
1135                            fallback_model.clone(),
1136                        ).await?;
1137                        yield LLMStreamEvent::Completed { response: Box::new(fallback_response) };
1138                        return;
1139                    }
1140                    Err(err) => Err(format_network_error("Ollama", &err))?,
1141                };
1142                buffer.extend_from_slice(&chunk);
1143
1144                while let Some(pos) = buffer.iter().position(|b| *b == b'\n') {
1145                    let line_bytes: Vec<u8> = buffer.drain(..=pos).collect();
1146                    let line = std::str::from_utf8(&line_bytes)
1147                        .map_err(|err| LLMError::Provider {
1148                            message: format!("Invalid UTF-8 in Ollama stream: {err}"),
1149                            metadata: None,
1150                        })?;
1151                    let line = line.trim();
1152
1153                    if line.is_empty() {
1154                        continue;
1155                    }
1156
1157                    let parsed = parse_stream_chunk(line)?;
1158
1159                    if let Some(error) = parsed.error {
1160                        Err(LLMError::Provider {
1161                            message: error,
1162                            metadata: None,
1163                        })?;
1164                    }
1165
1166                    if let Some(message) = parsed.message {
1167                        if let Some(reasoning_details) = message.reasoning_details.as_deref() {
1168                            aggregator.set_reasoning_details(reasoning_details);
1169                        }
1170
1171                        let has_explicit_thinking = message
1172                            .thinking
1173                            .as_ref()
1174                            .map(|v| !v.is_empty())
1175                            .unwrap_or(false);
1176
1177                        if let Some(thinking) = message.thinking
1178                            && let Some(delta) = aggregator.handle_reasoning(&thinking) {
1179                                yield LLMStreamEvent::Reasoning { delta };
1180                            }
1181
1182                        if let Some(content) = message.content {
1183                            for event in aggregator.handle_content(&content) {
1184                                match &event {
1185                                    LLMStreamEvent::Reasoning { .. }
1186                                        if has_explicit_thinking || any_interleaved =>
1187                                    {
1188                                    }
1189                                    _ => yield event,
1190                                }
1191                            }
1192                        }
1193
1194                        if let Some(tool_calls) = message.tool_calls {
1195                            let tool_calls_json: Vec<Value> = tool_calls
1196                                .into_iter()
1197                                .map(|tc| serde_json::to_value(tc).unwrap_or(Value::Null))
1198                                .filter(|v| !v.is_null())
1199                                .collect();
1200                            aggregator.handle_tool_calls(&tool_calls_json);
1201                        }
1202                    }
1203
1204                    if parsed.done {
1205                        prompt_tokens = parsed.prompt_eval_count;
1206                        completion_tokens = parsed.eval_count;
1207                        finish_reason = parsed.done_reason;
1208                        completed = true;
1209                    }
1210                }
1211
1212                if completed {
1213                    break;
1214                }
1215            }
1216
1217            if !completed {
1218                Err(LLMError::Provider {
1219                    message: "Ollama stream ended without completion signal".to_string(),
1220                    metadata: None,
1221                })?;
1222            }
1223
1224            let mut response = aggregator.finalize();
1225            if let Some(pt) = prompt_tokens {
1226                let mut usage = response.usage.unwrap_or_default();
1227                usage.prompt_tokens = pt;
1228                if let Some(ct) = completion_tokens {
1229                    usage.completion_tokens = ct;
1230                    usage.total_tokens = pt + ct;
1231                }
1232                response.usage = Some(usage);
1233            }
1234            if let Some(fr) = finish_reason {
1235                response.finish_reason = crate::llm::providers::common::map_finish_reason_common(&fr);
1236            }
1237            if response.reasoning.is_none()
1238                && let Some(details) = response.reasoning_details.as_ref()
1239            {
1240                response.reasoning = extract_reasoning_text_from_serialized_details(details);
1241            }
1242
1243            yield LLMStreamEvent::Completed { response: Box::new(response) };
1244        };
1245
1246        Ok(Box::pin(stream))
1247    }
1248
1249    fn supported_models(&self) -> Vec<String> {
1250        models::ollama::SUPPORTED_MODELS
1251            .iter()
1252            .map(|model| model.to_string())
1253            .collect()
1254    }
1255
1256    fn validate_request(&self, request: &LLMRequest) -> Result<(), LLMError> {
1257        if let Some(tool_choice) = &request.tool_choice {
1258            match tool_choice {
1259                ToolChoice::Auto | ToolChoice::None => {}
1260                _ => {
1261                    return Err(LLMError::InvalidRequest {
1262                        message: "Ollama does not support explicit tool_choice overrides"
1263                            .to_string(),
1264                        metadata: None,
1265                    });
1266                }
1267            }
1268        }
1269
1270        if request.parallel_tool_calls.is_some() || request.parallel_tool_config.is_some() {
1271            return Err(LLMError::InvalidRequest {
1272                message: "Ollama does not support parallel tool configuration".to_string(),
1273                metadata: None,
1274            });
1275        }
1276
1277        for message in &request.messages {
1278            if matches!(message.role, MessageRole::Tool) && message.tool_call_id.is_none() {
1279                return Err(LLMError::InvalidRequest {
1280                    message: "Ollama tool responses must include tool_call_id".to_string(),
1281                    metadata: None,
1282                });
1283            }
1284        }
1285
1286        Ok(())
1287    }
1288}
1289
1290#[async_trait]
1291impl LLMClient for OllamaProvider {
1292    async fn generate(&mut self, prompt: &str) -> Result<LLMResponse, LLMError> {
1293        let mut request = self.parse_client_prompt(prompt);
1294        if request.model.is_empty() {
1295            request.model = self.model.clone();
1296        }
1297        Ok(LLMProvider::generate(self, request).await?)
1298    }
1299
1300    fn model_id(&self) -> &str {
1301        &self.model
1302    }
1303}
1304
1305#[cfg(test)]
1306mod tests {
1307    use super::*;
1308    use crate::config::types::ReasoningEffortLevel;
1309    use crate::llm::provider::{ContentPart, Message, MessageContent};
1310    use serde_json::json;
1311
1312    fn test_provider() -> OllamaProvider {
1313        OllamaProvider::from_config(
1314            None,
1315            Some("test-model".to_string()),
1316            Some("http://localhost".to_string()),
1317            None,
1318            None,
1319            None,
1320            None,
1321        )
1322    }
1323
1324    #[test]
1325    fn build_payload_includes_images() {
1326        let provider = test_provider();
1327        let parts = vec![
1328            ContentPart::text("see ".to_string()),
1329            ContentPart::image("BASE64DATA".to_string(), "image/png".to_string()),
1330        ];
1331        let request = LLMRequest {
1332            model: "test-model".to_string(),
1333            messages: vec![Message::user_with_parts(parts)],
1334            ..Default::default()
1335        };
1336
1337        let payload = provider.build_payload(&request, false).unwrap();
1338        assert_eq!(payload.messages.len(), 1);
1339        let message = &payload.messages[0];
1340        assert_eq!(message.content.as_deref(), Some("see "));
1341        assert_eq!(
1342            message.images.as_ref(),
1343            Some(&vec!["BASE64DATA".to_string()])
1344        );
1345    }
1346
1347    #[test]
1348    fn build_payload_omits_images_when_none_present() {
1349        let provider = test_provider();
1350        let content = MessageContent::text("no images".to_string());
1351        let request = LLMRequest {
1352            model: "test-model".to_string(),
1353            messages: vec![Message::user(content.as_text().into_owned())],
1354            ..Default::default()
1355        };
1356
1357        let payload = provider.build_payload(&request, false).unwrap();
1358        assert_eq!(payload.messages.len(), 1);
1359        let message = &payload.messages[0];
1360        assert_eq!(message.content.as_deref(), Some("no images"));
1361        assert!(message.images.is_none());
1362    }
1363
1364    #[test]
1365    fn build_payload_minimax_tool_followup_omits_tool_call_id() {
1366        let provider = test_provider();
1367        let tool_call_id = "direct_run_pty_cmd_1".to_string();
1368        let request = LLMRequest {
1369            model: models::ollama::MINIMAX_M25_CLOUD.to_string(),
1370            messages: vec![
1371                Message::assistant_with_tools(
1372                    String::new(),
1373                    vec![ToolCall::function(
1374                        tool_call_id.clone(),
1375                        "run_pty_cmd".to_string(),
1376                        "{\"command\":\"cargo fmt\"}".to_string(),
1377                    )],
1378                ),
1379                Message::tool_response(
1380                    tool_call_id,
1381                    "{\"output\":\"\",\"exit_code\":0}".to_string(),
1382                ),
1383            ],
1384            reasoning_effort: Some(ReasoningEffortLevel::Low),
1385            ..Default::default()
1386        };
1387
1388        let payload = provider.build_payload(&request, false).unwrap();
1389        assert_eq!(payload.messages.len(), 2);
1390        assert_eq!(payload.messages[1].role, "tool");
1391        assert_eq!(
1392            payload.messages[1].tool_name.as_deref(),
1393            Some("run_pty_cmd")
1394        );
1395        assert!(payload.messages[1].tool_call_id.is_none());
1396        assert!(payload.think.is_none());
1397    }
1398
1399    #[test]
1400    fn build_payload_non_minimax_tool_followup_keeps_tool_call_id() {
1401        let provider = test_provider();
1402        let tool_call_id = "direct_run_pty_cmd_1".to_string();
1403        let request = LLMRequest {
1404            model: models::ollama::GPT_OSS_20B_CLOUD.to_string(),
1405            messages: vec![
1406                Message::assistant_with_tools(
1407                    String::new(),
1408                    vec![ToolCall::function(
1409                        tool_call_id.clone(),
1410                        "run_pty_cmd".to_string(),
1411                        "{\"command\":\"cargo fmt\"}".to_string(),
1412                    )],
1413                ),
1414                Message::tool_response(
1415                    tool_call_id.clone(),
1416                    "{\"output\":\"\",\"exit_code\":0}".to_string(),
1417                ),
1418            ],
1419            reasoning_effort: Some(ReasoningEffortLevel::Low),
1420            ..Default::default()
1421        };
1422
1423        let payload = provider.build_payload(&request, false).unwrap();
1424        assert_eq!(payload.messages.len(), 2);
1425        assert_eq!(payload.messages[1].role, "tool");
1426        assert_eq!(
1427            payload.messages[1].tool_name.as_deref(),
1428            Some("run_pty_cmd")
1429        );
1430        assert_eq!(
1431            payload.messages[1].tool_call_id.as_deref(),
1432            Some(tool_call_id.as_str())
1433        );
1434        assert_eq!(payload.think, Some(Value::String("low".to_string())));
1435    }
1436
1437    #[test]
1438    fn build_payload_hoists_history_system_directives_into_system_prompt() {
1439        let provider = test_provider();
1440        let request = LLMRequest {
1441            model: models::ollama::MINIMAX_M25_CLOUD.to_string(),
1442            system_prompt: Some(std::sync::Arc::new(
1443                "stable system instructions".to_string(),
1444            )),
1445            messages: vec![
1446                Message::user("explore architecture".to_string()),
1447                Message::system(
1448                    "Previous turn already completed tool execution. Reuse the latest tool outputs in history instead of rerunning the same exploration.".to_string(),
1449                ),
1450            ],
1451            ..Default::default()
1452        };
1453
1454        let payload = provider.build_payload(&request, false).unwrap();
1455        assert_eq!(payload.messages.len(), 2);
1456        assert_eq!(payload.messages[0].role, "system");
1457        assert!(
1458            payload.messages[0]
1459                .content
1460                .as_deref()
1461                .unwrap_or("")
1462                .contains("stable system instructions")
1463        );
1464        assert!(
1465            payload.messages[0]
1466                .content
1467                .as_deref()
1468                .unwrap_or("")
1469                .contains("[History Directives]")
1470        );
1471        assert!(
1472            payload.messages[0]
1473                .content
1474                .as_deref()
1475                .unwrap_or("")
1476                .contains("Previous turn already completed tool execution")
1477        );
1478        assert_eq!(payload.messages[1].role, "user");
1479        assert_eq!(
1480            payload.messages[1].content.as_deref(),
1481            Some("explore architecture")
1482        );
1483    }
1484
1485    #[test]
1486    fn build_payload_promotes_history_system_directive_without_base_system_prompt() {
1487        let provider = test_provider();
1488        let request = LLMRequest {
1489            model: models::ollama::MINIMAX_M25_CLOUD.to_string(),
1490            messages: vec![
1491                Message::system(
1492                    "Repeated read-only exploration hit the per-turn family cap. Scheduling a final recovery pass without more tools.".to_string(),
1493                ),
1494                Message::user("summarize the architecture".to_string()),
1495            ],
1496            ..Default::default()
1497        };
1498
1499        let payload = provider.build_payload(&request, false).unwrap();
1500        assert_eq!(payload.messages.len(), 2);
1501        assert_eq!(payload.messages[0].role, "system");
1502        assert!(
1503            payload.messages[0]
1504                .content
1505                .as_deref()
1506                .unwrap_or("")
1507                .contains("[History Directives]")
1508        );
1509        assert!(
1510            payload.messages[0]
1511                .content
1512                .as_deref()
1513                .unwrap_or("")
1514                .contains("Repeated read-only exploration hit the per-turn family cap")
1515        );
1516        assert_eq!(payload.messages[1].role, "user");
1517    }
1518
1519    #[test]
1520    fn build_payload_recovers_balanced_prefix_from_malformed_history_tool_arguments() {
1521        let provider = test_provider();
1522        let request = LLMRequest {
1523            model: "test-model".to_string(),
1524            messages: vec![Message::assistant_with_tools(
1525                String::new(),
1526                vec![ToolCall::function(
1527                    "tool_call_0".to_string(),
1528                    "unified_file".to_string(),
1529                    "{\"action\":\"read\",\"path\":\"docs/ARCHITECTURE.md\",\"offset\":1,\"limit\":100}{\"action\":\"read\",\"path\":\"README.md\"}"
1530                        .to_string(),
1531                )],
1532            )],
1533            ..Default::default()
1534        };
1535
1536        let payload = provider
1537            .build_payload(&request, false)
1538            .expect("payload should recover malformed history tool arguments");
1539
1540        let tool_calls = payload.messages[0]
1541            .tool_calls
1542            .as_ref()
1543            .expect("tool calls should be present");
1544        assert_eq!(tool_calls.len(), 1);
1545        assert_eq!(
1546            tool_calls[0].function.arguments,
1547            Some(json!({
1548                "action": "read",
1549                "path": "docs/ARCHITECTURE.md",
1550                "offset": 1,
1551                "limit": 100
1552            }))
1553        );
1554    }
1555
1556    #[test]
1557    fn build_payload_rehydrates_glm_interleaved_history_into_content() {
1558        let provider = test_provider();
1559        let request = LLMRequest {
1560            model: models::ollama::GLM_5_CLOUD.to_string(),
1561            messages: vec![
1562                Message::assistant("done".to_string()).with_reasoning(Some("trace".to_string())),
1563            ],
1564            ..Default::default()
1565        };
1566
1567        let payload = provider.build_payload(&request, false).unwrap();
1568
1569        assert_eq!(
1570            payload.messages[0].content.as_deref(),
1571            Some("<think>trace</think>done")
1572        );
1573        assert!(payload.messages[0].thinking.is_none());
1574    }
1575
1576    #[test]
1577    fn build_payload_replays_assistant_reasoning_as_ollama_thinking() {
1578        let provider = test_provider();
1579        let request = LLMRequest {
1580            model: models::ollama::GPT_OSS_20B.to_string(),
1581            messages: vec![
1582                Message::assistant("need a tool".to_string())
1583                    .with_reasoning(Some("reasoning trace".to_string())),
1584            ],
1585            ..Default::default()
1586        };
1587
1588        let payload = provider.build_payload(&request, false).unwrap();
1589
1590        assert_eq!(payload.messages[0].content.as_deref(), Some("need a tool"));
1591        assert_eq!(
1592            payload.messages[0].thinking.as_deref(),
1593            Some("reasoning trace")
1594        );
1595    }
1596
1597    #[test]
1598    fn build_payload_includes_apply_patch_as_normal_tool() {
1599        let provider = test_provider();
1600        let request = LLMRequest {
1601            model: "test-model".to_string(),
1602            messages: vec![Message::user("patch this file".to_string())],
1603            tools: Some(std::sync::Arc::new(vec![ToolDefinition::apply_patch(
1604                "Apply VT Code patches".to_string(),
1605            )])),
1606            ..Default::default()
1607        };
1608
1609        let payload = provider.build_payload(&request, false).unwrap();
1610        let tools = payload.tools.expect("tools should be present");
1611        assert_eq!(tools.len(), 1);
1612        assert_eq!(tools[0].function_name(), "apply_patch");
1613    }
1614
1615    #[test]
1616    fn response_payload_preserves_reasoning_details() {
1617        let parsed = OllamaChatResponse {
1618            message: Some(OllamaResponseMessage {
1619                role: Some("assistant".to_string()),
1620                content: Some("answer".to_string()),
1621                thinking: None,
1622                reasoning_details: Some(vec![json!({
1623                    "type": "reasoning.text",
1624                    "text": "step one"
1625                })]),
1626                tool_calls: None,
1627            }),
1628            done: true,
1629            done_reason: Some("stop".to_string()),
1630            prompt_eval_count: Some(1),
1631            eval_count: Some(2),
1632            error: None,
1633        };
1634
1635        let response = OllamaProvider::response_from_chat_payload("test-model".to_string(), parsed)
1636            .expect("response should parse");
1637        assert_eq!(response.reasoning.as_deref(), Some("step one"));
1638        assert!(response.reasoning_details.is_some());
1639
1640        let first_detail = response
1641            .reasoning_details
1642            .as_ref()
1643            .and_then(|details| details.first())
1644            .expect("reasoning detail should exist");
1645        let parsed_detail: Value =
1646            serde_json::from_str(first_detail).expect("reasoning detail should be json");
1647        assert_eq!(parsed_detail["type"], "reasoning.text");
1648    }
1649
1650    #[test]
1651    fn tags_response_accepts_partial_model_summaries() {
1652        let parsed: OllamaTagsResponse = serde_json::from_value(json!({
1653            "models": [
1654                { "model": "qwen3:8b" }
1655            ]
1656        }))
1657        .expect("partial model summaries should parse");
1658
1659        let names: Vec<String> = parsed
1660            .models
1661            .into_iter()
1662            .filter_map(|model| model.name.or(model.model))
1663            .collect();
1664        assert_eq!(names, vec!["qwen3:8b".to_string()]);
1665    }
1666
1667    #[test]
1668    fn wire_api_responses_for_dev_build() {
1669        assert_eq!(
1670            wire_api_for_version(&Version::new(0, 0, 0)),
1671            OllamaWireApi::Responses,
1672        );
1673    }
1674
1675    #[test]
1676    fn wire_api_responses_for_exact_threshold() {
1677        assert_eq!(
1678            wire_api_for_version(&Version::new(0, 13, 3)),
1679            OllamaWireApi::Responses,
1680        );
1681    }
1682
1683    #[test]
1684    fn wire_api_responses_for_above_threshold() {
1685        assert_eq!(
1686            wire_api_for_version(&Version::new(0, 14, 0)),
1687            OllamaWireApi::Responses,
1688        );
1689        assert_eq!(
1690            wire_api_for_version(&Version::new(1, 0, 0)),
1691            OllamaWireApi::Responses,
1692        );
1693    }
1694
1695    #[test]
1696    fn wire_api_chat_for_below_threshold() {
1697        assert_eq!(
1698            wire_api_for_version(&Version::new(0, 13, 2)),
1699            OllamaWireApi::Chat,
1700        );
1701        assert_eq!(
1702            wire_api_for_version(&Version::new(0, 12, 0)),
1703            OllamaWireApi::Chat,
1704        );
1705        assert_eq!(
1706            wire_api_for_version(&Version::new(0, 1, 0)),
1707            OllamaWireApi::Chat,
1708        );
1709    }
1710}