// FIXME: Update this file to be type safe and remove this and next line // @ts-strict-ignore import { Component, OnDestroy, OnInit } from "@angular/core"; import { ActivatedRoute, Router } from "@angular/router"; import { concatMap, firstValueFrom, lastValueFrom, Subject, takeUntil } from "rxjs"; import { OrganizationUserApiService } from "@bitwarden/admin-console/common"; import { UserNamePipe } from "@bitwarden/angular/pipes/user-name.pipe"; import { ApiService } from "@bitwarden/common/abstractions/api.service"; import { OrganizationApiServiceAbstraction } from "@bitwarden/common/admin-console/abstractions/organization/organization-api.service.abstraction"; import { getOrganizationById, OrganizationService, } from "@bitwarden/common/admin-console/abstractions/organization/organization.service.abstraction"; import { ProviderService } from "@bitwarden/common/admin-console/abstractions/provider.service"; import { Organization } from "@bitwarden/common/admin-console/models/domain/organization"; import { AccountService } from "@bitwarden/common/auth/abstractions/account.service"; import { getUserId } from "@bitwarden/common/auth/services/account.service"; import { ProductTierType } from "@bitwarden/common/billing/enums"; import { OrganizationSubscriptionResponse } from "@bitwarden/common/billing/models/response/organization-subscription.response"; import { EventSystemUser } from "@bitwarden/common/enums"; import { FeatureFlag } from "@bitwarden/common/enums/feature-flag.enum"; import { EventResponse } from "@bitwarden/common/models/response/event.response"; import { EventView } from "@bitwarden/common/models/view/event.view"; import { ConfigService } from "@bitwarden/common/platform/abstractions/config/config.service"; import { FileDownloadService } from "@bitwarden/common/platform/abstractions/file-download/file-download.service"; import { I18nService } from "@bitwarden/common/platform/abstractions/i18n.service"; import { LogService } from "@bitwarden/common/platform/abstractions/log.service"; import { PlatformUtilsService } from "@bitwarden/common/platform/abstractions/platform-utils.service"; import { DialogService, ToastService } from "@bitwarden/components"; import { ChangePlanDialogResultType, openChangePlanDialog, } from "../../../billing/organizations/change-plan-dialog.component"; import { EventService } from "../../../core"; import { EventExportService } from "../../../tools/event-export"; import { BaseEventsComponent } from "../../common/base.events.component"; import { placeholderEvents } from "./placeholder-events"; const EVENT_SYSTEM_USER_TO_TRANSLATION: Record = { [EventSystemUser.SCIM]: null, // SCIM acronym not able to be translated so just display SCIM [EventSystemUser.DomainVerification]: "domainVerification", [EventSystemUser.PublicApi]: "publicApi", }; @Component({ selector: "app-org-events", templateUrl: "events.component.html", }) export class EventsComponent extends BaseEventsComponent implements OnInit, OnDestroy { exportFileName = "org-events"; organizationId: string; organization: Organization; organizationSubscription: OrganizationSubscriptionResponse; placeholderEvents = placeholderEvents as EventView[]; private orgUsersUserIdMap = new Map(); private destroy$ = new Subject(); readonly ProductTierType = ProductTierType; protected isBreadcrumbEventLogsEnabled$ = this.configService.getFeatureFlag$( FeatureFlag.PM12276_BreadcrumbEventLogs, ); constructor( private apiService: ApiService, private route: ActivatedRoute, eventService: EventService, i18nService: I18nService, exportService: EventExportService, platformUtilsService: PlatformUtilsService, private router: Router, logService: LogService, private userNamePipe: UserNamePipe, private organizationService: OrganizationService, private organizationUserApiService: OrganizationUserApiService, private organizationApiService: OrganizationApiServiceAbstraction, private providerService: ProviderService, fileDownloadService: FileDownloadService, toastService: ToastService, private accountService: AccountService, private dialogService: DialogService, private configService: ConfigService, ) { super( eventService, i18nService, exportService, platformUtilsService, logService, fileDownloadService, toastService, ); } async ngOnInit() { const userId = await firstValueFrom(getUserId(this.accountService.activeAccount$)); this.route.params .pipe( concatMap(async (params) => { this.organizationId = params.organizationId; this.organization = await firstValueFrom( this.organizationService .organizations$(userId) .pipe(getOrganizationById(this.organizationId)), ); if (!this.organization.useEvents) { this.eventsForm.get("start").disable(); this.eventsForm.get("end").disable(); this.organizationSubscription = await this.organizationApiService.getSubscription( this.organizationId, ); } await this.load(); }), takeUntil(this.destroy$), ) .subscribe(); } async load() { const response = await this.organizationUserApiService.getAllMiniUserDetails( this.organizationId, ); response.data.forEach((u) => { const name = this.userNamePipe.transform(u); this.orgUsersUserIdMap.set(u.userId, { name: name, email: u.email }); }); if (this.organization.providerId != null) { try { const provider = await this.providerService.get(this.organization.providerId); if ( provider != null && (await this.providerService.get(this.organization.providerId)).canManageUsers ) { const providerUsersResponse = await this.apiService.getProviderUsers( this.organization.providerId, ); providerUsersResponse.data.forEach((u) => { const name = this.userNamePipe.transform(u); this.orgUsersUserIdMap.set(u.userId, { name: `${name} (${this.organization.providerName})`, email: u.email, }); }); } } catch (e) { this.logService.warning(e); } } await this.refreshEvents(); this.loaded = true; } protected requestEvents(startDate: string, endDate: string, continuationToken: string) { return this.apiService.getEventsOrganization( this.organizationId, startDate, endDate, continuationToken, ); } protected getUserName(r: EventResponse, userId: string) { if (r.installationId != null) { return { name: `Installation: ${r.installationId}`, }; } if (userId != null) { if (this.orgUsersUserIdMap.has(userId)) { return this.orgUsersUserIdMap.get(userId); } if (r.providerId != null && r.providerId === this.organization.providerId) { return { name: this.organization.providerName, }; } } if (r.systemUser != null) { const systemUserI18nKey: string = EVENT_SYSTEM_USER_TO_TRANSLATION[r.systemUser]; if (systemUserI18nKey) { return { name: this.i18nService.t(systemUserI18nKey), }; } else { return { name: EventSystemUser[r.systemUser], }; } } if (r.serviceAccountId) { return { name: this.i18nService.t("machineAccount") + " " + this.getShortId(r.serviceAccountId), }; } return null; } private getShortId(id: string) { return id?.substring(0, 8); } async changePlan() { const reference = openChangePlanDialog(this.dialogService, { data: { organizationId: this.organizationId, subscription: this.organizationSubscription, productTierType: this.organization.productTierType, }, }); const result = await lastValueFrom(reference.closed); if (result === ChangePlanDialogResultType.Closed) { return; } await this.load(); } ngOnDestroy() { this.destroy$.next(); this.destroy$.complete(); } }