2025-03-04 22:53:31 +13:00
|
|
|
use std::{collections::HashMap, fmt};
|
2024-11-21 10:47:32 -06:00
|
|
|
|
|
|
|
use anyhow::Result;
|
|
|
|
use schemars::JsonSchema;
|
|
|
|
use serde::{Deserialize, Serialize};
|
|
|
|
|
2025-03-08 03:53:34 +13:00
|
|
|
use super::{
|
|
|
|
memory::{self, EnvironmentRef},
|
|
|
|
MetaSettings,
|
|
|
|
};
|
2024-11-21 10:47:32 -06:00
|
|
|
use crate::{
|
|
|
|
errors::KclErrorDetails,
|
2025-01-07 19:10:53 -08:00
|
|
|
execution::{
|
2025-02-21 10:24:12 -05:00
|
|
|
ExecState, ExecutorContext, Face, Helix, ImportedGeometry, Metadata, Plane, Sketch, SketchSet, Solid, SolidSet,
|
|
|
|
TagIdentifier,
|
2025-01-07 19:10:53 -08:00
|
|
|
},
|
2024-12-17 15:23:00 +13:00
|
|
|
parsing::{
|
2025-02-14 13:03:23 +13:00
|
|
|
ast::types::{
|
2025-02-27 15:58:58 +13:00
|
|
|
DefaultParamVal, FunctionExpression, KclNone, Literal, LiteralValue, Node,
|
|
|
|
PrimitiveType as AstPrimitiveType, TagDeclarator, TagNode, Type,
|
2025-02-14 13:03:23 +13:00
|
|
|
},
|
2024-12-17 15:23:00 +13:00
|
|
|
token::NumericSuffix,
|
|
|
|
},
|
2025-02-22 20:16:29 +13:00
|
|
|
std::{args::Arg, StdFnProps},
|
|
|
|
CompilationError, KclError, ModuleId, SourceRange,
|
2024-11-21 10:47:32 -06:00
|
|
|
};
|
|
|
|
|
2024-11-26 09:51:43 -06:00
|
|
|
pub type KclObjectFields = HashMap<String, KclValue>;
|
|
|
|
|
2024-11-21 10:47:32 -06:00
|
|
|
/// Any KCL value.
|
|
|
|
#[derive(Debug, Clone, Deserialize, Serialize, PartialEq, ts_rs::TS, JsonSchema)]
|
|
|
|
#[ts(export)]
|
|
|
|
#[serde(tag = "type")]
|
|
|
|
pub enum KclValue {
|
|
|
|
Uuid {
|
|
|
|
value: ::uuid::Uuid,
|
|
|
|
#[serde(rename = "__meta")]
|
|
|
|
meta: Vec<Metadata>,
|
|
|
|
},
|
|
|
|
Bool {
|
|
|
|
value: bool,
|
|
|
|
#[serde(rename = "__meta")]
|
|
|
|
meta: Vec<Metadata>,
|
|
|
|
},
|
|
|
|
Number {
|
|
|
|
value: f64,
|
2025-02-14 13:03:23 +13:00
|
|
|
ty: NumericType,
|
2024-11-21 10:47:32 -06:00
|
|
|
#[serde(rename = "__meta")]
|
|
|
|
meta: Vec<Metadata>,
|
|
|
|
},
|
|
|
|
String {
|
|
|
|
value: String,
|
|
|
|
#[serde(rename = "__meta")]
|
|
|
|
meta: Vec<Metadata>,
|
|
|
|
},
|
|
|
|
Array {
|
|
|
|
value: Vec<KclValue>,
|
|
|
|
#[serde(rename = "__meta")]
|
|
|
|
meta: Vec<Metadata>,
|
|
|
|
},
|
|
|
|
Object {
|
2024-11-26 09:51:43 -06:00
|
|
|
value: KclObjectFields,
|
2024-11-21 10:47:32 -06:00
|
|
|
#[serde(rename = "__meta")]
|
|
|
|
meta: Vec<Metadata>,
|
|
|
|
},
|
|
|
|
TagIdentifier(Box<TagIdentifier>),
|
2024-12-05 17:56:49 +13:00
|
|
|
TagDeclarator(crate::parsing::ast::types::BoxNode<TagDeclarator>),
|
2025-01-22 09:42:09 +13:00
|
|
|
Plane {
|
|
|
|
value: Box<Plane>,
|
|
|
|
},
|
|
|
|
Face {
|
|
|
|
value: Box<Face>,
|
|
|
|
},
|
2024-11-21 10:47:32 -06:00
|
|
|
Sketch {
|
|
|
|
value: Box<Sketch>,
|
|
|
|
},
|
|
|
|
Sketches {
|
|
|
|
value: Vec<Box<Sketch>>,
|
|
|
|
},
|
2025-01-22 09:42:09 +13:00
|
|
|
Solid {
|
|
|
|
value: Box<Solid>,
|
|
|
|
},
|
2024-11-21 10:47:32 -06:00
|
|
|
Solids {
|
|
|
|
value: Vec<Box<Solid>>,
|
|
|
|
},
|
2025-01-22 09:42:09 +13:00
|
|
|
Helix {
|
|
|
|
value: Box<Helix>,
|
|
|
|
},
|
2024-11-21 10:47:32 -06:00
|
|
|
ImportedGeometry(ImportedGeometry),
|
|
|
|
#[ts(skip)]
|
|
|
|
Function {
|
|
|
|
#[serde(skip)]
|
2025-02-22 20:16:29 +13:00
|
|
|
value: FunctionSource,
|
2024-11-21 10:47:32 -06:00
|
|
|
#[serde(rename = "__meta")]
|
|
|
|
meta: Vec<Metadata>,
|
|
|
|
},
|
2024-12-17 09:38:32 +13:00
|
|
|
Module {
|
|
|
|
value: ModuleId,
|
|
|
|
#[serde(rename = "__meta")]
|
|
|
|
meta: Vec<Metadata>,
|
|
|
|
},
|
2025-03-08 03:53:34 +13:00
|
|
|
#[ts(skip)]
|
|
|
|
Type {
|
|
|
|
#[serde(skip)]
|
|
|
|
value: Option<(PrimitiveType, StdFnProps)>,
|
|
|
|
#[serde(rename = "__meta")]
|
|
|
|
meta: Vec<Metadata>,
|
|
|
|
},
|
2024-11-21 10:47:32 -06:00
|
|
|
KclNone {
|
|
|
|
value: KclNone,
|
|
|
|
#[serde(rename = "__meta")]
|
|
|
|
meta: Vec<Metadata>,
|
|
|
|
},
|
2025-02-12 10:22:56 +13:00
|
|
|
// Only used for memory management. Should never be visible outside of the memory module.
|
|
|
|
Tombstone {
|
|
|
|
value: (),
|
|
|
|
#[serde(rename = "__meta")]
|
|
|
|
meta: Vec<Metadata>,
|
|
|
|
},
|
2024-11-21 10:47:32 -06:00
|
|
|
}
|
|
|
|
|
2025-02-22 20:16:29 +13:00
|
|
|
#[derive(Debug, Clone, PartialEq, Default)]
|
|
|
|
pub enum FunctionSource {
|
|
|
|
#[default]
|
|
|
|
None,
|
|
|
|
Std {
|
|
|
|
func: crate::std::StdFn,
|
|
|
|
props: StdFnProps,
|
|
|
|
},
|
|
|
|
User {
|
|
|
|
ast: crate::parsing::ast::types::BoxNode<FunctionExpression>,
|
|
|
|
memory: EnvironmentRef,
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
impl JsonSchema for FunctionSource {
|
|
|
|
fn schema_name() -> String {
|
|
|
|
"FunctionSource".to_owned()
|
|
|
|
}
|
|
|
|
|
|
|
|
fn json_schema(gen: &mut schemars::gen::SchemaGenerator) -> schemars::schema::Schema {
|
|
|
|
// TODO: Actually generate a reasonable schema.
|
|
|
|
gen.subschema_for::<()>()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-11-21 10:47:32 -06:00
|
|
|
impl From<SketchSet> for KclValue {
|
|
|
|
fn from(sg: SketchSet) -> Self {
|
|
|
|
match sg {
|
|
|
|
SketchSet::Sketch(value) => KclValue::Sketch { value },
|
|
|
|
SketchSet::Sketches(value) => KclValue::Sketches { value },
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl From<Vec<Box<Sketch>>> for KclValue {
|
|
|
|
fn from(sg: Vec<Box<Sketch>>) -> Self {
|
|
|
|
KclValue::Sketches { value: sg }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl From<SolidSet> for KclValue {
|
|
|
|
fn from(eg: SolidSet) -> Self {
|
|
|
|
match eg {
|
2025-01-22 09:42:09 +13:00
|
|
|
SolidSet::Solid(eg) => KclValue::Solid { value: eg },
|
2024-11-21 10:47:32 -06:00
|
|
|
SolidSet::Solids(egs) => KclValue::Solids { value: egs },
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl From<Vec<Box<Solid>>> for KclValue {
|
|
|
|
fn from(eg: Vec<Box<Solid>>) -> Self {
|
|
|
|
if eg.len() == 1 {
|
2025-01-22 09:42:09 +13:00
|
|
|
KclValue::Solid { value: eg[0].clone() }
|
2024-11-21 10:47:32 -06:00
|
|
|
} else {
|
|
|
|
KclValue::Solids { value: eg }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
impl From<KclValue> for Vec<SourceRange> {
|
|
|
|
fn from(item: KclValue) -> Self {
|
|
|
|
match item {
|
2024-12-03 16:39:51 +13:00
|
|
|
KclValue::TagDeclarator(t) => vec![SourceRange::new(t.start, t.end, t.module_id)],
|
2024-11-21 10:47:32 -06:00
|
|
|
KclValue::TagIdentifier(t) => to_vec_sr(&t.meta),
|
2025-01-22 09:42:09 +13:00
|
|
|
KclValue::Solid { value } => to_vec_sr(&value.meta),
|
2024-11-21 10:47:32 -06:00
|
|
|
KclValue::Solids { value } => value.iter().flat_map(|eg| to_vec_sr(&eg.meta)).collect(),
|
|
|
|
KclValue::Sketch { value } => to_vec_sr(&value.meta),
|
|
|
|
KclValue::Sketches { value } => value.iter().flat_map(|eg| to_vec_sr(&eg.meta)).collect(),
|
2025-01-22 09:42:09 +13:00
|
|
|
KclValue::Helix { value } => to_vec_sr(&value.meta),
|
2024-11-21 10:47:32 -06:00
|
|
|
KclValue::ImportedGeometry(i) => to_vec_sr(&i.meta),
|
|
|
|
KclValue::Function { meta, .. } => to_vec_sr(&meta),
|
2025-01-22 09:42:09 +13:00
|
|
|
KclValue::Plane { value } => to_vec_sr(&value.meta),
|
|
|
|
KclValue::Face { value } => to_vec_sr(&value.meta),
|
2024-11-21 10:47:32 -06:00
|
|
|
KclValue::Bool { meta, .. } => to_vec_sr(&meta),
|
|
|
|
KclValue::Number { meta, .. } => to_vec_sr(&meta),
|
|
|
|
KclValue::String { meta, .. } => to_vec_sr(&meta),
|
|
|
|
KclValue::Array { meta, .. } => to_vec_sr(&meta),
|
|
|
|
KclValue::Object { meta, .. } => to_vec_sr(&meta),
|
2024-12-17 09:38:32 +13:00
|
|
|
KclValue::Module { meta, .. } => to_vec_sr(&meta),
|
2024-11-21 10:47:32 -06:00
|
|
|
KclValue::Uuid { meta, .. } => to_vec_sr(&meta),
|
2025-03-08 03:53:34 +13:00
|
|
|
KclValue::Type { meta, .. } => to_vec_sr(&meta),
|
2024-11-21 10:47:32 -06:00
|
|
|
KclValue::KclNone { meta, .. } => to_vec_sr(&meta),
|
2025-02-12 10:22:56 +13:00
|
|
|
KclValue::Tombstone { .. } => unreachable!("Tombstone SourceRange"),
|
2024-11-21 10:47:32 -06:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn to_vec_sr(meta: &[Metadata]) -> Vec<SourceRange> {
|
|
|
|
meta.iter().map(|m| m.source_range).collect()
|
|
|
|
}
|
|
|
|
|
|
|
|
impl From<&KclValue> for Vec<SourceRange> {
|
|
|
|
fn from(item: &KclValue) -> Self {
|
|
|
|
match item {
|
2024-12-03 16:39:51 +13:00
|
|
|
KclValue::TagDeclarator(t) => vec![SourceRange::new(t.start, t.end, t.module_id)],
|
2024-11-21 10:47:32 -06:00
|
|
|
KclValue::TagIdentifier(t) => to_vec_sr(&t.meta),
|
2025-01-22 09:42:09 +13:00
|
|
|
KclValue::Solid { value } => to_vec_sr(&value.meta),
|
2024-11-21 10:47:32 -06:00
|
|
|
KclValue::Solids { value } => value.iter().flat_map(|eg| to_vec_sr(&eg.meta)).collect(),
|
|
|
|
KclValue::Sketch { value } => to_vec_sr(&value.meta),
|
|
|
|
KclValue::Sketches { value } => value.iter().flat_map(|eg| to_vec_sr(&eg.meta)).collect(),
|
2025-01-22 09:42:09 +13:00
|
|
|
KclValue::Helix { value } => to_vec_sr(&value.meta),
|
2024-11-21 10:47:32 -06:00
|
|
|
KclValue::ImportedGeometry(i) => to_vec_sr(&i.meta),
|
|
|
|
KclValue::Function { meta, .. } => to_vec_sr(meta),
|
2025-01-22 09:42:09 +13:00
|
|
|
KclValue::Plane { value } => to_vec_sr(&value.meta),
|
|
|
|
KclValue::Face { value } => to_vec_sr(&value.meta),
|
2024-11-21 10:47:32 -06:00
|
|
|
KclValue::Bool { meta, .. } => to_vec_sr(meta),
|
|
|
|
KclValue::Number { meta, .. } => to_vec_sr(meta),
|
|
|
|
KclValue::String { meta, .. } => to_vec_sr(meta),
|
|
|
|
KclValue::Uuid { meta, .. } => to_vec_sr(meta),
|
|
|
|
KclValue::Array { meta, .. } => to_vec_sr(meta),
|
|
|
|
KclValue::Object { meta, .. } => to_vec_sr(meta),
|
2024-12-17 09:38:32 +13:00
|
|
|
KclValue::Module { meta, .. } => to_vec_sr(meta),
|
2024-11-21 10:47:32 -06:00
|
|
|
KclValue::KclNone { meta, .. } => to_vec_sr(meta),
|
2025-03-08 03:53:34 +13:00
|
|
|
KclValue::Type { meta, .. } => to_vec_sr(meta),
|
2025-02-12 10:22:56 +13:00
|
|
|
KclValue::Tombstone { .. } => unreachable!("Tombstone &SourceRange"),
|
2024-11-21 10:47:32 -06:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2025-03-08 03:53:34 +13:00
|
|
|
impl From<&KclValue> for SourceRange {
|
|
|
|
fn from(item: &KclValue) -> Self {
|
|
|
|
let v: Vec<_> = item.into();
|
|
|
|
v.into_iter().next().unwrap_or_default()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-11-21 10:47:32 -06:00
|
|
|
impl KclValue {
|
2024-11-26 12:27:09 -06:00
|
|
|
pub(crate) fn metadata(&self) -> Vec<Metadata> {
|
|
|
|
match self {
|
|
|
|
KclValue::Uuid { value: _, meta } => meta.clone(),
|
|
|
|
KclValue::Bool { value: _, meta } => meta.clone(),
|
2025-02-14 13:03:23 +13:00
|
|
|
KclValue::Number { meta, .. } => meta.clone(),
|
2024-11-26 12:27:09 -06:00
|
|
|
KclValue::String { value: _, meta } => meta.clone(),
|
|
|
|
KclValue::Array { value: _, meta } => meta.clone(),
|
|
|
|
KclValue::Object { value: _, meta } => meta.clone(),
|
|
|
|
KclValue::TagIdentifier(x) => x.meta.clone(),
|
|
|
|
KclValue::TagDeclarator(x) => vec![x.metadata()],
|
2025-01-22 09:42:09 +13:00
|
|
|
KclValue::Plane { value } => value.meta.clone(),
|
|
|
|
KclValue::Face { value } => value.meta.clone(),
|
2024-11-26 12:27:09 -06:00
|
|
|
KclValue::Sketch { value } => value.meta.clone(),
|
|
|
|
KclValue::Sketches { value } => value.iter().flat_map(|sketch| &sketch.meta).copied().collect(),
|
2025-01-22 09:42:09 +13:00
|
|
|
KclValue::Solid { value } => value.meta.clone(),
|
2024-11-26 12:27:09 -06:00
|
|
|
KclValue::Solids { value } => value.iter().flat_map(|sketch| &sketch.meta).copied().collect(),
|
2025-01-22 09:42:09 +13:00
|
|
|
KclValue::Helix { value } => value.meta.clone(),
|
2024-11-26 12:27:09 -06:00
|
|
|
KclValue::ImportedGeometry(x) => x.meta.clone(),
|
|
|
|
KclValue::Function { meta, .. } => meta.clone(),
|
2024-12-17 09:38:32 +13:00
|
|
|
KclValue::Module { meta, .. } => meta.clone(),
|
2024-11-26 12:27:09 -06:00
|
|
|
KclValue::KclNone { meta, .. } => meta.clone(),
|
2025-03-08 03:53:34 +13:00
|
|
|
KclValue::Type { meta, .. } => meta.clone(),
|
2025-02-12 10:22:56 +13:00
|
|
|
KclValue::Tombstone { .. } => unreachable!("Tombstone Metadata"),
|
2024-11-26 12:27:09 -06:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-12-16 13:10:31 -05:00
|
|
|
pub(crate) fn function_def_source_range(&self) -> Option<SourceRange> {
|
2025-02-22 20:16:29 +13:00
|
|
|
let KclValue::Function {
|
|
|
|
value: FunctionSource::User { ast, .. },
|
|
|
|
..
|
|
|
|
} = self
|
|
|
|
else {
|
2024-12-16 13:10:31 -05:00
|
|
|
return None;
|
|
|
|
};
|
|
|
|
// TODO: It would be nice if we could extract the source range starting
|
|
|
|
// at the fn, but that's the variable declaration.
|
2025-02-22 20:16:29 +13:00
|
|
|
Some(ast.as_source_range())
|
2024-12-16 13:10:31 -05:00
|
|
|
}
|
|
|
|
|
2024-11-26 12:27:09 -06:00
|
|
|
pub(crate) fn get_solid_set(&self) -> Result<SolidSet> {
|
|
|
|
match self {
|
2025-01-22 09:42:09 +13:00
|
|
|
KclValue::Solid { value } => Ok(SolidSet::Solid(value.clone())),
|
2024-11-26 12:27:09 -06:00
|
|
|
KclValue::Solids { value } => Ok(SolidSet::Solids(value.clone())),
|
|
|
|
KclValue::Array { value, .. } => {
|
|
|
|
let solids: Vec<_> = value
|
|
|
|
.iter()
|
|
|
|
.enumerate()
|
|
|
|
.map(|(i, v)| {
|
|
|
|
v.as_solid().map(|v| v.to_owned()).map(Box::new).ok_or_else(|| {
|
|
|
|
anyhow::anyhow!(
|
|
|
|
"expected this array to only contain solids, but element {i} was actually {}",
|
|
|
|
v.human_friendly_type()
|
|
|
|
)
|
|
|
|
})
|
|
|
|
})
|
|
|
|
.collect::<Result<_, _>>()?;
|
|
|
|
Ok(SolidSet::Solids(solids))
|
|
|
|
}
|
|
|
|
_ => anyhow::bail!("Not a solid or solids: {:?}", self),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-12-05 21:04:40 -06:00
|
|
|
#[allow(unused)]
|
|
|
|
pub(crate) fn none() -> Self {
|
|
|
|
Self::KclNone {
|
|
|
|
value: Default::default(),
|
|
|
|
meta: Default::default(),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-11-26 12:27:09 -06:00
|
|
|
/// Human readable type name used in error messages. Should not be relied
|
|
|
|
/// on for program logic.
|
|
|
|
pub(crate) fn human_friendly_type(&self) -> &'static str {
|
|
|
|
match self {
|
|
|
|
KclValue::Uuid { .. } => "Unique ID (uuid)",
|
|
|
|
KclValue::TagDeclarator(_) => "TagDeclarator",
|
|
|
|
KclValue::TagIdentifier(_) => "TagIdentifier",
|
2025-01-22 09:42:09 +13:00
|
|
|
KclValue::Solid { .. } => "Solid",
|
2024-11-26 12:27:09 -06:00
|
|
|
KclValue::Solids { .. } => "Solids",
|
|
|
|
KclValue::Sketch { .. } => "Sketch",
|
|
|
|
KclValue::Sketches { .. } => "Sketches",
|
2025-01-22 09:42:09 +13:00
|
|
|
KclValue::Helix { .. } => "Helix",
|
2024-11-26 12:27:09 -06:00
|
|
|
KclValue::ImportedGeometry(_) => "ImportedGeometry",
|
|
|
|
KclValue::Function { .. } => "Function",
|
2025-01-22 09:42:09 +13:00
|
|
|
KclValue::Plane { .. } => "Plane",
|
|
|
|
KclValue::Face { .. } => "Face",
|
2024-11-26 12:27:09 -06:00
|
|
|
KclValue::Bool { .. } => "boolean (true/false value)",
|
|
|
|
KclValue::Number { .. } => "number",
|
|
|
|
KclValue::String { .. } => "string (text)",
|
|
|
|
KclValue::Array { .. } => "array (list)",
|
|
|
|
KclValue::Object { .. } => "object",
|
2024-12-17 09:38:32 +13:00
|
|
|
KclValue::Module { .. } => "module",
|
2025-03-08 03:53:34 +13:00
|
|
|
KclValue::Type { .. } => "type",
|
2024-11-26 12:27:09 -06:00
|
|
|
KclValue::KclNone { .. } => "None",
|
2025-02-12 10:22:56 +13:00
|
|
|
KclValue::Tombstone { .. } => "TOMBSTONE",
|
2024-11-26 12:27:09 -06:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2025-02-14 13:03:23 +13:00
|
|
|
pub(crate) fn from_literal(literal: Node<Literal>, settings: &MetaSettings) -> Self {
|
|
|
|
let meta = vec![literal.metadata()];
|
|
|
|
match literal.inner.value {
|
|
|
|
LiteralValue::Number { value, suffix } => KclValue::Number {
|
|
|
|
value,
|
|
|
|
meta,
|
|
|
|
ty: NumericType::from_parsed(suffix, settings),
|
|
|
|
},
|
2024-12-05 21:04:40 -06:00
|
|
|
LiteralValue::String(value) => KclValue::String { value, meta },
|
|
|
|
LiteralValue::Bool(value) => KclValue::Bool { value, meta },
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2025-02-14 13:03:23 +13:00
|
|
|
pub(crate) fn from_default_param(param: DefaultParamVal, settings: &MetaSettings) -> Self {
|
|
|
|
match param {
|
|
|
|
DefaultParamVal::Literal(lit) => Self::from_literal(lit, settings),
|
|
|
|
DefaultParamVal::KclNone(none) => KclValue::KclNone {
|
|
|
|
value: none,
|
|
|
|
meta: Default::default(),
|
|
|
|
},
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2025-02-12 10:22:56 +13:00
|
|
|
pub(crate) fn map_env_ref(&self, env_map: &HashMap<EnvironmentRef, EnvironmentRef>) -> Self {
|
|
|
|
let mut result = self.clone();
|
2025-02-20 19:33:21 +13:00
|
|
|
if let KclValue::Function {
|
2025-02-22 20:16:29 +13:00
|
|
|
value: FunctionSource::User { ref mut memory, .. },
|
2025-02-20 19:33:21 +13:00
|
|
|
..
|
|
|
|
} = result
|
|
|
|
{
|
2025-02-12 10:22:56 +13:00
|
|
|
if let Some(new) = env_map.get(memory) {
|
|
|
|
*memory = *new;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
result
|
|
|
|
}
|
|
|
|
|
2024-11-21 10:47:32 -06:00
|
|
|
/// Put the number into a KCL value.
|
2024-11-26 12:27:09 -06:00
|
|
|
pub const fn from_number(f: f64, meta: Vec<Metadata>) -> Self {
|
2025-02-14 13:03:23 +13:00
|
|
|
Self::Number {
|
|
|
|
value: f,
|
|
|
|
meta,
|
|
|
|
ty: NumericType::Unknown,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub const fn from_number_with_type(f: f64, ty: NumericType, meta: Vec<Metadata>) -> Self {
|
|
|
|
Self::Number { value: f, meta, ty }
|
2024-11-21 10:47:32 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Put the point into a KCL value.
|
2025-02-14 13:03:23 +13:00
|
|
|
pub fn from_point2d(p: [f64; 2], ty: NumericType, meta: Vec<Metadata>) -> Self {
|
2024-11-21 10:47:32 -06:00
|
|
|
Self::Array {
|
|
|
|
value: vec![
|
|
|
|
Self::Number {
|
|
|
|
value: p[0],
|
|
|
|
meta: meta.clone(),
|
2025-02-14 13:03:23 +13:00
|
|
|
ty: ty.clone(),
|
2024-11-21 10:47:32 -06:00
|
|
|
},
|
|
|
|
Self::Number {
|
|
|
|
value: p[1],
|
|
|
|
meta: meta.clone(),
|
2025-02-14 13:03:23 +13:00
|
|
|
ty,
|
2024-11-21 10:47:32 -06:00
|
|
|
},
|
|
|
|
],
|
|
|
|
meta,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub(crate) fn as_usize(&self) -> Option<usize> {
|
|
|
|
match self {
|
2024-11-25 10:50:43 +13:00
|
|
|
KclValue::Number { value, .. } => crate::try_f64_to_usize(*value),
|
2024-11-21 10:47:32 -06:00
|
|
|
_ => None,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn as_int(&self) -> Option<i64> {
|
2024-11-25 10:50:43 +13:00
|
|
|
match self {
|
|
|
|
KclValue::Number { value, .. } => crate::try_f64_to_i64(*value),
|
|
|
|
_ => None,
|
2024-11-21 10:47:32 -06:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-11-26 09:51:43 -06:00
|
|
|
pub fn as_object(&self) -> Option<&KclObjectFields> {
|
2024-11-21 10:47:32 -06:00
|
|
|
if let KclValue::Object { value, meta: _ } = &self {
|
|
|
|
Some(value)
|
|
|
|
} else {
|
|
|
|
None
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-11-26 09:51:43 -06:00
|
|
|
pub fn into_object(self) -> Option<KclObjectFields> {
|
2024-11-21 10:47:32 -06:00
|
|
|
if let KclValue::Object { value, meta: _ } = self {
|
|
|
|
Some(value)
|
|
|
|
} else {
|
|
|
|
None
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn as_str(&self) -> Option<&str> {
|
|
|
|
if let KclValue::String { value, meta: _ } = &self {
|
|
|
|
Some(value)
|
|
|
|
} else {
|
|
|
|
None
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn as_array(&self) -> Option<&[KclValue]> {
|
|
|
|
if let KclValue::Array { value, meta: _ } = &self {
|
|
|
|
Some(value)
|
|
|
|
} else {
|
|
|
|
None
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn as_point2d(&self) -> Option<[f64; 2]> {
|
|
|
|
let arr = self.as_array()?;
|
|
|
|
if arr.len() != 2 {
|
|
|
|
return None;
|
|
|
|
}
|
|
|
|
let x = arr[0].as_f64()?;
|
|
|
|
let y = arr[1].as_f64()?;
|
|
|
|
Some([x, y])
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn as_uuid(&self) -> Option<uuid::Uuid> {
|
|
|
|
if let KclValue::Uuid { value, meta: _ } = &self {
|
|
|
|
Some(*value)
|
|
|
|
} else {
|
|
|
|
None
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn as_plane(&self) -> Option<&Plane> {
|
2025-01-22 09:42:09 +13:00
|
|
|
if let KclValue::Plane { value } = &self {
|
2024-11-21 10:47:32 -06:00
|
|
|
Some(value)
|
|
|
|
} else {
|
|
|
|
None
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn as_solid(&self) -> Option<&Solid> {
|
2025-01-22 09:42:09 +13:00
|
|
|
if let KclValue::Solid { value } = &self {
|
2024-11-21 10:47:32 -06:00
|
|
|
Some(value)
|
|
|
|
} else {
|
|
|
|
None
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2025-02-12 10:22:56 +13:00
|
|
|
pub fn as_sketch(&self) -> Option<&Sketch> {
|
|
|
|
if let KclValue::Sketch { value } = self {
|
|
|
|
Some(value)
|
|
|
|
} else {
|
|
|
|
None
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-11-21 10:47:32 -06:00
|
|
|
pub fn as_f64(&self) -> Option<f64> {
|
2025-02-14 13:03:23 +13:00
|
|
|
if let KclValue::Number { value, .. } = &self {
|
2024-11-21 10:47:32 -06:00
|
|
|
Some(*value)
|
|
|
|
} else {
|
|
|
|
None
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn as_bool(&self) -> Option<bool> {
|
|
|
|
if let KclValue::Bool { value, meta: _ } = &self {
|
|
|
|
Some(*value)
|
|
|
|
} else {
|
|
|
|
None
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// If this value fits in a u32, return it.
|
|
|
|
pub fn get_u32(&self, source_ranges: Vec<SourceRange>) -> Result<u32, KclError> {
|
|
|
|
let u = self.as_int().and_then(|n| u64::try_from(n).ok()).ok_or_else(|| {
|
|
|
|
KclError::Semantic(KclErrorDetails {
|
|
|
|
message: "Expected an integer >= 0".to_owned(),
|
|
|
|
source_ranges: source_ranges.clone(),
|
|
|
|
})
|
|
|
|
})?;
|
|
|
|
u32::try_from(u).map_err(|_| {
|
|
|
|
KclError::Semantic(KclErrorDetails {
|
|
|
|
message: "Number was too big".to_owned(),
|
|
|
|
source_ranges,
|
|
|
|
})
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
/// If this value is of type function, return it.
|
2025-02-22 20:16:29 +13:00
|
|
|
pub fn get_function(&self) -> Option<&FunctionSource> {
|
|
|
|
match self {
|
|
|
|
KclValue::Function { value, .. } => Some(value),
|
|
|
|
_ => None,
|
|
|
|
}
|
2024-11-21 10:47:32 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Get a tag identifier from a memory item.
|
|
|
|
pub fn get_tag_identifier(&self) -> Result<TagIdentifier, KclError> {
|
|
|
|
match self {
|
|
|
|
KclValue::TagIdentifier(t) => Ok(*t.clone()),
|
|
|
|
_ => Err(KclError::Semantic(KclErrorDetails {
|
|
|
|
message: format!("Not a tag identifier: {:?}", self),
|
|
|
|
source_ranges: self.clone().into(),
|
|
|
|
})),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Get a tag declarator from a memory item.
|
|
|
|
pub fn get_tag_declarator(&self) -> Result<TagNode, KclError> {
|
|
|
|
match self {
|
|
|
|
KclValue::TagDeclarator(t) => Ok((**t).clone()),
|
|
|
|
_ => Err(KclError::Semantic(KclErrorDetails {
|
|
|
|
message: format!("Not a tag declarator: {:?}", self),
|
|
|
|
source_ranges: self.clone().into(),
|
|
|
|
})),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Get an optional tag from a memory item.
|
|
|
|
pub fn get_tag_declarator_opt(&self) -> Result<Option<TagNode>, KclError> {
|
|
|
|
match self {
|
|
|
|
KclValue::TagDeclarator(t) => Ok(Some((**t).clone())),
|
|
|
|
_ => Err(KclError::Semantic(KclErrorDetails {
|
|
|
|
message: format!("Not a tag declarator: {:?}", self),
|
|
|
|
source_ranges: self.clone().into(),
|
|
|
|
})),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// If this KCL value is a bool, retrieve it.
|
|
|
|
pub fn get_bool(&self) -> Result<bool, KclError> {
|
|
|
|
let Self::Bool { value: b, .. } = self else {
|
|
|
|
return Err(KclError::Type(KclErrorDetails {
|
|
|
|
source_ranges: self.into(),
|
|
|
|
message: format!("Expected bool, found {}", self.human_friendly_type()),
|
|
|
|
}));
|
|
|
|
};
|
|
|
|
Ok(*b)
|
|
|
|
}
|
|
|
|
|
2025-02-27 15:58:58 +13:00
|
|
|
/// True if `self` has a type which is a subtype of `ty` without coercion.
|
|
|
|
pub fn has_type(&self, ty: &RuntimeType) -> bool {
|
2025-03-04 22:53:31 +13:00
|
|
|
let Some(self_ty) = self.principal_type() else {
|
2025-02-27 15:58:58 +13:00
|
|
|
return false;
|
|
|
|
};
|
|
|
|
|
|
|
|
self_ty.subtype(ty)
|
|
|
|
}
|
|
|
|
|
2025-03-04 22:53:31 +13:00
|
|
|
pub fn principal_type(&self) -> Option<RuntimeType> {
|
2025-02-27 15:58:58 +13:00
|
|
|
match self {
|
|
|
|
KclValue::Bool { .. } => Some(RuntimeType::Primitive(PrimitiveType::Boolean)),
|
|
|
|
KclValue::Number { ty, .. } => Some(RuntimeType::Primitive(PrimitiveType::Number(ty.clone()))),
|
|
|
|
KclValue::String { .. } => Some(RuntimeType::Primitive(PrimitiveType::String)),
|
|
|
|
KclValue::Object { value, .. } => {
|
|
|
|
let properties = value
|
|
|
|
.iter()
|
2025-03-04 22:53:31 +13:00
|
|
|
.map(|(k, v)| v.principal_type().map(|t| (k.clone(), t)))
|
2025-02-27 15:58:58 +13:00
|
|
|
.collect::<Option<Vec<_>>>()?;
|
|
|
|
Some(RuntimeType::Object(properties))
|
|
|
|
}
|
|
|
|
KclValue::Plane { .. } => Some(RuntimeType::Primitive(PrimitiveType::Plane)),
|
|
|
|
KclValue::Sketch { .. } => Some(RuntimeType::Primitive(PrimitiveType::Sketch)),
|
|
|
|
KclValue::Sketches { .. } => Some(RuntimeType::Array(PrimitiveType::Sketch)),
|
|
|
|
KclValue::Solid { .. } => Some(RuntimeType::Primitive(PrimitiveType::Solid)),
|
|
|
|
KclValue::Solids { .. } => Some(RuntimeType::Array(PrimitiveType::Solid)),
|
|
|
|
KclValue::Array { value, .. } => Some(RuntimeType::Tuple(
|
|
|
|
value
|
|
|
|
.iter()
|
2025-03-04 22:53:31 +13:00
|
|
|
.map(|v| v.principal_type().and_then(RuntimeType::primitive))
|
2025-02-27 15:58:58 +13:00
|
|
|
.collect::<Option<Vec<_>>>()?,
|
|
|
|
)),
|
|
|
|
KclValue::Face { .. } => None,
|
2025-03-08 03:53:34 +13:00
|
|
|
KclValue::Helix { .. }
|
|
|
|
| KclValue::ImportedGeometry(..)
|
|
|
|
| KclValue::Function { .. }
|
|
|
|
| KclValue::Module { .. }
|
|
|
|
| KclValue::TagIdentifier(_)
|
|
|
|
| KclValue::TagDeclarator(_)
|
|
|
|
| KclValue::KclNone { .. }
|
|
|
|
| KclValue::Type { .. }
|
|
|
|
| KclValue::Uuid { .. }
|
|
|
|
| KclValue::Tombstone { .. } => None,
|
2025-02-27 15:58:58 +13:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-11-21 10:47:32 -06:00
|
|
|
/// If this memory item is a function, call it with the given arguments, return its val as Ok.
|
|
|
|
/// If it's not a function, return Err.
|
|
|
|
pub async fn call_fn(
|
|
|
|
&self,
|
2024-11-21 13:10:03 -05:00
|
|
|
args: Vec<Arg>,
|
2024-11-21 10:47:32 -06:00
|
|
|
exec_state: &mut ExecState,
|
|
|
|
ctx: ExecutorContext,
|
2025-02-20 19:33:21 +13:00
|
|
|
source_range: SourceRange,
|
2024-11-21 10:47:32 -06:00
|
|
|
) -> Result<Option<KclValue>, KclError> {
|
2025-02-22 20:16:29 +13:00
|
|
|
match self {
|
|
|
|
KclValue::Function {
|
|
|
|
value: FunctionSource::Std { func, props },
|
|
|
|
..
|
|
|
|
} => {
|
|
|
|
if props.deprecated {
|
|
|
|
exec_state.warn(CompilationError::err(
|
|
|
|
source_range,
|
|
|
|
format!(
|
|
|
|
"`{}` is deprecated, see the docs for a recommended replacement",
|
|
|
|
props.name
|
|
|
|
),
|
|
|
|
));
|
|
|
|
}
|
2025-03-05 12:03:32 +13:00
|
|
|
exec_state.mut_stack().push_new_env_for_rust_call();
|
2025-02-22 20:16:29 +13:00
|
|
|
let args = crate::std::Args::new(
|
|
|
|
args,
|
|
|
|
source_range,
|
|
|
|
ctx.clone(),
|
|
|
|
exec_state.mod_local.pipe_value.clone().map(Arg::synthetic),
|
|
|
|
);
|
|
|
|
let result = func(exec_state, args).await.map(Some);
|
2025-03-05 12:03:32 +13:00
|
|
|
exec_state.mut_stack().pop_env();
|
2025-02-22 20:16:29 +13:00
|
|
|
result
|
|
|
|
}
|
|
|
|
KclValue::Function {
|
|
|
|
value: FunctionSource::User { ast, memory },
|
|
|
|
..
|
|
|
|
} => crate::execution::exec_ast::call_user_defined_function(args, *memory, ast, exec_state, &ctx).await,
|
|
|
|
_ => Err(KclError::Semantic(KclErrorDetails {
|
|
|
|
message: "cannot call this because it isn't a function".to_string(),
|
|
|
|
source_ranges: vec![source_range],
|
|
|
|
})),
|
2024-11-21 10:47:32 -06:00
|
|
|
}
|
|
|
|
}
|
2024-12-09 22:11:16 -06:00
|
|
|
|
|
|
|
/// If this is a function, call it by applying keyword arguments.
|
|
|
|
/// If it's not a function, returns an error.
|
|
|
|
pub async fn call_fn_kw(
|
|
|
|
&self,
|
|
|
|
args: crate::std::Args,
|
|
|
|
exec_state: &mut ExecState,
|
|
|
|
ctx: ExecutorContext,
|
|
|
|
callsite: SourceRange,
|
|
|
|
) -> Result<Option<KclValue>, KclError> {
|
2025-02-22 20:16:29 +13:00
|
|
|
match self {
|
|
|
|
KclValue::Function {
|
|
|
|
value: FunctionSource::Std { func: _, props },
|
|
|
|
..
|
|
|
|
} => {
|
|
|
|
if props.deprecated {
|
|
|
|
exec_state.warn(CompilationError::err(
|
|
|
|
callsite,
|
|
|
|
format!(
|
|
|
|
"`{}` is deprecated, see the docs for a recommended replacement",
|
|
|
|
props.name
|
|
|
|
),
|
|
|
|
));
|
|
|
|
}
|
|
|
|
todo!("Implement KCL stdlib fns with keyword args");
|
|
|
|
}
|
|
|
|
KclValue::Function {
|
|
|
|
value: FunctionSource::User { ast, memory },
|
|
|
|
..
|
|
|
|
} => {
|
|
|
|
crate::execution::exec_ast::call_user_defined_function_kw(args.kw_args, *memory, ast, exec_state, &ctx)
|
|
|
|
.await
|
|
|
|
}
|
|
|
|
_ => Err(KclError::Semantic(KclErrorDetails {
|
2024-12-09 22:11:16 -06:00
|
|
|
message: "cannot call this because it isn't a function".to_string(),
|
|
|
|
source_ranges: vec![callsite],
|
2025-02-22 20:16:29 +13:00
|
|
|
})),
|
2024-12-09 22:11:16 -06:00
|
|
|
}
|
|
|
|
}
|
2025-03-04 22:53:31 +13:00
|
|
|
|
|
|
|
pub fn value_str(&self) -> Option<String> {
|
|
|
|
match self {
|
|
|
|
KclValue::Bool { value, .. } => Some(format!("{value}")),
|
|
|
|
KclValue::Number { value, .. } => Some(format!("{value}")),
|
|
|
|
KclValue::String { value, .. } => Some(format!("'{value}'")),
|
|
|
|
KclValue::Uuid { value, .. } => Some(format!("{value}")),
|
|
|
|
KclValue::TagDeclarator(tag) => Some(format!("${}", tag.name)),
|
|
|
|
KclValue::TagIdentifier(tag) => Some(format!("${}", tag.value)),
|
|
|
|
// TODO better Array and Object stringification
|
|
|
|
KclValue::Array { .. } => Some("[...]".to_owned()),
|
|
|
|
KclValue::Object { .. } => Some("{ ... }".to_owned()),
|
|
|
|
KclValue::Module { .. }
|
|
|
|
| KclValue::Solid { .. }
|
|
|
|
| KclValue::Solids { .. }
|
|
|
|
| KclValue::Sketch { .. }
|
|
|
|
| KclValue::Sketches { .. }
|
|
|
|
| KclValue::Helix { .. }
|
|
|
|
| KclValue::ImportedGeometry(_)
|
|
|
|
| KclValue::Function { .. }
|
|
|
|
| KclValue::Plane { .. }
|
|
|
|
| KclValue::Face { .. }
|
|
|
|
| KclValue::KclNone { .. }
|
2025-03-08 03:53:34 +13:00
|
|
|
| KclValue::Type { .. }
|
2025-03-04 22:53:31 +13:00
|
|
|
| KclValue::Tombstone { .. } => None,
|
|
|
|
}
|
|
|
|
}
|
2024-11-21 10:47:32 -06:00
|
|
|
}
|
2024-12-17 15:23:00 +13:00
|
|
|
|
2025-02-27 15:58:58 +13:00
|
|
|
#[derive(Debug, Clone, PartialEq)]
|
|
|
|
pub enum RuntimeType {
|
|
|
|
Primitive(PrimitiveType),
|
|
|
|
Array(PrimitiveType),
|
|
|
|
Tuple(Vec<PrimitiveType>),
|
|
|
|
Object(Vec<(String, RuntimeType)>),
|
|
|
|
}
|
|
|
|
|
|
|
|
impl RuntimeType {
|
2025-03-08 03:53:34 +13:00
|
|
|
pub fn from_parsed(
|
|
|
|
value: Type,
|
|
|
|
exec_state: &mut ExecState,
|
|
|
|
source_range: SourceRange,
|
|
|
|
) -> Result<Option<Self>, CompilationError> {
|
|
|
|
Ok(match value {
|
|
|
|
Type::Primitive(pt) => {
|
|
|
|
PrimitiveType::from_parsed(pt, exec_state, source_range)?.map(RuntimeType::Primitive)
|
|
|
|
}
|
|
|
|
Type::Array(pt) => PrimitiveType::from_parsed(pt, exec_state, source_range)?.map(RuntimeType::Array),
|
|
|
|
Type::Object { properties } => properties
|
|
|
|
.into_iter()
|
|
|
|
.map(|p| {
|
|
|
|
let pt = match p.type_ {
|
|
|
|
Some(t) => t,
|
|
|
|
None => return Ok(None),
|
|
|
|
};
|
|
|
|
Ok(RuntimeType::from_parsed(pt.inner, exec_state, source_range)?
|
|
|
|
.map(|ty| (p.identifier.inner.name, ty)))
|
|
|
|
})
|
|
|
|
.collect::<Result<Option<Vec<_>>, CompilationError>>()?
|
|
|
|
.map(RuntimeType::Object),
|
|
|
|
})
|
2025-02-27 15:58:58 +13:00
|
|
|
}
|
|
|
|
|
|
|
|
// Subtype with no coercion, including refining numeric types.
|
|
|
|
fn subtype(&self, sup: &RuntimeType) -> bool {
|
|
|
|
use RuntimeType::*;
|
|
|
|
|
|
|
|
match (self, sup) {
|
|
|
|
// TODO arrays could be covariant
|
|
|
|
(Primitive(t1), Primitive(t2)) | (Array(t1), Array(t2)) => t1 == t2,
|
|
|
|
(Tuple(t1), Tuple(t2)) => t1 == t2,
|
|
|
|
(Tuple(t1), Array(t2)) => t1.iter().all(|t| t == t2),
|
|
|
|
// TODO record subtyping - subtype can be larger, fields can be covariant.
|
|
|
|
(Object(t1), Object(t2)) => t1 == t2,
|
|
|
|
_ => false,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn primitive(self) -> Option<PrimitiveType> {
|
|
|
|
match self {
|
|
|
|
RuntimeType::Primitive(t) => Some(t),
|
|
|
|
_ => None,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2025-03-04 22:53:31 +13:00
|
|
|
impl fmt::Display for RuntimeType {
|
|
|
|
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
|
|
|
|
match self {
|
|
|
|
RuntimeType::Primitive(t) => t.fmt(f),
|
|
|
|
RuntimeType::Array(t) => write!(f, "[{t}]"),
|
|
|
|
RuntimeType::Tuple(ts) => write!(
|
|
|
|
f,
|
|
|
|
"[{}]",
|
|
|
|
ts.iter().map(|t| t.to_string()).collect::<Vec<_>>().join(", ")
|
|
|
|
),
|
|
|
|
RuntimeType::Object(items) => write!(
|
|
|
|
f,
|
|
|
|
"{{ {} }}",
|
|
|
|
items
|
|
|
|
.iter()
|
|
|
|
.map(|(n, t)| format!("{n}: {t}"))
|
|
|
|
.collect::<Vec<_>>()
|
|
|
|
.join(", ")
|
|
|
|
),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2025-02-27 15:58:58 +13:00
|
|
|
#[derive(Debug, Clone, PartialEq)]
|
|
|
|
pub enum PrimitiveType {
|
|
|
|
Number(NumericType),
|
|
|
|
String,
|
|
|
|
Boolean,
|
|
|
|
Sketch,
|
|
|
|
Solid,
|
|
|
|
Plane,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl PrimitiveType {
|
2025-03-08 03:53:34 +13:00
|
|
|
fn from_parsed(
|
|
|
|
value: AstPrimitiveType,
|
|
|
|
exec_state: &mut ExecState,
|
|
|
|
source_range: SourceRange,
|
|
|
|
) -> Result<Option<Self>, CompilationError> {
|
|
|
|
Ok(match value {
|
2025-02-27 15:58:58 +13:00
|
|
|
AstPrimitiveType::String => Some(PrimitiveType::String),
|
|
|
|
AstPrimitiveType::Boolean => Some(PrimitiveType::Boolean),
|
2025-03-08 03:53:34 +13:00
|
|
|
AstPrimitiveType::Number(suffix) => Some(PrimitiveType::Number(NumericType::from_parsed(
|
|
|
|
suffix,
|
|
|
|
&exec_state.mod_local.settings,
|
|
|
|
))),
|
|
|
|
AstPrimitiveType::Named(name) => {
|
|
|
|
let ty_val = exec_state
|
|
|
|
.stack()
|
|
|
|
.get(&format!("{}{}", memory::TYPE_PREFIX, name.name), source_range)
|
|
|
|
.map_err(|_| CompilationError::err(source_range, format!("Unknown type: {}", name.name)))?;
|
|
|
|
|
|
|
|
let (ty, _) = match ty_val {
|
|
|
|
KclValue::Type { value: Some(ty), .. } => ty,
|
|
|
|
_ => unreachable!(),
|
|
|
|
};
|
|
|
|
|
|
|
|
Some(ty.clone())
|
|
|
|
}
|
2025-02-27 15:58:58 +13:00
|
|
|
_ => None,
|
2025-03-08 03:53:34 +13:00
|
|
|
})
|
2025-02-27 15:58:58 +13:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2025-03-04 22:53:31 +13:00
|
|
|
impl fmt::Display for PrimitiveType {
|
|
|
|
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
|
|
|
|
match self {
|
|
|
|
PrimitiveType::Number(NumericType::Known(unit)) => write!(f, "number({unit})"),
|
|
|
|
PrimitiveType::Number(_) => write!(f, "number"),
|
|
|
|
PrimitiveType::String => write!(f, "string"),
|
|
|
|
PrimitiveType::Boolean => write!(f, "bool"),
|
|
|
|
PrimitiveType::Sketch => write!(f, "Sketch"),
|
|
|
|
PrimitiveType::Solid => write!(f, "Solid"),
|
|
|
|
PrimitiveType::Plane => write!(f, "Plane"),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2025-02-14 13:03:23 +13:00
|
|
|
#[derive(Debug, Clone, Deserialize, Serialize, PartialEq, ts_rs::TS, JsonSchema)]
|
|
|
|
#[ts(export)]
|
|
|
|
#[serde(tag = "type")]
|
|
|
|
pub enum NumericType {
|
|
|
|
// Specified by the user (directly or indirectly)
|
|
|
|
Known(UnitType),
|
|
|
|
// Unspecified, using defaults
|
|
|
|
Default { len: UnitLen, angle: UnitAngle },
|
|
|
|
// Exceeded the ability of the type system to track.
|
|
|
|
Unknown,
|
|
|
|
// Type info has been explicitly cast away.
|
|
|
|
Any,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl NumericType {
|
|
|
|
pub fn count() -> Self {
|
|
|
|
NumericType::Known(UnitType::Count)
|
|
|
|
}
|
|
|
|
|
2025-02-20 10:12:37 +13:00
|
|
|
/// Combine two types when we expect them to be equal.
|
|
|
|
pub fn combine_eq(self, other: &NumericType) -> NumericType {
|
2025-02-14 13:03:23 +13:00
|
|
|
if &self == other {
|
|
|
|
self
|
|
|
|
} else {
|
|
|
|
NumericType::Unknown
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2025-02-20 10:12:37 +13:00
|
|
|
/// Combine n types when we expect them to be equal.
|
|
|
|
///
|
|
|
|
/// Precondition: tys.len() > 0
|
|
|
|
pub fn combine_n_eq(tys: &[NumericType]) -> NumericType {
|
|
|
|
let ty0 = tys[0].clone();
|
|
|
|
for t in &tys[1..] {
|
|
|
|
if t != &ty0 {
|
|
|
|
return NumericType::Unknown;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
ty0
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Combine two types in addition-like operations.
|
|
|
|
pub fn combine_add(a: NumericType, b: NumericType) -> NumericType {
|
|
|
|
if a == b {
|
|
|
|
return a;
|
|
|
|
}
|
|
|
|
NumericType::Unknown
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Combine two types in multiplication-like operations.
|
|
|
|
pub fn combine_mul(a: NumericType, b: NumericType) -> NumericType {
|
|
|
|
if a == NumericType::count() {
|
|
|
|
return b;
|
|
|
|
}
|
|
|
|
if b == NumericType::count() {
|
|
|
|
return a;
|
|
|
|
}
|
|
|
|
NumericType::Unknown
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Combine two types in division-like operations.
|
|
|
|
pub fn combine_div(a: NumericType, b: NumericType) -> NumericType {
|
|
|
|
if b == NumericType::count() {
|
|
|
|
return a;
|
|
|
|
}
|
|
|
|
NumericType::Unknown
|
|
|
|
}
|
|
|
|
|
2025-02-14 13:03:23 +13:00
|
|
|
pub fn from_parsed(suffix: NumericSuffix, settings: &super::MetaSettings) -> Self {
|
|
|
|
match suffix {
|
|
|
|
NumericSuffix::None => NumericType::Default {
|
|
|
|
len: settings.default_length_units,
|
|
|
|
angle: settings.default_angle_units,
|
|
|
|
},
|
|
|
|
NumericSuffix::Count => NumericType::Known(UnitType::Count),
|
|
|
|
NumericSuffix::Mm => NumericType::Known(UnitType::Length(UnitLen::Mm)),
|
|
|
|
NumericSuffix::Cm => NumericType::Known(UnitType::Length(UnitLen::Cm)),
|
|
|
|
NumericSuffix::M => NumericType::Known(UnitType::Length(UnitLen::M)),
|
|
|
|
NumericSuffix::Inch => NumericType::Known(UnitType::Length(UnitLen::Inches)),
|
|
|
|
NumericSuffix::Ft => NumericType::Known(UnitType::Length(UnitLen::Feet)),
|
|
|
|
NumericSuffix::Yd => NumericType::Known(UnitType::Length(UnitLen::Yards)),
|
|
|
|
NumericSuffix::Deg => NumericType::Known(UnitType::Angle(UnitAngle::Degrees)),
|
|
|
|
NumericSuffix::Rad => NumericType::Known(UnitType::Angle(UnitAngle::Radians)),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl From<UnitLen> for NumericType {
|
|
|
|
fn from(value: UnitLen) -> Self {
|
|
|
|
NumericType::Known(UnitType::Length(value))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl From<UnitAngle> for NumericType {
|
|
|
|
fn from(value: UnitAngle) -> Self {
|
|
|
|
NumericType::Known(UnitType::Angle(value))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Debug, Clone, Copy, Deserialize, Serialize, PartialEq, Eq, ts_rs::TS, JsonSchema)]
|
|
|
|
#[ts(export)]
|
|
|
|
#[serde(tag = "type")]
|
|
|
|
pub enum UnitType {
|
|
|
|
Count,
|
|
|
|
Length(UnitLen),
|
|
|
|
Angle(UnitAngle),
|
|
|
|
}
|
|
|
|
|
2025-03-04 22:53:31 +13:00
|
|
|
impl std::fmt::Display for UnitType {
|
|
|
|
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
|
|
|
|
match self {
|
|
|
|
UnitType::Count => write!(f, "_"),
|
|
|
|
UnitType::Length(l) => l.fmt(f),
|
|
|
|
UnitType::Angle(a) => a.fmt(f),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-12-17 15:23:00 +13:00
|
|
|
// TODO called UnitLen so as not to clash with UnitLength in settings)
|
2025-02-19 19:48:27 -08:00
|
|
|
/// A unit of length.
|
2025-01-31 13:11:15 -08:00
|
|
|
#[derive(Debug, Default, Clone, Copy, Deserialize, Serialize, PartialEq, ts_rs::TS, JsonSchema, Eq)]
|
2024-12-17 15:23:00 +13:00
|
|
|
#[ts(export)]
|
|
|
|
#[serde(tag = "type")]
|
|
|
|
pub enum UnitLen {
|
2025-01-31 13:11:15 -08:00
|
|
|
#[default]
|
2024-12-17 15:23:00 +13:00
|
|
|
Mm,
|
|
|
|
Cm,
|
|
|
|
M,
|
|
|
|
Inches,
|
|
|
|
Feet,
|
|
|
|
Yards,
|
|
|
|
}
|
|
|
|
|
2025-01-31 13:11:15 -08:00
|
|
|
impl std::fmt::Display for UnitLen {
|
|
|
|
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
|
|
|
|
match self {
|
|
|
|
UnitLen::Mm => write!(f, "mm"),
|
|
|
|
UnitLen::Cm => write!(f, "cm"),
|
|
|
|
UnitLen::M => write!(f, "m"),
|
|
|
|
UnitLen::Inches => write!(f, "in"),
|
|
|
|
UnitLen::Feet => write!(f, "ft"),
|
|
|
|
UnitLen::Yards => write!(f, "yd"),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-12-17 15:23:00 +13:00
|
|
|
impl TryFrom<NumericSuffix> for UnitLen {
|
|
|
|
type Error = ();
|
|
|
|
|
|
|
|
fn try_from(suffix: NumericSuffix) -> std::result::Result<Self, Self::Error> {
|
|
|
|
match suffix {
|
|
|
|
NumericSuffix::Mm => Ok(Self::Mm),
|
|
|
|
NumericSuffix::Cm => Ok(Self::Cm),
|
|
|
|
NumericSuffix::M => Ok(Self::M),
|
|
|
|
NumericSuffix::Inch => Ok(Self::Inches),
|
|
|
|
NumericSuffix::Ft => Ok(Self::Feet),
|
|
|
|
NumericSuffix::Yd => Ok(Self::Yards),
|
|
|
|
_ => Err(()),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2025-01-17 07:55:01 +13:00
|
|
|
impl From<crate::UnitLength> for UnitLen {
|
|
|
|
fn from(unit: crate::UnitLength) -> Self {
|
|
|
|
match unit {
|
|
|
|
crate::UnitLength::Cm => UnitLen::Cm,
|
|
|
|
crate::UnitLength::Ft => UnitLen::Feet,
|
|
|
|
crate::UnitLength::In => UnitLen::Inches,
|
|
|
|
crate::UnitLength::M => UnitLen::M,
|
|
|
|
crate::UnitLength::Mm => UnitLen::Mm,
|
|
|
|
crate::UnitLength::Yd => UnitLen::Yards,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2025-01-22 15:23:55 +13:00
|
|
|
impl From<UnitLen> for crate::UnitLength {
|
|
|
|
fn from(unit: UnitLen) -> Self {
|
|
|
|
match unit {
|
|
|
|
UnitLen::Cm => crate::UnitLength::Cm,
|
|
|
|
UnitLen::Feet => crate::UnitLength::Ft,
|
|
|
|
UnitLen::Inches => crate::UnitLength::In,
|
|
|
|
UnitLen::M => crate::UnitLength::M,
|
|
|
|
UnitLen::Mm => crate::UnitLength::Mm,
|
|
|
|
UnitLen::Yards => crate::UnitLength::Yd,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2025-02-13 06:24:27 +13:00
|
|
|
impl From<UnitLen> for kittycad_modeling_cmds::units::UnitLength {
|
|
|
|
fn from(unit: UnitLen) -> Self {
|
|
|
|
match unit {
|
|
|
|
UnitLen::Cm => kittycad_modeling_cmds::units::UnitLength::Centimeters,
|
|
|
|
UnitLen::Feet => kittycad_modeling_cmds::units::UnitLength::Feet,
|
|
|
|
UnitLen::Inches => kittycad_modeling_cmds::units::UnitLength::Inches,
|
|
|
|
UnitLen::M => kittycad_modeling_cmds::units::UnitLength::Meters,
|
|
|
|
UnitLen::Mm => kittycad_modeling_cmds::units::UnitLength::Millimeters,
|
|
|
|
UnitLen::Yards => kittycad_modeling_cmds::units::UnitLength::Yards,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2025-02-19 19:48:27 -08:00
|
|
|
/// A unit of angle.
|
2025-01-17 07:55:01 +13:00
|
|
|
#[derive(Debug, Default, Clone, Copy, Deserialize, Serialize, PartialEq, ts_rs::TS, JsonSchema, Eq)]
|
2024-12-17 15:23:00 +13:00
|
|
|
#[ts(export)]
|
|
|
|
#[serde(tag = "type")]
|
|
|
|
pub enum UnitAngle {
|
2025-01-17 07:55:01 +13:00
|
|
|
#[default]
|
2024-12-17 15:23:00 +13:00
|
|
|
Degrees,
|
|
|
|
Radians,
|
|
|
|
}
|
|
|
|
|
2025-01-31 13:11:15 -08:00
|
|
|
impl std::fmt::Display for UnitAngle {
|
|
|
|
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
|
|
|
|
match self {
|
|
|
|
UnitAngle::Degrees => write!(f, "deg"),
|
|
|
|
UnitAngle::Radians => write!(f, "rad"),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-12-17 15:23:00 +13:00
|
|
|
impl TryFrom<NumericSuffix> for UnitAngle {
|
|
|
|
type Error = ();
|
|
|
|
|
|
|
|
fn try_from(suffix: NumericSuffix) -> std::result::Result<Self, Self::Error> {
|
|
|
|
match suffix {
|
|
|
|
NumericSuffix::Deg => Ok(Self::Degrees),
|
|
|
|
NumericSuffix::Rad => Ok(Self::Radians),
|
|
|
|
_ => Err(()),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|