1use std::iter;
2use std::time::Duration;
3
4use md5::{Digest, Md5};
5use serde::Serialize;
6use time::OffsetDateTime;
7use url::Url;
8
9use crate::actions::Method;
10use crate::actions::OSSAction;
11use crate::signing::sign;
12use crate::sorting_iter::SortingIterator;
13use crate::{Bucket, Credentials, Map};
14
15#[derive(Debug, Clone)]
21pub struct DeleteObjects<'a, I> {
22 bucket: &'a Bucket,
23 credentials: Option<&'a Credentials>,
24 objects: I,
25 quiet: bool,
26
27 query: Map<'a>,
28 headers: Map<'a>,
29}
30
31impl<'a, I> DeleteObjects<'a, I> {
32 #[inline]
33 pub fn new(bucket: &'a Bucket, credentials: Option<&'a Credentials>, objects: I) -> Self {
34 Self {
35 bucket,
36 credentials,
37 objects,
38 quiet: false,
39 query: Map::new(),
40 headers: Map::new(),
41 }
42 }
43
44 pub fn quiet(&self) -> bool {
45 self.quiet
46 }
47
48 pub fn set_quiet(&mut self, quiet: bool) {
49 self.quiet = quiet;
50 }
51}
52
53#[derive(Debug, Clone, Default)]
54pub struct ObjectIdentifier {
55 pub key: String,
56 pub version_id: Option<String>,
57}
58
59impl ObjectIdentifier {
60 pub fn new(key: String) -> Self {
61 Self {
62 key,
63 ..Default::default()
64 }
65 }
66}
67
68impl<'a, I> DeleteObjects<'a, I>
69where
70 I: Iterator<Item = &'a ObjectIdentifier>,
71{
72 pub fn body_with_md5(self) -> (String, String) {
73 #[derive(Serialize)]
74 #[serde(rename = "Delete")]
75 struct DeleteSerde<'a> {
76 #[serde(rename = "Object")]
77 objects: Vec<Object<'a>>,
78 #[serde(rename = "Quiet")]
79 quiet: Option<bool>,
80 }
81 #[derive(Serialize)]
82 #[serde(rename = "Delete")]
83 struct Object<'a> {
84 #[serde(rename = "$value")]
85 nodes: Vec<Node<'a>>,
86 }
87
88 #[derive(Serialize)]
89 enum Node<'a> {
90 Key(&'a str),
91 VersionId(&'a str),
92 }
93
94 let objects: Vec<Object<'a>> = self
95 .objects
96 .map(|o| {
97 let mut nodes = vec![Node::Key(o.key.as_str())];
98 if let Some(ref version_id) = o.version_id {
99 nodes.push(Node::VersionId(version_id.as_str()));
100 }
101 Object { nodes }
102 })
103 .collect();
104
105 let req = DeleteSerde {
106 objects,
107 quiet: self.quiet.then(|| true),
108 };
109
110 let body = quick_xml::se::to_string(&req).unwrap();
111
112 let content_md5 = crate::base64::encode(Md5::digest(body.as_bytes()));
113 (body, content_md5)
114 }
115}
116
117impl<'a, I> OSSAction<'a> for DeleteObjects<'a, I>
118where
119 I: Iterator<Item = &'a ObjectIdentifier>,
120{
121 const METHOD: Method = Method::Post;
122
123 fn query_mut(&mut self) -> &mut Map<'a> {
124 &mut self.query
125 }
126
127 fn headers_mut(&mut self) -> &mut Map<'a> {
128 &mut self.headers
129 }
130
131 fn sign_with_time(&self, expires_in: Duration, time: &OffsetDateTime) -> Url {
132 let url = self.bucket.base_url().clone();
133 let query = SortingIterator::new(iter::once(("delete", "1")), self.query.iter());
134
135 match self.credentials {
136 Some(credentials) => sign(
137 time,
138 Method::Post,
139 url,
140 credentials.key(),
141 credentials.secret(),
142 credentials.token(),
143 self.bucket.region(),
144 expires_in.as_secs(),
145 query,
146 self.headers.iter(),
147 ),
148 None => crate::signing::util::add_query_params(url, query),
149 }
150 }
151}
152
153#[cfg(test)]
154mod tests {
155 use pretty_assertions::assert_eq;
156 use time::OffsetDateTime;
157
158 use crate::{Bucket, Credentials, UrlStyle};
159
160 use super::*;
161
162 #[test]
163 fn oss_example() {
164 let date = OffsetDateTime::from_unix_timestamp(1369353600).unwrap();
166 let expires_in = Duration::from_secs(86400);
167
168 let endpoint = "https://oss-cn-hangzhou.aliyuncs.com".parse().unwrap();
169 let bucket = Bucket::new(
170 endpoint,
171 UrlStyle::VirtualHost,
172 "examplebucket",
173 "cn-hangzhou",
174 )
175 .unwrap();
176 let credentials = Credentials::new(
177 "access_key_id",
178 "access_key_secret",
179 );
180
181 let objects = [
182 ObjectIdentifier {
183 key: "123".to_owned(),
184 ..Default::default()
185 },
186 ObjectIdentifier {
187 key: "456".to_owned(),
188 version_id: Some("ver1234".to_owned()),
189 },
190 ];
191 let action = DeleteObjects::new(&bucket, Some(&credentials), objects.iter());
192
193 let url = action.sign_with_time(expires_in, &date);
194 let expected = "https://examplebucket.oss-cn-hangzhou.aliyuncs.com/?delete=1&x-oss-additional-headers=host&x-oss-credential=access_key_id%2F20130524%2Fcn-hangzhou%2Foss%2Faliyun_v4_request&x-oss-date=20130524T000000Z&x-oss-expires=86400&x-oss-signature-version=OSS4-HMAC-SHA256&x-oss-signature=b2d1a8a3818834f42fd4b828f5a461d74dc396ffef3540f834ed872fba693cfe";
195
196 assert_eq!(expected, url.as_str());
197 }
198
199 #[test]
200 fn anonymous_custom_query() {
201 let expires_in = Duration::from_secs(86400);
202
203 let endpoint = "https://oss-cn-hangzhou.aliyuncs.com".parse().unwrap();
204 let bucket = Bucket::new(
205 endpoint,
206 UrlStyle::VirtualHost,
207 "examplebucket",
208 "cn-hangzhou",
209 )
210 .unwrap();
211
212 let objects = [
213 ObjectIdentifier {
214 key: "123".to_owned(),
215 ..Default::default()
216 },
217 ObjectIdentifier {
218 key: "456".to_owned(),
219 version_id: Some("ver1234".to_owned()),
220 },
221 ];
222 let action = DeleteObjects::new(&bucket, None, objects.iter());
223 let url = action.sign(expires_in);
224 let expected = "https://examplebucket.oss-cn-hangzhou.aliyuncs.com/?delete=1";
225
226 assert_eq!(expected, url.as_str());
227 }
228}