1use anyhow::{Context as _, Result, anyhow};
2use futures::{AsyncBufReadExt, AsyncReadExt, StreamExt, io::BufReader, stream::BoxStream};
3use http_client::{AsyncBody, HttpClient, Method, Request as HttpRequest, http};
4use serde::{Deserialize, Serialize};
5use serde_json::Value;
6use std::{convert::TryFrom, time::Duration};
7
8pub const LMSTUDIO_API_URL: &str = "http://localhost:1234/api/v0";
9
10#[derive(Clone, Copy, Serialize, Deserialize, Debug, Eq, PartialEq)]
11#[serde(rename_all = "lowercase")]
12pub enum Role {
13 User,
14 Assistant,
15 System,
16 Tool,
17}
18
19impl TryFrom<String> for Role {
20 type Error = anyhow::Error;
21
22 fn try_from(value: String) -> Result<Self> {
23 match value.as_str() {
24 "user" => Ok(Self::User),
25 "assistant" => Ok(Self::Assistant),
26 "system" => Ok(Self::System),
27 "tool" => Ok(Self::Tool),
28 _ => anyhow::bail!("invalid role '{value}'"),
29 }
30 }
31}
32
33impl From<Role> for String {
34 fn from(val: Role) -> Self {
35 match val {
36 Role::User => "user".to_owned(),
37 Role::Assistant => "assistant".to_owned(),
38 Role::System => "system".to_owned(),
39 Role::Tool => "tool".to_owned(),
40 }
41 }
42}
43
44#[cfg_attr(feature = "schemars", derive(schemars::JsonSchema))]
45#[derive(Clone, Debug, Default, Serialize, Deserialize, PartialEq)]
46pub struct Model {
47 pub name: String,
48 pub display_name: Option<String>,
49 pub max_tokens: u64,
50 pub supports_tool_calls: bool,
51 pub supports_images: bool,
52}
53
54impl Model {
55 pub fn new(
56 name: &str,
57 display_name: Option<&str>,
58 max_tokens: Option<u64>,
59 supports_tool_calls: bool,
60 supports_images: bool,
61 ) -> Self {
62 Self {
63 name: name.to_owned(),
64 display_name: display_name.map(|s| s.to_owned()),
65 max_tokens: max_tokens.unwrap_or(2048),
66 supports_tool_calls,
67 supports_images,
68 }
69 }
70
71 pub fn id(&self) -> &str {
72 &self.name
73 }
74
75 pub fn display_name(&self) -> &str {
76 self.display_name.as_ref().unwrap_or(&self.name)
77 }
78
79 pub fn max_token_count(&self) -> u64 {
80 self.max_tokens
81 }
82
83 pub fn supports_tool_calls(&self) -> bool {
84 self.supports_tool_calls
85 }
86}
87
88#[derive(Debug, Serialize, Deserialize)]
89#[serde(untagged)]
90pub enum ToolChoice {
91 Auto,
92 Required,
93 None,
94 Other(ToolDefinition),
95}
96
97#[derive(Clone, Deserialize, Serialize, Debug)]
98#[serde(tag = "type", rename_all = "snake_case")]
99pub enum ToolDefinition {
100 #[allow(dead_code)]
101 Function { function: FunctionDefinition },
102}
103
104#[derive(Clone, Debug, Serialize, Deserialize)]
105pub struct FunctionDefinition {
106 pub name: String,
107 pub description: Option<String>,
108 pub parameters: Option<Value>,
109}
110
111#[derive(Serialize, Deserialize, Debug)]
112#[serde(tag = "role", rename_all = "lowercase")]
113pub enum ChatMessage {
114 Assistant {
115 #[serde(default)]
116 content: Option<MessageContent>,
117 #[serde(default, skip_serializing_if = "Vec::is_empty")]
118 tool_calls: Vec<ToolCall>,
119 },
120 User {
121 content: MessageContent,
122 },
123 System {
124 content: MessageContent,
125 },
126 Tool {
127 content: MessageContent,
128 tool_call_id: String,
129 },
130}
131
132#[derive(Serialize, Deserialize, Debug, Eq, PartialEq)]
133#[serde(untagged)]
134pub enum MessageContent {
135 Plain(String),
136 Multipart(Vec<MessagePart>),
137}
138
139impl MessageContent {
140 pub fn empty() -> Self {
141 MessageContent::Multipart(vec![])
142 }
143
144 pub fn push_part(&mut self, part: MessagePart) {
145 match self {
146 MessageContent::Plain(text) => {
147 *self =
148 MessageContent::Multipart(vec![MessagePart::Text { text: text.clone() }, part]);
149 }
150 MessageContent::Multipart(parts) if parts.is_empty() => match part {
151 MessagePart::Text { text } => *self = MessageContent::Plain(text),
152 MessagePart::Image { .. } => *self = MessageContent::Multipart(vec![part]),
153 },
154 MessageContent::Multipart(parts) => parts.push(part),
155 }
156 }
157}
158
159impl From<Vec<MessagePart>> for MessageContent {
160 fn from(mut parts: Vec<MessagePart>) -> Self {
161 if let [MessagePart::Text { text }] = parts.as_mut_slice() {
162 MessageContent::Plain(std::mem::take(text))
163 } else {
164 MessageContent::Multipart(parts)
165 }
166 }
167}
168
169#[derive(Serialize, Deserialize, Debug, Eq, PartialEq)]
170#[serde(tag = "type", rename_all = "snake_case")]
171pub enum MessagePart {
172 Text {
173 text: String,
174 },
175 #[serde(rename = "image_url")]
176 Image {
177 image_url: ImageUrl,
178 },
179}
180
181#[derive(Serialize, Deserialize, Clone, Debug, Eq, PartialEq)]
182pub struct ImageUrl {
183 pub url: String,
184 #[serde(skip_serializing_if = "Option::is_none")]
185 pub detail: Option<String>,
186}
187
188#[derive(Serialize, Deserialize, Debug, Eq, PartialEq)]
189pub struct ToolCall {
190 pub id: String,
191 #[serde(flatten)]
192 pub content: ToolCallContent,
193}
194
195#[derive(Serialize, Deserialize, Debug, Eq, PartialEq)]
196#[serde(tag = "type", rename_all = "lowercase")]
197pub enum ToolCallContent {
198 Function { function: FunctionContent },
199}
200
201#[derive(Serialize, Deserialize, Debug, Eq, PartialEq)]
202pub struct FunctionContent {
203 pub name: String,
204 pub arguments: String,
205}
206
207#[derive(Serialize, Debug)]
208pub struct ChatCompletionRequest {
209 pub model: String,
210 pub messages: Vec<ChatMessage>,
211 pub stream: bool,
212 #[serde(skip_serializing_if = "Option::is_none")]
213 pub max_tokens: Option<i32>,
214 #[serde(skip_serializing_if = "Option::is_none")]
215 pub stop: Option<Vec<String>>,
216 #[serde(skip_serializing_if = "Option::is_none")]
217 pub temperature: Option<f32>,
218 #[serde(skip_serializing_if = "Vec::is_empty")]
219 pub tools: Vec<ToolDefinition>,
220 #[serde(skip_serializing_if = "Option::is_none")]
221 pub tool_choice: Option<ToolChoice>,
222}
223
224#[derive(Serialize, Deserialize, Debug)]
225pub struct ChatResponse {
226 pub id: String,
227 pub object: String,
228 pub created: u64,
229 pub model: String,
230 pub choices: Vec<ChoiceDelta>,
231}
232
233#[derive(Serialize, Deserialize, Debug)]
234pub struct ChoiceDelta {
235 pub index: u32,
236 pub delta: ResponseMessageDelta,
237 pub finish_reason: Option<String>,
238}
239
240#[derive(Serialize, Deserialize, Debug, Eq, PartialEq)]
241pub struct ToolCallChunk {
242 pub index: usize,
243 pub id: Option<String>,
244
245 // There is also an optional `type` field that would determine if a
246 // function is there. Sometimes this streams in with the `function` before
247 // it streams in the `type`
248 pub function: Option<FunctionChunk>,
249}
250
251#[derive(Serialize, Deserialize, Debug, Eq, PartialEq)]
252pub struct FunctionChunk {
253 pub name: Option<String>,
254 pub arguments: Option<String>,
255}
256
257#[derive(Serialize, Deserialize, Debug)]
258pub struct Usage {
259 pub prompt_tokens: u64,
260 pub completion_tokens: u64,
261 pub total_tokens: u64,
262}
263
264#[derive(Debug, Default, Clone, Deserialize, PartialEq)]
265#[serde(transparent)]
266pub struct Capabilities(Vec<String>);
267
268impl Capabilities {
269 pub fn supports_tool_calls(&self) -> bool {
270 self.0.iter().any(|cap| cap == "tool_use")
271 }
272
273 pub fn supports_images(&self) -> bool {
274 self.0.iter().any(|cap| cap == "vision")
275 }
276}
277
278#[derive(Serialize, Deserialize, Debug)]
279pub struct LmStudioError {
280 pub message: String,
281}
282
283#[derive(Serialize, Deserialize, Debug)]
284#[serde(untagged)]
285pub enum ResponseStreamResult {
286 Ok(ResponseStreamEvent),
287 Err { error: LmStudioError },
288}
289
290#[derive(Serialize, Deserialize, Debug)]
291pub struct ResponseStreamEvent {
292 pub created: u32,
293 pub model: String,
294 pub object: String,
295 pub choices: Vec<ChoiceDelta>,
296 pub usage: Option<Usage>,
297}
298
299#[derive(Deserialize)]
300pub struct ListModelsResponse {
301 pub data: Vec<ModelEntry>,
302}
303
304#[derive(Clone, Debug, Deserialize, PartialEq)]
305pub struct ModelEntry {
306 pub id: String,
307 pub object: String,
308 pub r#type: ModelType,
309 pub publisher: String,
310 pub arch: Option<String>,
311 pub compatibility_type: CompatibilityType,
312 pub quantization: Option<String>,
313 pub state: ModelState,
314 pub max_context_length: Option<u64>,
315 pub loaded_context_length: Option<u64>,
316 #[serde(default)]
317 pub capabilities: Capabilities,
318}
319
320#[derive(Clone, Debug, Serialize, Deserialize, PartialEq)]
321#[serde(rename_all = "lowercase")]
322pub enum ModelType {
323 Llm,
324 Embeddings,
325 Vlm,
326}
327
328#[derive(Clone, Debug, Serialize, Deserialize, PartialEq)]
329#[serde(rename_all = "kebab-case")]
330pub enum ModelState {
331 Loaded,
332 Loading,
333 NotLoaded,
334}
335
336#[derive(Clone, Debug, Serialize, Deserialize, PartialEq)]
337#[serde(rename_all = "lowercase")]
338pub enum CompatibilityType {
339 Gguf,
340 Mlx,
341}
342
343#[derive(Serialize, Deserialize, Debug, Eq, PartialEq)]
344pub struct ResponseMessageDelta {
345 pub role: Option<Role>,
346 pub content: Option<String>,
347 #[serde(default, skip_serializing_if = "Option::is_none")]
348 pub reasoning_content: Option<String>,
349 #[serde(default, skip_serializing_if = "Option::is_none")]
350 pub tool_calls: Option<Vec<ToolCallChunk>>,
351}
352
353pub async fn complete(
354 client: &dyn HttpClient,
355 api_url: &str,
356 request: ChatCompletionRequest,
357) -> Result<ChatResponse> {
358 let uri = format!("{api_url}/chat/completions");
359 let request_builder = HttpRequest::builder()
360 .method(Method::POST)
361 .uri(uri)
362 .header("Content-Type", "application/json");
363
364 let serialized_request = serde_json::to_string(&request)?;
365 let request = request_builder.body(AsyncBody::from(serialized_request))?;
366
367 let mut response = client.send(request).await?;
368 if response.status().is_success() {
369 let mut body = Vec::new();
370 response.body_mut().read_to_end(&mut body).await?;
371 let response_message: ChatResponse = serde_json::from_slice(&body)?;
372 Ok(response_message)
373 } else {
374 let mut body = Vec::new();
375 response.body_mut().read_to_end(&mut body).await?;
376 let body_str = std::str::from_utf8(&body)?;
377 anyhow::bail!(
378 "Failed to connect to API: {} {}",
379 response.status(),
380 body_str
381 );
382 }
383}
384
385pub async fn stream_chat_completion(
386 client: &dyn HttpClient,
387 api_url: &str,
388 request: ChatCompletionRequest,
389) -> Result<BoxStream<'static, Result<ResponseStreamEvent>>> {
390 let uri = format!("{api_url}/chat/completions");
391 let request_builder = http::Request::builder()
392 .method(Method::POST)
393 .uri(uri)
394 .header("Content-Type", "application/json");
395
396 let request = request_builder.body(AsyncBody::from(serde_json::to_string(&request)?))?;
397 let mut response = client.send(request).await?;
398 if response.status().is_success() {
399 let reader = BufReader::new(response.into_body());
400 Ok(reader
401 .lines()
402 .filter_map(|line| async move {
403 match line {
404 Ok(line) => {
405 let line = line.strip_prefix("data: ")?;
406 if line == "[DONE]" {
407 None
408 } else {
409 match serde_json::from_str(line) {
410 Ok(ResponseStreamResult::Ok(response)) => Some(Ok(response)),
411 Ok(ResponseStreamResult::Err { error, .. }) => {
412 Some(Err(anyhow!(error.message)))
413 }
414 Err(error) => Some(Err(anyhow!(error))),
415 }
416 }
417 }
418 Err(error) => Some(Err(anyhow!(error))),
419 }
420 })
421 .boxed())
422 } else {
423 let mut body = String::new();
424 response.body_mut().read_to_string(&mut body).await?;
425 anyhow::bail!(
426 "Failed to connect to LM Studio API: {} {}",
427 response.status(),
428 body,
429 );
430 }
431}
432
433pub async fn get_models(
434 client: &dyn HttpClient,
435 api_url: &str,
436 _: Option<Duration>,
437) -> Result<Vec<ModelEntry>> {
438 let uri = format!("{api_url}/models");
439 let request_builder = HttpRequest::builder()
440 .method(Method::GET)
441 .uri(uri)
442 .header("Accept", "application/json");
443
444 let request = request_builder.body(AsyncBody::default())?;
445
446 let mut response = client.send(request).await?;
447
448 let mut body = String::new();
449 response.body_mut().read_to_string(&mut body).await?;
450
451 anyhow::ensure!(
452 response.status().is_success(),
453 "Failed to connect to LM Studio API: {} {}",
454 response.status(),
455 body,
456 );
457 let response: ListModelsResponse =
458 serde_json::from_str(&body).context("Unable to parse LM Studio models response")?;
459 Ok(response.data)
460}
461
462#[cfg(test)]
463mod tests {
464 use super::*;
465
466 #[test]
467 fn test_image_message_part_serialization() {
468 let image_part = MessagePart::Image {
469 image_url: ImageUrl {
470 url: "".to_string(),
471 detail: None,
472 },
473 };
474
475 let json = serde_json::to_string(&image_part).unwrap();
476 println!("Serialized image part: {}", json);
477
478 // Verify the structure matches what LM Studio expects
479 let expected_structure = r#"{"type":"image_url","image_url":{"url":""}}"#;
480 assert_eq!(json, expected_structure);
481 }
482
483 #[test]
484 fn test_text_message_part_serialization() {
485 let text_part = MessagePart::Text {
486 text: "Hello, world!".to_string(),
487 };
488
489 let json = serde_json::to_string(&text_part).unwrap();
490 println!("Serialized text part: {}", json);
491
492 let expected_structure = r#"{"type":"text","text":"Hello, world!"}"#;
493 assert_eq!(json, expected_structure);
494 }
495}