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

Reference key storage api #447

Merged
merged 37 commits into from
Mar 2, 2022
Merged

Conversation

lexnv
Copy link
Collaborator

@lexnv lexnv commented Feb 14, 2022

A StorageEntryType::Plain type requires no changes as it receives no arguments.

Changes

  1. Structs receive a named lifetime and take a reference to what they previously contained.
    If they contain multiple fields, all are bounded to the same lifetime.
    With the exception of the storage::Account, which is an associated type for DefaultAccountData trait's implementation.
pub struct PendingSwap(pub runtime_types::polkadot_parachain::primitives::Id)

pub struct PendingSwap<'a>(pub &'a runtime_types::polkadot_parachain::primitives::Id);
  1. StorageEntry trait is implemented for an anonymous lifetime.
impl ::subxt::StorageEntry for PendingSwap {

impl ::subxt::StorageEntry for PendingSwap<'_> {
  1. Storage API uses reference for key arguments
pub async fn pending_swap(
  &self,
  _0: runtime_types::polkadot_parachain::primitives::Id,
  hash: ::core::option::Option<T::Hash>,
)

pub async fn pending_swap(
  &self,
  _0: &runtime_types::polkadot_parachain::primitives::Id,
  hash: ::core::option::Option<T::Hash>,
)
  1. Vector references are replaced by slices
_0: &::std::vec::Vec<::core::primitive::u8>

0: &[::core::primitive::u8]
  1. Lastly, the iter methods bound the lifetime of the generated structs to the named lifetime of the StorageClient.
pub async fn pending_swap_iter(
    &self,
    hash: ::core::option::Option<T::Hash>,
) -> ::core::result::Result<
    ::subxt::KeyIter<'a, T, PendingSwap>,
    ::subxt::BasicError,
> {

pub async fn pending_swap_iter(
    &self,
    hash: ::core::option::Option<T::Hash>,
) -> ::core::result::Result<
    ::subxt::KeyIter<'a, T, PendingSwap<'a>>,
    ::subxt::BasicError,
> {

closes #411.

Example Output

Staking active era: index: 1, start: Some(1645114260002)
  Alice account id:        d43593c715fdd31c61141abd04a99fd6822c8558854ccde39a5684e7a56da27d (5GrwvaEF...)
  Alice//stash account id: be5ddb1579b72e84524fc29e78609e3caf42e85aa118ebfe0b0ad404b5bdd25f (5GNJqTPy...)
    account controlled by: d43593c715fdd31c61141abd04a99fd6822c8558854ccde39a5684e7a56da27d (5GrwvaEF...)
Era reward points: EraRewardPoints { total: 20740, individual: {be5ddb1579b72e84524fc29e78609e3caf42e85aa118ebfe0b0ad404b5bdd25f (5GNJqTPy...): 20740} }

Next Steps

  • Fix clippy due to &Vec<_>
  • Add storage example

polkadot commit-hash: d96d3bea85
polkadot tag: v0.9.16-rc2

Signed-off-by: Alexandru Vasile <alexandru.vasile@parity.io>
Signed-off-by: Alexandru Vasile <alexandru.vasile@parity.io>
Signed-off-by: Alexandru Vasile <alexandru.vasile@parity.io>
Signed-off-by: Alexandru Vasile <alexandru.vasile@parity.io>
Signed-off-by: Alexandru Vasile <alexandru.vasile@parity.io>
Signed-off-by: Alexandru Vasile <alexandru.vasile@parity.io>
cli/src/main.rs Outdated Show resolved Hide resolved
This reverts commit 2970d05.

Revert "cli: Fix polkadot.rs license check"

This reverts commit 6fe8818.
}
_ => {
let should_ref = storage_entry.name != "Account";
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

So I think this is the case because of the DefaultAccountData thing that's generated not taking a ref?

I don't really like the inconsistency this would introduce in the storage API (this call doesn't take a ref but other similar calls do). I think it'd be good to see whether the generated DefaultAccountData impl can be tweaked to take/allow the ref. @ascjones any thoughts on this?

Copy link
Collaborator

@jsdw jsdw left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Looking good overall! I'm wondering whether we can avoid the inconsistency/hardcoded string w.r.t that Account entry though :)

Signed-off-by: Alexandru Vasile <alexandru.vasile@parity.io>
@jsdw
Copy link
Collaborator

jsdw commented Feb 15, 2022

So I had a bit of a dig into this area, and it looks like the purpose of AccountData is basically to generate the correct storage key, and then decode a nonce from the storage value that comes back when node storage is queried with that key (the nonce being the number of transactions from the account, and is needed to be provided with each transaction).

See

A::storage_entry(signer.account_id().clone().into());
.

By default this trait is implemented for some default enum which points at the Account storage key (see

impl ::subxt::AccountData for #default_impl_name {
). This makes it hard or impossible to use an Account struct that has a lifetime at the mo, which is why you currently are not adding a lifetime to a struct that looks like "Account".

In order to have a consistent storage API where everything provided is a reference (rather than having some exceptional thing that is not a reference), one approach I think would be to generate a version of the "system::Account" type with a reference to be consistent with the rest of the API, but to also generate a version of it without a reference. This non-reference version could then be used by that default AccountData impl instead :)

You'll need to look for the correct type to do this with sortof like you do now, but it's worth making sure that the entire path of the type is correct (ie it is ["system", "Account"], and not for instance ["balances", "Account"]) (I had to do the same thing here:

.find(|&ty| ty.ty().path().segments() == ["sp_runtime", "DispatchError"])
).

@ascjones does that sound reasonable to you?

Signed-off-by: Alexandru Vasile <alexandru.vasile@parity.io>
Signed-off-by: Alexandru Vasile <alexandru.vasile@parity.io>
Signed-off-by: Alexandru Vasile <alexandru.vasile@parity.io>
Signed-off-by: Alexandru Vasile <alexandru.vasile@parity.io>
Signed-off-by: Alexandru Vasile <alexandru.vasile@parity.io>
Signed-off-by: Alexandru Vasile <alexandru.vasile@parity.io>
Signed-off-by: Alexandru Vasile <alexandru.vasile@parity.io>
let ty_path = type_gen.resolve_type_path(key.id(), &[]);
let fields = vec![(format_ident!("_0"), ty_path.clone())];
// `::system::storage::Account` was utilized as associated type `StorageEntry`
// for `::subxt::AccountData` implementation of generated `DefaultAccountData`.
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Suggested change
// for `::subxt::AccountData` implementation of generated `DefaultAccountData`.
// for `::subxt::AccountData` implementation by the generated `DefaultAccountData`.

@@ -74,12 +74,15 @@ fn generate_storage_entry_fns(
storage_entry: &StorageEntryMetadata<PortableForm>,
) -> (TokenStream2, TokenStream2) {
let entry_struct_ident = format_ident!("{}", storage_entry.name);
let (fields, entry_struct, constructor, key_impl) = match storage_entry.ty {
let is_account_wrapper = pallet.name == "System" && storage_entry.name == "Account";
let wrapper_struct_ident = format_ident!("{}DefaultData", storage_entry.name);
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

nitpicking: I'm not super fond of the name AccountDefaultData; I'd lean towards something like AccountOwned I guess.. but I'm not too fussed :D

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

AccountOwned sounds better :D Thanks

Signed-off-by: Alexandru Vasile <alexandru.vasile@parity.io>
Signed-off-by: Alexandru Vasile <alexandru.vasile@parity.io>
Signed-off-by: Alexandru Vasile <alexandru.vasile@parity.io>
Signed-off-by: Alexandru Vasile <alexandru.vasile@parity.io>
Signed-off-by: Alexandru Vasile <alexandru.vasile@parity.io>
Signed-off-by: Alexandru Vasile <alexandru.vasile@parity.io>
Signed-off-by: Alexandru Vasile <alexandru.vasile@parity.io>
match self {
TypePath::Type(ty) => {
match ty.ty.type_def() {
TypeDef::Sequence(_) => {
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I think things like HashMap and HashSet and such also encode to a TypeDef::Sequence, so you'll probably also need to check that the .path() == &["Vec"] :)

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I think we currently substitute any Sequence<ty> with std::vec<ty> (from type_path.rs). It does make sense to constrain it further, but the ty.ty.path() is empty in this case. Any other ways we could constrain it to just vec?

Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

As we discussed, my comment wasn't correct in fact, and it looks like only Vec and VecDeque are represented as Sequences, and so I think this is ok!

@jsdw jsdw requested a review from ascjones March 1, 2022 10:30
lexnv added 4 commits March 1, 2022 14:15
Signed-off-by: Alexandru Vasile <alexandru.vasile@parity.io>
Signed-off-by: Alexandru Vasile <alexandru.vasile@parity.io>
Signed-off-by: Alexandru Vasile <alexandru.vasile@parity.io>
TypePath::Type(ty) => {
match ty.ty.type_def() {
TypeDef::Sequence(_) => {
if ty.params.is_empty() {
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Looking at the scale-info crate it looks like you could more safely do:

TypeDef::Sequence(seq) => {
    let type_param = seq.type_param()
    // ...

I'd also lean towards not bothering to quote! the type_param before returning it (I think it'll work just fine if you don't), but I'm not too bothered either way!

Also as a matter of personal taste I'd probably avoid the indentation with some early returns, a bit like:

let ty = match self {
    TypePath::Type(ty) => ty,
    _ => return None
};

let sequence = match ty.ty.type_def() {
    TypeDef::Sequence(seq) => seq,
    _ => return None;
};

// ...

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

That makes a lot of sense, and indeed the code looks a lot better 😄

I did give the type_param() a try, but it seems to return a UntrackedSymbol<TypeId> which does not implement the ToToken trait.
Looking below in the code generation, it seems under TypeDef::Sequence there is the assumption of always having at least one type_param. I've removed the is_empty() check here for consistency.
Wondering if we could run into problems along the line due to this assumption. 🤔

// You should have received a copy of the GNU General Public License
// along with subxt. If not, see <http://www.gnu.org/licenses/>.

//! To run this example, a local polkadot node should be running. Example verified against polkadot 0.9.13-82616422d0-aarch64-macos.
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Question: Is this "Example verified against $version" correct for you? I'd either remove it or, better, use polkadot --version to put whatever version you're using in (if it's different) :)

Copy link
Collaborator

@jsdw jsdw Mar 1, 2022

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Or, looking at the below, perhaps the example is using a substrate instance rather than a polkadot one (which the other examples are using)? I'd prolly stick to polkadot just to be the same as the other examples (and consistent with the comments) :)

let controller_acc = api
.storage()
.staking()
.bonded(&alice_stash_id, None)
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Cool!

I'm wondering whether it is worth renaming this ewxample to fetch_staking_details.rs so that it lines up with the other fetch example (both are about fetching stuff out of storage)?

.await?;
let bob_pre = api
.storage()
.system()
.account(bob.account_id().clone(), None)
.account(bob.account_id(), None)
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

It's nice not having to clone these any more!

let ctx = test_context().await;
let current_era_result = ctx
let cxt = test_context().await;
let eras = 0;
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I think it's just era :)

@@ -79,7 +79,7 @@ async fn storage_n_mapish_key_is_properly_created(
};

// This is what the generated code hashes a `session().key_owner(..)` key into:
let actual_key_bytes = KeyOwner(KeyTypeId([1, 2, 3, 4]), vec![5u8, 6, 7, 8])
let actual_key_bytes = KeyOwner(&KeyTypeId([1, 2, 3, 4]), &[5u8, 6, 7, 8])
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Nice!

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Thanks! Having the API as reference does improve the ergonomics IMO :D

lexnv added 5 commits March 1, 2022 18:17
Signed-off-by: Alexandru Vasile <alexandru.vasile@parity.io>
Signed-off-by: Alexandru Vasile <alexandru.vasile@parity.io>
Signed-off-by: Alexandru Vasile <alexandru.vasile@parity.io>
Signed-off-by: Alexandru Vasile <alexandru.vasile@parity.io>
Signed-off-by: Alexandru Vasile <alexandru.vasile@parity.io>
match ty.ty.type_def() {
TypeDef::Sequence(_) => Some(&ty.params[0]),
_ => None,
}
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Looking neater; nice :)

// Get Alice' Stash account ID
//
// *Note*: This is pre-funded, as declared in the [genesis state]
// (https://github.com/substrate-developer-hub/substrate-node-template/blob/main/node/src/chain_spec.rs#L49)
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Because it's polkadot and not substrate, I think the link would point to something more like:

https://github.com/paritytech/polkadot/blob/dc784f9b47e4681897cfd477b4f0760330875a87/node/service/src/chain_spec.rs#L1277

But in other examples we've taken for granted that some test accounts have funds in, so I wouldn't bother with a comment about it myself :)

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Thanks, i ve removed it for now :D

Copy link
Collaborator

@jsdw jsdw left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

You'll need to merge in the latest stuff on master and then re-generate that polkadot.rs file, but otherwise lgtm!

lexnv added 3 commits March 2, 2022 11:12
Signed-off-by: Alexandru Vasile <alexandru.vasile@parity.io>
Signed-off-by: Alexandru Vasile <alexandru.vasile@parity.io>
Copy link
Contributor

@ascjones ascjones left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This is a huge usability improvement!

While I'm not a huge fan of the speciaI cased codegen for the Account storage item, I can see that the AccountOwned workaround is necessary to make this work with the current storage API, and I can't come up with anything better.

However we should keep an open mind as to how we can improve this in the future to make the codegen less convoluted.

One avenue to explore would be to modify fetch and fetch_or_default to accept preconstructed StorageKeys, and have the macro construct those keys inline instead of generating and calling StorageEntry::key. Then could modify the AccountData trait to have a method which constructs the StorageKey from the AccountId.

This will all need a bit more thought, so not in the scope of this PR.

@lexnv lexnv merged commit 08369f3 into paritytech:master Mar 2, 2022
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

Change storage key arguments (and call arguments?) to being references
3 participants