Exchange Online custom RBAC Role with App Authentication (OAuth2)
Hi All,
I've already blogged about Exchange Online PowerShell V2 and Modern Auth with Application in AzureAD (Step-by-step)
- Azure AD Application Registration
- Permission: Office 365 Exchange Online > Exchange.ManageAsApp
- Add AAD App to Azure AD Role "Exchange Administrator" or "Exchange Recipient Administrator"
In this Article it's about custom RBAC Roles and Application Authentication
- Azure AD Application Registration
- Permission: Office 365 Exchange Online > Exchange.ManageAsApp
- Create a Exchange Online Service Principal for the Azure AD App
- Create a RoleGroup and Assign ServicePrincipal (and CustomRecipientWriteScope if you want)
Notes from the field: Using app-only authentication with customized RBAC roles in Exchange Online
I've created an Azure AD App Registration
data:image/s3,"s3://crabby-images/07f9d/07f9df3bd001903ed6d3c021bb09c953d6832a65" alt=""
Uploaded a Certificate
data:image/s3,"s3://crabby-images/116a6/116a61c6b286f091eef55acf0b42f2b1d336ecf0" alt=""
Add Permission
Add Permission > APIs my organization uses > Office 365 Exchange Online > Exchange.ManageAsApp
data:image/s3,"s3://crabby-images/fd5cf/fd5cfaf939b47fdb7073deab4ee048e66ddf903a" alt=""
Grant Admin Consent
data:image/s3,"s3://crabby-images/fdff3/fdff3423b90e078843e735177fd838c8812d356f" alt=""
You need to create an Exchange Service Principal.
For that you need the ObjectID of the Enterprise Application
data:image/s3,"s3://crabby-images/192f1/192f139357fe30627473a75d34709b120aeab728" alt=""
###############################################################################
# Get AzureAD Application with Microsoft.Graph PowerShell
###############################################################################
Connect-MgGraph -Scopes 'Application.Read.All'
$ServicePrincipalDetails = Get-MgServicePrincipal -Filter "DisplayName eq 'Demo-EXO-RBAC-PS'"
$ServicePrincipalDetails
###############################################################################
# Create Exchange Service Principal
###############################################################################
Connect-ExchangeOnline
New-ServicePrincipal -AppId $ServicePrincipalDetails.AppId -ServiceId $ServicePrincipalDetails.Id -DisplayName "EXO Serviceprincipal $($ServicePrincipalDetails.Displayname)"
Get-ServicePrincipal | where {$_.AppId -eq "341772e9-4f7a-4444-9b2c-66620d27aec0"}
# Get AzureAD Application with Microsoft.Graph PowerShell
###############################################################################
Connect-MgGraph -Scopes 'Application.Read.All'
$ServicePrincipalDetails = Get-MgServicePrincipal -Filter "DisplayName eq 'Demo-EXO-RBAC-PS'"
$ServicePrincipalDetails
###############################################################################
# Create Exchange Service Principal
###############################################################################
Connect-ExchangeOnline
New-ServicePrincipal -AppId $ServicePrincipalDetails.AppId -ServiceId $ServicePrincipalDetails.Id -DisplayName "EXO Serviceprincipal $($ServicePrincipalDetails.Displayname)"
Get-ServicePrincipal | where {$_.AppId -eq "341772e9-4f7a-4444-9b2c-66620d27aec0"}
data:image/s3,"s3://crabby-images/a2145/a214528aaf86922cc2be24e7b561f0463e4d2a44" alt=""
Create the Management Scope. Please note that only the "Sitzungszimmer" Mailbox matches the Management Scope
###############################################################################
#Magagement Scope
###############################################################################
Filterable properties for the RecipientFilter parameter on Exchange cmdlets
https://learn.microsoft.com/en-us/powershell/exchange/recipientfilter-properties?view=exchange-ps
Get-ManagementScope
Get-Recipient -RecipientPreviewFilter "(City -eq 'Zürich') -and (RecipientTypeDetails -eq 'RoomMailbox')"
#Magagement Scope
###############################################################################
Filterable properties for the RecipientFilter parameter on Exchange cmdlets
https://learn.microsoft.com/en-us/powershell/exchange/recipientfilter-properties?view=exchange-ps
Get-ManagementScope
Get-Recipient -RecipientPreviewFilter "(City -eq 'Zürich') -and (RecipientTypeDetails -eq 'RoomMailbox')"
data:image/s3,"s3://crabby-images/c9510/c95100bc72f296af21870295f2d6c4502cfd4234" alt=""
I've already documented how to create an Management Role in Exchange and Exchange Online
###############################################################################
#Get-ManagementRole
###############################################################################
Get-ManagementRole -Identity "ICE-UserPhoto"
Get-ManagementRoleEntry -Identity "ICE-UserPhoto\*"
#Get-ManagementRole
###############################################################################
Get-ManagementRole -Identity "ICE-UserPhoto"
Get-ManagementRoleEntry -Identity "ICE-UserPhoto\*"
data:image/s3,"s3://crabby-images/0f740/0f740c2ebf18579c4f7386931d515d27968bdd35" alt=""
Now let's put it all together. Assign the App the Role "Ice-UserPhoto" and assign the ResourceScope "ZH Rooms"
###############################################################################
#New-RoleGroup
###############################################################################
$AppID = "341772e9-4f7a-4444-9b2c-66620d27aec0"
$SP = Get-ServicePrincipal | where {$_.AppId -eq $AppID}
$ServiceId = $SP.ServiceId
New-RoleGroup -Name 'Icewolf-UserPhoto' -Roles "ICE-UserPhoto" -CustomRecipientWriteScope "ZH Rooms"
Add-RoleGroupMember -Identity "Icewolf-UserPhoto" -Member $ServiceId
#New-RoleGroup
###############################################################################
$AppID = "341772e9-4f7a-4444-9b2c-66620d27aec0"
$SP = Get-ServicePrincipal | where {$_.AppId -eq $AppID}
$ServiceId = $SP.ServiceId
New-RoleGroup -Name 'Icewolf-UserPhoto' -Roles "ICE-UserPhoto" -CustomRecipientWriteScope "ZH Rooms"
Add-RoleGroupMember -Identity "Icewolf-UserPhoto" -Member $ServiceId
data:image/s3,"s3://crabby-images/1c71b/1c71b65274eb71a6c0d80b6c59e1aa737034b3f8" alt=""
Note that this Action will trigger an Alert
data:image/s3,"s3://crabby-images/c4b9a/c4b9a1fb17dcc3555a5cac67a722f463c556ae9d" alt=""
###############################################################################
#Get-RoleGroup
###############################################################################
$AppID = "341772e9-4f7a-4444-9b2c-66620d27aec0"
$SP = Get-ServicePrincipal | where {$_.AppId -eq $AppID}
$ServiceId = $SP.ServiceId
Get-RoleGroup | where {$_.Members -Match $ServiceId} | fl
#Get-RoleGroup
###############################################################################
$AppID = "341772e9-4f7a-4444-9b2c-66620d27aec0"
$SP = Get-ServicePrincipal | where {$_.AppId -eq $AppID}
$ServiceId = $SP.ServiceId
Get-RoleGroup | where {$_.Members -Match $ServiceId} | fl
data:image/s3,"s3://crabby-images/9d530/9d530b3a8cb342feee4156ae526337491d32bbb9" alt=""
###############################################################################
#Get-ManagementRoleAssignment
###############################################################################
Get-ManagementRoleAssignment | where {$_.Role -match "ICE-UserPhoto"} | fl
#Get-ManagementRoleAssignment
###############################################################################
Get-ManagementRoleAssignment | where {$_.Role -match "ICE-UserPhoto"} | fl
data:image/s3,"s3://crabby-images/9d530/9d530b3a8cb342feee4156ae526337491d32bbb9" alt=""
You can see the Permissions also in the Classic Exchange Online Admin Center
data:image/s3,"s3://crabby-images/d9ecd/d9ecdc11ab0c167ded6143d390562501c2736c40" alt=""
I recently noticed that there exists a new Menu Item Roles > Admin Roles in the Exchange Admin Center
data:image/s3,"s3://crabby-images/52a6b/52a6b1218ce0d2d0bddc927d879cbdd6f7073f44" alt=""
data:image/s3,"s3://crabby-images/bac09/bac094d364a82cea1bb9469c8c8f9828a7149b7c" alt=""
data:image/s3,"s3://crabby-images/fa972/fa97298d4bacd4e75f706a51d2f359e978c9b29a" alt=""
Let's connect with the App and the Certificate
###############################################################################
#Connect-ExchangeOnline with AppId and Certificate
###############################################################################
$AppID = "341772e9-4f7a-4444-9b2c-66620d27aec0"
$CertificateThumbprint = "07eff3918f47995eb53b91848f69b5c0e78622fd"
$TenantId = "icewolfch.onmicrosoft.com"
Connect-ExchangeOnline -AppId $AppID -CertificateThumbprint $CertificateThumbprint -Organization $TenantId
#Connect-ExchangeOnline with AppId and Certificate
###############################################################################
$AppID = "341772e9-4f7a-4444-9b2c-66620d27aec0"
$CertificateThumbprint = "07eff3918f47995eb53b91848f69b5c0e78622fd"
$TenantId = "icewolfch.onmicrosoft.com"
Connect-ExchangeOnline -AppId $AppID -CertificateThumbprint $CertificateThumbprint -Organization $TenantId
Get-ConnectionInformation
Get-Comand -Module <Module>
data:image/s3,"s3://crabby-images/f064a/f064ad2777f890cc262395e2648b04b25a485e02" alt=""
Let's try it. I will set a UserPhoto for a Mailbox that is inside the RecipientWriteScope
Set-UserPhoto -Identity Sitzungszimmer@icewolf.ch -PictureData ([System.IO.File]::ReadAllBytes("E:\Temp\AvatarBaby.jpg"))
Get-UserPhoto -Identity Sitzungszimmer@icewolf.ch
Get-UserPhoto -Identity Sitzungszimmer@icewolf.ch
data:image/s3,"s3://crabby-images/66e8f/66e8fc994d5efce9d985e88c61565691bd80fccf" alt=""
What threw me off, at first was that get works also for Mailboxes outside the Management Scope - but remember the Parameter is "CustomRecipientWriteScope"
Get-UserPhoto -Identity Sitzungszimmer@icewolf.ch
Get-UserPhoto -Identity a.bohren@icewolf.ch
data:image/s3,"s3://crabby-images/ecef1/ecef13acfe63793900f9e29aaf43c86e359487c8" alt=""
It's not possible to Set-UserPhoto for a Mailbox that is Outside the Management Scope / CustomRecipientWriteScope.
And you also get an Error, if no Picture is set on the Mailbox
Set-UserPhoto -Identity SitzungszimmerEiger@icewolf.ch -PictureData ([System.IO.File]::ReadAllBytes("E:\Temp\AvatarBaby.jpg"))
Get-UserPhoto -Identity SitzungszimmerEiger
data:image/s3,"s3://crabby-images/e8fda/e8fdaf5921e0f2de21404488f890e26d2f8c1435" alt=""
Regards
Andres Bohren
data:image/s3,"s3://crabby-images/41807/418077381cbae835a1a7d22a0fcf1948f7a73b5b" alt=""