use std::i64;
use std::io::Read;
use serde_json::{Map, Value};
use crate::error::{DecodeError, Error as AvroError};
pub const MAX_ALLOCATION_BYTES: usize = 512 * 1024 * 1024;
#[derive(Debug, Clone, Copy, Eq, PartialEq)]
pub enum TsUnit {
Millis,
Micros,
}
pub trait MapHelper {
fn string(&self, key: &str) -> Option<String>;
fn name(&self) -> Option<String> {
self.string("name")
}
fn doc(&self) -> Option<String> {
self.string("doc")
}
}
impl MapHelper for Map<String, Value> {
fn string(&self, key: &str) -> Option<String> {
self.get(key)
.and_then(|v| v.as_str())
.map(|v| v.to_string())
}
}
pub fn read_long<R: Read>(reader: &mut R) -> Result<i64, AvroError> {
zag_i64(reader)
}
pub fn zig_i32(n: i32, buffer: &mut Vec<u8>) {
zig_i64(n as i64, buffer)
}
pub fn zig_i64(n: i64, buffer: &mut Vec<u8>) {
encode_variable(((n << 1) ^ (n >> 63)) as u64, buffer)
}
pub fn zag_i32<R: Read>(reader: &mut R) -> Result<i32, AvroError> {
let i = zag_i64(reader)?;
if i < i64::from(i32::min_value()) || i > i64::from(i32::max_value()) {
Err(AvroError::Decode(DecodeError::I32OutOfRange(i)))
} else {
Ok(i as i32)
}
}
pub fn zag_i64<R: Read>(reader: &mut R) -> Result<i64, AvroError> {
let z = decode_variable(reader)?;
Ok(if z & 0x1 == 0 {
(z >> 1) as i64
} else {
!(z >> 1) as i64
})
}
fn encode_variable(mut z: u64, buffer: &mut Vec<u8>) {
loop {
if z <= 0x7F {
buffer.push((z & 0x7F) as u8);
break;
} else {
buffer.push((0x80 | (z & 0x7F)) as u8);
z >>= 7;
}
}
}
fn decode_variable<R: Read>(reader: &mut R) -> Result<u64, AvroError> {
let mut i = 0u64;
let mut buf = [0u8; 1];
let mut j = 0;
loop {
if j > 9 {
return Err(AvroError::Decode(DecodeError::IntDecodeOverflow));
}
reader.read_exact(&mut buf[..])?;
i |= (u64::from(buf[0] & 0x7F)) << (j * 7);
if (buf[0] >> 7) == 0 {
break;
} else {
j += 1;
}
}
Ok(i)
}
pub fn safe_len(len: usize) -> Result<usize, AvroError> {
if len <= MAX_ALLOCATION_BYTES {
Ok(len)
} else {
Err(AvroError::Allocation {
attempted: len,
allowed: MAX_ALLOCATION_BYTES,
})
}
}
#[cfg(test)]
mod tests {
use super::*;
use mz_ore::assert_err;
#[mz_ore::test]
fn test_zigzag() {
let mut a = Vec::new();
let mut b = Vec::new();
zig_i32(42i32, &mut a);
zig_i64(42i64, &mut b);
assert_eq!(a, b);
}
#[mz_ore::test]
fn test_zig_i64() {
let mut s = Vec::new();
zig_i64(2_147_483_647_i64, &mut s);
assert_eq!(s, [254, 255, 255, 255, 15]);
s.clear();
zig_i64(2_147_483_648_i64, &mut s);
assert_eq!(s, [128, 128, 128, 128, 16]);
s.clear();
zig_i64(-2_147_483_648_i64, &mut s);
assert_eq!(s, [255, 255, 255, 255, 15]);
s.clear();
zig_i64(-2_147_483_649_i64, &mut s);
assert_eq!(s, [129, 128, 128, 128, 16]);
s.clear();
zig_i64(i64::MAX, &mut s);
assert_eq!(s, [254, 255, 255, 255, 255, 255, 255, 255, 255, 1]);
s.clear();
zig_i64(i64::MIN, &mut s);
assert_eq!(s, [255, 255, 255, 255, 255, 255, 255, 255, 255, 1]);
}
#[mz_ore::test]
fn test_zig_i32() {
let mut s = Vec::new();
zig_i32(1_073_741_823_i32, &mut s);
assert_eq!(s, [254, 255, 255, 255, 7]);
s.clear();
zig_i32(-1_073_741_824_i32, &mut s);
assert_eq!(s, [255, 255, 255, 255, 7]);
s.clear();
zig_i32(1_073_741_824_i32, &mut s);
assert_eq!(s, [128, 128, 128, 128, 8]);
s.clear();
zig_i32(-1_073_741_825_i32, &mut s);
assert_eq!(s, [129, 128, 128, 128, 8]);
s.clear();
zig_i32(2_147_483_647_i32, &mut s);
assert_eq!(s, [254, 255, 255, 255, 15]);
s.clear();
zig_i32(-2_147_483_648_i32, &mut s);
assert_eq!(s, [255, 255, 255, 255, 15]);
}
#[mz_ore::test]
fn test_overflow() {
let causes_left_shift_overflow: &[u8] = &[0xe1, 0xe1, 0xe1, 0xe1, 0xe1];
assert_err!(decode_variable(&mut &causes_left_shift_overflow[..]));
}
#[mz_ore::test]
fn test_safe_len() {
assert_eq!(42usize, safe_len(42usize).unwrap());
assert_err!(safe_len(1024 * 1024 * 1024));
}
}