
* 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>
104 lines
3.6 KiB
TypeScript
104 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 { CSSResult, TemplateResult, css, html } from "lit";
|
|
import { customElement, property } from "lit/decorators.js";
|
|
|
|
import PFCard from "@patternfly/patternfly/components/Card/card.css";
|
|
|
|
import { Event, EventsApi } from "@goauthentik/api";
|
|
|
|
@customElement("ak-recent-events")
|
|
export class RecentEventsCard extends Table<Event> {
|
|
@property()
|
|
order = "-created";
|
|
|
|
@property({ type: Number })
|
|
pageSize = 10;
|
|
|
|
async apiEndpoint(): Promise<PaginatedResponse<Event>> {
|
|
return new EventsApi(DEFAULT_CONFIG).eventsEventsList({
|
|
...(await this.defaultEndpointConfig()),
|
|
pageSize: this.pageSize,
|
|
});
|
|
}
|
|
|
|
static get styles(): CSSResult[] {
|
|
return super.styles.concat(
|
|
PFCard,
|
|
css`
|
|
.pf-c-card__title {
|
|
--pf-c-card__title--FontFamily: var(
|
|
--pf-global--FontFamily--heading--sans-serif
|
|
);
|
|
--pf-c-card__title--FontSize: var(--pf-global--FontSize--md);
|
|
--pf-c-card__title--FontWeight: var(--pf-global--FontWeight--bold);
|
|
}
|
|
* {
|
|
word-break: break-all;
|
|
}
|
|
`,
|
|
);
|
|
}
|
|
|
|
columns(): TableColumn[] {
|
|
return [
|
|
new TableColumn(msg("Action"), "action"),
|
|
new TableColumn(msg("User"), "user"),
|
|
new TableColumn(msg("Creation Date"), "created"),
|
|
new TableColumn(msg("Client IP"), "client_ip"),
|
|
new TableColumn(msg("Brand"), "brand_name"),
|
|
];
|
|
}
|
|
|
|
renderToolbar(): TemplateResult {
|
|
return html`<div class="pf-c-card__title">
|
|
<i class="pf-icon pf-icon-catalog"></i> ${msg("Recent events")}
|
|
</div>`;
|
|
}
|
|
|
|
row(item: EventWithContext): SlottedTemplateResult[] {
|
|
return [
|
|
html`<div><a href="${`#/events/log/${item.pk}`}">${actionToLabel(item.action)}</a></div>
|
|
<small>${item.app}</small>`,
|
|
renderEventUser(item),
|
|
html`<div>${formatElapsedTime(item.created)}</div>
|
|
<small>${item.created.toLocaleString()}</small>`,
|
|
html` <div>${item.clientIp || msg("-")}</div>
|
|
<small>${EventGeo(item)}</small>`,
|
|
html`<span>${item.brand?.name || msg("-")}</span>`,
|
|
];
|
|
}
|
|
|
|
renderEmpty(inner?: SlottedTemplateResult): TemplateResult {
|
|
if (this.error) {
|
|
return super.renderEmpty(inner);
|
|
}
|
|
|
|
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-recent-events": RecentEventsCard;
|
|
}
|
|
}
|