use nodedb_sql::ddl_ast::TenantSelector;
use pgwire::api::results::{Response, Tag};
use pgwire::error::PgWireResult;
use crate::control::security::audit::AuditEvent;
use crate::control::security::identity::{AuthenticatedIdentity, Role};
use crate::control::server::pgwire::types::{parse_role, require_tenant_admin, sqlstate_error};
use crate::control::state::SharedState;
use crate::types::TenantId;
fn resolve_tenant_selector(
state: &SharedState,
selector: &TenantSelector,
) -> PgWireResult<TenantId> {
match selector {
TenantSelector::Id(id) => Ok(TenantId::new(*id)),
TenantSelector::Name(name) => {
let catalog =
state.credentials.catalog().as_ref().ok_or_else(|| {
sqlstate_error("42704", &format!("tenant '{name}' not found"))
})?;
let stored = catalog
.find_tenant_by_name(name)
.map_err(|e| sqlstate_error("XX000", &format!("catalog read: {e}")))?
.ok_or_else(|| sqlstate_error("42704", &format!("tenant '{name}' not found")))?;
Ok(TenantId::new(stored.tenant_id))
}
}
}
pub fn create_user(
state: &SharedState,
identity: &AuthenticatedIdentity,
username: &str,
password: &str,
role_name: Option<&str>,
tenant: Option<&TenantSelector>,
if_not_exists: bool,
) -> PgWireResult<Vec<Response>> {
require_tenant_admin(identity, "create users")?;
if username.is_empty() {
return Err(sqlstate_error(
"42601",
"syntax: CREATE USER <name> WITH PASSWORD '<password>' [ROLE <role>] [TENANT <id>]",
));
}
if if_not_exists && state.credentials.get_user(username).is_some() {
return Ok(vec![Response::Execution(Tag::new("CREATE USER"))]);
}
if password.is_empty() {
return Err(sqlstate_error(
"42601",
"password must be a single-quoted string",
));
}
let role = role_name.map(parse_role).unwrap_or(Role::ReadWrite);
let tenant_id = if let Some(selector) = tenant {
if !identity.is_superuser {
return Err(sqlstate_error("42501", "only superuser can assign tenants"));
}
resolve_tenant_selector(state, selector)?
} else {
identity.tenant_id
};
let stored = state
.credentials
.prepare_user(username, password, tenant_id, vec![role])
.map_err(|e| sqlstate_error("42710", &e.to_string()))?;
let entry = crate::control::catalog_entry::CatalogEntry::PutUser(Box::new(stored.clone()));
let log_index = crate::control::metadata_proposer::propose_catalog_entry(state, &entry)
.map_err(|e| sqlstate_error("XX000", &format!("metadata propose: {e}")))?;
if log_index == 0 {
if let Some(catalog) = state.credentials.catalog() {
catalog
.put_user(&stored)
.map_err(|e| sqlstate_error("XX000", &format!("catalog write: {e}")))?;
}
state.credentials.install_replicated_user(&stored, None);
} else {
if state.credentials.get_user(username).is_none() {
return Err(sqlstate_error(
"40001",
"transient: metadata entry truncated by leader change, retry",
));
}
}
state.audit_record(
AuditEvent::PrivilegeChange,
Some(tenant_id),
&identity.username,
&format!("created user '{username}' in tenant {tenant_id}"),
);
Ok(vec![Response::Execution(Tag::new("CREATE USER"))])
}