abstract_std/objects/
namespace.rsuse std::fmt::Display;
use cosmwasm_std::StdResult;
use cw_storage_plus::{Key, KeyDeserialize, Prefixer, PrimaryKey};
use schemars::JsonSchema;
use serde::{Deserialize, Serialize};
use super::module::validate_name;
use crate::{AbstractError, AbstractResult};
pub const ABSTRACT_NAMESPACE: &str = "abstract";
#[derive(Deserialize, Serialize, Clone, Debug, PartialEq, JsonSchema)]
pub struct Namespace(String);
impl Namespace {
pub fn new(namespace: &str) -> AbstractResult<Self> {
validate_name(namespace)?;
Ok(Self(namespace.to_owned()))
}
pub fn unchecked(namespace: impl ToString) -> Self {
Self(namespace.to_string())
}
pub fn as_str(&self) -> &str {
&self.0
}
pub fn validate(&self) -> AbstractResult<()> {
validate_name(&self.0)?;
Ok(())
}
pub fn from_id(module_id: &str) -> AbstractResult<Self> {
let parts: Vec<&str> = module_id.split(':').collect();
if parts.len() != 2 {
return Err(AbstractError::FormattingError {
object: "module_id".to_string(),
expected: "namespace:module".to_string(),
actual: module_id.to_string(),
});
}
Self::new(parts[0])
}
}
impl TryFrom<&str> for Namespace {
type Error = AbstractError;
fn try_from(namespace: &str) -> AbstractResult<Self> {
Self::new(namespace)
}
}
impl TryFrom<String> for Namespace {
type Error = AbstractError;
fn try_from(namespace: String) -> AbstractResult<Self> {
Self::try_from(&namespace)
}
}
impl TryFrom<&String> for Namespace {
type Error = AbstractError;
fn try_from(namespace: &String) -> AbstractResult<Self> {
Self::new(namespace)
}
}
impl Display for Namespace {
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
write!(f, "{}", self.0)
}
}
impl KeyDeserialize for &Namespace {
type Output = Namespace;
const KEY_ELEMS: u16 = 1;
#[inline(always)]
fn from_vec(value: Vec<u8>) -> StdResult<Self::Output> {
Ok(Namespace(String::from_vec(value)?))
}
}
impl PrimaryKey<'_> for Namespace {
type Prefix = ();
type SubPrefix = ();
type Suffix = Self;
type SuperSuffix = Self;
fn key(&self) -> Vec<cw_storage_plus::Key> {
self.0.key()
}
}
impl Prefixer<'_> for Namespace {
fn prefix(&self) -> Vec<Key> {
self.0.prefix()
}
}
impl KeyDeserialize for Namespace {
type Output = Namespace;
const KEY_ELEMS: u16 = 1;
#[inline(always)]
fn from_vec(value: Vec<u8>) -> StdResult<Self::Output> {
Ok(Namespace(String::from_vec(value)?))
}
}
#[cfg(test)]
mod test {
#![allow(clippy::needless_borrows_for_generic_args)]
use super::*;
#[coverage_helper::test]
fn test_namespace() {
let namespace = Namespace::new("test").unwrap();
assert_eq!(namespace.as_str(), "test");
}
#[coverage_helper::test]
fn test_from_string() {
let namespace = Namespace::try_from("test".to_string()).unwrap();
assert_eq!(namespace.as_str(), "test");
}
#[coverage_helper::test]
fn test_from_str() {
let namespace = Namespace::try_from("test").unwrap();
assert_eq!(namespace.as_str(), "test");
}
#[coverage_helper::test]
fn test_from_ref_string() {
let namespace = Namespace::try_from(&"test".to_string()).unwrap();
assert_eq!(namespace.as_str(), "test");
}
#[coverage_helper::test]
fn test_to_string() {
let namespace = Namespace::new("test").unwrap();
assert_eq!(namespace.to_string(), "test".to_string());
}
#[coverage_helper::test]
fn string_key_works() {
let k = &Namespace::new("test").unwrap();
let path = k.key();
assert_eq!(1, path.len());
assert_eq!(b"test", path[0].as_ref());
let joined = k.joined_key();
assert_eq!(joined, b"test")
}
}