minio-rs/tests/s3/bucket_replication.rs
Henk-Jan Lebbink c450081b2e
Refactor string parameters to typed wrapper structs (#200)
- 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
2026-01-27 14:12:49 +01:00

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:?}"),
}
}