The script generates a report that includes information on Microsoft 365 licenses. The report will include numbers of available and used licenses in all registered tenants. To execute the script, create a report with corresponding custom columns. The report should have no scope or parameters
Parameters:
- $sumLicensesColumn - Specifies the identifier of the custom column that will store the full number of licenses (sum of available and used). To get the identifier:
- In the Report-specific columns section, on the Columns tab, right-click the custom column.
- In the context menu, navigate to Copy and click Column ID.
- The column identifier will be copied to clipboard.
- $availableLicensesColumn - Specifies the identifier of the custom column that will store the number of available licenses.
- $usedLicensesColumn - Specifies the identifier of the custom column that will store the number of used licenses.
- $tenantColumn - Specifies the identifier of the custom column that will store the name of the Microsoft 365 tenant.
- $licensseSKUs - Specifies the SKU part numbers of the licenses to be included into the report. To include all licenses ib all tenants, set the variable to an empty array.
PowerShell
$sumLicensesColumn = "{d69bd562-5ba2-4302-90da-02d27d4bd8a7}" # TODO: modify me
$availableLicensesColumn = "{e148141d-755f-4bc8-bf40-6e5f1cfc44ad}" # TODO: modify me
$usedLicensesColumn = "{c1d48810-9fdd-4de3-ab5e-d6c2c9245eba}" # TODO: modify me
$tenantColumn = "{98882249-54a0-4f99-be40-da24c1221c44}" # TODO: modify me
$licensseSKUs = @("DEVELOPERPACK_E5", "FLOW_FREE") # TODO: modify me
# Find all Microsoft 365 tenants
$configurationContainerPath = $Context.GetWellKnownContainerPath("CloudServicesO365")
$tenantSearcher = $Context.BindToObject($configurationContainerPath)
$tenantSearcher.Criteria = New-AdmCriteria "adm-O365Tenant"
$tenantSearcher.SearchScope = "ADS_SCOPE_SUBTREE"
try
{
# Execute search
$tenantSearchResultIterator = $tenantSearcher.ExecuteSearch()
$tenants = $tenantSearchResultIterator.FetchAll()
foreach ($tenantID in $tenants)
{
# Bind to the Tenant
$tenant = $Context.BindToObject($tenantID.AdsPath)
foreach ($sku in $tenant.Skus)
{
if ($licensseSKUs.Count -ne 0 -and $licensseSKUs -notcontains $sku.SkuPartNumber)
{
continue
}
# Get license plan display name
if (-not([System.String]::IsNullOrEmpty($sku.CustomDisplayName)))
{
$skuDisplayName = $sku.CustomDisplayName
}
else
{
$skuDisplayName = $sku.DefaultDisplayName
}
$Context.Items.Add(-1, $skuDisplayName, "License", @{
$sumLicensesColumn = $sku.TotalUnits;
$availableLicensesColumn = $sku.TotalUnits - $sku.ConsumedUnits;
$usedLicensesColumn = $sku.ConsumedUnits;
$tenantColumn = $tenant.TenantName
})
}
}
}
finally
{
# Release resources
if ($tenantSearchResultIterator){ $tenantSearchResultIterator.Dispose() }
}
$licenseNameToSkuPartNumber = @{}
It seems like it should be populated with data later but is not.
Thank you for pointing this out. You are right, the line is not required in the script. We removed it.
Is there a way to filter and display only the Active SKUs, excluding all Disabled, Expired, etc.?
Unfortunately, there is no such possibility.
Thank you!
Yes, we updated the script accordingly (the $licensseSKUs variable was added).