From 77ebf53d20fbf4527352039a9f137d8a8f11e715 Mon Sep 17 00:00:00 2001 From: Takeshi Nakatani Date: Sun, 3 Mar 2024 11:36:53 +0000 Subject: [PATCH] Improved to add option for Session Token Period Second --- README.md | 4 +++ awscred_func.cpp | 70 ++++++++++++++++++++++++++++++++++++++++++++++-- 2 files changed, 72 insertions(+), 2 deletions(-) diff --git a/README.md b/README.md index bfb50ff..fac7556 100644 --- a/README.md +++ b/README.md @@ -76,6 +76,10 @@ _These options are the same as the log level defined in `aws-sdk-cpp`(Aws::Utils - SSOProfile(SSOProf) Specify the SSO profile name. _(mainly the name written in sso-session in `.aws/config`.)_ _This DSO cannot handle that authentication callback when it comes to SSO, so it is a temporary token acquisition._ +- TokenPeriodSecond(PeriodSec) +Specify the validity period of the Session Token in seconds. +_If this option is specified, the Session Token will be considered valid for this validity period(in seconds), starting from the first time this Token is read._ +_User cannot set an expiration date for Credentials(`.aws/` or environment variables), so if this value is not set, the expiration date will indicate a long time in the future._ If you want to specify multiple options above, please specify them using a comma(`,`) as a delimiter. diff --git a/awscred_func.cpp b/awscred_func.cpp index 8d75a3a..bfe1d66 100644 --- a/awscred_func.cpp +++ b/awscred_func.cpp @@ -97,6 +97,56 @@ static Aws::String& GetSSOProfile() return ssoprofile; } +//---------------------------------------------------------- +// Auxiliary Valid period seconds +//---------------------------------------------------------- +// [NOTE] About Session Token Expiration +// There is no key in .aws/config or .aws/credential that +// specifies the SessionToken Expiration.(missing keys like +// aws_session_expiration) +// This library only provides a function to load Credentials +// and does not have a function to obtain SessionTokens, so +// SessionTokens(and AccessKeys, Secrets, etc.) can only be +// obtained from Credentials (environment variables, files, +// etc). +// This means that unless you pass the Expriation externally, +// you won't know its expiration date. +// Therefore, it can only be passed as a constant value as +// an option to this library.(This may change in the future) +// +static int64_t periodsec = -1; + +static bool SetValidPeriodSec(int64_t sec) +{ + if(-1 != periodsec){ + return false; + } + if(sec <= 0 || (60 * 60 * 24 * 365 * 5) < sec){ // Maximum is 5 years + return false; + } + periodsec = sec; + + return true; +} + +static const Aws::Utils::DateTime& GetExparationByValidPeriod(const Aws::String& sessionToken, const Aws::Utils::DateTime& exp) +{ + static Aws::String targetSessionToken; + static Aws::Utils::DateTime targetExpiration; + + if(-1 == periodsec){ + return exp; + } + if(targetSessionToken != sessionToken){ + // Update new session token + int64_t expms = exp.Millis(); + int64_t maxms = Aws::Utils::DateTime::CurrentTimeMillis() + (periodsec * 1000); + targetExpiration = Aws::Utils::DateTime(std::min(expms, maxms)); + targetSessionToken = sessionToken; + } + return targetExpiration; +} + //---------------------------------------------------------- // Export interface functions //---------------------------------------------------------- @@ -169,6 +219,22 @@ bool InitS3fsCredential(const char* popts, char** pperrstr) } ssoprofile = strValue.c_str(); + }else if(0 == strcasecmp(strLowkey.c_str(), "TokenPeriodSecond") || 0 == strcasecmp(strLowkey.c_str(), "PeriodSec")){ + if(strValue.empty()){ + if(pperrstr){ + *pperrstr = strdup("Option(SSOProfile) value is empty."); + } + return false; + } + int64_t periodsec = static_cast(stoll(strValue)); + + if(!SetValidPeriodSec(periodsec)){ + if(pperrstr){ + *pperrstr = strdup("Failed to set Session Token Period Seconds."); + } + return false; + } + }else if(0 == strcasecmp(strLowkey.c_str(), "LogLevel")){ if(0 == strcasecmp(strValue.c_str(), "Off")){ if(isSetLogLevel){ @@ -367,13 +433,13 @@ bool UpdateS3fsCredential(char** ppaccess_key_id, char** ppserect_access_key, ch Aws::String accessKeyId = credentials.GetAWSAccessKeyId(); Aws::String secretKey = credentials.GetAWSSecretKey(); Aws::String sessionToken= credentials.GetSessionToken(); - Aws::Utils::DateTime expiration = credentials.GetExpiration(); + Aws::Utils::DateTime expiration = GetExparationByValidPeriod(sessionToken, credentials.GetExpiration()); // Set result buffers *ppaccess_key_id = strdup(accessKeyId.c_str()); *ppserect_access_key= strdup(secretKey.c_str()); *ppaccess_token = strdup(sessionToken.c_str()); - *ptoken_expire = expiration.Millis() / 1000; // msec to unittime(s) + *ptoken_expire = static_cast(expiration.Seconds()); // For debug if(Aws::Utils::Logging::LogLevel::Info <= options.loggingOptions.logLevel){