use serde::{Deserialize, Serialize};
use vantage_vista::{NoExtras, VistaSpec};
#[derive(Debug, Clone, Default, Serialize, Deserialize)]
#[serde(deny_unknown_fields)]
pub struct MysqlTableExtras {
#[serde(default, skip_serializing_if = "Option::is_none")]
pub mysql: Option<MysqlTableBlock>,
}
#[derive(Debug, Clone, Default, Serialize, Deserialize)]
#[serde(deny_unknown_fields)]
pub struct MysqlTableBlock {
#[serde(default, skip_serializing_if = "Option::is_none")]
pub table: Option<String>,
#[serde(default, skip_serializing_if = "Option::is_none")]
pub rhai: Option<String>,
#[serde(default, skip_serializing_if = "Option::is_none")]
pub base: Option<String>,
#[serde(default, skip_serializing_if = "Option::is_none")]
pub inherit: Option<InheritBlock>,
}
#[derive(Debug, Clone, Default, Serialize, Deserialize)]
#[serde(deny_unknown_fields)]
pub struct InheritBlock {
#[serde(default, skip_serializing_if = "Vec::is_empty")]
pub columns: Vec<String>,
#[serde(default, skip_serializing_if = "Vec::is_empty")]
pub relations: Vec<String>,
}
#[derive(Debug, Clone, Default, Serialize, Deserialize)]
#[serde(deny_unknown_fields)]
pub struct MysqlColumnExtras {
#[serde(default, skip_serializing_if = "Option::is_none")]
pub mysql: Option<MysqlColumnBlock>,
}
#[derive(Debug, Clone, Default, Serialize, Deserialize)]
#[serde(deny_unknown_fields)]
pub struct MysqlColumnBlock {
#[serde(default, skip_serializing_if = "Option::is_none")]
pub column: Option<String>,
}
pub type MysqlVistaSpec = VistaSpec<MysqlTableExtras, MysqlColumnExtras, NoExtras>;
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn yaml_parses_into_mysql_vista_spec() {
let yaml = r#"
name: client
columns:
id:
type: int
flags: [id]
name:
type: string
flags: [title]
mysql:
table: clients
"#;
let spec: MysqlVistaSpec = serde_yaml_ng::from_str(yaml).unwrap();
assert_eq!(spec.name, "client");
assert_eq!(
spec.driver.mysql.as_ref().and_then(|m| m.table.as_deref()),
Some("clients")
);
}
#[test]
fn yaml_rejects_unknown_mysql_block_field() {
let yaml = r#"
name: client
columns:
id: { type: int, flags: [id] }
mysql:
table: clients
bogus: 1
"#;
let err = serde_yaml_ng::from_str::<MysqlVistaSpec>(yaml).unwrap_err();
assert!(err.to_string().contains("bogus") || err.to_string().contains("unknown"));
}
#[test]
fn yaml_parses_base_inherit_and_rhai() {
let yaml = r#"
name: vip_clients
columns: {}
mysql:
base: client
inherit:
columns: [id, name]
relations: [orders]
rhai: |
base.where(expr("vip = true"))
"#;
let spec: MysqlVistaSpec = serde_yaml_ng::from_str(yaml).unwrap();
let block = spec.driver.mysql.as_ref().unwrap();
assert_eq!(block.base.as_deref(), Some("client"));
assert!(block.rhai.as_ref().unwrap().contains("base.where"));
let inherit = block.inherit.as_ref().unwrap();
assert_eq!(inherit.columns, vec!["id", "name"]);
assert_eq!(inherit.relations, vec!["orders"]);
}
}