1
0
Fork 0
forked from 2sys/phonograph
phonograph/interim-server/src/routes/relations_single/remove_field_handler.rs

99 lines
2.7 KiB
Rust
Raw Normal View History

2025-11-12 23:00:30 +00:00
use axum::{
debug_handler,
extract::{Path, State},
response::Response,
};
use interim_models::{field::Field, portal::Portal};
use interim_pgtypes::{escape_identifier, pg_class::PgClass};
use serde::Deserialize;
use sqlx::{postgres::types::Oid, query};
use uuid::Uuid;
use validator::Validate;
use crate::{
app::{App, AppDbConn},
errors::{AppError, bad_request},
extractors::ValidatedForm,
navigator::{Navigator, NavigatorPage},
user::CurrentUser,
workspace_pooler::{RoleAssignment, WorkspacePooler},
};
#[derive(Debug, Deserialize)]
pub(super) struct PathParams {
portal_id: Uuid,
rel_oid: u32,
workspace_id: Uuid,
}
#[derive(Debug, Deserialize, Validate)]
pub(super) struct FormBody {
field_id: Uuid,
/// Expects "true" for truthy, else falsy.
delete_data: String,
}
/// HTTP POST handler for removing an existing [`Field`].
///
/// This handler expects 3 path parameters with the structure described by
/// [`PathParams`].
#[debug_handler(state = App)]
pub(super) async fn post(
AppDbConn(mut app_db): AppDbConn,
State(mut pooler): State<WorkspacePooler>,
CurrentUser(user): CurrentUser,
navigator: Navigator,
Path(PathParams {
portal_id,
rel_oid,
workspace_id,
}): Path<PathParams>,
ValidatedForm(FormBody {
delete_data,
field_id,
}): ValidatedForm<FormBody>,
) -> Result<Response, AppError> {
// FIXME CSRF
// FIXME ensure workspace corresponds to rel/portal, and that user has
// permission to access/alter both as needed.
// Ensure field exists and belongs to portal.
let field = Field::belonging_to_portal(portal_id)
.with_id(field_id)
.fetch_one(&mut app_db)
.await?;
if delete_data == "true" && field.name.starts_with('_') {
return Err(bad_request!("cannot delete data for a system column"));
}
field.delete(&mut app_db).await?;
if delete_data == "true" {
let mut workspace_client = pooler
.acquire_for(workspace_id, RoleAssignment::User(user.id))
.await?;
let portal = Portal::with_id(portal_id).fetch_one(&mut app_db).await?;
let rel = PgClass::with_oid(portal.class_oid)
.fetch_one(&mut workspace_client)
.await?;
query(&format!(
"alter table {ident} drop column if exists {col_esc}",
ident = rel.get_identifier(),
col_esc = escape_identifier(&field.name),
))
.execute(workspace_client.get_conn())
.await?;
}
Ok(navigator
.portal_page()
.workspace_id(workspace_id)
.rel_oid(Oid(rel_oid))
.portal_id(portal_id)
.build()?
.redirect_to())
}