feat(console): stripe (#1590)

* prepare request for sub service

* show detail, i18n

* form dialog

* country

* stripe button

* feature

* flex

* rm logs

* org creationdate, lint

* rm log
This commit is contained in:
Max Peintner 2021-04-15 18:30:50 +02:00 committed by GitHub
parent c2fedbbfc6
commit 6a3a541848
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
272 changed files with 2270 additions and 193 deletions

View File

@ -25,6 +25,7 @@ import { AuthConfig, OAuthModule, OAuthStorage } from 'angular-oauth2-oidc';
import { QuicklinkModule } from 'ngx-quicklink'; import { QuicklinkModule } from 'ngx-quicklink';
import { OnboardingModule } from 'src/app/modules/onboarding/onboarding.module'; import { OnboardingModule } from 'src/app/modules/onboarding/onboarding.module';
import { RegExpPipeModule } from 'src/app/pipes/regexp-pipe/regexp-pipe.module'; import { RegExpPipeModule } from 'src/app/pipes/regexp-pipe/regexp-pipe.module';
import { SubscriptionService } from 'src/app/services/subscription.service';
import { environment } from '../environments/environment'; import { environment } from '../environments/environment';
import { AppRoutingModule } from './app-routing.module'; import { AppRoutingModule } from './app-routing.module';
@ -178,6 +179,7 @@ const authConfig: AuthConfig = {
GrpcService, GrpcService,
AuthenticationService, AuthenticationService,
GrpcAuthService, GrpcAuthService,
SubscriptionService,
{ provide: 'windowObject', useValue: window }, { provide: 'windowObject', useValue: window },
], ],
bootstrap: [AppComponent], bootstrap: [AppComponent],

File diff suppressed because it is too large Load Diff

View File

@ -1,6 +1,39 @@
<app-detail-layout [backRouterLink]="[ serviceType === FeatureServiceType.ADMIN ? '/iam/policies' : '/org']" <app-detail-layout [backRouterLink]="[ serviceType === FeatureServiceType.ADMIN ? '/iam/policies' : '/org']"
[title]="'ZITADEL ' +features?.tier.name + ' ' + ('FEATURES.TITLE' | translate)" [description]="'FEATURES.DESCRIPTION' | translate"> [title]="('FEATURES.TITLE' | translate)" [description]="'FEATURES.DESCRIPTION' | translate">
<p class="default" *ngIf="isDefault"> {{'POLICY.DEFAULTLABEL' | translate}}</p>
<h2>{{'FEATURES.TIER.TITLE' | translate}}</h2>
<p *ngIf="serviceType === FeatureServiceType.MGMT" class="tier-desc">{{'FEATURES.TIER.DESCRIPTION' | translate}}
{{'FEATURES.TIER.QUESTIONS' | translate}} <a href="mailto:support@zitadel.ch">support@zitadel.ch</a>.</p>
<div class="detail">
<p class="title">{{'FEATURES.TIER.NAME' | translate}}</p>
<p>{{features?.tier?.name}}</p>
</div>
<ng-container *ngIf="serviceType === FeatureServiceType.MGMT">
<mat-spinner class="spinner" diameter="20" *ngIf="customerLoading || stripeLoading"></mat-spinner>
<div class="detail" *ngIf="stripeCustomer">
<p class="title">{{'FEATURES.TIER.DETAILS' | translate}}
<a (click)="updateCustomer()">{{'ACTIONS.EDIT' | translate}}</a>
</p>
<p>{{stripeCustomer?.contact}}</p>
<p *ngIf="stripeCustomer.company">{{stripeCustomer?.company}}</p>
<p>{{stripeCustomer?.address}}</p>
<p *ngIf="stripeCustomer?.postal_code || stripeCustomer?.city || stripeCustomer?.country">
{{stripeCustomer?.postal_code}} {{stripeCustomer?.city}} {{stripeCustomer?.country}}
<img *ngIf="customerCountry" height="20px" width="30px"
style="margin-right: 1rem; border-radius: 2px; vertical-align: middle;"
src="../../../assets/flags/{{customerCountry.isoCode.toLowerCase()}}.png" />
</p>
</div>
<p class="error" *ngIf="stripeCustomer && !customerValid">{{'FEATURES.TIER.CUSTOMERINVALID' | translate}}</p>
<div class="current-tier">
<a [disabled]="!org.id || !customerValid" mat-raised-button [href]="stripeURL" target="_blank"
alt="change tier">{{'FEATURES.TIER.BTN' | translate}}</a>
</div>
</ng-container>
<ng-template appHasRole [appHasRole]="['iam.features.delete']"> <ng-template appHasRole [appHasRole]="['iam.features.delete']">
<button *ngIf="serviceType === FeatureServiceType.MGMT && !isDefault" <button *ngIf="serviceType === FeatureServiceType.MGMT && !isDefault"
@ -9,12 +42,15 @@
</button> </button>
</ng-template> </ng-template>
<div class="content" *ngIf="features"> <div class="divider"></div>
<p class="default" *ngIf="isDefault"> {{'POLICY.DEFAULTLABEL' | translate}}</p>
<div class="content" *ngIf="features">
<div class="row"> <div class="row">
<span class="left-desc">{{'FEATURES.DATA.AUDITLOGRETENTION' | translate}}</span> <span class="left-desc">{{'FEATURES.DATA.AUDITLOGRETENTION' | translate}}</span>
<span class="fill-space"></span> <span class="fill-space"></span>
<span>{{features.auditLogRetention | timestampToRetention }} {{'FEATURES.RETENTIONHOURS' | translate}}</span> <span>{{features.auditLogRetention | timestampToRetention }} {{'FEATURES.RETENTIONHOURS' |
translate}}</span>
</div> </div>
<div class="row"> <div class="row">
<span class="left-desc">{{'FEATURES.DATA.LOGINPOLICYUSERNAMELOGIN' | translate}}</span> <span class="left-desc">{{'FEATURES.DATA.LOGINPOLICYUSERNAMELOGIN' | translate}}</span>

View File

@ -1,8 +1,66 @@
.default { .default {
color: var(--color-main); color: var(--color-main);
font-size: 14px;
}
.tier-desc {
color: var(--grey);
font-size: 14px;
margin-top: 0; margin-top: 0;
} }
.detail {
margin-bottom: 1rem;
p {
margin: 0;
font-size: 14px;
display: flex;
align-items: center;
}
.title {
font-size: 14px;
color: var(--grey);
margin-bottom: .5rem;
a {
margin-left: .5rem;
cursor: pointer;
}
}
img {
height: 15px;
width: auto;
margin-left: .5rem;
}
}
.spinner {
margin: .5rem;
}
.error {
color: #f44336;
font-size: 14px;
}
.current-tier {
display: flex;
align-items: center;
margin-bottom: 1rem;
}
.divider {
height: 1px;
width: 100%;
background-color: var(--grey);
opacity: .5;
margin: .5rem 0;
display: block;
}
.content { .content {
padding-top: 1rem; padding-top: 1rem;
display: flex; display: flex;

View File

@ -1,11 +1,12 @@
import { Component, Injector, OnDestroy, Type } from '@angular/core'; import { Component, Injector, OnDestroy, Type } from '@angular/core';
import { MatDialog } from '@angular/material/dialog';
import { ActivatedRoute } from '@angular/router'; import { ActivatedRoute } from '@angular/router';
import { Subscription } from 'rxjs'; import { Subscription } from 'rxjs';
import { switchMap } from 'rxjs/operators'; import { switchMap } from 'rxjs/operators';
import { import {
GetOrgFeaturesResponse, GetOrgFeaturesResponse,
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 { 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';
@ -13,132 +14,206 @@ 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';
import { ManagementService } from 'src/app/services/mgmt.service'; import { ManagementService } from 'src/app/services/mgmt.service';
import { StorageService } from 'src/app/services/storage.service'; import { StorageService } from 'src/app/services/storage.service';
import { StripeCustomer, SubscriptionService } from 'src/app/services/subscription.service';
import { ToastService } from 'src/app/services/toast.service'; import { ToastService } from 'src/app/services/toast.service';
import { COUNTRIES, Country } from './country';
import { PaymentInfoDialogComponent } from './payment-info-dialog/payment-info-dialog.component';
export enum FeatureServiceType { export enum FeatureServiceType {
MGMT, MGMT,
ADMIN, ADMIN,
} }
@Component({ @Component({
selector: 'app-features', selector: 'app-features',
templateUrl: './features.component.html', templateUrl: './features.component.html',
styleUrls: ['./features.component.scss'], styleUrls: ['./features.component.scss'],
}) })
export class FeaturesComponent implements OnDestroy { export class FeaturesComponent implements OnDestroy {
private managementService!: ManagementService; private managementService!: ManagementService;
public serviceType!: FeatureServiceType; public serviceType!: FeatureServiceType;
public features!: Features.AsObject; public features!: Features.AsObject;
private sub: Subscription = new Subscription(); private sub: Subscription = new Subscription();
private org!: Org.AsObject; public org!: Org.AsObject;
public FeatureServiceType: any = FeatureServiceType; public FeatureServiceType: any = FeatureServiceType;
constructor(
private route: ActivatedRoute, public customerLoading: boolean = false;
private toast: ToastService, public stripeLoading: boolean = false;
private sessionStorage: StorageService, public stripeURL: string = '';
private injector: Injector, public stripeCustomer!: StripeCustomer;
private adminService: AdminService,
) { constructor(
const temporg = this.sessionStorage.getItem('organization') as Org.AsObject; private route: ActivatedRoute,
if (temporg) { private toast: ToastService,
this.org = temporg; private sessionStorage: StorageService,
} private injector: Injector,
this.sub = this.route.data.pipe(switchMap(data => { private adminService: AdminService,
this.serviceType = data.serviceType; private subService: SubscriptionService,
if (this.serviceType === FeatureServiceType.MGMT) { private dialog: MatDialog,
this.managementService = this.injector.get(ManagementService as Type<ManagementService>); ) {
} const temporg = this.sessionStorage.getItem('organization') as Org.AsObject;
return this.route.params; if (temporg) {
})).subscribe(_ => { this.org = temporg;
this.fetchData(); }
this.sub = this.route.data.pipe(switchMap(data => {
this.serviceType = data.serviceType;
if (this.serviceType === FeatureServiceType.MGMT) {
this.managementService = this.injector.get(ManagementService as Type<ManagementService>);
}
return this.route.params;
})).subscribe(_ => {
this.fetchData();
});
if (this.serviceType === FeatureServiceType.MGMT) {
this.customerLoading = true;
this.subService.getCustomer(this.org.id)
.then(payload => {
this.customerLoading = false;
this.stripeCustomer = payload;
if (this.customerValid) {
this.getLinkToStripe();
}
})
.catch(error => {
this.customerLoading = false;
console.error(error);
}); });
} }
}
public ngOnDestroy(): void { public ngOnDestroy(): void {
this.sub.unsubscribe(); this.sub.unsubscribe();
} }
public fetchData(): void { public updateCustomer(): void {
this.getData().then(resp => { const dialogRefPhone = this.dialog.open(PaymentInfoDialogComponent, {
if (resp?.features) { data: {
this.features = resp.features; customer: this.stripeCustomer,
} },
width: '400px',
});
dialogRefPhone.afterClosed().subscribe(customer => {
if (customer) {
console.log(customer);
this.stripeCustomer = customer;
this.subService.updateCustomer(this.org.id, customer).then(() => {
this.getLinkToStripe();
}).catch(console.error);
}
});
}
public getLinkToStripe(): void {
if (this.serviceType === FeatureServiceType.MGMT) {
this.stripeLoading = true;
this.subService.getLink(this.org.id, window.location.href)
.then(payload => {
this.stripeLoading = false;
console.log(payload);
this.stripeURL = payload.redirect_url;
})
.catch(error => {
this.stripeLoading = false;
console.error(error);
}); });
} }
}
private async getData(): Promise<GetFeaturesResponse.AsObject | GetOrgFeaturesResponse.AsObject | undefined> { public fetchData(): void {
switch (this.serviceType) { this.getData().then(resp => {
case FeatureServiceType.MGMT: if (resp?.features) {
return this.managementService.getFeatures(); this.features = resp.features;
case FeatureServiceType.ADMIN: }
if (this.org?.id) { });
return this.adminService.getDefaultFeatures(); }
}
break; private async getData(): Promise<GetFeaturesResponse.AsObject | GetOrgFeaturesResponse.AsObject | undefined> {
switch (this.serviceType) {
case FeatureServiceType.MGMT:
return this.managementService.getFeatures();
case FeatureServiceType.ADMIN:
if (this.org?.id) {
return this.adminService.getDefaultFeatures();
} }
break;
} }
}
public savePolicy(): void { public savePolicy(): void {
switch (this.serviceType) { switch (this.serviceType) {
case FeatureServiceType.MGMT: case FeatureServiceType.MGMT:
const req = new SetOrgFeaturesRequest(); const req = new SetOrgFeaturesRequest();
req.setOrgId(this.org.id); req.setOrgId(this.org.id);
req.setLoginPolicyUsernameLogin(this.features.loginPolicyUsernameLogin); req.setLoginPolicyUsernameLogin(this.features.loginPolicyUsernameLogin);
req.setLoginPolicyRegistration(this.features.loginPolicyRegistration); req.setLoginPolicyRegistration(this.features.loginPolicyRegistration);
req.setLoginPolicyIdp(this.features.loginPolicyIdp); req.setLoginPolicyIdp(this.features.loginPolicyIdp);
req.setLoginPolicyFactors(this.features.loginPolicyFactors); req.setLoginPolicyFactors(this.features.loginPolicyFactors);
req.setLoginPolicyPasswordless(this.features.loginPolicyPasswordless); req.setLoginPolicyPasswordless(this.features.loginPolicyPasswordless);
req.setPasswordComplexityPolicy(this.features.passwordComplexityPolicy); req.setPasswordComplexityPolicy(this.features.passwordComplexityPolicy);
req.setLabelPolicy(this.features.labelPolicy); req.setLabelPolicy(this.features.labelPolicy);
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;
case FeatureServiceType.ADMIN: case FeatureServiceType.ADMIN:
// update Default org iam policy? // update Default org iam policy?
const dreq = new SetDefaultFeaturesRequest(); const dreq = new SetDefaultFeaturesRequest();
dreq.setLoginPolicyUsernameLogin(this.features.loginPolicyUsernameLogin); dreq.setLoginPolicyUsernameLogin(this.features.loginPolicyUsernameLogin);
dreq.setLoginPolicyRegistration(this.features.loginPolicyRegistration); dreq.setLoginPolicyRegistration(this.features.loginPolicyRegistration);
dreq.setLoginPolicyIdp(this.features.loginPolicyIdp); dreq.setLoginPolicyIdp(this.features.loginPolicyIdp);
dreq.setLoginPolicyFactors(this.features.loginPolicyFactors); dreq.setLoginPolicyFactors(this.features.loginPolicyFactors);
dreq.setLoginPolicyPasswordless(this.features.loginPolicyPasswordless); dreq.setLoginPolicyPasswordless(this.features.loginPolicyPasswordless);
dreq.setPasswordComplexityPolicy(this.features.passwordComplexityPolicy); dreq.setPasswordComplexityPolicy(this.features.passwordComplexityPolicy);
dreq.setLabelPolicy(this.features.labelPolicy); dreq.setLabelPolicy(this.features.labelPolicy);
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;
}
} }
}
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);
}); });
}
} }
}
public get isDefault(): boolean { public get isDefault(): boolean {
if (this.features && this.serviceType === FeatureServiceType.MGMT) { if (this.features && this.serviceType === FeatureServiceType.MGMT) {
return this.features.isDefault; return this.features.isDefault;
} else { } else {
return false; return false;
}
} }
}
get customerValid(): boolean {
return !!this.stripeCustomer?.contact &&
!!this.stripeCustomer?.address &&
!!this.stripeCustomer?.city &&
!!this.stripeCustomer?.postal_code;
}
get customerCountry(): Country | undefined {
return COUNTRIES.find(country => country.isoCode === this.stripeCustomer.country);
}
} }

View File

@ -1,8 +1,10 @@
import { CommonModule } from '@angular/common'; import { CommonModule } from '@angular/common';
import { NgModule } from '@angular/core'; import { NgModule } from '@angular/core';
import { FormsModule } from '@angular/forms'; import { FormsModule, ReactiveFormsModule } from '@angular/forms';
import { MatButtonModule } from '@angular/material/button'; import { MatButtonModule } from '@angular/material/button';
import { MatIconModule } from '@angular/material/icon'; import { MatIconModule } from '@angular/material/icon';
import { MatProgressSpinnerModule } from '@angular/material/progress-spinner';
import { MatSelectModule } from '@angular/material/select';
import { MatSlideToggleModule } from '@angular/material/slide-toggle'; import { MatSlideToggleModule } from '@angular/material/slide-toggle';
import { MatTooltipModule } from '@angular/material/tooltip'; import { MatTooltipModule } from '@angular/material/tooltip';
import { TranslateModule } from '@ngx-translate/core'; import { TranslateModule } from '@ngx-translate/core';
@ -11,29 +13,37 @@ import { DetailLayoutModule } from 'src/app/modules/detail-layout/detail-layout.
import { InputModule } from 'src/app/modules/input/input.module'; import { InputModule } from 'src/app/modules/input/input.module';
import { HasRolePipeModule } from 'src/app/pipes/has-role-pipe/has-role-pipe.module'; import { HasRolePipeModule } from 'src/app/pipes/has-role-pipe/has-role-pipe.module';
import { import {
TimestampToRetentionPipeModule, TimestampToRetentionPipeModule,
} from 'src/app/pipes/timestamp-to-retention-pipe/timestamp-to-retention-pipe.module'; } from 'src/app/pipes/timestamp-to-retention-pipe/timestamp-to-retention-pipe.module';
import { FormFieldModule } from '../form-field/form-field.module';
import { InfoSectionModule } from '../info-section/info-section.module'; import { InfoSectionModule } from '../info-section/info-section.module';
import { FeaturesRoutingModule } from './features-routing.module'; import { FeaturesRoutingModule } from './features-routing.module';
import { FeaturesComponent } from './features.component'; import { FeaturesComponent } from './features.component';
import { PaymentInfoDialogComponent } from './payment-info-dialog/payment-info-dialog.component';
@NgModule({ @NgModule({
declarations: [ declarations: [
FeaturesComponent, FeaturesComponent,
PaymentInfoDialogComponent,
], ],
imports: [ imports: [
FeaturesRoutingModule, FeaturesRoutingModule,
CommonModule, CommonModule,
FormsModule, FormsModule,
ReactiveFormsModule,
InputModule, InputModule,
MatButtonModule, MatButtonModule,
FormFieldModule,
InputModule,
HasRoleModule, HasRoleModule,
MatSlideToggleModule, MatSlideToggleModule,
MatSelectModule,
MatIconModule, MatIconModule,
HasRoleModule, HasRoleModule,
HasRolePipeModule, HasRolePipeModule,
MatTooltipModule, MatTooltipModule,
MatProgressSpinnerModule,
InfoSectionModule, InfoSectionModule,
TranslateModule, TranslateModule,
DetailLayoutModule, DetailLayoutModule,

View File

@ -0,0 +1,60 @@
<h1 mat-dialog-title>
<span class="title">{{'FEATURES.TIER.DETAILS' | translate}} {{data?.number}}</span>
</h1>
<p class="desc">{{'FEATURES.TIER.DETAILS_DESC' | translate}}</p>
<mat-spinner class="spinner" *ngIf="stripeLoading" diameter="20"></mat-spinner>
<div mat-dialog-content>
<form [formGroup]="form">
<cnsl-form-field class="formfield">
<cnsl-label>{{ 'FEATURES.TIER.CONTACT' | translate }}</cnsl-label>
<input cnslInput name="contact" formControlName="contact" />
</cnsl-form-field>
<cnsl-form-field class="formfield">
<cnsl-label>{{ 'FEATURES.TIER.COMPANY' | translate }}</cnsl-label>
<input cnslInput name="company" formControlName="company" />
</cnsl-form-field>
<cnsl-form-field class="formfield">
<cnsl-label>{{ 'FEATURES.TIER.ADDRESS' | translate }}</cnsl-label>
<input cnslInput name="address" formControlName="address" />
</cnsl-form-field>
<cnsl-form-field class="formfield">
<cnsl-label>{{ 'FEATURES.TIER.CITY' | translate }}</cnsl-label>
<input cnslInput name="city" formControlName="city" />
</cnsl-form-field>
<cnsl-form-field class="formfield">
<cnsl-label>{{ 'FEATURES.TIER.POSTAL_CODE' | translate }}</cnsl-label>
<input cnslInput name="postal_code" formControlName="postal_code" />
</cnsl-form-field>
<cnsl-form-field class="formfield">
<cnsl-label>{{ 'FEATURES.TIER.COUNTRY' | translate }}</cnsl-label>
<mat-select formControlName="country" name="country" (selectionChange)="changeCountry($event)">
<mat-option *ngFor="let country of COUNTRIES" [value]="country.isoCode" class="center-row">
<div class="center">
<div class="img-wrapper">
<img height="20px" width="30px"
style="margin-right: 1rem; border-radius: 2px; vertical-align: middle;"
src="../../../../assets/flags/{{country.isoCode.toLowerCase()}}.png" />
</div>
{{country.isoCode}} <span class="lighter">|&nbsp;{{country.name}}</span>
</div>
</mat-option>
</mat-select>
</cnsl-form-field>
</form>
</div>
<div mat-dialog-actions class="action">
<button cdkFocusInitial color="primary" mat-button class="ok-button" (click)="dialogRef.close()">
{{'ACTIONS.CLOSE' | translate}}
</button>
<button [disabled]="form.invalid" cdkFocusInitial color="primary" mat-raised-button class="ok-button"
(click)="submitAndCloseDialog()">
{{'ACTIONS.OK' | translate}}
</button>
</div>

View File

@ -0,0 +1,47 @@
h1 {
font-size: 1.5rem;
margin-top: 0;
}
.desc {
font-size: 14px;
color: var(--grey);
}
.formfield {
width: 100%;
}
.action {
display: flex;
justify-content: flex-end;
.ok-button {
margin-left: .5rem;
}
button {
border-radius: .5rem;
}
}
img {
height: 15px;
width: auto;
}
.img-wrapper {
width: 50px;
display: flex;
}
.lighter {
font-size: 12px;
color: var(--grey);
padding: 0 .5rem;
}
.center {
display: flex;
align-items: center;
}

View File

@ -0,0 +1,25 @@
import { ComponentFixture, TestBed, waitForAsync } from '@angular/core/testing';
import { PaymentInfoDialogComponent } from './payment-info-dialog.component';
describe('PaymentInfoDialogComponent', () => {
let component: PaymentInfoDialogComponent;
let fixture: ComponentFixture<PaymentInfoDialogComponent>;
beforeEach(waitForAsync(() => {
TestBed.configureTestingModule({
declarations: [PaymentInfoDialogComponent],
})
.compileComponents();
}));
beforeEach(() => {
fixture = TestBed.createComponent(PaymentInfoDialogComponent);
component = fixture.componentInstance;
fixture.detectChanges();
});
it('should create', () => {
expect(component).toBeTruthy();
});
});

View File

@ -0,0 +1,91 @@
import { Component, Inject } from '@angular/core';
import { AbstractControl, FormBuilder, FormGroup, Validators } from '@angular/forms';
import { MAT_DIALOG_DATA, MatDialogRef } from '@angular/material/dialog';
import { MatSelectChange } from '@angular/material/select';
import { SubscriptionService } from 'src/app/services/subscription.service';
import { COUNTRIES, Country } from '../country';
function compare(a: Country, b: Country): number {
if (a.isoCode < b.isoCode) {
return -1;
}
if (a.isoCode > b.isoCode) {
return 1;
}
return 0;
}
@Component({
selector: 'app-payment-info-dialog',
templateUrl: './payment-info-dialog.component.html',
styleUrls: ['./payment-info-dialog.component.scss'],
})
export class PaymentInfoDialogComponent {
public stripeLoading: boolean = false;
public COUNTRIES: Country[] = COUNTRIES.sort(compare);
public form!: FormGroup;
private orgId: string = '';
constructor(
private subService: SubscriptionService,
private fb: FormBuilder,
public dialogRef: MatDialogRef<PaymentInfoDialogComponent>,
@Inject(MAT_DIALOG_DATA) public data: any,
) {
this.orgId = data.orgId;
this.form = this.fb.group({
contact: ['', [Validators.required]],
company: ['', []],
address: ['', [Validators.required]],
city: ['', [Validators.required]],
postal_code: ['', [Validators.required]],
country: ['', [Validators.required]],
});
if (data.customer) {
this.form.patchValue(data.customer);
}
if (!data.customer?.country) {
this.form.get('country')?.setValue('CH');
}
this.getLink();
}
public getLink(): void {
if (this.orgId) {
this.stripeLoading = true;
this.subService.getLink(this.orgId, window.location.href)
.then(payload => {
this.stripeLoading = false;
console.log(payload);
if (payload.redirect_url) {
window.open(payload.redirect_url, '_blank');
}
})
.catch(error => {
this.stripeLoading = false;
console.error(error);
});
}
}
public changeCountry(selection: MatSelectChange): void {
const country = COUNTRIES.find(c => c.isoCode === selection.value);
if (country && country.phoneCode !== undefined && this.phone && this.phone.value !== `+${country.phoneCode}`) {
this.phone.setValue(`+${country.phoneCode}`);
}
}
submitAndCloseDialog(): void {
this.dialogRef.close(this.form.value);
}
get phone(): AbstractControl | null {
return this.form.get('phone');
}
}

View File

@ -39,6 +39,16 @@
<td mat-cell *matCellDef="let org"> {{org.name}} </td> <td mat-cell *matCellDef="let org"> {{org.name}} </td>
</ng-container> </ng-container>
<ng-container matColumnDef="creationDate">
<th mat-header-cell *matHeaderCellDef mat-sort-header
[ngClass]="{'search-active': this.orgSearchKey == OrgListSearchKey.NAME}">
{{ 'ORG.PAGES.CREATIONDATE' | translate }}
</th>
<td mat-cell *matCellDef="let org">
{{org.details?.creationDate | timestampToDate | localizedDate: 'EEE dd. MMM YYYY, HH:mm'}}
</td>
</ng-container>
<tr mat-header-row *matHeaderRowDef="displayedColumns"></tr> <tr mat-header-row *matHeaderRowDef="displayedColumns"></tr>
<tr (click)="setAndNavigateToOrg(row)" mat-row *matRowDef="let row; columns: displayedColumns;"></tr> <tr (click)="setAndNavigateToOrg(row)" mat-row *matRowDef="let row; columns: displayedColumns;"></tr>
</table> </table>

View File

@ -11,102 +11,102 @@ import { Org, OrgNameQuery, OrgQuery } from 'src/app/proto/generated/zitadel/org
import { GrpcAuthService } from 'src/app/services/grpc-auth.service'; import { GrpcAuthService } from 'src/app/services/grpc-auth.service';
enum OrgListSearchKey { enum OrgListSearchKey {
NAME = 'NAME', NAME = 'NAME',
} }
@Component({ @Component({
selector: 'app-org-list', selector: 'app-org-list',
templateUrl: './org-list.component.html', templateUrl: './org-list.component.html',
styleUrls: ['./org-list.component.scss'], styleUrls: ['./org-list.component.scss'],
animations: [ animations: [
enterAnimations, enterAnimations,
], ],
}) })
export class OrgListComponent implements AfterViewInit { export class OrgListComponent implements AfterViewInit {
public orgSearchKey: OrgListSearchKey | undefined = undefined; public orgSearchKey: OrgListSearchKey | undefined = undefined;
@ViewChild(MatPaginator) public paginator!: MatPaginator; @ViewChild(MatPaginator) public paginator!: MatPaginator;
@ViewChild(MatSort) sort!: MatSort; @ViewChild(MatSort) sort!: MatSort;
@ViewChild('input') public filter!: Input; @ViewChild('input') public filter!: Input;
public dataSource!: MatTableDataSource<Org.AsObject>; public dataSource!: MatTableDataSource<Org.AsObject>;
public displayedColumns: string[] = ['select', 'id', 'name']; public displayedColumns: string[] = ['select', 'id', 'name', 'creationDate'];
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();
public activeOrg!: Org.AsObject; public activeOrg!: Org.AsObject;
public OrgListSearchKey: any = OrgListSearchKey; public OrgListSearchKey: any = OrgListSearchKey;
constructor( constructor(
private authService: GrpcAuthService, private authService: GrpcAuthService,
private router: Router, private router: Router,
) { ) {
this.loadOrgs(10, 0); this.loadOrgs(10, 0);
this.authService.getActiveOrg().then(org => this.activeOrg = org); this.authService.getActiveOrg().then(org => this.activeOrg = org);
}
public ngAfterViewInit(): void {
this.loadOrgs(10, 0);
}
public loadOrgs(limit: number, offset: number, filter?: string): void {
this.loadingSubject.next(true);
let query;
if (filter) {
query = new OrgQuery();
const orgNameQuery = new OrgNameQuery();
orgNameQuery.setMethod(TextQueryMethod.TEXT_QUERY_METHOD_CONTAINS_IGNORE_CASE);
orgNameQuery.setName(filter);
query.setNameQuery(orgNameQuery);
} }
public ngAfterViewInit(): void { from(this.authService.listMyProjectOrgs(limit, offset, query ? [query] : undefined)).pipe(
this.loadOrgs(10, 0); map(resp => {
return resp.resultList;
}),
catchError(() => of([])),
finalize(() => this.loadingSubject.next(false)),
).subscribe(views => {
this.dataSource = new MatTableDataSource(views);
this.dataSource.paginator = this.paginator;
this.dataSource.sort = this.sort;
});
}
public selectOrg(item: Org.AsObject, event?: any): void {
this.authService.setActiveOrg(item);
}
public refresh(): void {
this.loadOrgs(this.paginator.length, this.paginator.pageSize * this.paginator.pageIndex);
}
public setFilter(key: OrgListSearchKey): void {
setTimeout(() => {
if (this.filter) {
(this.filter as any).nativeElement.focus();
}
}, 100);
if (this.orgSearchKey !== key) {
this.orgSearchKey = key;
} else {
this.orgSearchKey = undefined;
this.refresh();
} }
}
public loadOrgs(limit: number, offset: number, filter?: string): void { public applyFilter(event: Event): void {
this.loadingSubject.next(true); const filterValue = (event.target as HTMLInputElement).value;
let query; this.loadOrgs(
if (filter) { this.paginator.pageSize,
query = new OrgQuery(); this.paginator.pageIndex * this.paginator.pageSize,
const orgNameQuery = new OrgNameQuery(); filterValue.trim().toLowerCase(),
orgNameQuery.setMethod(TextQueryMethod.TEXT_QUERY_METHOD_CONTAINS_IGNORE_CASE); );
orgNameQuery.setName(filter); }
query.setNameQuery(orgNameQuery);
}
from(this.authService.listMyProjectOrgs(limit, offset, query ? [query] : undefined)).pipe( public setAndNavigateToOrg(org: Org.AsObject): void {
map(resp => { this.authService.setActiveOrg(org);
return resp.resultList; this.router.navigate(['/org']);
}), }
catchError(() => of([])),
finalize(() => this.loadingSubject.next(false)),
).subscribe(views => {
this.dataSource = new MatTableDataSource(views);
this.dataSource.paginator = this.paginator;
this.dataSource.sort = this.sort;
});
}
public selectOrg(item: Org.AsObject, event?: any): void {
this.authService.setActiveOrg(item);
}
public refresh(): void {
this.loadOrgs(this.paginator.length, this.paginator.pageSize * this.paginator.pageIndex);
}
public setFilter(key: OrgListSearchKey): void {
setTimeout(() => {
if (this.filter) {
(this.filter as any).nativeElement.focus();
}
}, 100);
if (this.orgSearchKey !== key) {
this.orgSearchKey = key;
} else {
this.orgSearchKey = undefined;
this.refresh();
}
}
public applyFilter(event: Event): void {
const filterValue = (event.target as HTMLInputElement).value;
this.loadOrgs(
this.paginator.pageSize,
this.paginator.pageIndex * this.paginator.pageSize,
filterValue.trim().toLowerCase(),
);
}
public setAndNavigateToOrg(org: Org.AsObject): void {
this.authService.setActiveOrg(org);
this.router.navigate(['/org']);
}
} }

View File

@ -0,0 +1,89 @@
import { HttpClient } from '@angular/common/http';
import { Injectable } from '@angular/core';
import { StorageService } from './storage.service';
const authorizationKey = 'Authorization';
const bearerPrefix = 'Bearer';
const accessTokenStorageKey = 'access_token';
export interface StripeCustomer {
contact: string;
company?: string;
address: string;
city: string;
postal_code: string;
country: string;
}
@Injectable({
providedIn: 'root',
})
export class SubscriptionService {
constructor(private http: HttpClient, private storageService: StorageService) { }
public getLink(orgId: string, redirectURI: string): Promise<any> {
return this.http.get('./assets/environment.json')
.toPromise().then((data: any) => {
if (data && data.subscriptionServiceUrl) {
const serviceUrl = data.subscriptionServiceUrl;
const accessToken = this.storageService.getItem(accessTokenStorageKey);
return this.http.get(`${serviceUrl}/redirect`, {
headers: {
// 'Content-Type': 'application/json; charset=utf-8',
[authorizationKey]: `${bearerPrefix} ${accessToken}`,
},
params: {
'org': orgId,
'return_url': encodeURI(redirectURI),
'country': 'ch',
},
}).toPromise();
} else {
return Promise.reject('Could not load environment');
}
});
}
public getCustomer(orgId: string): Promise<any> {
return this.http.get('./assets/environment.json')
.toPromise().then((data: any) => {
if (data && data.subscriptionServiceUrl) {
const serviceUrl = data.subscriptionServiceUrl;
const accessToken = this.storageService.getItem(accessTokenStorageKey);
return this.http.get(`${serviceUrl}/customer`, {
headers: {
// 'Content-Type': 'application/json; charset=utf-8',
[authorizationKey]: `${bearerPrefix} ${accessToken}`,
},
params: {
'org': orgId,
},
}).toPromise();
} else {
return Promise.reject('Could not load environment');
}
});
}
public updateCustomer(orgId: string, body: StripeCustomer): Promise<any> {
return this.http.get('./assets/environment.json')
.toPromise().then((data: any) => {
if (data && data.subscriptionServiceUrl) {
const serviceUrl = data.subscriptionServiceUrl;
const accessToken = this.storageService.getItem(accessTokenStorageKey);
return this.http.put(`${serviceUrl}/customer`, body, {
headers: {
// 'Content-Type': 'application/json; charset=utf-8',
[authorizationKey]: `${bearerPrefix} ${accessToken}`,
},
params: {
'org': orgId,
},
}).toPromise();
} else {
return Promise.reject('Could not load environment');
}
});
}
}

View File

@ -2,6 +2,7 @@
"authServiceUrl": "https://api.zitadel.io", "authServiceUrl": "https://api.zitadel.io",
"mgmtServiceUrl": "https://api.zitadel.io", "mgmtServiceUrl": "https://api.zitadel.io",
"adminServiceUrl":"https://api.zitadel.io", "adminServiceUrl":"https://api.zitadel.io",
"subscriptionServiceUrl":"https://sub.zitadel.io",
"issuer": "https://issuer.zitadel.io", "issuer": "https://issuer.zitadel.io",
"clientid": "69234247558357051@zitadel" "clientid": "69234247558357051@zitadel"
} }

Binary file not shown.

After

Width:  |  Height:  |  Size: 652 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 170 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 620 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 735 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 646 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 688 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 122 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 309 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 549 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 1.9 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 327 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 759 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 133 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 630 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 281 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 279 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 243 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 461 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 381 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 258 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 172 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 283 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 106 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 188 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 691 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 169 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 2.2 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 918 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 955 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 547 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 159 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 750 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 289 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 1.3 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 260 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 172 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 452 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 2.2 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 406 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 593 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 449 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 327 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 296 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 172 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 165 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 722 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 285 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 245 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 315 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 158 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 109 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 356 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 407 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 296 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 720 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 572 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 341 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 106 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 531 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 203 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 515 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 422 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 405 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 1.0 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 158 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 366 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 374 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 584 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 682 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 596 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 546 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 186 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 876 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 1.0 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 269 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 260 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 165 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 109 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 99 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 384 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 358 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 1.7 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 383 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 594 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 359 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 424 B

Some files were not shown because too many files have changed in this diff Show More