feat(console): limited actions (#3164)

* max count features

* deactivate, activate

* actions, limited

* disable without permission, show action state in flow

Co-authored-by: Livio Amstutz <livio.a@gmail.com>
This commit is contained in:
Max Peintner 2022-02-07 14:53:35 +01:00 committed by GitHub
parent 78af86db98
commit 3bf9adece5
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
13 changed files with 276 additions and 85 deletions

View File

@ -265,6 +265,8 @@
</mat-slide-toggle> </mat-slide-toggle>
</div> </div>
<p class="feature-section">{{'FEATURES.HEADERS.FLOWS' | translate}}</p>
<div class="row"> <div class="row">
<div class="featureavatar pink"> <div class="featureavatar pink">
<i class="icon las la-exchange-alt"></i> <i class="icon las la-exchange-alt"></i>
@ -272,9 +274,25 @@
<span class="left-desc">{{'FEATURES.DATA.FLOWS' | translate}}</span> <span class="left-desc">{{'FEATURES.DATA.FLOWS' | translate}}</span>
<span class="fill-space"></span> <span class="fill-space"></span>
<template [ngTemplateOutlet]="templateRef" [ngTemplateOutletContext]="{active: features.actions}"></template> <template [ngTemplateOutlet]="templateRef" [ngTemplateOutletContext]="{active: features.actions}"></template>
<mat-slide-toggle class="toggle" color="primary" name="hasNumber" ngDefaultControl [(ngModel)]="features.actions" </div>
*ngIf="(['iam.features.write'] | hasRole | async)">
</mat-slide-toggle> <div class="row">
<span class="fill-space"></span>
<cnsl-form-field class="flow-select">
<cnsl-label>{{ 'FEATURES.DATA.FLOW.TYPE' | translate }}</cnsl-label>
<mat-select [(ngModel)]="features.actionsAllowed"
[disabled]="(['iam.features.write'] | hasRole | async) === false">
<mat-option *ngFor="let allowedType of actionsSelection" [value]="allowedType">
{{ 'FEATURES.DATA.FLOW.ACTIONSALLOWED.'+allowedType | translate}}
</mat-option>
</mat-select>
</cnsl-form-field>
<cnsl-form-field *ngIf="features.actionsAllowed === ActionsAllowed.ACTIONS_ALLOWED_MAX" class="flow-count">
<cnsl-label>{{ 'FEATURES.DATA.FLOW.COUNT' | translate }}</cnsl-label>
<input cnslInput type="number" [(ngModel)]="features.maxActions"
[disabled]="(['iam.features.write'] | hasRole | async) === false" ngDefaultControl />
</cnsl-form-field>
</div> </div>
</div> </div>

View File

@ -17,10 +17,10 @@
.title { .title {
font-size: 14px; font-size: 14px;
color: var(--grey); color: var(--grey);
margin-bottom: .5rem; margin-bottom: 0.5rem;
a { a {
margin-left: .5rem; margin-left: 0.5rem;
cursor: pointer; cursor: pointer;
} }
} }
@ -30,19 +30,19 @@
align-items: center; align-items: center;
a { a {
margin-left: .5rem; margin-left: 0.5rem;
} }
} }
img { img {
height: 15px; height: 15px;
width: auto; width: auto;
margin-left: .5rem; margin-left: 0.5rem;
} }
} }
.spinner { .spinner {
margin: .5rem; margin: 0.5rem;
} }
.error { .error {
@ -60,8 +60,8 @@
height: 1px; height: 1px;
width: 100%; width: 100%;
background-color: var(--grey); background-color: var(--grey);
opacity: .5; opacity: 0.5;
margin: .5rem 0; margin: 0.5rem 0;
display: block; display: block;
} }
@ -80,7 +80,7 @@
.row { .row {
display: flex; display: flex;
align-items: center; align-items: center;
padding: .3rem 0; padding: 0.3rem 0;
.featureavatar { .featureavatar {
margin-right: 1rem; margin-right: 1rem;
@ -137,7 +137,7 @@
} }
.left-desc { .left-desc {
font-size: .9rem; font-size: 0.9rem;
margin-right: 1rem; margin-right: 1rem;
} }
@ -149,6 +149,18 @@
display: flex; display: flex;
align-items: center; align-items: center;
} }
.flow-select {
flex-shrink: 1;
min-width: 150px;
margin-left: 1rem;
}
.flow-count {
flex-shrink: 1;
width: 70px;
margin-left: 1rem;
}
} }
} }

View File

@ -8,7 +8,7 @@ import {
SetDefaultFeaturesRequest, SetDefaultFeaturesRequest,
SetOrgFeaturesRequest, SetOrgFeaturesRequest,
} from 'src/app/proto/generated/zitadel/admin_pb'; } from 'src/app/proto/generated/zitadel/admin_pb';
import { Features } from 'src/app/proto/generated/zitadel/features_pb'; import { ActionsAllowed, Features } from 'src/app/proto/generated/zitadel/features_pb';
import { GetFeaturesResponse } from 'src/app/proto/generated/zitadel/management_pb'; import { GetFeaturesResponse } from 'src/app/proto/generated/zitadel/management_pb';
import { Org } from 'src/app/proto/generated/zitadel/org_pb'; import { Org } from 'src/app/proto/generated/zitadel/org_pb';
import { AdminService } from 'src/app/services/admin.service'; import { AdminService } from 'src/app/services/admin.service';
@ -46,6 +46,13 @@ export class FeaturesComponent implements OnDestroy {
public stripeURL: string = ''; public stripeURL: string = '';
public stripeCustomer!: StripeCustomer; public stripeCustomer!: StripeCustomer;
public actionsSelection: any = [
ActionsAllowed.ACTIONS_ALLOWED_NOT_ALLOWED,
ActionsAllowed.ACTIONS_ALLOWED_MAX,
ActionsAllowed.ACTIONS_ALLOWED_UNLIMITED,
];
public ActionsAllowed: any = ActionsAllowed;
constructor( constructor(
private route: ActivatedRoute, private route: ActivatedRoute,
private toast: ToastService, private toast: ToastService,
@ -60,27 +67,32 @@ export class FeaturesComponent implements OnDestroy {
if (temporg) { if (temporg) {
this.org = temporg; this.org = temporg;
} }
this.sub = this.route.data.pipe(switchMap(data => { this.sub = this.route.data
.pipe(
switchMap((data) => {
this.serviceType = data.serviceType; this.serviceType = data.serviceType;
if (this.serviceType === FeatureServiceType.MGMT) { if (this.serviceType === FeatureServiceType.MGMT) {
this.managementService = this.injector.get(ManagementService as Type<ManagementService>); this.managementService = this.injector.get(ManagementService as Type<ManagementService>);
} }
return this.route.params; return this.route.params;
})).subscribe(_ => { }),
)
.subscribe((_) => {
this.fetchData(); this.fetchData();
}); });
if (this.serviceType === FeatureServiceType.MGMT) { if (this.serviceType === FeatureServiceType.MGMT) {
this.customerLoading = true; this.customerLoading = true;
this.subService.getCustomer(this.org.id) this.subService
.then(payload => { .getCustomer(this.org.id)
.then((payload) => {
this.customerLoading = false; this.customerLoading = false;
this.stripeCustomer = payload; this.stripeCustomer = payload;
if (this.customerValid) { if (this.customerValid) {
this.getLinkToStripe(); this.getLinkToStripe();
} }
}) })
.catch(error => { .catch((error) => {
this.customerLoading = false; this.customerLoading = false;
console.error(error); console.error(error);
}); });
@ -99,13 +111,16 @@ export class FeaturesComponent implements OnDestroy {
width: '400px', width: '400px',
}); });
dialogRefPhone.afterClosed().subscribe(customer => { dialogRefPhone.afterClosed().subscribe((customer) => {
if (customer) { if (customer) {
console.log(customer); console.log(customer);
this.stripeCustomer = customer; this.stripeCustomer = customer;
this.subService.setCustomer(this.org.id, customer).then(() => { this.subService
.setCustomer(this.org.id, customer)
.then(() => {
this.getLinkToStripe(); this.getLinkToStripe();
}).catch(console.error); })
.catch(console.error);
} }
}); });
} }
@ -113,12 +128,13 @@ export class FeaturesComponent implements OnDestroy {
public getLinkToStripe(): void { public getLinkToStripe(): void {
if (this.serviceType === FeatureServiceType.MGMT) { if (this.serviceType === FeatureServiceType.MGMT) {
this.stripeLoading = true; this.stripeLoading = true;
this.subService.getLink(this.org.id, window.location.href) this.subService
.then(payload => { .getLink(this.org.id, window.location.href)
.then((payload) => {
this.stripeLoading = false; this.stripeLoading = false;
this.stripeURL = payload.redirect_url; this.stripeURL = payload.redirect_url;
}) })
.catch(error => { .catch((error) => {
this.stripeLoading = false; this.stripeLoading = false;
console.error(error); console.error(error);
}); });
@ -126,7 +142,7 @@ export class FeaturesComponent implements OnDestroy {
} }
public fetchData(): void { public fetchData(): void {
this.getData().then(resp => { this.getData().then((resp) => {
if (resp?.features) { if (resp?.features) {
this.features = resp.features; this.features = resp.features;
} }
@ -167,11 +183,16 @@ export class FeaturesComponent implements OnDestroy {
req.setPrivacyPolicy(this.features.privacyPolicy); req.setPrivacyPolicy(this.features.privacyPolicy);
req.setMetadataUser(this.features.metadataUser); req.setMetadataUser(this.features.metadataUser);
req.setLockoutPolicy(this.features.lockoutPolicy); req.setLockoutPolicy(this.features.lockoutPolicy);
req.setActions(this.features.actions); // req.setActions(this.features.actions);
req.setActionsAllowed(this.features.actionsAllowed);
req.setMaxActions(this.features.maxActions);
this.adminService.setOrgFeatures(req).then(() => { this.adminService
.setOrgFeatures(req)
.then(() => {
this.toast.showInfo('POLICY.TOAST.SET', true); this.toast.showInfo('POLICY.TOAST.SET', true);
}).catch(error => { })
.catch((error) => {
this.toast.showError(error); this.toast.showError(error);
}); });
break; break;
@ -192,11 +213,16 @@ export class FeaturesComponent implements OnDestroy {
dreq.setCustomTextMessage(this.features.customTextMessage); dreq.setCustomTextMessage(this.features.customTextMessage);
dreq.setMetadataUser(this.features.metadataUser); dreq.setMetadataUser(this.features.metadataUser);
dreq.setLockoutPolicy(this.features.lockoutPolicy); dreq.setLockoutPolicy(this.features.lockoutPolicy);
dreq.setActions(this.features.actions); // dreq.setActions(this.features.actions);
dreq.setActionsAllowed(this.features.actionsAllowed);
dreq.setMaxActions(this.features.maxActions);
this.adminService.setDefaultFeatures(dreq).then(() => { this.adminService
.setDefaultFeatures(dreq)
.then(() => {
this.toast.showInfo('POLICY.TOAST.SET', true); this.toast.showInfo('POLICY.TOAST.SET', true);
}).catch(error => { })
.catch((error) => {
this.toast.showError(error); this.toast.showError(error);
}); });
break; break;
@ -205,12 +231,15 @@ export class FeaturesComponent implements OnDestroy {
public resetFeatures(): void { public resetFeatures(): void {
if (this.serviceType === FeatureServiceType.MGMT) { if (this.serviceType === FeatureServiceType.MGMT) {
this.adminService.resetOrgFeatures(this.org.id).then(() => { this.adminService
.resetOrgFeatures(this.org.id)
.then(() => {
this.toast.showInfo('POLICY.TOAST.RESETSUCCESS', true); this.toast.showInfo('POLICY.TOAST.RESETSUCCESS', true);
setTimeout(() => { setTimeout(() => {
this.fetchData(); this.fetchData();
}, 1000); }, 1000);
}).catch(error => { })
.catch((error) => {
this.toast.showError(error); this.toast.showError(error);
}); });
} }
@ -225,13 +254,15 @@ export class FeaturesComponent implements OnDestroy {
} }
get customerValid(): boolean { get customerValid(): boolean {
return !!this.stripeCustomer?.contact && return (
!!this.stripeCustomer?.contact &&
!!this.stripeCustomer?.address && !!this.stripeCustomer?.address &&
!!this.stripeCustomer?.city && !!this.stripeCustomer?.city &&
!!this.stripeCustomer?.postal_code; !!this.stripeCustomer?.postal_code
);
} }
get customerCountry(): Country | undefined { get customerCountry(): Country | undefined {
return COUNTRIES.find(country => country.isoCode === this.stripeCustomer.country); return COUNTRIES.find((country) => country.isoCode === this.stripeCustomer.country);
} }
} }

View File

@ -1,10 +1,19 @@
<cnsl-refresh-table [loading]="loading$ | async" (refreshed)="refreshPage()" [dataSize]="dataSource?.data?.length ?? 0" <cnsl-refresh-table [loading]="loading$ | async" (refreshed)="refreshPage()" [dataSize]="dataSource?.data?.length ?? 0"
[timestamp]="actionsResult?.details?.viewTimestamp" [selection]="selection"> [timestamp]="actionsResult?.details?.viewTimestamp" [selection]="selection">
<div actions> <div actions *ngIf="selection.isEmpty()">
<a color="primary" mat-raised-button (click)="openAddAction()"> <a color="primary" mat-raised-button (click)="openAddAction()">
<mat-icon class="icon">add</mat-icon>{{ 'ACTIONS.NEW' | translate }} <mat-icon class="icon">add</mat-icon>{{ 'ACTIONS.NEW' | translate }}
</a> </a>
</div> </div>
<div actions *ngIf="!selection.isEmpty()">
<button class="action-state-btn" mat-stroked-button (click)="deactivateSelection()">
{{ 'ACTIONS.DEACTIVATE' | translate }}
</button>
<button class="action-state-btn" mat-stroked-button (click)="activateSelection()">
{{ 'ACTIONS.REACTIVATE' | translate }}
</button>
</div>
<div class="table-wrapper"> <div class="table-wrapper">
<table class="table" mat-table [dataSource]="dataSource"> <table class="table" mat-table [dataSource]="dataSource">

View File

@ -1,3 +1,6 @@
.action-state-btn {
margin-left: 0.5rem;
}
.table-wrapper { .table-wrapper {
overflow: auto; overflow: auto;

View File

@ -137,4 +137,40 @@ export class ActionTableComponent implements OnInit {
public refreshPage(): void { public refreshPage(): void {
this.getData(this.paginator.pageSize, this.paginator.pageIndex * this.paginator.pageSize); this.getData(this.paginator.pageSize, this.paginator.pageIndex * this.paginator.pageSize);
} }
public deactivateSelection(): Promise<void> {
const prom = this.selection.selected.map((action) => {
return this.mgmtService.deactivateAction(action.id);
});
return Promise.all(prom)
.then(() => {
this.selection.clear();
this.toast.showInfo('FLOWS.TOAST.ACTIONDEACTIVATED', true);
this.getData(10, 0);
})
.catch((error) => {
this.selection.clear();
this.toast.showError(error);
this.getData(10, 0);
});
}
public activateSelection(): Promise<void> {
const prom = this.selection.selected.map((action) => {
return this.mgmtService.reactivateAction(action.id);
});
return Promise.all(prom)
.then(() => {
this.selection.clear();
this.toast.showInfo('FLOWS.TOAST.ACTIONREACTIVATED', true);
this.getData(10, 0);
})
.catch((error) => {
this.selection.clear();
this.toast.showError(error);
this.getData(10, 0);
});
}
} }

View File

@ -2,6 +2,9 @@
<h1>{{ 'FLOWS.TITLE' | translate }}</h1> <h1>{{ 'FLOWS.TITLE' | translate }}</h1>
<p class="desc">{{'FLOWS.DESCRIPTION' | translate }}</p> <p class="desc">{{'FLOWS.DESCRIPTION' | translate }}</p>
<cnsl-info-section class="max-actions" *ngIf="maxActions">{{'FLOWS.ACTIONSMAX' | translate: ({value: maxActions}) }}
</cnsl-info-section>
<cnsl-info-section *ngIf="(['actions'] | hasFeature | async) === false" [featureLink]="['/org/features']" class="info" <cnsl-info-section *ngIf="(['actions'] | hasFeature | async) === false" [featureLink]="['/org/features']" class="info"
[type]="InfoSectionType.WARN"> [type]="InfoSectionType.WARN">
<span [innerHTML]="'FEATURES.NOTAVAILABLE' | translate: ({value: 'actions'})"></span> <span [innerHTML]="'FEATURES.NOTAVAILABLE' | translate: ({value: 'actions'})"></span>
@ -44,11 +47,15 @@
<mat-icon svgIcon="mdi_arrow_right_bottom" class="icon"></mat-icon> <mat-icon svgIcon="mdi_arrow_right_bottom" class="icon"></mat-icon>
<span>{{'FLOWS.TRIGGERTYPES.'+trigger.triggerType | translate}}</span> <span>{{'FLOWS.TRIGGERTYPES.'+trigger.triggerType | translate}}</span>
<span class="fill-space"></span> <span class="fill-space"></span>
<div class="action-wrapper" cdkDropList (cdkDropListDropped)="drop(i, trigger.actionsList, $event)"> <div class="flow-action-wrapper" cdkDropList (cdkDropListDropped)="drop(i, trigger.actionsList, $event)">
<div cdkDrag cdkDragLockAxis="y" cdkDragBoundary=".action-wrapper" class="action" <div cdkDrag cdkDragLockAxis="y" cdkDragBoundary=".action-wrapper" class="flow-action"
*ngFor="let action of trigger.actionsList"> *ngFor="let action of trigger.actionsList">
<i class="las la-code"></i> <i class="las la-code"></i>
<span>{{action.name}}</span> <span class="flow-action-name">{{action.name}}</span>
<span class="fill-space"></span>
<span class="state"
[ngClass]="{'active': action.state === ActionState.ACTION_STATE_ACTIVE,'inactive': action.state === ActionState.ACTION_STATE_INACTIVE }">
{{'FLOWS.STATES.'+action.state | translate}}</span>
</div> </div>
</div> </div>
</div> </div>

View File

@ -18,7 +18,7 @@ h1 {
} }
i { i {
margin-left: .5rem; margin-left: 0.5rem;
} }
} }
@ -29,8 +29,8 @@ h1 {
.flow-type { .flow-type {
padding: 1rem 1rem; padding: 1rem 1rem;
margin: .5rem 0; margin: 0.5rem 0;
border-radius: .5rem; border-radius: 0.5rem;
display: flex; display: flex;
align-items: center; align-items: center;
justify-content: space-between; justify-content: space-between;
@ -39,7 +39,7 @@ h1 {
.topelements { .topelements {
border: 3px solid var(--color-main); border: 3px solid var(--color-main);
border-radius: 1rem; border-radius: 1rem;
padding: 0 .5rem; padding: 0 0.5rem;
} }
.trigger-wrapper { .trigger-wrapper {
@ -57,13 +57,13 @@ h1 {
} }
.trigger { .trigger {
padding: .5rem 1rem; padding: 0.5rem 1rem;
border-radius: .5rem; border-radius: 0.5rem;
display: flex; display: flex;
align-items: center; align-items: center;
background: var(--color-main); background: var(--color-main);
color: white; color: white;
margin: .5rem 0; margin: 0.5rem 0;
min-height: 40px; min-height: 40px;
.icon { .icon {
@ -74,18 +74,27 @@ h1 {
flex: 1; flex: 1;
} }
.action-wrapper { .flow-action-wrapper {
padding: 0 .5rem; padding: 0 0.5rem;
margin: 0;
.action { .flow-action {
display: flex; display: flex;
align-items: center; align-items: center;
font-size: 14px; font-size: 14px;
padding: .5rem 0; padding: 0.5rem 0;
cursor: move; cursor: move;
.flow-action-name {
margin-right: 1rem;
}
.fill-space {
flex: 1;
}
i { i {
margin-right: .5rem; margin-right: 0.5rem;
} }
} }
} }
@ -105,13 +114,13 @@ h1 {
display: flex; display: flex;
align-items: center; align-items: center;
font-size: 14px; font-size: 14px;
border-radius: .5rem; border-radius: 0.5rem;
padding: 0 .5rem; padding: 0 0.5rem;
background-color: var(--color-main); background-color: var(--color-main);
box-shadow: 0 5px 5px -3px rgba(0, 0, 0, .2), 0 8px 10px 1px rgba(0, 0, 0, .14), 0 3px 14px 2px rgba(0, 0, 0, .12); box-shadow: 0 5px 5px -3px rgba(0, 0, 0, 0.2), 0 8px 10px 1px rgba(0, 0, 0, 0.14), 0 3px 14px 2px rgba(0, 0, 0, 0.12);
i { i {
margin-right: .5rem; margin-right: 0.5rem;
} }
} }
@ -120,5 +129,5 @@ h1 {
} }
.cdk-drag-animating { .cdk-drag-animating {
transition: transform 250ms cubic-bezier(0, 0, .2, 1); transition: transform 250ms cubic-bezier(0, 0, 0.2, 1);
} }

View File

@ -4,7 +4,8 @@ import { FormControl } from '@angular/forms';
import { MatDialog } from '@angular/material/dialog'; import { MatDialog } from '@angular/material/dialog';
import { InfoSectionType } from 'src/app/modules/info-section/info-section.component'; import { InfoSectionType } from 'src/app/modules/info-section/info-section.component';
import { WarnDialogComponent } from 'src/app/modules/warn-dialog/warn-dialog.component'; import { WarnDialogComponent } from 'src/app/modules/warn-dialog/warn-dialog.component';
import { Action, Flow, FlowType, TriggerType } from 'src/app/proto/generated/zitadel/action_pb'; import { Action, ActionState, Flow, FlowType, TriggerType } from 'src/app/proto/generated/zitadel/action_pb';
import { ActionsAllowed } from 'src/app/proto/generated/zitadel/features_pb';
import { SetTriggerActionsRequest } from 'src/app/proto/generated/zitadel/management_pb'; import { SetTriggerActionsRequest } from 'src/app/proto/generated/zitadel/management_pb';
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';
@ -26,14 +27,28 @@ export class ActionsComponent {
public selection: Action.AsObject[] = []; public selection: Action.AsObject[] = [];
public InfoSectionType: any = InfoSectionType; public InfoSectionType: any = InfoSectionType;
public maxActions: number | null = null;
public ActionState: any = ActionState;
constructor(private mgmtService: ManagementService, private dialog: MatDialog, private toast: ToastService) { constructor(private mgmtService: ManagementService, private dialog: MatDialog, private toast: ToastService) {
this.mgmtService.getFeatures().then((featuresResp) => {
if (featuresResp && featuresResp.features) {
const features = featuresResp.features;
this.maxActions =
features && features.actionsAllowed === ActionsAllowed.ACTIONS_ALLOWED_MAX
? features.maxActions
: features.actionsAllowed === ActionsAllowed.ACTIONS_ALLOWED_MAX
? null
: 0;
}
});
this.loadFlow(); this.loadFlow();
} }
private loadFlow() { private loadFlow() {
this.mgmtService.getFlow(this.flowType).then((flowResponse) => { this.mgmtService.getFlow(this.flowType).then((flowResponse) => {
if (flowResponse.flow) this.flow = flowResponse.flow; if (flowResponse.flow) this.flow = flowResponse.flow;
console.log(this.flow);
}); });
} }
@ -92,7 +107,6 @@ export class ActionsComponent {
} }
saveFlow(index: number) { saveFlow(index: number) {
console.log(this.flow.triggerActionsList[index].actionsList.map((action) => action.id));
this.mgmtService this.mgmtService
.setTriggerActions( .setTriggerActions(
this.flow.triggerActionsList[index].actionsList.map((action) => action.id), this.flow.triggerActionsList[index].actionsList.map((action) => action.id),

View File

@ -74,6 +74,8 @@ import {
ClearFlowResponse, ClearFlowResponse,
CreateActionRequest, CreateActionRequest,
CreateActionResponse, CreateActionResponse,
DeactivateActionRequest,
DeactivateActionResponse,
DeactivateAppRequest, DeactivateAppRequest,
DeactivateAppResponse, DeactivateAppResponse,
DeactivateOrgIDPRequest, DeactivateOrgIDPRequest,
@ -238,6 +240,8 @@ import {
ListUserMetadataResponse, ListUserMetadataResponse,
ListUsersRequest, ListUsersRequest,
ListUsersResponse, ListUsersResponse,
ReactivateActionRequest,
ReactivateActionResponse,
ReactivateAppRequest, ReactivateAppRequest,
ReactivateAppResponse, ReactivateAppResponse,
ReactivateOrgIDPRequest, ReactivateOrgIDPRequest,
@ -919,6 +923,18 @@ export class ManagementService {
return this.grpcService.mgmt.deleteAction(req, null).then((resp) => resp.toObject()); return this.grpcService.mgmt.deleteAction(req, null).then((resp) => resp.toObject());
} }
public deactivateAction(id: string): Promise<DeactivateActionResponse.AsObject> {
const req = new DeactivateActionRequest();
req.setId(id);
return this.grpcService.mgmt.deactivateAction(req, null).then((resp) => resp.toObject());
}
public reactivateAction(id: string): Promise<ReactivateActionResponse.AsObject> {
const req = new ReactivateActionRequest();
req.setId(id);
return this.grpcService.mgmt.reactivateAction(req, null).then((resp) => resp.toObject());
}
public listActions( public listActions(
limit?: number, limit?: number,
offset?: number, offset?: number,

View File

@ -564,6 +564,7 @@
"FLOWTYPE": "Flow Typ", "FLOWTYPE": "Flow Typ",
"TRIGGERTYPE": "Trigger Typ", "TRIGGERTYPE": "Trigger Typ",
"ACTIONS": "Aktionen", "ACTIONS": "Aktionen",
"ACTIONSMAX": "Basierend auf Ihrem Tier steht Ihnen eine begrenzte Anzahl von Aktionen ({{value}}) zur Verfügung. Stellen Sie sicher, dass Sie diejenigen deaktivieren, die Sie nicht benötigen, oder erwägen Sie ein Upgrade.",
"DIALOG": { "DIALOG": {
"ADD": { "ADD": {
"TITLE": "Aktion erstellen" "TITLE": "Aktion erstellen"
@ -581,7 +582,9 @@
} }
}, },
"TOAST": { "TOAST": {
"ACTIONSSET": "Aktionen gesetzt" "ACTIONSSET": "Aktionen gesetzt",
"ACTIONREACTIVATED": "Aktionen erfolgreich reaktiviert",
"ACTIONDEACTIVATED": "Aktionen erfolgreich deaktiviert"
} }
}, },
"IAM": { "IAM": {
@ -756,7 +759,16 @@
"CUSTOMTEXTMESSAGE": "Benutzerdefinierte Benachrichtigungstexte", "CUSTOMTEXTMESSAGE": "Benutzerdefinierte Benachrichtigungstexte",
"PRIVACYPOLICY": "Benutzerdefinierte Datenschutzrichtlinie und AGB", "PRIVACYPOLICY": "Benutzerdefinierte Datenschutzrichtlinie und AGB",
"METADATAUSER": "User Metadata", "METADATAUSER": "User Metadata",
"FLOWS": "Aktionen und Abläufe" "FLOWS": "Aktionen und Abläufe",
"FLOW": {
"ACTIONSALLOWED": {
"0": "Nicht erlaubt",
"1": "Limitierte Aktionen",
"2": "Unlimitiert"
},
"TYPE": "Verwendungsart",
"COUNT": "Anzahl"
}
}, },
"TIERSTATES": { "TIERSTATES": {
"0": "Aktiv", "0": "Aktiv",

View File

@ -564,6 +564,7 @@
"FLOWTYPE": "Flow Type", "FLOWTYPE": "Flow Type",
"TRIGGERTYPE": "Trigger Type", "TRIGGERTYPE": "Trigger Type",
"ACTIONS": "Actions", "ACTIONS": "Actions",
"ACTIONSMAX": "Based on your Tier, you have available a limited Number of Actions ({{value}}). Make sure to deaktivate those you are not in need or consider upgrading your tier.",
"DIALOG": { "DIALOG": {
"ADD": { "ADD": {
"TITLE": "Create an Action" "TITLE": "Create an Action"
@ -581,7 +582,9 @@
} }
}, },
"TOAST": { "TOAST": {
"ACTIONSSET": "Actions set" "ACTIONSSET": "Actions set",
"ACTIONREACTIVATED": "Actions reactivated with success",
"ACTIONDEACTIVATED": "Actions deactivated with success"
} }
}, },
"IAM": { "IAM": {
@ -756,7 +759,16 @@
"CUSTOMTEXTMESSAGE": "Custom notification mail texts", "CUSTOMTEXTMESSAGE": "Custom notification mail texts",
"PRIVACYPOLICY": "Custom Privacy Policy and TOS Links", "PRIVACYPOLICY": "Custom Privacy Policy and TOS Links",
"METADATAUSER": "User Metadata", "METADATAUSER": "User Metadata",
"FLOWS": "Actions and Flows" "FLOWS": "Actions and Flows",
"FLOW": {
"ACTIONSALLOWED": {
"0": "Not allowed",
"1": "Limited Actions",
"2": "Unlimited Actions"
},
"TYPE": "Type of use",
"COUNT": "Number"
}
}, },
"TIERSTATES": { "TIERSTATES": {
"0": "Active", "0": "Active",

View File

@ -564,6 +564,7 @@
"FLOWTYPE": "Tipo processo", "FLOWTYPE": "Tipo processo",
"TRIGGERTYPE": "Tipo trigger", "TRIGGERTYPE": "Tipo trigger",
"ACTIONS": "Azioni", "ACTIONS": "Azioni",
"ACTIONSMAX": "In base al tuo tier, hai a disposizione un numero limitato di azioni ({{value}}). Assicurati di disattivare quelli di cui non hai bisogno o considera di fare un upgrade.",
"DIALOG": { "DIALOG": {
"ADD": { "ADD": {
"TITLE": "Crea azione" "TITLE": "Crea azione"
@ -581,7 +582,9 @@
} }
}, },
"TOAST": { "TOAST": {
"ACTIONSSET": "Azioni salvate!" "ACTIONSSET": "Azioni salvate!",
"ACTIONREACTIVATED": "Azioni riattivati con successo",
"ACTIONDEACTIVATED": "Azioni disattivati con successo"
} }
}, },
"IAM": { "IAM": {
@ -756,7 +759,16 @@
"CUSTOMTEXTMESSAGE": "Testi email personalizzati", "CUSTOMTEXTMESSAGE": "Testi email personalizzati",
"PRIVACYPOLICY": "Link personalizzati all'informativa sulla privacy e ai TOS", "PRIVACYPOLICY": "Link personalizzati all'informativa sulla privacy e ai TOS",
"METADATAUSER": "Metadati utente", "METADATAUSER": "Metadati utente",
"FLOWS": "Azioni e processi" "FLOWS": "Azioni e processi",
"FLOW": {
"ACTIONSALLOWED": {
"0": "Non abilitato",
"1": "Numero limitato",
"2": "Illimitato"
},
"COUNT": "Anzahl",
"TYPE": "Tipo di utilizzo"
}
}, },
"TIERSTATES": { "TIERSTATES": {
"0": "Attivo", "0": "Attivo",