opendal/services/webdav/
lister.rs

1// Licensed to the Apache Software Foundation (ASF) under one
2// or more contributor license agreements.  See the NOTICE file
3// distributed with this work for additional information
4// regarding copyright ownership.  The ASF licenses this file
5// to you under the Apache License, Version 2.0 (the
6// "License"); you may not use this file except in compliance
7// with the License.  You may obtain a copy of the License at
8//
9//   http://www.apache.org/licenses/LICENSE-2.0
10//
11// Unless required by applicable law or agreed to in writing,
12// software distributed under the License is distributed on an
13// "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
14// KIND, either express or implied.  See the License for the
15// specific language governing permissions and limitations
16// under the License.
17
18use std::sync::Arc;
19
20use http::StatusCode;
21
22use super::core::*;
23use super::error::*;
24use crate::raw::*;
25use crate::*;
26
27pub struct WebdavLister {
28    core: Arc<WebdavCore>,
29
30    path: String,
31    args: OpList,
32}
33
34impl WebdavLister {
35    pub fn new(core: Arc<WebdavCore>, path: &str, args: OpList) -> Self {
36        Self {
37            core,
38            path: path.to_string(),
39            args,
40        }
41    }
42}
43
44impl oio::PageList for WebdavLister {
45    async fn next_page(&self, ctx: &mut oio::PageContext) -> Result<()> {
46        let resp = self.core.webdav_list(&self.path, &self.args).await?;
47
48        // jfrog artifactory's webdav services have some strange behavior.
49        // We add this flag to check if the server is jfrog artifactory.
50        //
51        // Example: `"x-jfrog-version": "Artifactory/7.77.5 77705900"`
52        let is_jfrog_artifactory = if let Some(v) = resp.headers().get("x-jfrog-version") {
53            v.to_str().unwrap_or_default().starts_with("Artifactory")
54        } else {
55            false
56        };
57
58        let bs = if resp.status().is_success() {
59            resp.into_body()
60        } else if resp.status() == StatusCode::NOT_FOUND && self.path.ends_with('/') {
61            ctx.done = true;
62            return Ok(());
63        } else {
64            return Err(parse_error(resp));
65        };
66
67        let result: Multistatus = deserialize_multistatus(&bs.to_bytes())?;
68
69        for res in result.response {
70            let mut path = res
71                .href
72                .strip_prefix(&self.core.server_path)
73                .unwrap_or(&res.href)
74                .to_string();
75
76            let meta = parse_propstat(&res.propstat)?;
77
78            // Append `/` to path if it's a dir
79            if !path.ends_with('/') && meta.is_dir() {
80                path += "/"
81            }
82
83            let decoded_path = percent_decode_path(&path);
84            let normalized_path = if self.core.root != decoded_path {
85                build_rel_path(&self.core.root, &decoded_path)
86            } else {
87                "/".to_owned()
88            };
89
90            // HACKS! HACKS! HACKS!
91            //
92            // jfrog artifactory will generate a virtual checksum file for each file.
93            // The checksum file can't be stated, but can be listed and read.
94            // We ignore the checksum files to avoid listing unexpected files.
95            if is_jfrog_artifactory && meta.content_type() == Some("application/x-checksum") {
96                continue;
97            }
98
99            ctx.entries
100                .push_back(oio::Entry::new(&normalized_path, meta))
101        }
102        ctx.done = true;
103
104        Ok(())
105    }
106}