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
199 lines
6.8 KiB
Rust
199 lines
6.8 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 minio::s3::builders::VersioningStatus;
|
|
use minio::s3::client::DEFAULT_REGION;
|
|
use minio::s3::error::{Error, S3ServerError};
|
|
use minio::s3::minio_error_response::MinioErrorCode;
|
|
use minio::s3::response::{
|
|
DeleteBucketReplicationResponse, GetBucketReplicationResponse, GetBucketVersioningResponse,
|
|
PutBucketPolicyResponse, PutBucketReplicationResponse, PutBucketVersioningResponse,
|
|
};
|
|
use minio::s3::response_traits::{HasBucket, HasRegion};
|
|
use minio::s3::types::{BucketName, ReplicationConfig, S3Api};
|
|
use minio_common::example::{
|
|
create_bucket_policy_config_example_for_replication, create_bucket_replication_config_example,
|
|
};
|
|
use minio_common::test_context::TestContext;
|
|
|
|
#[minio_macros::test(skip_if_express)]
|
|
async fn bucket_replication_s3(ctx: TestContext, bucket: BucketName) {
|
|
let ctx2 = TestContext::new_from_env();
|
|
let (bucket2, cleanup2) = ctx2.create_bucket_helper().await;
|
|
|
|
{
|
|
let resp: PutBucketVersioningResponse = ctx
|
|
.client
|
|
.put_bucket_versioning(&bucket, VersioningStatus::Enabled)
|
|
.unwrap()
|
|
.build()
|
|
.send()
|
|
.await
|
|
.unwrap();
|
|
assert_eq!(resp.bucket(), Some(&bucket));
|
|
assert_eq!(resp.region(), &*DEFAULT_REGION);
|
|
|
|
let resp: PutBucketVersioningResponse = ctx
|
|
.client
|
|
.put_bucket_versioning(bucket2.as_str(), VersioningStatus::Enabled)
|
|
.unwrap()
|
|
.build()
|
|
.send()
|
|
.await
|
|
.unwrap();
|
|
assert_eq!(
|
|
resp.bucket(),
|
|
Some(&BucketName::try_from(bucket2.as_str()).unwrap())
|
|
);
|
|
assert_eq!(resp.region(), &*DEFAULT_REGION);
|
|
|
|
let resp: GetBucketVersioningResponse = ctx
|
|
.client
|
|
.get_bucket_versioning(&bucket)
|
|
.unwrap()
|
|
.build()
|
|
.send()
|
|
.await
|
|
.unwrap();
|
|
assert_eq!(resp.status().unwrap(), Some(VersioningStatus::Enabled));
|
|
assert_eq!(resp.bucket(), Some(&bucket));
|
|
assert_eq!(resp.region(), &*DEFAULT_REGION);
|
|
|
|
if false {
|
|
//TODO: to allow replication policy needs to be applied, but this fails
|
|
let config: String = create_bucket_policy_config_example_for_replication();
|
|
let _resp: PutBucketPolicyResponse = ctx
|
|
.client
|
|
.put_bucket_policy(&bucket)
|
|
.unwrap()
|
|
.config(config.clone())
|
|
.build()
|
|
.send()
|
|
.await
|
|
.unwrap();
|
|
|
|
let _resp: PutBucketPolicyResponse = ctx
|
|
.client
|
|
.put_bucket_policy(bucket2.as_str())
|
|
.unwrap()
|
|
.config(config.clone())
|
|
.build()
|
|
.send()
|
|
.await
|
|
.unwrap();
|
|
}
|
|
}
|
|
|
|
if false {
|
|
let config: ReplicationConfig = create_bucket_replication_config_example(&bucket2);
|
|
|
|
//TODO setup permissions that allow replication
|
|
// TODO panic: called `Result::unwrap()` on an `Err` value: S3Error(ErrorResponse { code: "XMinioAdminRemoteTargetNotFoundError", message: "The remote target does not exist",
|
|
let resp: PutBucketReplicationResponse = ctx
|
|
.client
|
|
.put_bucket_replication(&bucket)
|
|
.unwrap()
|
|
.replication_config(config.clone())
|
|
.build()
|
|
.send()
|
|
.await
|
|
.unwrap();
|
|
//println!("response of setting replication: resp={:?}", resp);
|
|
assert_eq!(resp.bucket(), Some(&bucket));
|
|
assert_eq!(resp.region(), &*DEFAULT_REGION);
|
|
|
|
let resp: GetBucketReplicationResponse = ctx
|
|
.client
|
|
.get_bucket_replication(&bucket)
|
|
.unwrap()
|
|
.build()
|
|
.send()
|
|
.await
|
|
.unwrap();
|
|
//assert_eq!(resp.config, config); //TODO
|
|
assert_eq!(resp.bucket(), Some(&bucket));
|
|
assert_eq!(resp.region(), &*DEFAULT_REGION);
|
|
|
|
// TODO called `Result::unwrap()` on an `Err` value: S3Error(ErrorResponse { code: "XMinioAdminRemoteTargetNotFoundError", message: "The remote target does not exist",
|
|
let resp: DeleteBucketReplicationResponse = ctx
|
|
.client
|
|
.delete_bucket_replication(&bucket)
|
|
.unwrap()
|
|
.build()
|
|
.send()
|
|
.await
|
|
.unwrap();
|
|
println!("response of deleting replication: resp={resp:?}");
|
|
}
|
|
let _resp: GetBucketVersioningResponse = ctx
|
|
.client
|
|
.get_bucket_versioning(&bucket)
|
|
.unwrap()
|
|
.build()
|
|
.send()
|
|
.await
|
|
.unwrap();
|
|
cleanup2.cleanup().await;
|
|
//println!("response of getting replication: resp={:?}", resp);
|
|
}
|
|
|
|
#[minio_macros::test(skip_if_not_express)]
|
|
async fn bucket_replication_s3express(ctx: TestContext, bucket: BucketName) {
|
|
let config: ReplicationConfig = create_bucket_replication_config_example(&bucket);
|
|
|
|
let resp: Result<PutBucketReplicationResponse, Error> = ctx
|
|
.client
|
|
.put_bucket_replication(&bucket)
|
|
.unwrap()
|
|
.replication_config(config.clone())
|
|
.build()
|
|
.send()
|
|
.await;
|
|
match resp {
|
|
Err(Error::S3Server(S3ServerError::S3Error(e))) => {
|
|
assert_eq!(e.code(), MinioErrorCode::NotSupported)
|
|
}
|
|
v => panic!("Expected error S3Error(NotSupported): but got {v:?}"),
|
|
}
|
|
|
|
let resp: Result<GetBucketReplicationResponse, Error> = ctx
|
|
.client
|
|
.get_bucket_replication(&bucket)
|
|
.unwrap()
|
|
.build()
|
|
.send()
|
|
.await;
|
|
match resp {
|
|
Err(Error::S3Server(S3ServerError::S3Error(e))) => {
|
|
assert_eq!(e.code(), MinioErrorCode::NotSupported)
|
|
}
|
|
v => panic!("Expected error S3Error(NotSupported): but got {v:?}"),
|
|
}
|
|
|
|
let resp: Result<DeleteBucketReplicationResponse, Error> = ctx
|
|
.client
|
|
.delete_bucket_replication(&bucket)
|
|
.unwrap()
|
|
.build()
|
|
.send()
|
|
.await;
|
|
match resp {
|
|
Err(Error::S3Server(S3ServerError::S3Error(e))) => {
|
|
assert_eq!(e.code(), MinioErrorCode::NotSupported)
|
|
}
|
|
v => panic!("Expected error S3Error(NotSupported): but got {v:?}"),
|
|
}
|
|
}
|