forked from Orchid/orchid
129 lines
3.8 KiB
Rust
129 lines
3.8 KiB
Rust
use std::borrow::Cow;
|
|
use std::ops::Deref;
|
|
use std::pin::Pin;
|
|
use std::rc::Rc;
|
|
|
|
use futures::AsyncWrite;
|
|
use orchid_api_derive::Coding;
|
|
use orchid_api_traits::{Encode, Request};
|
|
use orchid_base::error::{OrcRes, mk_errv};
|
|
use orchid_base::format::{FmtCtx, FmtUnit};
|
|
use orchid_base::interner::Tok;
|
|
use orchid_extension::atom::{AtomMethod, Atomic, MethodSetBuilder, Supports, TAtom};
|
|
use orchid_extension::atom_owned::{DeserializeCtx, OwnedAtom, OwnedVariant};
|
|
use orchid_extension::context::i;
|
|
use orchid_extension::conv::TryFromExpr;
|
|
use orchid_extension::expr::Expr;
|
|
|
|
use crate::std::string::to_string::ToStringMethod;
|
|
|
|
#[derive(Copy, Clone, Debug, Coding)]
|
|
pub struct StringGetVal;
|
|
impl Request for StringGetVal {
|
|
type Response = Rc<String>;
|
|
}
|
|
impl AtomMethod for StringGetVal {
|
|
const NAME: &str = "std::string_get_val";
|
|
}
|
|
impl Supports<StringGetVal> for StrAtom {
|
|
async fn handle(&self, _: StringGetVal) -> <StringGetVal as Request>::Response { self.0.clone() }
|
|
}
|
|
impl Supports<ToStringMethod> for StrAtom {
|
|
async fn handle(&self, _: ToStringMethod) -> <ToStringMethod as Request>::Response {
|
|
self.0.as_str().to_string()
|
|
}
|
|
}
|
|
|
|
#[derive(Clone)]
|
|
pub struct StrAtom(Rc<String>);
|
|
impl Atomic for StrAtom {
|
|
type Variant = OwnedVariant;
|
|
type Data = ();
|
|
fn reg_reqs() -> MethodSetBuilder<Self> { MethodSetBuilder::new().handle::<StringGetVal>() }
|
|
}
|
|
impl StrAtom {
|
|
pub fn new(str: Rc<String>) -> Self { Self(str) }
|
|
}
|
|
impl Deref for StrAtom {
|
|
type Target = str;
|
|
fn deref(&self) -> &Self::Target { &self.0 }
|
|
}
|
|
impl OwnedAtom for StrAtom {
|
|
type Refs = ();
|
|
async fn val(&self) -> Cow<'_, Self::Data> { Cow::Owned(()) }
|
|
async fn serialize(&self, sink: Pin<&mut (impl AsyncWrite + ?Sized)>) -> Self::Refs {
|
|
self.deref().encode(sink).await
|
|
}
|
|
async fn print_atom<'a>(&'a self, _: &'a (impl FmtCtx + ?Sized + 'a)) -> FmtUnit {
|
|
format!("{:?}", &*self.0).into()
|
|
}
|
|
async fn deserialize(mut ctx: impl DeserializeCtx, _: Self::Refs) -> Self {
|
|
Self::new(Rc::new(ctx.read::<String>().await))
|
|
}
|
|
}
|
|
|
|
#[derive(Debug, Clone)]
|
|
pub struct IntStrAtom(pub(crate) Tok<String>);
|
|
impl Atomic for IntStrAtom {
|
|
type Variant = OwnedVariant;
|
|
type Data = orchid_api::TStr;
|
|
}
|
|
impl From<Tok<String>> for IntStrAtom {
|
|
fn from(value: Tok<String>) -> Self { Self(value) }
|
|
}
|
|
impl OwnedAtom for IntStrAtom {
|
|
type Refs = ();
|
|
async fn val(&self) -> Cow<'_, Self::Data> { Cow::Owned(self.0.to_api()) }
|
|
async fn print_atom<'a>(&'a self, _: &'a (impl FmtCtx + ?Sized + 'a)) -> FmtUnit {
|
|
format!("{:?}i", *self.0).into()
|
|
}
|
|
async fn serialize(&self, write: Pin<&mut (impl AsyncWrite + ?Sized)>) {
|
|
self.0.encode(write).await
|
|
}
|
|
async fn deserialize(mut dctx: impl DeserializeCtx, _: ()) -> Self {
|
|
let s = dctx.decode::<String>().await;
|
|
Self(i().i(&s).await)
|
|
}
|
|
}
|
|
impl TryFromExpr for IntStrAtom {
|
|
async fn try_from_expr(expr: Expr) -> OrcRes<Self> {
|
|
Ok(IntStrAtom(i().ex(TAtom::<IntStrAtom>::try_from_expr(expr).await?.value).await))
|
|
}
|
|
}
|
|
impl Supports<ToStringMethod> for IntStrAtom {
|
|
async fn handle(&self, _: ToStringMethod) -> <ToStringMethod as Request>::Response {
|
|
self.0.as_str().to_string()
|
|
}
|
|
}
|
|
|
|
#[derive(Clone)]
|
|
pub struct OrcString {
|
|
kind: OrcStringKind,
|
|
}
|
|
|
|
#[derive(Clone)]
|
|
pub enum OrcStringKind {
|
|
Val(TAtom<StrAtom>),
|
|
Int(TAtom<IntStrAtom>),
|
|
}
|
|
impl OrcString {
|
|
pub async fn get_string(&self) -> Rc<String> {
|
|
match &self.kind {
|
|
OrcStringKind::Int(tok) => i().ex(**tok).await.rc(),
|
|
OrcStringKind::Val(atom) => atom.request(StringGetVal).await,
|
|
}
|
|
}
|
|
}
|
|
|
|
impl TryFromExpr for OrcString {
|
|
async fn try_from_expr(expr: Expr) -> OrcRes<OrcString> {
|
|
if let Ok(v) = TAtom::<StrAtom>::try_from_expr(expr.clone()).await {
|
|
return Ok(OrcString { kind: OrcStringKind::Val(v) });
|
|
}
|
|
match TAtom::<IntStrAtom>::try_from_expr(expr).await {
|
|
Ok(t) => Ok(OrcString { kind: OrcStringKind::Int(t) }),
|
|
Err(e) => Err(mk_errv(i().i("A string was expected").await, "", e.pos_iter())),
|
|
}
|
|
}
|
|
}
|