1
use crate::server::{
2
    dataset::{Dataset, PredicateObject},
3
    DatasetSummary, DatasetVersionSummary,
4
};
5
use askama::Template;
6
use sophia::api::{prefix::PrefixMapPair, term::IriRef, MownStr};
7
use std::collections::BTreeSet;
8

            
9
2
#[derive(Template)]
10
2
#[template(path = "index.html")]
11
pub(crate) struct IndexTemplate<'a> {
12
    pub datasets: &'a [DatasetSummary<'a>],
13
}
14

            
15
1
#[derive(Template)]
16
1
#[template(path = "dataset.html")]
17
pub(crate) struct DatasetTemplate<'a> {
18
    pub dataset: &'a Dataset,
19
    pub versions: &'a [DatasetVersionSummary],
20
}
21

            
22
1063
#[derive(Template)]
23
1
#[template(path = "version.html")]
24
pub(crate) struct VersionTemplate<'a> {
25
    pub dataset: &'a DatasetSummary<'a>,
26
    pub version: &'a DatasetVersionSummary,
27
    pub subjects: BTreeSet<&'a IriRef<MownStr<'a>>>,
28
    pub datasets: &'a Vec<Dataset>,
29
    pub prefixes: &'a Vec<PrefixMapPair>,
30
}
31

            
32
6
#[derive(Template)]
33
1
#[template(path = "subject.html")]
34
pub(crate) struct SubjectTemplate<'a> {
35
    pub dataset: &'a DatasetSummary<'a>,
36
    pub subject: String,
37
    pub version: &'a DatasetVersionSummary,
38
    pub pos: Vec<PredicateObject<'a>>,
39
    pub datasets: &'a Vec<Dataset>,
40
    pub prefixes: &'a Vec<PrefixMapPair>,
41
}
42

            
43
mod filters {
44
    use std::{collections::BTreeSet, fmt};
45

            
46
    use askama::Error::Fmt;
47
    use chrono::{DateTime, Utc};
48
    use sophia::{
49
        api::{
50
            prefix::{PrefixMap, PrefixMapPair},
51
            term::IriRef,
52
            MownStr,
53
        },
54
        iri::Iri,
55
    };
56

            
57
    use crate::{
58
        constants::LABEL_RDFS_IRI,
59
        server::{dataset::Dataset, get_dataset_version},
60
    };
61

            
62
    /// Filter to replace the prefix in a resource IRI
63
124
    pub(crate) fn replace_prefix<T: std::fmt::Display>(
64
124
        s: T,
65
124
        prefixes: &Vec<PrefixMapPair>,
66
124
    ) -> ::askama::Result<String> {
67
124
        let prefix_found = prefixes.get_prefixed_pair(Iri::new_unchecked(s.to_string()));
68

            
69
124
        let out = if let Some(ps) = prefix_found {
70
122
            format!("{}:{}", ps.0.as_str(), ps.1)
71
        } else {
72
2
            s.to_string()
73
        };
74

            
75
124
        Ok(out)
76
124
    }
77

            
78
    /// Filter to find and display the rdfs:label resource for a subject
79
123
    pub(crate) fn rdfslabel<T: std::fmt::Display>(
80
123
        s: T,
81
123
        datasets: &[Dataset],
82
123
        id: &&str,
83
123
        version: &DateTime<Utc>,
84
123
    ) -> ::askama::Result<String> {
85
123
        let subject = Iri::new_unchecked(s.to_string());
86
123
        let Ok((_, dataset_version)) = get_dataset_version(datasets, id, *version) else {
87
            return Err(Fmt(fmt::Error));
88
        };
89
123
        let pos = dataset_version.pos(&subject);
90
123

            
91
123
        let find_target_po = pos
92
123
            .iter()
93
379
            .find(|po| po.predicate.as_str() == LABEL_RDFS_IRI);
94
123
        let Some(target_po) = find_target_po else {
95
7
            return Ok("".to_string());
96
        };
97
116
        let Some(target_object) = target_po.object_literal else {
98
            return Ok("".to_string());
99
        };
100

            
101
116
        Ok(target_object.to_string())
102
123
    }
103

            
104
    /// Filter to sort subjects by prefix
105
    ///
106
    /// Each prefix becomes a category.  
107
    /// This filter has a list of hardcoded prefixes that it will place in front of the other prefixes.  
108
    /// This filter returns a tuple of two vectors:
109
    ///   1. the list of names for each prefix/category
110
    ///   2. the index of the prefix/category assigned to each subject
111
1
    pub(crate) fn categorize_subjects<'a>(
112
1
        subs: &BTreeSet<&'a IriRef<MownStr<'a>>>,
113
1
        prefixes: &Vec<PrefixMapPair>,
114
1
    ) -> ::askama::Result<(Vec<String>, Vec<usize>)> {
115
1
        let special_categories: Vec<&str> = vec!["proposal", "nlnet"];
116
1
        let mut categories: Vec<String> = prefixes
117
1
            .iter()
118
7
            .map(|(p, _)| p.to_string())
119
7
            .filter(|cat| !special_categories.contains(&cat.as_str()))
120
1
            .collect();
121
1

            
122
1
        let cat_assigns: Vec<usize> = subs
123
1
            .iter()
124
118
            .map(|s| {
125
118
                let prefix_found = prefixes.get_prefixed_pair(Iri::new_unchecked(s.to_string()));
126

            
127
118
                if let Some((prefix, _)) = prefix_found {
128
116
                    if let Some(idx) = special_categories
129
116
                        .iter()
130
232
                        .position(|cat| cat == &prefix.as_str())
131
                    {
132
                        return idx;
133
116
                    } else if let Some(idx) =
134
362
                        categories.iter().position(|cat| cat == prefix.as_str())
135
                    {
136
116
                        return idx + special_categories.len();
137
                    }
138
2
                }
139

            
140
2
                categories.len() + special_categories.len()
141
118
            })
142
1
            .collect();
143
1

            
144
1
        let mut all_categories = vec!["Proposals".to_string(), "NLnet".to_string()];
145
1
        let mut others = vec!["Other".to_string()];
146
1
        all_categories.append(&mut categories);
147
1
        all_categories.append(&mut others);
148
1

            
149
1
        Ok((all_categories, cat_assigns))
150
1
    }
151
}