use sqlx::PgPool;
use systemprompt_identifiers::Actor;
use crate::authz::types::DecisionTag;
pub const AUDIT_WRITE_FAILED_TOTAL: &str = "governance_audit_write_failed_total";
#[derive(Debug)]
pub struct GovernanceDecisionRecord<'a> {
pub id: &'a str,
pub actor: &'a Actor,
pub session_id: &'a str,
pub tool_name: &'a str,
pub agent_id: Option<&'a str>,
pub agent_scope: &'a str,
pub decision: DecisionTag,
pub policy: &'a str,
pub reason: &'a str,
pub evaluated_rules: &'a serde_json::Value,
pub plugin_id: Option<&'a str>,
pub act_chain: &'a [Actor],
}
#[derive(Debug, Clone)]
pub struct GovernanceDecisionRepository {
pool: std::sync::Arc<PgPool>,
}
impl GovernanceDecisionRepository {
pub const fn from_pool(pool: std::sync::Arc<PgPool>) -> Self {
Self { pool }
}
pub fn pool(&self) -> &PgPool {
&self.pool
}
pub async fn insert(&self, record: &GovernanceDecisionRecord<'_>) -> Result<(), sqlx::Error> {
insert_governance_decision(&self.pool, record).await
}
}
pub async fn insert_governance_decision(
pool: &PgPool,
record: &GovernanceDecisionRecord<'_>,
) -> Result<(), sqlx::Error> {
let actor_kind = record.actor.kind.tag();
let actor_id = record.actor.kind.actor_id(&record.actor.user_id);
let act_chain =
serde_json::to_value(record.act_chain).unwrap_or_else(|_| serde_json::json!([]));
let result = sqlx::query!(
"INSERT INTO governance_decisions (id, user_id, session_id, tool_name, agent_id, \
agent_scope, decision, policy, reason, evaluated_rules, plugin_id, actor_kind, actor_id, \
act_chain) VALUES ($1, $2, $3, $4, $5, $6, $7, $8, $9, $10, $11, $12, $13, $14)",
record.id,
record.actor.user_id.as_str(),
record.session_id,
record.tool_name,
record.agent_id,
record.agent_scope,
record.decision.as_str(),
record.policy,
record.reason,
record.evaluated_rules,
record.plugin_id,
actor_kind.as_str(),
actor_id,
act_chain,
)
.execute(pool)
.await;
if let Err(error) = &result {
tracing::error!(
error = %error,
actor_kind = actor_kind.as_str(),
actor_id,
policy = record.policy,
decision = record.decision.as_str(),
session_id = record.session_id,
"governance_decisions insert failed; audit row dropped"
);
metrics::counter!(
AUDIT_WRITE_FAILED_TOTAL,
"actor_kind" => actor_kind.as_str(),
"decision" => record.decision.as_str(),
"policy" => record.policy.to_owned(),
)
.increment(1);
}
result.map(|_| ())
}