|
17 | 17 |
|
18 | 18 | use std::fmt;
|
19 | 19 | use std::fmt::Debug;
|
| 20 | +use std::fmt::Display; |
20 | 21 | use std::fmt::Formatter;
|
21 | 22 | use std::fmt::Write;
|
22 | 23 | use std::sync::atomic;
|
23 | 24 | use std::sync::atomic::AtomicBool;
|
24 | 25 | use std::time::Duration;
|
25 | 26 |
|
| 27 | +use base64::prelude::BASE64_STANDARD; |
| 28 | +use base64::Engine; |
26 | 29 | use bytes::Bytes;
|
27 | 30 | use http::header::HeaderName;
|
28 | 31 | use http::header::CACHE_CONTROL;
|
@@ -88,6 +91,7 @@ pub struct S3Core {
|
88 | 91 | pub credential_loaded: AtomicBool,
|
89 | 92 | pub client: HttpClient,
|
90 | 93 | pub batch_max_operations: usize,
|
| 94 | + pub checksum_algorithm: Option<ChecksumAlgorithm>, |
91 | 95 | }
|
92 | 96 |
|
93 | 97 | impl Debug for S3Core {
|
@@ -246,6 +250,35 @@ impl S3Core {
|
246 | 250 |
|
247 | 251 | req
|
248 | 252 | }
|
| 253 | + |
| 254 | + pub fn insert_checksum_header( |
| 255 | + &self, |
| 256 | + mut req: http::request::Builder, |
| 257 | + body: &Buffer, |
| 258 | + ) -> http::request::Builder { |
| 259 | + if let Some(checksum_algorithm) = self.checksum_algorithm.as_ref() { |
| 260 | + let checksum = match checksum_algorithm { |
| 261 | + ChecksumAlgorithm::Crc32c => { |
| 262 | + let mut crc = 0u32; |
| 263 | + body.clone() |
| 264 | + .for_each(|b| crc = crc32c::crc32c_append(crc, &b)); |
| 265 | + BASE64_STANDARD.encode(crc.to_be_bytes()) |
| 266 | + } |
| 267 | + }; |
| 268 | + req = req.header(checksum_algorithm.to_header_name(), checksum); |
| 269 | + } |
| 270 | + req |
| 271 | + } |
| 272 | + |
| 273 | + pub fn insert_checksum_type_header( |
| 274 | + &self, |
| 275 | + mut req: http::request::Builder, |
| 276 | + ) -> http::request::Builder { |
| 277 | + if let Some(checksum_algorithm) = self.checksum_algorithm.as_ref() { |
| 278 | + req = req.header("x-amz-checksum-algorithm", checksum_algorithm.to_string()); |
| 279 | + } |
| 280 | + req |
| 281 | + } |
249 | 282 | }
|
250 | 283 |
|
251 | 284 | impl S3Core {
|
@@ -408,6 +441,9 @@ impl S3Core {
|
408 | 441 | // Set SSE headers.
|
409 | 442 | req = self.insert_sse_headers(req, true);
|
410 | 443 |
|
| 444 | + // Set Checksum header. |
| 445 | + req = self.insert_checksum_header(req, &body); |
| 446 | + |
411 | 447 | // Set body
|
412 | 448 | let req = req.body(body).map_err(new_request_build_error)?;
|
413 | 449 |
|
@@ -573,6 +609,9 @@ impl S3Core {
|
573 | 609 | // Set SSE headers.
|
574 | 610 | let req = self.insert_sse_headers(req, true);
|
575 | 611 |
|
| 612 | + // Set SSE headers. |
| 613 | + let req = self.insert_checksum_type_header(req); |
| 614 | + |
576 | 615 | let mut req = req.body(Buffer::new()).map_err(new_request_build_error)?;
|
577 | 616 |
|
578 | 617 | self.sign(&mut req).await?;
|
@@ -605,6 +644,9 @@ impl S3Core {
|
605 | 644 | // Set SSE headers.
|
606 | 645 | req = self.insert_sse_headers(req, true);
|
607 | 646 |
|
| 647 | + // Set Checksum header. |
| 648 | + req = self.insert_checksum_header(req, &body); |
| 649 | + |
608 | 650 | // Set body
|
609 | 651 | let req = req.body(body).map_err(new_request_build_error)?;
|
610 | 652 |
|
@@ -821,6 +863,28 @@ pub struct OutputCommonPrefix {
|
821 | 863 | pub prefix: String,
|
822 | 864 | }
|
823 | 865 |
|
| 866 | +pub enum ChecksumAlgorithm { |
| 867 | + Crc32c, |
| 868 | +} |
| 869 | +impl ChecksumAlgorithm { |
| 870 | + pub fn to_header_name(&self) -> HeaderName { |
| 871 | + match self { |
| 872 | + Self::Crc32c => HeaderName::from_static("x-amz-checksum-crc32c"), |
| 873 | + } |
| 874 | + } |
| 875 | +} |
| 876 | +impl Display for ChecksumAlgorithm { |
| 877 | + fn fmt(&self, f: &mut Formatter<'_>) -> fmt::Result { |
| 878 | + write!( |
| 879 | + f, |
| 880 | + "{}", |
| 881 | + match self { |
| 882 | + Self::Crc32c => "CRC32C", |
| 883 | + } |
| 884 | + ) |
| 885 | + } |
| 886 | +} |
| 887 | + |
824 | 888 | #[cfg(test)]
|
825 | 889 | mod tests {
|
826 | 890 | use bytes::Buf;
|
|
0 commit comments