use hyper::header; use serde_json::Value; use client::response::{FromResponse, ParseError}; use token::{Token, Lifetime}; /// The bearer token type. /// /// See [RFC 6750](http://tools.ietf.org/html/rfc6750). #[derive(Debug, Clone, PartialEq, Eq, Serialize, Deserialize)] pub struct Bearer { access_token: String, scope: Option, lifetime: L, } impl Token for Bearer { fn access_token(&self) -> &str { &self.access_token } fn scope(&self) -> Option<&str> { self.scope.as_ref().map(|s| &s[..]) } fn lifetime(&self) -> &L { &self.lifetime } } impl<'a, L: Lifetime> Into> for &'a Bearer { fn into(self) -> header::Authorization { header::Authorization(header::Bearer { token: self.access_token.clone() }) } } impl Bearer { fn from_response_and_lifetime(json: &Value, lifetime: L) -> Result { let obj = json.as_object().ok_or(ParseError::ExpectedType("object"))?; let token_type = obj.get("token_type") .and_then(Value::as_str) .ok_or(ParseError::ExpectedFieldType("token_type", "string"))?; if token_type != "Bearer" && token_type != "bearer" { return Err(ParseError::ExpectedFieldValue("token_type", "Bearer")); } let access_token = obj.get("access_token") .and_then(Value::as_str) .ok_or(ParseError::ExpectedFieldType("access_token", "string"))?; let scope = obj.get("scope").and_then(Value::as_str); Ok(Bearer { access_token: access_token.into(), scope: scope.map(Into::into), lifetime: lifetime, }) } } impl FromResponse for Bearer { fn from_response(json: &Value) -> Result { let lifetime = FromResponse::from_response(json)?; Bearer::from_response_and_lifetime(json, lifetime) } fn from_response_inherit(json: &Value, prev: &Self) -> Result { let lifetime = FromResponse::from_response_inherit(json, &prev.lifetime)?; Bearer::from_response_and_lifetime(json, lifetime) } } #[cfg(test)] mod tests { use chrono::{Utc, Duration}; use client::response::{FromResponse, ParseError}; use token::{Static, Refresh}; use super::Bearer; #[test] fn from_response_with_invalid_token_type() { let json = r#"{"token_type":"MAC","access_token":"aaaaaaaa"}"#.parse().unwrap(); assert_eq!( ParseError::ExpectedFieldValue("token_type", "Bearer"), Bearer::::from_response(&json).unwrap_err() ); } #[test] fn from_response_capital_b() { let json = r#"{"token_type":"Bearer","access_token":"aaaaaaaa"}"#.parse().unwrap(); assert_eq!( Bearer { access_token: String::from("aaaaaaaa"), scope: None, lifetime: Static, }, Bearer::::from_response(&json).unwrap() ); } #[test] fn from_response_little_b() { let json = r#"{"token_type":"bearer","access_token":"aaaaaaaa"}"#.parse().unwrap(); assert_eq!( Bearer { access_token: String::from("aaaaaaaa"), scope: None, lifetime: Static, }, Bearer::::from_response(&json).unwrap() ); } #[test] fn from_response_with_scope() { let json = r#"{"token_type":"Bearer","access_token":"aaaaaaaa","scope":"foo"}"# .parse() .unwrap(); assert_eq!( Bearer { access_token: String::from("aaaaaaaa"), scope: Some(String::from("foo")), lifetime: Static, }, Bearer::::from_response(&json).unwrap() ); } #[test] fn from_response_refresh() { let json = r#" { "token_type":"Bearer", "access_token":"aaaaaaaa", "expires_in":3600, "refresh_token":"bbbbbbbb" } "#.parse().unwrap(); let bearer = Bearer::::from_response(&json).unwrap(); assert_eq!("aaaaaaaa", bearer.access_token); assert_eq!(None, bearer.scope); let refresh = bearer.lifetime; assert_eq!("bbbbbbbb", refresh.refresh_token()); assert!(refresh.expires() > &Utc::now()); assert!(refresh.expires() <= &(Utc::now() + Duration::seconds(3600))); } #[test] fn from_response_inherit_refresh() { let json = r#" { "token_type":"Bearer", "access_token":"aaaaaaaa", "expires_in":3600, "refresh_token":"bbbbbbbb" } "#.parse().unwrap(); let prev = Bearer::::from_response(&json).unwrap(); let json = r#" { "token_type":"Bearer", "access_token":"cccccccc", "expires_in":3600 } "#.parse().unwrap(); let bearer = Bearer::::from_response_inherit(&json, &prev).unwrap(); assert_eq!("cccccccc", bearer.access_token); assert_eq!(None, bearer.scope); let refresh = bearer.lifetime; assert_eq!("bbbbbbbb", refresh.refresh_token()); assert!(refresh.expires() > &Utc::now()); assert!(refresh.expires() <= &(Utc::now() + Duration::seconds(3600))); } }