par_term_config/
profile.rs1use serde::{Deserialize, Serialize};
7use std::collections::HashMap;
8
9fn default_refresh_interval_secs() -> u64 {
12 1800
13}
14
15fn default_max_size_bytes() -> usize {
16 1_048_576
17}
18
19fn default_fetch_timeout_secs() -> u64 {
20 10
21}
22
23fn default_true() -> bool {
24 true
25}
26
27#[derive(Debug, Clone, Default, Serialize, Deserialize, PartialEq)]
29#[serde(rename_all = "snake_case")]
30pub enum ConflictResolution {
31 #[default]
33 LocalWins,
34 RemoteWins,
36}
37
38impl ConflictResolution {
39 pub fn variants() -> &'static [ConflictResolution] {
41 &[
42 ConflictResolution::LocalWins,
43 ConflictResolution::RemoteWins,
44 ]
45 }
46
47 pub fn display_name(&self) -> &'static str {
49 match self {
50 ConflictResolution::LocalWins => "Local Wins",
51 ConflictResolution::RemoteWins => "Remote Wins",
52 }
53 }
54}
55
56#[derive(Debug, Clone, Serialize, Deserialize, PartialEq)]
58pub struct DynamicProfileSource {
59 pub url: String,
61
62 #[serde(default, skip_serializing_if = "HashMap::is_empty")]
64 pub headers: HashMap<String, String>,
65
66 #[serde(default = "default_refresh_interval_secs")]
68 pub refresh_interval_secs: u64,
69
70 #[serde(default = "default_max_size_bytes")]
72 pub max_size_bytes: usize,
73
74 #[serde(default = "default_fetch_timeout_secs")]
76 pub fetch_timeout_secs: u64,
77
78 #[serde(default = "default_true")]
80 pub enabled: bool,
81
82 #[serde(default)]
84 pub conflict_resolution: ConflictResolution,
85}
86
87impl Default for DynamicProfileSource {
88 fn default() -> Self {
89 Self {
90 url: String::new(),
91 headers: HashMap::new(),
92 refresh_interval_secs: default_refresh_interval_secs(),
93 max_size_bytes: default_max_size_bytes(),
94 fetch_timeout_secs: default_fetch_timeout_secs(),
95 enabled: true,
96 conflict_resolution: ConflictResolution::default(),
97 }
98 }
99}
100
101#[cfg(test)]
102mod tests {
103 use super::*;
104
105 #[test]
106 fn test_default_source() {
107 let source = DynamicProfileSource::default();
108
109 assert_eq!(source.url, "");
110 assert!(source.headers.is_empty());
111 assert_eq!(source.refresh_interval_secs, 1800);
112 assert_eq!(source.max_size_bytes, 1_048_576);
113 assert_eq!(source.fetch_timeout_secs, 10);
114 assert!(source.enabled);
115 assert_eq!(source.conflict_resolution, ConflictResolution::LocalWins);
116 }
117
118 #[test]
119 fn test_serialize_deserialize_roundtrip() {
120 let mut headers = HashMap::new();
121 headers.insert("Authorization".to_string(), "Bearer tok123".to_string());
122 headers.insert("X-Custom".to_string(), "value".to_string());
123
124 let source = DynamicProfileSource {
125 url: "https://example.com/profiles.yaml".to_string(),
126 headers,
127 refresh_interval_secs: 900,
128 max_size_bytes: 512_000,
129 fetch_timeout_secs: 15,
130 enabled: false,
131 conflict_resolution: ConflictResolution::RemoteWins,
132 };
133
134 let yaml = serde_yaml::to_string(&source).expect("serialize");
135 let deserialized: DynamicProfileSource = serde_yaml::from_str(&yaml).expect("deserialize");
136
137 assert_eq!(deserialized.url, source.url);
138 assert_eq!(deserialized.headers, source.headers);
139 assert_eq!(
140 deserialized.refresh_interval_secs,
141 source.refresh_interval_secs
142 );
143 assert_eq!(deserialized.max_size_bytes, source.max_size_bytes);
144 assert_eq!(deserialized.fetch_timeout_secs, source.fetch_timeout_secs);
145 assert_eq!(deserialized.enabled, source.enabled);
146 assert_eq!(deserialized.conflict_resolution, source.conflict_resolution);
147 }
148
149 #[test]
150 fn test_deserialize_minimal_yaml() {
151 let yaml = "url: https://example.com/profiles.yaml\n";
152 let source: DynamicProfileSource = serde_yaml::from_str(yaml).expect("deserialize minimal");
153
154 assert_eq!(source.url, "https://example.com/profiles.yaml");
155 assert!(source.headers.is_empty());
156 assert_eq!(source.refresh_interval_secs, 1800);
157 assert_eq!(source.max_size_bytes, 1_048_576);
158 assert_eq!(source.fetch_timeout_secs, 10);
159 assert!(source.enabled);
160 assert_eq!(source.conflict_resolution, ConflictResolution::LocalWins);
161 }
162
163 #[test]
164 fn test_conflict_resolution_display() {
165 assert_eq!(ConflictResolution::LocalWins.display_name(), "Local Wins");
166 assert_eq!(ConflictResolution::RemoteWins.display_name(), "Remote Wins");
167 }
168
169 #[test]
170 fn test_conflict_resolution_variants() {
171 let variants = ConflictResolution::variants();
172 assert_eq!(variants.len(), 2);
173 assert_eq!(variants[0], ConflictResolution::LocalWins);
174 assert_eq!(variants[1], ConflictResolution::RemoteWins);
175 }
176}