opendal/services/gcs/
delete.rs1use super::core::*;
19use super::error::parse_error;
20use crate::raw::oio::BatchDeleteResult;
21use crate::raw::*;
22use crate::*;
23use http::StatusCode;
24use std::sync::Arc;
25
26pub struct GcsDeleter {
27 core: Arc<GcsCore>,
28}
29
30impl GcsDeleter {
31 pub fn new(core: Arc<GcsCore>) -> Self {
32 Self { core }
33 }
34}
35
36impl oio::BatchDelete for GcsDeleter {
37 async fn delete_once(&self, path: String, _: OpDelete) -> Result<()> {
38 let resp = self.core.gcs_delete_object(&path).await?;
39
40 if resp.status().is_success() || resp.status() == StatusCode::NOT_FOUND {
42 Ok(())
43 } else {
44 Err(parse_error(resp))
45 }
46 }
47
48 async fn delete_batch(&self, batch: Vec<(String, OpDelete)>) -> Result<BatchDeleteResult> {
49 let paths: Vec<String> = batch.into_iter().map(|(p, _)| p).collect();
50 let resp = self.core.gcs_delete_objects(paths.clone()).await?;
51
52 let status = resp.status();
53
54 if status != StatusCode::OK {
57 return Err(parse_error(resp));
58 }
59
60 let boundary = parse_multipart_boundary(resp.headers())?.ok_or_else(|| {
61 Error::new(
62 ErrorKind::Unexpected,
63 "gcs batch delete response content type is empty",
64 )
65 })?;
66 let multipart: Multipart<MixedPart> = Multipart::new()
67 .with_boundary(boundary)
68 .parse(resp.into_body().to_bytes())?;
69 let parts = multipart.into_parts();
70
71 let mut batched_result = BatchDeleteResult::default();
72
73 for (i, part) in parts.into_iter().enumerate() {
74 let resp = part.into_response();
75 let path = paths[i].clone();
77
78 if resp.status().is_success() || resp.status() == StatusCode::NOT_FOUND {
80 batched_result.succeeded.push((path, OpDelete::default()));
81 } else {
82 batched_result
83 .failed
84 .push((path, OpDelete::default(), parse_error(resp)));
85 }
86 }
87
88 if batched_result.succeeded.is_empty() {
90 let err = batched_result.failed.remove(0).2;
91 return Err(err);
92 }
93
94 Ok(batched_result)
95 }
96}