use crate::{
meta::{region::ProvideRegion, token::TokenProviderChain},
provider_config::ProviderConfig,
};
use aws_credential_types::provider::{future, token::ProvideToken};
pub async fn default_provider() -> impl ProvideToken {
DefaultTokenChain::builder().build().await
}
#[derive(Debug)]
pub struct DefaultTokenChain {
provider_chain: TokenProviderChain,
}
impl DefaultTokenChain {
pub fn builder() -> Builder {
Builder::default()
}
}
impl ProvideToken for DefaultTokenChain {
fn provide_token<'a>(&'a self) -> future::ProvideToken<'a>
where
Self: 'a,
{
self.provider_chain.provide_token()
}
}
#[derive(Debug, Default)]
pub struct Builder {
profile_file_builder: crate::profile::token::Builder,
region_override: Option<Box<dyn ProvideRegion>>,
region_chain: crate::default_provider::region::Builder,
conf: Option<ProviderConfig>,
}
impl Builder {
pub fn region(mut self, region: impl ProvideRegion + 'static) -> Self {
self.set_region(Some(region));
self
}
pub fn set_region(&mut self, region: Option<impl ProvideRegion + 'static>) -> &mut Self {
self.region_override = region.map(|provider| Box::new(provider) as _);
self
}
pub fn profile_name(mut self, name: &str) -> Self {
self.profile_file_builder = self.profile_file_builder.profile_name(name);
self.region_chain = self.region_chain.profile_name(name);
self
}
pub(crate) fn configure(mut self, config: ProviderConfig) -> Self {
self.region_chain = self.region_chain.configure(&config);
self.conf = Some(config);
self
}
pub async fn build(self) -> DefaultTokenChain {
let region = match self.region_override {
Some(provider) => provider.region().await,
None => self.region_chain.build().region().await,
};
let conf = self.conf.unwrap_or_default().with_region(region);
let provider_chain = TokenProviderChain::first_try(
"Profile",
self.profile_file_builder.configure(&conf).build(),
);
DefaultTokenChain { provider_chain }
}
}