mirror of
https://github.com/zitadel/zitadel.git
synced 2025-02-28 21:37:24 +00:00
feat(console): org metadata, mfa i18n, defer load of actions (#4471)
* user metadata refactor * metadata module, set remove interface * refresh list * mfa, passwordless i18n * i18n, metadata table Co-authored-by: Livio Spring <livio.a@gmail.com>
This commit is contained in:
parent
531c30a031
commit
05ad3b4ef0
@ -1,22 +1,19 @@
|
|||||||
<div class="title-row">
|
<div class="title-row">
|
||||||
<h1 class="metadata-title">{{ 'USER.METADATA.TITLE' | translate }}</h1>
|
<h1 class="metadata-title">{{ 'METADATA.TITLE' | translate }}</h1>
|
||||||
<span class="fill-space"></span>
|
<span class="fill-space"></span>
|
||||||
<p *ngIf="ts" class="ts cnsl-secondary-text">{{ ts | timestampToDate | localizedDate: 'dd. MMM, HH:mm' }}</p>
|
<p *ngIf="ts" class="ts cnsl-secondary-text">{{ ts | timestampToDate | localizedDate: 'dd. MMM, HH:mm' }}</p>
|
||||||
<mat-spinner *ngIf="loading" diameter="20"></mat-spinner>
|
<mat-spinner *ngIf="loading" diameter="20"></mat-spinner>
|
||||||
<button class="icon-button" mat-icon-button (click)="load()">
|
|
||||||
<mat-icon class="icon">refresh</mat-icon>
|
|
||||||
</button>
|
|
||||||
</div>
|
</div>
|
||||||
<p class="desc">{{ 'USER.METADATA.DESCRIPTION' | translate }}</p>
|
<p class="desc">{{ 'METADATA.DESCRIPTION' | translate }}</p>
|
||||||
<div mat-dialog-content class="metadata-dialog-content">
|
<div mat-dialog-content class="metadata-dialog-content">
|
||||||
<form *ngFor="let md of metadata; index as i" (ngSubmit)="saveElement(i)">
|
<form *ngFor="let md of metadata; index as i" (ngSubmit)="saveElement(i)">
|
||||||
<div class="content">
|
<div class="content">
|
||||||
<cnsl-form-field #key id="key{{ i }}" class="formfield">
|
<cnsl-form-field #key id="key{{ i }}" class="formfield">
|
||||||
<cnsl-label>{{ 'USER.METADATA.KEY' | translate }}</cnsl-label>
|
<cnsl-label>{{ 'METADATA.KEY' | translate }}</cnsl-label>
|
||||||
<input cnslInput [(ngModel)]="md.key" [ngModelOptions]="{ standalone: true }" />
|
<input cnslInput [(ngModel)]="md.key" [ngModelOptions]="{ standalone: true }" />
|
||||||
</cnsl-form-field>
|
</cnsl-form-field>
|
||||||
<cnsl-form-field #value id="value{{ i }}" class="formfield">
|
<cnsl-form-field #value id="value{{ i }}" class="formfield">
|
||||||
<cnsl-label>{{ 'USER.METADATA.VALUE' | translate }}</cnsl-label>
|
<cnsl-label>{{ 'METADATA.VALUE' | translate }}</cnsl-label>
|
||||||
<input cnslInput [(ngModel)]="md.value" [ngModelOptions]="{ standalone: true }" />
|
<input cnslInput [(ngModel)]="md.value" [ngModelOptions]="{ standalone: true }" />
|
||||||
</cnsl-form-field>
|
</cnsl-form-field>
|
||||||
|
|
@ -3,7 +3,6 @@ import { MAT_DIALOG_DATA, MatDialogRef } from '@angular/material/dialog';
|
|||||||
import { Buffer } from 'buffer';
|
import { Buffer } from 'buffer';
|
||||||
import { Timestamp } from 'google-protobuf/google/protobuf/timestamp_pb';
|
import { Timestamp } from 'google-protobuf/google/protobuf/timestamp_pb';
|
||||||
import { Metadata } from 'src/app/proto/generated/zitadel/metadata_pb';
|
import { Metadata } from 'src/app/proto/generated/zitadel/metadata_pb';
|
||||||
import { GrpcAuthService } from 'src/app/services/grpc-auth.service';
|
|
||||||
import { ManagementService } from 'src/app/services/mgmt.service';
|
import { ManagementService } from 'src/app/services/mgmt.service';
|
||||||
import { ToastService } from 'src/app/services/toast.service';
|
import { ToastService } from 'src/app/services/toast.service';
|
||||||
|
|
||||||
@ -14,61 +13,15 @@ import { ToastService } from 'src/app/services/toast.service';
|
|||||||
})
|
})
|
||||||
export class MetadataDialogComponent {
|
export class MetadataDialogComponent {
|
||||||
public metadata: Partial<Metadata.AsObject>[] = [];
|
public metadata: Partial<Metadata.AsObject>[] = [];
|
||||||
public injData: any = {};
|
public loading: boolean = false;
|
||||||
public loading: boolean = true;
|
|
||||||
public ts!: Timestamp.AsObject | undefined;
|
public ts!: Timestamp.AsObject | undefined;
|
||||||
|
|
||||||
constructor(
|
constructor(
|
||||||
private managementService: ManagementService,
|
|
||||||
private authService: GrpcAuthService,
|
|
||||||
private toast: ToastService,
|
private toast: ToastService,
|
||||||
public dialogRef: MatDialogRef<MetadataDialogComponent>,
|
public dialogRef: MatDialogRef<MetadataDialogComponent>,
|
||||||
@Inject(MAT_DIALOG_DATA) public data: any,
|
@Inject(MAT_DIALOG_DATA) public data: any,
|
||||||
) {
|
) {
|
||||||
this.injData = data;
|
this.metadata = data.metadata;
|
||||||
this.load();
|
|
||||||
}
|
|
||||||
|
|
||||||
public load(): void {
|
|
||||||
this.loadMetadata()
|
|
||||||
.then(() => {
|
|
||||||
this.loading = false;
|
|
||||||
if (this.metadata.length === 0) {
|
|
||||||
this.addEntry();
|
|
||||||
}
|
|
||||||
})
|
|
||||||
.catch((error) => {
|
|
||||||
this.loading = false;
|
|
||||||
this.toast.showError(error);
|
|
||||||
if (this.metadata.length === 0) {
|
|
||||||
this.addEntry();
|
|
||||||
}
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
public loadMetadata(): Promise<void> {
|
|
||||||
this.loading = true;
|
|
||||||
if (this.injData.userId) {
|
|
||||||
return this.managementService.listUserMetadata(this.injData.userId).then((resp) => {
|
|
||||||
this.metadata = resp.resultList.map((md) => {
|
|
||||||
return {
|
|
||||||
key: md.key,
|
|
||||||
value: Buffer.from(md.value as string, 'base64'),
|
|
||||||
};
|
|
||||||
});
|
|
||||||
this.ts = resp.details?.viewTimestamp;
|
|
||||||
});
|
|
||||||
} else {
|
|
||||||
return this.authService.listMyMetadata().then((resp) => {
|
|
||||||
this.metadata = resp.resultList.map((md) => {
|
|
||||||
return {
|
|
||||||
key: md.key,
|
|
||||||
value: Buffer.from(md.value as string, 'base64'),
|
|
||||||
};
|
|
||||||
});
|
|
||||||
this.ts = resp.details?.viewTimestamp;
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public addEntry(): void {
|
public addEntry(): void {
|
||||||
@ -104,24 +57,24 @@ export class MetadataDialogComponent {
|
|||||||
|
|
||||||
public setMetadata(key: string, value: string): void {
|
public setMetadata(key: string, value: string): void {
|
||||||
if (key && value) {
|
if (key && value) {
|
||||||
this.managementService
|
this.data
|
||||||
.setUserMetadata(key, btoa(value), this.injData.userId)
|
.setFcn(key, value)
|
||||||
.then(() => {
|
.then(() => {
|
||||||
this.toast.showInfo('USER.METADATA.SETSUCCESS', true);
|
this.toast.showInfo('METADATA.SETSUCCESS', true);
|
||||||
})
|
})
|
||||||
.catch((error) => {
|
.catch((error: any) => {
|
||||||
this.toast.showError(error);
|
this.toast.showError(error);
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public removeMetadata(key: string): Promise<void> {
|
public removeMetadata(key: string): Promise<void> {
|
||||||
return this.managementService
|
return this.data
|
||||||
.removeUserMetadata(key, this.injData.userId)
|
.removeFcn(key)
|
||||||
.then((resp) => {
|
.then((resp: any) => {
|
||||||
this.toast.showInfo('USER.METADATA.REMOVESUCCESS', true);
|
this.toast.showInfo('METADATA.REMOVESUCCESS', true);
|
||||||
})
|
})
|
||||||
.catch((error) => {
|
.catch((error: any) => {
|
||||||
this.toast.showError(error);
|
this.toast.showError(error);
|
||||||
});
|
});
|
||||||
}
|
}
|
40
console/src/app/modules/metadata/metadata.module.ts
Normal file
40
console/src/app/modules/metadata/metadata.module.ts
Normal file
@ -0,0 +1,40 @@
|
|||||||
|
import { CommonModule } from '@angular/common';
|
||||||
|
import { NgModule } from '@angular/core';
|
||||||
|
import { FormsModule } from '@angular/forms';
|
||||||
|
import { MatButtonModule } from '@angular/material/button';
|
||||||
|
import { MatDialogModule } from '@angular/material/dialog';
|
||||||
|
import { MatIconModule } from '@angular/material/icon';
|
||||||
|
import { MatProgressSpinnerModule } from '@angular/material/progress-spinner';
|
||||||
|
import { MatTableModule } from '@angular/material/table';
|
||||||
|
import { MatTooltipModule } from '@angular/material/tooltip';
|
||||||
|
import { TranslateModule } from '@ngx-translate/core';
|
||||||
|
import { LocalizedDatePipeModule } from 'src/app/pipes/localized-date-pipe/localized-date-pipe.module';
|
||||||
|
import { TimestampToDatePipeModule } from 'src/app/pipes/timestamp-to-date-pipe/timestamp-to-date-pipe.module';
|
||||||
|
import { CardModule } from '../card/card.module';
|
||||||
|
|
||||||
|
import { InputModule } from '../input/input.module';
|
||||||
|
import { RefreshTableModule } from '../refresh-table/refresh-table.module';
|
||||||
|
import { MetadataDialogComponent } from './metadata-dialog/metadata-dialog.component';
|
||||||
|
import { MetadataComponent } from './metadata/metadata.component';
|
||||||
|
|
||||||
|
@NgModule({
|
||||||
|
declarations: [MetadataComponent, MetadataDialogComponent],
|
||||||
|
imports: [
|
||||||
|
CommonModule,
|
||||||
|
MatDialogModule,
|
||||||
|
MatProgressSpinnerModule,
|
||||||
|
CardModule,
|
||||||
|
MatButtonModule,
|
||||||
|
TranslateModule,
|
||||||
|
InputModule,
|
||||||
|
MatIconModule,
|
||||||
|
MatTooltipModule,
|
||||||
|
FormsModule,
|
||||||
|
LocalizedDatePipeModule,
|
||||||
|
TimestampToDatePipeModule,
|
||||||
|
RefreshTableModule,
|
||||||
|
MatTableModule,
|
||||||
|
],
|
||||||
|
exports: [MetadataComponent, MetadataDialogComponent],
|
||||||
|
})
|
||||||
|
export class MetadataModule {}
|
@ -0,0 +1,36 @@
|
|||||||
|
<cnsl-card class="metadata-details" title="{{ 'METADATA.TITLE' | translate }}">
|
||||||
|
<mat-spinner card-actions class="spinner" diameter="20" *ngIf="loading"></mat-spinner>
|
||||||
|
|
||||||
|
<cnsl-refresh-table [loading]="loading$ | async" (refreshed)="refresh.emit()" [dataSize]="dataSource.data.length">
|
||||||
|
<button actions [disabled]="disabled" mat-raised-button color="primary" class="edit" (click)="editClicked.emit()">
|
||||||
|
{{ 'ACTIONS.EDIT' | translate }}
|
||||||
|
</button>
|
||||||
|
|
||||||
|
<table class="table" mat-table [dataSource]="dataSource">
|
||||||
|
<ng-container matColumnDef="key">
|
||||||
|
<th mat-header-cell *matHeaderCellDef>{{ 'METADATA.KEY' | translate }}</th>
|
||||||
|
<td mat-cell *matCellDef="let metadata">
|
||||||
|
<span *ngIf="metadata?.key" class="centered">
|
||||||
|
{{ metadata.key }}
|
||||||
|
</span>
|
||||||
|
</td>
|
||||||
|
</ng-container>
|
||||||
|
|
||||||
|
<ng-container matColumnDef="value">
|
||||||
|
<th mat-header-cell *matHeaderCellDef>{{ 'METADATA.VALUE' | translate }}</th>
|
||||||
|
<td mat-cell *matCellDef="let metadata">
|
||||||
|
<span *ngIf="metadata?.value" class="centered">
|
||||||
|
{{ metadata.value }}
|
||||||
|
</span>
|
||||||
|
</td>
|
||||||
|
</ng-container>
|
||||||
|
|
||||||
|
<tr mat-header-row *matHeaderRowDef="displayedColumns"></tr>
|
||||||
|
<tr class="highlight" mat-row *matRowDef="let row; columns: displayedColumns"></tr>
|
||||||
|
</table>
|
||||||
|
<div *ngIf="(loading$ | async) === false && !dataSource?.data?.length" class="no-content-row">
|
||||||
|
<i class="las la-exclamation"></i>
|
||||||
|
<span>{{ 'USER.MFA.EMPTY' | translate }}</span>
|
||||||
|
</div>
|
||||||
|
</cnsl-refresh-table>
|
||||||
|
</cnsl-card>
|
@ -1,16 +1,19 @@
|
|||||||
.metadata-details {
|
.metadata-details {
|
||||||
padding-bottom: 1rem;
|
padding-bottom: 1rem;
|
||||||
|
|
||||||
.metadata-actions {
|
.refresh {
|
||||||
display: flex;
|
margin-left: 0.5rem;
|
||||||
align-items: center;
|
font-size: 1.2rem;
|
||||||
justify-content: flex-end;
|
|
||||||
|
|
||||||
.edit {
|
i {
|
||||||
font-size: 14px;
|
font-size: 1.2rem;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.edit {
|
||||||
|
font-size: 14px;
|
||||||
|
}
|
||||||
|
|
||||||
.meta-row {
|
.meta-row {
|
||||||
display: flex;
|
display: flex;
|
||||||
margin-bottom: 0.5rem;
|
margin-bottom: 0.5rem;
|
@ -0,0 +1,34 @@
|
|||||||
|
import { Component, EventEmitter, Input, OnChanges, Output, SimpleChanges, ViewChild } from '@angular/core';
|
||||||
|
import { MatSort } from '@angular/material/sort';
|
||||||
|
import { MatTable, MatTableDataSource } from '@angular/material/table';
|
||||||
|
import { BehaviorSubject, Observable } from 'rxjs';
|
||||||
|
import { Metadata } from 'src/app/proto/generated/zitadel/metadata_pb';
|
||||||
|
|
||||||
|
@Component({
|
||||||
|
selector: 'cnsl-metadata',
|
||||||
|
templateUrl: './metadata.component.html',
|
||||||
|
styleUrls: ['./metadata.component.scss'],
|
||||||
|
})
|
||||||
|
export class MetadataComponent implements OnChanges {
|
||||||
|
@Input() public metadata: Metadata.AsObject[] = [];
|
||||||
|
@Input() public disabled: boolean = false;
|
||||||
|
@Input() public loading: boolean = false;
|
||||||
|
@Output() public editClicked: EventEmitter<void> = new EventEmitter();
|
||||||
|
@Output() public refresh: EventEmitter<void> = new EventEmitter();
|
||||||
|
|
||||||
|
public displayedColumns: string[] = ['key', 'value'];
|
||||||
|
private loadingSubject: BehaviorSubject<boolean> = new BehaviorSubject<boolean>(false);
|
||||||
|
public loading$: Observable<boolean> = this.loadingSubject.asObservable();
|
||||||
|
|
||||||
|
@ViewChild(MatTable) public table!: MatTable<Metadata.AsObject>;
|
||||||
|
@ViewChild(MatSort) public sort!: MatSort;
|
||||||
|
public dataSource: MatTableDataSource<Metadata.AsObject> = new MatTableDataSource<Metadata.AsObject>([]);
|
||||||
|
|
||||||
|
constructor() {}
|
||||||
|
|
||||||
|
ngOnChanges(changes: SimpleChanges): void {
|
||||||
|
if (changes.metadata?.currentValue) {
|
||||||
|
this.dataSource = new MatTableDataSource<Metadata.AsObject>(changes.metadata.currentValue);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@ -84,7 +84,8 @@ export class ActionTableComponent implements OnInit {
|
|||||||
.deleteAction(action.id)
|
.deleteAction(action.id)
|
||||||
.then(() => {
|
.then(() => {
|
||||||
this.toast.showInfo('FLOWS.DIALOG.DELETEACTION.DELETE_SUCCESS', true);
|
this.toast.showInfo('FLOWS.DIALOG.DELETEACTION.DELETE_SUCCESS', true);
|
||||||
this.getData(20, 0);
|
|
||||||
|
this.refreshPage();
|
||||||
})
|
})
|
||||||
.catch((error: any) => {
|
.catch((error: any) => {
|
||||||
this.toast.showError(error);
|
this.toast.showError(error);
|
||||||
@ -152,7 +153,9 @@ export class ActionTableComponent implements OnInit {
|
|||||||
}
|
}
|
||||||
|
|
||||||
public refreshPage(): void {
|
public refreshPage(): void {
|
||||||
this.getData(this.paginator.pageSize, this.paginator.pageIndex * this.paginator.pageSize);
|
setTimeout(() => {
|
||||||
|
this.getData(this.paginator.pageSize, this.paginator.pageIndex * this.paginator.pageSize);
|
||||||
|
}, 1000);
|
||||||
}
|
}
|
||||||
|
|
||||||
public deactivateSelection(): Promise<void> {
|
public deactivateSelection(): Promise<void> {
|
||||||
|
@ -46,6 +46,13 @@
|
|||||||
<cnsl-settings-grid [type]="PolicyComponentServiceType.MGMT"></cnsl-settings-grid>
|
<cnsl-settings-grid [type]="PolicyComponentServiceType.MGMT"></cnsl-settings-grid>
|
||||||
</ng-container>
|
</ng-container>
|
||||||
|
|
||||||
|
<cnsl-metadata
|
||||||
|
[metadata]="metadata"
|
||||||
|
[disabled]="(['org.write'] | hasRole | async) === false"
|
||||||
|
(editClicked)="editMetadata()"
|
||||||
|
(refresh)="loadMetadata()"
|
||||||
|
></cnsl-metadata>
|
||||||
|
|
||||||
<ng-template #nopolicyreadpermission>
|
<ng-template #nopolicyreadpermission>
|
||||||
<div class="no-permission-warn-wrapper">
|
<div class="no-permission-warn-wrapper">
|
||||||
<cnsl-info-section class="info-section-warn" [fitWidth]="true" [type]="InfoSectionType.ALERT">{{
|
<cnsl-info-section class="info-section-warn" [fitWidth]="true" [type]="InfoSectionType.ALERT">{{
|
||||||
|
@ -1,20 +1,23 @@
|
|||||||
import { Component, OnDestroy, OnInit } from '@angular/core';
|
import { Component, OnDestroy, OnInit } from '@angular/core';
|
||||||
import { MatDialog } from '@angular/material/dialog';
|
import { MatDialog } from '@angular/material/dialog';
|
||||||
import { ActivatedRoute, Router } from '@angular/router';
|
import { Router } from '@angular/router';
|
||||||
import { BehaviorSubject, from, Observable, of, Subject, takeUntil } from 'rxjs';
|
import { BehaviorSubject, from, Observable, of, Subject, takeUntil } from 'rxjs';
|
||||||
import { catchError, finalize, map, take } from 'rxjs/operators';
|
import { catchError, finalize, map } from 'rxjs/operators';
|
||||||
import { CreationType, MemberCreateDialogComponent } from 'src/app/modules/add-member-dialog/member-create-dialog.component';
|
import { CreationType, MemberCreateDialogComponent } from 'src/app/modules/add-member-dialog/member-create-dialog.component';
|
||||||
import { ChangeType } from 'src/app/modules/changes/changes.component';
|
import { ChangeType } from 'src/app/modules/changes/changes.component';
|
||||||
import { InfoSectionType } from 'src/app/modules/info-section/info-section.component';
|
import { InfoSectionType } from 'src/app/modules/info-section/info-section.component';
|
||||||
|
import { MetadataDialogComponent } from 'src/app/modules/metadata/metadata-dialog/metadata-dialog.component';
|
||||||
import { PolicyComponentServiceType } from 'src/app/modules/policies/policy-component-types.enum';
|
import { PolicyComponentServiceType } from 'src/app/modules/policies/policy-component-types.enum';
|
||||||
import { WarnDialogComponent } from 'src/app/modules/warn-dialog/warn-dialog.component';
|
import { WarnDialogComponent } from 'src/app/modules/warn-dialog/warn-dialog.component';
|
||||||
import { Member } from 'src/app/proto/generated/zitadel/member_pb';
|
import { Member } from 'src/app/proto/generated/zitadel/member_pb';
|
||||||
|
import { Metadata } from 'src/app/proto/generated/zitadel/metadata_pb';
|
||||||
import { Org, OrgState } from 'src/app/proto/generated/zitadel/org_pb';
|
import { Org, OrgState } from 'src/app/proto/generated/zitadel/org_pb';
|
||||||
import { User } from 'src/app/proto/generated/zitadel/user_pb';
|
import { User } from 'src/app/proto/generated/zitadel/user_pb';
|
||||||
import { Breadcrumb, BreadcrumbService, BreadcrumbType } from 'src/app/services/breadcrumb.service';
|
import { Breadcrumb, BreadcrumbService, BreadcrumbType } from 'src/app/services/breadcrumb.service';
|
||||||
import { GrpcAuthService } from 'src/app/services/grpc-auth.service';
|
import { GrpcAuthService } from 'src/app/services/grpc-auth.service';
|
||||||
import { ManagementService } from 'src/app/services/mgmt.service';
|
import { ManagementService } from 'src/app/services/mgmt.service';
|
||||||
import { ToastService } from 'src/app/services/toast.service';
|
import { ToastService } from 'src/app/services/toast.service';
|
||||||
|
import { Buffer } from 'buffer';
|
||||||
|
|
||||||
@Component({
|
@Component({
|
||||||
selector: 'cnsl-org-detail',
|
selector: 'cnsl-org-detail',
|
||||||
@ -28,6 +31,9 @@ export class OrgDetailComponent implements OnInit, OnDestroy {
|
|||||||
public OrgState: any = OrgState;
|
public OrgState: any = OrgState;
|
||||||
public ChangeType: any = ChangeType;
|
public ChangeType: any = ChangeType;
|
||||||
|
|
||||||
|
public metadata: Metadata.AsObject[] = [];
|
||||||
|
public loadingMetadata: boolean = true;
|
||||||
|
|
||||||
// members
|
// members
|
||||||
private loadingSubject: BehaviorSubject<boolean> = new BehaviorSubject<boolean>(false);
|
private loadingSubject: BehaviorSubject<boolean> = new BehaviorSubject<boolean>(false);
|
||||||
public loading$: Observable<boolean> = this.loadingSubject.asObservable();
|
public loading$: Observable<boolean> = this.loadingSubject.asObservable();
|
||||||
@ -36,6 +42,7 @@ export class OrgDetailComponent implements OnInit, OnDestroy {
|
|||||||
private destroy$: Subject<void> = new Subject();
|
private destroy$: Subject<void> = new Subject();
|
||||||
|
|
||||||
public InfoSectionType: any = InfoSectionType;
|
public InfoSectionType: any = InfoSectionType;
|
||||||
|
|
||||||
constructor(
|
constructor(
|
||||||
auth: GrpcAuthService,
|
auth: GrpcAuthService,
|
||||||
private dialog: MatDialog,
|
private dialog: MatDialog,
|
||||||
@ -52,11 +59,13 @@ export class OrgDetailComponent implements OnInit, OnDestroy {
|
|||||||
|
|
||||||
auth.activeOrgChanged.pipe(takeUntil(this.destroy$)).subscribe((org) => {
|
auth.activeOrgChanged.pipe(takeUntil(this.destroy$)).subscribe((org) => {
|
||||||
this.getData();
|
this.getData();
|
||||||
|
this.loadMetadata();
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
public ngOnInit(): void {
|
public ngOnInit(): void {
|
||||||
this.getData();
|
this.getData();
|
||||||
|
this.loadMetadata();
|
||||||
}
|
}
|
||||||
|
|
||||||
public ngOnDestroy(): void {
|
public ngOnDestroy(): void {
|
||||||
@ -188,4 +197,40 @@ export class OrgDetailComponent implements OnInit, OnDestroy {
|
|||||||
this.membersSubject.next(members);
|
this.membersSubject.next(members);
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public loadMetadata(): Promise<any> | void {
|
||||||
|
this.loadingMetadata = true;
|
||||||
|
return this.mgmtService
|
||||||
|
.listOrgMetadata()
|
||||||
|
.then((resp) => {
|
||||||
|
this.loadingMetadata = false;
|
||||||
|
this.metadata = resp.resultList.map((md) => {
|
||||||
|
return {
|
||||||
|
key: md.key,
|
||||||
|
value: Buffer.from(md.value as string, 'base64').toString('ascii'),
|
||||||
|
};
|
||||||
|
});
|
||||||
|
})
|
||||||
|
.catch((error) => {
|
||||||
|
this.loadingMetadata = false;
|
||||||
|
this.toast.showError(error);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
public editMetadata(): void {
|
||||||
|
const setFcn = (key: string, value: string): Promise<any> => this.mgmtService.setOrgMetadata(key, btoa(value));
|
||||||
|
const removeFcn = (key: string): Promise<any> => this.mgmtService.removeOrgMetadata(key);
|
||||||
|
|
||||||
|
const dialogRef = this.dialog.open(MetadataDialogComponent, {
|
||||||
|
data: {
|
||||||
|
metadata: this.metadata,
|
||||||
|
setFcn: setFcn,
|
||||||
|
removeFcn: removeFcn,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
dialogRef.afterClosed().subscribe(() => {
|
||||||
|
this.loadMetadata();
|
||||||
|
});
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
@ -18,6 +18,7 @@ import { InfoRowModule } from 'src/app/modules/info-row/info-row.module';
|
|||||||
import { InfoSectionModule } from 'src/app/modules/info-section/info-section.module';
|
import { InfoSectionModule } from 'src/app/modules/info-section/info-section.module';
|
||||||
import { InputModule } from 'src/app/modules/input/input.module';
|
import { InputModule } from 'src/app/modules/input/input.module';
|
||||||
import { MetaLayoutModule } from 'src/app/modules/meta-layout/meta-layout.module';
|
import { MetaLayoutModule } from 'src/app/modules/meta-layout/meta-layout.module';
|
||||||
|
import { MetadataModule } from 'src/app/modules/metadata/metadata.module';
|
||||||
import { SettingsGridModule } from 'src/app/modules/settings-grid/settings-grid.module';
|
import { SettingsGridModule } from 'src/app/modules/settings-grid/settings-grid.module';
|
||||||
import { SharedModule } from 'src/app/modules/shared/shared.module';
|
import { SharedModule } from 'src/app/modules/shared/shared.module';
|
||||||
import { TopViewModule } from 'src/app/modules/top-view/top-view.module';
|
import { TopViewModule } from 'src/app/modules/top-view/top-view.module';
|
||||||
@ -53,6 +54,7 @@ import { OrgRoutingModule } from './org-routing.module';
|
|||||||
MatMenuModule,
|
MatMenuModule,
|
||||||
ChangesModule,
|
ChangesModule,
|
||||||
MatProgressSpinnerModule,
|
MatProgressSpinnerModule,
|
||||||
|
MetadataModule,
|
||||||
TranslateModule,
|
TranslateModule,
|
||||||
SharedModule,
|
SharedModule,
|
||||||
SettingsGridModule,
|
SettingsGridModule,
|
||||||
|
@ -25,7 +25,7 @@
|
|||||||
color="primary"
|
color="primary"
|
||||||
matTooltip="{{ 'ACTIONS.NEW' | translate }}"
|
matTooltip="{{ 'ACTIONS.NEW' | translate }}"
|
||||||
>
|
>
|
||||||
<i class="icon las la-fingerprint"></i>
|
<mat-icon>add</mat-icon>
|
||||||
{{ 'USER.PASSWORDLESS.U2F' | translate }}
|
{{ 'USER.PASSWORDLESS.U2F' | translate }}
|
||||||
</button>
|
</button>
|
||||||
<table class="table" mat-table [dataSource]="dataSource">
|
<table class="table" mat-table [dataSource]="dataSource">
|
||||||
|
@ -3,7 +3,7 @@
|
|||||||
<div *ngIf="!showSent && !showQR">
|
<div *ngIf="!showSent && !showQR">
|
||||||
<p>{{ 'USER.PASSWORDLESS.DIALOG.ADD_DESCRIPTION' | translate }}</p>
|
<p>{{ 'USER.PASSWORDLESS.DIALOG.ADD_DESCRIPTION' | translate }}</p>
|
||||||
|
|
||||||
<div class="desc">
|
<div class="passwordless-desc">
|
||||||
<i class="icon las la-plus-circle"></i>
|
<i class="icon las la-plus-circle"></i>
|
||||||
<p class="cnsl-secondary-text">{{ 'USER.PASSWORDLESS.DIALOG.NEW_DESCRIPTION' | translate }}</p>
|
<p class="cnsl-secondary-text">{{ 'USER.PASSWORDLESS.DIALOG.NEW_DESCRIPTION' | translate }}</p>
|
||||||
</div>
|
</div>
|
||||||
@ -19,7 +19,7 @@
|
|||||||
|
|
||||||
<p class="error">{{ error }}</p>
|
<p class="error">{{ error }}</p>
|
||||||
|
|
||||||
<div class="desc">
|
<div class="passwordless-desc">
|
||||||
<i class="icon las la-paper-plane"></i>
|
<i class="icon las la-paper-plane"></i>
|
||||||
<p class="cnsl-secondary-text">{{ 'USER.PASSWORDLESS.DIALOG.SEND_DESCRIPTION' | translate }}</p>
|
<p class="cnsl-secondary-text">{{ 'USER.PASSWORDLESS.DIALOG.SEND_DESCRIPTION' | translate }}</p>
|
||||||
</div>
|
</div>
|
||||||
@ -28,7 +28,7 @@
|
|||||||
{{ 'USER.PASSWORDLESS.DIALOG.SEND' | translate }}
|
{{ 'USER.PASSWORDLESS.DIALOG.SEND' | translate }}
|
||||||
</button>
|
</button>
|
||||||
|
|
||||||
<div class="desc">
|
<div class="passwordless-desc">
|
||||||
<i class="icon las la-qrcode"></i>
|
<i class="icon las la-qrcode"></i>
|
||||||
<p class="cnsl-secondary-text">{{ 'USER.PASSWORDLESS.DIALOG.QRCODE_DESCRIPTION' | translate }}</p>
|
<p class="cnsl-secondary-text">{{ 'USER.PASSWORDLESS.DIALOG.QRCODE_DESCRIPTION' | translate }}</p>
|
||||||
</div>
|
</div>
|
||||||
|
@ -12,7 +12,7 @@
|
|||||||
margin: 0;
|
margin: 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
.desc {
|
.passwordless-desc {
|
||||||
display: flex;
|
display: flex;
|
||||||
align-items: center;
|
align-items: center;
|
||||||
padding-top: 1rem;
|
padding-top: 1rem;
|
||||||
|
@ -130,7 +130,13 @@
|
|||||||
</ng-container>
|
</ng-container>
|
||||||
|
|
||||||
<ng-container *ngIf="currentSetting === 'metadata'">
|
<ng-container *ngIf="currentSetting === 'metadata'">
|
||||||
<cnsl-metadata *ngIf="user && user.id" [userId]="user.id"></cnsl-metadata>
|
<cnsl-metadata
|
||||||
|
[metadata]="metadata"
|
||||||
|
[disabled]="(['user.write:' + user.id, 'user.write'] | hasRole | async) === false"
|
||||||
|
*ngIf="user && user.id"
|
||||||
|
(editClicked)="editMetadata()"
|
||||||
|
(refresh)="loadMetadata()"
|
||||||
|
></cnsl-metadata>
|
||||||
</ng-container>
|
</ng-container>
|
||||||
</cnsl-sidenav>
|
</cnsl-sidenav>
|
||||||
|
|
||||||
|
@ -6,15 +6,18 @@ import { ActivatedRoute, Params } from '@angular/router';
|
|||||||
import { TranslateService } from '@ngx-translate/core';
|
import { TranslateService } from '@ngx-translate/core';
|
||||||
import { Subscription, take } from 'rxjs';
|
import { Subscription, take } from 'rxjs';
|
||||||
import { ChangeType } from 'src/app/modules/changes/changes.component';
|
import { ChangeType } from 'src/app/modules/changes/changes.component';
|
||||||
|
import { MetadataDialogComponent } from 'src/app/modules/metadata/metadata-dialog/metadata-dialog.component';
|
||||||
import { SidenavSetting } from 'src/app/modules/sidenav/sidenav.component';
|
import { SidenavSetting } from 'src/app/modules/sidenav/sidenav.component';
|
||||||
import { UserGrantContext } from 'src/app/modules/user-grants/user-grants-datasource';
|
import { UserGrantContext } from 'src/app/modules/user-grants/user-grants-datasource';
|
||||||
import { WarnDialogComponent } from 'src/app/modules/warn-dialog/warn-dialog.component';
|
import { WarnDialogComponent } from 'src/app/modules/warn-dialog/warn-dialog.component';
|
||||||
|
import { Metadata } from 'src/app/proto/generated/zitadel/metadata_pb';
|
||||||
import { Email, Gender, Phone, Profile, User, UserState } from 'src/app/proto/generated/zitadel/user_pb';
|
import { Email, Gender, Phone, Profile, User, UserState } from 'src/app/proto/generated/zitadel/user_pb';
|
||||||
import { AuthenticationService } from 'src/app/services/authentication.service';
|
import { AuthenticationService } from 'src/app/services/authentication.service';
|
||||||
import { Breadcrumb, BreadcrumbService, BreadcrumbType } from 'src/app/services/breadcrumb.service';
|
import { Breadcrumb, BreadcrumbService, BreadcrumbType } from 'src/app/services/breadcrumb.service';
|
||||||
import { GrpcAuthService } from 'src/app/services/grpc-auth.service';
|
import { GrpcAuthService } from 'src/app/services/grpc-auth.service';
|
||||||
|
import { ManagementService } from 'src/app/services/mgmt.service';
|
||||||
import { ToastService } from 'src/app/services/toast.service';
|
import { ToastService } from 'src/app/services/toast.service';
|
||||||
|
import { Buffer } from 'buffer';
|
||||||
import { EditDialogComponent, EditDialogType } from './edit-dialog/edit-dialog.component';
|
import { EditDialogComponent, EditDialogType } from './edit-dialog/edit-dialog.component';
|
||||||
|
|
||||||
@Component({
|
@Component({
|
||||||
@ -30,6 +33,7 @@ export class AuthUserDetailComponent implements OnDestroy {
|
|||||||
private subscription: Subscription = new Subscription();
|
private subscription: Subscription = new Subscription();
|
||||||
|
|
||||||
public loading: boolean = false;
|
public loading: boolean = false;
|
||||||
|
public loadingMetadata: boolean = false;
|
||||||
|
|
||||||
public ChangeType: any = ChangeType;
|
public ChangeType: any = ChangeType;
|
||||||
public userLoginMustBeDomain: boolean = false;
|
public userLoginMustBeDomain: boolean = false;
|
||||||
@ -38,6 +42,8 @@ export class AuthUserDetailComponent implements OnDestroy {
|
|||||||
public USERGRANTCONTEXT: UserGrantContext = UserGrantContext.USER;
|
public USERGRANTCONTEXT: UserGrantContext = UserGrantContext.USER;
|
||||||
public refreshChanges$: EventEmitter<void> = new EventEmitter();
|
public refreshChanges$: EventEmitter<void> = new EventEmitter();
|
||||||
|
|
||||||
|
public metadata: Metadata.AsObject[] = [];
|
||||||
|
|
||||||
public settingsList: SidenavSetting[] = [
|
public settingsList: SidenavSetting[] = [
|
||||||
{ id: 'general', i18nKey: 'USER.SETTINGS.GENERAL' },
|
{ id: 'general', i18nKey: 'USER.SETTINGS.GENERAL' },
|
||||||
{ id: 'idp', i18nKey: 'USER.SETTINGS.IDP' },
|
{ id: 'idp', i18nKey: 'USER.SETTINGS.IDP' },
|
||||||
@ -55,6 +61,7 @@ export class AuthUserDetailComponent implements OnDestroy {
|
|||||||
public userService: GrpcAuthService,
|
public userService: GrpcAuthService,
|
||||||
private dialog: MatDialog,
|
private dialog: MatDialog,
|
||||||
private auth: AuthenticationService,
|
private auth: AuthenticationService,
|
||||||
|
private mgmt: ManagementService,
|
||||||
private breadcrumbService: BreadcrumbService,
|
private breadcrumbService: BreadcrumbService,
|
||||||
private mediaMatcher: MediaMatcher,
|
private mediaMatcher: MediaMatcher,
|
||||||
private _location: Location,
|
private _location: Location,
|
||||||
@ -104,6 +111,8 @@ export class AuthUserDetailComponent implements OnDestroy {
|
|||||||
if (resp.user) {
|
if (resp.user) {
|
||||||
this.user = resp.user;
|
this.user = resp.user;
|
||||||
|
|
||||||
|
this.loadMetadata();
|
||||||
|
|
||||||
this.breadcrumbService.setBreadcrumb([
|
this.breadcrumbService.setBreadcrumb([
|
||||||
new Breadcrumb({
|
new Breadcrumb({
|
||||||
type: BreadcrumbType.AUTHUSER,
|
type: BreadcrumbType.AUTHUSER,
|
||||||
@ -337,4 +346,45 @@ export class AuthUserDetailComponent implements OnDestroy {
|
|||||||
}
|
}
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public loadMetadata(): Promise<any> | void {
|
||||||
|
if (this.user) {
|
||||||
|
this.loadingMetadata = true;
|
||||||
|
return this.mgmt
|
||||||
|
.listUserMetadata(this.user.id)
|
||||||
|
.then((resp) => {
|
||||||
|
this.loadingMetadata = false;
|
||||||
|
this.metadata = resp.resultList.map((md) => {
|
||||||
|
return {
|
||||||
|
key: md.key,
|
||||||
|
value: Buffer.from(md.value as string, 'base64').toString('ascii'),
|
||||||
|
};
|
||||||
|
});
|
||||||
|
})
|
||||||
|
.catch((error) => {
|
||||||
|
this.loadingMetadata = false;
|
||||||
|
this.toast.showError(error);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public editMetadata(): void {
|
||||||
|
if (this.user && this.user.id) {
|
||||||
|
const setFcn = (key: string, value: string): Promise<any> =>
|
||||||
|
this.mgmt.setUserMetadata(key, Buffer.from(value).toString('base64'), this.user?.id ?? '');
|
||||||
|
const removeFcn = (key: string): Promise<any> => this.mgmt.removeUserMetadata(key, this.user?.id ?? '');
|
||||||
|
|
||||||
|
const dialogRef = this.dialog.open(MetadataDialogComponent, {
|
||||||
|
data: {
|
||||||
|
metadata: this.metadata,
|
||||||
|
setFcn: setFcn,
|
||||||
|
removeFcn: removeFcn,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
dialogRef.afterClosed().subscribe(() => {
|
||||||
|
this.loadMetadata();
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
@ -1,25 +0,0 @@
|
|||||||
<cnsl-card class="metadata-details" title="{{ 'USER.METADATA.TITLE' | translate }}">
|
|
||||||
<div class="metadata-actions">
|
|
||||||
<mat-spinner class="spinner" diameter="20" *ngIf="loading"></mat-spinner>
|
|
||||||
<button
|
|
||||||
[disabled]="(['user.write:' + userId, 'user.write'] | hasRole | async) === false"
|
|
||||||
mat-raised-button
|
|
||||||
color="primary"
|
|
||||||
class="edit"
|
|
||||||
(click)="editMetadata()"
|
|
||||||
>
|
|
||||||
{{ 'ACTIONS.EDIT' | translate }}
|
|
||||||
</button>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<ng-container *ngIf="metadata?.length; else emptyList">
|
|
||||||
<div class="metadata-set" *ngFor="let md of metadata">
|
|
||||||
<span class="first cnsl-secondary-text">{{ md.key }}</span>
|
|
||||||
<span class="second">{{ md.value }}</span>
|
|
||||||
</div>
|
|
||||||
</ng-container>
|
|
||||||
|
|
||||||
<ng-template #emptyList>
|
|
||||||
<p class="empty-desc cnsl-secondary-text">{{ 'USER.METADATA.EMPTY' | translate }}</p>
|
|
||||||
</ng-template>
|
|
||||||
</cnsl-card>
|
|
@ -1,55 +0,0 @@
|
|||||||
import { Component, Input, OnInit } from '@angular/core';
|
|
||||||
import { MatDialog } from '@angular/material/dialog';
|
|
||||||
import { Metadata } from 'src/app/proto/generated/zitadel/metadata_pb';
|
|
||||||
import { ManagementService } from 'src/app/services/mgmt.service';
|
|
||||||
import { ToastService } from 'src/app/services/toast.service';
|
|
||||||
|
|
||||||
import { MetadataDialogComponent } from '../metadata-dialog/metadata-dialog.component';
|
|
||||||
|
|
||||||
@Component({
|
|
||||||
selector: 'cnsl-metadata',
|
|
||||||
templateUrl: './metadata.component.html',
|
|
||||||
styleUrls: ['./metadata.component.scss'],
|
|
||||||
})
|
|
||||||
export class MetadataComponent implements OnInit {
|
|
||||||
@Input() userId: string = '';
|
|
||||||
public metadata: Metadata.AsObject[] = [];
|
|
||||||
public loading: boolean = false;
|
|
||||||
|
|
||||||
constructor(private dialog: MatDialog, private service: ManagementService, private toast: ToastService) {}
|
|
||||||
|
|
||||||
ngOnInit(): void {
|
|
||||||
this.loadMetadata();
|
|
||||||
}
|
|
||||||
|
|
||||||
public editMetadata(): void {
|
|
||||||
const dialogRef = this.dialog.open(MetadataDialogComponent, {
|
|
||||||
data: {
|
|
||||||
userId: this.userId,
|
|
||||||
},
|
|
||||||
});
|
|
||||||
|
|
||||||
dialogRef.afterClosed().subscribe(() => {
|
|
||||||
this.loadMetadata();
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
public loadMetadata(): Promise<any> {
|
|
||||||
this.loading = true;
|
|
||||||
return (this.service as ManagementService)
|
|
||||||
.listUserMetadata(this.userId)
|
|
||||||
.then((resp) => {
|
|
||||||
this.loading = false;
|
|
||||||
this.metadata = resp.resultList.map((md) => {
|
|
||||||
return {
|
|
||||||
key: md.key,
|
|
||||||
value: atob(md.value as string),
|
|
||||||
};
|
|
||||||
});
|
|
||||||
})
|
|
||||||
.catch((error) => {
|
|
||||||
this.loading = false;
|
|
||||||
this.toast.showError(error);
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
|
@ -52,12 +52,11 @@ import { ContactComponent } from './contact/contact.component';
|
|||||||
import { DetailFormMachineModule } from './detail-form-machine/detail-form-machine.module';
|
import { DetailFormMachineModule } from './detail-form-machine/detail-form-machine.module';
|
||||||
import { DetailFormModule } from './detail-form/detail-form.module';
|
import { DetailFormModule } from './detail-form/detail-form.module';
|
||||||
import { ExternalIdpsComponent } from './external-idps/external-idps.component';
|
import { ExternalIdpsComponent } from './external-idps/external-idps.component';
|
||||||
import { MetadataDialogComponent } from './metadata-dialog/metadata-dialog.component';
|
|
||||||
import { MetadataComponent } from './metadata/metadata.component';
|
|
||||||
import { PasswordComponent } from './password/password.component';
|
import { PasswordComponent } from './password/password.component';
|
||||||
import { PasswordlessComponent } from './user-detail/passwordless/passwordless.component';
|
import { PasswordlessComponent } from './user-detail/passwordless/passwordless.component';
|
||||||
import { UserDetailComponent } from './user-detail/user-detail.component';
|
import { UserDetailComponent } from './user-detail/user-detail.component';
|
||||||
import { UserMfaComponent } from './user-detail/user-mfa/user-mfa.component';
|
import { UserMfaComponent } from './user-detail/user-mfa/user-mfa.component';
|
||||||
|
import { MetadataModule } from 'src/app/modules/metadata/metadata.module';
|
||||||
|
|
||||||
@NgModule({
|
@NgModule({
|
||||||
declarations: [
|
declarations: [
|
||||||
@ -76,8 +75,6 @@ import { UserMfaComponent } from './user-detail/user-mfa/user-mfa.component';
|
|||||||
DialogU2FComponent,
|
DialogU2FComponent,
|
||||||
DialogPasswordlessComponent,
|
DialogPasswordlessComponent,
|
||||||
AuthFactorDialogComponent,
|
AuthFactorDialogComponent,
|
||||||
MetadataDialogComponent,
|
|
||||||
MetadataComponent,
|
|
||||||
],
|
],
|
||||||
imports: [
|
imports: [
|
||||||
ChangesModule,
|
ChangesModule,
|
||||||
@ -95,6 +92,7 @@ import { UserMfaComponent } from './user-detail/user-mfa/user-mfa.component';
|
|||||||
ShowTokenDialogModule,
|
ShowTokenDialogModule,
|
||||||
MetaLayoutModule,
|
MetaLayoutModule,
|
||||||
MatCheckboxModule,
|
MatCheckboxModule,
|
||||||
|
MetadataModule,
|
||||||
TopViewModule,
|
TopViewModule,
|
||||||
HasRolePipeModule,
|
HasRolePipeModule,
|
||||||
UserGrantsModule,
|
UserGrantsModule,
|
||||||
|
@ -206,7 +206,13 @@
|
|||||||
</ng-container>
|
</ng-container>
|
||||||
|
|
||||||
<ng-container *ngIf="currentSetting && currentSetting === 'metadata'">
|
<ng-container *ngIf="currentSetting && currentSetting === 'metadata'">
|
||||||
<cnsl-metadata *ngIf="user" [userId]="user.id"></cnsl-metadata>
|
<cnsl-metadata
|
||||||
|
[metadata]="metadata"
|
||||||
|
[disabled]="(['user.write:' + user.id, 'user.write'] | hasRole | async) === false"
|
||||||
|
*ngIf="user && user.id"
|
||||||
|
(editClicked)="editMetadata()"
|
||||||
|
(refresh)="loadMetadata(user.id)"
|
||||||
|
></cnsl-metadata>
|
||||||
</ng-container>
|
</ng-container>
|
||||||
</div>
|
</div>
|
||||||
</cnsl-sidenav>
|
</cnsl-sidenav>
|
||||||
|
@ -7,6 +7,7 @@ import { TranslateService } from '@ngx-translate/core';
|
|||||||
import { take } from 'rxjs/operators';
|
import { take } from 'rxjs/operators';
|
||||||
import { ChangeType } from 'src/app/modules/changes/changes.component';
|
import { ChangeType } from 'src/app/modules/changes/changes.component';
|
||||||
import { InfoSectionType } from 'src/app/modules/info-section/info-section.component';
|
import { InfoSectionType } from 'src/app/modules/info-section/info-section.component';
|
||||||
|
import { MetadataDialogComponent } from 'src/app/modules/metadata/metadata-dialog/metadata-dialog.component';
|
||||||
import { SidenavSetting } from 'src/app/modules/sidenav/sidenav.component';
|
import { SidenavSetting } from 'src/app/modules/sidenav/sidenav.component';
|
||||||
import { UserGrantContext } from 'src/app/modules/user-grants/user-grants-datasource';
|
import { UserGrantContext } from 'src/app/modules/user-grants/user-grants-datasource';
|
||||||
import { WarnDialogComponent } from 'src/app/modules/warn-dialog/warn-dialog.component';
|
import { WarnDialogComponent } from 'src/app/modules/warn-dialog/warn-dialog.component';
|
||||||
@ -16,7 +17,7 @@ import { Email, Gender, Machine, Phone, Profile, User, UserState } from 'src/app
|
|||||||
import { Breadcrumb, BreadcrumbService, BreadcrumbType } from 'src/app/services/breadcrumb.service';
|
import { Breadcrumb, BreadcrumbService, BreadcrumbType } from 'src/app/services/breadcrumb.service';
|
||||||
import { ManagementService } from 'src/app/services/mgmt.service';
|
import { ManagementService } from 'src/app/services/mgmt.service';
|
||||||
import { ToastService } from 'src/app/services/toast.service';
|
import { ToastService } from 'src/app/services/toast.service';
|
||||||
|
import { Buffer } from 'buffer';
|
||||||
import { EditDialogComponent, EditDialogType } from '../auth-user-detail/edit-dialog/edit-dialog.component';
|
import { EditDialogComponent, EditDialogType } from '../auth-user-detail/edit-dialog/edit-dialog.component';
|
||||||
import { ResendEmailDialogComponent } from '../auth-user-detail/resend-email-dialog/resend-email-dialog.component';
|
import { ResendEmailDialogComponent } from '../auth-user-detail/resend-email-dialog/resend-email-dialog.component';
|
||||||
|
|
||||||
@ -42,7 +43,9 @@ export class UserDetailComponent implements OnInit {
|
|||||||
public languages: string[] = ['de', 'en', 'it', 'fr'];
|
public languages: string[] = ['de', 'en', 'it', 'fr'];
|
||||||
|
|
||||||
public ChangeType: any = ChangeType;
|
public ChangeType: any = ChangeType;
|
||||||
|
|
||||||
public loading: boolean = true;
|
public loading: boolean = true;
|
||||||
|
public loadingMetadata: boolean = true;
|
||||||
|
|
||||||
public UserState: any = UserState;
|
public UserState: any = UserState;
|
||||||
public copied: string = '';
|
public copied: string = '';
|
||||||
@ -113,6 +116,7 @@ export class UserDetailComponent implements OnInit {
|
|||||||
this.mgmtUserService
|
this.mgmtUserService
|
||||||
.getUserByID(id)
|
.getUserByID(id)
|
||||||
.then((resp) => {
|
.then((resp) => {
|
||||||
|
this.loadMetadata(id);
|
||||||
this.loading = false;
|
this.loading = false;
|
||||||
if (resp.user) {
|
if (resp.user) {
|
||||||
this.user = resp.user;
|
this.user = resp.user;
|
||||||
@ -129,17 +133,6 @@ export class UserDetailComponent implements OnInit {
|
|||||||
this.loading = false;
|
this.loading = false;
|
||||||
this.toast.showError(err);
|
this.toast.showError(err);
|
||||||
});
|
});
|
||||||
|
|
||||||
this.mgmtUserService
|
|
||||||
.listUserMetadata(id, 0, 100, [])
|
|
||||||
.then((resp) => {
|
|
||||||
if (resp.resultList) {
|
|
||||||
this.metadata = resp.resultList;
|
|
||||||
}
|
|
||||||
})
|
|
||||||
.catch((err) => {
|
|
||||||
console.error(err);
|
|
||||||
});
|
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -448,4 +441,43 @@ export class UserDetailComponent implements OnInit {
|
|||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public loadMetadata(id: string): Promise<any> | void {
|
||||||
|
this.loadingMetadata = true;
|
||||||
|
return this.mgmtUserService
|
||||||
|
.listUserMetadata(id)
|
||||||
|
.then((resp) => {
|
||||||
|
this.loadingMetadata = false;
|
||||||
|
this.metadata = resp.resultList.map((md) => {
|
||||||
|
return {
|
||||||
|
key: md.key,
|
||||||
|
value: Buffer.from(md.value as string, 'base64').toString('ascii'),
|
||||||
|
};
|
||||||
|
});
|
||||||
|
})
|
||||||
|
.catch((error) => {
|
||||||
|
this.loadingMetadata = false;
|
||||||
|
this.toast.showError(error);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
public editMetadata(): void {
|
||||||
|
if (this.user) {
|
||||||
|
const setFcn = (key: string, value: string): Promise<any> =>
|
||||||
|
this.mgmtUserService.setUserMetadata(key, Buffer.from(value).toString('base64'), this.user.id);
|
||||||
|
const removeFcn = (key: string): Promise<any> => this.mgmtUserService.removeUserMetadata(key, this.user.id);
|
||||||
|
|
||||||
|
const dialogRef = this.dialog.open(MetadataDialogComponent, {
|
||||||
|
data: {
|
||||||
|
metadata: this.metadata,
|
||||||
|
setFcn: setFcn,
|
||||||
|
removeFcn: removeFcn,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
dialogRef.afterClosed().subscribe(() => {
|
||||||
|
this.loadMetadata(this.user.id);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
@ -219,6 +219,8 @@ import {
|
|||||||
ListOrgMemberRolesResponse,
|
ListOrgMemberRolesResponse,
|
||||||
ListOrgMembersRequest,
|
ListOrgMembersRequest,
|
||||||
ListOrgMembersResponse,
|
ListOrgMembersResponse,
|
||||||
|
ListOrgMetadataRequest,
|
||||||
|
ListOrgMetadataResponse,
|
||||||
ListPersonalAccessTokensRequest,
|
ListPersonalAccessTokensRequest,
|
||||||
ListPersonalAccessTokensResponse,
|
ListPersonalAccessTokensResponse,
|
||||||
ListProjectChangesRequest,
|
ListProjectChangesRequest,
|
||||||
@ -303,6 +305,8 @@ import {
|
|||||||
RemoveOrgIDPResponse,
|
RemoveOrgIDPResponse,
|
||||||
RemoveOrgMemberRequest,
|
RemoveOrgMemberRequest,
|
||||||
RemoveOrgMemberResponse,
|
RemoveOrgMemberResponse,
|
||||||
|
RemoveOrgMetadataRequest,
|
||||||
|
RemoveOrgMetadataResponse,
|
||||||
RemovePersonalAccessTokenRequest,
|
RemovePersonalAccessTokenRequest,
|
||||||
RemovePersonalAccessTokenResponse,
|
RemovePersonalAccessTokenResponse,
|
||||||
RemoveProjectGrantMemberRequest,
|
RemoveProjectGrantMemberRequest,
|
||||||
@ -371,6 +375,8 @@ import {
|
|||||||
SetCustomVerifyPhoneMessageTextRequest,
|
SetCustomVerifyPhoneMessageTextRequest,
|
||||||
SetCustomVerifyPhoneMessageTextResponse,
|
SetCustomVerifyPhoneMessageTextResponse,
|
||||||
SetHumanInitialPasswordRequest,
|
SetHumanInitialPasswordRequest,
|
||||||
|
SetOrgMetadataRequest,
|
||||||
|
SetOrgMetadataResponse,
|
||||||
SetPrimaryOrgDomainRequest,
|
SetPrimaryOrgDomainRequest,
|
||||||
SetPrimaryOrgDomainResponse,
|
SetPrimaryOrgDomainResponse,
|
||||||
SetTriggerActionsRequest,
|
SetTriggerActionsRequest,
|
||||||
@ -1374,6 +1380,26 @@ export class ManagementService {
|
|||||||
return this.grpcService.mgmt.listUserMetadata(req, null).then((resp) => resp.toObject());
|
return this.grpcService.mgmt.listUserMetadata(req, null).then((resp) => resp.toObject());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public listOrgMetadata(
|
||||||
|
offset?: number,
|
||||||
|
limit?: number,
|
||||||
|
queryList?: MetadataQuery[],
|
||||||
|
): Promise<ListOrgMetadataResponse.AsObject> {
|
||||||
|
const req = new ListOrgMetadataRequest();
|
||||||
|
|
||||||
|
const metadata = new ListQuery();
|
||||||
|
if (offset) {
|
||||||
|
metadata.setOffset(offset);
|
||||||
|
}
|
||||||
|
if (limit) {
|
||||||
|
metadata.setLimit(limit);
|
||||||
|
}
|
||||||
|
if (queryList) {
|
||||||
|
req.setQueriesList(queryList);
|
||||||
|
}
|
||||||
|
return this.grpcService.mgmt.listOrgMetadata(req, null).then((resp) => resp.toObject());
|
||||||
|
}
|
||||||
|
|
||||||
public getUserMetadata(userId: string, key: string): Promise<GetUserMetadataResponse.AsObject> {
|
public getUserMetadata(userId: string, key: string): Promise<GetUserMetadataResponse.AsObject> {
|
||||||
const req = new GetUserMetadataRequest();
|
const req = new GetUserMetadataRequest();
|
||||||
req.setId(userId);
|
req.setId(userId);
|
||||||
@ -1389,6 +1415,13 @@ export class ManagementService {
|
|||||||
return this.grpcService.mgmt.setUserMetadata(req, null).then((resp) => resp.toObject());
|
return this.grpcService.mgmt.setUserMetadata(req, null).then((resp) => resp.toObject());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public setOrgMetadata(key: string, value: string): Promise<SetOrgMetadataResponse.AsObject> {
|
||||||
|
const req = new SetOrgMetadataRequest();
|
||||||
|
req.setKey(key);
|
||||||
|
req.setValue(value);
|
||||||
|
return this.grpcService.mgmt.setOrgMetadata(req, null).then((resp) => resp.toObject());
|
||||||
|
}
|
||||||
|
|
||||||
public bulkSetUserMetadata(
|
public bulkSetUserMetadata(
|
||||||
list: BulkSetUserMetadataRequest.Metadata[],
|
list: BulkSetUserMetadataRequest.Metadata[],
|
||||||
userId: string,
|
userId: string,
|
||||||
@ -1406,6 +1439,12 @@ export class ManagementService {
|
|||||||
return this.grpcService.mgmt.removeUserMetadata(req, null).then((resp) => resp.toObject());
|
return this.grpcService.mgmt.removeUserMetadata(req, null).then((resp) => resp.toObject());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public removeOrgMetadata(key: string): Promise<RemoveOrgMetadataResponse.AsObject> {
|
||||||
|
const req = new RemoveOrgMetadataRequest();
|
||||||
|
req.setKey(key);
|
||||||
|
return this.grpcService.mgmt.removeOrgMetadata(req, null).then((resp) => resp.toObject());
|
||||||
|
}
|
||||||
|
|
||||||
public removeUser(id: string): Promise<RemoveUserResponse.AsObject> {
|
public removeUser(id: string): Promise<RemoveUserResponse.AsObject> {
|
||||||
const req = new RemoveUserRequest();
|
const req = new RemoveUserRequest();
|
||||||
req.setId(id);
|
req.setId(id);
|
||||||
|
@ -294,7 +294,7 @@
|
|||||||
"TITLE": "Passwortlose Authentifizierungsmethoden",
|
"TITLE": "Passwortlose Authentifizierungsmethoden",
|
||||||
"DESCRIPTION": "Füge WebAuthn kompatible Authentifikatoren hinzu um dich passwortlos anzumelden.",
|
"DESCRIPTION": "Füge WebAuthn kompatible Authentifikatoren hinzu um dich passwortlos anzumelden.",
|
||||||
"MANAGE_DESCRIPTION": "Verwalte die Multifaktor-Merkmale Deiner Benutzer.",
|
"MANAGE_DESCRIPTION": "Verwalte die Multifaktor-Merkmale Deiner Benutzer.",
|
||||||
"U2F": "Authentifikator hinzufügen",
|
"U2F": "Methode hinzufügen",
|
||||||
"U2F_DIALOG_TITLE": "Authentifikator hinzufügen",
|
"U2F_DIALOG_TITLE": "Authentifikator hinzufügen",
|
||||||
"U2F_DIALOG_DESCRIPTION": "Gib einen Namen für den von dir verwendeten Login an.",
|
"U2F_DIALOG_DESCRIPTION": "Gib einen Namen für den von dir verwendeten Login an.",
|
||||||
"U2F_SUCCESS": "Passwortlos erfolgreich erstellt!",
|
"U2F_SUCCESS": "Passwortlos erfolgreich erstellt!",
|
||||||
@ -326,17 +326,6 @@
|
|||||||
"NEW": "Hinzufügen"
|
"NEW": "Hinzufügen"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"METADATA": {
|
|
||||||
"TITLE": "Metadata",
|
|
||||||
"DESCRIPTION": "",
|
|
||||||
"KEY": "Schlüssel",
|
|
||||||
"VALUE": "Wert",
|
|
||||||
"ADD": "Neues Element",
|
|
||||||
"SAVE": "Speichern",
|
|
||||||
"EMPTY": "Keine Metadaten",
|
|
||||||
"SETSUCCESS": "Element erfolgreich gespeichert",
|
|
||||||
"REMOVESUCCESS": "Element erfolgreich gelöscht"
|
|
||||||
},
|
|
||||||
"MFA": {
|
"MFA": {
|
||||||
"TABLETYPE": "Typ",
|
"TABLETYPE": "Typ",
|
||||||
"TABLESTATE": "Status",
|
"TABLESTATE": "Status",
|
||||||
@ -346,7 +335,7 @@
|
|||||||
"DESCRIPTION": "Füge einen zusätzlichen Faktor hinzu, um Dein Konto optimal zu schützen.",
|
"DESCRIPTION": "Füge einen zusätzlichen Faktor hinzu, um Dein Konto optimal zu schützen.",
|
||||||
"MANAGE_DESCRIPTION": "Verwalte die Multifaktor-Merkmale Deiner Benutzer.",
|
"MANAGE_DESCRIPTION": "Verwalte die Multifaktor-Merkmale Deiner Benutzer.",
|
||||||
"ADD": "Faktor hinzufügen",
|
"ADD": "Faktor hinzufügen",
|
||||||
"OTP": "OTP (One-Time Password)",
|
"OTP": "Authentikator App für OTP (One-Time Password)",
|
||||||
"OTP_DIALOG_TITLE": "OTP hinzufügen",
|
"OTP_DIALOG_TITLE": "OTP hinzufügen",
|
||||||
"OTP_DIALOG_DESCRIPTION": "Scanne den QR-Code mit einer Authenticator App und verifiziere den erhaltenen Code, um OTP zu aktivieren.",
|
"OTP_DIALOG_DESCRIPTION": "Scanne den QR-Code mit einer Authenticator App und verifiziere den erhaltenen Code, um OTP zu aktivieren.",
|
||||||
"U2F": "Fingerabdruck, Security Key, Face ID oder andere",
|
"U2F": "Fingerabdruck, Security Key, Face ID oder andere",
|
||||||
@ -639,6 +628,17 @@
|
|||||||
"DELETED": "Personal Access Token gelöscht."
|
"DELETED": "Personal Access Token gelöscht."
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"METADATA": {
|
||||||
|
"TITLE": "Metadata",
|
||||||
|
"DESCRIPTION": "",
|
||||||
|
"KEY": "Schlüssel",
|
||||||
|
"VALUE": "Wert",
|
||||||
|
"ADD": "Neues Element",
|
||||||
|
"SAVE": "Speichern",
|
||||||
|
"EMPTY": "Keine Metadaten",
|
||||||
|
"SETSUCCESS": "Element erfolgreich gespeichert",
|
||||||
|
"REMOVESUCCESS": "Element erfolgreich gelöscht"
|
||||||
|
},
|
||||||
"FLOWS": {
|
"FLOWS": {
|
||||||
"TITLE": "Aktionen und Abläufe",
|
"TITLE": "Aktionen und Abläufe",
|
||||||
"DESCRIPTION": "Hinterlege scripts die bei einem bestimmten Event ausgeführt werden.",
|
"DESCRIPTION": "Hinterlege scripts die bei einem bestimmten Event ausgeführt werden.",
|
||||||
|
@ -294,7 +294,7 @@
|
|||||||
"TITLE": "Passwordless Authentication",
|
"TITLE": "Passwordless Authentication",
|
||||||
"DESCRIPTION": "Add WebAuthn based Authentication Methods to log onto ZITADEL passwordless.",
|
"DESCRIPTION": "Add WebAuthn based Authentication Methods to log onto ZITADEL passwordless.",
|
||||||
"MANAGE_DESCRIPTION": "Manage the second factor methods of your users.",
|
"MANAGE_DESCRIPTION": "Manage the second factor methods of your users.",
|
||||||
"U2F": "Add authenticator",
|
"U2F": "Add method",
|
||||||
"U2F_DIALOG_TITLE": "Verify authenticator",
|
"U2F_DIALOG_TITLE": "Verify authenticator",
|
||||||
"U2F_DIALOG_DESCRIPTION": "Enter a name for your used passwordless Login",
|
"U2F_DIALOG_DESCRIPTION": "Enter a name for your used passwordless Login",
|
||||||
"U2F_SUCCESS": "Passwordless Auth created successfully!",
|
"U2F_SUCCESS": "Passwordless Auth created successfully!",
|
||||||
@ -326,17 +326,6 @@
|
|||||||
"NEW": "Add New"
|
"NEW": "Add New"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"METADATA": {
|
|
||||||
"TITLE": "Metadata",
|
|
||||||
"DESCRIPTION": "",
|
|
||||||
"KEY": "Key",
|
|
||||||
"VALUE": "Value",
|
|
||||||
"ADD": "New Entry",
|
|
||||||
"SAVE": "Save",
|
|
||||||
"EMPTY": "No metadata",
|
|
||||||
"SETSUCCESS": "Element saved successfully",
|
|
||||||
"REMOVESUCCESS": "Element deleted successfully"
|
|
||||||
},
|
|
||||||
"MFA": {
|
"MFA": {
|
||||||
"TABLETYPE": "Type",
|
"TABLETYPE": "Type",
|
||||||
"TABLESTATE": "Status",
|
"TABLESTATE": "Status",
|
||||||
@ -346,7 +335,7 @@
|
|||||||
"DESCRIPTION": "Add a second factor to ensure optimal security for your account.",
|
"DESCRIPTION": "Add a second factor to ensure optimal security for your account.",
|
||||||
"MANAGE_DESCRIPTION": "Manage the second factor methods of your users.",
|
"MANAGE_DESCRIPTION": "Manage the second factor methods of your users.",
|
||||||
"ADD": "Add Factor",
|
"ADD": "Add Factor",
|
||||||
"OTP": "OTP (One-Time Password)",
|
"OTP": "Authenticator App for OTP (One-Time Password)",
|
||||||
"OTP_DIALOG_TITLE": "Add OTP",
|
"OTP_DIALOG_TITLE": "Add OTP",
|
||||||
"OTP_DIALOG_DESCRIPTION": "Scan the QR code with an authenticator app and enter the code below to verify and activate the OTP method.",
|
"OTP_DIALOG_DESCRIPTION": "Scan the QR code with an authenticator app and enter the code below to verify and activate the OTP method.",
|
||||||
"U2F": "Fingerprint, Security Keys, Face ID and other",
|
"U2F": "Fingerprint, Security Keys, Face ID and other",
|
||||||
@ -639,6 +628,17 @@
|
|||||||
"DELETED": "Token deleted with success."
|
"DELETED": "Token deleted with success."
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"METADATA": {
|
||||||
|
"TITLE": "Metadata",
|
||||||
|
"DESCRIPTION": "",
|
||||||
|
"KEY": "Key",
|
||||||
|
"VALUE": "Value",
|
||||||
|
"ADD": "New Entry",
|
||||||
|
"SAVE": "Save",
|
||||||
|
"EMPTY": "No metadata",
|
||||||
|
"SETSUCCESS": "Element saved successfully",
|
||||||
|
"REMOVESUCCESS": "Element deleted successfully"
|
||||||
|
},
|
||||||
"FLOWS": {
|
"FLOWS": {
|
||||||
"TITLE": "Actions and Flows",
|
"TITLE": "Actions and Flows",
|
||||||
"DESCRIPTION": "Define scripts to execute on a certain event.",
|
"DESCRIPTION": "Define scripts to execute on a certain event.",
|
||||||
|
@ -294,7 +294,7 @@
|
|||||||
"TITLE": "Authentification sans mot de passe",
|
"TITLE": "Authentification sans mot de passe",
|
||||||
"DESCRIPTION": "Ajoutez des méthodes d'authentification basées sur WebAuthn pour vous connecter à ZITADEL sans mot de passe.",
|
"DESCRIPTION": "Ajoutez des méthodes d'authentification basées sur WebAuthn pour vous connecter à ZITADEL sans mot de passe.",
|
||||||
"MANAGE_DESCRIPTION": "Gérez les méthodes de second facteur de vos utilisateurs.",
|
"MANAGE_DESCRIPTION": "Gérez les méthodes de second facteur de vos utilisateurs.",
|
||||||
"U2F": "Ajouter un authentifiant",
|
"U2F": "Ajouter une méthode",
|
||||||
"U2F_DIALOG_TITLE": "Vérifier l'authentifiant",
|
"U2F_DIALOG_TITLE": "Vérifier l'authentifiant",
|
||||||
"U2F_DIALOG_DESCRIPTION": "Entrez un nom pour votre connexion sans mot de passe utilisée",
|
"U2F_DIALOG_DESCRIPTION": "Entrez un nom pour votre connexion sans mot de passe utilisée",
|
||||||
"U2F_SUCCESS": "Auth sans mot de passe créé avec succès !",
|
"U2F_SUCCESS": "Auth sans mot de passe créé avec succès !",
|
||||||
@ -326,17 +326,6 @@
|
|||||||
"NEW": "Ajouter un nouveau"
|
"NEW": "Ajouter un nouveau"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"METADATA": {
|
|
||||||
"TITLE": "Métadonnées",
|
|
||||||
"DESCRIPTION": "",
|
|
||||||
"KEY": "Clé",
|
|
||||||
"VALUE": "Valeur",
|
|
||||||
"ADD": "Nouvelle entrée",
|
|
||||||
"SAVE": "Enregistrer",
|
|
||||||
"EMPTY": "Pas de métadonnées",
|
|
||||||
"SETSUCCESS": "Élément sauvegardé avec succès",
|
|
||||||
"REMOVESUCCESS": "Élément supprimé avec succès"
|
|
||||||
},
|
|
||||||
"MFA": {
|
"MFA": {
|
||||||
"TABLETYPE": "Type",
|
"TABLETYPE": "Type",
|
||||||
"TABLESTATE": "Statut",
|
"TABLESTATE": "Statut",
|
||||||
@ -346,7 +335,7 @@
|
|||||||
"DESCRIPTION": "Ajoutez un second facteur pour garantir une sécurité optimale de votre compte.",
|
"DESCRIPTION": "Ajoutez un second facteur pour garantir une sécurité optimale de votre compte.",
|
||||||
"MANAGE_DESCRIPTION": "Gérez les méthodes de second facteur de vos utilisateurs.",
|
"MANAGE_DESCRIPTION": "Gérez les méthodes de second facteur de vos utilisateurs.",
|
||||||
"ADD": "Ajouter un facteur",
|
"ADD": "Ajouter un facteur",
|
||||||
"OTP": "OTP (mot de passe à usage unique)",
|
"OTP": "Application d'authentification pour OTP (One-time password)",
|
||||||
"OTP_DIALOG_TITLE": "Ajouter un OTP",
|
"OTP_DIALOG_TITLE": "Ajouter un OTP",
|
||||||
"OTP_DIALOG_DESCRIPTION": "Scannez le code QR avec une application d'authentification et saisissez le code ci-dessous pour vérifier et activer la méthode OTP.",
|
"OTP_DIALOG_DESCRIPTION": "Scannez le code QR avec une application d'authentification et saisissez le code ci-dessous pour vérifier et activer la méthode OTP.",
|
||||||
"U2F": "Empreinte digitale, clés de sécurité, Face ID et autres",
|
"U2F": "Empreinte digitale, clés de sécurité, Face ID et autres",
|
||||||
@ -639,6 +628,17 @@
|
|||||||
"DELETED": "Jeton supprimé avec succès."
|
"DELETED": "Jeton supprimé avec succès."
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"METADATA": {
|
||||||
|
"TITLE": "Métadonnées",
|
||||||
|
"DESCRIPTION": "",
|
||||||
|
"KEY": "Clé",
|
||||||
|
"VALUE": "Valeur",
|
||||||
|
"ADD": "Nouvelle entrée",
|
||||||
|
"SAVE": "Enregistrer",
|
||||||
|
"EMPTY": "Pas de métadonnées",
|
||||||
|
"SETSUCCESS": "Élément sauvegardé avec succès",
|
||||||
|
"REMOVESUCCESS": "Élément supprimé avec succès"
|
||||||
|
},
|
||||||
"FLOWS": {
|
"FLOWS": {
|
||||||
"TITLE": "Actions et flux",
|
"TITLE": "Actions et flux",
|
||||||
"DESCRIPTION": "Définissez des scripts à exécuter lors d'un certain événement.",
|
"DESCRIPTION": "Définissez des scripts à exécuter lors d'un certain événement.",
|
||||||
|
@ -294,7 +294,7 @@
|
|||||||
"TITLE": "Autenticazione passwordless",
|
"TITLE": "Autenticazione passwordless",
|
||||||
"DESCRIPTION": "Aggiungi i metodi di autenticazione basati su WebAuthn per accedere a ZITADEL senza password.",
|
"DESCRIPTION": "Aggiungi i metodi di autenticazione basati su WebAuthn per accedere a ZITADEL senza password.",
|
||||||
"MANAGE_DESCRIPTION": "Gestisci i metodi del secondo fattore dei vostri utenti.",
|
"MANAGE_DESCRIPTION": "Gestisci i metodi del secondo fattore dei vostri utenti.",
|
||||||
"U2F": "Aggiungi autenticatore",
|
"U2F": "Aggiungi metodo",
|
||||||
"U2F_DIALOG_TITLE": "Verifica autenticatore",
|
"U2F_DIALOG_TITLE": "Verifica autenticatore",
|
||||||
"U2F_DIALOG_DESCRIPTION": "Inserisci un nome per il tuo authenticatore o dispositivo usato.",
|
"U2F_DIALOG_DESCRIPTION": "Inserisci un nome per il tuo authenticatore o dispositivo usato.",
|
||||||
"U2F_SUCCESS": "Autorizzazione passwordless creata con successo!",
|
"U2F_SUCCESS": "Autorizzazione passwordless creata con successo!",
|
||||||
@ -326,17 +326,6 @@
|
|||||||
"NEW": "Aggiungi nuovo"
|
"NEW": "Aggiungi nuovo"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"METADATA": {
|
|
||||||
"TITLE": "Metadati",
|
|
||||||
"DESCRIPTION": "",
|
|
||||||
"KEY": "Chiave",
|
|
||||||
"VALUE": "Valore",
|
|
||||||
"ADD": "Nuova voce",
|
|
||||||
"SAVE": "Salva",
|
|
||||||
"EMPTY": "Nessun metadato",
|
|
||||||
"SETSUCCESS": "Salvato con successo",
|
|
||||||
"REMOVESUCCESS": "Rimosso con successo"
|
|
||||||
},
|
|
||||||
"MFA": {
|
"MFA": {
|
||||||
"TABLETYPE": "Tipo",
|
"TABLETYPE": "Tipo",
|
||||||
"TABLESTATE": "Stato",
|
"TABLESTATE": "Stato",
|
||||||
@ -346,7 +335,7 @@
|
|||||||
"DESCRIPTION": "Aggiungi un secondo fattore per garantire la sicurezza ottimale del tuo account.",
|
"DESCRIPTION": "Aggiungi un secondo fattore per garantire la sicurezza ottimale del tuo account.",
|
||||||
"MANAGE_DESCRIPTION": "Gestite i metodi del secondo fattore dei vostri utenti.",
|
"MANAGE_DESCRIPTION": "Gestite i metodi del secondo fattore dei vostri utenti.",
|
||||||
"ADD": "Aggiungi fattore",
|
"ADD": "Aggiungi fattore",
|
||||||
"OTP": "OTP (One-Time Password)",
|
"OTP": "App di autenticazione per OTP (One-Time Password)",
|
||||||
"OTP_DIALOG_TITLE": "Aggiungi OTP",
|
"OTP_DIALOG_TITLE": "Aggiungi OTP",
|
||||||
"OTP_DIALOG_DESCRIPTION": "Scansiona il codice QR con un'app di autenticazione e inserisci il codice nel campo sottostante per verificare e attivare il metodo OTP.",
|
"OTP_DIALOG_DESCRIPTION": "Scansiona il codice QR con un'app di autenticazione e inserisci il codice nel campo sottostante per verificare e attivare il metodo OTP.",
|
||||||
"U2F": "Impronta digitale, chiave di sicurezza, Face ID e altri",
|
"U2F": "Impronta digitale, chiave di sicurezza, Face ID e altri",
|
||||||
@ -639,6 +628,17 @@
|
|||||||
"DELETED": "Token eliminato con successo."
|
"DELETED": "Token eliminato con successo."
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"METADATA": {
|
||||||
|
"TITLE": "Metadati",
|
||||||
|
"DESCRIPTION": "",
|
||||||
|
"KEY": "Chiave",
|
||||||
|
"VALUE": "Valore",
|
||||||
|
"ADD": "Nuova voce",
|
||||||
|
"SAVE": "Salva",
|
||||||
|
"EMPTY": "Nessun metadato",
|
||||||
|
"SETSUCCESS": "Salvato con successo",
|
||||||
|
"REMOVESUCCESS": "Rimosso con successo"
|
||||||
|
},
|
||||||
"FLOWS": {
|
"FLOWS": {
|
||||||
"TITLE": "Azioni e Processi",
|
"TITLE": "Azioni e Processi",
|
||||||
"DESCRIPTION": "Esegui processi su certi eventi.",
|
"DESCRIPTION": "Esegui processi su certi eventi.",
|
||||||
|
@ -294,7 +294,7 @@
|
|||||||
"TITLE": "无密码身份验证",
|
"TITLE": "无密码身份验证",
|
||||||
"DESCRIPTION": "添加基于 WebAuthn 的身份验证方法以无密码登录 ZITADEL。",
|
"DESCRIPTION": "添加基于 WebAuthn 的身份验证方法以无密码登录 ZITADEL。",
|
||||||
"MANAGE_DESCRIPTION": "管理用户的第二因素认证方式。",
|
"MANAGE_DESCRIPTION": "管理用户的第二因素认证方式。",
|
||||||
"U2F": "添加验证器",
|
"U2F": "添加方法",
|
||||||
"U2F_DIALOG_TITLE": "身份验证器",
|
"U2F_DIALOG_TITLE": "身份验证器",
|
||||||
"U2F_DIALOG_DESCRIPTION": "输入您使用的无密码登录名",
|
"U2F_DIALOG_DESCRIPTION": "输入您使用的无密码登录名",
|
||||||
"U2F_SUCCESS": "无密码验证创建成功!",
|
"U2F_SUCCESS": "无密码验证创建成功!",
|
||||||
@ -326,17 +326,6 @@
|
|||||||
"NEW": "添加"
|
"NEW": "添加"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"METADATA": {
|
|
||||||
"TITLE": "元数据",
|
|
||||||
"DESCRIPTION": "",
|
|
||||||
"KEY": "键",
|
|
||||||
"VALUE": "值",
|
|
||||||
"ADD": "新条目",
|
|
||||||
"SAVE": "保存",
|
|
||||||
"EMPTY": "暂无元数据",
|
|
||||||
"SETSUCCESS": "键值对保存成功",
|
|
||||||
"REMOVESUCCESS": "键值对删除成功"
|
|
||||||
},
|
|
||||||
"MFA": {
|
"MFA": {
|
||||||
"TABLETYPE": "类型",
|
"TABLETYPE": "类型",
|
||||||
"TABLESTATE": "状态",
|
"TABLESTATE": "状态",
|
||||||
@ -346,7 +335,7 @@
|
|||||||
"DESCRIPTION": "添加第二个因素以确保您帐户的最佳安全性。",
|
"DESCRIPTION": "添加第二个因素以确保您帐户的最佳安全性。",
|
||||||
"MANAGE_DESCRIPTION": "管理用户的第二因素身份认证方式。",
|
"MANAGE_DESCRIPTION": "管理用户的第二因素身份认证方式。",
|
||||||
"ADD": "添加因子",
|
"ADD": "添加因子",
|
||||||
"OTP": "一次性密码 (OTP)",
|
"OTP": "用于 OTP 的身份验证器应用程序",
|
||||||
"OTP_DIALOG_TITLE": "添加 OTP",
|
"OTP_DIALOG_TITLE": "添加 OTP",
|
||||||
"OTP_DIALOG_DESCRIPTION": "使用验证器应用程序扫描二维码并输入下面的代码以验证并激活 OTP 方法。",
|
"OTP_DIALOG_DESCRIPTION": "使用验证器应用程序扫描二维码并输入下面的代码以验证并激活 OTP 方法。",
|
||||||
"U2F": "指纹、安全密钥、Face ID 等",
|
"U2F": "指纹、安全密钥、Face ID 等",
|
||||||
@ -639,6 +628,17 @@
|
|||||||
"DELETED": "成功删除令牌。"
|
"DELETED": "成功删除令牌。"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"METADATA": {
|
||||||
|
"TITLE": "元数据",
|
||||||
|
"DESCRIPTION": "",
|
||||||
|
"KEY": "键",
|
||||||
|
"VALUE": "值",
|
||||||
|
"ADD": "新条目",
|
||||||
|
"SAVE": "保存",
|
||||||
|
"EMPTY": "暂无元数据",
|
||||||
|
"SETSUCCESS": "键值对保存成功",
|
||||||
|
"REMOVESUCCESS": "键值对删除成功"
|
||||||
|
},
|
||||||
"FLOWS": {
|
"FLOWS": {
|
||||||
"TITLE": "动作和流程",
|
"TITLE": "动作和流程",
|
||||||
"DESCRIPTION": "定义要在特定事件上需要执行的脚本。",
|
"DESCRIPTION": "定义要在特定事件上需要执行的脚本。",
|
||||||
|
Loading…
x
Reference in New Issue
Block a user