summaryrefslogtreecommitdiffstats
path: root/src/frontend/json.rs
blob: 39e86f854f9df8c2c416a7363296fa8f62a1d3a0 (plain)
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
use std::io::Stdout;
use std::io::Write;
use std::ops::Deref;

use librepology::v1::types::Package;
use librepology::v1::types::Problem;
use librepology::v1::types::Repo;
use failure::Fallible as Result;
use failure::Error;

use crate::frontend::Frontend;
use crate::backend::Backend;
use crate::compare::ComparePackage;
use librepology::v1::api::Api;

pub struct JsonFrontend(Stdout);

/// A Frontend that serializes the data to JSON
///
/// Useful for piping the data as structured data to another program.
///
/// # Warning
///
/// This frontend does _not_ maintain compatibility with repolocli itself. That means that piping
/// output from repolocli to repolocli is _NOT_ supported by this frontend.
///
impl JsonFrontend {
    pub fn new(stdout: Stdout) -> Self {
        JsonFrontend(stdout)
    }

    fn write(&self, output: String) -> Result<()> {
        let mut outlock = self.0.lock();
        writeln!(outlock, "{}", output).map_err(Error::from)
    }
}

impl Frontend for JsonFrontend {
    fn list_packages(&self, packages: Vec<Package>) -> Result<()> {
        self.write(serde_json::ser::to_string_pretty(&packages).map_err(Error::from)?)

    }

    fn list_problems(&self, problems: Vec<Problem>) -> Result<()> {
        self.write(serde_json::ser::to_string_pretty(&problems).map_err(Error::from)?)
    }

    fn compare_packages(&self, packages: Vec<ComparePackage>, backend: &Backend, filter_repos: Vec<Repo>) -> Result<()> {
        #[derive(Serialize)]
        struct PackageComp {
            // not optimal, as we have to clone the inner variables from the package
            // but using references is too complicated right now
            package_name: String,
            local_version: String,
            comparisons: Vec<CompareTarget>,
        }

        #[derive(Serialize)]
        struct CompareTarget {
            version: String,
            repo: String,
        }

        let mut output: Vec<PackageComp> = vec![];

        for package in packages.iter() {

            let comparisons = backend
                .project(package.name().deref())?
                .into_iter()
                .filter(|p| filter_repos.contains(p.repo()))
                .map(|upstream_package| CompareTarget {
                    version: upstream_package.version().deref().clone(),
                    repo: upstream_package.repo().deref().clone(),
                })
                .collect();

            output.push(PackageComp {
                package_name: package.name().clone(),
                local_version: package.version().clone(),
                comparisons,
            });
        }

        self.write(serde_json::ser::to_string_pretty(&output)?)
    }
}