1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
// Copyright 2019-2022 Parity Technologies (UK) Ltd.
// This file is dual-licensed as Apache-2.0 or GPL-3.0.
// see LICENSE for license details.

use super::ConstantAddress;
use crate::{
    client::OfflineClientT,
    error::Error,
    metadata::{
        DecodeWithMetadata,
        MetadataError,
    },
    Config,
};
use derivative::Derivative;

/// A client for accessing constants.
#[derive(Derivative)]
#[derivative(Clone(bound = "Client: Clone"))]
pub struct ConstantsClient<T, Client> {
    client: Client,
    _marker: std::marker::PhantomData<T>,
}

impl<T, Client> ConstantsClient<T, Client> {
    /// Create a new [`ConstantsClient`].
    pub fn new(client: Client) -> Self {
        Self {
            client,
            _marker: std::marker::PhantomData,
        }
    }
}

impl<T: Config, Client: OfflineClientT<T>> ConstantsClient<T, Client> {
    /// Run the validation logic against some constant address you'd like to access. Returns `Ok(())`
    /// if the address is valid (or if it's not possible to check since the address has no validation hash).
    /// Return an error if the address was not valid or something went wrong trying to validate it (ie
    /// the pallet or constant in question do not exist at all).
    pub fn validate<Address: ConstantAddress>(
        &self,
        address: &Address,
    ) -> Result<(), Error> {
        if let Some(actual_hash) = address.validation_hash() {
            let expected_hash = self
                .client
                .metadata()
                .constant_hash(address.pallet_name(), address.constant_name())?;
            if actual_hash != expected_hash {
                return Err(MetadataError::IncompatibleConstantMetadata(
                    address.pallet_name().into(),
                    address.constant_name().into(),
                )
                .into())
            }
        }
        Ok(())
    }

    /// Access the constant at the address given, returning the type defined by this address.
    /// This is probably used with addresses given from static codegen, although you can manually
    /// construct your own, too.
    pub fn at<Address: ConstantAddress>(
        &self,
        address: &Address,
    ) -> Result<<Address::Target as DecodeWithMetadata>::Target, Error> {
        let metadata = self.client.metadata();

        // 1. Validate constant shape if hash given:
        self.validate(address)?;

        // 2. Attempt to decode the constant into the type given:
        let pallet = metadata.pallet(address.pallet_name())?;
        let constant = pallet.constant(address.constant_name())?;
        let value = Address::Target::decode_with_metadata(
            &mut &*constant.value,
            constant.ty.id(),
            &metadata,
        )?;
        Ok(value)
    }
}