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
use crate::error::Result;
use crate::query::facet::FacetCount;
use crate::query::works::{WorksCombined, WorksFilter, WorksQuery};
use crate::query::*;
use std::borrow::Cow;

/// filters supported for the /funders route
#[derive(Debug, PartialEq, Eq, Clone, Serialize, Deserialize)]
pub enum FundersFilter {
    /// funders located in specified country
    Location(String),
}

impl FundersFilter {
    /// the key name for the filter element
    pub fn name(&self) -> &str {
        match self {
            FundersFilter::Location(_) => "location",
        }
    }
}

impl ParamFragment for FundersFilter {
    fn key(&self) -> Cow<str> {
        Cow::Borrowed(self.name())
    }

    fn value(&self) -> Option<Cow<str>> {
        match self {
            FundersFilter::Location(s) => Some(Cow::Borrowed(s.as_str())),
        }
    }
}

impl Filter for FundersFilter {}

impl_common_query!(FundersQuery, FundersFilter);

/// constructs the request payload for the `/funders` route
#[derive(Debug, Clone, Serialize, Deserialize)]
pub enum Funders {
    /// target a specific funder at `/funder/{id}`
    Identifier(String),
    /// target all funders that match the query at `/funders?query...`
    Query(FundersQuery),
    /// target a `Work` for a specific funder at `/funders/{id}/works?query..`
    Works(WorksCombined),
}

impl CrossrefRoute for Funders {
    fn route(&self) -> Result<String> {
        match self {
            Funders::Identifier(s) => Ok(format!("{}/{}", Component::Funders.route()?, s)),
            Funders::Query(query) => {
                let query = query.route()?;
                if query.is_empty() {
                    Component::Funders.route()
                } else {
                    Ok(format!("{}?{}", Component::Funders.route()?, query))
                }
            }
            Funders::Works(combined) => {
                let query = combined.query.route()?;
                if query.is_empty() {
                    Ok(format!(
                        "{}/{}/{}",
                        Component::Funders.route()?,
                        combined.id,
                        Component::Works.route()?
                    ))
                } else {
                    Ok(format!(
                        "{}/{}/{}?{}",
                        Component::Funders.route()?,
                        combined.id,
                        Component::Works.route()?,
                        query
                    ))
                }
            }
        }
    }
}

impl CrossrefQuery for Funders {
    fn resource_component(self) -> ResourceComponent {
        ResourceComponent::Funders(self)
    }
}