#[cfg(server)]
use super::admin_auth::AdminAuthenticatedUser;
use crate::adapters::{AdminDatabase, AdminRecord, AdminSite};
use crate::types::MutationResponse;
#[cfg(server)]
use reinhardt_di::Depends;
#[cfg(server)]
use reinhardt_pages::server_fn::ServerFnRequest;
use reinhardt_pages::server_fn::{ServerFnError, server_fn};
#[cfg(server)]
use super::audit;
#[cfg(server)]
use super::error::{AdminAuth, MapServerFnError, ModelPermission};
#[cfg(server)]
use super::security::{require_csrf_token, sanitize_mutation_values};
#[cfg(server)]
use super::validation::validate_mutation_data;
#[server_fn]
pub async fn create_record(
model_name: String,
request: crate::types::MutationRequest,
#[inject] site: Depends<AdminSite>,
#[inject] db: Depends<AdminDatabase>,
#[inject] http_request: ServerFnRequest,
#[inject] AdminAuthenticatedUser(user): AdminAuthenticatedUser,
) -> Result<crate::types::MutationResponse, ServerFnError> {
require_csrf_token(&request.csrf_token, &http_request.inner().headers)?;
let auth = AdminAuth::from_request(&http_request);
let model_admin = site.get_model_admin(&model_name).map_server_fn_error()?;
auth.require_model_permission(model_admin.as_ref(), user.as_ref(), ModelPermission::Add)
.await?;
let table_name = model_admin.table_name();
let pk_field = model_admin.pk_field();
validate_mutation_data(&request.data, model_admin.as_ref(), false).map_server_fn_error()?;
let mut sanitized_data = request.data;
sanitize_mutation_values(&mut sanitized_data);
inject_auto_timestamps(&mut sanitized_data, table_name);
let user_id = auth.user_id().unwrap_or("unknown").to_string();
let result = db
.create::<AdminRecord>(table_name, Some(pk_field), sanitized_data.clone())
.await
.map_server_fn_error();
let success = result.is_ok();
audit::log_create(&user_id, &model_name, &sanitized_data, success);
let affected = result?;
Ok(MutationResponse {
success: true,
message: format!("{} created successfully", model_name),
affected: Some(affected),
data: None,
})
}
#[cfg(server)]
pub(crate) fn inject_auto_timestamps(
data: &mut std::collections::HashMap<String, serde_json::Value>,
table_name: &str,
) {
use crate::server::type_inference::find_model_by_table_name;
let Some(model) = find_model_by_table_name(table_name) else {
return;
};
let now = chrono::Utc::now();
for (field_name, meta) in &model.fields {
let is_auto_now = meta
.params
.get("auto_now")
.is_some_and(|v| v == "true" || v == "True");
let is_auto_now_add = meta
.params
.get("auto_now_add")
.is_some_and(|v| v == "true" || v == "True");
if is_auto_now || is_auto_now_add {
let value = match &meta.field_type {
reinhardt_db::migrations::FieldType::Date => {
serde_json::Value::String(now.format("%Y-%m-%d").to_string())
}
reinhardt_db::migrations::FieldType::Time => {
serde_json::Value::String(now.format("%H:%M:%S").to_string())
}
_ => {
serde_json::Value::String(now.format("%Y-%m-%dT%H:%M:%S%.6fZ").to_string())
}
};
data.insert(field_name.clone(), value);
}
}
}
#[cfg(server)]
pub(crate) fn inject_auto_now_timestamps(
data: &mut std::collections::HashMap<String, serde_json::Value>,
table_name: &str,
) {
use crate::server::type_inference::find_model_by_table_name;
let Some(model) = find_model_by_table_name(table_name) else {
return;
};
let now = chrono::Utc::now();
for (field_name, meta) in &model.fields {
let is_auto_now = meta
.params
.get("auto_now")
.is_some_and(|v| v == "true" || v == "True");
if is_auto_now {
let value = match &meta.field_type {
reinhardt_db::migrations::FieldType::Date => {
serde_json::Value::String(now.format("%Y-%m-%d").to_string())
}
reinhardt_db::migrations::FieldType::Time => {
serde_json::Value::String(now.format("%H:%M:%S").to_string())
}
_ => serde_json::Value::String(now.format("%Y-%m-%dT%H:%M:%S%.6fZ").to_string()),
};
data.insert(field_name.clone(), value);
}
}
}