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
use crate::{AvroResult, Error};
use num_bigint::{BigInt, Sign};
#[derive(Debug, Clone)]
pub struct Decimal {
value: BigInt,
len: usize,
}
impl PartialEq for Decimal {
fn eq(&self, other: &Self) -> bool {
self.value == other.value
}
}
impl Decimal {
pub(crate) fn len(&self) -> usize {
self.len
}
fn to_vec(&self) -> AvroResult<Vec<u8>> {
self.to_sign_extended_bytes_with_len(self.len)
}
pub(crate) fn to_sign_extended_bytes_with_len(&self, len: usize) -> AvroResult<Vec<u8>> {
let sign_byte = 0xFF * u8::from(self.value.sign() == Sign::Minus);
let mut decimal_bytes = vec![sign_byte; len];
let raw_bytes = self.value.to_signed_bytes_be();
let num_raw_bytes = raw_bytes.len();
let start_byte_index = len.checked_sub(num_raw_bytes).ok_or(Error::SignExtend {
requested: len,
needed: num_raw_bytes,
})?;
decimal_bytes[start_byte_index..].copy_from_slice(&raw_bytes);
Ok(decimal_bytes)
}
}
impl std::convert::TryFrom<&Decimal> for Vec<u8> {
type Error = Error;
fn try_from(decimal: &Decimal) -> Result<Self, Self::Error> {
decimal.to_vec()
}
}
impl std::convert::TryFrom<Decimal> for Vec<u8> {
type Error = Error;
fn try_from(decimal: Decimal) -> Result<Self, Self::Error> {
decimal.to_vec()
}
}
impl<T: AsRef<[u8]>> From<T> for Decimal {
fn from(bytes: T) -> Self {
let bytes_ref = bytes.as_ref();
Self {
value: BigInt::from_signed_bytes_be(bytes_ref),
len: bytes_ref.len(),
}
}
}
#[cfg(test)]
mod tests {
use super::*;
use pretty_assertions::assert_eq;
use std::convert::TryFrom;
#[test]
fn test_decimal_from_bytes_from_ref_decimal() {
let input = vec![1, 24];
let d = Decimal::from(&input);
let output = <Vec<u8>>::try_from(&d).unwrap();
assert_eq!(output, input);
}
#[test]
fn test_decimal_from_bytes_from_owned_decimal() {
let input = vec![1, 24];
let d = Decimal::from(&input);
let output = <Vec<u8>>::try_from(d).unwrap();
assert_eq!(output, input);
}
}