use reqsign_core::Context;
use reqsign_core::ProvideCredential;
use reqsign_core::Result;
use super::super::constants::*;
use super::super::credential::Credential;
#[derive(Debug, Default, Clone)]
pub struct EnvCredentialProvider;
impl EnvCredentialProvider {
pub fn new() -> Self {
Self
}
}
impl ProvideCredential for EnvCredentialProvider {
type Credential = Credential;
async fn provide_credential(&self, ctx: &Context) -> Result<Option<Self::Credential>> {
let env = ctx.env_vars();
let access_key_id = env.get(ENV_ACCESS_KEY_ID).cloned();
let secret_access_key = env.get(ENV_SECRET_ACCESS_KEY).cloned();
let security_token = env.get(ENV_SESSION_TOKEN).cloned();
match (access_key_id, secret_access_key) {
(Some(ak), Some(sk)) => {
let mut cred = Credential::new(&ak, &sk);
if let Some(token) = security_token {
cred = cred.with_session_token(&token);
}
Ok(Some(cred))
}
_ => Ok(None),
}
}
}
#[cfg(test)]
mod tests {
use super::*;
use reqsign_core::StaticEnv;
use std::collections::HashMap;
#[tokio::test]
async fn test_env_credential_provider() -> anyhow::Result<()> {
let envs = HashMap::from([
(ENV_ACCESS_KEY_ID.to_string(), "test_access_key".to_string()),
(
ENV_SECRET_ACCESS_KEY.to_string(),
"test_secret_key".to_string(),
),
]);
let ctx = Context::new().with_env(StaticEnv {
home_dir: None,
envs,
});
let provider = EnvCredentialProvider::new();
let cred = provider.provide_credential(&ctx).await?;
assert!(cred.is_some());
let cred = cred.unwrap();
assert_eq!(cred.access_key_id, "test_access_key");
assert_eq!(cred.secret_access_key, "test_secret_key");
assert!(cred.session_token.is_none());
Ok(())
}
#[tokio::test]
async fn test_env_credential_provider_with_session_token() -> anyhow::Result<()> {
let envs = HashMap::from([
(ENV_ACCESS_KEY_ID.to_string(), "test_access_key".to_string()),
(
ENV_SECRET_ACCESS_KEY.to_string(),
"test_secret_key".to_string(),
),
(
ENV_SESSION_TOKEN.to_string(),
"test_session_token".to_string(),
),
]);
let ctx = Context::new().with_env(StaticEnv {
home_dir: None,
envs,
});
let provider = EnvCredentialProvider::new();
let cred = provider.provide_credential(&ctx).await?;
assert!(cred.is_some());
let cred = cred.unwrap();
assert_eq!(cred.access_key_id, "test_access_key");
assert_eq!(cred.secret_access_key, "test_secret_key");
assert_eq!(cred.session_token, Some("test_session_token".to_string()));
Ok(())
}
#[tokio::test]
async fn test_env_credential_provider_missing_credentials() -> anyhow::Result<()> {
let ctx = Context::new().with_env(StaticEnv::default());
let provider = EnvCredentialProvider::new();
let cred = provider.provide_credential(&ctx).await?;
assert!(cred.is_none());
Ok(())
}
#[tokio::test]
async fn test_env_credential_provider_partial_credentials() -> anyhow::Result<()> {
let envs = HashMap::from([(ENV_ACCESS_KEY_ID.to_string(), "test_access_key".to_string())]);
let ctx = Context::new().with_env(StaticEnv {
home_dir: None,
envs,
});
let provider = EnvCredentialProvider::new();
let cred = provider.provide_credential(&ctx).await?;
assert!(cred.is_none());
Ok(())
}
}