328 lines
15 KiB
HTML
328 lines
15 KiB
HTML
<div class="header-container">
|
|
<button mat-icon-button color="primary" (click)="iniciarTour()" matTooltip="Ayuda">
|
|
<mat-icon>help</mat-icon>
|
|
</button>
|
|
|
|
<div class="header-container-title">
|
|
<h2 joyrideStep="tracesTitleStep" text="{{ 'tracesTitleStepText' | translate }}">{{ 'adminCommandsTitle' |
|
|
translate }}</h2>
|
|
</div>
|
|
|
|
<div class="header-actions">
|
|
<button class="action-button secondary" (click)="exportToCSV()" matTooltip="Exportar a CSV">
|
|
<mat-icon>download</mat-icon>
|
|
{{ 'exportCSV' | translate }}
|
|
</button>
|
|
<button class="action-button" (click)="resetFilters(commandSearchInput, commandStatusInput, commandClientInput)"
|
|
joyrideStep="resetFiltersStep" text="{{ 'resetFiltersStepText' | translate }}">
|
|
<mat-icon>refresh</mat-icon>
|
|
{{ 'resetFilters' | translate }}
|
|
</button>
|
|
</div>
|
|
</div>
|
|
|
|
<div class="stats-container" *ngIf="!loading">
|
|
<div class="stat-card stat-total clickable"
|
|
[class.active-filter]="isFilterActive('total')"
|
|
(click)="filterByTotal()"
|
|
matTooltip="Ver todas las trazas">
|
|
<div class="stat-number">{{ totalStats.total }}</div>
|
|
<div class="stat-label">{{ 'totalTraces' | translate }}</div>
|
|
</div>
|
|
<div class="stat-card stat-today clickable"
|
|
[class.active-filter]="isFilterActive('today')"
|
|
(click)="filterByToday()"
|
|
matTooltip="Ver trazas de hoy">
|
|
<div class="stat-number">{{ getStatusCount('today') }}</div>
|
|
<div class="stat-label">{{ 'todayTraces' | translate }}</div>
|
|
</div>
|
|
<div class="stat-card stat-success clickable"
|
|
[class.active-filter]="isFilterActive('success')"
|
|
(click)="filterBySuccess()"
|
|
matTooltip="Ver trazas exitosas">
|
|
<div class="stat-number">{{ getStatusCount('success') }}</div>
|
|
<div class="stat-label">{{ 'successful' | translate }}</div>
|
|
</div>
|
|
<div class="stat-card stat-failed clickable"
|
|
[class.active-filter]="isFilterActive('failed')"
|
|
(click)="filterByFailed()"
|
|
matTooltip="Ver trazas fallidas">
|
|
<div class="stat-number">{{ getStatusCount('failed') }}</div>
|
|
<div class="stat-label">{{ 'failed' | translate }}</div>
|
|
</div>
|
|
<div class="stat-card stat-in-progress clickable"
|
|
[class.active-filter]="isFilterActive('in-progress')"
|
|
(click)="filterByInProgress()"
|
|
matTooltip="Ver trazas en progreso">
|
|
<div class="stat-number">{{ getStatusCount('in-progress') }}</div>
|
|
<div class="stat-label">{{ 'inProgress' | translate }}</div>
|
|
</div>
|
|
</div>
|
|
|
|
<div class="filters-section" joyrideStep="filtersStep" text="{{ 'filtersStepText' | translate }}">
|
|
<div class="filters-header">
|
|
<h3>{{ 'filters' | translate }}</h3>
|
|
<div class="filters-header-actions">
|
|
<div *ngIf="activeFilter" class="active-filter-indicator">
|
|
<mat-icon>filter_alt</mat-icon>
|
|
<span>Filtro activo: {{ getActiveFilterLabel() }}</span>
|
|
<button mat-icon-button (click)="resetFilters(commandSearchInput, commandStatusInput, commandClientInput)"
|
|
matTooltip="Limpiar filtros">
|
|
<mat-icon>clear</mat-icon>
|
|
</button>
|
|
</div>
|
|
<button mat-button color="primary" (click)="toggleFilters()">
|
|
<mat-icon>{{ showAdvancedFilters ? 'expand_less' : 'expand_more' }}</mat-icon>
|
|
{{ showAdvancedFilters ? 'hideAdvanced' : 'showAdvanced' | translate }}
|
|
</button>
|
|
</div>
|
|
</div>
|
|
|
|
<div class="search-container" [class.expanded]="showAdvancedFilters">
|
|
<div class="filter-row">
|
|
<mat-form-field appearance="fill" class="search-select">
|
|
<input type="text" matInput [formControl]="clientControl" [matAutocomplete]="clientAuto" #commandClientInput
|
|
placeholder="{{ 'filterClientPlaceholder' | translate }}">
|
|
<mat-autocomplete #clientAuto="matAutocomplete" [displayWith]="displayFnClient"
|
|
(optionSelected)="onOptionClientSelected($event.option.value)">
|
|
<mat-option *ngFor="let client of filteredClients | async" [value]="client">
|
|
<div class="client-option">
|
|
<span class="client-name">{{ client.name }}</span>
|
|
<span class="client-details">{{ client.ip }} — {{ client.mac }}</span>
|
|
</div>
|
|
</mat-option>
|
|
</mat-autocomplete>
|
|
<button *ngIf="commandClientInput.value" mat-icon-button matSuffix aria-label="Clear input search"
|
|
(click)="clearClientFilter($event, commandClientInput)">
|
|
<mat-icon>close</mat-icon>
|
|
</button>
|
|
<mat-hint>{{ 'enterClientName' | translate }}</mat-hint>
|
|
</mat-form-field>
|
|
|
|
<mat-form-field appearance="fill" class="search-select">
|
|
<mat-label>{{ 'commandSelectStepText' | translate }}</mat-label>
|
|
<mat-select (selectionChange)="onOptionCommandSelected($event.value)" #commandSearchInput>
|
|
<mat-option *ngFor="let command of filteredCommands2" [value]="command">
|
|
{{ translateCommand(command.name) }}
|
|
</mat-option>
|
|
</mat-select>
|
|
<button *ngIf="commandSearchInput.value" mat-icon-button matSuffix aria-label="Clear input search"
|
|
(click)="clearCommandFilter($event, commandSearchInput)">
|
|
<mat-icon>close</mat-icon>
|
|
</button>
|
|
</mat-form-field>
|
|
|
|
<mat-form-field appearance="fill" class="search-boolean">
|
|
<mat-label>{{ 'status' | translate }}</mat-label>
|
|
<mat-select (selectionChange)="onOptionStatusSelected($event.value)" placeholder="Seleccionar opción"
|
|
#commandStatusInput>
|
|
<mat-option [value]="'failed'">
|
|
<div class="status-option">
|
|
<div class="status-indicator failed"></div>
|
|
{{ 'failed' | translate }}
|
|
</div>
|
|
</mat-option>
|
|
<mat-option [value]="'pending'">
|
|
<div class="status-option">
|
|
<div class="status-indicator pending"></div>
|
|
{{ 'pending' | translate }}
|
|
</div>
|
|
</mat-option>
|
|
<mat-option [value]="'in-progress'">
|
|
<div class="status-option">
|
|
<div class="status-indicator in-progress"></div>
|
|
{{ 'inProgress' | translate }}
|
|
</div>
|
|
</mat-option>
|
|
<mat-option [value]="'success'">
|
|
<div class="status-option">
|
|
<div class="status-indicator success"></div>
|
|
{{ 'success' | translate }}
|
|
</div>
|
|
</mat-option>
|
|
<mat-option [value]="'cancelled'">
|
|
<div class="status-option">
|
|
<div class="status-indicator cancelled"></div>
|
|
{{ 'cancelled' | translate }}
|
|
</div>
|
|
</mat-option>
|
|
</mat-select>
|
|
<button *ngIf="commandStatusInput.value" mat-icon-button matSuffix aria-label="Clear input search"
|
|
(click)="clearStatusFilter($event, commandStatusInput)">
|
|
<mat-icon>close</mat-icon>
|
|
</button>
|
|
</mat-form-field>
|
|
</div>
|
|
|
|
<div class="filter-row advanced-filters" *ngIf="showAdvancedFilters">
|
|
<mat-form-field appearance="fill" class="search-date">
|
|
<mat-label>{{ 'fromDate' | translate }}</mat-label>
|
|
<input matInput [matDatepicker]="fromPicker" [(ngModel)]="filters['startDate']"
|
|
(dateChange)="onDateFilterChange()" [max]="today">
|
|
<mat-datepicker-toggle matSuffix [for]="fromPicker"></mat-datepicker-toggle>
|
|
<mat-datepicker #fromPicker></mat-datepicker>
|
|
</mat-form-field>
|
|
|
|
<mat-form-field appearance="fill" class="search-date">
|
|
<mat-label>{{ 'toDate' | translate }}</mat-label>
|
|
<input matInput [matDatepicker]="toPicker" [(ngModel)]="filters['endDate']" (dateChange)="onDateFilterChange()"
|
|
[max]="today">
|
|
<mat-datepicker-toggle matSuffix [for]="toPicker"></mat-datepicker-toggle>
|
|
<mat-datepicker #toPicker></mat-datepicker>
|
|
</mat-form-field>
|
|
|
|
<mat-form-field appearance="fill" class="search-select">
|
|
<mat-label>{{ 'sortBy' | translate }}</mat-label>
|
|
<mat-select [(ngModel)]="sortBy" (selectionChange)="onSortChange()">
|
|
<mat-option value="executedAt">{{ 'executionDate' | translate }}</mat-option>
|
|
<mat-option value="status">{{ 'status' | translate }}</mat-option>
|
|
<mat-option value="command">{{ 'command' | translate }}</mat-option>
|
|
<mat-option value="client">{{ 'client' | translate }}</mat-option>
|
|
</mat-select>
|
|
</mat-form-field>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
|
|
<app-loading [isLoading]="loading"></app-loading>
|
|
|
|
<div *ngIf="!loading" class="table-container">
|
|
<div class="table-header">
|
|
<div class="table-info">
|
|
<span>{{ 'showingResults' | translate: { from: getPaginationFrom(), to: getPaginationTo(), total: getPaginationTotal() } }}</span>
|
|
</div>
|
|
<div class="table-actions">
|
|
<button mat-icon-button (click)="refreshData()" matTooltip="{{ 'refresh' | translate }}">
|
|
<mat-icon>refresh</mat-icon>
|
|
</button>
|
|
</div>
|
|
</div>
|
|
|
|
<table mat-table [dataSource]="traces" class="mat-elevation-z8">
|
|
<ng-container *ngFor="let column of columns" [matColumnDef]="column.columnDef">
|
|
<th mat-header-cell *matHeaderCellDef>
|
|
<div class="column-header">
|
|
<span>{{ column.header }}</span>
|
|
<button *ngIf="column.sortable !== false" mat-icon-button (click)="sortColumn(column.columnDef)" class="sort-button">
|
|
<mat-icon>{{ getSortIcon(column.columnDef) }}</mat-icon>
|
|
</button>
|
|
</div>
|
|
</th>
|
|
<td mat-cell *matCellDef="let trace">
|
|
|
|
<ng-container [ngSwitch]="column.columnDef">
|
|
<ng-container *ngSwitchCase="'status'">
|
|
<ng-container *ngIf="trace.status === 'in-progress' && trace.progress; else statusChip">
|
|
<div class="progress-container">
|
|
<mat-progress-bar class="example-margin" [mode]="mode" [value]="trace.progress"
|
|
[bufferValue]="bufferValue">
|
|
</mat-progress-bar>
|
|
<span class="progress-text">{{trace.progress}}%</span>
|
|
<button mat-icon-button
|
|
(click)="cancelTrace(trace)" class="cancel-button" matTooltip="{{ 'cancelTask' | translate }}">
|
|
<mat-icon>cancel</mat-icon>
|
|
</button>
|
|
</div>
|
|
</ng-container>
|
|
<ng-template #statusChip>
|
|
<div class="status-progress-flex" joyrideStep="tracesProgressStep"
|
|
text="{{ 'tracesProgressStepText' | translate }}">
|
|
<mat-chip [ngClass]="{
|
|
'chip-failed': trace.status === 'failed',
|
|
'chip-success': trace.status === 'success',
|
|
'chip-pending': trace.status === 'pending',
|
|
'chip-in-progress': trace.status === 'in-progress',
|
|
'chip-cancelled': trace.status === 'cancelled'
|
|
}">
|
|
{{
|
|
trace.status === 'failed' ? ('failed' | translate) :
|
|
trace.status === 'in-progress' ? ('inProgress' | translate) :
|
|
trace.status === 'success' ? ('successful' | translate) :
|
|
trace.status === 'pending' ? ('pending' | translate) :
|
|
trace.status === 'cancelled' ? ('cancelled' | translate) :
|
|
trace.status
|
|
}}
|
|
</mat-chip>
|
|
<button *ngIf="trace.status === 'in-progress'" mat-icon-button
|
|
(click)="cancelTrace(trace)" class="cancel-button" matTooltip="{{ 'cancelTask' | translate }}">
|
|
<mat-icon>cancel</mat-icon>
|
|
</button>
|
|
</div>
|
|
</ng-template>
|
|
</ng-container>
|
|
|
|
<ng-container *ngSwitchCase="'command'">
|
|
<div class="command-cell">
|
|
<span class="command-name">{{ translateCommand(trace.command) }}</span>
|
|
<span class="command-id"
|
|
[class.clickable-job-id]="trace.status === 'in-progress'"
|
|
(click)="trace.status === 'in-progress' ? openLogsInNewTab(trace) : null"
|
|
[matTooltip]="trace.status === 'in-progress' ? 'Ver logs' : ''">
|
|
{{ trace.jobId }}
|
|
</span>
|
|
</div>
|
|
</ng-container>
|
|
|
|
<ng-container *ngSwitchCase="'client'">
|
|
<div class="client-cell">
|
|
<span class="client-name">{{ trace.client?.name }}</span>
|
|
<span class="client-ip">{{ trace.client?.ip }}</span>
|
|
</div>
|
|
</ng-container>
|
|
|
|
<ng-container *ngSwitchCase="'executedAt'">
|
|
<div class="date-cell">
|
|
<span class="date-time">{{ trace.executedAt |date: 'dd/MM/yyyy hh:mm:ss'}}</span>
|
|
<span class="date-relative" *ngIf="getRelativeTime(trace.executedAt)">{{ getRelativeTime(trace.executedAt) }}</span>
|
|
</div>
|
|
</ng-container>
|
|
|
|
<ng-container *ngSwitchCase="'finishedAt'">
|
|
<div class="date-cell">
|
|
<span class="date-time">{{ trace.finishedAt |date: 'dd/MM/yyyy hh:mm:ss'}}</span>
|
|
<span class="date-relative" *ngIf="getRelativeTime(trace.finishedAt)">{{ getRelativeTime(trace.finishedAt) }}</span>
|
|
</div>
|
|
</ng-container>
|
|
<ng-container *ngSwitchDefault>
|
|
{{ column.cell(trace) }}
|
|
</ng-container>
|
|
|
|
</ng-container>
|
|
</td>
|
|
</ng-container>
|
|
|
|
<ng-container matColumnDef="information">
|
|
<th mat-header-cell *matHeaderCellDef style="text-align: center;">{{ 'informationLabel' | translate }}</th>
|
|
<td mat-cell *matCellDef="let trace" style="text-align: center;" joyrideStep="tracesInfoStep"
|
|
text="{{ 'tracesInfoStepText' | translate }}">
|
|
<div class="action-buttons">
|
|
<button mat-icon-button color="primary" [disabled]="!trace.input || trace.input.length === 0"
|
|
(click)="openInputModal(trace.input)" matTooltip="{{ 'viewInput' | translate }}">
|
|
<mat-icon>
|
|
<span class="material-symbols-outlined">mode_comment</span>
|
|
</mat-icon>
|
|
</button>
|
|
<button mat-icon-button color="primary" [disabled]="!trace.output" (click)="openOutputModal(trace.output)"
|
|
matTooltip="{{ 'viewOutput' | translate }}">
|
|
<mat-icon>
|
|
<span class="material-symbols-outlined">info</span>
|
|
</mat-icon>
|
|
</button>
|
|
|
|
</div>
|
|
</td>
|
|
</ng-container>
|
|
|
|
<tr mat-header-row *matHeaderRowDef="displayedColumns"></tr>
|
|
<tr mat-row *matRowDef="let row; columns: displayedColumns;"
|
|
[class.selected-row]="selectedTrace?.id === row.id"
|
|
(click)="selectTrace(row)"></tr>
|
|
</table>
|
|
</div>
|
|
|
|
<div class="paginator-container" joyrideStep="paginationStep" text="{{ 'paginationStepText' | translate }}">
|
|
<mat-paginator [length]="length" [pageSize]="itemsPerPage" [pageIndex]="page" [pageSizeOptions]="pageSizeOptions"
|
|
(page)="onPageChange($event)">
|
|
</mat-paginator>
|
|
</div>
|