chiark / gitweb /
server: get token, wip hmac work
[hippotat.git] / src / utils.rs
1 // Copyright 2021 Ian Jackson and contributors to Hippotat
2 // SPDX-License-Identifier: GPL-3.0-or-later
3 // There is NO WARRANTY.
4
5 use crate::prelude::*;
6
7 #[ext(pub)]
8 impl<T> T where T: Debug {
9   fn to_debug(&self) -> String { format!("{:?}", self) }
10 }
11
12 #[ext(pub)]
13 impl<T,E> Result<T,E> where AE: From<E> {
14   fn dcontext<D:Debug>(self, d: D) -> anyhow::Result<T> {
15     self.map_err(|e| AE::from(e)).with_context(|| d.to_debug())
16   }
17 }
18
19 #[derive(Error,Debug)]
20 pub enum ReadLimitedError {
21   #[error("maximum size {limit} exceeded")]
22   Truncated { sofar: Box<[u8]>, limit: usize },
23
24   #[error("HTTP error {0}")]
25   Hyper(#[from] hyper::Error),
26 }
27
28 impl ReadLimitedError {
29   pub fn discard_data(&mut self) { match self {
30     ReadLimitedError::Truncated { sofar,.. } => { mem::take(sofar); },
31     _ => { },
32   } }
33 }
34 #[ext(pub)]
35 impl<T> Result<T,ReadLimitedError> {
36   fn discard_data(self) -> Self {
37     self.map_err(|mut e| { e.discard_data(); e })
38   }
39 }
40
41 #[throws(ReadLimitedError)]
42 pub async fn read_limited_bytes<S>(limit: usize, stream: &mut S) -> Box<[u8]>
43 where S: futures::Stream<Item=Result<hyper::body::Bytes,hyper::Error>>
44          + Debug + Unpin,
45       // we also require that the Stream is cancellation-safe
46 {
47   let mut accum = vec![];
48   while let Some(item) = stream.next().await {
49     let b = item?;
50     accum.extend(b);
51     if accum.len() > limit {
52       throw!(ReadLimitedError::Truncated { limit, sofar: accum.into() })
53     }
54   }
55   accum.into()
56 }
57
58 pub fn time_t_now() -> u64 {
59   SystemTime::now()
60     .duration_since(UNIX_EPOCH)
61     .unwrap_or_else(|_| Duration::default()) // clock is being weird
62     .as_secs()
63 }
64
65 use sha2::Digest as _;
66
67 pub type HmacH = sha2::Sha256;
68 pub const HMAC_B: usize = 64;
69 pub const HMAC_L: usize = 32;
70
71 pub fn token_hmac(key: &[u8], message: &[u8]) -> [u8; HMAC_L] {
72   let key = {
73     let mut padded = [0; HMAC_B];
74     if key.len() > padded.len() {
75       let digest: [u8; HMAC_L] = HmacH::digest(key).into();
76       padded[0..HMAC_L].copy_from_slice(&digest);
77     } else {
78       padded[0.. key.len()].copy_from_slice(key);
79     }
80     padded
81   };
82   let mut ikey = key;  for k in &mut ikey { *k ^= 0x36; }
83   let mut okey = key;  for k in &mut okey { *k ^= 0x5C; }
84
85 //dbg!(&key, &ikey, &okey);
86
87   let h1 = HmacH::new()
88     .chain(&ikey)
89     .chain(message)
90     .finalize();
91   let h2 = HmacH::new()
92     .chain(&okey)
93     .chain(h1)
94     .finalize();
95   h2.into()
96 }
97
98 #[test]
99 fn hmac_test_vectors(){
100   // C&P from RFC 4231
101   let vectors = r#"
102    Key =          0b0b0b0b0b0b0b0b0b0b0b0b0b0b0b0b
103                   0b0b0b0b                          (20 bytes)
104    Data =         4869205468657265                  ("Hi There")
105
106    HMAC-SHA-256 = b0344c61d8db38535ca8afceaf0bf12b
107                   881dc200c9833da726e9376c2e32cff7
108
109     
110    Key =          4a656665                          ("Jefe")
111    Data =         7768617420646f2079612077616e7420  ("what do ya want ")
112                   666f72206e6f7468696e673f          ("for nothing?")
113
114    HMAC-SHA-256 = 5bdcc146bf60754e6a042426089575c7
115                   5a003f089d2739839dec58b964ec3843
116
117
118    Key =          aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa
119                   aaaaaaaa                          (20 bytes)
120    Data =         dddddddddddddddddddddddddddddddd
121                   dddddddddddddddddddddddddddddddd
122                   dddddddddddddddddddddddddddddddd
123                   dddd                              (50 bytes)
124
125    HMAC-SHA-256 = 773ea91e36800e46854db8ebd09181a7
126                   2959098b3ef8c122d9635514ced565fe
127
128
129    Key =          0102030405060708090a0b0c0d0e0f10
130                   111213141516171819                (25 bytes)
131    Data =         cdcdcdcdcdcdcdcdcdcdcdcdcdcdcdcd
132                   cdcdcdcdcdcdcdcdcdcdcdcdcdcdcdcd
133                   cdcdcdcdcdcdcdcdcdcdcdcdcdcdcdcd
134                   cdcd                              (50 bytes)
135
136    HMAC-SHA-256 = 82558a389a443c0ea4cc819899f2083a
137                   85f0faa3e578f8077a2e3ff46729665b
138
139
140
141    Key =          aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa
142                   aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa
143                   aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa
144                   aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa
145                   aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa
146                   aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa
147                   aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa
148                   aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa
149                   aaaaaa                            (131 bytes)
150    Data =         54657374205573696e67204c61726765  ("Test Using Large")
151                   72205468616e20426c6f636b2d53697a  ("r Than Block-Siz")
152                   65204b6579202d2048617368204b6579  ("e Key - Hash Key")
153                   204669727374                      (" First")
154
155    HMAC-SHA-256 = 60e431591ee0b67f0d8a26aacbf5b77f
156                   8e0bc6213728c5140546040f0ee37f54
157
158    Key =          aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa
159                   aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa
160                   aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa
161                   aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa
162                   aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa
163                   aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa
164                   aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa
165                   aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa
166                   aaaaaa                            (131 bytes)
167    Data =         54686973206973206120746573742075  ("This is a test u")
168                   73696e672061206c6172676572207468  ("sing a larger th")
169                   616e20626c6f636b2d73697a65206b65  ("an block-size ke")
170                   7920616e642061206c61726765722074  ("y and a larger t")
171                   68616e20626c6f636b2d73697a652064  ("han block-size d")
172                   6174612e20546865206b6579206e6565  ("ata. The key nee")
173                   647320746f2062652068617368656420  ("ds to be hashed ")
174                   6265666f7265206265696e6720757365  ("before being use")
175                   642062792074686520484d414320616c  ("d by the HMAC al")
176                   676f726974686d2e                  ("gorithm.")
177
178    HMAC-SHA-256 = 9b09ffa71b942fcb27635fbcd5b0e944
179                   bfdc63644f0713938a7f51535c3a35e2
180 "#;
181   let vectors = regex_replace_all!{
182     r#"\(.*\)"#,
183     vectors.trim_end(),
184     |_| "",
185   };
186   let vectors = regex_replace_all!{
187     r#" *\n                  "#,
188     &vectors,
189     |_| "",
190   };
191   let vectors = regex_replace_all!{
192     r#"\s*\n"#,
193     &vectors,
194     |_| "\n",
195   };
196   let mut lines = vectors.split('\n');
197   assert_eq!( lines.next().unwrap(), "" );
198   let mut get = |prefix| {
199     let l = lines.next()?;
200     dbg!(l);
201     let b = l.strip_prefix(prefix).unwrap().as_bytes().chunks(2)
202       .map(|s| str::from_utf8(s).unwrap())
203       .map(|s| { assert_eq!(s.len(), 2); u8::from_str_radix(s,16).unwrap() })
204       .collect::<Vec<u8>>();
205     Some(b)
206   };
207   while let Some(key) = get("   Key =          ") {
208     let data = get("   Data =         ").unwrap();
209     let exp = get("   HMAC-SHA-256 = ").unwrap();
210     let got = token_hmac(&key, &data);
211     assert_eq!(&got[..], &exp);
212   }
213 }