use crate::Endpoint;
use serde::{Deserialize, Serialize};
use std::{
borrow::Cow,
collections::{BTreeMap, HashMap},
};
use typed_builder::TypedBuilder;
use super::send_email::{Attachment, Header, SendEmailResponse, TrackLink};
#[derive(Debug, Clone, PartialEq, Serialize, Deserialize, Default)]
pub struct TemplateModel {
#[serde(flatten)]
model: HashMap<String, serde_json::Value>,
}
impl TemplateModel {
pub fn insert<K, V>(&mut self, key: K, value: V)
where
K: Into<String>,
V: Serialize,
{
self.model
.insert(key.into(), serde_json::to_value(value).unwrap());
}
pub fn remove<K>(&mut self, key: K)
where
K: Into<String>,
{
self.model.remove(&key.into());
}
pub fn into_inner(self) -> HashMap<String, serde_json::Value> {
self.model
}
}
impl<K: Into<String>, V: Serialize> From<HashMap<K, V>> for TemplateModel {
fn from(model: HashMap<K, V>) -> Self {
Self {
model: model
.into_iter()
.map(|(k, v)| (k.into(), serde_json::to_value(v).unwrap()))
.collect(),
}
}
}
impl<K: Into<String>, V: Serialize> From<BTreeMap<K, V>> for TemplateModel {
fn from(model: BTreeMap<K, V>) -> Self {
Self {
model: model
.into_iter()
.map(|(k, v)| (k.into(), serde_json::to_value(v).unwrap()))
.collect(),
}
}
}
#[cfg(feature = "indexmap")]
impl<K: Into<String>, V: Serialize> From<indexmap::IndexMap<K, V>> for TemplateModel {
fn from(model: indexmap::IndexMap<K, V>) -> Self {
Self {
model: model
.into_iter()
.map(|(k, v)| (k.into(), serde_json::to_value(v).unwrap()))
.collect(),
}
}
}
#[derive(Debug, Clone, PartialEq, Serialize, Deserialize, Default)]
#[serde(rename_all = "PascalCase")]
#[derive(TypedBuilder)]
pub struct SendEmailWithTemplateRequest {
#[builder(setter(into))]
pub from: String,
#[builder(setter(into))]
pub to: String,
#[serde(skip_serializing_if = "Option::is_none")]
#[builder(default, setter(into, strip_option))]
pub template_id: Option<i64>,
#[serde(skip_serializing_if = "Option::is_none")]
#[builder(default, setter(into, strip_option))]
pub template_alias: Option<String>,
#[builder(default, setter(into))]
pub template_model: TemplateModel,
#[serde(skip_serializing_if = "Option::is_none")]
#[builder(default, setter(into, strip_option))]
pub cc: Option<String>,
#[serde(skip_serializing_if = "Option::is_none")]
#[builder(default, setter(into, strip_option))]
pub bcc: Option<String>,
#[serde(skip_serializing_if = "Option::is_none")]
#[builder(default, setter(into, strip_option))]
pub tag: Option<String>,
#[serde(skip_serializing_if = "Option::is_none")]
#[builder(default, setter(into, strip_option))]
pub reply_to: Option<String>,
#[serde(skip_serializing_if = "Option::is_none")]
#[builder(default, setter(into, strip_option))]
pub headers: Option<Vec<Header>>,
#[serde(skip_serializing_if = "Option::is_none")]
#[builder(default, setter(into, strip_option))]
pub track_opens: Option<bool>,
#[serde(skip_serializing_if = "Option::is_none")]
#[builder(default, setter(into, strip_option))]
pub track_links: Option<TrackLink>,
#[serde(skip_serializing_if = "Option::is_none")]
#[builder(default, setter(into, strip_option))]
pub attachments: Option<Vec<Attachment>>,
#[serde(skip_serializing_if = "Option::is_none")]
#[builder(default, setter(into, strip_option))]
pub metadata: Option<HashMap<String, String>>,
#[serde(skip_serializing_if = "Option::is_none")]
#[builder(default, setter(into, strip_option))]
pub message_stream: Option<String>,
}
impl Endpoint for SendEmailWithTemplateRequest {
type Request = SendEmailWithTemplateRequest;
type Response = SendEmailResponse;
fn endpoint(&self) -> Cow<'static, str> {
"/email/withTemplate".into()
}
fn body(&self) -> &Self::Request {
self
}
}
#[cfg(test)]
mod tests {
use std::collections::HashMap;
use httptest::matchers::request;
use httptest::{Expectation, Server, responders::*};
use serde_json::json;
use super::{SendEmailWithTemplateRequest, TemplateModel};
use crate::Query;
use crate::reqwest::PostmarkClient;
#[tokio::test]
pub async fn send_email_test() {
let server = Server::run();
server.expect(
Expectation::matching(request::method_path("POST", "/email/withTemplate"))
.respond_with(json_encoded(json!({
"To": "receiver@example.com",
"SubmittedAt": "2014-02-17T07:25:01.4178645-05:00",
"MessageID": "0a129aee-e1cd-480d-b08d-4f48548ff48d",
"ErrorCode": 0_i64,
"Message": "OK"
}))),
);
let client = PostmarkClient::builder()
.base_url(server.url("/").to_string())
.build();
let mut model = TemplateModel::default();
model.insert("name", "Ferris");
model.insert("favorite_food", ["algae", "seaweed", "shrimp", "cpp"]);
let req = SendEmailWithTemplateRequest::builder()
.from("pa@example.com")
.to("mathieu@example.com")
.template_alias("my_template".to_string())
.template_model(model)
.build();
req.execute(&client)
.await
.expect("Should get a response and be able to json decode it");
}
#[tokio::test]
pub async fn send_email_test_should_not_error_on_postmark_error() {
let server = Server::run();
server.expect(
Expectation::matching(request::method_path("POST", "/email/withTemplate")).respond_with(
json_encoded(json!({
"ErrorCode": 406_i64,
"Message": "You tried to send to a recipient that has been marked as inactive. Found inactive addresses: example@example.com. Inactive recipients are ones that have generated a hard bounce, a spam complaint, or a manual suppression. " })),
),
);
let client = PostmarkClient::builder()
.base_url(server.url("/").to_string())
.build();
let mut nested_map = HashMap::new();
nested_map.insert("code".to_string(), 123_i64);
let mut template = HashMap::new();
template.insert("nested".to_string(), nested_map);
let req = SendEmailWithTemplateRequest::builder()
.from("pa@example.com")
.to("mathieu@example.com")
.template_id(123456)
.template_model(TemplateModel::from(template))
.build();
req.execute(&client)
.await
.expect("Should get a response and be able to json decode it");
}
}