dprint_plugin_markdown/configuration/
builder.rs1use dprint_core::configuration::ConfigKeyMap;
2use dprint_core::configuration::ConfigKeyValue;
3use dprint_core::configuration::GlobalConfiguration;
4use dprint_core::configuration::NewLineKind;
5
6use super::*;
7
8#[derive(Default)]
20pub struct ConfigurationBuilder {
21 pub(super) config: ConfigKeyMap,
22 global_config: Option<GlobalConfiguration>,
23}
24
25impl ConfigurationBuilder {
26 pub fn new() -> Self {
28 Self::default()
29 }
30
31 pub fn build(&self) -> Configuration {
33 if let Some(global_config) = &self.global_config {
34 resolve_config(self.config.clone(), global_config).config
35 } else {
36 resolve_config(self.config.clone(), &Default::default()).config
37 }
38 }
39
40 pub fn global_config(&mut self, global_config: GlobalConfiguration) -> &mut Self {
42 self.global_config = Some(global_config);
43 self
44 }
45
46 pub fn line_width(&mut self, value: u32) -> &mut Self {
49 self.insert("lineWidth", (value as i32).into())
50 }
51
52 pub fn new_line_kind(&mut self, value: NewLineKind) -> &mut Self {
55 self.insert("newLineKind", value.to_string().into())
56 }
57
58 pub fn text_wrap(&mut self, value: TextWrap) -> &mut Self {
61 self.insert("textWrap", value.to_string().into())
62 }
63
64 pub fn emphasis_kind(&mut self, value: EmphasisKind) -> &mut Self {
67 self.insert("emphasisKind", value.to_string().into())
68 }
69
70 pub fn strong_kind(&mut self, value: StrongKind) -> &mut Self {
73 self.insert("strongKind", value.to_string().into())
74 }
75
76 pub fn unordered_list_kind(&mut self, value: UnorderedListKind) -> &mut Self {
79 self.insert("unorderedListKind", value.to_string().into())
80 }
81
82 pub fn heading_kind(&mut self, value: HeadingKind) -> &mut Self {
85 self.insert("headingKind", value.to_string().into())
86 }
87
88 pub fn list_indent_kind(&mut self, value: ListIndentKind) -> &mut Self {
91 self.insert("listIndentKind", value.to_string().into())
92 }
93
94 pub fn ignore_directive(&mut self, value: &str) -> &mut Self {
97 self.insert("ignoreDirective", value.to_string().into())
98 }
99
100 pub fn ignore_file_directive(&mut self, value: &str) -> &mut Self {
103 self.insert("ignoreFileDirective", value.to_string().into())
104 }
105
106 pub fn ignore_start_directive(&mut self, value: &str) -> &mut Self {
109 self.insert("ignoreStartDirective", value.to_string().into())
110 }
111
112 pub fn ignore_end_directive(&mut self, value: &str) -> &mut Self {
115 self.insert("ignoreEndDirective", value.to_string().into())
116 }
117
118 pub fn deno(&mut self) -> &mut Self {
119 self
120 .text_wrap(TextWrap::Always)
121 .ignore_directive("deno-fmt-ignore")
122 .ignore_start_directive("deno-fmt-ignore-start")
123 .ignore_end_directive("deno-fmt-ignore-end")
124 .ignore_file_directive("deno-fmt-ignore-file")
125 }
126
127 #[cfg(test)]
128 pub(super) fn get_inner_config(&self) -> ConfigKeyMap {
129 self.config.clone()
130 }
131
132 fn insert(&mut self, name: &str, value: ConfigKeyValue) -> &mut Self {
133 self.config.insert(String::from(name), value);
134 self
135 }
136}
137
138#[cfg(test)]
139mod tests {
140 use dprint_core::configuration::resolve_global_config;
141 use dprint_core::configuration::NewLineKind;
142
143 use super::*;
144
145 #[test]
146 fn check_all_values_set() {
147 let mut config = ConfigurationBuilder::new();
148 config
149 .new_line_kind(NewLineKind::CarriageReturnLineFeed)
150 .line_width(90)
151 .text_wrap(TextWrap::Always)
152 .emphasis_kind(EmphasisKind::Asterisks)
153 .strong_kind(StrongKind::Underscores)
154 .unordered_list_kind(UnorderedListKind::Asterisks)
155 .heading_kind(HeadingKind::Atx)
156 .list_indent_kind(ListIndentKind::PythonMarkdown)
157 .ignore_directive("test")
158 .ignore_file_directive("test")
159 .ignore_start_directive("test")
160 .ignore_end_directive("test");
161
162 let inner_config = config.get_inner_config();
163 assert_eq!(inner_config.len(), 12);
164 let diagnostics = resolve_config(inner_config, &Default::default()).diagnostics;
165 assert_eq!(diagnostics.len(), 0);
166 }
167
168 #[test]
169 fn handle_global_config() {
170 let mut global_config = ConfigKeyMap::new();
171 global_config.insert(String::from("lineWidth"), 90.into());
172 global_config.insert(String::from("newLineKind"), "crlf".into());
173 global_config.insert(String::from("useTabs"), true.into());
174 let global_config = resolve_global_config(&mut global_config).config;
175 let mut config_builder = ConfigurationBuilder::new();
176 let config = config_builder.global_config(global_config).build();
177 assert_eq!(config.line_width, 90);
178 assert_eq!(config.new_line_kind == NewLineKind::CarriageReturnLineFeed, true);
179 }
180
181 #[test]
182 fn use_markdown_defaults_when_global_not_set() {
183 let global_config = GlobalConfiguration::default();
184 let mut config_builder = ConfigurationBuilder::new();
185 let config = config_builder.global_config(global_config).build();
186 assert_eq!(config.line_width, 80); assert_eq!(config.new_line_kind == NewLineKind::LineFeed, true);
188 }
189
190 #[test]
191 fn tags_valid_object() {
192 let mut config = ConfigKeyMap::new();
193 let mut tags_obj = ConfigKeyMap::new();
194 tags_obj.insert("markdown".into(), "md".into());
195 tags_obj.insert("JSX".into(), "tsx".into());
196 config.insert("tags".into(), ConfigKeyValue::Object(tags_obj));
197
198 let result = resolve_config(config, &Default::default());
199 assert_eq!(result.diagnostics.len(), 0);
200 assert_eq!(result.config.tags.get("markdown").unwrap(), "md");
201 assert_eq!(result.config.tags.get("jsx").unwrap(), "tsx");
203 assert!(result.config.tags.get("JSX").is_none());
204 }
205
206 #[test]
207 fn tags_extension_with_period() {
208 let mut config = ConfigKeyMap::new();
209 let mut tags_obj = ConfigKeyMap::new();
210 tags_obj.insert("markdown".into(), ".md".into());
211 config.insert("tags".into(), ConfigKeyValue::Object(tags_obj));
212
213 let result = resolve_config(config, &Default::default());
214 assert_eq!(result.diagnostics.len(), 1);
215 assert_eq!(result.diagnostics[0].property_name, "tags.markdown");
216 assert!(result.diagnostics[0].message.contains("without a period"));
217 }
218
219 #[test]
220 fn tags_non_string_value() {
221 let mut config = ConfigKeyMap::new();
222 let mut tags_obj = ConfigKeyMap::new();
223 tags_obj.insert("markdown".into(), true.into());
224 config.insert("tags".into(), ConfigKeyValue::Object(tags_obj));
225
226 let result = resolve_config(config, &Default::default());
227 assert_eq!(result.diagnostics.len(), 1);
228 assert_eq!(result.diagnostics[0].property_name, "tags.markdown");
229 assert!(result.diagnostics[0].message.contains("Expected string value"));
230 }
231
232 #[test]
233 fn tags_not_an_object() {
234 let mut config = ConfigKeyMap::new();
235 config.insert("tags".into(), "not_an_object".into());
236
237 let result = resolve_config(config, &Default::default());
238 assert_eq!(result.diagnostics.len(), 1);
239 assert_eq!(result.diagnostics[0].property_name, "tags");
240 assert!(result.diagnostics[0].message.contains("Expected an object"));
241 }
242}