use crate::prelude::ResourceGraphHelper;
use cloud_terrastodon_azure_types::prelude::RoleDefinition;
use cloud_terrastodon_command::CacheKey;
use cloud_terrastodon_command::CacheableCommand;
use cloud_terrastodon_command::async_trait;
use eyre::Result;
use std::path::PathBuf;
use tracing::debug;
#[must_use = "This is a future request, you must .await it"]
pub struct RoleDefinitionListRequest;
pub fn fetch_all_role_definitions() -> RoleDefinitionListRequest {
RoleDefinitionListRequest
}
#[async_trait]
impl CacheableCommand for RoleDefinitionListRequest {
type Output = Vec<RoleDefinition>;
fn cache_key(&self) -> CacheKey {
CacheKey::new(PathBuf::from_iter([
"az",
"resource_graph",
"role-definitions",
]))
}
async fn run(self) -> Result<Self::Output> {
debug!("Fetching role definitions");
let role_definitions = ResourceGraphHelper::new(
r#"authorizationresources
| where type =~ "microsoft.authorization/roledefinitions"
| project id, properties
| extend
assignable_scopes = properties.assignableScopes,
description = properties.description,
permissions = properties.permissions,
display_name = properties.roleName,
['kind'] = properties.type
| project-away properties"#,
Some(self.cache_key()),
)
.collect_all()
.await?;
debug!("Found {} role definitions", role_definitions.len());
Ok(role_definitions)
}
}
cloud_terrastodon_command::impl_cacheable_into_future!(RoleDefinitionListRequest);
#[cfg(test)]
mod tests {
use super::*;
use cloud_terrastodon_azure_types::prelude::RolePermissionAction;
use cloud_terrastodon_azure_types::prelude::Scope;
use eyre::eyre;
#[tokio::test]
async fn it_works() -> Result<()> {
let results = fetch_all_role_definitions().await?;
println!("Found {} role definitions:", results.len());
for role in results {
println!(
"- {:?} {} ({})",
role.kind,
role.display_name,
role.id.short_form()
);
}
Ok(())
}
#[test]
fn understanding_context() -> Result<()> {
let e1 = eyre!("Something went wrong! (base error)").wrap_err("e1 context");
let e2 = e1.wrap_err("e2 context");
println!("{e2:#}\n=====\n{e2:#?}\n=====");
Ok(())
}
#[tokio::test]
async fn key_vaults() -> Result<()> {
let role_definitions = fetch_all_role_definitions().await?;
let key_vault_secrets_officer_id = "b86a8fe4-44ce-4948-aee5-eccb2c155cd7";
let key_vault_secrets_officer = role_definitions
.iter()
.find(|rd| rd.id.short_form() == key_vault_secrets_officer_id)
.ok_or_else(|| eyre!("Couldn't find Key Vault Secrets Officer role definition"))?;
let permission = "Microsoft.KeyVault/vaults/secrets/readMetadata/action";
assert!(key_vault_secrets_officer.satisfies(&[], &[RolePermissionAction::new(permission)]));
Ok(())
}
}