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
use crate::pagination::{Cursor, Pagination};
use crate::Model;
use async_graphql::connection::{Connection, CursorType, Edge};
use async_graphql::{
    InputValueError, InputValueResult, OutputType, Scalar, ScalarType, SimpleObject, Value,
};
use std::fmt::Debug;

#[Scalar]
impl ScalarType for Cursor {
    fn parse(value: Value) -> InputValueResult<Self> {
        if let Value::String(value) = &value {
            let cursor =
                Cursor::decode(value).map_err(|e| InputValueError::custom(e.to_string()))?;
            Ok(cursor)
        } else {
            Err(InputValueError::expected_type(value))
        }
    }

    fn to_value(&self) -> Value {
        let value = self.encode();
        Value::String(value)
    }
}

impl CursorType for Cursor {
    type Error = anyhow::Error;

    fn decode_cursor(s: &str) -> Result<Self, Self::Error> {
        Self::decode(s)
    }

    fn encode_cursor(&self) -> String {
        self.encode()
    }
}

#[derive(SimpleObject, Debug, Clone)]
pub struct AdditionalFields {
    pub total_nodes: i64,
}

impl<T> From<Pagination<T>> for Connection<Cursor, T, AdditionalFields>
where
    T: OutputType + Model + Debug,
{
    fn from(value: Pagination<T>) -> Self {
        let mut connection = Connection::with_additional_fields(
            value.has_previous,
            value.has_next,
            AdditionalFields {
                total_nodes: value.total_nodes,
            },
        );

        connection.edges = value
            .items
            .into_iter()
            .map(|item| Edge::new(item.cursor(), item))
            .collect::<Vec<_>>();

        connection
    }
}