gix_config/file/section/
mod.rs1use std::{borrow::Cow, ops::Deref};
2
3use bstr::{BStr, BString, ByteSlice};
4use smallvec::SmallVec;
5
6use crate::{
7 file,
8 file::{Metadata, Section, SectionMut},
9 parse,
10 parse::{section, Event},
11};
12
13pub(crate) mod body;
14pub use body::{Body, BodyIter};
15use gix_features::threading::OwnShared;
16
17use crate::file::{
18 write::{extract_newline, platform_newline},
19 SectionId,
20};
21
22impl<'a> Deref for Section<'a> {
23 type Target = Body<'a>;
24
25 fn deref(&self) -> &Self::Target {
26 &self.body
27 }
28}
29
30impl<'a> Section<'a> {
32 pub fn new(
34 name: impl Into<Cow<'a, str>>,
35 subsection: impl Into<Option<Cow<'a, BStr>>>,
36 meta: impl Into<OwnShared<file::Metadata>>,
37 ) -> Result<Self, parse::section::header::Error> {
38 Ok(Section {
39 header: parse::section::Header::new(name, subsection)?,
40 body: Default::default(),
41 meta: meta.into(),
42 id: SectionId::default(),
43 })
44 }
45}
46
47impl<'a> Section<'a> {
49 pub fn header(&self) -> §ion::Header<'a> {
51 &self.header
52 }
53
54 pub fn id(&self) -> SectionId {
57 self.id
58 }
59
60 pub fn body(&self) -> &Body<'a> {
62 &self.body
63 }
64
65 #[must_use]
69 pub fn to_bstring(&self) -> BString {
70 let mut buf = Vec::new();
71 self.write_to(&mut buf).expect("io error impossible");
72 buf.into()
73 }
74
75 pub fn write_to(&self, mut out: &mut dyn std::io::Write) -> std::io::Result<()> {
78 self.header.write_to(&mut *out)?;
79
80 if self.body.0.is_empty() {
81 return Ok(());
82 }
83
84 let nl = self
85 .body
86 .as_ref()
87 .iter()
88 .find_map(extract_newline)
89 .unwrap_or_else(|| platform_newline());
90
91 if !self
92 .body
93 .as_ref()
94 .iter()
95 .take_while(|e| !matches!(e, Event::SectionValueName(_)))
96 .any(|e| e.to_bstr_lossy().contains_str(nl))
97 {
98 out.write_all(nl)?;
99 }
100
101 let mut saw_newline_after_value = true;
102 let mut in_key_value_pair = false;
103 for (idx, event) in self.body.as_ref().iter().enumerate() {
104 match event {
105 Event::SectionValueName(_) => {
106 if !saw_newline_after_value {
107 out.write_all(nl)?;
108 }
109 saw_newline_after_value = false;
110 in_key_value_pair = true;
111 }
112 Event::Newline(_) if !in_key_value_pair => {
113 saw_newline_after_value = true;
114 }
115 Event::Value(_) | Event::ValueDone(_) => {
116 in_key_value_pair = false;
117 }
118 _ => {}
119 }
120 event.write_to(&mut out)?;
121 if let Event::ValueNotDone(_) = event {
122 if self
123 .body
124 .0
125 .get(idx + 1)
126 .filter(|e| matches!(e, Event::Newline(_)))
127 .is_none()
128 {
129 out.write_all(nl)?;
130 }
131 }
132 }
133 Ok(())
134 }
135
136 pub fn meta(&self) -> &Metadata {
138 &self.meta
139 }
140
141 pub fn to_mut(&mut self, newline: SmallVec<[u8; 2]>) -> SectionMut<'_, 'a> {
143 SectionMut::new(self, newline)
144 }
145}