|
// Licensed to the .NET Foundation under one or more agreements.
// The .NET Foundation licenses this file to you under the MIT license.
using System.Globalization;
using Aspire.Dashboard.Components.Dialogs;
using Aspire.Dashboard.Components.Layout;
using Aspire.Dashboard.Configuration;
using Aspire.Dashboard.Extensions;
using Aspire.Dashboard.Model;
using Aspire.Dashboard.Model.Otlp;
using Aspire.Dashboard.Otlp.Model;
using Aspire.Dashboard.Otlp.Storage;
using Aspire.Dashboard.Resources;
using Aspire.Dashboard.Utils;
using Microsoft.AspNetCore.Components;
using Microsoft.Extensions.Options;
using Microsoft.FluentUI.AspNetCore.Components;
using Microsoft.JSInterop;
namespace Aspire.Dashboard.Components.Pages;
public partial class StructuredLogs : IPageWithSessionAndUrlState<StructuredLogs.StructuredLogsPageViewModel, StructuredLogs.StructuredLogsPageState>
{
private const string ResourceColumn = nameof(ResourceColumn);
private const string LogLevelColumn = nameof(LogLevelColumn);
private const string TimestampColumn = nameof(TimestampColumn);
private const string MessageColumn = nameof(MessageColumn);
private const string TraceColumn = nameof(TraceColumn);
private const string ActionsColumn = nameof(ActionsColumn);
private SelectViewModel<ResourceTypeDetails> _allApplication = default!;
private TotalItemsFooter _totalItemsFooter = default!;
private int _totalItemsCount;
private List<OtlpApplication> _applications = default!;
private List<SelectViewModel<ResourceTypeDetails>> _applicationViewModels = default!;
private List<SelectViewModel<LogLevel?>> _logLevels = default!;
private Subscription? _applicationsSubscription;
private Subscription? _logsSubscription;
private bool _applicationChanged;
private string? _elementIdBeforeDetailsViewOpened;
private AspirePageContentLayout? _contentLayout;
private string _filter = string.Empty;
private FluentDataGrid<OtlpLogEntry> _dataGrid = null!;
private GridColumnManager _manager = null!;
private IList<GridColumn> _gridColumns = null!;
private ColumnResizeLabels _resizeLabels = ColumnResizeLabels.Default;
private ColumnSortLabels _sortLabels = ColumnSortLabels.Default;
public string BasePath => DashboardUrls.StructuredLogsBasePath;
public string SessionStorageKey => BrowserStorageKeys.StructuredLogsPageState;
public StructuredLogsPageViewModel PageViewModel { get; set; } = null!;
[Inject]
public required TelemetryRepository TelemetryRepository { get; init; }
[Inject]
public required StructuredLogsViewModel ViewModel { get; init; }
[Inject]
public required IDialogService DialogService { get; init; }
[Inject]
public required ISessionStorage SessionStorage { get; init; }
[Inject]
public required NavigationManager NavigationManager { get; init; }
[Inject]
public required BrowserTimeProvider TimeProvider { get; init; }
[Inject]
public required ILogger<StructuredLogs> Logger { get; init; }
[Inject]
public required DimensionManager DimensionManager { get; set; }
[Inject]
public required IOptions<DashboardOptions> DashboardOptions { get; init; }
[Inject]
public required IMessageService MessageService { get; init; }
[CascadingParameter]
public required ViewportInformation ViewportInformation { get; set; }
[Parameter]
public string? ApplicationName { get; set; }
[Parameter]
[SupplyParameterFromQuery]
public string? TraceId { get; set; }
[Parameter]
[SupplyParameterFromQuery]
public string? SpanId { get; set; }
[Parameter]
[SupplyParameterFromQuery(Name = "logLevel")]
public string? LogLevelText { get; set; }
[Parameter]
[SupplyParameterFromQuery(Name = "filters")]
public string? SerializedFilters { get; set; }
public StructureLogsDetailsViewModel? SelectedLogEntry { get; set; }
private async ValueTask<GridItemsProviderResult<OtlpLogEntry>> GetData(GridItemsProviderRequest<OtlpLogEntry> request)
{
ViewModel.StartIndex = request.StartIndex;
ViewModel.Count = request.Count ?? DashboardUIHelpers.DefaultDataGridResultCount;
var logs = ViewModel.GetLogs();
if (DashboardOptions.Value.TelemetryLimits.MaxLogCount == logs.TotalItemCount && !TelemetryRepository.HasDisplayedMaxLogLimitMessage)
{
await MessageService.ShowMessageBarAsync(options =>
{
options.Title = Loc[nameof(Dashboard.Resources.StructuredLogs.MessageExceededLimitTitle)];
options.Body = string.Format(CultureInfo.InvariantCulture, Loc[nameof(Dashboard.Resources.StructuredLogs.MessageExceededLimitBody)], DashboardOptions.Value.TelemetryLimits.MaxLogCount);
options.Intent = MessageIntent.Info;
options.Section = "MessagesTop";
options.AllowDismiss = true;
});
TelemetryRepository.HasDisplayedMaxLogLimitMessage = true;
}
// Updating the total item count as a field doesn't work because it isn't updated with the grid.
// The workaround is to explicitly update and refresh the control.
_totalItemsCount = logs.TotalItemCount;
_totalItemsFooter.UpdateDisplayedCount(_totalItemsCount);
TelemetryRepository.MarkViewedErrorLogs(ViewModel.ApplicationKey);
return GridItemsProviderResult.From(logs.Items, logs.TotalItemCount);
}
protected override Task OnInitializedAsync()
{
(_resizeLabels, _sortLabels) = DashboardUIHelpers.CreateGridLabels(ControlsStringsLoc);
_gridColumns = [
new GridColumn(Name: ResourceColumn, DesktopWidth: "2fr", MobileWidth: "1fr"),
new GridColumn(Name: LogLevelColumn, DesktopWidth: "1fr"),
new GridColumn(Name: TimestampColumn, DesktopWidth: "1.5fr"),
new GridColumn(Name: MessageColumn, DesktopWidth: "5fr", "2.5fr"),
new GridColumn(Name: TraceColumn, DesktopWidth: "1fr"),
new GridColumn(Name: ActionsColumn, DesktopWidth: "1fr", MobileWidth: "0.8fr")
];
if (!string.IsNullOrEmpty(TraceId))
{
ViewModel.AddFilter(new TelemetryFilter
{
Field = KnownStructuredLogFields.TraceIdField, Condition = FilterCondition.Equals, Value = TraceId
});
}
if (!string.IsNullOrEmpty(SpanId))
{
ViewModel.AddFilter(new TelemetryFilter
{
Field = KnownStructuredLogFields.SpanIdField, Condition = FilterCondition.Equals, Value = SpanId
});
}
_allApplication = new()
{
Id = null,
Name = ControlsStringsLoc[nameof(Dashboard.Resources.ControlsStrings.LabelAll)]
};
_logLevels = new List<SelectViewModel<LogLevel?>>
{
new SelectViewModel<LogLevel?> { Id = null, Name = ControlsStringsLoc[nameof(Dashboard.Resources.ControlsStrings.LabelAll)] },
new SelectViewModel<LogLevel?> { Id = LogLevel.Trace, Name = "Trace" },
new SelectViewModel<LogLevel?> { Id = LogLevel.Debug, Name = "Debug" },
new SelectViewModel<LogLevel?> { Id = LogLevel.Information, Name = "Information" },
new SelectViewModel<LogLevel?> { Id = LogLevel.Warning, Name = "Warning" },
new SelectViewModel<LogLevel?> { Id = LogLevel.Error, Name = "Error" },
new SelectViewModel<LogLevel?> { Id = LogLevel.Critical, Name = "Critical" },
};
PageViewModel = new StructuredLogsPageViewModel
{
SelectedLogLevel = _logLevels[0],
SelectedApplication = _allApplication
};
UpdateApplications();
_applicationsSubscription = TelemetryRepository.OnNewApplications(() => InvokeAsync(() =>
{
UpdateApplications();
StateHasChanged();
}));
return Task.CompletedTask;
}
protected override async Task OnParametersSetAsync()
{
if (await this.InitializeViewModelAsync())
{
return;
}
UpdateSubscription();
}
private void UpdateApplications()
{
_applications = TelemetryRepository.GetApplications();
_applicationViewModels = ApplicationsSelectHelpers.CreateApplications(_applications);
_applicationViewModels.Insert(0, _allApplication);
}
private Task HandleSelectedApplicationChangedAsync()
{
_applicationChanged = true;
return this.AfterViewModelChangedAsync(_contentLayout, waitToApplyMobileChange: true);
}
private async Task HandleSelectedLogLevelChangedAsync()
{
_applicationChanged = true;
await ClearSelectedLogEntryAsync();
await this.AfterViewModelChangedAsync(_contentLayout, waitToApplyMobileChange: true);
}
private void UpdateSubscription()
{
// Subscribe to updates.
if (_logsSubscription is null || _logsSubscription.ApplicationKey != PageViewModel.SelectedApplication.Id?.GetApplicationKey())
{
_logsSubscription?.Dispose();
_logsSubscription = TelemetryRepository.OnNewLogs(PageViewModel.SelectedApplication.Id?.GetApplicationKey(), SubscriptionType.Read, async () =>
{
ViewModel.ClearData();
await InvokeAsync(_dataGrid.SafeRefreshDataAsync);
});
}
}
private async Task OnShowPropertiesAsync(OtlpLogEntry entry, string? buttonId)
{
_elementIdBeforeDetailsViewOpened = buttonId;
if (SelectedLogEntry?.LogEntry.InternalId == entry.InternalId)
{
await ClearSelectedLogEntryAsync();
}
else
{
var logEntryViewModel = new StructureLogsDetailsViewModel
{
LogEntry = entry
};
SelectedLogEntry = logEntryViewModel;
}
}
private async Task ClearSelectedLogEntryAsync(bool causedByUserAction = false)
{
SelectedLogEntry = null;
if (_elementIdBeforeDetailsViewOpened is not null && causedByUserAction)
{
await JS.InvokeVoidAsync("focusElement", _elementIdBeforeDetailsViewOpened);
}
_elementIdBeforeDetailsViewOpened = null;
}
private async Task OpenFilterAsync(TelemetryFilter? entry)
{
if (_contentLayout is not null)
{
await _contentLayout.CloseMobileToolbarAsync();
}
var title = entry is not null ? FilterLoc[nameof(StructuredFiltering.DialogTitleEditFilter)] : FilterLoc[nameof(StructuredFiltering.DialogTitleAddFilter)];
var parameters = new DialogParameters
{
OnDialogResult = DialogService.CreateDialogCallback(this, HandleFilterDialog),
Title = title,
Alignment = HorizontalAlignment.Right,
PrimaryAction = null,
SecondaryAction = null,
Width = "450px"
};
var data = new FilterDialogViewModel
{
Filter = entry,
PropertyKeys = TelemetryRepository.GetLogPropertyKeys(PageViewModel.SelectedApplication.Id?.GetApplicationKey()),
KnownKeys = KnownStructuredLogFields.AllFields,
GetFieldValues = TelemetryRepository.GetLogsFieldValues
};
await DialogService.ShowPanelAsync<FilterDialog>(data, parameters);
}
private async Task HandleFilterDialog(DialogResult result)
{
if (result.Data is FilterDialogResult filterResult && filterResult.Filter is TelemetryFilter filter)
{
if (filterResult.Delete)
{
ViewModel.RemoveFilter(filter);
}
else if (filterResult.Add)
{
ViewModel.AddFilter(filter);
await ClearSelectedLogEntryAsync();
}
}
await this.AfterViewModelChangedAsync(_contentLayout, waitToApplyMobileChange: true);
}
private async Task HandleAfterFilterBindAsync()
{
ViewModel.FilterText = _filter;
await InvokeAsync(_dataGrid.SafeRefreshDataAsync);
if (string.IsNullOrEmpty(_filter))
{
return;
}
await ClearSelectedLogEntryAsync();
}
private string GetResourceName(OtlpApplicationView app) => OtlpApplication.GetResourceName(app.Application, _applications);
private string GetRowClass(OtlpLogEntry entry)
{
if (entry.InternalId == SelectedLogEntry?.LogEntry.InternalId)
{
return "selected-row";
}
else
{
return $"log-row-{entry.Severity.ToString().ToLowerInvariant()}";
}
}
protected override async Task OnAfterRenderAsync(bool firstRender)
{
if (_applicationChanged)
{
await JS.InvokeVoidAsync("resetContinuousScrollPosition");
_applicationChanged = false;
}
if (firstRender)
{
await JS.InvokeVoidAsync("initializeContinuousScroll");
DimensionManager.OnViewportInformationChanged += OnBrowserResize;
}
}
private void OnBrowserResize(object? o, EventArgs args)
{
InvokeAsync(async () =>
{
await JS.InvokeVoidAsync("resetContinuousScrollPosition");
await JS.InvokeVoidAsync("initializeContinuousScroll");
});
}
public void Dispose()
{
_applicationsSubscription?.Dispose();
_logsSubscription?.Dispose();
DimensionManager.OnViewportInformationChanged -= OnBrowserResize;
}
public string GetUrlFromSerializableViewModel(StructuredLogsPageState serializable)
{
var filters = (serializable.Filters.Count > 0) ? TelemetryFilterFormatter.SerializeFiltersToString(serializable.Filters) : null;
var url = DashboardUrls.StructuredLogsUrl(
resource: serializable.SelectedApplication,
logLevel: serializable.LogLevelText,
filters: filters);
return url;
}
public StructuredLogsPageState ConvertViewModelToSerializable()
{
return new StructuredLogsPageState
{
LogLevelText = PageViewModel.SelectedLogLevel.Id?.ToString().ToLowerInvariant(),
SelectedApplication = PageViewModel.SelectedApplication.Id is not null ? PageViewModel.SelectedApplication.Name : null,
Filters = ViewModel.Filters
};
}
public async Task UpdateViewModelFromQueryAsync(StructuredLogsPageViewModel viewModel)
{
viewModel.SelectedApplication = _applicationViewModels.GetApplication(Logger, ApplicationName, canSelectGrouping: true, _allApplication);
ViewModel.ApplicationKey = PageViewModel.SelectedApplication.Id?.GetApplicationKey();
if (LogLevelText is not null && Enum.TryParse<LogLevel>(LogLevelText, ignoreCase: true, out var logLevel))
{
PageViewModel.SelectedLogLevel = _logLevels.SingleOrDefault(e => e.Id == logLevel) ?? _logLevels[0];
}
else
{
PageViewModel.SelectedLogLevel = _logLevels[0];
}
ViewModel.LogLevel = PageViewModel.SelectedLogLevel.Id;
if (SerializedFilters is not null)
{
var filters = TelemetryFilterFormatter.DeserializeFiltersFromString(SerializedFilters);
if (filters.Count > 0)
{
ViewModel.ClearFilters();
foreach (var filter in filters)
{
ViewModel.AddFilter(filter);
}
}
}
await InvokeAsync(_dataGrid.SafeRefreshDataAsync);
}
private Task ClearStructureLogs(ApplicationKey? key)
{
TelemetryRepository.ClearStructuredLogs(key);
return Task.CompletedTask;
}
public class StructuredLogsPageViewModel
{
public required SelectViewModel<ResourceTypeDetails> SelectedApplication { get; set; }
public SelectViewModel<LogLevel?> SelectedLogLevel { get; set; } = default!;
}
public class StructuredLogsPageState
{
public string? SelectedApplication { get; set; }
public string? LogLevelText { get; set; }
public required IReadOnlyCollection<TelemetryFilter> Filters { get; set; }
}
}
|