AzureDevOpsDsc.psm1
#Region '.\prefix.ps1' -1 $script:dscResourceCommonModulePath = Join-Path -Path $PSScriptRoot -ChildPath 'Modules\DscResource.Common' Import-Module -Name $script:dscResourceCommonModulePath # Import nested, 'AzureDevOpsDsc.Common' module $script:azureDevOpsDscCommonModulePath = Join-Path -Path $PSScriptRoot -ChildPath 'Modules\AzureDevOpsDsc.Common' Import-Module -Name $script:azureDevOpsDscCommonModulePath # Define localization data $script:localizedData = Get-LocalizedData -DefaultUICulture 'en-US' #EndRegion '.\prefix.ps1' 12 #Region '.\Enum\Ensure.ps1' -1 <# .SYNOPSIS Defines whether the DSC resource should be Present or Absent. #> enum Ensure { Present Absent } #EndRegion '.\Enum\Ensure.ps1' 10 #Region '.\Enum\RequiredAction.ps1' -1 <# .SYNOPSIS Defines the `RequiredAction` of the DSC resource. #> enum RequiredAction { None Get New Set Remove Test Error } #EndRegion '.\Enum\RequiredAction.ps1' 15 #Region '.\Classes\001.DscResourceBase.ps1' -1 <# .SYNOPSIS Defines a base class from which other DSC resources inherit from. #> class DscResourceBase { hidden [System.String]GetDscResourceKey() { [System.String]$dscResourceKeyPropertyName = $this.GetDscResourceKeyPropertyName() if ([String]::IsNullOrWhiteSpace($dscResourceKeyPropertyName)) { $errorMessage = "Cannot obtain a 'DscResourceKey' value for the '$($this.GetType().Name)' instance." New-InvalidOperationException -Message $errorMessage } return $this."$dscResourceKeyPropertyName" } hidden [System.String]GetDscResourceKeyPropertyName() { [System.String[]]$dscResourceKeyPropertyNames = @() [Type]$thisType = $this.GetType() [System.Reflection.PropertyInfo[]]$thisProperties = $thisType.GetProperties() $thisProperties | ForEach-Object { [System.Reflection.PropertyInfo]$propertyInfo = $_ $PropertyName = $_.Name $propertyInfo.GetCustomAttributes($true) | ForEach-Object { if ($_.TypeId.Name -eq 'DscPropertyAttribute' -and $_.Key -eq $true) { $dscResourceKeyPropertyNames += $PropertyName } } } if ($null -eq $dscResourceKeyPropertyNames -or $dscResourceKeyPropertyNames.Count -eq 0) { $errorMessage = "Could not obtain a 'DscResourceDscKey' property for type '$($this.GetType().Name)'." New-InvalidOperationException -Message $errorMessage } elseif ($dscResourceKeyPropertyNames.Count -gt 1) { $errorMessage = "Obtained more than 1 property for type '$($this.GetType().Name)' that was marked as a 'Key'. There must only be 1 property on the class set as the 'Key' for DSC." New-InvalidOperationException -Message $errorMessage } return $dscResourceKeyPropertyNames[0] } hidden [System.String[]]GetDscResourcePropertyNames() { [System.String[]]$thisDscPropertyNames = @() [Type]$thisType = $this.GetType() [System.Reflection.PropertyInfo[]]$thisProperties = $thisType.GetProperties() $thisProperties | ForEach-Object { $propertyInfo = $_ $PropertyName = $_.Name $propertyInfo.GetCustomAttributes($true) | ForEach-Object { if ($_.TypeId.Name -eq 'DscPropertyAttribute') { $thisDscPropertyNames += $PropertyName } } } return $thisDscPropertyNames } hidden [System.String[]]GetDscResourcePropertyNamesWithNoSetSupport() { return @() } } #EndRegion '.\Classes\001.DscResourceBase.ps1' 89 #Region '.\Classes\002.AzDevOpsApiDscResourceBase.ps1' -1 <# .SYNOPSIS Defines a base class from which other DSC resources that use the AzureDevOps API inherit from. #> class AzDevOpsApiDscResourceBase : DscResourceBase { [System.String]$ResourceName = $this.GetResourceName() hidden [System.String]GetResourceName() { # Assumes a naming convention is followed between the DSC # resource name and the name of the resource within the API return $this.GetType().ToString().Replace('AzDevOps','') } <# .NOTES When creating an object via the Azure DevOps API, the ID (if provided) is ignored and Azure DevOps creates/generates the Id (a GUID) which can then be used for the object. As a result, only existing resources from the API will have a ResourceId and new resources to be created via the API do not need one providing. #> [System.String]$ResourceId = $this.GetResourceId() [System.String]$ResourceIdPropertyName = $this.GetResourceIdPropertyName() hidden [System.String]GetResourceId() { return $this."$($this.ResourceIdPropertyName)" } hidden [System.String]GetResourceIdPropertyName() { return "$($this.ResourceName)Id" } <# .NOTES When creating an object via the Azure DevOps API, the 'Key' of the object will be another, alternate, unique key/identifier to the 'ResourceId' but this will be specific to the resource. This 'Key' can be used to determine an 'Id' of a new resource that has been added. #> [System.String]$ResourceKey = $this.GetResourceKey() [System.String]$ResourceKeyPropertyName = $this.GetResourceKeyPropertyName() hidden [System.String]GetResourceKey() { [System.String]$keyPropertyName = $this.ResourceKeyPropertyName if ([System.String]::IsNullOrWhiteSpace($keyPropertyName)) { return $null } return $this."$keyPropertyName" } hidden [System.String]GetResourceKeyPropertyName() { # Use same property as the DSC Resource 'Key' return $this.GetDscResourceKeyPropertyName() } hidden [System.String]GetResourceFunctionName([RequiredAction]$RequiredAction) { if ($RequiredAction -in @( [RequiredAction]::Get, [RequiredAction]::New, [RequiredAction]::Set, [RequiredAction]::Remove, [RequiredAction]::Test)) { return "$($RequiredAction)-AzDevOps$($this.ResourceName)" } return $null } } #EndRegion '.\Classes\002.AzDevOpsApiDscResourceBase.ps1' 89 #Region '.\Classes\003.AzDevOpsDscResourceBase.ps1' -1 <# .SYNOPSIS Defines a base class from which other AzureDevOps DSC resources inherit from. #> class AzDevOpsDscResourceBase : AzDevOpsApiDscResourceBase { [DscProperty()] [Alias('Uri')] [System.String] $ApiUri [DscProperty()] [Alias('PersonalAccessToken')] [System.String] $Pat [DscProperty()] [Ensure] $Ensure hidden [Hashtable]GetDscCurrentStateObjectGetParameters() { # Setup a default set of parameters to pass into the resource/object's 'Get' method $getParameters = @{ ApiUri = $this.ApiUri Pat = $this.Pat "$($this.GetResourceKeyPropertyName())" = $this.GetResourceKey() } # If there is an available 'ResourceId' value, add it to the parameters/hashtable if (![System.String]::IsNullOrWhiteSpace($this.GetResourceId())) { $getParameters."$($this.GetResourceIdPropertyName())" = $this.GetResourceId() } return $getParameters } hidden [PsObject]GetDscCurrentStateResourceObject([Hashtable]$GetParameters) { # Obtain the 'Get' function name for the object, then invoke it $thisResourceGetFunctionName = $this.GetResourceFunctionName(([RequiredAction]::Get)) return $(& $thisResourceGetFunctionName @GetParameters) } hidden [System.Management.Automation.PSObject]GetDscCurrentStateObject() { $getParameters = $this.GetDscCurrentStateObjectGetParameters() $dscCurrentStateResourceObject = $this.GetDscCurrentStateResourceObject($getParameters) # If no object was returned (i.e it does not exist), create a default/empty object if ($null -eq $dscCurrentStateResourceObject) { return New-Object -TypeName 'System.Management.Automation.PSObject' -Property @{ Ensure = [Ensure]::Absent } } return $dscCurrentStateResourceObject } hidden [Hashtable]GetDscCurrentStateProperties() { # Obtain 'CurrentStateResourceObject' and pass into overidden function of inheriting class return $this.GetDscCurrentStateProperties($this.GetDscCurrentStateObject()) } # This method must be overidden by inheriting class(es) hidden [Hashtable]GetDscCurrentStateProperties([PSCustomObject]$CurrentResourceObject) { # Obtain the type of $this object. Throw an exception if this is being called from the base class method. $thisType = $this.GetType() if ($thisType -eq [AzDevOpsDscResourceBase]) { $errorMessage = "Method 'GetCurrentState()' in '$($thisType.Name)' must be overidden and called by an inheriting class." New-InvalidOperationException -Message $errorMessage } return $null } hidden [Hashtable]GetDscDesiredStateProperties() { [Hashtable]$dscDesiredStateProperties = @{} # Obtain all DSC-related properties, and add them and their values to the hashtable output $this.GetDscResourcePropertyNames() | ForEach-Object { $dscDesiredStateProperties."$_" = $this."$_" } return $dscDesiredStateProperties } hidden [RequiredAction]GetDscRequiredAction() { [Hashtable]$currentProperties = $this.GetDscCurrentStateProperties() [Hashtable]$desiredProperties = $this.GetDscDesiredStateProperties() [System.String[]]$dscPropertyNamesWithNoSetSupport = $this.GetDscResourcePropertyNamesWithNoSetSupport() [System.String[]]$dscPropertyNamesToCompare = $this.GetDscResourcePropertyNames() # Update 'Id' property: # Set $desiredProperties."$IdPropertyName" to $currentProperties."$IdPropertyName" if it's desired # value is blank/null but it's current/existing value is known (and can be recovered from $currentProperties). # # This ensures that alternate keys (typically ResourceIds) not provided in the DSC configuration do not flag differences [System.String]$IdPropertyName = $this.GetResourceIdPropertyName() if ([System.String]::IsNullOrWhiteSpace($desiredProperties[$IdPropertyName]) -and ![System.String]::IsNullOrWhiteSpace($currentProperties[$IdPropertyName])) { $desiredProperties."$IdPropertyName" = $currentProperties."$IdPropertyName" } # Perform logic with 'Ensure' (to determine whether resource should be created or dropped (or updated, if already [Ensure]::Present but property values differ) $dscRequiredAction = [RequiredAction]::None switch ($desiredProperties.Ensure) { ([Ensure]::Present) { # If not already present, or different to expected/desired - return [RequiredAction]::New (i.e. Resource needs creating) if ($null -eq $currentProperties -or $($currentProperties.Ensure) -ne [Ensure]::Present) { $dscRequiredAction = [RequiredAction]::New Write-Verbose "DscActionRequired='$dscRequiredAction'" break } # Changes made by DSC to the following properties are unsupported by the resource (other than when creating a [RequiredAction]::New resource) if ($dscPropertyNamesWithNoSetSupport.Count -gt 0) { $dscPropertyNamesWithNoSetSupport | ForEach-Object { if ($($currentProperties[$_].ToString()) -ne $($desiredProperties[$_].ToString())) { $errorMessage = "The '$($this.GetType().Name)', DSC Resource does not support changes for/to the '$_' property." New-InvalidOperationException -Message $errorMessage } } } # Compare all properties ('Current' vs 'Desired') if ($dscPropertyNamesToCompare.Count -gt 0) { $dscPropertyNamesToCompare | ForEach-Object { if ($($currentProperties."$_") -ne $($desiredProperties."$_")) { Write-Verbose "DscPropertyValueMismatch='$_'" $dscRequiredAction = [RequiredAction]::Set } } if ($dscRequiredAction -eq [RequiredAction]::Set) { Write-Verbose "DscActionRequired='$dscRequiredAction'" break } } # Otherwise, no changes to make (i.e. The desired state is already achieved) return $dscRequiredAction break } ([Ensure]::Absent) { # If currently/already present - return $false (i.e. state is incorrect) if ($null -ne $currentProperties -and $currentProperties.Ensure -ne [Ensure]::Absent) { $dscRequiredAction = [RequiredAction]::Remove Write-Verbose "DscActionRequired='$dscRequiredAction'" break } # Otherwise, no changes to make (i.e. The desired state is already achieved) Write-Verbose "DscActionRequired='$dscRequiredAction'" return $dscRequiredAction break } default { $errorMessage = "Could not obtain a valid 'Ensure' value within '$($this.GetResourceName())' Test() function. Value was '$($desiredProperties.Ensure)'." New-InvalidOperationException -Message $errorMessage } } return $dscRequiredAction } hidden [Hashtable]GetDesiredStateParameters([Hashtable]$CurrentStateProperties, [Hashtable]$DesiredStateProperties, [RequiredAction]$RequiredAction) { [Hashtable]$desiredStateParameters = $DesiredStateProperties [System.String]$IdPropertyName = $this.GetResourceIdPropertyName() # If actions required are 'None' or 'Error', return a $null value if ($RequiredAction -in @([RequiredAction]::None, [RequiredAction]::Error)) { return $null } # If the desired state/action is to remove the resource, generate/return a minimal set of parameters required to remove the resource elseif ($RequiredAction -eq [RequiredAction]::Remove) { return @{ ApiUri = $DesiredStateProperties.ApiUri Pat = $DesiredStateProperties.Pat Force = $true # Set this from the 'Current' state as we would expect this to have an existing key/ID value to use "$IdPropertyName" = $CurrentStateProperties."$IdPropertyName" } } # If the desired state/action is to add/new or update/set the resource, start with the values in the $DesiredStateProperties variable, and amend elseif ($RequiredAction -in @([RequiredAction]::New, [RequiredAction]::Set)) { # Set $desiredParameters."$IdPropertyName" to $CurrentStateProperties."$IdPropertyName" if it's known and can be recovered from existing resource if ([System.String]::IsNullOrWhiteSpace($desiredStateParameters."$IdPropertyName") -and ![System.String]::IsNullOrWhiteSpace($CurrentStateProperties."$IdPropertyName")) { $desiredStateParameters."$IdPropertyName" = $CurrentStateProperties."$IdPropertyName" } # Alternatively, if $desiredParameters."$IdPropertyName" is null/empty, remove the key (as we don't want to pass an empty/null parameter) elseif ([System.String]::IsNullOrWhiteSpace($desiredStateParameters."$IdPropertyName")) { $desiredStateParameters.Remove($IdPropertyName) } # Do not need/want this passing as a parameter (the action taken will determine the desired state) $desiredStateParameters.Remove('Ensure') # Add this to 'Force' subsequent function call $desiredStateParameters.Force = $true # Some DSC properties are only supported for 'New' and 'Remove' actions, but not 'Set' ones (these need to be removed) [System.String[]]$unsupportedForSetPropertyNames = $this.GetDscResourcePropertyNamesWithNoSetSupport() if ($RequiredAction -eq [RequiredAction]::Set -and $unsupportedForSetPropertyNames.Count -gt 0) { $unsupportedForSetPropertyNames | ForEach-Object { $desiredStateParameters.Remove($_) } } } else { $errorMessage = "A required action of '$RequiredAction' has not been catered for in GetDesiredStateParameters() method." New-InvalidOperationException -Message $errorMessage } return $desiredStateParameters } hidden [System.Boolean]TestDesiredState() { return ($this.GetDscRequiredAction() -eq [RequiredAction]::None) } [System.Boolean] Test() { # TestDesiredState() will throw an exception in certain expected circumstances. Return $false if this occurs. try { return $this.TestDesiredState() } catch { return $false } } [Int32]GetPostSetWaitTimeMs() { return 2000 } [void] SetToDesiredState() { [RequiredAction]$dscRequiredAction = $this.GetDscRequiredAction() if ($dscRequiredAction -in @([RequiredAction]::'New', [RequiredAction]::'Set', [RequiredAction]::'Remove')) { $dscCurrentStateProperties = $this.GetDscCurrentStateProperties() $dscDesiredStateProperties = $this.GetDscDesiredStateProperties() $dscRequiredActionFunctionName = $this.GetResourceFunctionName($dscRequiredAction) $dscDesiredStateParameters = $this.GetDesiredStateParameters($dscCurrentStateProperties, $dscDesiredStateProperties, $dscRequiredAction) & $dscRequiredActionFunctionName @dscDesiredStateParameters | Out-Null Start-Sleep -Milliseconds $($this.GetPostSetWaitTimeMs()) } } [void] Set() { $this.SetToDesiredState() } } #EndRegion '.\Classes\003.AzDevOpsDscResourceBase.ps1' 315 #Region '.\Classes\010.AzDevOpsProject.ps1' -1 <# .SYNOPSIS A DSC Resource for Azure DevOps that represents the 'Project' resource. .DESCRIPTION A DSC Resource for Azure DevOps that represents the 'Project' resource. .PARAMETER ProjectId The 'Id' of the Azure DevOps, 'Project' resource. .PARAMETER ProjectName The 'Name' of the Azure DevOps, 'Project' resource. .PARAMETER ProjectDescription The 'Description' of the Azure DevOps, 'Project' resource. .PARAMETER SourceControlType The 'SourceControlType' of the Azure DevOps, 'Project' resource. If the 'Project' resource already exists in Azure DevOps, the parameter `SourceControlType` cannot be used to change to another type. #> [DscResource()] [Diagnostics.CodeAnalysis.SuppressMessageAttribute('PSDSCStandardDSCFunctionsInResource', '', Justification='Test() and Set() method are inherited from base, "AzDevOpsDscResourceBase" class')] class AzDevOpsProject : AzDevOpsDscResourceBase { [DscProperty()] [Alias('Id')] [System.String]$ProjectId [DscProperty(Key, Mandatory)] [Alias('Name')] [System.String]$ProjectName [DscProperty()] [Alias('Description')] [System.String]$ProjectDescription [DscProperty()] [ValidateSet('Git', 'Tfvc')] [System.String]$SourceControlType [AzDevOpsProject] Get() { return [AzDevOpsProject]$($this.GetDscCurrentStateProperties()) } hidden [System.String[]]GetDscResourcePropertyNamesWithNoSetSupport() { return @('SourceControlType') } hidden [Hashtable]GetDscCurrentStateProperties([PSCustomObject]$CurrentResourceObject) { $properties = @{ Pat = $this.Pat ApiUri = $this.ApiUri Ensure = [Ensure]::Absent } if ($null -ne $CurrentResourceObject) { if (![System.String]::IsNullOrWhiteSpace($CurrentResourceObject.id)) { $properties.Ensure = [Ensure]::Present } $properties.ProjectId = $CurrentResourceObject.id $properties.ProjectName = $CurrentResourceObject.name $properties.ProjectDescription = $CurrentResourceObject.description $properties.SourceControlType = $CurrentResourceObject.capabilities.versioncontrol.sourceControlType } return $properties } } #EndRegion '.\Classes\010.AzDevOpsProject.ps1' 79 |