use std::fmt::{Display, Formatter};
use ct_codecs::{Base64UrlSafeNoPadding, Decoder};
use http::uri::Uri;
use crate::{
error::WebPushError,
http_ece::{ContentEncoding, HttpEce},
vapid::VapidSignature,
};
#[derive(Debug, Deserialize, Serialize, Clone, Eq, PartialEq, Ord, PartialOrd, Default, Hash)]
pub struct SubscriptionKeys {
pub p256dh: String,
pub auth: String,
}
#[derive(Debug, Deserialize, Serialize, Clone, Eq, PartialEq, Ord, PartialOrd, Default, Hash)]
pub struct SubscriptionInfo {
pub endpoint: String,
pub keys: SubscriptionKeys,
}
impl SubscriptionInfo {
pub fn new<S>(endpoint: S, p256dh: S, auth: S) -> SubscriptionInfo
where
S: Into<String>,
{
SubscriptionInfo {
endpoint: endpoint.into(),
keys: SubscriptionKeys {
p256dh: p256dh.into(),
auth: auth.into(),
},
}
}
}
#[derive(Debug, PartialEq)]
pub struct WebPushPayload {
pub content: Vec<u8>,
pub crypto_headers: Vec<(&'static str, String)>,
pub content_encoding: ContentEncoding,
}
#[derive(Debug, Deserialize, Serialize, Copy, Clone, Eq, PartialEq, Ord, PartialOrd, Default, Hash)]
#[serde(rename_all = "kebab-case")]
pub enum Urgency {
VeryLow,
Low,
#[default]
Normal,
High,
}
impl Display for Urgency {
fn fmt(&self, f: &mut Formatter<'_>) -> std::fmt::Result {
let str = match self {
Urgency::VeryLow => "very-low",
Urgency::Low => "low",
Urgency::Normal => "normal",
Urgency::High => "high",
};
f.write_str(str)
}
}
#[derive(Debug)]
pub struct WebPushMessage {
pub endpoint: Uri,
pub ttl: u32,
pub urgency: Option<Urgency>,
pub topic: Option<String>,
pub payload: Option<WebPushPayload>,
}
struct WebPushPayloadBuilder<'a> {
pub content: &'a [u8],
pub encoding: ContentEncoding,
}
pub struct WebPushMessageBuilder<'a> {
subscription_info: &'a SubscriptionInfo,
payload: Option<WebPushPayloadBuilder<'a>>,
ttl: u32,
urgency: Option<Urgency>,
topic: Option<String>,
vapid_signature: Option<VapidSignature>,
}
impl<'a> WebPushMessageBuilder<'a> {
pub fn new(subscription_info: &'a SubscriptionInfo) -> WebPushMessageBuilder<'a> {
WebPushMessageBuilder {
subscription_info,
ttl: 2_419_200,
urgency: None,
topic: None,
payload: None,
vapid_signature: None,
}
}
pub fn set_ttl(&mut self, ttl: u32) {
self.ttl = ttl;
}
pub fn set_urgency(&mut self, urgency: Urgency) {
self.urgency = Some(urgency);
}
pub fn set_topic(&mut self, topic: String) {
self.topic = Some(topic);
}
pub fn set_vapid_signature(&mut self, vapid_signature: VapidSignature) {
self.vapid_signature = Some(vapid_signature);
}
pub fn set_payload(&mut self, encoding: ContentEncoding, content: &'a [u8]) {
self.payload = Some(WebPushPayloadBuilder { content, encoding });
}
pub fn build(self) -> Result<WebPushMessage, WebPushError> {
let endpoint: Uri = self.subscription_info.endpoint.parse()?;
let topic: Option<String> = self
.topic
.map(|topic| {
if topic.len() > 32 {
Err(WebPushError::InvalidTopic)
} else if topic.chars().all(is_base64url_char) {
Ok(topic)
} else {
Err(WebPushError::InvalidTopic)
}
})
.transpose()?;
if let Some(payload) = self.payload {
let p256dh = Base64UrlSafeNoPadding::decode_to_vec(&self.subscription_info.keys.p256dh, None)
.map_err(|_| WebPushError::InvalidCryptoKeys)?;
let auth = Base64UrlSafeNoPadding::decode_to_vec(&self.subscription_info.keys.auth, None)
.map_err(|_| WebPushError::InvalidCryptoKeys)?;
let http_ece = HttpEce::new(payload.encoding, &p256dh, &auth, self.vapid_signature);
Ok(WebPushMessage {
endpoint,
ttl: self.ttl,
urgency: self.urgency,
topic,
payload: Some(http_ece.encrypt(payload.content)?),
})
} else {
Ok(WebPushMessage {
endpoint,
ttl: self.ttl,
urgency: self.urgency,
topic,
payload: None,
})
}
}
}
fn is_base64url_char(c: char) -> bool {
c.is_ascii_uppercase() || c.is_ascii_lowercase() || c.is_ascii_digit() || (c == '-' || c == '_')
}