summaryrefslogtreecommitdiff
path: root/src/template.rs
blob: 0ac03cd4d45922e5cf8d65badb2f86f370c9aaa0 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123


use std::collections::HashMap;
use serde_json::{json, Value};
use crate::{
	parameter::Parameter,
	PResult,
	perr
};

#[derive(Debug, PartialEq, Eq, Clone, Hash)]
pub struct EntityType(pub String);

#[derive(Debug, Clone, Copy, Hash, PartialEq, Eq)]
pub enum SaveOption {
	Default,
	False,
	Always
}

#[derive(Debug, Clone, PartialEq)]
pub struct Template {
	pub name: EntityType,
	pub args: Vec<Parameter>,
	pub kwargs: HashMap<String, Parameter>,
	pub save: SaveOption,
}


impl Template {
	
	pub fn new(name: &str, kwargs: HashMap<String, Parameter>) -> Self {
		Self {
			name: EntityType(name.to_string()),
			args: Vec::new(),
			kwargs,
			save: SaveOption::Default
		}
	}
	
	pub fn empty(name: &str) -> Self {
		Self::new(name, HashMap::new())
	}
	
	pub fn should_save(&self) -> bool {
		match self.save {
			SaveOption::Default | SaveOption::Always => true,
			SaveOption::False => false
		}
	}
	
	pub fn from_entity_type(typ: EntityType) -> Self {
		Self {
			name: typ,
			args: Vec::new(),
			kwargs: HashMap::new(),
			save: SaveOption::Default
		}
	}
	
	pub fn unsaved(mut self) -> Self {
		if self.save == SaveOption::Default {
			self.save = SaveOption::False
		}
		self
	}
	
	pub fn merge(mut self, other: Template) -> Self {
		if self.save == SaveOption::Default {
			self.save = other.save;
		}
		for (key, value) in other.kwargs {
			self.kwargs.entry(key).or_insert(value);
		}
		self
	}
	
	pub fn from_json(v: &Value) -> PResult<Template> {
		let val = match v {
			Value::String(s) => json!({"type": s}),
			Value::Array(_) => json!({
				"type": v.get(0).ok_or(perr!("index 0 not in template array {:?}", v))?,
				"kwargs": v.get(1).ok_or(perr!("index 1 not in template array {:?}", v))?
			}),
			Value::Object(_) => v.clone(),
			_ => return Err(perr!("invalid template {:?}", v))
		};
			
		let name = EntityType(val.get("type").ok_or(perr!("template doesn't have 'type'"))?.as_str().ok_or(perr!("template type not a string"))?.to_string());
		let mut args = Vec::new();
		for arg in val.get("args").unwrap_or(&json!([])).as_array().ok_or(perr!("template args not an array"))? {
			args.push(Parameter::guess_from_json(arg).ok_or(perr!("template arg {:?} not a parameter", arg))?);
		}
		let mut kwargs = HashMap::new();
		for (key, arg) in val.get("kwargs").unwrap_or(&json!({})).as_object().ok_or(perr!("template kwargs not a json object"))? {
			kwargs.insert(key.to_string(), Parameter::guess_from_json(arg).ok_or(perr!("template kwarg {}: {:?} not a parameter", key, arg))?);
		}
		let save = 
			if let Some(saveval) = val.get("save") {
				if saveval.as_bool().ok_or(perr!("save not a bool"))? {
					SaveOption::Always
				} else {
					SaveOption::False
				}
			} else {
				SaveOption::Default
			};
		Ok(Template {name, args, kwargs, save})
	}
	
	pub fn to_json(&self) -> Value {
		if self.args.is_empty() && self.kwargs.is_empty() {
			return json!(self.name.0);
		}
		let jsonargs: Vec<Value> = self.args.iter().map(|a| a.to_json()).collect();
		let jsonkwargs: HashMap<&String, Value> = self.kwargs.iter().map(|(k, a)| (k, a.to_json())).collect();
		json!({
			"type": self.name.0,
			"args": jsonargs,
			"kwargs": jsonkwargs
		})
	}
}