Recently we have been running some Microsoft Graph API queries and were not getting back all the results expected.
Microsoft Graph seems to have a limit of 1000 results for signin logs, when the limit is reached graph will then start paging the result and adding them to @odata.nextLink property.
There is a Microsoft doc on paging in Graph.
Paging Microsoft Graph data in your app – Microsoft Graph | Microsoft Docs
There are different limits depending on the API.
Below is an extract from the paging doc.
Paging behavior varies across different Microsoft Graph APIs. Consider the following when working with paged data:
- Different APIs might have different default and maximum page sizes.
- Different APIs might behave differently if you specify a page size (via the
$top
query parameter) that exceeds the maximum page size for that API. Depending on the API, the requested page size might be ignored, it might default to the maximum page size for that API, or Microsoft Graph might return an error. - Not all resources or relationships support paging. For example, queries against directoryRoles do not support paging. This includes reading role objects themselves as well as role members.
The @odata.nextLink then needs to be called to get the addtional results.
When we run a query and the results are under the limit there will be two objects returned
@odata.context and value

If we run a query that goes over the max results limit of 1000 the value object is changed to @odata.nextLink

If we use count on the $Results variable it returns 996 items.
To query the addtional results we need to loop through each page to extract the results.
To loop through the results we can create a new variable with the @odata.nextLink and use while loop to go through the variable to get results from the pages.
When we do this the $results count go up to 1142

Below is the full script to get the addtional results.
$ApplicationID = ""
$TenatDomainName = ""
$AccessSecret = Read-Host "Enter Secret"
$Body = @{
Grant_Type = "client_credentials"
Scope = "https://graph.microsoft.com/.default"
client_Id = $ApplicationID
Client_Secret = $AccessSecret
}
$ConnectGraph = Invoke-RestMethod -Uri "https://login.microsoftonline.com/$TenatDomainName/oauth2/v2.0/token" `
-Method POST -Body $Body
$token = $ConnectGraph.access_token
$LoginUrl = "https://graph.microsoft.com/v1.0/auditLogs/signIns"
$LoginResults = Invoke-RestMethod -Headers @{Authorization = "Bearer $($Token)" } -Uri $LoginUrl -Method Get
$Results = @()
$Results += $LoginResults.value
$Pages = $LoginResults.'@odata.nextLink'
while($null -ne $Pages) {
Write-Warning "Checking Next page"
$Addtional = Invoke-RestMethod -Headers @{Authorization = "Bearer $($Token)" } -Uri $Pages -Method Get
if ($Pages){
$Pages = $Addtional."@odata.nextLink"
}
$Results += $Addtional.value
}
If using the Microsoft Graph SDK the process is much simpler.
Once connect to using Connect-Mgraph we can then run
Get-MgAuditLogSignIn |
Select-Object UserDisplayName,UserPrincipalName,IPAddress,AppDisplayName,ResourceDisplayName,ClientAppUsed,CreatedDateTime

The $signins results variable returns the same 996 results as we got with the above invoke-restmethod graph api query.
To get all the addtional results all we need to do is add the -All parameter and this should then return all results.

This shows the two different method to get the larger results from GraphAPI queries.