data_modelling_sdk/models/
column.rs1use serde::{Deserialize, Serialize};
4use std::collections::HashMap;
5
6#[derive(Debug, Clone, Serialize, Deserialize, PartialEq)]
8pub struct ForeignKey {
9 pub table_id: String,
11 pub column_name: String,
13}
14
15#[derive(Debug, Clone, Serialize, Deserialize, PartialEq)]
28pub struct Column {
29 pub name: String,
31 pub data_type: String,
33 #[serde(default = "default_true")]
35 pub nullable: bool,
36 #[serde(default)]
38 pub primary_key: bool,
39 #[serde(default)]
41 pub secondary_key: bool,
42 #[serde(skip_serializing_if = "Option::is_none")]
44 pub composite_key: Option<String>,
45 #[serde(skip_serializing_if = "Option::is_none")]
47 pub foreign_key: Option<ForeignKey>,
48 #[serde(default)]
50 pub constraints: Vec<String>,
51 #[serde(default)]
53 pub description: String,
54 #[serde(default)]
56 pub errors: Vec<HashMap<String, serde_json::Value>>,
57 #[serde(default)]
59 pub quality: Vec<HashMap<String, serde_json::Value>>,
60 #[serde(default)]
62 pub enum_values: Vec<String>,
63 #[serde(default)]
65 pub column_order: i32,
66}
67
68fn default_true() -> bool {
69 true
70}
71
72impl Column {
73 pub fn new(name: String, data_type: String) -> Self {
92 Self {
93 name,
94 data_type: normalize_data_type(&data_type),
95 nullable: true,
96 primary_key: false,
97 secondary_key: false,
98 composite_key: None,
99 foreign_key: None,
100 constraints: Vec::new(),
101 description: String::new(),
102 errors: Vec::new(),
103 quality: Vec::new(),
104 enum_values: Vec::new(),
105 column_order: 0,
106 }
107 }
108}
109
110fn normalize_data_type(data_type: &str) -> String {
111 if data_type.is_empty() {
112 return data_type.to_string();
113 }
114
115 let upper = data_type.to_uppercase();
116
117 if upper.starts_with("STRUCT") {
119 if let Some(start) = data_type.find('<')
120 && let Some(end) = data_type.rfind('>')
121 {
122 let inner = &data_type[start + 1..end];
123 return format!("STRUCT<{}>", inner);
124 }
125 return format!("STRUCT{}", &data_type[6..]);
126 } else if upper.starts_with("ARRAY") {
127 if let Some(start) = data_type.find('<')
128 && let Some(end) = data_type.rfind('>')
129 {
130 let inner = &data_type[start + 1..end];
131 return format!("ARRAY<{}>", inner);
132 }
133 return format!("ARRAY{}", &data_type[5..]);
134 } else if upper.starts_with("MAP") {
135 if let Some(start) = data_type.find('<')
136 && let Some(end) = data_type.rfind('>')
137 {
138 let inner = &data_type[start + 1..end];
139 return format!("MAP<{}>", inner);
140 }
141 return format!("MAP{}", &data_type[3..]);
142 }
143
144 upper
145}