bitwarden_core/auth/
jwt_token.rs1use std::str::FromStr;
2
3use base64::{engine::general_purpose::URL_SAFE_NO_PAD, Engine};
4use thiserror::Error;
5
6#[derive(serde::Deserialize)]
14pub struct JwtToken {
15 pub exp: u64,
17 pub sub: String,
19 pub email: Option<String>,
21 pub organization: Option<String>,
23 pub scope: Vec<String>,
25}
26
27#[allow(missing_docs)]
29#[derive(Debug, Error)]
30pub enum JwtTokenParseError {
31 #[error("JWT token parse error: {0}")]
32 Parse(#[from] serde_json::Error),
33 #[error("JWT token decode error: {0}")]
34 Decode(#[from] base64::DecodeError),
35
36 #[error("JWT token has an invalid number of parts")]
37 InvalidParts,
38}
39
40impl FromStr for JwtToken {
41 type Err = JwtTokenParseError;
42
43 fn from_str(s: &str) -> Result<Self, Self::Err> {
47 let split = s.split('.').collect::<Vec<_>>();
48 if split.len() != 3 {
49 return Err(Self::Err::InvalidParts);
50 }
51 let decoded = URL_SAFE_NO_PAD.decode(split[1])?;
52 Ok(serde_json::from_slice(&decoded)?)
53 }
54}
55
56#[cfg(test)]
57mod tests {
58 use crate::auth::jwt_token::JwtToken;
59
60 #[test]
61 fn can_decode_jwt() {
62 let jwt = "eyJhbGciOiJSUzI1NiIsImtpZCI6IjMwMURENkE1MEU4NEUxRDA5MUM4MUQzQjAwQkY5MDEwQz\
63 g1REJEOUFSUzI1NiIsInR5cCI6ImF0K2p3dCIsIng1dCI6Ik1CM1dwUTZFNGRDUnlCMDdBTC1RRU1oZHZabyJ9.eyJu\
64 YmYiOjE2NzUxMDM1NzcsImV4cCI6MTY3NTEwNzE3NywiaXNzIjoiaHR0cDovL2xvY2FsaG9zdCIsImNsaWVudF9pZCI\
65 6IndlYiIsInN1YiI6ImUyNWQzN2YzLWI2MDMtNDBkZS04NGJhLWFmOTYwMTJmNWE0MiIsImF1dGhfdGltZSI6MTY3NT\
66 EwMzU0OSwiaWRwIjoiYml0d2FyZGVuIiwicHJlbWl1bSI6ZmFsc2UsImVtYWlsIjoidGVzdEBiaXR3YXJkZW4uY29tI\
67 iwiZW1haWxfdmVyaWZpZWQiOnRydWUsInNzdGFtcCI6IkUzNElDWVhRUFRDS01EVldBREZDNktHNDJCQldJRDdJIiwi\
68 bmFtZSI6IlRlc3QiLCJvcmdvd25lciI6ImY0ZTQ0YTdmLTExOTAtNDMyYS05ZDRhLWFmOTYwMTMxMjdjYiIsImRldml\
69 jZSI6Ijg5Mjg5M2FiLWRkNDMtNDUwYS04NGI1LWFhOWM1YjdiYjJkOCIsImp0aSI6IkEzMkVFNjY5NDdEQzlDNUE2MT\
70 IwRURBRTIwNzc5OUJFIiwiaWF0IjoxNjc1MTAzNTc3LCJzY29wZSI6WyJhcGkiLCJvZmZsaW5lX2FjY2VzcyJdLCJhb\
71 XIiOlsiQXBwbGljYXRpb24iXX0.AyDkKvjmyaSPQViQSa2sGTKIkDGrUAtDmwpE57K4DDWT0QvwDe7FMktmwiF4LH36\
72 wx_FnpH21VI1pzwJeTHXtaz3niANJtQZjzGFsNAna_95vrsxZC2YizgGlt6mX4YIGmAw9DiYrmaN0BvQOEm_caV_u6f\
73 a30iz9Kvjxf7cpzeZvPEysxGpB3k3TRYTkFUdV43HiXdhXMBhyyOpFU6Fk6yA41y7-8bGYc5mYGknWktmPD9Yx-1xKL\
74 ftFja1SnCoLPWvDeK60lqWZQiT4tZHCYJ7m0bBNCccYHc2Kk2Bo5-UoyDxazPwsqMxeNfjlaUuj3o5N_uQ-4n_gVbeA\
75 qWV2wrel5UhYjWnczMSLBtt9p0W35kkBPt3ZAnRWMtQMPNH04p-_L6cG-Xu6lDksBTwaavcmtnCKG8V91826EiQ8MrF\
76 wGWQRZV6tPKTDAYCgSAZGBY3QDmPGT5BeFcg5Ag_nYYIIifKP-kv10v_N-TOcT3NeGBOUlAZ-9m7iT7Rk3vC--SDZdA\
77 U5turoBFiiPL2XXfAjM7P0r7J91gfXc0FaD6I2jDxOmym5h7Yn5phLsbC2NlIXkZp54dKHICenPl4ve6ndDIJacVeS5\
78 f3LEddAPV8cAFza4DjA8pZJLFrMyRvMXcL_PjKF8qPVzqVWh03lfJ4clOIxR2gOuWIc902Y5E";
79
80 let token: JwtToken = jwt.parse().unwrap();
81 assert_eq!(token.exp, 1675107177);
82 assert_eq!(token.sub, "e25d37f3-b603-40de-84ba-af96012f5a42");
83 assert_eq!(token.email.as_deref(), Some("[email protected]"));
84 assert_eq!(token.organization.as_deref(), None);
85 assert_eq!(token.scope[0], "api");
86 assert_eq!(token.scope[1], "offline_access");
87 }
88}