230 lines
8 KiB
Rust
230 lines
8 KiB
Rust
use axum::{
|
|
extract::{Path, State},
|
|
http::StatusCode,
|
|
Json,
|
|
};
|
|
use chrono::{DateTime, Utc};
|
|
use diesel::prelude::*;
|
|
use serde::{Deserialize, Serialize};
|
|
use std::sync::Arc;
|
|
use uuid::Uuid;
|
|
|
|
use crate::core::shared::schema::marketing_templates;
|
|
use crate::core::shared::state::AppState;
|
|
use crate::core::bot::get_default_bot;
|
|
|
|
#[derive(Debug, Clone, Serialize, Deserialize, Queryable, Insertable, AsChangeset)]
|
|
#[diesel(table_name = marketing_templates)]
|
|
pub struct MarketingTemplate {
|
|
pub id: Uuid,
|
|
pub org_id: Uuid,
|
|
pub bot_id: Uuid,
|
|
pub name: String,
|
|
pub channel: String,
|
|
pub subject: Option<String>,
|
|
pub body: Option<String>,
|
|
pub media_url: Option<String>,
|
|
pub ai_prompt: Option<String>,
|
|
pub variables: serde_json::Value,
|
|
pub approved: Option<bool>,
|
|
pub meta_template_id: Option<String>,
|
|
pub created_at: DateTime<Utc>,
|
|
pub updated_at: Option<DateTime<Utc>>,
|
|
}
|
|
|
|
#[derive(Debug, Deserialize)]
|
|
pub struct CreateTemplateRequest {
|
|
pub name: String,
|
|
pub channel: String,
|
|
pub subject: Option<String>,
|
|
pub body: Option<String>,
|
|
pub media_url: Option<String>,
|
|
pub ai_prompt: Option<String>,
|
|
pub variables: Option<serde_json::Value>,
|
|
}
|
|
|
|
#[derive(Debug, Deserialize)]
|
|
pub struct UpdateTemplateRequest {
|
|
pub name: Option<String>,
|
|
pub channel: Option<String>,
|
|
pub subject: Option<String>,
|
|
pub body: Option<String>,
|
|
pub media_url: Option<String>,
|
|
pub ai_prompt: Option<String>,
|
|
pub variables: Option<serde_json::Value>,
|
|
pub approved: Option<bool>,
|
|
}
|
|
|
|
fn get_bot_context(state: &AppState) -> (Uuid, Uuid) {
|
|
use diesel::prelude::*;
|
|
use crate::core::shared::schema::bots;
|
|
|
|
let Ok(mut conn) = state.conn.get() else {
|
|
return (Uuid::nil(), Uuid::nil());
|
|
};
|
|
let (bot_id, _bot_name) = get_default_bot(&mut conn);
|
|
|
|
let org_id = bots::table
|
|
.filter(bots::id.eq(bot_id))
|
|
.select(bots::org_id)
|
|
.first::<Option<Uuid>>(&mut conn)
|
|
.unwrap_or(None)
|
|
.unwrap_or(Uuid::nil());
|
|
|
|
(org_id, bot_id)
|
|
}
|
|
|
|
pub async fn list_templates(
|
|
State(state): State<Arc<AppState>>,
|
|
) -> Result<Json<Vec<MarketingTemplate>>, (StatusCode, String)> {
|
|
let mut conn = state.conn.get().map_err(|e| {
|
|
(StatusCode::INTERNAL_SERVER_ERROR, format!("DB error: {e}"))
|
|
})?;
|
|
|
|
let (org_id, bot_id) = get_bot_context(&state);
|
|
|
|
let templates: Vec<MarketingTemplate> = marketing_templates::table
|
|
.filter(marketing_templates::org_id.eq(org_id))
|
|
.filter(marketing_templates::bot_id.eq(bot_id))
|
|
.order(marketing_templates::created_at.desc())
|
|
.load(&mut conn)
|
|
.map_err(|e| (StatusCode::INTERNAL_SERVER_ERROR, format!("Query error: {e}")))?;
|
|
|
|
Ok(Json(templates))
|
|
}
|
|
|
|
pub async fn get_template(
|
|
State(state): State<Arc<AppState>>,
|
|
Path(id): Path<Uuid>,
|
|
) -> Result<Json<MarketingTemplate>, (StatusCode, String)> {
|
|
let mut conn = state.conn.get().map_err(|e| {
|
|
(StatusCode::INTERNAL_SERVER_ERROR, format!("DB error: {e}"))
|
|
})?;
|
|
|
|
let template: MarketingTemplate = marketing_templates::table
|
|
.filter(marketing_templates::id.eq(id))
|
|
.first(&mut conn)
|
|
.map_err(|_| (StatusCode::NOT_FOUND, "Template not found".to_string()))?;
|
|
|
|
Ok(Json(template))
|
|
}
|
|
|
|
pub async fn create_template(
|
|
State(state): State<Arc<AppState>>,
|
|
Json(req): Json<CreateTemplateRequest>,
|
|
) -> Result<Json<MarketingTemplate>, (StatusCode, String)> {
|
|
let mut conn = state.conn.get().map_err(|e| {
|
|
(StatusCode::INTERNAL_SERVER_ERROR, format!("DB error: {e}"))
|
|
})?;
|
|
|
|
let (org_id, bot_id) = get_bot_context(&state);
|
|
let id = Uuid::new_v4();
|
|
let now = Utc::now();
|
|
|
|
let template = MarketingTemplate {
|
|
id,
|
|
org_id,
|
|
bot_id,
|
|
name: req.name,
|
|
channel: req.channel,
|
|
subject: req.subject,
|
|
body: req.body,
|
|
media_url: req.media_url,
|
|
ai_prompt: req.ai_prompt,
|
|
variables: req.variables.unwrap_or(serde_json::json!({})),
|
|
approved: Some(false),
|
|
meta_template_id: None,
|
|
created_at: now,
|
|
updated_at: Some(now),
|
|
};
|
|
|
|
diesel::insert_into(marketing_templates::table)
|
|
.values(&template)
|
|
.execute(&mut conn)
|
|
.map_err(|e| (StatusCode::INTERNAL_SERVER_ERROR, format!("Insert error: {e}")))?;
|
|
|
|
Ok(Json(template))
|
|
}
|
|
|
|
pub async fn update_template(
|
|
State(state): State<Arc<AppState>>,
|
|
Path(id): Path<Uuid>,
|
|
Json(req): Json<UpdateTemplateRequest>,
|
|
) -> Result<Json<MarketingTemplate>, (StatusCode, String)> {
|
|
let mut conn = state.conn.get().map_err(|e| {
|
|
(StatusCode::INTERNAL_SERVER_ERROR, format!("DB error: {e}"))
|
|
})?;
|
|
|
|
let now = Utc::now();
|
|
|
|
if let Some(name) = req.name {
|
|
diesel::update(marketing_templates::table.filter(marketing_templates::id.eq(id)))
|
|
.set(marketing_templates::name.eq(name))
|
|
.execute(&mut conn)
|
|
.map_err(|e| (StatusCode::INTERNAL_SERVER_ERROR, format!("Update error: {e}")))?;
|
|
}
|
|
if let Some(channel) = req.channel {
|
|
diesel::update(marketing_templates::table.filter(marketing_templates::id.eq(id)))
|
|
.set(marketing_templates::channel.eq(channel))
|
|
.execute(&mut conn)
|
|
.map_err(|e| (StatusCode::INTERNAL_SERVER_ERROR, format!("Update error: {e}")))?;
|
|
}
|
|
if let Some(subject) = req.subject {
|
|
diesel::update(marketing_templates::table.filter(marketing_templates::id.eq(id)))
|
|
.set(marketing_templates::subject.eq(Some(subject)))
|
|
.execute(&mut conn)
|
|
.map_err(|e| (StatusCode::INTERNAL_SERVER_ERROR, format!("Update error: {e}")))?;
|
|
}
|
|
if let Some(body) = req.body {
|
|
diesel::update(marketing_templates::table.filter(marketing_templates::id.eq(id)))
|
|
.set(marketing_templates::body.eq(Some(body)))
|
|
.execute(&mut conn)
|
|
.map_err(|e| (StatusCode::INTERNAL_SERVER_ERROR, format!("Update error: {e}")))?;
|
|
}
|
|
if let Some(media_url) = req.media_url {
|
|
diesel::update(marketing_templates::table.filter(marketing_templates::id.eq(id)))
|
|
.set(marketing_templates::media_url.eq(Some(media_url)))
|
|
.execute(&mut conn)
|
|
.map_err(|e| (StatusCode::INTERNAL_SERVER_ERROR, format!("Update error: {e}")))?;
|
|
}
|
|
if let Some(ai_prompt) = req.ai_prompt {
|
|
diesel::update(marketing_templates::table.filter(marketing_templates::id.eq(id)))
|
|
.set(marketing_templates::ai_prompt.eq(Some(ai_prompt)))
|
|
.execute(&mut conn)
|
|
.map_err(|e| (StatusCode::INTERNAL_SERVER_ERROR, format!("Update error: {e}")))?;
|
|
}
|
|
if let Some(variables) = req.variables {
|
|
diesel::update(marketing_templates::table.filter(marketing_templates::id.eq(id)))
|
|
.set(marketing_templates::variables.eq(variables))
|
|
.execute(&mut conn)
|
|
.map_err(|e| (StatusCode::INTERNAL_SERVER_ERROR, format!("Update error: {e}")))?;
|
|
}
|
|
if let Some(approved) = req.approved {
|
|
diesel::update(marketing_templates::table.filter(marketing_templates::id.eq(id)))
|
|
.set(marketing_templates::approved.eq(Some(approved)))
|
|
.execute(&mut conn)
|
|
.map_err(|e| (StatusCode::INTERNAL_SERVER_ERROR, format!("Update error: {e}")))?;
|
|
}
|
|
|
|
diesel::update(marketing_templates::table.filter(marketing_templates::id.eq(id)))
|
|
.set(marketing_templates::updated_at.eq(Some(now)))
|
|
.execute(&mut conn)
|
|
.map_err(|e| (StatusCode::INTERNAL_SERVER_ERROR, format!("Update error: {e}")))?;
|
|
|
|
get_template(State(state), Path(id)).await
|
|
}
|
|
|
|
pub async fn delete_template(
|
|
State(state): State<Arc<AppState>>,
|
|
Path(id): Path<Uuid>,
|
|
) -> Result<Json<serde_json::Value>, (StatusCode, String)> {
|
|
let mut conn = state.conn.get().map_err(|e| {
|
|
(StatusCode::INTERNAL_SERVER_ERROR, format!("DB error: {e}"))
|
|
})?;
|
|
|
|
diesel::delete(marketing_templates::table.filter(marketing_templates::id.eq(id)))
|
|
.execute(&mut conn)
|
|
.map_err(|e| (StatusCode::INTERNAL_SERVER_ERROR, format!("Delete error: {e}")))?;
|
|
|
|
Ok(Json(serde_json::json!({ "status": "deleted" })))
|
|
}
|