in core/src/services/cos/backend.rs [168:307]
fn build(self) -> Result<impl Access> {
debug!("backend build started: {:?}", &self);
let root = normalize_root(&self.config.root.unwrap_or_default());
debug!("backend use root {}", root);
let bucket = match &self.config.bucket {
Some(bucket) => Ok(bucket.to_string()),
None => Err(
Error::new(ErrorKind::ConfigInvalid, "The bucket is misconfigured")
.with_context("service", Scheme::Cos),
),
}?;
debug!("backend use bucket {}", &bucket);
let uri = match &self.config.endpoint {
Some(endpoint) => endpoint.parse::<Uri>().map_err(|err| {
Error::new(ErrorKind::ConfigInvalid, "endpoint is invalid")
.with_context("service", Scheme::Cos)
.with_context("endpoint", endpoint)
.set_source(err)
}),
None => Err(Error::new(ErrorKind::ConfigInvalid, "endpoint is empty")
.with_context("service", Scheme::Cos)),
}?;
let scheme = match uri.scheme_str() {
Some(scheme) => scheme.to_string(),
None => "https".to_string(),
};
// If endpoint contains bucket name, we should trim them.
let endpoint = uri.host().unwrap().replace(&format!("//{bucket}."), "//");
debug!("backend use endpoint {}", &endpoint);
let mut cfg = TencentCosConfig::default();
if !self.config.disable_config_load {
cfg = cfg.from_env();
}
if let Some(v) = self.config.secret_id {
cfg.secret_id = Some(v);
}
if let Some(v) = self.config.secret_key {
cfg.secret_key = Some(v);
}
let cred_loader = TencentCosCredentialLoader::new(GLOBAL_REQWEST_CLIENT.clone(), cfg);
let signer = TencentCosSigner::new();
Ok(CosBackend {
core: Arc::new(CosCore {
info: {
let am = AccessorInfo::default();
am.set_scheme(Scheme::Cos)
.set_root(&root)
.set_name(&bucket)
.set_native_capability(Capability {
stat: true,
stat_with_if_match: true,
stat_with_if_none_match: true,
stat_has_cache_control: true,
stat_has_content_length: true,
stat_has_content_type: true,
stat_has_content_encoding: true,
stat_has_content_range: true,
stat_with_version: self.config.enable_versioning,
stat_has_etag: true,
stat_has_content_md5: true,
stat_has_last_modified: true,
stat_has_content_disposition: true,
stat_has_version: true,
stat_has_user_metadata: true,
read: true,
read_with_if_match: true,
read_with_if_none_match: true,
read_with_version: self.config.enable_versioning,
write: true,
write_can_empty: true,
write_can_append: true,
write_can_multi: true,
write_with_content_type: true,
write_with_cache_control: true,
write_with_content_disposition: true,
// Cos doesn't support forbid overwrite while version has been enabled.
write_with_if_not_exists: !self.config.enable_versioning,
// The min multipart size of COS is 1 MiB.
//
// ref: <https://www.tencentcloud.com/document/product/436/14112>
write_multi_min_size: Some(1024 * 1024),
// The max multipart size of COS is 5 GiB.
//
// ref: <https://www.tencentcloud.com/document/product/436/14112>
write_multi_max_size: if cfg!(target_pointer_width = "64") {
Some(5 * 1024 * 1024 * 1024)
} else {
Some(usize::MAX)
},
write_with_user_metadata: true,
delete: true,
delete_with_version: self.config.enable_versioning,
copy: true,
list: true,
list_with_recursive: true,
list_with_versions: self.config.enable_versioning,
list_with_deleted: self.config.enable_versioning,
list_has_content_length: true,
presign: true,
presign_stat: true,
presign_read: true,
presign_write: true,
shared: true,
..Default::default()
});
// allow deprecated api here for compatibility
#[allow(deprecated)]
if let Some(client) = self.http_client {
am.update_http_client(|_| client);
}
am.into()
},
bucket: bucket.clone(),
root,
endpoint: format!("{}://{}.{}", &scheme, &bucket, &endpoint),
signer,
loader: cred_loader,
}),
})
}