use std::sync::{Arc, OnceLock};
use async_trait::async_trait;
use authz_resolver_sdk::{AuthZResolverPluginClient, AuthZResolverPluginSpecV1};
use modkit::Module;
use modkit::client_hub::ClientScope;
use modkit::context::ModuleCtx;
use modkit::gts::BaseModkitPluginV1;
use tenant_resolver_sdk::TenantResolverClient;
use tracing::info;
use types_registry_sdk::{RegisterResult, TypesRegistryClient};
use crate::config::TrAuthZPluginConfig;
use crate::domain::Service;
#[modkit::module(
name = "tr-authz-plugin",
deps = ["types-registry", "tenant-resolver"]
)]
pub struct TrAuthZPlugin {
service: OnceLock<Arc<Service>>,
}
impl Default for TrAuthZPlugin {
fn default() -> Self {
Self {
service: OnceLock::new(),
}
}
}
#[async_trait]
impl Module for TrAuthZPlugin {
async fn init(&self, ctx: &ModuleCtx) -> anyhow::Result<()> {
let cfg: TrAuthZPluginConfig = ctx.config_or_default()?;
info!(
vendor = %cfg.vendor,
priority = cfg.priority,
"Loaded TR AuthZ plugin configuration"
);
let instance_id = AuthZResolverPluginSpecV1::gts_make_instance_id(
"hyperspot.builtin.tr_authz_resolver.plugin.v1",
);
let tr: Arc<dyn TenantResolverClient> =
ctx.client_hub().get::<dyn TenantResolverClient>()?;
let registry = ctx.client_hub().get::<dyn TypesRegistryClient>()?;
let instance = BaseModkitPluginV1::<AuthZResolverPluginSpecV1> {
id: instance_id.clone(),
vendor: cfg.vendor.clone(),
priority: cfg.priority,
properties: AuthZResolverPluginSpecV1,
};
let instance_json = serde_json::to_value(&instance)?;
let results = registry.register(vec![instance_json]).await?;
RegisterResult::ensure_all_ok(&results)?;
let service = Arc::new(Service::new(tr));
self.service
.set(service.clone())
.map_err(|_| anyhow::anyhow!("{} module already initialized", Self::MODULE_NAME))?;
let api: Arc<dyn AuthZResolverPluginClient> = service;
ctx.client_hub()
.register_scoped::<dyn AuthZResolverPluginClient>(
ClientScope::gts_id(&instance_id),
api,
);
info!(instance_id = %instance_id);
Ok(())
}
}