diff --git a/crates/meilisearch/tests/vector/settings.rs b/crates/meilisearch/tests/vector/settings.rs index c89102de9..2aae67ebf 100644 --- a/crates/meilisearch/tests/vector/settings.rs +++ b/crates/meilisearch/tests/vector/settings.rs @@ -255,3 +255,155 @@ async fn reset_embedder_documents() { } "###); } + +#[actix_rt::test] +async fn ollama_url_checks() { + let server = super::get_server_vector().await; + let index = server.index("doggo"); + + let (response, code) = index + .update_settings(json!({ + "embedders": { "ollama": {"source": "ollama", "model": "toto", "dimensions": 1, "url": "http://localhost:11434/api/embeddings"}}, + })) + .await; + snapshot!(code, @"202 Accepted"); + let response = server.wait_task(response.uid()).await; + + snapshot!(response, @r###" + { + "uid": "[uid]", + "batchUid": "[batch_uid]", + "indexUid": "doggo", + "status": "succeeded", + "type": "settingsUpdate", + "canceledBy": null, + "details": { + "embedders": { + "ollama": { + "source": "ollama", + "model": "toto", + "dimensions": 1, + "url": "[url]" + } + } + }, + "error": null, + "duration": "[duration]", + "enqueuedAt": "[date]", + "startedAt": "[date]", + "finishedAt": "[date]" + } + "###); + + let (response, code) = index + .update_settings(json!({ + "embedders": { "ollama": {"source": "ollama", "model": "toto", "dimensions": 1, "url": "http://localhost:11434/api/embed"}}, + })) + .await; + snapshot!(code, @"202 Accepted"); + let response = server.wait_task(response.uid()).await; + + snapshot!(response, @r###" + { + "uid": "[uid]", + "batchUid": "[batch_uid]", + "indexUid": "doggo", + "status": "succeeded", + "type": "settingsUpdate", + "canceledBy": null, + "details": { + "embedders": { + "ollama": { + "source": "ollama", + "model": "toto", + "dimensions": 1, + "url": "[url]" + } + } + }, + "error": null, + "duration": "[duration]", + "enqueuedAt": "[date]", + "startedAt": "[date]", + "finishedAt": "[date]" + } + "###); + + let (response, code) = index + .update_settings(json!({ + "embedders": { "ollama": {"source": "ollama", "model": "toto", "dimensions": 1, "url": "http://localhost:11434/api/embedd"}}, + })) + .await; + snapshot!(code, @"202 Accepted"); + let response = server.wait_task(response.uid()).await; + + snapshot!(response, @r###" + { + "uid": "[uid]", + "batchUid": "[batch_uid]", + "indexUid": "doggo", + "status": "failed", + "type": "settingsUpdate", + "canceledBy": null, + "details": { + "embedders": { + "ollama": { + "source": "ollama", + "model": "toto", + "dimensions": 1, + "url": "[url]" + } + } + }, + "error": { + "message": "Index `doggo`: Error while generating embeddings: user error: unsupported Ollama URL.\n - For `ollama` sources, the URL must end with `/api/embed` or `/api/embeddings`\n - Got `http://localhost:11434/api/embedd`", + "code": "vector_embedding_error", + "type": "invalid_request", + "link": "https://docs.meilisearch.com/errors#vector_embedding_error" + }, + "duration": "[duration]", + "enqueuedAt": "[date]", + "startedAt": "[date]", + "finishedAt": "[date]" + } + "###); + + let (response, code) = index + .update_settings(json!({ + "embedders": { "ollama": {"source": "ollama", "model": "toto", "dimensions": 1, "url": "http://localhost:11434/v1/embeddings"}}, + })) + .await; + snapshot!(code, @"202 Accepted"); + let response = server.wait_task(response.uid()).await; + + snapshot!(response, @r###" + { + "uid": "[uid]", + "batchUid": "[batch_uid]", + "indexUid": "doggo", + "status": "failed", + "type": "settingsUpdate", + "canceledBy": null, + "details": { + "embedders": { + "ollama": { + "source": "ollama", + "model": "toto", + "dimensions": 1, + "url": "[url]" + } + } + }, + "error": { + "message": "Index `doggo`: Error while generating embeddings: user error: unsupported Ollama URL.\n - For `ollama` sources, the URL must end with `/api/embed` or `/api/embeddings`\n - Got `http://localhost:11434/v1/embeddings`", + "code": "vector_embedding_error", + "type": "invalid_request", + "link": "https://docs.meilisearch.com/errors#vector_embedding_error" + }, + "duration": "[duration]", + "enqueuedAt": "[date]", + "startedAt": "[date]", + "finishedAt": "[date]" + } + "###); +} diff --git a/crates/milli/src/vector/error.rs b/crates/milli/src/vector/error.rs index 5edabed0d..d1b2516f5 100644 --- a/crates/milli/src/vector/error.rs +++ b/crates/milli/src/vector/error.rs @@ -67,7 +67,7 @@ pub enum EmbedErrorKind { #[error("could not authenticate against {embedding} server{server_reply}{hint}", embedding=match *.1 { ConfigurationSource::User => "embedding", ConfigurationSource::OpenAi => "OpenAI", - ConfigurationSource::Ollama => "ollama" + ConfigurationSource::Ollama => "Ollama" }, server_reply=option_info(.0.as_deref(), "server replied with "), hint=match *.1 { @@ -306,6 +306,10 @@ impl NewEmbedderError { fault: FaultSource::User, } } + + pub(crate) fn ollama_unsupported_url(url: String) -> NewEmbedderError { + Self { kind: NewEmbedderErrorKind::OllamaUnsupportedUrl(url), fault: FaultSource::User } + } } #[derive(Debug, thiserror::Error)] @@ -369,6 +373,8 @@ pub enum NewEmbedderErrorKind { LoadModel(candle_core::Error), #[error("{0}")] CouldNotParseTemplate(String), + #[error("unsupported Ollama URL.\n - For `ollama` sources, the URL must end with `/api/embed` or `/api/embeddings`\n - Got `{0}`")] + OllamaUnsupportedUrl(String), } pub struct PossibleEmbeddingMistakes { diff --git a/crates/milli/src/vector/ollama.rs b/crates/milli/src/vector/ollama.rs index 7ee775cbf..fb0f3fb82 100644 --- a/crates/milli/src/vector/ollama.rs +++ b/crates/milli/src/vector/ollama.rs @@ -38,26 +38,46 @@ impl EmbedderOptions { dimensions, } } + + fn into_rest_embedder_config(self) -> Result { + let url = self.url.unwrap_or_else(get_ollama_path); + let model = self.embedding_model.as_str(); + + // **warning**: do not swap these two `if`s, as the second one is always true when the first one is. + let (request, response) = if url.ends_with("/api/embeddings") { + ( + serde_json::json!({"model": model, "input": [super::rest::REQUEST_PLACEHOLDER, super::rest::REPEAT_PLACEHOLDER]}), + serde_json::json!({"embeddings": [super::rest::RESPONSE_PLACEHOLDER, super::rest::REPEAT_PLACEHOLDER]}), + ) + } else if url.ends_with("/api/embed") { + ( + serde_json::json!({ + "model": model, + "prompt": super::rest::REQUEST_PLACEHOLDER, + }), + serde_json::json!({ + "embedding": super::rest::RESPONSE_PLACEHOLDER, + }), + ) + } else { + return Err(NewEmbedderError::ollama_unsupported_url(url)); + }; + Ok(RestEmbedderOptions { + api_key: self.api_key, + dimensions: self.dimensions, + distribution: self.distribution, + url, + request, + response, + headers: Default::default(), + }) + } } impl Embedder { pub fn new(options: EmbedderOptions) -> Result { - let model = options.embedding_model.as_str(); let rest_embedder = match RestEmbedder::new( - RestEmbedderOptions { - api_key: options.api_key, - dimensions: options.dimensions, - distribution: options.distribution, - url: options.url.unwrap_or_else(get_ollama_path), - request: serde_json::json!({ - "model": model, - "prompt": super::rest::REQUEST_PLACEHOLDER, - }), - response: serde_json::json!({ - "embedding": super::rest::RESPONSE_PLACEHOLDER, - }), - headers: Default::default(), - }, + options.into_rest_embedder_config()?, super::rest::ConfigurationSource::Ollama, ) { Ok(embedder) => embedder,