mirror of
				https://github.com/meilisearch/meilisearch.git
				synced 2025-10-26 13:36:27 +00:00 
			
		
		
		
	WIP multi embedders
fixed template bugs
This commit is contained in:
		| @@ -110,7 +110,6 @@ impl Prompt { | ||||
|         }; | ||||
|  | ||||
|         // render template with special object that's OK with `doc.*` and `fields.*` | ||||
|         /// FIXME: doesn't work for nested objects e.g. `doc.a.b` | ||||
|         this.template | ||||
|             .render(&template_checker::TemplateChecker) | ||||
|             .map_err(NewPromptError::invalid_fields_in_template)?; | ||||
| @@ -142,3 +141,80 @@ pub enum PromptFallbackStrategy { | ||||
|     #[default] | ||||
|     Error, | ||||
| } | ||||
|  | ||||
| #[cfg(test)] | ||||
| mod test { | ||||
|     use super::Prompt; | ||||
|     use crate::error::FaultSource; | ||||
|     use crate::prompt::error::{NewPromptError, NewPromptErrorKind}; | ||||
|  | ||||
|     #[test] | ||||
|     fn default_template() { | ||||
|         // does not panic | ||||
|         Prompt::default(); | ||||
|     } | ||||
|  | ||||
|     #[test] | ||||
|     fn empty_template() { | ||||
|         Prompt::new("".into(), None, None).unwrap(); | ||||
|     } | ||||
|  | ||||
|     #[test] | ||||
|     fn template_ok() { | ||||
|         Prompt::new("{{doc.title}}: {{doc.overview}}".into(), None, None).unwrap(); | ||||
|     } | ||||
|  | ||||
|     #[test] | ||||
|     fn template_syntax() { | ||||
|         assert!(matches!( | ||||
|             Prompt::new("{{doc.title: {{doc.overview}}".into(), None, None), | ||||
|             Err(NewPromptError { | ||||
|                 kind: NewPromptErrorKind::CannotParseTemplate(_), | ||||
|                 fault: FaultSource::User | ||||
|             }) | ||||
|         )); | ||||
|     } | ||||
|  | ||||
|     #[test] | ||||
|     fn template_missing_doc() { | ||||
|         assert!(matches!( | ||||
|             Prompt::new("{{title}}: {{overview}}".into(), None, None), | ||||
|             Err(NewPromptError { | ||||
|                 kind: NewPromptErrorKind::InvalidFieldsInTemplate(_), | ||||
|                 fault: FaultSource::User | ||||
|             }) | ||||
|         )); | ||||
|     } | ||||
|  | ||||
|     #[test] | ||||
|     fn template_nested_doc() { | ||||
|         Prompt::new("{{doc.actor.firstName}}: {{doc.actor.lastName}}".into(), None, None).unwrap(); | ||||
|     } | ||||
|  | ||||
|     #[test] | ||||
|     fn template_fields() { | ||||
|         Prompt::new("{% for field in fields %}{{field}}{% endfor %}".into(), None, None).unwrap(); | ||||
|     } | ||||
|  | ||||
|     #[test] | ||||
|     fn template_fields_ok() { | ||||
|         Prompt::new( | ||||
|             "{% for field in fields %}{{field.name}}: {{field.value}}{% endfor %}".into(), | ||||
|             None, | ||||
|             None, | ||||
|         ) | ||||
|         .unwrap(); | ||||
|     } | ||||
|  | ||||
|     #[test] | ||||
|     fn template_fields_invalid() { | ||||
|         assert!(matches!( | ||||
|             // intentionally garbled field | ||||
|             Prompt::new("{% for field in fields %}{{field.vaelu}} {% endfor %}".into(), None, None), | ||||
|             Err(NewPromptError { | ||||
|                 kind: NewPromptErrorKind::InvalidFieldsInTemplate(_), | ||||
|                 fault: FaultSource::User | ||||
|             }) | ||||
|         )); | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -1,7 +1,7 @@ | ||||
| use liquid::model::{ | ||||
|     ArrayView, DisplayCow, KStringCow, ObjectRender, ObjectSource, State, Value as LiquidValue, | ||||
| }; | ||||
| use liquid::{ObjectView, ValueView}; | ||||
| use liquid::{Object, ObjectView, ValueView}; | ||||
|  | ||||
| #[derive(Debug)] | ||||
| pub struct TemplateChecker; | ||||
| @@ -31,11 +31,11 @@ impl ObjectView for DummyField { | ||||
|     } | ||||
|  | ||||
|     fn values<'k>(&'k self) -> Box<dyn Iterator<Item = &'k dyn ValueView> + 'k> { | ||||
|         Box::new(std::iter::empty()) | ||||
|         Box::new(vec![DUMMY_VALUE.as_view(), DUMMY_VALUE.as_view()].into_iter()) | ||||
|     } | ||||
|  | ||||
|     fn iter<'k>(&'k self) -> Box<dyn Iterator<Item = (KStringCow<'k>, &'k dyn ValueView)> + 'k> { | ||||
|         Box::new(std::iter::empty()) | ||||
|         Box::new(self.keys().zip(self.values())) | ||||
|     } | ||||
|  | ||||
|     fn contains_key(&self, index: &str) -> bool { | ||||
| @@ -69,7 +69,12 @@ impl ValueView for DummyField { | ||||
|     } | ||||
|  | ||||
|     fn query_state(&self, state: State) -> bool { | ||||
|         DUMMY_VALUE.query_state(state) | ||||
|         match state { | ||||
|             State::Truthy => true, | ||||
|             State::DefaultValue => false, | ||||
|             State::Empty => false, | ||||
|             State::Blank => false, | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     fn to_kstr(&self) -> KStringCow<'_> { | ||||
| @@ -77,7 +82,10 @@ impl ValueView for DummyField { | ||||
|     } | ||||
|  | ||||
|     fn to_value(&self) -> LiquidValue { | ||||
|         LiquidValue::Nil | ||||
|         let mut this = Object::new(); | ||||
|         this.insert("name".into(), LiquidValue::Nil); | ||||
|         this.insert("value".into(), LiquidValue::Nil); | ||||
|         LiquidValue::Object(this) | ||||
|     } | ||||
|  | ||||
|     fn as_object(&self) -> Option<&dyn ObjectView> { | ||||
| @@ -103,7 +111,12 @@ impl ValueView for DummyFields { | ||||
|     } | ||||
|  | ||||
|     fn query_state(&self, state: State) -> bool { | ||||
|         DUMMY_VALUE.query_state(state) | ||||
|         match state { | ||||
|             State::Truthy => true, | ||||
|             State::DefaultValue => false, | ||||
|             State::Empty => false, | ||||
|             State::Blank => false, | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     fn to_kstr(&self) -> KStringCow<'_> { | ||||
| @@ -111,7 +124,7 @@ impl ValueView for DummyFields { | ||||
|     } | ||||
|  | ||||
|     fn to_value(&self) -> LiquidValue { | ||||
|         LiquidValue::Nil | ||||
|         LiquidValue::Array(vec![DummyField.to_value()]) | ||||
|     } | ||||
|  | ||||
|     fn as_array(&self) -> Option<&dyn ArrayView> { | ||||
| @@ -125,15 +138,15 @@ impl ArrayView for DummyFields { | ||||
|     } | ||||
|  | ||||
|     fn size(&self) -> i64 { | ||||
|         i64::MAX | ||||
|         u16::MAX as i64 | ||||
|     } | ||||
|  | ||||
|     fn values<'k>(&'k self) -> Box<dyn Iterator<Item = &'k dyn ValueView> + 'k> { | ||||
|         Box::new(std::iter::empty()) | ||||
|         Box::new(std::iter::once(DummyField.as_value())) | ||||
|     } | ||||
|  | ||||
|     fn contains_key(&self, _index: i64) -> bool { | ||||
|         true | ||||
|     fn contains_key(&self, index: i64) -> bool { | ||||
|         index < self.size() | ||||
|     } | ||||
|  | ||||
|     fn get(&self, _index: i64) -> Option<&dyn ValueView> { | ||||
| @@ -167,7 +180,8 @@ impl ObjectView for DummyDoc { | ||||
|     } | ||||
|  | ||||
|     fn get<'s>(&'s self, _index: &str) -> Option<&'s dyn ValueView> { | ||||
|         Some(DUMMY_VALUE.as_view()) | ||||
|         // Recursively sends itself | ||||
|         Some(self) | ||||
|     } | ||||
| } | ||||
|  | ||||
| @@ -189,7 +203,12 @@ impl ValueView for DummyDoc { | ||||
|     } | ||||
|  | ||||
|     fn query_state(&self, state: State) -> bool { | ||||
|         DUMMY_VALUE.query_state(state) | ||||
|         match state { | ||||
|             State::Truthy => true, | ||||
|             State::DefaultValue => false, | ||||
|             State::Empty => false, | ||||
|             State::Blank => false, | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     fn to_kstr(&self) -> KStringCow<'_> { | ||||
|   | ||||
		Reference in New Issue
	
	Block a user