DSCResources/MSFT_SPExcelServiceApp/MSFT_SPExcelServiceApp.psm1
$script:resourceModulePath = Split-Path -Path (Split-Path -Path $PSScriptRoot -Parent) -Parent $script:modulesFolderPath = Join-Path -Path $script:resourceModulePath -ChildPath 'Modules' $script:resourceHelperModulePath = Join-Path -Path $script:modulesFolderPath -ChildPath 'SharePointDsc.Util' Import-Module -Name (Join-Path -Path $script:resourceHelperModulePath -ChildPath 'SharePointDsc.Util.psm1') $Script:TrustLocationProperties = @( "Address", "LocationType", "IncludeChildren", "SessionTimeout", "ShortSessionTimeout", "NewWorkbookSessionTimeout", "RequestDurationMax", "ChartRenderDurationMax", "WorkbookSizeMax", "ChartAndImageSizeMax", "AutomaticVolatileFunctionCacheLifetime", "DefaultWorkbookCalcMode", "ExternalDataAllowed", "WarnOnDataRefresh", "DisplayGranularExtDataErrors", "AbortOnRefreshOnOpenFail", "PeriodicExtDataCacheLifetime", "ManualExtDataCacheLifetime", "ConcurrentDataRequestsPerSessionMax", "UdfsAllowed", "Description", "RESTExternalDataAllowed" ) $Script:ServiceAppObjectType = "Microsoft.Office.Excel.Server.MossHost.ExcelServerWebServiceApplication" function Get-TargetResource { [CmdletBinding()] [OutputType([System.Collections.Hashtable])] param ( [Parameter(Mandatory = $true)] [System.String] $Name, [Parameter(Mandatory = $true)] [System.String] $ApplicationPool, [Parameter()] [Microsoft.Management.Infrastructure.CimInstance[]] $TrustedFileLocations, [Parameter()] [System.Boolean] $CachingOfUnusedFilesEnable, [Parameter()] [System.Boolean] $CrossDomainAccessAllowed, [Parameter()] [ValidateSet("None", "Connection")] [System.String] $EncryptedUserConnectionRequired, [Parameter()] [System.UInt32] $ExternalDataConnectionLifetime, [Parameter()] [ValidateSet("UseImpersonation", "UseFileAccessAccount")] [System.String] $FileAccessMethod, [Parameter()] [ValidateSet("RoundRobin", "Local", "WorkbookURL")] [System.String] $LoadBalancingScheme, [Parameter()] [System.UInt32] $MemoryCacheThreshold, [Parameter()] [System.Int32] $PrivateBytesMax, [Parameter()] [System.UInt32] $SessionsPerUserMax, [Parameter()] [System.UInt32] $SiteCollectionAnonymousSessionsMax, [Parameter()] [System.Boolean] $TerminateProcessOnAccessViolation, [Parameter()] [System.UInt32] $ThrottleAccessViolationsPerSiteCollection, [Parameter()] [System.String] $UnattendedAccountApplicationId, [Parameter()] [System.Int32] $UnusedObjectAgeMax, [Parameter()] [System.String] $WorkbookCache, [Parameter()] [System.UInt32] $WorkbookCacheSizeMax, [Parameter()] [ValidateSet("Present", "Absent")] [System.String] $Ensure = "Present", [Parameter()] [System.Management.Automation.PSCredential] $InstallAccount ) Write-Verbose -Message "Getting Excel Services Application '$Name'" if ((Get-SPDscInstalledProductVersion).FileMajorPart -ne 15) { throw [Exception] ("Only SharePoint 2013 is supported to deploy Excel Services " + ` "service applications via DSC, as SharePoint 2016 and SharePoint 2019 deprecated " + ` "this service. See " + ` "https://technet.microsoft.com/en-us/library/mt346112(v=office.16).aspx " + ` "for more info.") } $result = Invoke-SPDscCommand -Credential $InstallAccount ` -Arguments @($PSBoundParameters, $Script:ServiceAppObjectType) ` -ScriptBlock { $params = $args[0] $serviceAppObjectType = $args[1] $serviceApps = Get-SPServiceApplication -Name $params.Name ` -ErrorAction SilentlyContinue $nullReturn = @{ Name = $params.Name ApplicationPool = $params.ApplicationPool Ensure = "Absent" InstallAccount = $params.InstallAccount } if ($null -eq $serviceApps) { return $nullReturn } $serviceApp = $serviceApps | Where-Object -FilterScript { $_.GetType().FullName -eq $serviceAppObjectType } if ($null -eq $serviceApp) { return $nullReturn } else { $fileLocations = Get-SPExcelFileLocation -ExcelServiceApplication $serviceApp $fileLocationsToReturn = @() $fileLocations | ForEach-Object -Process { $fileLocationsToReturn += @{ Address = $_.Address LocationType = $_.LocationType IncludeChildren = [Convert]::ToBoolean($_.IncludeChildren) SessionTimeout = $_.SessionTimeout ShortSessionTimeout = $_.ShortSessionTimeout NewWorkbookSessionTimeout = $_.NewWorkbookSessionTimeout RequestDurationMax = $_.RequestDurationMax ChartRenderDurationMax = $_.ChartRenderDurationMax WorkbookSizeMax = $_.WorkbookSizeMax ChartAndImageSizeMax = $_.ChartAndImageSizeMax AutomaticVolatileFunctionCacheLifetime = $_.AutomaticVolatileFunctionCacheLifetime DefaultWorkbookCalcMode = $_.DefaultWorkbookCalcMode ExternalDataAllowed = $_.ExternalDataAllowed WarnOnDataRefresh = [Convert]::ToBoolean($_.WarnOnDataRefresh) DisplayGranularExtDataErrors = [Convert]::ToBoolean($_.DisplayGranularExtDataErrors) AbortOnRefreshOnOpenFail = [Convert]::ToBoolean($_.AbortOnRefreshOnOpenFail) PeriodicExtDataCacheLifetime = $_.PeriodicExtDataCacheLifetime ManualExtDataCacheLifetime = $_.ManualExtDataCacheLifetime ConcurrentDataRequestsPerSessionMax = $_.ConcurrentDataRequestsPerSessionMax UdfsAllowed = [Convert]::ToBoolean($_.UdfsAllowed) Description = $_.Description RESTExternalDataAllowed = [Convert]::ToBoolean($_.RESTExternalDataAllowed) } } $returnVal = @{ Name = $serviceApp.DisplayName ApplicationPool = $serviceApp.ApplicationPool.Name Ensure = "Present" TrustedFileLocations = $fileLocationsToReturn CachingOfUnusedFilesEnable = $serviceApp.CachingOfUnusedFilesEnable CrossDomainAccessAllowed = $serviceApp.CrossDomainAccessAllowed EncryptedUserConnectionRequired = $serviceApp.EncryptedUserConnectionRequired ExternalDataConnectionLifetime = $serviceApp.ExternalDataConnectionLifetime FileAccessMethod = $serviceApp.FileAccessMethod LoadBalancingScheme = $serviceApp.LoadBalancingScheme MemoryCacheThreshold = $serviceApp.MemoryCacheThreshold PrivateBytesMax = $serviceApp.PrivateBytesMax SessionsPerUserMax = $serviceApp.SessionsPerUserMax SiteCollectionAnonymousSessionsMax = $serviceApp.SiteCollectionAnonymousSessionsMax TerminateProcessOnAccessViolation = $serviceApp.TerminateProcessOnAccessViolation ThrottleAccessViolationsPerSiteCollection = $serviceApp.ThrottleAccessViolationsPerSiteCollection UnattendedAccountApplicationId = $serviceApp.UnattendedAccountApplicationId UnusedObjectAgeMax = $serviceApp.UnusedObjectAgeMax WorkbookCache = $serviceApp.WorkbookCache WorkbookCacheSizeMax = $serviceApp.WorkbookCacheSizeMax InstallAccount = $params.InstallAccount } return $returnVal } } return $result } function Set-TargetResource { [CmdletBinding()] param ( [Parameter(Mandatory = $true)] [System.String] $Name, [Parameter(Mandatory = $true)] [System.String] $ApplicationPool, [Parameter()] [Microsoft.Management.Infrastructure.CimInstance[]] $TrustedFileLocations, [Parameter()] [System.Boolean] $CachingOfUnusedFilesEnable, [Parameter()] [System.Boolean] $CrossDomainAccessAllowed, [Parameter()] [ValidateSet("None", "Connection")] [System.String] $EncryptedUserConnectionRequired, [Parameter()] [System.UInt32] $ExternalDataConnectionLifetime, [Parameter()] [ValidateSet("UseImpersonation", "UseFileAccessAccount")] [System.String] $FileAccessMethod, [Parameter()] [ValidateSet("RoundRobin", "Local", "WorkbookURL")] [System.String] $LoadBalancingScheme, [Parameter()] [System.UInt32] $MemoryCacheThreshold, [Parameter()] [System.Int32] $PrivateBytesMax, [Parameter()] [System.UInt32] $SessionsPerUserMax, [Parameter()] [System.UInt32] $SiteCollectionAnonymousSessionsMax, [Parameter()] [System.Boolean] $TerminateProcessOnAccessViolation, [Parameter()] [System.UInt32] $ThrottleAccessViolationsPerSiteCollection, [Parameter()] [System.String] $UnattendedAccountApplicationId, [Parameter()] [System.Int32] $UnusedObjectAgeMax, [Parameter()] [System.String] $WorkbookCache, [Parameter()] [System.UInt32] $WorkbookCacheSizeMax, [Parameter()] [ValidateSet("Present", "Absent")] [System.String] $Ensure = "Present", [Parameter()] [System.Management.Automation.PSCredential] $InstallAccount ) Write-Verbose -Message "Setting Excel Services Application '$Name'" if ((Get-SPDscInstalledProductVersion).FileMajorPart -ne 15) { throw [Exception] ("Only SharePoint 2013 is supported to deploy Excel Services " + ` "service applications via DSC, as SharePoint 2016 and SharePoint 2019 have deprecated " + ` "this service. See " + ` "https://technet.microsoft.com/en-us/library/mt346112(v=office.16).aspx " + ` "for more info.") } $result = Get-TargetResource @PSBoundParameters if ($result.Ensure -eq "Absent" -and $Ensure -eq "Present") { Write-Verbose -Message "Creating Excel Services Application $Name" Invoke-SPDscCommand -Credential $InstallAccount ` -Arguments $PSBoundParameters ` -ScriptBlock { $params = $args[0] New-SPExcelServiceApplication -Name $params.Name ` -ApplicationPool $params.ApplicationPool ` -Default } } if ($Ensure -eq "Present") { Write-Verbose -Message "Updating settings for Excel Services Application $Name" Invoke-SPDscCommand -Credential $InstallAccount ` -Arguments $PSBoundParameters ` -ScriptBlock { $params = $args[0] $params.Add("Identity", $params.Name) # Remove parameters that do not belong on the set method @("InstallAccount", "Ensure", "TrustedFileLocations", "Name", "ApplicationPool") | ForEach-Object -Process { if ($params.ContainsKey($_) -eq $true) { $params.Remove($_) | Out-Null } } Set-SPExcelServiceApplication @params } # Update trusted locations if ($null -ne $TrustedFileLocations) { $TrustedFileLocations | ForEach-Object -Process { $desiredLocation = $_ $matchingCurrentValue = $result.TrustedFileLocations | Where-Object -FilterScript { $_.Address -eq $desiredLocation.Address } if ($null -eq $matchingCurrentValue) { Write-Verbose -Message "Adding trusted location '$($desiredLocation.Address)' to service app" Invoke-SPDscCommand -Credential $InstallAccount ` -Arguments @($PSBoundParameters, $desiredLocation, $Script:TrustLocationProperties, $Script:ServiceAppObjectType) ` -ScriptBlock { $params = $args[0] $desiredLocation = $args[1] $trustLocationProperties = $args[2] $serviceAppObjectType = $args[3] $newArgs = @{ } $trustLocationProperties | ForEach-Object -Process { if ($null -ne $desiredLocation.$_) { $newArgs.Add($_, $desiredLocation.$_) } } $serviceApp = Get-SPServiceApplication -Name $params.Name | Where-Object -FilterScript { $_.GetType().FullName -eq $serviceAppObjectType } $newArgs.Add("ExcelServiceApplication", $serviceApp) New-SPExcelFileLocation @newArgs } } else { Write-Verbose -Message "Updating trusted location '$($desiredLocation.Address)' in service app" Invoke-SPDscCommand -Credential $InstallAccount ` -Arguments @($PSBoundParameters, $desiredLocation, $Script:TrustLocationProperties, $Script:ServiceAppObjectType) ` -ScriptBlock { $params = $args[0] $desiredLocation = $args[1] $trustLocationProperties = $args[2] $serviceAppObjectType = $args[3] $updateArgs = @{ } $trustLocationProperties | ForEach-Object -Process { if ($null -ne $desiredLocation.$_) { $updateArgs.Add($_, $desiredLocation.$_) } } $serviceApp = Get-SPServiceApplication -Name $params.Name | Where-Object -FilterScript { $_.GetType().FullName -eq $serviceAppObjectType } $updateArgs.Add("Identity", $desiredLocation.Address) $updateArgs.Add("ExcelServiceApplication", $serviceApp) Set-SPExcelFileLocation @updateArgs } } } # Remove unlisted trusted locations $result.TrustedFileLocations | ForEach-Object -Process { $currentLocation = $_ $matchingDesiredValue = $TrustedFileLocations | Where-Object -FilterScript { $_.Address -eq $currentLocation.Address } if ($null -eq $matchingDesiredValue) { Write-Verbose -Message "Removing trusted location '$($currentLocation.Address)' from service app" Invoke-SPDscCommand -Credential $InstallAccount ` -Arguments @($Name, $currentLocation) ` -ScriptBlock { $name = $args[0] $currentLocation = $args[1] Remove-SPExcelFileLocation -ExcelServiceApplication $name -Identity $currentLocation.Address -Confirm:$false } } } } } if ($Ensure -eq "Absent") { Write-Verbose -Message "Removing Excel Service Application $Name" Invoke-SPDscCommand -Credential $InstallAccount ` -Arguments @($PSBoundParameters, $Script:ServiceAppObjectType) ` -ScriptBlock { $params = $args[0] $serviceAppObjectType = $args[1] $serviceApp = Get-SPServiceApplication -Name $params.Name | Where-Object -FilterScript { $_.GetType().FullName -eq $serviceAppObjectType } $proxies = Get-SPServiceApplicationProxy foreach ($proxyInstance in $proxies) { if ($serviceApp.IsConnected($proxyInstance)) { $proxyInstance.Delete() } } Remove-SPServiceApplication -Identity $serviceApp -Confirm:$false } } } function Test-TargetResource { [CmdletBinding()] [OutputType([System.Boolean])] param ( [Parameter(Mandatory = $true)] [System.String] $Name, [Parameter(Mandatory = $true)] [System.String] $ApplicationPool, [Parameter()] [Microsoft.Management.Infrastructure.CimInstance[]] $TrustedFileLocations, [Parameter()] [System.Boolean] $CachingOfUnusedFilesEnable, [Parameter()] [System.Boolean] $CrossDomainAccessAllowed, [Parameter()] [ValidateSet("None", "Connection")] [System.String] $EncryptedUserConnectionRequired, [Parameter()] [System.UInt32] $ExternalDataConnectionLifetime, [Parameter()] [ValidateSet("UseImpersonation", "UseFileAccessAccount")] [System.String] $FileAccessMethod, [Parameter()] [ValidateSet("RoundRobin", "Local", "WorkbookURL")] [System.String] $LoadBalancingScheme, [Parameter()] [System.UInt32] $MemoryCacheThreshold, [Parameter()] [System.Int32] $PrivateBytesMax, [Parameter()] [System.UInt32] $SessionsPerUserMax, [Parameter()] [System.UInt32] $SiteCollectionAnonymousSessionsMax, [Parameter()] [System.Boolean] $TerminateProcessOnAccessViolation, [Parameter()] [System.UInt32] $ThrottleAccessViolationsPerSiteCollection, [Parameter()] [System.String] $UnattendedAccountApplicationId, [Parameter()] [System.Int32] $UnusedObjectAgeMax, [Parameter()] [System.String] $WorkbookCache, [Parameter()] [System.UInt32] $WorkbookCacheSizeMax, [Parameter()] [ValidateSet("Present", "Absent")] [System.String] $Ensure = "Present", [Parameter()] [System.Management.Automation.PSCredential] $InstallAccount ) Write-Verbose -Message "Testing Excel Services Application '$Name'" $PSBoundParameters.Ensure = $Ensure if ((Get-SPDscInstalledProductVersion).FileMajorPart -ne 15) { throw [Exception] ("Only SharePoint 2013 is supported to deploy Excel Services " + ` "service applications via DSC, as SharePoint 2016 and SharePoint 2019 have deprecated " + ` "this service. See " + ` "https://technet.microsoft.com/en-us/library/mt346112(v=office.16).aspx " + ` "for more info.") } $CurrentValues = Get-TargetResource @PSBoundParameters Write-Verbose -Message "Current Values: $(Convert-SPDscHashtableToString -Hashtable $CurrentValues)" Write-Verbose -Message "Target Values: $(Convert-SPDscHashtableToString -Hashtable $PSBoundParameters)" $mainCheck = Test-SPDscParameterState -CurrentValues $CurrentValues ` -Source $($MyInvocation.MyCommand.Source) ` -DesiredValues $PSBoundParameters ` -ValuesToCheck @( "Ensure", "CachingOfUnusedFilesEnable", "CrossDomainAccessAllowed", "EncryptedUserConnectionRequired", "ExternalDataConnectionLifetime", "FileAccessMethod", "LoadBalancingScheme", "MemoryCacheThreshold", "PrivateBytesMax", "SessionsPerUserMax", "SiteCollectionAnonymousSessionsMax", "TerminateProcessOnAccessViolation", "ThrottleAccessViolationsPerSiteCollection", "UnattendedAccountApplicationId", "UnusedObjectAgeMax", "WorkbookCache", "WorkbookCacheSizeMax" ) if ($Ensure -eq "Present" -and $mainCheck -eq $true -and $null -ne $TrustedFileLocations) { # Check that all the desired types are in the current values and match $locationCheck = $TrustedFileLocations | ForEach-Object -Process { $desiredLocation = $_ $matchingCurrentValue = $CurrentValues.TrustedFileLocations | Where-Object -FilterScript { $_.Address -eq $desiredLocation.Address } if ($null -eq $matchingCurrentValue) { $message = ("Trusted file location '$($_.Address)' was not found " + ` "in the Excel service app. Desired state is false.") Write-Verbose -Message $message Add-SPDscEvent -Message $message -EntryType 'Error' -EventID 1 -Source $MyInvocation.MyCommand.Source return $false } else { $Script:TrustLocationProperties | ForEach-Object -Process { if ($desiredLocation.CimInstanceProperties.Name -contains $_) { if ($desiredLocation.$_ -ne $matchingCurrentValue.$_) { $message = ("Trusted file location '$($desiredLocation.Address)' did not match " + ` "desired property '$_'. Desired value is " + ` "'$($desiredLocation.$_)' but the current value is " + ` "'$($matchingCurrentValue.$_)'") Write-Verbose -Message $message Add-SPDscEvent -Message $message -EntryType 'Error' -EventID 1 -Source $MyInvocation.MyCommand.Source return $false } } } } return $true } if ($locationCheck -contains $false) { Write-Verbose -Message "Test-TargetResource returned false" return $false } # Check that any other existing trusted locations are in the desired state $locationCheck = $CurrentValues.TrustedFileLocations | ForEach-Object -Process { $currentLocation = $_ $matchingDesiredValue = $TrustedFileLocations | Where-Object -FilterScript { $_.Address -eq $currentLocation.Address } if ($null -eq $matchingDesiredValue) { $message = ("Existing trusted file location '$($_.Address)' was not " + ` "found in the desired state for this service " + ` "application. Desired state is false.") Write-Verbose -Message $message Add-SPDscEvent -Message $message -EntryType 'Error' -EventID 1 -Source $MyInvocation.MyCommand.Source return $false } return $true } if ($locationCheck -contains $false) { Write-Verbose -Message "Test-TargetResource returned false" return $false } # at this point if no other value has been returned, all desired entires exist and are # correct, and no existing entries exist that are not in desired state, so return true Write-Verbose -Message "Test-TargetResource returned true" return $true } else { Write-Verbose -Message "Test-TargetResource returned $mainCheck" return $mainCheck } } Export-ModuleMember -Function *-TargetResource |