use std::borrow::Cow;
use crate::data;
use crate::error::{ApiError, Result};
use crate::{EndpointResult, Token, TokenStorage};
#[derive(Serialize, Clone, Hash, Debug)]
pub struct LoginArgs<'a> {
email: Cow<'a, str>,
password: Cow<'a, str>,
}
impl<'a> LoginArgs<'a> {
pub fn new<T, U>(email: T, password: U) -> Self
where
T: Into<Cow<'a, str>>,
U: Into<Cow<'a, str>>,
{
LoginArgs {
email: email.into(),
password: password.into(),
}
}
}
#[derive(serde_derive::Deserialize, Clone, Hash, Debug)]
pub(crate) struct Response {
ok: i32,
token: Option<String>,
}
#[must_use = "LoggedIn does not do anything unless registered in a token store"]
#[derive(Clone, Hash, Debug)]
pub struct LoggedIn {
pub token: Token,
_non_exhaustive: (),
}
impl LoggedIn {
pub fn return_to(self, storage: &TokenStorage) {
storage.set(self.token);
}
}
impl EndpointResult for LoggedIn {
type RequestResult = Response;
type ErrorResult = data::ApiError;
fn from_raw(raw: Response) -> Result<LoggedIn> {
let Response { ok, token } = raw;
if ok != 1 {
return Err(ApiError::NotOk(ok).into());
}
match token {
Some(token) => Ok(LoggedIn {
token: token.into(),
_non_exhaustive: (),
}),
None => Err(ApiError::MissingField("token").into()),
}
}
}
#[cfg(test)]
mod tests {
use super::LoggedIn;
use crate::EndpointResult;
use serde_json;
fn test_parse(json: serde_json::Value) {
let response = serde_json::from_value(json).unwrap();
let _ = LoggedIn::from_raw(response).unwrap();
}
#[test]
fn parse_sample_login_success() {
test_parse(json! ({
"ok": 1,
"token": "c07924d3f556a355eba7cd59f4c21f670fda76c2",
}));
}
}