-
Notifications
You must be signed in to change notification settings - Fork 11
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Co-authored-by: Christopher Kolstad <chriswk@getunleash.io>
- Loading branch information
Showing
9 changed files
with
1,172 additions
and
82 deletions.
There are no files selected for viewing
Large diffs are not rendered by default.
Oops, something went wrong.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,177 @@ | ||
use std::collections::HashMap; | ||
|
||
use async_trait::async_trait; | ||
use unleash_types::client_features::ClientFeatures; | ||
|
||
use super::EdgePersistence; | ||
use crate::{ | ||
error::EdgeError, | ||
types::{EdgeResult, EdgeToken}, | ||
}; | ||
use aws_sdk_s3::{ | ||
self as s3, | ||
error::SdkError, | ||
operation::{get_object::GetObjectError, put_object::PutObjectError}, | ||
primitives::{ByteStream, SdkBody}, | ||
}; | ||
|
||
pub const FEATURES_KEY: &str = "/unleash-features.json"; | ||
pub const TOKENS_KEY: &str = "/unleash-tokens.json"; | ||
|
||
pub struct S3Persister { | ||
client: s3::Client, | ||
bucket: String, | ||
} | ||
|
||
impl S3Persister { | ||
pub fn new_with_config(bucket_name: &str, config: s3::config::Config) -> Self { | ||
let client = s3::Client::from_conf(config); | ||
Self { | ||
client, | ||
bucket: bucket_name.to_string(), | ||
} | ||
} | ||
pub async fn new_from_env(bucket_name: &str) -> Self { | ||
let shared_config = aws_config::load_from_env().await; | ||
let client = s3::Client::new(&shared_config); | ||
Self { | ||
client, | ||
bucket: bucket_name.to_string(), | ||
} | ||
} | ||
} | ||
|
||
impl From<SdkError<GetObjectError>> for EdgeError { | ||
fn from(err: SdkError<GetObjectError>) -> Self { | ||
EdgeError::PersistenceError(format!("failed to get object {}", err)) | ||
} | ||
} | ||
|
||
impl From<SdkError<PutObjectError>> for EdgeError { | ||
fn from(err: SdkError<PutObjectError>) -> Self { | ||
EdgeError::PersistenceError(format!("failed to put object {}", err)) | ||
} | ||
} | ||
|
||
impl S3Persister { | ||
async fn create_bucket_if_not_exists(&self) -> EdgeResult<()> { | ||
match self | ||
.client | ||
.create_bucket() | ||
.bucket(&self.bucket) | ||
.send() | ||
.await | ||
{ | ||
Ok(_) => Ok(()), | ||
Err(err) => { | ||
if err.to_string().contains("BucketAlreadyOwnedByYou") | ||
|| err.to_string().contains("BucketAlreadyExists") | ||
{ | ||
Ok(()) | ||
} else { | ||
Err(EdgeError::PersistenceError(format!( | ||
"Failed to create bucket: {}", | ||
err | ||
))) | ||
} | ||
} | ||
} | ||
} | ||
} | ||
|
||
#[async_trait] | ||
impl EdgePersistence for S3Persister { | ||
async fn load_tokens(&self) -> EdgeResult<Vec<EdgeToken>> { | ||
let response = self | ||
.client | ||
.get_object() | ||
.bucket(self.bucket.clone()) | ||
.key(TOKENS_KEY) | ||
.response_content_type("application/json") | ||
.send() | ||
.await?; | ||
let data = response.body.collect().await.expect("Failed data"); | ||
serde_json::from_slice(&data.to_vec()) | ||
.map_err(|_| EdgeError::PersistenceError("Failed to deserialize tokens".to_string())) | ||
} | ||
|
||
async fn save_tokens(&self, tokens: Vec<EdgeToken>) -> EdgeResult<()> { | ||
self.create_bucket_if_not_exists().await?; | ||
let body_data = serde_json::to_vec(&tokens) | ||
.map_err(|_| EdgeError::PersistenceError("Failed to serialize tokens".to_string())) | ||
.map(SdkBody::from)?; | ||
let byte_stream = aws_sdk_s3::primitives::ByteStream::new(body_data); | ||
self.client | ||
.put_object() | ||
.bucket(self.bucket.clone()) | ||
.key(TOKENS_KEY) | ||
.body(byte_stream) | ||
.send() | ||
.await | ||
.map(|_| ()) | ||
.map_err(|err| { | ||
dbg!(err); | ||
EdgeError::PersistenceError("Failed to save tokens".to_string()) | ||
}) | ||
} | ||
|
||
async fn load_features(&self) -> EdgeResult<HashMap<String, ClientFeatures>> { | ||
let query = self | ||
.client | ||
.get_object() | ||
.bucket(self.bucket.clone()) | ||
.key(FEATURES_KEY) | ||
.response_content_type("application/json") | ||
.send() | ||
.await | ||
.map_err(|err| { | ||
if err.to_string().contains("NoSuchKey") { | ||
return EdgeError::PersistenceError("No features found".to_string()); | ||
} | ||
dbg!(err); | ||
EdgeError::PersistenceError("Failed to load features".to_string()) | ||
}); | ||
match query { | ||
Ok(response) => { | ||
let data = response.body.collect().await.expect("Failed data"); | ||
let deser: Vec<(String, ClientFeatures)> = serde_json::from_slice(&data.to_vec()) | ||
.map_err(|_| { | ||
EdgeError::PersistenceError("Failed to deserialize features".to_string()) | ||
})?; | ||
Ok(deser | ||
.iter() | ||
.cloned() | ||
.collect::<HashMap<String, ClientFeatures>>()) | ||
} | ||
Err(e) => { | ||
eprintln!("Err Arg, failed to read features"); | ||
dbg!(e); | ||
Ok(HashMap::new()) | ||
} | ||
} | ||
} | ||
|
||
async fn save_features(&self, features: Vec<(String, ClientFeatures)>) -> EdgeResult<()> { | ||
self.create_bucket_if_not_exists().await?; | ||
let body_data = serde_json::to_vec(&features) | ||
.map_err(|_| EdgeError::PersistenceError("Failed to serialize features".to_string()))?; | ||
let byte_stream = ByteStream::new(SdkBody::from(body_data)); | ||
match self | ||
.client | ||
.put_object() | ||
.bucket(self.bucket.clone()) | ||
.key(FEATURES_KEY) | ||
.body(byte_stream) | ||
.send() | ||
.await | ||
{ | ||
Ok(_) => Ok(()), | ||
Err(s3_err) => { | ||
dbg!(s3_err); | ||
Err(EdgeError::PersistenceError( | ||
"Failed to save features".to_string(), | ||
)) | ||
} | ||
} | ||
} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Oops, something went wrong.