Skip to main content

alien_core/resources/
azure_resource_group.rs

1use crate::error::{ErrorData, Result};
2use crate::resource::{ResourceDefinition, ResourceOutputsDefinition, ResourceRef, ResourceType};
3use alien_error::AlienError;
4use bon::Builder;
5use serde::{Deserialize, Serialize};
6use std::any::Any;
7use std::fmt::Debug;
8
9/// Represents an Azure Resource Group that acts as a logical container for Azure resources.
10#[derive(Debug, Clone, PartialEq, Eq, Serialize, Deserialize, Builder)]
11#[cfg_attr(feature = "openapi", derive(utoipa::ToSchema))]
12#[serde(rename_all = "camelCase", deny_unknown_fields)]
13#[builder(start_fn = new)]
14pub struct AzureResourceGroup {
15    /// Identifier for the resource group. Must contain only alphanumeric characters, hyphens, and underscores ([A-Za-z0-9-_]).
16    /// Maximum 64 characters.
17    #[builder(start_fn)]
18    pub id: String,
19}
20
21impl AzureResourceGroup {
22    /// The resource type identifier for Azure Resource Groups
23    pub const RESOURCE_TYPE: ResourceType = ResourceType::from_static("azure_resource_group");
24
25    /// Returns the resource group's unique identifier.
26    pub fn id(&self) -> &str {
27        &self.id
28    }
29}
30
31/// Outputs generated by a successfully provisioned Azure Resource Group.
32#[derive(Debug, Clone, PartialEq, Eq, Serialize, Deserialize)]
33#[cfg_attr(feature = "openapi", derive(utoipa::ToSchema))]
34#[serde(rename_all = "camelCase")]
35pub struct AzureResourceGroupOutputs {
36    /// The name of the resource group.
37    pub name: String,
38    /// The resource ID of the resource group.
39    pub resource_id: String,
40    /// The location/region of the resource group.
41    pub location: String,
42}
43
44#[typetag::serde(name = "azure_resource_group")]
45impl ResourceOutputsDefinition for AzureResourceGroupOutputs {
46    fn resource_type() -> ResourceType {
47        AzureResourceGroup::RESOURCE_TYPE.clone()
48    }
49
50    fn as_any(&self) -> &dyn Any {
51        self
52    }
53
54    fn box_clone(&self) -> Box<dyn ResourceOutputsDefinition> {
55        Box::new(self.clone())
56    }
57
58    fn outputs_eq(&self, other: &dyn ResourceOutputsDefinition) -> bool {
59        other.as_any().downcast_ref::<AzureResourceGroupOutputs>() == Some(self)
60    }
61}
62
63// Implementation of ResourceDefinition trait for AzureResourceGroup
64#[typetag::serde(name = "azure_resource_group")]
65impl ResourceDefinition for AzureResourceGroup {
66    fn resource_type() -> ResourceType {
67        Self::RESOURCE_TYPE.clone()
68    }
69
70    fn get_resource_type(&self) -> ResourceType {
71        Self::resource_type()
72    }
73
74    fn id(&self) -> &str {
75        &self.id
76    }
77
78    fn get_dependencies(&self) -> Vec<ResourceRef> {
79        Vec::new()
80    }
81
82    fn validate_update(&self, _new_config: &dyn ResourceDefinition) -> Result<()> {
83        Err(AlienError::new(ErrorData::InvalidResourceUpdate {
84            resource_id: self.id.clone(),
85            reason: "Azure resource groups cannot be updated once created".to_string(),
86        }))
87    }
88
89    fn as_any(&self) -> &dyn Any {
90        self
91    }
92
93    fn as_any_mut(&mut self) -> &mut dyn Any {
94        self
95    }
96
97    fn box_clone(&self) -> Box<dyn ResourceDefinition> {
98        Box::new(self.clone())
99    }
100
101    fn resource_eq(&self, other: &dyn ResourceDefinition) -> bool {
102        other.as_any().downcast_ref::<AzureResourceGroup>() == Some(self)
103    }
104}
105
106#[cfg(test)]
107mod tests {
108    use super::*;
109
110    #[test]
111    fn test_azure_resource_group_creation() {
112        let resource_group = AzureResourceGroup::new("my-rg".to_string()).build();
113        assert_eq!(resource_group.id, "my-rg");
114    }
115}