
* unrelated: dont show nested for user Signed-off-by: Jens Langhammer <jens@goauthentik.io> * unrelated: fix error when no extents in. map Signed-off-by: Jens Langhammer <jens@goauthentik.io> * events: use pending_user when possible Signed-off-by: Jens Langhammer <jens@goauthentik.io> * fix for identification stage "fake" user Signed-off-by: Jens Langhammer <jens@goauthentik.io> * better username rendering Signed-off-by: Jens Langhammer <jens@goauthentik.io> --------- Signed-off-by: Jens Langhammer <jens@goauthentik.io>
109 lines
3.6 KiB
TypeScript
109 lines
3.6 KiB
TypeScript
import { EventGeo, renderEventUser } from "@goauthentik/admin/events/utils";
|
|
import { DEFAULT_CONFIG } from "@goauthentik/common/api/config";
|
|
import { EventWithContext } from "@goauthentik/common/events";
|
|
import { actionToLabel } from "@goauthentik/common/labels";
|
|
import { formatElapsedTime } from "@goauthentik/common/temporal";
|
|
import "@goauthentik/components/ak-event-info";
|
|
import "@goauthentik/elements/Tabs";
|
|
import "@goauthentik/elements/buttons/Dropdown";
|
|
import "@goauthentik/elements/buttons/ModalButton";
|
|
import "@goauthentik/elements/buttons/SpinnerButton";
|
|
import { PaginatedResponse } from "@goauthentik/elements/table/Table";
|
|
import { Table, TableColumn } from "@goauthentik/elements/table/Table";
|
|
import { SlottedTemplateResult } from "@goauthentik/elements/types";
|
|
|
|
import { msg } from "@lit/localize";
|
|
import { PropertyValues, TemplateResult, html } from "lit";
|
|
import { customElement, property } from "lit/decorators.js";
|
|
|
|
import { Event, EventsApi } from "@goauthentik/api";
|
|
|
|
@customElement("ak-object-changelog")
|
|
export class ObjectChangelog extends Table<Event> {
|
|
expandable = true;
|
|
|
|
@property()
|
|
order = "-created";
|
|
|
|
@property()
|
|
targetModelPk!: string | number;
|
|
|
|
@property()
|
|
targetModelApp?: string;
|
|
|
|
@property()
|
|
targetModelName = "";
|
|
|
|
async apiEndpoint(): Promise<PaginatedResponse<Event>> {
|
|
let modelName = this.targetModelName;
|
|
let appName = this.targetModelApp;
|
|
if (this.targetModelName.indexOf(".") !== -1) {
|
|
const parts = this.targetModelName.split(".", 1);
|
|
appName = parts[0];
|
|
modelName = parts[1];
|
|
}
|
|
if (this.targetModelName === "") {
|
|
return Promise.reject();
|
|
}
|
|
return new EventsApi(DEFAULT_CONFIG).eventsEventsList({
|
|
...(await this.defaultEndpointConfig()),
|
|
action: "model_",
|
|
contextModelApp: appName,
|
|
contextModelName: modelName,
|
|
contextModelPk: this.targetModelPk.toString(),
|
|
});
|
|
}
|
|
|
|
columns(): TableColumn[] {
|
|
return [
|
|
new TableColumn(msg("Action"), "action"),
|
|
new TableColumn(msg("User"), "enabled"),
|
|
new TableColumn(msg("Creation Date"), "created"),
|
|
new TableColumn(msg("Client IP"), "client_ip"),
|
|
];
|
|
}
|
|
|
|
willUpdate(changedProperties: PropertyValues<this>) {
|
|
if (changedProperties.has("targetModelName") && this.targetModelName) {
|
|
this.fetch();
|
|
}
|
|
}
|
|
|
|
row(item: EventWithContext): SlottedTemplateResult[] {
|
|
return [
|
|
html`${actionToLabel(item.action)}`,
|
|
renderEventUser(item),
|
|
html`<div>${formatElapsedTime(item.created)}</div>
|
|
<small>${item.created.toLocaleString()}</small>`,
|
|
html`<div>${item.clientIp || msg("-")}</div>
|
|
<small>${EventGeo(item)}</small>`,
|
|
];
|
|
}
|
|
|
|
renderExpanded(item: Event): TemplateResult {
|
|
return html` <td role="cell" colspan="4">
|
|
<div class="pf-c-table__expandable-row-content">
|
|
<ak-event-info .event=${item as EventWithContext}></ak-event-info>
|
|
</div>
|
|
</td>
|
|
<td></td>
|
|
<td></td>
|
|
<td></td>`;
|
|
}
|
|
|
|
renderEmpty(): TemplateResult {
|
|
return super.renderEmpty(
|
|
html`<ak-empty-state
|
|
><span>${msg("No Events found.")}</span>
|
|
<div slot="body">${msg("No matching events could be found.")}</div>
|
|
</ak-empty-state>`,
|
|
);
|
|
}
|
|
}
|
|
|
|
declare global {
|
|
interface HTMLElementTagNameMap {
|
|
"ak-object-changelog": ObjectChangelog;
|
|
}
|
|
}
|