Skip to content

Commit

Permalink
Merge Access and AccessRef using a Cow
Browse files Browse the repository at this point in the history
mooooo
  • Loading branch information
nicopap committed Jul 5, 2023
1 parent b100069 commit f081fe1
Show file tree
Hide file tree
Showing 2 changed files with 73 additions and 89 deletions.
122 changes: 48 additions & 74 deletions crates/bevy_reflect/src/path/access.rs
Original file line number Diff line number Diff line change
@@ -1,4 +1,4 @@
use std::fmt;
use std::{borrow::Cow, fmt};

use super::{AccessError, ReflectPathError};
use crate::{Reflect, ReflectMut, ReflectRef, VariantType};
Expand All @@ -13,10 +13,7 @@ pub(super) enum Error<'a> {
access.kind(),
access.display_value(),
)]
Access {
ty: TypeShape,
access: AccessRef<'a>,
},
Access { ty: TypeShape, access: Access<'a> },

#[error("invalid type shape: expected {expected} but found a reflect {actual}")]
Type {
Expand All @@ -36,6 +33,10 @@ impl<'a> Error<'a> {
let error = AccessError(self);
ReflectPathError::InvalidAccess { offset, error }
}

fn access(ty: TypeShape, access: Access<'a>) -> Self {
Self::Access { ty, access }
}
}
impl Error<'static> {
fn bad_enum_variant(expected: TypeShape, actual: impl Into<TypeShape>) -> Self {
Expand Down Expand Up @@ -101,20 +102,7 @@ impl From<VariantType> for TypeShape {
}
}

/// A singular owned element access within a path.
///
/// Can be applied to a `dyn Reflect` to get a reference to the targeted element.
///
/// A path is composed of multiple accesses in sequence.
#[derive(Debug, Clone, PartialEq, Eq, Hash, PartialOrd, Ord)]
pub(super) enum Access {
Field(Box<str>),
FieldIndex(usize),
TupleIndex(usize),
ListIndex(usize),
}

impl fmt::Display for Access {
impl fmt::Display for Access<'_> {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
match self {
Access::Field(field) => write!(f, ".{field}"),
Expand All @@ -125,35 +113,21 @@ impl fmt::Display for Access {
}
}

impl Access {
pub(super) fn as_ref(&self) -> AccessRef<'_> {
match self {
Self::Field(value) => AccessRef::Field(value),
Self::FieldIndex(value) => AccessRef::FieldIndex(*value),
Self::TupleIndex(value) => AccessRef::TupleIndex(*value),
Self::ListIndex(value) => AccessRef::ListIndex(*value),
}
}
}

/// A singular borrowed element access within a path.
/// A singular element access within a path.
///
/// Can be applied to a `dyn Reflect` to get a reference to the targeted element.
///
/// Does not own the backing store it's sourced from.
/// For an owned version, you can convert one to an [`Access`] with [`AccessRef::to_owned`].
#[derive(Debug, Clone, Copy, PartialEq, Eq)]
pub(super) enum AccessRef<'a> {
Field(&'a str),
#[derive(Debug, Clone, PartialEq, Eq, PartialOrd, Ord, Hash)]
pub(super) enum Access<'a> {
Field(Cow<'a, str>),
FieldIndex(usize),
TupleIndex(usize),
ListIndex(usize),
}

impl<'a> AccessRef<'a> {
pub(super) fn to_owned(self) -> Access {
impl<'a> Access<'a> {
pub(super) fn into_owned(self) -> Access<'static> {
match self {
Self::Field(value) => Access::Field(value.to_string().into_boxed_str()),
Self::Field(value) => Access::Field(value.to_string().into()),
Self::FieldIndex(value) => Access::FieldIndex(value),
Self::TupleIndex(value) => Access::TupleIndex(value),
Self::ListIndex(value) => Access::ListIndex(value),
Expand All @@ -174,78 +148,78 @@ impl<'a> AccessRef<'a> {
}
}

pub(super) fn element(
self,
base: &dyn Reflect,
pub(super) fn element<'r>(
&self,
base: &'r dyn Reflect,
offset: usize,
) -> Result<&dyn Reflect, ReflectPathError<'a>> {
) -> Result<&'r dyn Reflect, ReflectPathError<'a>> {
let ty = base.reflect_ref().into();
self.element_inner(base)
.and_then(|maybe| maybe.ok_or(Error::Access { ty, access: self }))
.and_then(|maybe| maybe.ok_or(Error::access(ty, self.clone())))
.map_err(|err| err.with_offset(offset))
}

fn element_inner(self, base: &dyn Reflect) -> InnerResult<&dyn Reflect> {
fn element_inner<'r>(&self, base: &'r dyn Reflect) -> InnerResult<&'r dyn Reflect> {
use ReflectRef::*;
match (self, base.reflect_ref()) {
(Self::Field(field), Struct(struct_ref)) => Ok(struct_ref.field(field)),
(Self::Field(field), Struct(struct_ref)) => Ok(struct_ref.field(field.as_ref())),
(Self::Field(field), Enum(enum_ref)) => match enum_ref.variant_type() {
VariantType::Struct => Ok(enum_ref.field(field)),
VariantType::Struct => Ok(enum_ref.field(field.as_ref())),
actual => Err(Error::bad_enum_variant(TypeShape::Struct, actual)),
},
(Self::FieldIndex(index), Struct(struct_ref)) => Ok(struct_ref.field_at(index)),
(Self::FieldIndex(index), Enum(enum_ref)) => match enum_ref.variant_type() {
(&Self::FieldIndex(index), Struct(struct_ref)) => Ok(struct_ref.field_at(index)),
(&Self::FieldIndex(index), Enum(enum_ref)) => match enum_ref.variant_type() {
VariantType::Struct => Ok(enum_ref.field_at(index)),
actual => Err(Error::bad_enum_variant(TypeShape::Struct, actual)),
},
(Self::TupleIndex(index), TupleStruct(tuple)) => Ok(tuple.field(index)),
(Self::TupleIndex(index), Tuple(tuple)) => Ok(tuple.field(index)),
(Self::TupleIndex(index), Enum(enum_ref)) => match enum_ref.variant_type() {
(&Self::TupleIndex(index), TupleStruct(tuple)) => Ok(tuple.field(index)),
(&Self::TupleIndex(index), Tuple(tuple)) => Ok(tuple.field(index)),
(&Self::TupleIndex(index), Enum(enum_ref)) => match enum_ref.variant_type() {
VariantType::Tuple => Ok(enum_ref.field_at(index)),
actual => Err(Error::bad_enum_variant(TypeShape::Tuple, actual)),
},
(Self::ListIndex(index), List(list)) => Ok(list.get(index)),
(Self::ListIndex(index), Array(list)) => Ok(list.get(index)),
(Self::ListIndex(_), actual) => Err(Error::bad_type(TypeShape::List, actual)),
(&Self::ListIndex(index), List(list)) => Ok(list.get(index)),
(&Self::ListIndex(index), Array(list)) => Ok(list.get(index)),
(&Self::ListIndex(_), actual) => Err(Error::bad_type(TypeShape::List, actual)),
(_, actual) => Err(Error::bad_type(TypeShape::Struct, actual)),
}
}

pub(super) fn element_mut(
self,
base: &mut dyn Reflect,
pub(super) fn element_mut<'r>(
&self,
base: &'r mut dyn Reflect,
offset: usize,
) -> Result<&mut dyn Reflect, ReflectPathError<'a>> {
) -> Result<&'r mut dyn Reflect, ReflectPathError<'a>> {
let ty = base.reflect_ref().into();
self.element_inner_mut(base)
.and_then(|maybe| maybe.ok_or(Error::Access { ty, access: self }))
.and_then(|maybe| maybe.ok_or(Error::access(ty, self.clone())))
.map_err(|err| err.with_offset(offset))
}

fn element_inner_mut(self, base: &mut dyn Reflect) -> InnerResult<&mut dyn Reflect> {
fn element_inner_mut<'r>(&self, base: &'r mut dyn Reflect) -> InnerResult<&'r mut dyn Reflect> {
use ReflectMut::*;
let base_kind: TypeShape = base.reflect_ref().into();
let base_shape: TypeShape = base.reflect_ref().into();
match (self, base.reflect_mut()) {
(Self::Field(field), Struct(struct_mut)) => Ok(struct_mut.field_mut(field)),
(Self::Field(field), Struct(struct_mut)) => Ok(struct_mut.field_mut(field.as_ref())),
(Self::Field(field), Enum(enum_mut)) => match enum_mut.variant_type() {
VariantType::Struct => Ok(enum_mut.field_mut(field)),
VariantType::Struct => Ok(enum_mut.field_mut(field.as_ref())),
actual => Err(Error::bad_enum_variant(TypeShape::Struct, actual)),
},
(Self::FieldIndex(index), Struct(struct_mut)) => Ok(struct_mut.field_at_mut(index)),
(Self::FieldIndex(index), Enum(enum_mut)) => match enum_mut.variant_type() {
(&Self::FieldIndex(index), Struct(struct_mut)) => Ok(struct_mut.field_at_mut(index)),
(&Self::FieldIndex(index), Enum(enum_mut)) => match enum_mut.variant_type() {
VariantType::Struct => Ok(enum_mut.field_at_mut(index)),
actual => Err(Error::bad_enum_variant(TypeShape::Struct, actual)),
},
(Self::TupleIndex(index), TupleStruct(tuple)) => Ok(tuple.field_mut(index)),
(Self::TupleIndex(index), Tuple(tuple)) => Ok(tuple.field_mut(index)),
(Self::TupleIndex(index), Enum(enum_mut)) => match enum_mut.variant_type() {
(&Self::TupleIndex(index), TupleStruct(tuple)) => Ok(tuple.field_mut(index)),
(&Self::TupleIndex(index), Tuple(tuple)) => Ok(tuple.field_mut(index)),
(&Self::TupleIndex(index), Enum(enum_mut)) => match enum_mut.variant_type() {
VariantType::Tuple => Ok(enum_mut.field_at_mut(index)),
actual => Err(Error::bad_enum_variant(TypeShape::Tuple, actual)),
},
(Self::ListIndex(index), List(list)) => Ok(list.get_mut(index)),
(Self::ListIndex(index), Array(list)) => Ok(list.get_mut(index)),
(Self::ListIndex(_), _) => Err(Error::bad_type(TypeShape::List, base_kind)),
(_, _) => Err(Error::bad_type(TypeShape::Struct, base_kind)),
(&Self::ListIndex(index), List(list)) => Ok(list.get_mut(index)),
(&Self::ListIndex(index), Array(list)) => Ok(list.get_mut(index)),
(&Self::ListIndex(_), _) => Err(Error::bad_type(TypeShape::List, base_shape)),
(_, _) => Err(Error::bad_type(TypeShape::Struct, base_shape)),
}
}
}
40 changes: 25 additions & 15 deletions crates/bevy_reflect/src/path/mod.rs
Original file line number Diff line number Diff line change
Expand Up @@ -4,7 +4,7 @@ use std::fmt;
use std::num::ParseIntError;

use crate::Reflect;
use access::{Access, AccessRef};
use access::Access;
use thiserror::Error;

type ParseResult<T> = Result<T, ReflectPathParseError>;
Expand Down Expand Up @@ -291,7 +291,7 @@ pub struct ParsedPath(
/// index of the start of the access within the parsed path string.
///
/// The index is mainly used for more helpful error reporting.
Box<[(Access, usize)]>,
Box<[(Access<'static>, usize)]>,
);

impl ParsedPath {
Expand Down Expand Up @@ -343,7 +343,17 @@ impl ParsedPath {
pub fn parse(string: &str) -> Result<Self, ReflectPathError<'_>> {
let mut parts = Vec::new();
for (access, idx) in PathParser::new(string) {
parts.push((access?.to_owned(), idx));
parts.push((access?.into_owned(), idx));
}
Ok(Self(parts.into_boxed_slice()))
}

/// Similar to [`Self::parse`] but only works on `&'static str`
/// and does not allocate per named field.
pub fn parse_static(string: &'static str) -> Result<Self, ReflectPathError<'_>> {
let mut parts = Vec::new();
for (access, idx) in PathParser::new(string) {
parts.push((access?, idx));
}
Ok(Self(parts.into_boxed_slice()))
}
Expand All @@ -359,7 +369,7 @@ impl ParsedPath {
) -> Result<&'r dyn Reflect, ReflectPathError<'p>> {
let mut current = root;
for (access, current_index) in self.0.iter() {
current = access.as_ref().element(current, *current_index)?;
current = access.element(current, *current_index)?;
}
Ok(current)
}
Expand All @@ -375,7 +385,7 @@ impl ParsedPath {
) -> Result<&'r mut dyn Reflect, ReflectPathError<'p>> {
let mut current = root;
for (access, current_index) in self.0.iter() {
current = access.as_ref().element_mut(current, *current_index)?;
current = access.element_mut(current, *current_index)?;
}
Ok(current)
}
Expand Down Expand Up @@ -471,15 +481,15 @@ impl<'a> PathParser<'a> {
Some(ident)
}

fn token_to_access(&mut self, token: Token<'a>) -> ParseResult<AccessRef<'a>> {
fn token_to_access(&mut self, token: Token<'a>) -> ParseResult<Access<'a>> {
let current_offset = self.index;
match token {
Token::Dot => {
if let Some(Token::Ident(value)) = self.next_token() {
value
.parse::<usize>()
.map(AccessRef::TupleIndex)
.or(Ok(AccessRef::Field(value)))
.map(Access::TupleIndex)
.or(Ok(Access::Field(value.into())))
} else {
Err(ReflectPathParseError::ExpectedIdent {
offset: current_offset,
Expand All @@ -488,7 +498,7 @@ impl<'a> PathParser<'a> {
}
Token::CrossHatch => {
if let Some(Token::Ident(value)) = self.next_token() {
Ok(AccessRef::FieldIndex(value.parse::<usize>()?))
Ok(Access::FieldIndex(value.parse::<usize>()?))
} else {
Err(ReflectPathParseError::ExpectedIdent {
offset: current_offset,
Expand All @@ -497,7 +507,7 @@ impl<'a> PathParser<'a> {
}
Token::OpenBracket => {
let access = if let Some(Token::Ident(value)) = self.next_token() {
AccessRef::ListIndex(value.parse::<usize>()?)
Access::ListIndex(value.parse::<usize>()?)
} else {
return Err(ReflectPathParseError::ExpectedIdent {
offset: current_offset,
Expand All @@ -519,14 +529,14 @@ impl<'a> PathParser<'a> {
}),
Token::Ident(value) => value
.parse::<usize>()
.map(AccessRef::TupleIndex)
.or(Ok(AccessRef::Field(value))),
.map(Access::TupleIndex)
.or(Ok(Access::Field(value.into()))),
}
}
}

impl<'a> Iterator for PathParser<'a> {
type Item = (ParseResult<AccessRef<'a>>, usize);
type Item = (ParseResult<Access<'a>>, usize);

fn next(&mut self) -> Option<Self::Item> {
let token = self.next_token()?;
Expand Down Expand Up @@ -615,7 +625,7 @@ mod tests {
}

fn access_field(field: &'static str) -> Access {
Access::Field(field.to_string().into_boxed_str())
Access::Field(field.into())
}

#[test]
Expand Down Expand Up @@ -778,7 +788,7 @@ mod tests {
offset: 2,
error: AccessError(access::Error::Access {
ty: TypeShape::Struct,
access: AccessRef::Field("notreal"),
access: access_field("notreal"),
}),
}
);
Expand Down

0 comments on commit f081fe1

Please sign in to comment.