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#[derive(Debug, Clone, Copy, PartialEq, Eq)]
49pub enum OllamaWireApi {
50 Responses,
52 Chat,
54}
55
56pub struct WireApiDetection {
58 pub wire_api: OllamaWireApi,
59 pub version: Option<Version>,
60}
61
62static 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
69fn 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
81pub 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
115pub 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 let ollama_client = OllamaClient::try_from_base_url(&resolved_base_url).await?;
135
136 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 }
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
192pub 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 let tags_url = format!("{}/api/tags", resolved_base_url);
204
205 let client = http_client::create_client_with_timeout(std::time::Duration::from_secs(5));
207
208 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 let tags_response: OllamaTagsResponse = response
233 .json()
234 .await
235 .map_err(|e| anyhow::anyhow!("Failed to parse Ollama models response: {}", e))?;
236
237 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 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 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 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 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 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}