use derive_builder::Builder;
use crate::{util::debug_long_utf8, Embedding, SparseEmbedding};
type Document = String;
#[derive(Clone, Default, Builder, PartialEq)]
#[builder(setter(into))]
pub struct Query<State> {
    original: String,
    #[builder(default = "self.original.clone().unwrap_or_default()")]
    current: String,
    state: State,
    #[builder(default)]
    transformation_history: Vec<TransformationEvent>,
    #[builder(default)]
    pub embedding: Option<Embedding>,
    #[builder(default)]
    pub sparse_embedding: Option<SparseEmbedding>,
}
impl<T: std::fmt::Debug> std::fmt::Debug for Query<T> {
    fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
        f.debug_struct("Query")
            .field("original", &debug_long_utf8(&self.original, 100))
            .field("current", &debug_long_utf8(&self.current, 100))
            .field("state", &self.state)
            .field("transformation_history", &self.transformation_history)
            .field("embedding", &self.embedding.is_some())
            .finish()
    }
}
impl<T: Clone> Query<T> {
    pub fn builder() -> QueryBuilder<T> {
        QueryBuilder::default().clone()
    }
    pub fn original(&self) -> &str {
        &self.original
    }
    pub fn current(&self) -> &str {
        &self.current
    }
    fn transition_to<S>(self, new_state: S) -> Query<S> {
        Query {
            state: new_state,
            original: self.original,
            current: self.current,
            transformation_history: self.transformation_history,
            embedding: self.embedding,
            sparse_embedding: self.sparse_embedding,
        }
    }
    #[allow(dead_code)]
    pub fn history(&self) -> &Vec<TransformationEvent> {
        &self.transformation_history
    }
}
impl Query<states::Pending> {
    pub fn new(query: impl Into<String>) -> Self {
        Self {
            original: query.into(),
            ..Default::default()
        }
    }
    pub fn transformed_query(&mut self, new_query: impl Into<String>) {
        let new_query = new_query.into();
        self.transformation_history
            .push(TransformationEvent::Transformed {
                before: self.current.clone(),
                after: new_query.clone(),
            });
        self.current = new_query;
    }
    pub fn retrieved_documents(mut self, documents: Vec<Document>) -> Query<states::Retrieved> {
        self.transformation_history
            .push(TransformationEvent::Retrieved {
                before: self.current.clone(),
                after: String::new(),
                documents: documents.clone(),
            });
        let state = states::Retrieved { documents };
        self.current.clear();
        self.transition_to(state)
    }
}
impl Query<states::Retrieved> {
    pub fn new() -> Self {
        Self::default()
    }
    pub fn transformed_response(&mut self, new_response: impl Into<String>) {
        let new_response = new_response.into();
        self.transformation_history
            .push(TransformationEvent::Transformed {
                before: self.current.clone(),
                after: new_response.clone(),
            });
        self.current = new_response;
    }
    pub fn documents(&self) -> &[Document] {
        &self.state.documents
    }
    #[must_use]
    pub fn answered(self, answer: impl Into<String>) -> Query<states::Answered> {
        let state = states::Answered {
            answer: answer.into(),
        };
        self.transition_to(state)
    }
}
impl Query<states::Answered> {
    pub fn new() -> Self {
        Self::default()
    }
    pub fn answer(&self) -> &str {
        &self.state.answer
    }
}
pub mod states {
    use crate::util::debug_long_utf8;
    use super::Builder;
    use super::Document;
    #[derive(Debug, Default, Clone)]
    pub struct Pending;
    #[derive(Default, Clone, Builder, PartialEq)]
    #[builder(setter(into))]
    pub struct Retrieved {
        pub(crate) documents: Vec<Document>,
    }
    impl std::fmt::Debug for Retrieved {
        fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
            f.debug_struct("Retrieved")
                .field("num_documents", &self.documents.len())
                .field(
                    "documents",
                    &self
                        .documents
                        .iter()
                        .map(|d| debug_long_utf8(d, 100))
                        .collect::<Vec<_>>(),
                )
                .finish()
        }
    }
    #[derive(Default, Clone, Builder, PartialEq)]
    #[builder(setter(into))]
    pub struct Answered {
        pub(crate) answer: String,
    }
    impl std::fmt::Debug for Answered {
        fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
            f.debug_struct("Answered")
                .field("answer", &debug_long_utf8(&self.answer, 100))
                .finish()
        }
    }
}
impl<T: AsRef<str>> From<T> for Query<states::Pending> {
    fn from(original: T) -> Self {
        Self {
            original: original.as_ref().to_string(),
            current: original.as_ref().to_string(),
            state: states::Pending,
            ..Default::default()
        }
    }
}
#[derive(Clone, PartialEq)]
pub enum TransformationEvent {
    Transformed {
        before: String,
        after: String,
    },
    Retrieved {
        before: String,
        after: String,
        documents: Vec<Document>,
    },
}
impl std::fmt::Debug for TransformationEvent {
    fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
        match self {
            TransformationEvent::Transformed { before, after } => {
                write!(
                    f,
                    "Transformed: {} -> {}",
                    &debug_long_utf8(before, 100),
                    &debug_long_utf8(after, 100)
                )
            }
            TransformationEvent::Retrieved {
                before,
                after,
                documents,
            } => {
                write!(
                    f,
                    "Retrieved: {} -> {}\nDocuments: {:?}",
                    &debug_long_utf8(before, 100),
                    &debug_long_utf8(after, 100),
                    documents.len()
                )
            }
        }
    }
}
#[cfg(test)]
mod tests {
    use super::*;
    #[test]
    fn test_query_initial_state() {
        let query = Query::<states::Pending>::from("test query");
        assert_eq!(query.original(), "test query");
        assert_eq!(query.current(), "test query");
        assert_eq!(query.history().len(), 0);
    }
    #[test]
    fn test_query_transformed_query() {
        let mut query = Query::<states::Pending>::from("test query");
        query.transformed_query("new query");
        assert_eq!(query.current(), "new query");
        assert_eq!(query.history().len(), 1);
        if let TransformationEvent::Transformed { before, after } = &query.history()[0] {
            assert_eq!(before, "test query");
            assert_eq!(after, "new query");
        } else {
            panic!("Unexpected event in history");
        }
    }
    #[test]
    fn test_query_retrieved_documents() {
        let query = Query::<states::Pending>::from("test query");
        let documents = vec!["doc1".to_string(), "doc2".to_string()];
        let query = query.retrieved_documents(documents.clone());
        assert_eq!(query.documents(), &documents);
        assert_eq!(query.history().len(), 1);
        assert!(query.current().is_empty());
        if let TransformationEvent::Retrieved {
            before,
            after,
            documents: retrieved_docs,
        } = &query.history()[0]
        {
            assert_eq!(before, "test query");
            assert_eq!(after, "");
            assert_eq!(retrieved_docs, &documents);
        } else {
            panic!("Unexpected event in history");
        }
    }
    #[test]
    fn test_query_transformed_response() {
        let query = Query::<states::Pending>::from("test query");
        let documents = vec!["doc1".to_string(), "doc2".to_string()];
        let mut query = query.retrieved_documents(documents.clone());
        query.transformed_response("new response");
        assert_eq!(query.current(), "new response");
        assert_eq!(query.history().len(), 2);
        assert_eq!(query.documents(), &documents);
        assert_eq!(query.original, "test query");
        if let TransformationEvent::Transformed { before, after } = &query.history()[1] {
            assert_eq!(before, "");
            assert_eq!(after, "new response");
        } else {
            panic!("Unexpected event in history");
        }
    }
    #[test]
    fn test_query_answered() {
        let query = Query::<states::Pending>::from("test query");
        let documents = vec!["doc1".to_string(), "doc2".to_string()];
        let query = query.retrieved_documents(documents);
        let query = query.answered("the answer");
        assert_eq!(query.answer(), "the answer");
    }
}