use pgwire::api::results::{Response, Tag};
use pgwire::error::PgWireResult;
use crate::control::catalog_entry::CatalogEntry;
use crate::control::metadata_proposer::propose_catalog_entry;
use crate::control::security::audit::AuditEvent;
use crate::control::security::catalog::StoredTenant;
use crate::control::security::identity::{AuthenticatedIdentity, Role};
use crate::control::security::tenant::TenantQuota;
use crate::control::state::SharedState;
use crate::types::TenantId;
use super::super::super::types::sqlstate_error;
use super::super::parse_utils::strip_if_not_exists;
struct TenantOptions<'a> {
explicit_id: Option<u64>,
admin_override: Option<&'a str>,
}
fn parse_tenant_options<'a>(rest: &[&'a str]) -> PgWireResult<TenantOptions<'a>> {
let mut explicit_id = None;
let mut admin_override = None;
let mut i = 0;
while i < rest.len() {
if rest[i].eq_ignore_ascii_case("ID") && i + 1 < rest.len() {
let id: u64 = rest[i + 1]
.parse()
.map_err(|_| sqlstate_error("42601", "TENANT ID must be a numeric value"))?;
explicit_id = Some(id);
i += 2;
} else if rest[i].eq_ignore_ascii_case("WITH")
&& i + 2 < rest.len()
&& rest[i + 1].eq_ignore_ascii_case("ADMIN")
{
admin_override = Some(rest[i + 2]);
i += 3;
} else {
i += 1;
}
}
Ok(TenantOptions {
explicit_id,
admin_override,
})
}
pub fn create_tenant(
state: &SharedState,
identity: &AuthenticatedIdentity,
parts: &[&str],
) -> PgWireResult<Vec<Response>> {
if !identity.is_superuser {
return Err(sqlstate_error(
"42501",
"permission denied: only superuser can create tenants",
));
}
let (if_not_exists, parts) = strip_if_not_exists(parts, 2);
if parts.len() < 3 {
return Err(sqlstate_error(
"42601",
"syntax: CREATE TENANT [IF NOT EXISTS] <name> [ID <id>] [WITH ADMIN <user>]",
));
}
let name = parts[2];
let opts = parse_tenant_options(&parts[3..])?;
if if_not_exists
&& let Some(catalog) = state.credentials.catalog()
&& catalog
.find_tenant_by_name(name)
.map_err(|e| sqlstate_error("XX000", &format!("catalog read: {e}")))?
.is_some()
{
return Ok(vec![Response::Execution(Tag::new("CREATE TENANT"))]);
}
let tenant_id = {
let tenants = match state.tenants.lock() {
Ok(t) => t,
Err(p) => p.into_inner(),
};
match opts.explicit_id {
Some(id) => TenantId::new(id),
None => {
let count = tenants.tenant_count() as u64;
TenantId::new(count + 1)
}
}
};
let now = std::time::SystemTime::now()
.duration_since(std::time::UNIX_EPOCH)
.unwrap_or_default()
.as_secs();
let stored = StoredTenant {
tenant_id: tenant_id.as_u64(),
name: name.to_string(),
created_at: now,
is_active: true,
};
let entry = CatalogEntry::PutTenant(Box::new(stored.clone()));
let log_index = 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_tenant(&stored)
.map_err(|e| sqlstate_error("XX000", &format!("catalog write: {e}")))?;
}
let mut tenants = match state.tenants.lock() {
Ok(t) => t,
Err(p) => p.into_inner(),
};
if !tenants.has_quota(tenant_id) {
tenants.set_quota(tenant_id, TenantQuota::default());
}
}
let admin_name = opts
.admin_override
.map(str::to_string)
.unwrap_or_else(|| format!("{name}_admin"));
let admin_password = {
use sha2::{Digest, Sha256};
let mut hasher = Sha256::new();
hasher.update(name.as_bytes());
hasher.update(tenant_id.as_u64().to_le_bytes());
hasher.update(
std::time::SystemTime::now()
.duration_since(std::time::UNIX_EPOCH)
.unwrap_or_default()
.as_nanos()
.to_le_bytes(),
);
let hash = hasher.finalize();
let hex: String = hash.iter().take(12).map(|b| format!("{b:02x}")).collect();
format!("ndb_{hex}")
};
match state.credentials.create_user(
&admin_name,
&admin_password,
tenant_id,
vec![Role::TenantAdmin],
) {
Ok(_) => {
tracing::info!(tenant = %name, admin = %admin_name, "auto-created tenant admin");
}
Err(e) => {
tracing::warn!(tenant = %name, error = %e, "failed to auto-create tenant admin");
}
}
state.audit_record(
AuditEvent::TenantCreated,
Some(tenant_id),
&identity.username,
&format!("created tenant '{name}' (id {tenant_id}) with admin '{admin_name}'"),
);
Ok(vec![Response::Execution(Tag::new("CREATE TENANT"))])
}