Some checks failed
BotServer CI / build (push) Failing after 7m5s
175 lines
6.1 KiB
Rust
175 lines
6.1 KiB
Rust
pub mod campaigns;
|
|
pub mod lists;
|
|
pub mod templates;
|
|
pub mod triggers;
|
|
pub mod email;
|
|
pub mod whatsapp;
|
|
pub mod metrics;
|
|
pub mod ai;
|
|
|
|
use axum::{
|
|
body::Body,
|
|
extract::{Path, State},
|
|
http::{header, StatusCode},
|
|
response::Response,
|
|
routing::{get, post},
|
|
Router,
|
|
};
|
|
use chrono::Utc;
|
|
use diesel::prelude::*;
|
|
use std::sync::Arc;
|
|
|
|
use crate::core::shared::schema::email_tracking;
|
|
use crate::core::shared::state::AppState;
|
|
|
|
fn base64_decode(input: &str) -> Option<Vec<u8>> {
|
|
let chars: Vec<u8> = input
|
|
.chars()
|
|
.filter_map(|c| {
|
|
if c.is_ascii_alphanumeric() || c == '+' || c == '/' || c == '=' {
|
|
Some(c as u8)
|
|
} else {
|
|
None
|
|
}
|
|
})
|
|
.collect();
|
|
|
|
const DECODE_TABLE: [i8; 128] = [
|
|
-1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1,
|
|
-1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, 62, -1, -1,
|
|
-1, 63, 52, 53, 54, 55, 56, 57, 58, 59, 60, 61, -1, -1, -1, -1, -1, -1, -1, 0, 1, 2, 3,
|
|
4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 17, 18, 19, 20, 21, 22, 23, 24, 25, -1,
|
|
-1, -1, -1, -1, -1, 26, 27, 28, 29, 30, 31, 32, 33, 34, 35, 36, 37, 38, 39, 40, 41,
|
|
42, 43, 44, 45, 46, 47, 48, 49, 50, 51, -1, -1, -1, -1, -1, -1, -1,
|
|
];
|
|
|
|
let mut output = Vec::with_capacity(chars.len() * 3 / 4);
|
|
let mut buf = [0u8; 4];
|
|
let mut count = 0;
|
|
|
|
for &byte in chars.iter() {
|
|
if byte >= 128 {
|
|
return None;
|
|
}
|
|
let val = DECODE_TABLE[byte as usize];
|
|
if val < 0 {
|
|
continue;
|
|
}
|
|
buf[count] = val as u8;
|
|
count += 1;
|
|
if count == 4 {
|
|
output.push((buf[0] << 2) | (buf[1] >> 4));
|
|
output.push((buf[1] << 4) | (buf[2] >> 2));
|
|
output.push((buf[2] << 6) | buf[3]);
|
|
count = 0;
|
|
}
|
|
}
|
|
|
|
if count >= 2 {
|
|
output.push((buf[0] << 2) | (buf[1] >> 4));
|
|
if count > 2 {
|
|
output.push((buf[1] << 4) | (buf[2] >> 2));
|
|
}
|
|
}
|
|
|
|
Some(output)
|
|
}
|
|
|
|
pub async fn track_email_open_pixel(
|
|
State(state): State<Arc<AppState>>,
|
|
Path(token): Path<String>,
|
|
) -> Response {
|
|
let pixel = base64_decode("iVBORw0KGgoAAAANSUhEUgAAAAEAAAABCAYAAAAfFcSJAAAADUlEQVR42mNk+M9QDwADhgGAWjR9awAAAABJRU5ErkJggg==")
|
|
.unwrap_or_else(|| vec![0u8; 1]);
|
|
|
|
if let Ok(mut conn) = state.conn.get() {
|
|
if let Ok(token_uuid) = uuid::Uuid::parse_str(&token) {
|
|
let now = Utc::now();
|
|
let _ = diesel::update(
|
|
email_tracking::table.filter(email_tracking::open_token.eq(token_uuid)),
|
|
)
|
|
.set((
|
|
email_tracking::opened.eq(true),
|
|
email_tracking::opened_at.eq(Some(now)),
|
|
))
|
|
.execute(&mut conn);
|
|
|
|
log::info!("Email open tracked via pixel: token={}", token);
|
|
}
|
|
}
|
|
|
|
let mut response = Response::new(Body::from(pixel));
|
|
response.headers_mut().insert(
|
|
header::CONTENT_TYPE,
|
|
"image/png".parse().unwrap(),
|
|
);
|
|
response.headers_mut().insert(
|
|
header::CACHE_CONTROL,
|
|
"no-cache, no-store, must-revalidate".parse().unwrap(),
|
|
);
|
|
response
|
|
}
|
|
|
|
pub async fn track_email_click(
|
|
State(state): State<Arc<AppState>>,
|
|
Path((id, destination)): Path<(String, String)>,
|
|
) -> Response {
|
|
if let Ok(mut conn) = state.conn.get() {
|
|
if let Ok(tracking_id) = uuid::Uuid::parse_str(&id) {
|
|
let now = Utc::now();
|
|
let _ = diesel::update(
|
|
email_tracking::table.filter(email_tracking::id.eq(tracking_id)),
|
|
)
|
|
.set((
|
|
email_tracking::clicked.eq(true),
|
|
email_tracking::clicked_at.eq(Some(now)),
|
|
))
|
|
.execute(&mut conn);
|
|
|
|
log::info!("Email click tracked: tracking_id={}", tracking_id);
|
|
}
|
|
}
|
|
|
|
let destination = if destination.starts_with("http") {
|
|
destination
|
|
} else {
|
|
format!("/{}", destination)
|
|
};
|
|
|
|
let mut response = Response::new(Body::empty());
|
|
*response.status_mut() = StatusCode::FOUND;
|
|
response.headers_mut().insert(
|
|
header::LOCATION,
|
|
destination.parse().unwrap(),
|
|
);
|
|
response
|
|
}
|
|
|
|
pub fn configure_marketing_routes() -> Router<Arc<AppState>> {
|
|
Router::new()
|
|
.route("/api/crm/campaigns", get(campaigns::list_campaigns).post(campaigns::create_campaign))
|
|
.route("/api/crm/campaigns/:id", get(campaigns::get_campaign).put(campaigns::update_campaign).delete(campaigns::delete_campaign))
|
|
.route("/api/crm/campaigns/:id/send", post(campaigns::send_campaign))
|
|
|
|
.route("/api/crm/lists", get(lists::list_lists).post(lists::create_list))
|
|
.route("/api/crm/lists/:id", get(lists::get_list).put(lists::update_list).delete(lists::delete_list))
|
|
.route("/api/crm/lists/:id/refresh", post(lists::refresh_marketing_list))
|
|
|
|
.route("/api/crm/templates", get(templates::list_templates).post(templates::create_template))
|
|
.route("/api/crm/templates/:id", get(templates::get_template).put(templates::update_template).delete(templates::delete_template))
|
|
|
|
.route("/api/crm/email/track/open", post(triggers::track_email_open))
|
|
.route("/api/marketing/track/open/:token", get(track_email_open_pixel))
|
|
.route("/api/marketing/track/click/:id/*destination", get(track_email_click))
|
|
.route("/api/crm/email/send", post(email::send_email_api))
|
|
|
|
.route("/api/crm/whatsapp/send", post(whatsapp::send_whatsapp_api))
|
|
|
|
.route("/api/crm/metrics/campaign/:id", get(metrics::get_campaign_metrics_api))
|
|
.route("/api/crm/metrics/campaign/:id/channels", get(metrics::get_campaign_channel_breakdown_api))
|
|
.route("/api/crm/metrics/campaign/:id/timeseries/:interval", get(metrics::get_campaign_timeseries_api))
|
|
.route("/api/crm/metrics/aggregate", get(metrics::get_aggregate_metrics_api))
|
|
|
|
.route("/api/crm/ai/generate", post(ai::generate_content_api))
|
|
.route("/api/crm/ai/personalize", post(ai::personalize_api))
|
|
}
|