1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
use serde::{Deserialize, Serialize};

use crate::db::models::{challenges::Challenge, proposals::FullProposalInfo};

#[derive(Debug, Clone, Deserialize, Serialize)]
#[serde(rename_all = "kebab-case")]
pub struct SearchQuery {
    #[serde(flatten)]
    pub query: SearchCountQuery,
    pub limit: Option<u64>,
    pub offset: Option<u64>,
}

#[derive(Debug, Clone, Deserialize, Serialize)]
#[serde(rename_all = "kebab-case")]
pub struct SearchCountQuery {
    pub table: Table,
    #[serde(default)]
    pub filter: Vec<Constraint>,
    #[serde(default)]
    pub order_by: Vec<OrderBy>,
}

#[derive(Debug, Clone, Deserialize, Serialize)]
#[serde(rename_all = "kebab-case")]
#[serde(untagged)]
pub enum Constraint {
    Text {
        search: String,
        column: Column,
    },
    Range {
        lower: Option<i64>,
        upper: Option<i64>,
        column: Column,
    },
}

#[derive(Debug, Clone, Copy, Deserialize, Serialize)]
#[serde(rename_all = "kebab-case")]
pub enum OrderBy {
    Column {
        column: Column,
        #[serde(default)]
        descending: bool,
    },
    Random,
}

#[derive(Debug, Clone, Copy, Deserialize, Serialize)]
#[serde(rename_all = "kebab-case")]
pub enum Table {
    Challenges,
    Proposals,
}

#[derive(Debug, Clone, Copy, Deserialize, Serialize)]
#[serde(rename_all = "kebab-case")]
pub enum Column {
    Title,
    Type,
    Desc,
    Author,
    Funds,
    ImpactScore,
}

#[derive(Debug, Clone, Serialize, Deserialize)]
#[serde(untagged)] // should serialize as if it is either a `Vec<Challenge>` or `Vec<FullProposalInfo>`
pub enum SearchResponse {
    Challenge(Vec<Challenge>),
    Proposal(Vec<FullProposalInfo>),
}

impl SearchResponse {
    pub fn is_empty(&self) -> bool {
        match self {
            Self::Challenge(challenges) => challenges.is_empty(),
            Self::Proposal(proposals) => proposals.is_empty(),
        }
    }
}

#[cfg(test)]
mod tests {
    use serde_json::{from_value, json, to_string};

    use crate::db::models::proposals::test::get_test_proposal;

    use super::*;

    #[test]
    fn response_serializes_as_vec() {
        let response = SearchResponse::Proposal(vec![get_test_proposal("asdf")]);
        let s = to_string(&response).unwrap();
        assert!(s.starts_with('['));
        assert!(s.ends_with(']'));
    }

    #[test]
    fn filters_and_orders_are_optional() {
        from_value::<SearchQuery>(json!({"table": "proposals"})).unwrap();
        from_value::<SearchCountQuery>(json!({"table": "proposals"})).unwrap();
    }
}