@page "/user-audit" @attribute [Authorize] @inject IUserSessionService Session @inject ISessionManager SessionMgr @inject IUserAccessAuditService AuditSvc @inject UserAccessCsvExportService CsvExport @inject UserAccessHtmlExportService HtmlExport @inject WebExportService WebExport @rendermode InteractiveServer

User Access Audit

Find all permissions for one or more users across multiple sites.

@if (!Session.HasProfile) { return; }
@if (_running) { }
@if (!string.IsNullOrEmpty(_error)) {
@_error
} @if (_results.Count > 0) {
Audit Results @_results.Count
@foreach (var r in _results.Take(500)) { }
UserSiteObjectPermissionAccess TypeGranted Through
@r.UserDisplayName @r.SiteTitle @r.ObjectTitle (@r.ObjectType) @r.PermissionLevel @if (r.IsHighPrivilege) { High } @r.AccessType @r.GrantedThrough
@if (_results.Count > 500) {
Showing first 500. Export for full results.
}
} @code { private string _users = string.Empty, _sites = string.Empty; private bool _includeInherited, _includeSubsites, _scanFolders = true; private bool _running; private string _status = string.Empty, _error = string.Empty; private int _current, _total; private List _results = new(); private CancellationTokenSource? _cts; private async Task RunAudit() { _error = string.Empty; _results.Clear(); _running = true; _cts = new CancellationTokenSource(); var userList = _users.Split('\n', StringSplitOptions.RemoveEmptyEntries | StringSplitOptions.TrimEntries).ToList(); var siteList = _sites.Split('\n', StringSplitOptions.RemoveEmptyEntries | StringSplitOptions.TrimEntries) .Select(u => new SiteInfo(u, u.TrimEnd('/').Split('/').Last())).ToList(); if (!siteList.Any()) siteList.Add(new SiteInfo(Session.CurrentProfile!.TenantUrl, Session.CurrentProfile.Name)); var progress = new Progress(p => { _status = p.Message; _current = p.Current; _total = p.Total; InvokeAsync(StateHasChanged); }); try { var opts = new ScanOptions(_includeInherited, _scanFolders, 1, _includeSubsites); _results = (await AuditSvc.AuditUsersAsync(SessionMgr, Session.CurrentProfile!, userList, siteList, opts, progress, _cts.Token)).ToList(); _status = $"Found {_results.Count} access entries."; } catch (OperationCanceledException) { _status = "Cancelled."; } catch (Exception ex) { _error = ex.Message; } finally { _running = false; await InvokeAsync(StateHasChanged); } } private void Cancel() => _cts?.Cancel(); private async Task ExportCsv() { await WebExport.DownloadCsvAsync(CsvExport.BuildCsv(_results.FirstOrDefault()?.UserDisplayName ?? "Users", _results.FirstOrDefault()?.UserLogin ?? "", _results), $"user_audit_{DateTime.Now:yyyyMMdd_HHmmss}.csv"); } private async Task ExportHtml() { await WebExport.DownloadHtmlAsync(HtmlExport.BuildHtml(_results, branding: Session.CurrentBranding), $"user_audit_{DateTime.Now:yyyyMMdd_HHmmss}.html"); } }