sui_graphql_rpc/types/
base64.rs

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
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
// Copyright (c) Mysten Labs, Inc.
// SPDX-License-Identifier: Apache-2.0

use std::str::FromStr;

use async_graphql::*;
use fastcrypto::encoding::Base64 as FastCryptoBase64;
use fastcrypto::encoding::Encoding as FastCryptoEncoding;

#[derive(Clone, Debug, PartialEq, Eq)]
pub(crate) struct Base64(pub(crate) Vec<u8>);

/// String containing Base64-encoded binary data.
#[Scalar]
impl ScalarType for Base64 {
    fn parse(value: Value) -> InputValueResult<Self> {
        match value {
            Value::String(s) => Ok(Base64(
                FastCryptoBase64::decode(&s)
                    .map_err(|r| InputValueError::custom(format!("{r}")))?,
            )),
            _ => Err(InputValueError::expected_type(value)),
        }
    }

    fn to_value(&self) -> Value {
        Value::String(FastCryptoBase64::encode(self.0.clone()))
    }
}

impl FromStr for Base64 {
    type Err = InputValueError<String>;

    fn from_str(s: &str) -> Result<Self, Self::Err> {
        Ok(Base64(
            FastCryptoBase64::decode(s).map_err(|_| InputValueError::custom("Invalid Base64"))?,
        ))
    }
}

impl From<&Vec<u8>> for Base64 {
    fn from(bytes: &Vec<u8>) -> Self {
        Base64(bytes.clone())
    }
}

impl From<&[u8]> for Base64 {
    fn from(bytes: &[u8]) -> Self {
        Base64(bytes.to_vec())
    }
}

impl From<Vec<u8>> for Base64 {
    fn from(bytes: Vec<u8>) -> Self {
        Base64(bytes)
    }
}

#[cfg(test)]
mod tests {
    use super::*;
    use async_graphql::Value;

    fn assert_input_value_error<T, U>(result: Result<T, InputValueError<U>>) {
        match result {
            Err(InputValueError { .. }) => {}
            _ => panic!("Expected InputValueError"),
        }
    }

    #[test]
    fn test_parse_valid_base64() {
        let input = Value::String("SGVsbG8gd29ybGQ=".to_string());
        let parsed = <Base64 as ScalarType>::parse(input).unwrap();
        assert_eq!(parsed.0, b"Hello world");
    }

    #[test]
    fn test_parse_invalid_base64() {
        let input = Value::String("SGVsbG8gd29ybGQ@".to_string());
        let parsed = <Base64 as ScalarType>::parse(input);
        assert_input_value_error(parsed);
    }

    #[test]
    fn test_parse_invalid_boolean_value() {
        let input = Value::Boolean(true);
        let parsed = <Base64 as ScalarType>::parse(input);
        assert_input_value_error(parsed);
    }

    #[test]
    fn test_parse_invalid_number() {
        let input = Value::Number(1.into());
        let parsed = <Base64 as ScalarType>::parse(input);
        assert_input_value_error(parsed);
    }

    #[test]
    fn test_to_value() {
        let base64 = Base64(b"Hello world".to_vec());
        let value = <Base64 as ScalarType>::to_value(&base64);
        assert_eq!(value, Value::String("SGVsbG8gd29ybGQ=".to_string()));
    }

    #[test]
    fn test_from_str_valid() {
        let base64_str = "SGVsbG8gd29ybGQ=";
        let base64 = Base64::from_str(base64_str).unwrap();
        assert_eq!(base64.0, b"Hello world");
    }

    #[test]
    fn test_from_str_invalid() {
        let base64_str = "SGVsbG8gd29ybGQ@";
        let result = Base64::from_str(base64_str);
        assert_input_value_error(result);
    }

    #[test]
    fn test_from_vec_reference() {
        let vec = vec![1, 2, 3, 4, 5];
        let base64 = Base64::from(&vec);
        assert_eq!(base64.0, vec);
    }

    #[test]
    fn test_from_vec() {
        let vec = vec![1, 2, 3, 4, 5];
        let base64 = Base64::from(vec.clone());
        assert_eq!(base64.0, vec);
    }
}