1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
use crate::date_time::format_date;
use ring::{
digest::{self},
hmac::{self, Key, Tag},
};
use std::time::SystemTime;
#[allow(dead_code)]
pub(crate) fn sha256_hex_string(bytes: impl AsRef<[u8]>) -> String {
hex::encode(digest::digest(&digest::SHA256, bytes.as_ref()))
}
pub fn calculate_signature(signing_key: Tag, string_to_sign: &[u8]) -> String {
let s_key = Key::new(hmac::HMAC_SHA256, signing_key.as_ref());
let tag = hmac::sign(&s_key, string_to_sign);
hex::encode(tag)
}
pub fn generate_signing_key(
secret: &str,
time: SystemTime,
region: &str,
service: &str,
) -> hmac::Tag {
let secret = format!("AWS4{}", secret);
let secret = hmac::Key::new(hmac::HMAC_SHA256, secret.as_bytes());
let tag = hmac::sign(&secret, format_date(time).as_bytes());
let key = hmac::Key::new(hmac::HMAC_SHA256, tag.as_ref());
let tag = hmac::sign(&key, region.as_bytes());
let key = hmac::Key::new(hmac::HMAC_SHA256, tag.as_ref());
let tag = hmac::sign(&key, service.as_bytes());
let key = hmac::Key::new(hmac::HMAC_SHA256, tag.as_ref());
hmac::sign(&key, "aws4_request".as_bytes())
}
#[cfg(test)]
mod tests {
use super::{calculate_signature, generate_signing_key};
use crate::date_time::test_parsers::parse_date_time;
use crate::http_request::test::test_canonical_request;
use crate::sign::sha256_hex_string;
#[test]
fn test_signature_calculation() {
let secret = "wJalrXUtnFEMI/K7MDENG+bPxRfiCYEXAMPLEKEY";
let creq = test_canonical_request("iam");
let time = parse_date_time("20150830T123600Z").unwrap();
let derived_key = generate_signing_key(secret, time, "us-east-1", "iam");
let signature = calculate_signature(derived_key, creq.as_bytes());
let expected = "5d672d79c15b13162d9279b0855cfba6789a8edb4c82c400e06b5924a6f2b5d7";
assert_eq!(expected, &signature);
}
#[test]
fn sign_payload_empty_string() {
let expected = "e3b0c44298fc1c149afbf4c8996fb92427ae41e4649b934ca495991b7852b855";
let actual = sha256_hex_string(&[]);
assert_eq!(expected, actual);
}
}