mirror of
https://github.com/minio/minio-rs.git
synced 2026-03-10 05:32:51 +08:00
- Replaced raw string parameters with validated wrapper types (BucketName, ObjectName, Region, VersionId, etc.) following the "parse, don't validate" pattern - Bucket and object names are now validated at construction time, ensuring compile-time correctness - Added both relaxed (MinIO-compatible) and strict (AWS S3-compliant) validation modes for bucket names
125 lines
4.4 KiB
Rust
125 lines
4.4 KiB
Rust
// MinIO Rust Library for Amazon S3 Compatible Cloud Storage
|
|
// Copyright 2025 MinIO, Inc.
|
|
//
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
// you may not use this file except in compliance with the License.
|
|
// You may obtain a copy of the License at
|
|
//
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
//
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
// See the License for the specific language governing permissions and
|
|
// limitations under the License.
|
|
|
|
use async_std::io::ReadExt;
|
|
use minio::s3::builders::ObjectContent;
|
|
use minio::s3::response::{GetObjectResponse, PutObjectContentResponse};
|
|
use minio::s3::response_traits::{HasBucket, HasObject};
|
|
use minio::s3::types::{BucketName, ObjectKey, S3Api};
|
|
use minio::s3::utils::hex_encode;
|
|
use minio_common::rand_reader::RandReader;
|
|
use minio_common::test_context::TestContext;
|
|
use minio_common::utils::rand_object_name_utf8;
|
|
#[cfg(feature = "ring")]
|
|
use ring::digest::{Context, SHA256};
|
|
#[cfg(not(feature = "ring"))]
|
|
use sha2::{Digest, Sha256};
|
|
use std::path::PathBuf;
|
|
|
|
async fn get_hash(filename: &str) -> String {
|
|
#[cfg(feature = "ring")]
|
|
{
|
|
let mut context = Context::new(&SHA256);
|
|
let mut file = async_std::fs::File::open(filename).await.unwrap();
|
|
let mut buf = Vec::new();
|
|
file.read_to_end(&mut buf).await.unwrap();
|
|
context.update(&buf);
|
|
hex_encode(context.finish().as_ref())
|
|
}
|
|
#[cfg(not(feature = "ring"))]
|
|
{
|
|
let mut hasher = Sha256::new();
|
|
let mut file = async_std::fs::File::open(filename).await.unwrap();
|
|
let mut buf = Vec::new();
|
|
file.read_to_end(&mut buf).await.unwrap();
|
|
hasher.update(&buf);
|
|
hex_encode(hasher.finalize().as_ref())
|
|
}
|
|
}
|
|
|
|
async fn test_upload_download_object(ctx: &TestContext, bucket: &str, object: &str, size: u64) {
|
|
let mut file = async_std::fs::File::create(&object).await.unwrap();
|
|
|
|
async_std::io::copy(&mut RandReader::new(size), &mut file)
|
|
.await
|
|
.unwrap();
|
|
|
|
file.sync_all().await.unwrap();
|
|
|
|
let obj: ObjectContent = PathBuf::from(object).as_path().into();
|
|
|
|
let resp: PutObjectContentResponse = ctx
|
|
.client
|
|
.put_object_content(
|
|
BucketName::try_from(bucket).unwrap(),
|
|
ObjectKey::try_from(object).unwrap(),
|
|
obj,
|
|
)
|
|
.unwrap()
|
|
.build()
|
|
.send()
|
|
.await
|
|
.unwrap();
|
|
assert_eq!(resp.bucket(), Some(&BucketName::try_from(bucket).unwrap()));
|
|
assert_eq!(resp.object(), Some(&ObjectKey::try_from(object).unwrap()));
|
|
assert_eq!(resp.object_size(), size);
|
|
|
|
let filename: String = rand_object_name_utf8(20).to_string();
|
|
let resp: GetObjectResponse = ctx
|
|
.client
|
|
.get_object(
|
|
BucketName::try_from(bucket).unwrap(),
|
|
ObjectKey::try_from(object).unwrap(),
|
|
)
|
|
.unwrap()
|
|
.build()
|
|
.send()
|
|
.await
|
|
.unwrap();
|
|
assert_eq!(resp.bucket(), Some(&BucketName::try_from(bucket).unwrap()));
|
|
assert_eq!(resp.object(), Some(&ObjectKey::try_from(object).unwrap()));
|
|
|
|
// save the object to a file
|
|
resp.content()
|
|
.unwrap()
|
|
.to_file(PathBuf::from(&filename).as_path())
|
|
.await
|
|
.unwrap();
|
|
assert_eq!(get_hash(object).await, get_hash(&filename).await);
|
|
|
|
async_std::fs::remove_file(&object).await.unwrap();
|
|
async_std::fs::remove_file(&filename).await.unwrap();
|
|
}
|
|
|
|
/// Test uploading and downloading an object with a size that fits in a single part
|
|
#[minio_macros::test]
|
|
async fn upload_download_object_1(ctx: TestContext, bucket: BucketName) {
|
|
let object = rand_object_name_utf8(20);
|
|
test_upload_download_object(&ctx, bucket.as_str(), object.as_str(), 16).await;
|
|
}
|
|
|
|
/// Test uploading and downloading an object with a name that contains white space characters.
|
|
#[minio_macros::test]
|
|
async fn upload_download_object_2(ctx: TestContext, bucket: BucketName) {
|
|
test_upload_download_object(&ctx, bucket.as_str(), "a b+c", 16).await;
|
|
}
|
|
|
|
/// Test uploading and downloading an object with a size that needs multiple parts.
|
|
#[minio_macros::test]
|
|
async fn upload_download_object_3(ctx: TestContext, bucket: BucketName) {
|
|
let object = rand_object_name_utf8(20);
|
|
test_upload_download_object(&ctx, bucket.as_str(), object.as_str(), 16 + 5 * 1024 * 1024).await;
|
|
}
|