mirror of
https://github.com/google/flatbuffers.git
synced 2026-06-06 21:37:36 +00:00
* Cargo clippy lints * more lints * more lints * Restored a doc comment * Comment on float eps-eq and adjusted casting * Rust Flexbuffers * more serde tests, removed some unsafe * Redid serde to be map-like and Reader is Display * Moved iter from Reader to VectorReader * Serious quickcheck + bugs * wvo api * Made types smaller for a reasonable speedup * redid reading in a way that's a bit faster. Profiling shows the rust slowdown as building +10%, reading +20% * src/bin are developer binaries in rust * Root and Map width are not packed * key null check is debug only + doc changes * BuilderOptions * Documentation * Documentation * Moved tests to rust_usage_test * Moved rust flexbuffers samples to Flatbuffers/samples * Fixed RustTest * Fixed for Rust 1.37.0 * Upgraded to rust 1_40_0 * fixed a little-endian-only feature in a test * 1.40.0 * fixed some benchmarks for bigendian * Updated .bat file * misspelling * Gold Flexbuffer test. * Serialize,Deserialize, std::error::Error for Errors. * Undo rustfmt in integration_test.rs * from_slice instead of from_vec * Added comments to unsafe blocks * expanded on comment * bump Co-authored-by: CasperN <cneo@google.com>
251 lines
7.9 KiB
Rust
251 lines
7.9 KiB
Rust
// Copyright 2019 Google LLC
|
|
//
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
// you may not use this file except in compliance with the License.
|
|
// You may obtain a copy of the License at
|
|
//
|
|
// https://www.apache.org/licenses/LICENSE-2.0
|
|
//
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
// See the License for the specific language governing permissions and
|
|
// limitations under the License.
|
|
|
|
use super::Error;
|
|
use crate::{FlexBufferType, Reader, ReaderIterator};
|
|
use serde::de::{
|
|
DeserializeSeed, Deserializer, EnumAccess, IntoDeserializer, MapAccess, SeqAccess,
|
|
VariantAccess, Visitor,
|
|
};
|
|
|
|
/// Errors that may happen when deserializing a flexbuffer with serde.
|
|
#[derive(Debug, Clone, PartialEq, Eq)]
|
|
pub enum DeserializationError {
|
|
Reader(Error),
|
|
Serde(String),
|
|
}
|
|
|
|
impl std::error::Error for DeserializationError {}
|
|
impl std::fmt::Display for DeserializationError {
|
|
fn fmt(&self, f: &mut std::fmt::Formatter) -> Result<(), std::fmt::Error> {
|
|
match self {
|
|
Self::Reader(r) => write!(f, "Flexbuffer Read Error: {:?}", r),
|
|
Self::Serde(s) => write!(f, "Serde Error: {}", s),
|
|
}
|
|
}
|
|
}
|
|
impl serde::de::Error for DeserializationError {
|
|
fn custom<T>(msg: T) -> Self
|
|
where
|
|
T: std::fmt::Display,
|
|
{
|
|
Self::Serde(format!("{}", msg))
|
|
}
|
|
}
|
|
impl std::convert::From<super::Error> for DeserializationError {
|
|
fn from(e: super::Error) -> Self {
|
|
Self::Reader(e)
|
|
}
|
|
}
|
|
|
|
impl<'de> SeqAccess<'de> for ReaderIterator<'de> {
|
|
type Error = DeserializationError;
|
|
fn next_element_seed<T>(
|
|
&mut self,
|
|
seed: T,
|
|
) -> Result<Option<<T as DeserializeSeed<'de>>::Value>, Self::Error>
|
|
where
|
|
T: DeserializeSeed<'de>,
|
|
{
|
|
if let Some(elem) = self.next() {
|
|
seed.deserialize(elem).map(Some)
|
|
} else {
|
|
Ok(None)
|
|
}
|
|
}
|
|
fn size_hint(&self) -> Option<usize> {
|
|
Some(self.len())
|
|
}
|
|
}
|
|
|
|
struct EnumReader<'de> {
|
|
variant: &'de str,
|
|
value: Option<Reader<'de>>,
|
|
}
|
|
|
|
impl<'de> EnumAccess<'de> for EnumReader<'de> {
|
|
type Error = DeserializationError;
|
|
type Variant = Reader<'de>;
|
|
fn variant_seed<V>(self, seed: V) -> Result<(V::Value, Self::Variant), Self::Error>
|
|
where
|
|
V: DeserializeSeed<'de>,
|
|
{
|
|
seed.deserialize(self.variant.into_deserializer())
|
|
.map(|v| (v, self.value.unwrap_or_default()))
|
|
}
|
|
}
|
|
|
|
struct MapAccessor<'de> {
|
|
keys: ReaderIterator<'de>,
|
|
vals: ReaderIterator<'de>,
|
|
}
|
|
impl<'de> MapAccess<'de> for MapAccessor<'de> {
|
|
type Error = DeserializationError;
|
|
|
|
fn next_key_seed<K>(&mut self, seed: K) -> Result<Option<K::Value>, Self::Error>
|
|
where
|
|
K: DeserializeSeed<'de>,
|
|
{
|
|
if let Some(k) = self.keys.next() {
|
|
seed.deserialize(k).map(Some)
|
|
} else {
|
|
Ok(None)
|
|
}
|
|
}
|
|
fn next_value_seed<V>(&mut self, seed: V) -> Result<V::Value, Self::Error>
|
|
where
|
|
V: DeserializeSeed<'de>,
|
|
{
|
|
let val = self.vals.next().ok_or(Error::IndexOutOfBounds)?;
|
|
seed.deserialize(val)
|
|
}
|
|
}
|
|
|
|
impl<'de> VariantAccess<'de> for Reader<'de> {
|
|
type Error = DeserializationError;
|
|
fn unit_variant(self) -> Result<(), Self::Error> {
|
|
Ok(())
|
|
}
|
|
fn newtype_variant_seed<T>(self, seed: T) -> Result<T::Value, Self::Error>
|
|
where
|
|
T: DeserializeSeed<'de>,
|
|
{
|
|
seed.deserialize(self)
|
|
}
|
|
// Tuple variants have an internally tagged representation. They are vectors where Index 0 is
|
|
// the discriminant and index N is field N-1.
|
|
fn tuple_variant<V>(self, _len: usize, visitor: V) -> Result<V::Value, Self::Error>
|
|
where
|
|
V: Visitor<'de>,
|
|
{
|
|
visitor.visit_seq(self.as_vector().iter())
|
|
}
|
|
// Struct variants have an internally tagged representation. They are vectors where Index 0 is
|
|
// the discriminant and index N is field N-1.
|
|
fn struct_variant<V>(
|
|
self,
|
|
_fields: &'static [&'static str],
|
|
visitor: V,
|
|
) -> Result<V::Value, Self::Error>
|
|
where
|
|
V: Visitor<'de>,
|
|
{
|
|
let m = self.get_map()?;
|
|
visitor.visit_map(MapAccessor {
|
|
keys: m.keys_vector().iter(),
|
|
vals: m.iter_values(),
|
|
})
|
|
}
|
|
}
|
|
|
|
impl<'de> Deserializer<'de> for crate::Reader<'de> {
|
|
type Error = DeserializationError;
|
|
fn deserialize_any<V>(self, visitor: V) -> Result<V::Value, Self::Error>
|
|
where
|
|
V: Visitor<'de>,
|
|
{
|
|
use crate::BitWidth::*;
|
|
use crate::FlexBufferType::*;
|
|
match (self.flexbuffer_type(), self.bitwidth()) {
|
|
(Bool, _) => visitor.visit_bool(self.as_bool()),
|
|
(UInt, W8) => visitor.visit_u8(self.as_u8()),
|
|
(UInt, W16) => visitor.visit_u16(self.as_u16()),
|
|
(UInt, W32) => visitor.visit_u32(self.as_u32()),
|
|
(UInt, W64) => visitor.visit_u64(self.as_u64()),
|
|
(Int, W8) => visitor.visit_i8(self.as_i8()),
|
|
(Int, W16) => visitor.visit_i16(self.as_i16()),
|
|
(Int, W32) => visitor.visit_i32(self.as_i32()),
|
|
(Int, W64) => visitor.visit_i64(self.as_i64()),
|
|
(Float, W32) => visitor.visit_f32(self.as_f32()),
|
|
(Float, W64) => visitor.visit_f64(self.as_f64()),
|
|
(Float, _) => Err(Error::InvalidPackedType.into()), // f8 and f16 are not supported.
|
|
(Null, _) => visitor.visit_unit(),
|
|
(String, _) | (Key, _) => visitor.visit_borrowed_str(self.as_str()),
|
|
(Blob, _) => visitor.visit_borrowed_bytes(self.get_blob()?.0),
|
|
(Map, _) => {
|
|
let m = self.get_map()?;
|
|
visitor.visit_map(MapAccessor {
|
|
keys: m.keys_vector().iter(),
|
|
vals: m.iter_values(),
|
|
})
|
|
}
|
|
(ty, _) if ty.is_vector() => visitor.visit_seq(self.as_vector().iter()),
|
|
(ty, bw) => unreachable!("TODO deserialize_any {:?} {:?}.", ty, bw),
|
|
}
|
|
}
|
|
serde::forward_to_deserialize_any! {
|
|
bool i8 i16 i32 i64 u8 u16 u32 u64 f32 f64 str unit unit_struct bytes
|
|
ignored_any map identifier struct tuple tuple_struct seq string
|
|
}
|
|
fn deserialize_char<V>(self, visitor: V) -> Result<V::Value, Self::Error>
|
|
where
|
|
V: Visitor<'de>,
|
|
{
|
|
visitor.visit_char(self.as_u8() as char)
|
|
}
|
|
fn deserialize_byte_buf<V>(self, visitor: V) -> Result<V::Value, Self::Error>
|
|
where
|
|
V: Visitor<'de>,
|
|
{
|
|
visitor.visit_byte_buf(self.get_blob()?.0.to_vec())
|
|
}
|
|
fn deserialize_option<V>(self, visitor: V) -> Result<V::Value, Self::Error>
|
|
where
|
|
V: Visitor<'de>,
|
|
{
|
|
if self.flexbuffer_type() == FlexBufferType::Null {
|
|
visitor.visit_none()
|
|
} else {
|
|
visitor.visit_some(self)
|
|
}
|
|
}
|
|
fn deserialize_newtype_struct<V>(
|
|
self,
|
|
_name: &'static str,
|
|
visitor: V,
|
|
) -> Result<V::Value, Self::Error>
|
|
where
|
|
V: Visitor<'de>,
|
|
{
|
|
visitor.visit_newtype_struct(self)
|
|
}
|
|
fn deserialize_enum<V>(
|
|
self,
|
|
_name: &'static str,
|
|
_variants: &'static [&'static str],
|
|
visitor: V,
|
|
) -> Result<V::Value, Self::Error>
|
|
where
|
|
V: Visitor<'de>,
|
|
{
|
|
let (variant, value) = match self.fxb_type {
|
|
FlexBufferType::String => (self.as_str(), None),
|
|
FlexBufferType::Map => {
|
|
let m = self.get_map()?;
|
|
let variant = m.keys_vector().idx(0).get_key()?;
|
|
let value = Some(m.idx(0));
|
|
(variant, value)
|
|
}
|
|
_ => {
|
|
return Err(Error::UnexpectedFlexbufferType {
|
|
expected: FlexBufferType::Map,
|
|
actual: self.fxb_type,
|
|
}
|
|
.into());
|
|
}
|
|
};
|
|
visitor.visit_enum(EnumReader { variant, value })
|
|
}
|
|
}
|