Convert MS security baselines to Azure ARC Guest Configuration packages

In this post, I will show you how to easily convert official Microsoft Windows Security Baselines into Azure ARC Guest Configuration (DSC) packages. So you can easily deploy them to your ARC-managed clients or any DSC-managed client.
In general, you can use this guideline to convert any GPO to DSC!
Start with downloading Microsoft Security Compliance Toolkit 1.0 ZIP file which contains the mentioned baseline files.
Extract the downloaded ZIP to the
C:\ExtractedBaselines
folderInstall required PowerShell modules: 'GPRegistryPolicyParser', 'BaselineManagement', 'GPRegistryPolicyDsc', 'SecurityPolicyDsc', 'AuditPolicyDsc', 'PSDesiredStateConfiguration', ‘GuestConfiguration’
Use PowerShell Core!
# download & import essential modules 'GPRegistryPolicyParser', 'BaselineManagement', 'GPRegistryPolicyDsc', 'SecurityPolicyDsc', 'AuditPolicyDsc', 'PSDesiredStateConfiguration', 'GuestConfiguration' | % { Install-Module $_ } # make sure essential modules are in some of the path mentioned in the $env:PSModulePath!
Extract and organize baseline GPOs for better visibility
# organize GPOs for better visibility $extractedBaseline = "C:\ExtractedBaselines" $gpoBaselineDirectory = "C:\BaselineGpos" Remove-Item $gpoBaselineDirectory -Recurse -Force -ErrorAction SilentlyContinue [Void][System.IO.Directory]::CreateDirectory($gpoBaselineDirectory) foreach ($gpo in (Get-ChildItem "$extractedBaseline\GPOs" -Directory)) { $gpoName = ([regex]"<GPODisplayName><!\[CDATA\[(.+)\]\]></GPODisplayName>").Matches((gc "$($gpo.fullname)\bkupInfo.xml" -Raw)).captures.groups[1].value New-Item -Path $gpoBaselineDirectory -Name $gpoName -ItemType Directory -Force Copy-Item $gpo.fullname "$gpoBaselineDirectory\$gpoName" -Recurse -Force }
Create DSC configurations (
mof
andps1
) from the baseline GPOs# create DSC configuration from the baseline (mof and ps1) $dscDirectory = "C:\ConvertedBaselineGpos" Remove-Item $dscDirectory -Recurse -Force -ErrorAction SilentlyContinue [Void][System.IO.Directory]::CreateDirectory($gpoBaselineDirectory) Get-ChildItem $gpoBaselineDirectory | % { $confName = $_.Name -replace "-", "_" -replace "\s*" $confName # !BEWARE! creating of some localhost.mof can (probably will) end with an error https://github.com/microsoft/BaselineManagement?tab=readme-ov-file#known-gaps-in-capability # problematic ps1 parts have to be commented otherwise you will not be able to create DSC from it! try { $ErrorActionPreference = 'stop' $convertedGpo = ConvertFrom-GPO -Path $_.FullName -OutputConfigurationScript -OutputPath $dscDirectory -ConfigName $confName } catch { if ($_ -like "Invalid MOF definition*") { Write-Warning "In '$($convertedGpo.ConfigurationScript)' comment setting that contains property mentioned in this error '$_'.`n`nOtherwise you will not be able to generate guest configuration from it!" } else { Write-Error $_ } } $ErrorActionPreference = 'continue' # disable mof compilation part (last line) I will compile it by myself later $modifiedContent = Get-Content $convertedGpo.ConfigurationScript | ? { $_ -notlike "$confName -OutputPath *" } $modifiedContent -join "`n" | Set-Content $convertedGpo.ConfigurationScript -Force }
For converting of GPOs to DSC we are using the BaselineManagement module. But because DSC has some limitations, we have to comment on some of the settings in the generated
ps1
DSC scripts, to get working DSCmof
file later!This is also the time when you should modify those
ps1
files to suit your environmental needs
Now when we have the final
ps1
DSC scripts, we can generate DSCmof
files from them (if you want to use them outside of Azure ARC)# generate MOF file # dot source the configuration . <pathToGeneratedPs1Script> # call the configuration <nameOfTheConfigurationStoredInPs1Script> # result will be localhost.mof file
Or we can generate Guest Configuration packages. There is official documentation regarding this topic so just a simplified overview
# created ps1 DSC scripts saved in $dscDirectory have to be converted to guest configuration packages now # https://learn.microsoft.com/en-us/azure/governance/machine-configuration/how-to/develop-custom-package/overview # generate guest configuration package $param = @{ Name = $guestConfigurationName Path = $guestConfigTempDirectory Configuration = "$guestConfigTempDirectory\localhost.mof" FrequencyMinutes = 15 Type = $definedGuestConfigurationType # AuditAndSet, Audit FilesToInclude = '<pathTo_AuditPolicyDsc_Module>', '<pathTo_GPRegistryPolicyDsc_Module>', '<pathTo_SecurityPolicyDsc_Module>' # modules required by generated package to work correctly on destination machine Force = $true } "Generate guest configuration package '$guestConfigurationName'" New-GuestConfigurationPackage @param -ErrorAction continue # Upload package to Azure storage $param = @{ Container = $guestConfigContainerName File = $guestConfigFile Force = $true Blob = $policyBlobName } "Upload blob to Azure Storage" $guestConfigBlob = Set-AzStorageBlobContent @param # set package SAS URL $param = @{ StartTime = $startTime ExpiryTime = $endTime Container = $guestConfigContainerName Blob = $policyBlobName Permission = 'r' Context = $storageAccount.Context FullUri = $true } "Set blob SAS Url" $contentUri = New-AzStorageBlobSASToken @param # generate Azure policy template of guest configuration type locally $policyConfig = @{ PolicyId = (New-Guid).guid ContentUri = $contentUri DisplayName = $guestConfigurationName Description = $description Path = $guestConfigTempDirectory Platform = 'Windows' Mode = $azurePolicyType # ApplyAndMonitor, ApplyAndAutoCorrect, Audit PolicyVersion = $newPolicyVersion.tostring() } "Generate locally Azure policy template of the 'Guest Configuration' type ('$azurePolicyType' ver. $($newPolicyVersion.tostring()))" New-GuestConfigurationPolicy @policyConfig
Subscribe to my newsletter
Read articles from Ondrej Sebela directly inside your inbox. Subscribe to the newsletter, and don't miss out.
Written by

Ondrej Sebela
Ondrej Sebela
I work as System Administrator for more than 10 years now and I love to make my life easier by automating work & personal stuff via PowerShell (even silly things like food recipes list generation).