|
@@ -8,6 +8,9 @@ use sha2::Sha256;
|
|
|
|
|
|
type HmacSha256 = Hmac<Sha256>;
|
|
|
|
|
|
+/// Upload payload
|
|
|
+pub type TokenPayload = [u8; 10];
|
|
|
+
|
|
|
#[derive(thiserror::Error, Debug, Serialize)]
|
|
|
/// Error type
|
|
|
pub enum Error {
|
|
@@ -40,7 +43,7 @@ impl Default for TokenManager {
|
|
|
|
|
|
impl TokenManager {
|
|
|
/// Checks if the given token is valid and still not expired
|
|
|
- pub fn verify(&self, token: Token, update_token: &Option<Vec<u8>>) -> Result<(), Error> {
|
|
|
+ pub fn verify(&self, token: Token, update_token: &Option<TokenPayload>) -> Result<(), Error> {
|
|
|
if !token.is_valid() {
|
|
|
return Ok(());
|
|
|
}
|
|
@@ -52,9 +55,10 @@ impl TokenManager {
|
|
|
};
|
|
|
|
|
|
let mut mac = HmacSha256::new_from_slice(&self.0).expect("HMAC can take key of any size");
|
|
|
- mac.update(&token.to_bytes().unwrap_or_default());
|
|
|
+ mac.update(update_token);
|
|
|
+
|
|
|
let result = mac.finalize().into_bytes();
|
|
|
- if &result[..] != update_token {
|
|
|
+ if &result[..] != token.signature {
|
|
|
Err(Error::InvalidSignature)
|
|
|
} else {
|
|
|
Ok(())
|
|
@@ -62,22 +66,26 @@ impl TokenManager {
|
|
|
}
|
|
|
|
|
|
/// Creates a new instance of the token
|
|
|
- pub fn new_token(&self, duration: Duration) -> (Token, Vec<u8>) {
|
|
|
+ pub fn new_token(&self, owner: String, duration: Duration) -> (Token, TokenPayload) {
|
|
|
let mut rng = rand::thread_rng();
|
|
|
- let mut payload = [0u8; 10];
|
|
|
+ let mut payload = TokenPayload::default();
|
|
|
rng.fill(&mut payload);
|
|
|
|
|
|
- let token = Token {
|
|
|
- owner: "owner".to_string(),
|
|
|
- payload: payload.to_vec(),
|
|
|
- expires_at: Utc::now() + duration,
|
|
|
- };
|
|
|
- let token_bytes = token.to_bytes().unwrap_or_default();
|
|
|
let mut mac = HmacSha256::new_from_slice(&self.0).expect("HMAC can take key of any size");
|
|
|
- mac.update(&token_bytes);
|
|
|
- let proof = mac.finalize().into_bytes();
|
|
|
+ mac.update(&payload);
|
|
|
+
|
|
|
+ let signature = mac.finalize().into_bytes().into();
|
|
|
|
|
|
- (token, proof.to_vec())
|
|
|
+ (
|
|
|
+ // The token cannot be altered once it is commited, as the revision ID is the hash of
|
|
|
+ // the entire content, therefore it is safer to only HMAC the payload
|
|
|
+ Token {
|
|
|
+ expires_at: Utc::now() + duration,
|
|
|
+ owner,
|
|
|
+ signature,
|
|
|
+ },
|
|
|
+ payload,
|
|
|
+ )
|
|
|
}
|
|
|
}
|
|
|
|
|
@@ -92,7 +100,7 @@ pub struct Token {
|
|
|
deserialize_with = "crate::from_ts_microseconds"
|
|
|
)]
|
|
|
expires_at: DateTime<Utc>,
|
|
|
- payload: Vec<u8>,
|
|
|
+ signature: [u8; 32],
|
|
|
owner: String,
|
|
|
}
|
|
|
|