168 lines
4.4 KiB
Rust
168 lines
4.4 KiB
Rust
use std::{
|
|
fmt::{self, Display, Formatter},
|
|
sync::Arc,
|
|
};
|
|
|
|
use serde::{de::Visitor, Deserialize, Serialize};
|
|
|
|
use crate::{
|
|
built_in_identifiers::all_built_in_identifiers,
|
|
built_in_values::all_built_in_values,
|
|
error::{RuntimeError, SyntaxError, ValidationError},
|
|
AbstractTree, Context, Format, SyntaxNode, Type, Value,
|
|
};
|
|
|
|
/// A string by which a variable is known to a context.
|
|
///
|
|
/// Every variable is a key-value pair. An identifier holds the key part of that
|
|
/// pair. Its inner value can be used to retrieve a Value instance from a Map.
|
|
#[derive(Debug, Clone, Eq, PartialEq, PartialOrd, Ord, Hash)]
|
|
pub struct Identifier(Arc<String>);
|
|
|
|
impl Identifier {
|
|
pub fn new(key: &str) -> Self {
|
|
for built_in_identifier in all_built_in_identifiers() {
|
|
let identifier = built_in_identifier.get();
|
|
|
|
if &key == identifier.inner().as_ref() {
|
|
return identifier.clone();
|
|
}
|
|
}
|
|
|
|
Identifier(Arc::new(key.to_string()))
|
|
}
|
|
|
|
pub fn from_raw_parts(arc: Arc<String>) -> Self {
|
|
Identifier(arc)
|
|
}
|
|
|
|
pub fn inner(&self) -> &Arc<String> {
|
|
&self.0
|
|
}
|
|
|
|
pub fn contains(&self, string: &str) -> bool {
|
|
self.0.as_ref() == string
|
|
}
|
|
}
|
|
|
|
impl AbstractTree for Identifier {
|
|
fn from_syntax(
|
|
node: SyntaxNode,
|
|
source: &str,
|
|
_context: &Context,
|
|
) -> Result<Self, SyntaxError> {
|
|
SyntaxError::expect_syntax_node("identifier", node)?;
|
|
|
|
let text = &source[node.byte_range()];
|
|
|
|
debug_assert!(!text.is_empty());
|
|
|
|
Ok(Identifier::new(text))
|
|
}
|
|
|
|
fn validate(&self, _source: &str, context: &Context) -> Result<(), ValidationError> {
|
|
let variable_exists = context.add_allowance(self)?;
|
|
|
|
if variable_exists {
|
|
Ok(())
|
|
} else {
|
|
for built_in_value in all_built_in_values() {
|
|
if built_in_value.name() == self.inner().as_ref() {
|
|
return Ok(());
|
|
}
|
|
}
|
|
|
|
Err(ValidationError::VariableIdentifierNotFound(self.clone()))
|
|
}
|
|
}
|
|
|
|
fn expected_type(&self, context: &Context) -> Result<Type, ValidationError> {
|
|
if let Some(r#type) = context.get_type(self)? {
|
|
Ok(r#type)
|
|
} else {
|
|
for built_in_value in all_built_in_values() {
|
|
if built_in_value.name() == self.inner().as_ref() {
|
|
return Ok(built_in_value.get().r#type()?);
|
|
}
|
|
}
|
|
|
|
Err(ValidationError::VariableIdentifierNotFound(self.clone()))
|
|
}
|
|
}
|
|
|
|
fn run(&self, _source: &str, context: &Context) -> Result<Value, RuntimeError> {
|
|
if let Some(value) = context.get_value(self)? {
|
|
return Ok(value);
|
|
} else {
|
|
for built_in_value in all_built_in_values() {
|
|
if built_in_value.name() == self.inner().as_ref() {
|
|
return Ok(built_in_value.get().clone());
|
|
}
|
|
}
|
|
}
|
|
|
|
Err(RuntimeError::ValidationFailure(
|
|
ValidationError::VariableIdentifierNotFound(self.clone()),
|
|
))
|
|
}
|
|
}
|
|
|
|
impl Format for Identifier {
|
|
fn format(&self, output: &mut String, _indent_level: u8) {
|
|
output.push_str(&self.0);
|
|
}
|
|
}
|
|
|
|
impl From<String> for Identifier {
|
|
fn from(value: String) -> Self {
|
|
Identifier::from_raw_parts(Arc::new(value))
|
|
}
|
|
}
|
|
|
|
impl From<&str> for Identifier {
|
|
fn from(value: &str) -> Self {
|
|
Identifier::new(value)
|
|
}
|
|
}
|
|
|
|
impl Display for Identifier {
|
|
fn fmt(&self, f: &mut Formatter) -> fmt::Result {
|
|
write!(f, "{}", self.0)
|
|
}
|
|
}
|
|
|
|
impl Serialize for Identifier {
|
|
fn serialize<S>(&self, serializer: S) -> Result<S::Ok, S::Error>
|
|
where
|
|
S: serde::Serializer,
|
|
{
|
|
serializer.serialize_str(self.0.as_ref())
|
|
}
|
|
}
|
|
|
|
impl<'de> Deserialize<'de> for Identifier {
|
|
fn deserialize<D>(deserializer: D) -> Result<Self, D::Error>
|
|
where
|
|
D: serde::Deserializer<'de>,
|
|
{
|
|
deserializer.deserialize_string(IdentifierVisitor)
|
|
}
|
|
}
|
|
|
|
struct IdentifierVisitor;
|
|
|
|
impl<'de> Visitor<'de> for IdentifierVisitor {
|
|
type Value = Identifier;
|
|
|
|
fn expecting(&self, formatter: &mut fmt::Formatter) -> fmt::Result {
|
|
formatter.write_str("valid UFT-8 sequence")
|
|
}
|
|
|
|
fn visit_string<E>(self, v: String) -> Result<Self::Value, E>
|
|
where
|
|
E: serde::de::Error,
|
|
{
|
|
Ok(Identifier(Arc::new(v)))
|
|
}
|
|
}
|