Merge pull request 'Add report logos and configurable folder scan depth' (#2) from feat/report-logos-and-scan-depth into main
Reviewed-on: #2
This commit is contained in:
@@ -15,10 +15,12 @@
|
||||
|
||||
<div class="card">
|
||||
<div class="card-title">Scan Options</div>
|
||||
<div class="form-row">
|
||||
<div class="form-group">
|
||||
<label class="form-label">Site URL</label>
|
||||
<input class="form-input" @bind="_siteUrl" placeholder="@Session.CurrentProfile!.TenantUrl" />
|
||||
<SitePicker Profile="Session.CurrentProfile!" @bind-SelectedSites="_sites" />
|
||||
<div class="form-row mt-8">
|
||||
<div class="form-group" style="max-width:220px">
|
||||
<label class="form-label">Folder scan depth</label>
|
||||
<input class="form-input" type="number" min="0" max="20" @bind="_folderDepth" />
|
||||
<small class="text-muted">0 = libraries only. 1+ = drill into subfolders that many levels deep.</small>
|
||||
</div>
|
||||
</div>
|
||||
<div class="form-row">
|
||||
@@ -32,9 +34,10 @@
|
||||
<button class="btn btn-primary" @onclick="RunScan" disabled="@_running">
|
||||
@(_running ? "Scanning…" : "Scan Storage")
|
||||
</button>
|
||||
@if (_sites.Count > 0) { <span class="text-muted" style="align-self:center">@_sites.Count site(s) selected</span> }
|
||||
@if (_running) { <button class="btn btn-secondary" @onclick="Cancel">Cancel</button> }
|
||||
</div>
|
||||
<ProgressPanel IsRunning="_running" StatusMessage="_status" Current="_current" Total="_total" />
|
||||
<ProgressPanel IsRunning="_running" StatusMessage="@_status" Current="_current" Total="_total" />
|
||||
</div>
|
||||
|
||||
@if (!string.IsNullOrEmpty(_error)) { <div class="alert alert-error">@_error</div> }
|
||||
@@ -45,6 +48,7 @@
|
||||
<div class="flex-row">
|
||||
<div class="card-title">Storage Report <span class="count-badge">@_results.Count libraries</span></div>
|
||||
<div class="spacer"></div>
|
||||
<MergeModeSelect Value="_mergeMode" ValueChanged="v => _mergeMode = v" Visible="_bySite.Count > 1" />
|
||||
<button class="btn btn-secondary btn-sm" @onclick="ExportCsv">Export CSV</button>
|
||||
<button class="btn btn-secondary btn-sm" @onclick="ExportHtml">Export HTML</button>
|
||||
</div>
|
||||
@@ -79,28 +83,43 @@
|
||||
}
|
||||
|
||||
@code {
|
||||
private string _siteUrl = string.Empty;
|
||||
private List<SiteInfo> _sites = new();
|
||||
private bool _includeSubsites, _includeHidden = true, _includeRecycleBin = true;
|
||||
private int _folderDepth;
|
||||
private bool _running; private string _status = string.Empty, _error = string.Empty;
|
||||
private int _current, _total;
|
||||
private List<StorageNode> _results = new();
|
||||
private List<(string Label, IReadOnlyList<StorageNode> Results)> _bySite = new();
|
||||
private ReportMergeMode _mergeMode = ReportMergeMode.SingleMerged;
|
||||
private CancellationTokenSource? _cts;
|
||||
|
||||
private async Task RunScan()
|
||||
{
|
||||
_error = string.Empty; _results.Clear(); _running = true;
|
||||
_error = string.Empty; _results = new(); _bySite = new(); _running = true;
|
||||
_cts = new CancellationTokenSource();
|
||||
var siteUrl = string.IsNullOrWhiteSpace(_siteUrl) ? Session.CurrentProfile!.TenantUrl : _siteUrl.Trim();
|
||||
if (_sites.Count == 0) { _error = "Please select at least one site."; _running = false; return; }
|
||||
var progress = new Progress<OperationProgress>(p => { _status = p.Message; _current = p.Current; _total = p.Total; InvokeAsync(StateHasChanged); });
|
||||
try
|
||||
{
|
||||
var opts = new StorageScanOptions(IncludeSubsites: _includeSubsites, IncludeHiddenLibraries: _includeHidden, IncludeRecycleBin: _includeRecycleBin);
|
||||
_results = (await Elevation.RunAsync(async c =>
|
||||
var opts = new StorageScanOptions(IncludeSubsites: _includeSubsites, FolderDepth: Math.Clamp(_folderDepth, 0, 20), IncludeHiddenLibraries: _includeHidden, IncludeRecycleBin: _includeRecycleBin);
|
||||
var bySite = new List<(string, IReadOnlyList<StorageNode>)>();
|
||||
var flat = new List<StorageNode>();
|
||||
int i = 0;
|
||||
foreach (var site in _sites)
|
||||
{
|
||||
var ctx = await SessionMgr.GetOrCreateContextAsync(siteUrl, Session.CurrentProfile!, c);
|
||||
return await StorageSvc.CollectStorageAsync(ctx, opts, progress, c);
|
||||
}, _cts.Token)).ToList();
|
||||
_status = $"Complete: {_results.Count} nodes.";
|
||||
_cts.Token.ThrowIfCancellationRequested();
|
||||
_status = $"Scanning {site.Title} ({++i}/{_sites.Count})…";
|
||||
await InvokeAsync(StateHasChanged);
|
||||
var nodes = await Elevation.RunAsync(async c =>
|
||||
{
|
||||
var ctx = await SessionMgr.GetOrCreateContextAsync(site.Url, Session.CurrentProfile!, c);
|
||||
return await StorageSvc.CollectStorageAsync(ctx, opts, progress, c);
|
||||
}, _cts.Token);
|
||||
bySite.Add((site.Title, nodes));
|
||||
flat.AddRange(nodes);
|
||||
}
|
||||
_bySite = bySite; _results = flat;
|
||||
_status = $"Complete: {_results.Count} nodes across {_sites.Count} site(s).";
|
||||
}
|
||||
catch (OperationCanceledException) { _status = "Cancelled."; }
|
||||
catch (Exception ex) { _error = ex.Message; }
|
||||
@@ -111,12 +130,16 @@
|
||||
|
||||
private async Task ExportCsv()
|
||||
{
|
||||
var csv = CsvExport.BuildCsv(_results);
|
||||
await WebExport.DownloadCsvAsync(csv, $"storage_{DateTime.Now:yyyyMMdd_HHmmss}.csv");
|
||||
var ts = DateTime.Now.ToString("yyyyMMdd_HHmmss");
|
||||
var output = ReportMergeHelper.Build(_bySite, _mergeMode, "storage", ts, ReportFormat.Csv,
|
||||
rs => CsvExport.BuildCsv(rs));
|
||||
await WebExport.DownloadBytesAsync(output.Bytes, output.FileName, output.Mime);
|
||||
}
|
||||
private async Task ExportHtml()
|
||||
{
|
||||
var html = HtmlExport.BuildHtml(_results);
|
||||
await WebExport.DownloadHtmlAsync(html, $"storage_{DateTime.Now:yyyyMMdd_HHmmss}.html");
|
||||
var ts = DateTime.Now.ToString("yyyyMMdd_HHmmss");
|
||||
var output = ReportMergeHelper.Build(_bySite, _mergeMode, "storage", ts, ReportFormat.Html,
|
||||
rs => HtmlExport.BuildHtml(rs, Session.CurrentBranding));
|
||||
await WebExport.DownloadBytesAsync(output.Bytes, output.FileName, output.Mime);
|
||||
}
|
||||
}
|
||||
|
||||
Reference in New Issue
Block a user