diff --git a/Cargo.lock b/Cargo.lock
index 3bed5d4f35..df2b1eb619 100644
--- a/Cargo.lock
+++ b/Cargo.lock
@@ -7506,6 +7506,25 @@ dependencies = [
"unified-accounts-chain-extension-types",
]
+[[package]]
+name = "pallet-chain-extension-uniques"
+version = "0.1.0"
+dependencies = [
+ "frame-support",
+ "frame-system",
+ "log",
+ "num-traits",
+ "pallet-contracts",
+ "pallet-contracts-primitives",
+ "pallet-uniques",
+ "parity-scale-codec",
+ "scale-info",
+ "sp-core",
+ "sp-runtime",
+ "sp-std",
+ "uniques-chain-extension-types",
+]
+
[[package]]
name = "pallet-chain-extension-xvm"
version = "0.1.1"
@@ -15397,6 +15416,16 @@ dependencies = [
"sp-runtime",
]
+[[package]]
+name = "uniques-chain-extension-types"
+version = "0.1.0"
+dependencies = [
+ "frame-system",
+ "pallet-contracts",
+ "parity-scale-codec",
+ "scale-info",
+]
+
[[package]]
name = "universal-hash"
version = "0.4.1"
diff --git a/Cargo.toml b/Cargo.toml
index 3e7f41a8c7..e6024cc266 100644
--- a/Cargo.toml
+++ b/Cargo.toml
@@ -18,6 +18,7 @@ members = [
"chain-extensions/pallet-assets",
"chain-extensions/xvm",
"chain-extensions/unified-accounts",
+ "chain-extensions/pallet-uniques",
"chain-extensions/types/*",
"vendor/evm-tracing",
@@ -301,10 +302,12 @@ pallet-evm-precompile-unified-accounts = { path = "./precompiles/unified-account
pallet-chain-extension-xvm = { path = "./chain-extensions/xvm", default-features = false }
pallet-chain-extension-assets = { path = "./chain-extensions/pallet-assets", default-features = false }
pallet-chain-extension-unified-accounts = { path = "./chain-extensions/unified-accounts", default-features = false }
+pallet-chain-extension-uniques = { path = "./chain-extensions/pallet-uniques", default-features = false }
xvm-chain-extension-types = { path = "./chain-extensions/types/xvm", default-features = false }
assets-chain-extension-types = { path = "./chain-extensions/types/assets", default-features = false }
unified-accounts-chain-extension-types = { path = "./chain-extensions/types/unified-accounts", default-features = false }
+uniques-chain-extension-types = { path = "./chain-extensions/types/uniques", default-features = false }
precompile-utils = { path = "./precompiles/utils", default-features = false }
diff --git a/chain-extensions/pallet-uniques/Cargo.toml b/chain-extensions/pallet-uniques/Cargo.toml
new file mode 100644
index 0000000000..1d53acd3e2
--- /dev/null
+++ b/chain-extensions/pallet-uniques/Cargo.toml
@@ -0,0 +1,41 @@
+[package]
+name = "pallet-chain-extension-uniques"
+version = "0.1.0"
+license = "Apache-2.0"
+description = "Assets chain extension for WASM contracts"
+authors.workspace = true
+edition.workspace = true
+homepage.workspace = true
+repository.workspace = true
+
+[dependencies]
+uniques-chain-extension-types = { workspace = true }
+frame-support = { workspace = true }
+frame-system = { workspace = true }
+log = { workspace = true }
+num-traits = { workspace = true }
+pallet-uniques = { workspace = true }
+pallet-contracts = { workspace = true }
+pallet-contracts-primitives = { workspace = true }
+parity-scale-codec = { workspace = true }
+scale-info = { workspace = true }
+sp-core = { workspace = true }
+sp-runtime = { workspace = true }
+sp-std = { workspace = true }
+
+[features]
+default = ["std"]
+std = [
+ "parity-scale-codec/std",
+ "frame-support/std",
+ "frame-system/std",
+ "num-traits/std",
+ "pallet-contracts/std",
+ "pallet-contracts-primitives/std",
+ "scale-info/std",
+ "sp-std/std",
+ "sp-core/std",
+ "sp-runtime/std",
+ "pallet-uniques/std",
+ "uniques-chain-extension-types/std",
+]
diff --git a/chain-extensions/pallet-uniques/src/lib.rs b/chain-extensions/pallet-uniques/src/lib.rs
new file mode 100644
index 0000000000..ebf285853b
--- /dev/null
+++ b/chain-extensions/pallet-uniques/src/lib.rs
@@ -0,0 +1,211 @@
+// This file is part of Astar.
+
+// Copyright (C) 2019-2023 Stake Technologies Pte.Ltd.
+// SPDX-License-Identifier: GPL-3.0-or-later
+
+// Astar is free software: you can redistribute it and/or modify
+// it under the terms of the GNU General Public License as published by
+// the Free Software Foundation, either version 3 of the License, or
+// (at your option) any later version.
+
+// Astar is distributed in the hope that it will be useful,
+// but WITHOUT ANY WARRANTY; without even the implied warranty of
+// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+// GNU General Public License for more details.
+
+// You should have received a copy of the GNU General Public License
+// along with Astar. If not, see .
+
+#![cfg_attr(not(feature = "std"), no_std)]
+
+pub mod weights;
+
+use frame_support::traits::nonfungibles::{Inspect, InspectEnumerable};
+use pallet_contracts::chain_extension::{
+ ChainExtension, Environment, Ext, InitState, RetVal, SysConfig,
+};
+use parity_scale_codec::Encode;
+use sp_runtime::traits::StaticLookup;
+use sp_runtime::BoundedVec;
+use sp_runtime::DispatchError;
+use sp_std::marker::PhantomData;
+use sp_std::vec::Vec;
+use uniques_chain_extension_types::Outcome;
+
+type AccountIdLookup = <::Lookup as StaticLookup>::Source;
+
+enum UniquesFunc {
+ Owner,
+ CollectionOwner,
+ Attribute,
+ CollectionAttribute,
+ CanTransfer,
+ Collections,
+ Items,
+ Owned,
+ OwnedInCollection,
+}
+
+impl TryFrom for UniquesFunc {
+ type Error = DispatchError;
+
+ fn try_from(value: u16) -> Result {
+ match value {
+ 1 => Ok(UniquesFunc::Owner),
+ 2 => Ok(UniquesFunc::CollectionOwner),
+ 3 => Ok(UniquesFunc::Attribute),
+ 4 => Ok(UniquesFunc::CollectionAttribute),
+ 5 => Ok(UniquesFunc::CanTransfer),
+ 6 => Ok(UniquesFunc::Collections),
+ 7 => Ok(UniquesFunc::Items),
+ 8 => Ok(UniquesFunc::Owned),
+ 9 => Ok(UniquesFunc::OwnedInCollection),
+ _ => Err(DispatchError::Other(
+ "Unimplemented func_id for UniquesFunc",
+ )),
+ }
+ }
+}
+
+/// Pallet Uniques chain extension.
+pub struct UniquesExtension(PhantomData<(T, W)>);
+
+impl Default for UniquesExtension {
+ fn default() -> Self {
+ UniquesExtension(PhantomData)
+ }
+}
+
+impl ChainExtension for UniquesExtension
+where
+ T: pallet_uniques::Config + pallet_contracts::Config,
+ AccountIdLookup: From<::AccountId>,
+ ::AccountId: From<[u8; 32]>,
+ W: weights::WeightInfo,
+{
+ fn call(&mut self, env: Environment) -> Result
+ where
+ E: Ext,
+ {
+ let func_id = env.func_id().try_into()?;
+ let mut env = env.buf_in_buf_out();
+
+ match func_id {
+ UniquesFunc::Owner => {
+ let (collection_id, item): (
+ ::CollectionId,
+ ::ItemId,
+ ) = env.read_as()?;
+
+ let base_weight = ::owner();
+ env.charge_weight(base_weight)?;
+
+ let owner = pallet_uniques::Pallet::::owner(collection_id, item);
+ env.write(&owner.encode(), false, None)?;
+ }
+ UniquesFunc::CollectionOwner => {
+ let collection_id: ::CollectionId = env.read_as()?;
+
+ let base_weight = ::collection_owner();
+ env.charge_weight(base_weight)?;
+
+ let owner = pallet_uniques::Pallet::::collection_owner(collection_id);
+ env.write(&owner.encode(), false, None)?;
+ }
+ UniquesFunc::Attribute => {
+ let (collection_id, item, key): (
+ ::CollectionId,
+ ::ItemId,
+ BoundedVec::KeyLimit>,
+ ) = env.read_as()?;
+
+ let base_weight = ::attribute();
+ env.charge_weight(base_weight)?;
+
+ let attribute = pallet_uniques::Pallet::::attribute(&collection_id, &item, &key);
+ env.write(&attribute.encode(), false, None)?;
+ }
+ UniquesFunc::CollectionAttribute => {
+ let (collection_id, key): (
+ ::CollectionId,
+ BoundedVec::KeyLimit>,
+ ) = env.read_as()?;
+
+ let base_weight = ::collection_attribute();
+ env.charge_weight(base_weight)?;
+
+ let attribute =
+ pallet_uniques::Pallet::::collection_attribute(&collection_id, &key);
+ env.write(&attribute.encode(), false, None)?;
+ }
+ UniquesFunc::CanTransfer => {
+ let (collection_id, item): (
+ ::CollectionId,
+ ::ItemId,
+ ) = env.read_as()?;
+
+ let base_weight = ::can_transfer();
+ env.charge_weight(base_weight)?;
+
+ let can_transfer = pallet_uniques::Pallet::::can_transfer(&collection_id, &item);
+ env.write(&can_transfer.encode(), false, None)?;
+ }
+ UniquesFunc::Collections => {
+ let read_bound: u32 = env.read_as()?;
+
+ let base_weight = ::collections(read_bound);
+ env.charge_weight(base_weight)?;
+
+ let collections: Vec<::CollectionId> =
+ pallet_uniques::Pallet::::collections().collect();
+
+ env.write(&collections.encode(), false, None)?;
+ }
+ UniquesFunc::Items => {
+ let (collection_id, read_bound): (
+ ::CollectionId,
+ u32,
+ ) = env.read_as()?;
+
+ let base_weight = ::items(read_bound);
+ env.charge_weight(base_weight)?;
+
+ let items: Vec<::ItemId> =
+ pallet_uniques::Pallet::::items(&collection_id).collect();
+
+ env.write(&items.encode(), false, None)?;
+ }
+ UniquesFunc::Owned => {
+ let (who, read_bound): (T::AccountId, u32) = env.read_as()?;
+
+ let items: Vec<(
+ ::CollectionId,
+ ::ItemId,
+ )> = pallet_uniques::Pallet::::owned(&who).collect();
+
+ let base_weight = ::owned(read_bound);
+ env.charge_weight(base_weight)?;
+
+ env.write(&items.encode(), false, None)?;
+ }
+ UniquesFunc::OwnedInCollection => {
+ let (who, collection_id, read_bound): (
+ T::AccountId,
+ ::CollectionId,
+ u32,
+ ) = env.read_as()?;
+
+ let items: Vec<::ItemId> =
+ pallet_uniques::Pallet::::owned_in_collection(&collection_id, &who)
+ .collect();
+
+ let base_weight = ::owned_in_collection(read_bound);
+ env.charge_weight(base_weight)?;
+
+ env.write(&items.encode(), false, None)?;
+ }
+ }
+
+ Ok(RetVal::Converging(Outcome::Success as u32))
+ }
+}
diff --git a/chain-extensions/pallet-uniques/src/weights.rs b/chain-extensions/pallet-uniques/src/weights.rs
new file mode 100644
index 0000000000..c924ea45b5
--- /dev/null
+++ b/chain-extensions/pallet-uniques/src/weights.rs
@@ -0,0 +1,77 @@
+// This file is part of Astar.
+
+// Copyright (C) 2019-2023 Stake Technologies Pte.Ltd.
+// SPDX-License-Identifier: GPL-3.0-or-later
+
+// Astar is free software: you can redistribute it and/or modify
+// it under the terms of the GNU General Public License as published by
+// the Free Software Foundation, either version 3 of the License, or
+// (at your option) any later version.
+
+// Astar is distributed in the hope that it will be useful,
+// but WITHOUT ANY WARRANTY; without even the implied warranty of
+// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+// GNU General Public License for more details.
+
+// You should have received a copy of the GNU General Public License
+// along with Astar. If not, see .
+
+#![cfg_attr(rustfmt, rustfmt_skip)]
+#![allow(unused_parens)]
+#![allow(unused_imports)]
+
+use frame_support::{traits::Get, weights::{Weight, constants::RocksDbWeight}};
+use sp_std::marker::PhantomData;
+
+/// Weight functions needed for pallet-assets chain-extension.
+pub trait WeightInfo {
+ fn owner() -> Weight;
+ fn collection_owner() -> Weight;
+ fn attribute() -> Weight;
+ fn collection_attribute() -> Weight;
+ fn can_transfer() -> Weight;
+ fn collections(n: u32) -> Weight;
+ fn items(n: u32) -> Weight;
+ fn owned(n: u32) -> Weight;
+ fn owned_in_collection(n: u32) -> Weight;
+}
+
+/// Weights for pallet-uniques chain-extension
+pub struct SubstrateWeight(PhantomData);
+impl WeightInfo for SubstrateWeight {
+ fn owner() -> Weight {
+ T::DbWeight::get().reads(1 as u64)
+ }
+
+ fn collection_owner() -> Weight {
+ T::DbWeight::get().reads(1 as u64)
+ }
+
+ fn attribute() -> Weight {
+ T::DbWeight::get().reads(1 as u64)
+ }
+
+ fn collection_attribute() -> Weight {
+ T::DbWeight::get().reads(1 as u64)
+ }
+
+ fn can_transfer() -> Weight {
+ T::DbWeight::get().reads(1 as u64)
+ }
+
+ fn collections(n: u32) -> Weight {
+ T::DbWeight::get().reads(1 as u64).saturating_mul(n.into())
+ }
+
+ fn items(n: u32) -> Weight {
+ T::DbWeight::get().reads(1 as u64).saturating_mul(n.into())
+ }
+
+ fn owned(n: u32) -> Weight {
+ T::DbWeight::get().reads(1 as u64).saturating_mul(n.into())
+ }
+
+ fn owned_in_collection(n: u32) -> Weight {
+ T::DbWeight::get().reads(1 as u64).saturating_mul(n.into())
+ }
+}
diff --git a/chain-extensions/types/uniques/Cargo.toml b/chain-extensions/types/uniques/Cargo.toml
new file mode 100644
index 0000000000..6e46674671
--- /dev/null
+++ b/chain-extensions/types/uniques/Cargo.toml
@@ -0,0 +1,25 @@
+[package]
+name = "uniques-chain-extension-types"
+version = "0.1.0"
+license = "Apache-2.0"
+description = "Types definitions for assets chain-extension"
+authors.workspace = true
+edition.workspace = true
+homepage.workspace = true
+repository.workspace = true
+
+[dependencies]
+parity-scale-codec = { workspace = true }
+scale-info = { workspace = true }
+
+frame-system = { workspace = true }
+pallet-contracts = { workspace = true }
+
+[features]
+default = ["std"]
+std = [
+ "scale-info/std",
+ "parity-scale-codec/std",
+ "pallet-contracts/std",
+ "frame-system/std",
+]
diff --git a/chain-extensions/types/uniques/src/lib.rs b/chain-extensions/types/uniques/src/lib.rs
new file mode 100644
index 0000000000..675562901f
--- /dev/null
+++ b/chain-extensions/types/uniques/src/lib.rs
@@ -0,0 +1,37 @@
+#![cfg_attr(not(feature = "std"), no_std)]
+use parity_scale_codec::MaxEncodedLen;
+use parity_scale_codec::{Decode, Encode};
+
+#[derive(PartialEq, Eq, Copy, Clone, Encode, Decode, Debug)]
+#[cfg_attr(feature = "std", derive(scale_info::TypeInfo))]
+pub enum Outcome {
+ /// Success
+ Success = 0,
+ /// Origin Caller is not supported
+ OriginCannotBeCaller = 98,
+ /// Unknown error
+ RuntimeError = 99,
+}
+
+#[derive(Debug, Copy, Clone, PartialEq, Eq, Encode, Decode, MaxEncodedLen)]
+#[cfg_attr(feature = "std", derive(scale_info::TypeInfo))]
+pub enum Origin {
+ Caller,
+ Address,
+}
+
+impl Default for Origin {
+ fn default() -> Self {
+ Self::Address
+ }
+}
+
+#[macro_export]
+macro_rules! select_origin {
+ ($origin:expr, $account:expr) => {
+ match $origin {
+ Origin::Caller => return Ok(RetVal::Converging(Outcome::OriginCannotBeCaller as u32)),
+ Origin::Address => RawOrigin::Signed($account),
+ }
+ };
+}