DSCResources/MSFT_SPOSiteDesign/MSFT_SPOSiteDesign.psm1
function Get-TargetResource { [CmdletBinding()] [OutputType([System.Collections.Hashtable])] param ( [Parameter(Mandatory = $true)] [System.String] $Title, [Parameter()] [System.String[]] $SiteScriptNames, [Parameter()] [ValidateSet('CommunicationSite', 'TeamSite', 'GrouplessTeamSite')] [System.String] $WebTemplate, [Parameter()] [System.Boolean] $IsDefault, [Parameter()] [System.String] $PreviewImageAltText, [Parameter()] [System.String] $PreviewImageUrl, [Parameter()] [System.String] $Description, [Parameter()] [System.UInt32] $Version, [Parameter()] [ValidateSet('Present', 'Absent')] [System.String] $Ensure = 'Present', [Parameter()] [System.Management.Automation.PSCredential] $Credential, [Parameter()] [System.String] $ApplicationId, [Parameter()] [System.String] $TenantId, [Parameter()] [System.Management.Automation.PSCredential] $ApplicationSecret, [Parameter()] [System.String] $CertificatePath, [Parameter()] [System.Management.Automation.PSCredential] $CertificatePassword, [Parameter()] [System.String] $CertificateThumbprint, [Parameter()] [Switch] $ManagedIdentity ) Write-Verbose -Message "Getting configuration for SPO SiteDesign for $Title" $ConnectionMode = New-M365DSCConnection -Workload 'PNP' ` -InboundParameters $PSBoundParameters #Ensure the proper dependencies are installed in the current environment. Confirm-M365DSCDependencies #region Telemetry $ResourceName = $MyInvocation.MyCommand.ModuleName -replace 'MSFT_', '' $CommandName = $MyInvocation.MyCommand $data = Format-M365DSCTelemetryParameters -ResourceName $ResourceName ` -CommandName $CommandName ` -Parameters $PSBoundParameters Add-M365DSCTelemetryEvent -Data $data #endregion $nullReturn = $PSBoundParameters $nullReturn.Ensure = 'Absent' try { Write-Verbose -Message "Getting Site Design for $Title" $siteDesign = Get-PnPSiteDesign -Identity $Title -ErrorAction SilentlyContinue if ($null -eq $siteDesign) { Write-Verbose -Message "No Site Design found for $Title" return $nullReturn } $scriptTitles = @() foreach ($scriptId in $siteDesign.SiteScriptIds) { $siteScript = Get-PnPSiteScript -Identity $scriptId -ErrorAction SilentlyContinue if ($null -ne $siteScript) { $scriptTitles += $siteScript.Title } } ## Todo need to see if we can get this somehow from PNP module instead of hard coded in script ## https://github.com/SharePoint/PnP-PowerShell/blob/master/Commands/Enums/SiteWebTemplate.cs $webtemp = $null if ($siteDesign.WebTemplate -eq '64') { $webtemp = 'TeamSite' } elseif ($siteDesign.WebTemplate -eq '1') { $webtemp = 'GrouplessTeamSite' } else { $webtemp = 'CommunicationSite' } return @{ Title = $siteDesign.Title SiteScriptNames = $scriptTitles WebTemplate = $webtemp IsDefault = $siteDesign.IsDefault Description = $siteDesign.Description PreviewImageAltText = $siteDesign.PreviewImageAltText PreviewImageUrl = $siteDesign.PreviewImageUrl Version = $siteDesign.Version Ensure = 'Present' Credential = $Credential ApplicationId = $ApplicationId TenantId = $TenantId ApplicationSecret = $ApplicationSecret CertificatePassword = $CertificatePassword CertificatePath = $CertificatePath CertificateThumbprint = $CertificateThumbprint Managedidentity = $ManagedIdentity.IsPresent } } catch { New-M365DSCLogEntry -Message 'Error retrieving data:' ` -Exception $_ ` -Source $($MyInvocation.MyCommand.Source) ` -TenantId $TenantId ` -Credential $Credential return $nullReturn } } function Set-TargetResource { [CmdletBinding()] param ( [Parameter(Mandatory = $true)] [System.String] $Title, [Parameter()] [ValidateSet('CommunicationSite', 'TeamSite', 'GrouplessTeamSite')] [System.String] $WebTemplate, [Parameter()] [System.String[]] $SiteScriptNames, [Parameter()] [System.Boolean] $IsDefault, [Parameter()] [System.String] $PreviewImageAltText, [Parameter()] [System.String] $PreviewImageUrl, [Parameter()] [System.String] $Description, [Parameter()] [System.UInt32] $Version, [Parameter()] [ValidateSet('Present', 'Absent')] [System.String] $Ensure = 'Present', [Parameter()] [System.Management.Automation.PSCredential] $Credential, [Parameter()] [System.String] $ApplicationId, [Parameter()] [System.String] $TenantId, [Parameter()] [System.Management.Automation.PSCredential] $ApplicationSecret, [Parameter()] [System.String] $CertificatePath, [Parameter()] [System.Management.Automation.PSCredential] $CertificatePassword, [Parameter()] [System.String] $CertificateThumbprint, [Parameter()] [Switch] $ManagedIdentity ) Write-Verbose -Message "Setting configuration for SPO SiteDesign for $Title" #Ensure the proper dependencies are installed in the current environment. Confirm-M365DSCDependencies #region Telemetry $ResourceName = $MyInvocation.MyCommand.ModuleName -replace 'MSFT_', '' $CommandName = $MyInvocation.MyCommand $data = Format-M365DSCTelemetryParameters -ResourceName $ResourceName ` -CommandName $CommandName ` -Parameters $PSBoundParameters Add-M365DSCTelemetryEvent -Data $data #endregion $ConnectionMode = New-M365DSCConnection -Workload 'PNP' ` -InboundParameters $PSBoundParameters $curSiteDesign = Get-TargetResource @PSBoundParameters # Get list of site script names $scriptIds = @() foreach ($siteScriptName in $SiteScriptNames) { $siteScript = Get-PnPSiteScript | Where-Object -FilterScript { $_.Title -eq $siteScriptName } $scriptIds += $siteScript.Id } $CurrentParameters = $PSBoundParameters $CurrentParameters.Remove('Credential') | Out-Null $CurrentParameters.Remove('SiteScriptNames') | Out-Null $CurrentParameters.Remove('Ensure') | Out-Null $CurrentParameters.Remove('ApplicationId') | Out-Null $CurrentParameters.Remove('TenantId') | Out-Null $CurrentParameters.Remove('CertificatePath') | Out-Null $CurrentParameters.Remove('CertificatePassword') | Out-Null $CurrentParameters.Remove('CertificateThumbprint') | Out-Null $CurrentParameters.Remove('ManagedIdentity') | Out-Null $CurrentParameters.Remove('ApplicationSecret') | Out-Null $CurrentParameters.Add('SiteScriptIds', $scriptIds) if ($curSiteDesign.Ensure -eq 'Absent' -and 'Present' -eq $Ensure ) { $CurrentParameters.Remove('Version') Write-Verbose -Message "Adding new site design $Title" Add-PnPSiteDesign @CurrentParameters } elseif (($curSiteDesign.Ensure -eq 'Present' -and 'Present' -eq $Ensure)) { $siteDesign = Get-PnPSiteDesign -Identity $Title -ErrorAction SilentlyContinue if ($null -ne $siteDesign) { Write-Verbose -Message "Updating current site design $Title" Set-PnPSiteDesign -Identity $siteDesign.Id @CurrentParameters } } elseif (($Ensure -eq 'Absent' -and $curSiteDesign.Ensure -eq 'Present')) { $siteDesign = Get-PnPSiteDesign -Identity $Title -ErrorAction SilentlyContinue if ($null -ne $siteDesign) { Write-Verbose -Message "Removing site design $Title" Remove-PnPSiteDesign -Identity $siteDesign.Id -Force } } } function Test-TargetResource { [CmdletBinding()] [OutputType([System.Boolean])] param ( [Parameter(Mandatory = $true)] [System.String] $Title, [Parameter()] [ValidateSet('CommunicationSite', 'TeamSite', 'GrouplessTeamSite')] [System.String] $WebTemplate, [Parameter()] [System.String[]] $SiteScriptNames, [Parameter()] [System.Boolean] $isDefault, [Parameter()] [System.String] $PreviewImageAltText, [Parameter()] [System.String] $PreviewImageUrl, [Parameter()] [System.String] $Description, [Parameter()] [System.UInt32] $Version, [Parameter()] [ValidateSet('Present', 'Absent')] [System.String] $Ensure = 'Present', [Parameter()] [System.Management.Automation.PSCredential] $Credential, [Parameter()] [System.String] $ApplicationId, [Parameter()] [System.String] $TenantId, [Parameter()] [System.Management.Automation.PSCredential] $ApplicationSecret, [Parameter()] [System.String] $CertificatePath, [Parameter()] [System.Management.Automation.PSCredential] $CertificatePassword, [Parameter()] [System.String] $CertificateThumbprint, [Parameter()] [Switch] $ManagedIdentity ) #Ensure the proper dependencies are installed in the current environment. Confirm-M365DSCDependencies #region Telemetry $ResourceName = $MyInvocation.MyCommand.ModuleName -replace 'MSFT_', '' $CommandName = $MyInvocation.MyCommand $data = Format-M365DSCTelemetryParameters -ResourceName $ResourceName ` -CommandName $CommandName ` -Parameters $PSBoundParameters Add-M365DSCTelemetryEvent -Data $data #endregion Write-Verbose -Message "Testing configuration for SPO SiteDesign for $Title" $CurrentValues = Get-TargetResource @PSBoundParameters Write-Verbose -Message "Current Values: $(Convert-M365DscHashtableToString -Hashtable $CurrentValues)" Write-Verbose -Message "Target Values: $(Convert-M365DscHashtableToString -Hashtable $PSBoundParameters)" $ValuesToCheck = $PSBoundParameters $ValuesToCheck.Remove('Credential') | Out-Null $ValuesToCheck.Remove('ApplicationId') | Out-Null $ValuesToCheck.Remove('TenantId') | Out-Null $ValuesToCheck.Remove('CertificatePath') | Out-Null $ValuesToCheck.Remove('CertificatePassword') | Out-Null $ValuesToCheck.Remove('CertificateThumbprint') | Out-Null $ValuesToCheck.Remove('ManagedIdentity') | Out-Null $ValuesToCheck.Remove('ApplicationSecret') | Out-Null $TestResult = Test-M365DSCParameterState -CurrentValues $CurrentValues ` -Source $($MyInvocation.MyCommand.Source) ` -DesiredValues $PSBoundParameters ` -ValuesToCheck $ValuesToCheck.Keys Write-Verbose -Message "Test-TargetResource returned $TestResult" return $TestResult } function Export-TargetResource { [CmdletBinding()] [OutputType([System.String])] param ( [Parameter()] [System.Management.Automation.PSCredential] $Credential, [Parameter()] [System.String] $ApplicationId, [Parameter()] [System.String] $TenantId, [Parameter()] [System.Management.Automation.PSCredential] $ApplicationSecret, [Parameter()] [System.String] $CertificatePath, [Parameter()] [System.Management.Automation.PSCredential] $CertificatePassword, [Parameter()] [System.String] $CertificateThumbprint, [Parameter()] [Switch] $ManagedIdentity ) try { $ConnectionMode = New-M365DSCConnection -Workload 'PNP' ` -InboundParameters $PSBoundParameters #Ensure the proper dependencies are installed in the current environment. Confirm-M365DSCDependencies #region Telemetry $ResourceName = $MyInvocation.MyCommand.ModuleName -replace 'MSFT_', '' $CommandName = $MyInvocation.MyCommand $data = Format-M365DSCTelemetryParameters -ResourceName $ResourceName ` -CommandName $CommandName ` -Parameters $PSBoundParameters Add-M365DSCTelemetryEvent -Data $data #endregion $dscContent = '' $i = 1 [array]$designs = Get-PnPSiteDesign -ErrorAction Stop if ($designs.Length -eq 0) { Write-Host $Global:M365DSCEmojiGreenCheckMark } else { Write-Host "`r`n" -NoNewline } foreach ($design in $designs) { Write-Host " |---[$i/$($designs.Length)] $($design.Title)" -NoNewline $Params = @{ Title = $design.Title ApplicationId = $ApplicationId TenantId = $TenantId ApplicationSecret = $ApplicationSecret CertificatePassword = $CertificatePassword CertificatePath = $CertificatePath CertificateThumbprint = $CertificateThumbprint Managedidentity = $ManagedIdentity.IsPresent Credential = $Credential } $Results = Get-TargetResource @Params $Results = Update-M365DSCExportAuthenticationResults -ConnectionMode $ConnectionMode ` -Results $Results $currentDSCBlock = Get-M365DSCExportContentForResource -ResourceName $ResourceName ` -ConnectionMode $ConnectionMode ` -ModulePath $PSScriptRoot ` -Results $Results ` -Credential $Credential $dscContent += $currentDSCBlock Save-M365DSCPartialExport -Content $currentDSCBlock ` -FileName $Global:PartialExportFileName $i++ Write-Host $Global:M365DSCEmojiGreenCheckmark } return $dscContent } catch { Write-Host $Global:M365DSCEmojiRedX New-M365DSCLogEntry -Message 'Error during Export:' ` -Exception $_ ` -Source $($MyInvocation.MyCommand.Source) ` -TenantId $TenantId ` -Credential $Credential return '' } } Export-ModuleMember -Function *-TargetResource |