Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Construct PortableRegistry dynamically at runtime #164

Merged
merged 51 commits into from
Oct 7, 2022
Merged
Show file tree
Hide file tree
Changes from 1 commit
Commits
Show all changes
51 commits
Select commit Hold shift + click to select a range
8345a3c
WIP: allow constructing MetaTypes at runtime
ascjones Aug 11, 2022
3a14663
Remove fmt config options
ascjones Aug 11, 2022
1e03bd3
Add some docs
ascjones Aug 11, 2022
a68f28c
std feature String
ascjones Aug 11, 2022
da92e9d
fully qualified String
ascjones Aug 11, 2022
592736a
Fix features build
ascjones Aug 11, 2022
4500520
Generic string parameter for MetaType
ascjones Aug 15, 2022
a08ef8a
Remove license_template_path
ascjones Aug 15, 2022
3cacf78
Fmt
ascjones Aug 15, 2022
6476671
Fix fmt
ascjones Aug 15, 2022
6a1343b
Replace MetaFormString trait with cfg based string
ascjones Aug 16, 2022
42db0e3
Remove unused into
ascjones Aug 16, 2022
0c525c6
Fix more errors
ascjones Aug 16, 2022
7dfe55b
Now string types are the same across Meta/Portable, no longer necessa…
ascjones Aug 17, 2022
54ff5fc
Remove custom MetaType and TypeId
ascjones Aug 22, 2022
16e4e79
EXPERIMENT: make all type def fields public to allow construting Port…
ascjones Aug 22, 2022
c46186d
Rename test
ascjones Aug 22, 2022
5b47b1b
Fix up type ids in test
ascjones Aug 22, 2022
40b6047
Fix ui test
ascjones Aug 22, 2022
8255c99
*TEMPORARILY* pub registry for PortableType
ascjones Aug 23, 2022
79c19a9
add constructor for portableregistry
xermicus Aug 28, 2022
d8c3b5e
constructor for PortableType
xermicus Aug 28, 2022
0e4737f
implement remaining constructors for generic Form
xermicus Aug 28, 2022
8319875
make Type::new constructor public
xermicus Aug 28, 2022
ef4c3ee
make remaining constructor public
xermicus Aug 28, 2022
9e48542
add a new custom constructor for path
xermicus Sep 22, 2022
b900f54
Merge branch 'master' into aj/custom-meta-types
xermicus Sep 22, 2022
0b45a89
use new_custom in a test
xermicus Sep 22, 2022
6694dc0
Remove not required builder fn for MetaType
ascjones Sep 22, 2022
83a80ef
WIP propogate Form through builders
ascjones Oct 4, 2022
ae22be7
Fix up path builders
ascjones Oct 4, 2022
ababa57
More fixes
ascjones Oct 4, 2022
383399c
Fix up generated From impls
ascjones Oct 4, 2022
221c2f9
Fix up field builders
ascjones Oct 4, 2022
6caa3d6
Fix up field builder methods for MetaForm
ascjones Oct 4, 2022
a36540a
Rename constructor
ascjones Oct 4, 2022
b027ae8
Hide pub fields and add portable builder helper methods
ascjones Oct 4, 2022
1de0508
Fmt
ascjones Oct 4, 2022
f04d4ef
Explicity export PortableType
ascjones Oct 4, 2022
23c6cb9
Clippy
ascjones Oct 4, 2022
590bc58
Move PortableRegistry to own file, introduce PortableRegistryBuilder
ascjones Oct 5, 2022
b5b5a6f
Remove some stray `Str` type params
ascjones Oct 5, 2022
3c9d8dd
Implement PortableRegistryBuilder to manage ids
ascjones Oct 5, 2022
a2668a9
Default impl for PortableRegistryBuilder
ascjones Oct 5, 2022
220a59d
implement getter for registered types in PortableRegistryBuilder
xermicus Oct 6, 2022
07057ce
Rename new path constructor
ascjones Oct 7, 2022
07aaef8
Rename new_custom methods to new_portable
ascjones Oct 7, 2022
192d12c
Revert reordering of MetaForm
ascjones Oct 7, 2022
403c720
Move path construction back to MetaForm for non-breaking changes
ascjones Oct 7, 2022
1044c8d
Update path tests
ascjones Oct 7, 2022
c9149c2
TypeParameter::new_portable for non breaking change
ascjones Oct 7, 2022
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Next Next commit
WIP: allow constructing MetaTypes at runtime
  • Loading branch information
ascjones committed Aug 11, 2022
commit 8345a3c80795e2266b17dbbad582ae438ae8afd4
256 changes: 177 additions & 79 deletions src/build.rs

Large diffs are not rendered by default.

26 changes: 18 additions & 8 deletions src/form.rs
Original file line number Diff line number Diff line change
Expand Up @@ -29,14 +29,12 @@
//! Other forms, such as a portable form that is still bound to the registry
//! (also via lifetime tracking) are possible but current not needed.

use crate::prelude::{
any::TypeId,
fmt::Debug,
};
use crate::prelude::fmt::Debug;

use crate::{
interner::UntrackedSymbol,
meta_type::MetaType,
TypeId,
};

#[cfg(feature = "serde")]
Expand All @@ -51,20 +49,32 @@ pub trait Form {
/// The type representing the type.
type Type: PartialEq + Eq + PartialOrd + Ord + Clone + Debug;
/// The string type.
type String: AsRef<str> + PartialEq + Eq + PartialOrd + Ord + Clone + Debug;
type String: FormString;
}

/// todo: docs
pub trait FormString:
AsRef<str> + PartialEq + Eq + PartialOrd + Ord + Clone + Debug
{
}

impl FormString for &'static str {}
impl FormString for String {}

/// A meta meta-type.
///
/// Allows to be converted into other forms such as portable form
/// through the registry and `IntoPortable`.
#[cfg_attr(feature = "serde", derive(Serialize))]
#[derive(PartialEq, Eq, PartialOrd, Ord, Clone, Copy, Debug)]
pub enum MetaForm {}
pub struct MetaForm<S: FormString = &'static str>(core::marker::PhantomData<S>);

impl Form for MetaForm {
impl<S> Form for MetaForm<S>
where
S: FormString,
{
type Type = MetaType;
type String = &'static str;
type String = S;
}

/// Portable form that has its lifetime untracked in association to its interner.
Expand Down
19 changes: 19 additions & 0 deletions src/lib.rs
Original file line number Diff line number Diff line change
Expand Up @@ -355,6 +355,25 @@ pub use self::{
ty::*,
};

/// todo: docs
#[derive(Clone, Copy, Debug, Eq, PartialEq, Hash, Ord, PartialOrd)]
pub enum TypeId {
/// todo: docs
Rust(core::any::TypeId),
/// todo: docs
Custom(u64),
}

impl TypeId {
/// todo: docs
pub fn of<T>() -> Self
where
T: ?Sized + 'static,
{
Self::Rust(core::any::TypeId::of::<T>())
}
}

#[cfg(feature = "derive")]
pub use scale_info_derive::TypeInfo;

Expand Down
10 changes: 9 additions & 1 deletion src/meta_type.rs
Original file line number Diff line number Diff line change
Expand Up @@ -13,7 +13,6 @@
// limitations under the License.

use crate::prelude::{
any::TypeId,
cmp::Ordering,
fmt::{
Debug,
Expand All @@ -29,6 +28,7 @@ use crate::prelude::{
use crate::{
form::MetaForm,
Type,
TypeId,
TypeInfo,
};

Expand Down Expand Up @@ -96,6 +96,14 @@ impl MetaType {
}
}

/// todo: docs
pub fn new_custom(type_id: u64, fn_type_info: fn() -> Type<MetaForm>) -> Self {
ascjones marked this conversation as resolved.
Show resolved Hide resolved
Self {
fn_type_info,
type_id: TypeId::Custom(type_id),
}
}

/// Returns the meta type information.
pub fn type_info(&self) -> Type<MetaForm> {
(self.fn_type_info)()
Expand Down
4 changes: 2 additions & 2 deletions src/registry.rs
Original file line number Diff line number Diff line change
Expand Up @@ -24,7 +24,6 @@
//! prelude types that live in the so-called root namespace which is empty.

use crate::prelude::{
any::TypeId,
collections::BTreeMap,
fmt::Debug,
vec::Vec,
Expand All @@ -41,6 +40,7 @@ use crate::{
},
meta_type::MetaType,
Type,
TypeId,
};
use scale::Encode;

Expand Down Expand Up @@ -84,7 +84,7 @@ pub struct Registry {
/// The database where registered types reside.
///
/// The contents herein is used for serlialization.
types: BTreeMap<UntrackedSymbol<core::any::TypeId>, Type<PortableForm>>,
types: BTreeMap<UntrackedSymbol<TypeId>, Type<PortableForm>>,
}

impl Default for Registry {
Expand Down
21 changes: 21 additions & 0 deletions src/tests.rs
Original file line number Diff line number Diff line change
Expand Up @@ -14,6 +14,7 @@

use crate::{
build::*,
form::MetaForm,
prelude::{
borrow::Cow,
boxed::Box,
Expand Down Expand Up @@ -363,3 +364,23 @@ fn basic_enum_with_index() {

assert_type!(IndexedRustEnum, ty);
}

#[test]
fn runtime_meta_type() {
let mut fields = Fields::named();
let custom_u32_id = 1;
let custom_u32_type_info = <u32 as TypeInfo>::type_info;
for i in 0..3 {
fields.field_mut(|f| {
f.ty_meta(MetaType::new_custom(custom_u32_id, custom_u32_type_info))
.name(i.to_string())
.type_name("custom_u32".to_string())
})
}
let ty = TypeBuilder::<String>::default()
.path(
Path::<MetaForm<String>>::from_segments(vec!["MyCustomStruct".to_string()])
.unwrap(),
)
.composite(fields);
}
7 changes: 5 additions & 2 deletions src/ty/composite.rs
Original file line number Diff line number Diff line change
Expand Up @@ -89,11 +89,14 @@ impl IntoPortable for TypeDefComposite {
}
}

impl TypeDefComposite {
impl<T> TypeDefComposite<T>
where
T: Form,
{
/// Creates a new struct definition with named fields.
pub(crate) fn new<I>(fields: I) -> Self
where
I: IntoIterator<Item = Field>,
I: IntoIterator<Item = Field<T>>,
{
Self {
fields: fields.into_iter().collect(),
Expand Down
20 changes: 13 additions & 7 deletions src/ty/fields.rs
Original file line number Diff line number Diff line change
Expand Up @@ -16,12 +16,12 @@ use crate::{
build::FieldBuilder,
form::{
Form,
FormString,
MetaForm,
PortableForm,
},
prelude::vec::Vec,
IntoPortable,
MetaType,
Registry,
};
use scale::Encode;
Expand Down Expand Up @@ -110,20 +110,26 @@ impl IntoPortable for Field {
}
}

impl Field {
impl<T> Field<T>
where
T: Form,
{
/// Returns a new [`FieldBuilder`] for constructing a field.
pub fn builder() -> FieldBuilder {
pub fn builder<'a, Str>() -> FieldBuilder<'a, Str>
where
Str: FormString,
{
FieldBuilder::new()
}

/// Creates a new field.
///
/// Use this constructor if you want to instantiate from a given meta type.
pub fn new(
name: Option<&'static str>,
ty: MetaType,
type_name: Option<&'static str>,
docs: &[&'static str],
name: Option<T::String>,
ty: T::Type,
type_name: Option<T::String>,
docs: &[T::String],
) -> Self {
Self {
name,
Expand Down
17 changes: 11 additions & 6 deletions src/ty/mod.rs
Original file line number Diff line number Diff line change
Expand Up @@ -120,19 +120,24 @@ impl_from_type_def_for_type!(

impl Type {
/// Create a [`TypeBuilder`](`crate::build::TypeBuilder`) the public API for constructing a [`Type`]
pub fn builder() -> TypeBuilder {
pub fn builder() -> TypeBuilder<&'static str> {
TypeBuilder::default()
}
}

impl<F> Type<F>
where
F: Form,
{
pub(crate) fn new<I, D>(
path: Path,
path: Path<F>,
type_params: I,
type_def: D,
docs: Vec<&'static str>,
) -> Self
docs: Vec<F::String>,
) -> Type<F>
where
I: IntoIterator<Item = TypeParameter>,
D: Into<TypeDef>,
I: IntoIterator<Item = TypeParameter<F>>,
D: Into<TypeDef<F>>,
{
Self {
path,
Expand Down
30 changes: 21 additions & 9 deletions src/ty/path.rs
Original file line number Diff line number Diff line change
Expand Up @@ -25,6 +25,7 @@ use crate::prelude::{
use crate::{
form::{
Form,
FormString,
MetaForm,
PortableForm,
},
Expand Down Expand Up @@ -102,10 +103,15 @@ impl Path {
Self::from_segments(segments)
.expect("All path segments should be valid Rust identifiers")
}
}

impl<Str> Path<MetaForm<Str>>
where
Str: FormString,
{
/// Create an empty path for types which shall not be named
#[allow(unused)]
pub(crate) fn voldemort() -> Path {
pub(crate) fn voldemort() -> Path<MetaForm<Str>> {
Path {
segments: Vec::new(),
}
Expand All @@ -116,9 +122,9 @@ impl Path {
/// # Panics
///
/// - If the supplied ident is not a valid Rust identifier
pub(crate) fn prelude(ident: &'static str) -> Path {
Self::from_segments(vec![ident])
.unwrap_or_else(|_| panic!("{} is not a valid Rust identifier", ident))
pub(crate) fn prelude(ident: Str) -> Path<MetaForm<Str>> {
Self::from_segments(vec![ident.clone()])
.unwrap_or_else(|_| panic!("{:?} is not a valid Rust identifier", ident))
}

/// Create a Path from the given segments
Expand All @@ -127,15 +133,18 @@ impl Path {
///
/// - If no segments are supplied
/// - If any of the segments are invalid Rust identifiers
pub fn from_segments<I>(segments: I) -> Result<Path, PathError>
pub fn from_segments<I>(segments: I) -> Result<Path<MetaForm<Str>>, PathError>
where
I: IntoIterator<Item = &'static str>,
I: IntoIterator<Item = Str>,
{
let segments = segments.into_iter().collect::<Vec<_>>();
if segments.is_empty() {
return Err(PathError::MissingSegments)
}
if let Some(err_at) = segments.iter().position(|seg| !is_rust_identifier(seg)) {
if let Some(err_at) = segments
.iter()
.position(|seg| !is_rust_identifier(seg.as_ref()))
{
return Err(PathError::InvalidIdentifier { segment: err_at })
}
Ok(Path { segments })
Expand Down Expand Up @@ -186,7 +195,7 @@ mod tests {
#[test]
fn path_ok() {
assert_eq!(
Path::from_segments(vec!["hello"]),
Path::<MetaForm>::from_segments(vec!["hello"]),
ascjones marked this conversation as resolved.
Show resolved Hide resolved
Ok(Path {
segments: vec!["hello"]
})
Expand Down Expand Up @@ -217,7 +226,10 @@ mod tests {

#[test]
fn path_err() {
assert_eq!(Path::from_segments(vec![]), Err(PathError::MissingSegments));
assert_eq!(
Path::<MetaForm>::from_segments(vec![]),
Err(PathError::MissingSegments)
);
assert_eq!(
Path::from_segments(vec![""]),
Err(PathError::InvalidIdentifier { segment: 0 })
Expand Down
18 changes: 12 additions & 6 deletions src/ty/variant.rs
Original file line number Diff line number Diff line change
Expand Up @@ -101,11 +101,14 @@ impl IntoPortable for TypeDefVariant {
}
}

impl TypeDefVariant {
impl<T> TypeDefVariant<T>
where
T: Form,
{
/// Create a new `TypeDefVariant` with the given variants
pub fn new<I>(variants: I) -> Self
where
I: IntoIterator<Item = Variant>,
I: IntoIterator<Item = Variant<T>>,
{
Self {
variants: variants.into_iter().collect(),
Expand Down Expand Up @@ -184,13 +187,16 @@ impl IntoPortable for Variant {
}
}

impl Variant {
impl<T> Variant<T>
where
T: Form,
{
/// Creates a new variant.
pub(crate) fn new(
name: &'static str,
fields: Vec<Field<MetaForm>>,
name: T::String,
fields: Vec<Field<T>>,
index: u8,
docs: Vec<&'static str>,
docs: Vec<T::String>,
) -> Self {
Self {
name,
Expand Down