2022-10-11 01:10:49 -07:00
|
|
|
//! Resource identifiers.
|
2022-07-14 23:18:20 -07:00
|
|
|
|
2023-03-25 18:44:45 -07:00
|
|
|
use std::borrow::{Borrow, Cow};
|
2022-10-11 01:10:49 -07:00
|
|
|
use std::cmp::Ordering;
|
2022-10-12 03:53:59 -07:00
|
|
|
use std::fmt;
|
|
|
|
use std::fmt::Formatter;
|
2022-09-08 21:39:08 -07:00
|
|
|
use std::io::Write;
|
2022-04-14 14:55:45 -07:00
|
|
|
use std::str::FromStr;
|
|
|
|
|
2022-10-12 03:53:59 -07:00
|
|
|
use serde::de::Error as _;
|
|
|
|
use serde::{Deserialize, Deserializer, Serialize, Serializer};
|
2023-03-25 18:44:45 -07:00
|
|
|
use thiserror::Error;
|
2022-04-14 14:55:45 -07:00
|
|
|
|
2022-11-13 06:10:42 -08:00
|
|
|
use crate::{nbt, Decode, Encode};
|
2022-04-14 14:55:45 -07:00
|
|
|
|
2022-10-12 03:53:59 -07:00
|
|
|
/// A wrapper around a string type `S` which guarantees the wrapped string is a
|
|
|
|
/// valid resource identifier.
|
|
|
|
///
|
2022-10-11 01:10:49 -07:00
|
|
|
/// A resource identifier is a string divided into a "namespace" part and a
|
|
|
|
/// "path" part. For instance `minecraft:apple` and `valence:frobnicator` are
|
2022-10-13 22:28:12 -07:00
|
|
|
/// both valid identifiers. A string must match the regex
|
2023-03-25 18:44:45 -07:00
|
|
|
/// `^([a-z0-9_.-]+:)?[a-z0-9_.-\/]+$` to be successfully parsed.
|
2022-04-14 14:55:45 -07:00
|
|
|
///
|
2023-03-25 18:44:45 -07:00
|
|
|
/// While parsing, if the namespace part is left off (the part before and
|
|
|
|
/// including the colon) then "minecraft:" is inserted at the beginning of the
|
|
|
|
/// string.
|
|
|
|
#[derive(Copy, Clone, Eq, Ord, Hash)]
|
2022-10-12 03:53:59 -07:00
|
|
|
pub struct Ident<S> {
|
|
|
|
string: S,
|
2022-04-14 14:55:45 -07:00
|
|
|
}
|
|
|
|
|
2023-03-25 18:44:45 -07:00
|
|
|
/// The error type created when an [`Ident`] cannot be parsed from a
|
|
|
|
/// string. Contains the string that failed to parse.
|
|
|
|
#[derive(Clone, PartialEq, Eq, PartialOrd, Ord, Hash, Debug, Error)]
|
|
|
|
#[error("invalid resource identifier \"{0}\"")]
|
|
|
|
pub struct IdentError(pub String);
|
2022-04-14 14:55:45 -07:00
|
|
|
|
2023-03-25 18:44:45 -07:00
|
|
|
impl<'a> Ident<Cow<'a, str>> {
|
|
|
|
pub fn new(string: impl Into<Cow<'a, str>>) -> Result<Self, IdentError> {
|
|
|
|
parse(string.into())
|
2022-04-14 14:55:45 -07:00
|
|
|
}
|
2023-03-25 18:44:45 -07:00
|
|
|
}
|
2022-04-14 14:55:45 -07:00
|
|
|
|
2023-03-25 18:44:45 -07:00
|
|
|
impl<S> Ident<S> {
|
|
|
|
/// Internal API. Do not use.
|
|
|
|
#[doc(hidden)]
|
|
|
|
pub const fn new_unchecked(string: S) -> Self {
|
|
|
|
Self { string }
|
2022-04-14 14:55:45 -07:00
|
|
|
}
|
|
|
|
|
2023-03-25 18:44:45 -07:00
|
|
|
pub fn as_str(&self) -> &str
|
|
|
|
where
|
|
|
|
S: AsRef<str>,
|
|
|
|
{
|
2022-10-13 22:28:12 -07:00
|
|
|
self.string.as_ref()
|
2022-10-11 01:10:49 -07:00
|
|
|
}
|
|
|
|
|
2023-03-25 18:44:45 -07:00
|
|
|
pub fn as_str_ident(&self) -> Ident<&str>
|
|
|
|
where
|
|
|
|
S: AsRef<str>,
|
|
|
|
{
|
2022-10-12 03:53:59 -07:00
|
|
|
Ident {
|
2023-03-25 18:44:45 -07:00
|
|
|
string: self.as_str(),
|
2022-10-12 03:53:59 -07:00
|
|
|
}
|
2022-10-11 01:10:49 -07:00
|
|
|
}
|
|
|
|
|
2022-11-13 06:10:42 -08:00
|
|
|
pub fn into_inner(self) -> S {
|
|
|
|
self.string
|
|
|
|
}
|
|
|
|
|
2023-03-25 18:44:45 -07:00
|
|
|
/// Returns the namespace part of this resource identifier (the part before
|
|
|
|
/// the colon).
|
|
|
|
pub fn namespace(&self) -> &str
|
2022-10-11 01:10:49 -07:00
|
|
|
where
|
2023-03-25 18:44:45 -07:00
|
|
|
S: AsRef<str>,
|
2022-10-11 01:10:49 -07:00
|
|
|
{
|
2023-03-25 18:44:45 -07:00
|
|
|
self.namespace_and_path().0
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Returns the path part of this resource identifier (the part after the
|
|
|
|
/// colon).
|
|
|
|
pub fn path(&self) -> &str
|
|
|
|
where
|
|
|
|
S: AsRef<str>,
|
|
|
|
{
|
|
|
|
self.namespace_and_path().1
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn namespace_and_path(&self) -> (&str, &str)
|
|
|
|
where
|
|
|
|
S: AsRef<str>,
|
|
|
|
{
|
|
|
|
self.as_str()
|
|
|
|
.split_once(':')
|
|
|
|
.expect("invalid resource identifier")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn parse(string: Cow<str>) -> Result<Ident<Cow<str>>, IdentError> {
|
|
|
|
let check_namespace = |s: &str| {
|
|
|
|
!s.is_empty()
|
|
|
|
&& s.chars()
|
|
|
|
.all(|c| matches!(c, 'a'..='z' | '0'..='9' | '_' | '.' | '-'))
|
|
|
|
};
|
|
|
|
|
|
|
|
let check_path = |s: &str| {
|
|
|
|
!s.is_empty()
|
|
|
|
&& s.chars()
|
|
|
|
.all(|c| matches!(c, 'a'..='z' | '0'..='9' | '_' | '.' | '-' | '/'))
|
|
|
|
};
|
|
|
|
|
|
|
|
match string.split_once(':') {
|
|
|
|
Some((namespace, path)) if check_namespace(namespace) && check_path(path) => {
|
|
|
|
Ok(Ident { string })
|
2022-10-12 03:53:59 -07:00
|
|
|
}
|
2023-03-25 18:44:45 -07:00
|
|
|
None if check_path(&string) => Ok(Ident {
|
|
|
|
string: format!("minecraft:{string}").into(),
|
|
|
|
}),
|
|
|
|
_ => Err(IdentError(string.into())),
|
2022-04-14 14:55:45 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-03-25 18:44:45 -07:00
|
|
|
impl<S: AsRef<str>> AsRef<str> for Ident<S> {
|
|
|
|
fn as_ref(&self) -> &str {
|
|
|
|
self.string.as_ref()
|
2022-12-16 08:23:48 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-03-25 18:44:45 -07:00
|
|
|
impl<S> AsRef<S> for Ident<S> {
|
|
|
|
fn as_ref(&self) -> &S {
|
|
|
|
&self.string
|
2022-11-29 03:37:32 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-03-25 18:44:45 -07:00
|
|
|
impl<S: Borrow<str>> Borrow<str> for Ident<S> {
|
|
|
|
fn borrow(&self) -> &str {
|
|
|
|
self.string.borrow()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl From<Ident<String>> for String {
|
|
|
|
fn from(value: Ident<String>) -> Self {
|
|
|
|
value.into_inner()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<'a> From<Ident<Cow<'a, str>>> for Cow<'a, str> {
|
|
|
|
fn from(value: Ident<Cow<'a, str>>) -> Self {
|
|
|
|
value.into_inner()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<'a> From<Ident<Cow<'a, str>>> for Ident<String> {
|
|
|
|
fn from(value: Ident<Cow<'a, str>>) -> Self {
|
|
|
|
Self {
|
|
|
|
string: value.string.into(),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<'a> From<Ident<String>> for Ident<Cow<'a, str>> {
|
|
|
|
fn from(value: Ident<String>) -> Self {
|
|
|
|
Self {
|
2022-11-29 03:37:32 -08:00
|
|
|
string: value.string.into(),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<'a> From<Ident<&'a str>> for Ident<Cow<'a, str>> {
|
|
|
|
fn from(value: Ident<&'a str>) -> Self {
|
|
|
|
Ident {
|
2023-03-25 18:44:45 -07:00
|
|
|
string: value.string.into(),
|
2022-11-29 03:37:32 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-03-25 18:44:45 -07:00
|
|
|
impl<'a> From<Ident<&'a str>> for Ident<String> {
|
|
|
|
fn from(value: Ident<&'a str>) -> Self {
|
2022-11-29 03:37:32 -08:00
|
|
|
Ident {
|
2023-03-25 18:44:45 -07:00
|
|
|
string: value.string.into(),
|
2022-11-29 03:37:32 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-10-12 03:53:59 -07:00
|
|
|
impl FromStr for Ident<String> {
|
2023-03-25 18:44:45 -07:00
|
|
|
type Err = IdentError;
|
|
|
|
|
|
|
|
fn from_str(s: &str) -> Result<Self, Self::Err> {
|
|
|
|
Ok(Ident::new(s)?.into())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl FromStr for Ident<Cow<'static, str>> {
|
|
|
|
type Err = IdentError;
|
2022-04-14 14:55:45 -07:00
|
|
|
|
2022-10-12 03:53:59 -07:00
|
|
|
fn from_str(s: &str) -> Result<Self, Self::Err> {
|
2023-03-25 18:44:45 -07:00
|
|
|
Ident::<String>::try_from(s).map(From::from)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<'a> TryFrom<&'a str> for Ident<String> {
|
|
|
|
type Error = IdentError;
|
|
|
|
|
|
|
|
fn try_from(value: &'a str) -> Result<Self, Self::Error> {
|
|
|
|
Ok(Ident::new(value)?.into())
|
2022-10-11 01:10:49 -07:00
|
|
|
}
|
|
|
|
}
|
2022-04-14 14:55:45 -07:00
|
|
|
|
2022-10-12 03:53:59 -07:00
|
|
|
impl TryFrom<String> for Ident<String> {
|
2023-03-25 18:44:45 -07:00
|
|
|
type Error = IdentError;
|
2022-10-12 03:53:59 -07:00
|
|
|
|
|
|
|
fn try_from(value: String) -> Result<Self, Self::Error> {
|
2023-03-25 18:44:45 -07:00
|
|
|
Ok(Ident::new(value)?.into())
|
2022-04-14 14:55:45 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-03-25 18:44:45 -07:00
|
|
|
impl<'a> TryFrom<Cow<'a, str>> for Ident<String> {
|
|
|
|
type Error = IdentError;
|
|
|
|
|
|
|
|
fn try_from(value: Cow<'a, str>) -> Result<Self, Self::Error> {
|
|
|
|
Ok(Ident::new(value)?.into())
|
2022-10-11 01:10:49 -07:00
|
|
|
}
|
|
|
|
}
|
2022-04-14 14:55:45 -07:00
|
|
|
|
2023-03-25 18:44:45 -07:00
|
|
|
impl<'a> TryFrom<&'a str> for Ident<Cow<'a, str>> {
|
|
|
|
type Error = IdentError;
|
|
|
|
|
|
|
|
fn try_from(value: &'a str) -> Result<Self, Self::Error> {
|
|
|
|
Self::new(value)
|
2022-04-14 14:55:45 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-03-25 18:44:45 -07:00
|
|
|
impl<'a> TryFrom<String> for Ident<Cow<'a, str>> {
|
|
|
|
type Error = IdentError;
|
|
|
|
|
|
|
|
fn try_from(value: String) -> Result<Self, Self::Error> {
|
|
|
|
Self::new(value)
|
2022-04-14 14:55:45 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-03-25 18:44:45 -07:00
|
|
|
impl<'a> TryFrom<Cow<'a, str>> for Ident<Cow<'a, str>> {
|
|
|
|
type Error = IdentError;
|
2022-10-11 01:10:49 -07:00
|
|
|
|
2023-03-25 18:44:45 -07:00
|
|
|
fn try_from(value: Cow<'a, str>) -> Result<Self, Self::Error> {
|
|
|
|
Self::new(value)
|
2022-04-14 14:55:45 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-03-25 18:44:45 -07:00
|
|
|
impl<S: fmt::Debug> fmt::Debug for Ident<S> {
|
|
|
|
fn fmt(&self, f: &mut Formatter<'_>) -> fmt::Result {
|
|
|
|
self.string.fmt(f)
|
2022-04-14 14:55:45 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-03-25 18:44:45 -07:00
|
|
|
impl<S: fmt::Display> fmt::Display for Ident<S> {
|
|
|
|
fn fmt(&self, f: &mut Formatter<'_>) -> fmt::Result {
|
|
|
|
self.string.fmt(f)
|
2022-04-14 14:55:45 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-03-25 18:44:45 -07:00
|
|
|
impl<S, T> PartialEq<Ident<T>> for Ident<S>
|
2022-10-12 03:53:59 -07:00
|
|
|
where
|
2023-03-25 18:44:45 -07:00
|
|
|
S: PartialEq<T>,
|
2022-10-12 03:53:59 -07:00
|
|
|
{
|
2023-03-25 18:44:45 -07:00
|
|
|
fn eq(&self, other: &Ident<T>) -> bool {
|
|
|
|
self.string == other.string
|
2022-04-14 14:55:45 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-03-25 18:44:45 -07:00
|
|
|
impl<S, T> PartialOrd<Ident<T>> for Ident<S>
|
2022-10-12 03:53:59 -07:00
|
|
|
where
|
2023-03-25 18:44:45 -07:00
|
|
|
S: PartialOrd<T>,
|
2022-10-12 03:53:59 -07:00
|
|
|
{
|
2023-03-25 18:44:45 -07:00
|
|
|
fn partial_cmp(&self, other: &Ident<T>) -> Option<Ordering> {
|
|
|
|
self.string.partial_cmp(&other.string)
|
2022-04-14 14:55:45 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-10-12 03:53:59 -07:00
|
|
|
impl<S: Encode> Encode for Ident<S> {
|
2022-11-13 06:10:42 -08:00
|
|
|
fn encode(&self, w: impl Write) -> anyhow::Result<()> {
|
2023-03-25 18:44:45 -07:00
|
|
|
self.as_ref().encode(w)
|
2022-04-14 14:55:45 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-11-13 06:10:42 -08:00
|
|
|
impl<'a, S> Decode<'a> for Ident<S>
|
2022-10-12 03:53:59 -07:00
|
|
|
where
|
2023-03-25 18:44:45 -07:00
|
|
|
S: Decode<'a>,
|
|
|
|
Ident<S>: TryFrom<S, Error = IdentError>,
|
2022-10-12 03:53:59 -07:00
|
|
|
{
|
2022-11-13 06:10:42 -08:00
|
|
|
fn decode(r: &mut &'a [u8]) -> anyhow::Result<Self> {
|
2023-03-25 18:44:45 -07:00
|
|
|
Ok(Ident::try_from(S::decode(r)?)?)
|
2022-10-11 01:10:49 -07:00
|
|
|
}
|
2022-10-12 03:53:59 -07:00
|
|
|
}
|
2022-10-11 01:10:49 -07:00
|
|
|
|
2022-10-12 03:53:59 -07:00
|
|
|
impl<S> From<Ident<S>> for nbt::Value
|
|
|
|
where
|
|
|
|
S: Into<nbt::Value>,
|
|
|
|
{
|
2023-03-25 18:44:45 -07:00
|
|
|
fn from(value: Ident<S>) -> Self {
|
|
|
|
value.into_inner().into()
|
2022-04-14 14:55:45 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-03-25 18:44:45 -07:00
|
|
|
impl<T: Serialize> Serialize for Ident<T> {
|
|
|
|
fn serialize<S>(&self, serializer: S) -> Result<S::Ok, S::Error>
|
|
|
|
where
|
|
|
|
S: Serializer,
|
|
|
|
{
|
|
|
|
self.string.serialize(serializer)
|
2022-10-13 22:28:12 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-03-25 18:44:45 -07:00
|
|
|
impl<'de, S> Deserialize<'de> for Ident<S>
|
2022-10-13 22:28:12 -07:00
|
|
|
where
|
2023-03-25 18:44:45 -07:00
|
|
|
S: Deserialize<'de>,
|
|
|
|
Ident<S>: TryFrom<S, Error = IdentError>,
|
2022-10-13 22:28:12 -07:00
|
|
|
{
|
2023-03-25 18:44:45 -07:00
|
|
|
fn deserialize<D>(deserializer: D) -> Result<Self, D::Error>
|
|
|
|
where
|
|
|
|
D: Deserializer<'de>,
|
|
|
|
{
|
|
|
|
Ident::try_from(S::deserialize(deserializer)?).map_err(D::Error::custom)
|
2022-10-13 22:28:12 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-10-12 03:53:59 -07:00
|
|
|
/// Convenience macro for constructing an [`Ident<String>`] from a format
|
|
|
|
/// string.
|
2022-04-14 14:55:45 -07:00
|
|
|
///
|
2022-10-12 03:53:59 -07:00
|
|
|
/// The arguments to this macro are forwarded to [`std::format`].
|
2022-07-11 05:08:02 -07:00
|
|
|
///
|
|
|
|
/// # Panics
|
|
|
|
///
|
2022-10-12 03:53:59 -07:00
|
|
|
/// The macro will cause a panic if the formatted string is not a valid resource
|
2022-09-23 04:03:21 -07:00
|
|
|
/// identifier. See [`Ident`] for more information.
|
2022-07-11 05:08:02 -07:00
|
|
|
///
|
2022-10-12 03:53:59 -07:00
|
|
|
/// [`Ident<String>`]: [Ident]
|
|
|
|
///
|
2022-07-11 05:08:02 -07:00
|
|
|
/// # Examples
|
|
|
|
///
|
|
|
|
/// ```
|
2022-11-13 06:10:42 -08:00
|
|
|
/// use valence_protocol::ident;
|
2022-07-11 05:08:02 -07:00
|
|
|
///
|
|
|
|
/// let namespace = "my_namespace";
|
2022-10-12 03:53:59 -07:00
|
|
|
/// let path = "my_path";
|
2022-07-11 05:08:02 -07:00
|
|
|
///
|
2022-10-12 03:53:59 -07:00
|
|
|
/// let id = ident!("{namespace}:{path}");
|
|
|
|
///
|
|
|
|
/// assert_eq!(id.namespace(), "my_namespace");
|
|
|
|
/// assert_eq!(id.path(), "my_path");
|
2022-07-11 05:08:02 -07:00
|
|
|
/// ```
|
2022-04-14 14:55:45 -07:00
|
|
|
#[macro_export]
|
|
|
|
macro_rules! ident {
|
|
|
|
($($arg:tt)*) => {{
|
2023-03-25 18:44:45 -07:00
|
|
|
$crate::ident::Ident::<String>::try_from(::std::format!($($arg)*)).unwrap()
|
2022-04-14 14:55:45 -07:00
|
|
|
}}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[cfg(test)]
|
|
|
|
mod tests {
|
2022-10-11 14:05:00 -07:00
|
|
|
#[test]
|
|
|
|
fn check_namespace_and_path() {
|
|
|
|
let id = ident!("namespace:path");
|
|
|
|
assert_eq!(id.namespace(), "namespace");
|
|
|
|
assert_eq!(id.path(), "path");
|
|
|
|
}
|
|
|
|
|
2022-04-14 14:55:45 -07:00
|
|
|
#[test]
|
|
|
|
fn parse_valid() {
|
|
|
|
ident!("minecraft:whatever");
|
|
|
|
ident!("_what-ever55_:.whatever/whatever123456789_");
|
2022-10-11 01:10:49 -07:00
|
|
|
ident!("valence:frobnicator");
|
2022-04-14 14:55:45 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
#[should_panic]
|
|
|
|
fn parse_invalid_0() {
|
|
|
|
ident!("");
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
#[should_panic]
|
|
|
|
fn parse_invalid_1() {
|
|
|
|
ident!(":");
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
#[should_panic]
|
|
|
|
fn parse_invalid_2() {
|
|
|
|
ident!("foo:bar:baz");
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn equality() {
|
|
|
|
assert_eq!(ident!("minecraft:my.identifier"), ident!("my.identifier"));
|
|
|
|
}
|
|
|
|
}
|