opendal/raw/oio/list/
hierarchy_list.rs1use std::collections::HashSet;
19
20use crate::raw::*;
21use crate::*;
22
23pub struct HierarchyLister<P> {
34 lister: P,
35 path: String,
36 visited: HashSet<String>,
37 recursive: bool,
38}
39
40impl<P> HierarchyLister<P> {
41 pub fn new(lister: P, path: &str, recursive: bool) -> HierarchyLister<P> {
43 let path = if path == "/" {
44 "".to_string()
45 } else {
46 path.to_string()
47 };
48
49 HierarchyLister {
50 lister,
51 path,
52 visited: HashSet::default(),
53 recursive,
54 }
55 }
56
57 fn keep_entry(&mut self, e: &mut oio::Entry) -> bool {
64 if !e.path().starts_with(&self.path) {
69 return false;
70 }
71
72 if self.visited.contains(e.path()) {
74 return false;
75 }
76
77 let prefix_len = self.path.len();
78
79 let idx = if let Some(idx) = e.path()[prefix_len..].find('/') {
80 idx + prefix_len + 1
81 } else {
82 return true;
85 };
86
87 if idx == e.path().len() {
90 if !self.visited.contains(e.path()) {
91 self.visited.insert(e.path().to_string());
92 }
93 return true;
94 }
95
96 let has = {
99 let path = &e.path()[..idx];
100 self.visited.contains(path)
101 };
102 if !has {
103 let path = {
104 let path = &e.path()[..idx];
105 path.to_string()
106 };
107
108 e.set_path(&path);
109 e.set_mode(EntryMode::DIR);
110 self.visited.insert(path);
111
112 return true;
113 }
114
115 false
116 }
117}
118
119impl<P: oio::List> oio::List for HierarchyLister<P> {
120 async fn next(&mut self) -> Result<Option<oio::Entry>> {
121 loop {
122 let mut entry = match self.lister.next().await? {
123 Some(entry) => entry,
124 None => return Ok(None),
125 };
126
127 if self.recursive {
128 return Ok(Some(entry));
129 }
130 if self.keep_entry(&mut entry) {
131 return Ok(Some(entry));
132 }
133 }
134 }
135}
136
137impl<P: oio::BlockingList> oio::BlockingList for HierarchyLister<P> {
138 fn next(&mut self) -> Result<Option<oio::Entry>> {
139 loop {
140 let mut entry = match self.lister.next()? {
141 Some(entry) => entry,
142 None => return Ok(None),
143 };
144
145 if self.recursive {
146 return Ok(Some(entry));
147 }
148
149 if self.keep_entry(&mut entry) {
150 return Ok(Some(entry));
151 }
152 }
153 }
154}
155
156#[cfg(test)]
157mod tests {
158
159 use std::collections::HashSet;
160 use std::vec::IntoIter;
161
162 use log::debug;
163 use oio::BlockingList;
164
165 use super::*;
166
167 struct MockLister {
168 inner: IntoIter<&'static str>,
169 }
170
171 impl MockLister {
172 fn new(inner: Vec<&'static str>) -> Self {
173 Self {
174 inner: inner.into_iter(),
175 }
176 }
177 }
178
179 impl BlockingList for MockLister {
180 fn next(&mut self) -> Result<Option<oio::Entry>> {
181 let entry = self.inner.next().map(|path| {
182 if path.ends_with('/') {
183 oio::Entry::new(path, Metadata::new(EntryMode::DIR))
184 } else {
185 oio::Entry::new(path, Metadata::new(EntryMode::FILE))
186 }
187 });
188
189 Ok(entry)
190 }
191 }
192
193 #[test]
194 fn test_blocking_list() -> Result<()> {
195 let _ = tracing_subscriber::fmt().with_test_writer().try_init();
196
197 let lister = MockLister::new(vec![
198 "x/x/", "x/y/", "y/", "x/x/x", "y/y", "xy/", "z", "y/a",
199 ]);
200 let mut lister = HierarchyLister::new(lister, "", false);
201
202 let mut entries = Vec::default();
203
204 let mut set = HashSet::new();
205 while let Some(e) = lister.next()? {
206 debug!("got path {}", e.path());
207 assert!(
208 set.insert(e.path().to_string()),
209 "duplicated value: {}",
210 e.path()
211 );
212
213 entries.push(e)
214 }
215
216 assert_eq!(
217 entries[0],
218 oio::Entry::new("x/", Metadata::new(EntryMode::DIR))
219 );
220 assert_eq!(
221 entries[1],
222 oio::Entry::new("y/", Metadata::new(EntryMode::DIR))
223 );
224 assert_eq!(
225 entries[2],
226 oio::Entry::new("xy/", Metadata::new(EntryMode::DIR))
227 );
228 assert_eq!(
229 entries[3],
230 oio::Entry::new("z", Metadata::new(EntryMode::FILE))
231 );
232
233 Ok(())
234 }
235}