use chrono::{DateTime, Utc};
use serde::{Deserialize, Serialize};
use super::{DomainAuthInfo, DomainContact, HostAttr, XMLNS};
use crate::common::{NoExtension, ObjectStatus, StringValue};
use crate::request::{Command, Transaction};
impl<'a> Transaction<NoExtension> for DomainInfo<'a> {}
impl<'a> Command for DomainInfo<'a> {
type Response = DomainInfoResponse;
const COMMAND: &'static str = "info";
}
impl<'a> DomainInfo<'a> {
pub fn new(name: &'a str, auth_password: Option<&'a str>) -> Self {
Self {
info: DomainInfoRequestData {
xmlns: XMLNS,
domain: Domain { hosts: "all", name },
auth_info: auth_password.map(|password| DomainAuthInfo {
password: password.into(),
}),
},
}
}
}
#[derive(Serialize, Debug)]
pub struct Domain<'a> {
hosts: &'a str,
#[serde(rename = "$value")]
name: &'a str,
}
#[derive(Serialize, Debug)]
pub struct DomainInfoRequestData<'a> {
#[serde(rename = "xmlns:domain")]
xmlns: &'a str,
#[serde(rename = "domain:name")]
domain: Domain<'a>,
#[serde(rename = "domain:authInfo")]
auth_info: Option<DomainAuthInfo<'a>>,
}
#[derive(Serialize, Debug)]
pub struct DomainInfo<'a> {
#[serde(rename = "domain:info")]
info: DomainInfoRequestData<'a>,
}
#[derive(Deserialize, Debug)]
pub struct DomainNsList {
#[serde(rename = "hostObj")]
pub host_obj: Option<Vec<StringValue<'static>>>,
pub host_attr: Option<Vec<HostAttr<'static>>>,
}
#[derive(Deserialize, Debug)]
pub struct DomainInfoResponseData {
pub name: StringValue<'static>,
pub roid: StringValue<'static>,
#[serde(rename = "status")]
pub statuses: Option<Vec<ObjectStatus<'static>>>,
pub registrant: Option<StringValue<'static>>,
#[serde(rename = "contact")]
pub contacts: Option<Vec<DomainContact<'static>>>,
#[serde(rename = "ns")]
pub ns: Option<DomainNsList>,
#[serde(rename = "host")]
pub hosts: Option<Vec<StringValue<'static>>>,
#[serde(rename = "clID")]
pub client_id: StringValue<'static>,
#[serde(rename = "crID")]
pub creator_id: Option<StringValue<'static>>,
#[serde(rename = "crDate")]
pub created_at: Option<DateTime<Utc>>,
#[serde(rename = "exDate")]
pub expiring_at: Option<DateTime<Utc>>,
#[serde(rename = "upID")]
pub updater_id: Option<StringValue<'static>>,
#[serde(rename = "upDate")]
pub updated_at: Option<DateTime<Utc>>,
#[serde(rename = "trDate")]
pub transferred_at: Option<DateTime<Utc>>,
#[serde(rename = "authInfo")]
pub auth_info: Option<DomainAuthInfo<'static>>,
}
#[derive(Deserialize, Debug)]
pub struct DomainInfoResponse {
#[serde(rename = "infData")]
pub info_data: DomainInfoResponseData,
}
#[cfg(test)]
mod tests {
use super::DomainInfo;
use crate::response::ResultCode;
use crate::tests::{assert_serialized, response_from_file, CLTRID, SUCCESS_MSG, SVTRID};
use chrono::{TimeZone, Utc};
#[test]
fn command() {
let object = DomainInfo::new("eppdev.com", Some("2fooBAR"));
assert_serialized("request/domain/info.xml", &object);
}
#[test]
fn response() {
let object = response_from_file::<DomainInfo>("response/domain/info.xml");
let result = object.res_data().unwrap();
let auth_info = result.info_data.auth_info.as_ref().unwrap();
let ns_list = result.info_data.ns.as_ref().unwrap();
let ns = ns_list.host_obj.as_ref().unwrap();
let hosts = result.info_data.hosts.as_ref().unwrap();
let statuses = result.info_data.statuses.as_ref().unwrap();
let registrant = result.info_data.registrant.as_ref().unwrap();
let contacts = result.info_data.contacts.as_ref().unwrap();
assert_eq!(object.result.code, ResultCode::CommandCompletedSuccessfully);
assert_eq!(object.result.message, SUCCESS_MSG.into());
assert_eq!(result.info_data.name, "eppdev-1.com".into());
assert_eq!(result.info_data.roid, "125899511_DOMAIN_COM-VRSN".into());
assert_eq!(statuses[0].status, "ok".to_string());
assert_eq!(statuses[1].status, "clientTransferProhibited".to_string());
assert_eq!(*registrant, "eppdev-contact-2".into());
assert_eq!(contacts[0].id, "eppdev-contact-2".to_string());
assert_eq!(contacts[0].contact_type, "admin".to_string());
assert_eq!(contacts[1].id, "eppdev-contact-2".to_string());
assert_eq!(contacts[1].contact_type, "tech".to_string());
assert_eq!(contacts[2].id, "eppdev-contact-2".to_string());
assert_eq!(contacts[2].contact_type, "billing".to_string());
assert_eq!((*ns)[0], "ns1.eppdev-1.com".into());
assert_eq!((*ns)[1], "ns2.eppdev-1.com".into());
assert_eq!((*hosts)[0], "ns1.eppdev-1.com".into());
assert_eq!((*hosts)[1], "ns2.eppdev-1.com".into());
assert_eq!(result.info_data.client_id, "eppdev".into());
assert_eq!(
*result.info_data.creator_id.as_ref().unwrap(),
"SYSTEM".into()
);
assert_eq!(
*result.info_data.created_at.as_ref().unwrap(),
Utc.with_ymd_and_hms(2021, 7, 23, 15, 31, 20).unwrap()
);
assert_eq!(
*result.info_data.updater_id.as_ref().unwrap(),
"SYSTEM".into()
);
assert_eq!(
*result.info_data.updated_at.as_ref().unwrap(),
Utc.with_ymd_and_hms(2021, 7, 23, 15, 31, 21).unwrap()
);
assert_eq!(
*result.info_data.expiring_at.as_ref().unwrap(),
Utc.with_ymd_and_hms(2023, 7, 23, 15, 31, 20).unwrap()
);
assert_eq!(auth_info.password, "epP4uthd#v".into());
assert_eq!(object.tr_ids.client_tr_id.unwrap(), CLTRID.into());
assert_eq!(object.tr_ids.server_tr_id, SVTRID.into());
}
#[test]
fn response_alt() {
response_from_file::<DomainInfo>("response/domain/info_alt.xml");
}
}