From 241f2a943da8546b5251ed621653b43a106a2b9b Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Adolfo=20G=C3=B3mez=20Garc=C3=ADa?= Date: Thu, 12 Jan 2023 01:35:23 +0100 Subject: [PATCH] Moving observables to promises when possible --- .../components/service/service.component.ts | 20 +++--- src/app/gui/modal/modal.component.html | 7 +- src/app/gui/modal/modal.component.ts | 69 ++++--------------- src/app/gui/uds-gui.service.ts | 12 ++-- src/app/helpers/plugin.ts | 34 ++++----- src/app/pages/services/services.component.ts | 2 +- src/app/uds-api.service-type.ts | 12 ++-- src/app/uds-api.service.ts | 60 +++++++++------- src/styles.scss | 39 ++++++----- 9 files changed, 110 insertions(+), 145 deletions(-) diff --git a/src/app/gui/components/service/service.component.ts b/src/app/gui/components/service/service.component.ts index 26bdbfa..d8507cb 100644 --- a/src/app/gui/components/service/service.component.ts +++ b/src/app/gui/components/service/service.component.ts @@ -141,16 +141,14 @@ export class ServiceComponent implements OnInit { type === 'release' ? django.gettext('Service released') : django.gettext('Service reseted'); - this.api.gui - .yesno(title, django.gettext('Are you sure?')) - .subscribe((val) => { - if (val) { - this.api.action(type, this.service.id).subscribe((service) => { - if (service) { - this.api.gui.alert(title, action); - } - }); - } - }); + this.api.gui.yesno(title, django.gettext('Are you sure?')).then((val) => { + if (val) { + this.api.action(type, this.service.id).then((service) => { + if (service) { + this.api.gui.alert(title, action); + } + }); + } + }); } } diff --git a/src/app/gui/modal/modal.component.html b/src/app/gui/modal/modal.component.html index 7d921c0..ab49619 100644 --- a/src/app/gui/modal/modal.component.html +++ b/src/app/gui/modal/modal.component.html @@ -1,8 +1,7 @@

- - - - + + + diff --git a/src/app/gui/modal/modal.component.ts b/src/app/gui/modal/modal.component.ts index 4552370..1ec55ad 100644 --- a/src/app/gui/modal/modal.component.ts +++ b/src/app/gui/modal/modal.component.ts @@ -15,7 +15,7 @@ export interface ModalData { title: string; body: string; autoclose?: number; - checkClose?: Observable; + checkClose?: Promise; type: DialogType; username?: string; domain?: string; @@ -30,50 +30,17 @@ export interface ModalData { export class ModalComponent implements OnInit { extra: string; subscription: Subscription; - yesno: Observable; - yes: () => void; - no: () => void; - close: () => void; + yesno: Promise; + resolver: (value: boolean) => void; constructor(public dialogRef: MatDialogRef, @Inject(MAT_DIALOG_DATA) public data: ModalData) { - this.subscription = null; - this.resetCallbacks(); // Notifies on case of yes or not to subscriber - this.yesno = new Observable((observer) => { - this.yes = () => { - observer.next(true); - observer.complete(); - }; - this.no = () => { - observer.next(false); - observer.complete(); - }; - this.close = () => { - this.doClose(); - observer.next(false); - observer.complete(); - }; - const self = this; - return {unsubscribe: () => self.resetCallbacks()}; + this.yesno = new Promise((resolve) => { + this.resolver = resolve; }); } - resetCallbacks() { - this.yes = this.no = () => this.close(); - this.close = () => this.doClose(); - } - - /** - * Invoked on closed modal component - * This ensures that we stop subscription to interval - */ - closed(): void { - if (this.subscription !== null) { - this.subscription.unsubscribe(); - } - } - - doClose(): void { + close() { this.dialogRef.close(); } @@ -86,13 +53,13 @@ export class ModalComponent implements OnInit { this.extra = ' (' + Math.floor(miliseconds / 1000) + ' ' + django.gettext('seconds') + ') '; } - initAlert() { + async initAlert(): Promise { if (this.data.autoclose > 0) { this.dialogRef.afterClosed().subscribe(res => { - this.closed(); + this.close(); }); this.setExtra(this.data.autoclose); - this.subscription = interval(1000).subscribe(n => { + interval(1000).subscribe(n => { const rem = this.data.autoclose - (n + 1) * 1000; this.setExtra(rem); if (rem <= 0) { @@ -103,26 +70,14 @@ export class ModalComponent implements OnInit { this.dialogRef.close(); }, this.data.autoclose);*/ } else if (this.data.checkClose) { - this.dialogRef.afterClosed().subscribe(res => { - this.closed(); - }); - this.subscription = this.data.checkClose.subscribe(res => { - // Invoke the close after, result in fact is not important - window.setTimeout(() => { - this.doClose(); - }); - }); - + await this.data.checkClose; + this.close(); } } - initYesNo() { - // data.autoclose is not used - } - ngOnInit() { if ( this.data.type === DialogType.yesno ) { - this.initYesNo(); + ; } else { this.initAlert(); } diff --git a/src/app/gui/uds-gui.service.ts b/src/app/gui/uds-gui.service.ts index 58ca1cc..68dde34 100644 --- a/src/app/gui/uds-gui.service.ts +++ b/src/app/gui/uds-gui.service.ts @@ -3,7 +3,9 @@ import { Injectable } from '@angular/core'; import { ModalComponent, DialogType } from './modal/modal.component'; import { CredentialsModalComponent } from './credentials-modal/credentials-modal.component'; import { MatDialog } from '@angular/material/dialog'; -import { Observable } from 'rxjs'; +import { Observable, firstValueFrom } from 'rxjs'; + +const toPromise = (observable: Observable): Promise => firstValueFrom(observable); @Injectable() export class UDSGuiService { @@ -13,7 +15,7 @@ export class UDSGuiService { title: string, message: string, autoclose = 0, - checkClose: Observable = null + checkClose: Promise = null ) { const width = window.innerWidth < 800 ? '80%' : '40%'; const dialogRef = this.dialog.open(ModalComponent, { @@ -30,7 +32,7 @@ export class UDSGuiService { return dialogRef; } - yesno(title: string, message: string) { + yesno(title: string, message: string): Promise { const width = window.innerWidth < 800 ? '80%' : '40%'; const dialogRef = this.dialog.open(ModalComponent, { width, @@ -41,13 +43,13 @@ export class UDSGuiService { return dialogRef.componentInstance.yesno; } - askCredentials(username: string, domain: string): Observable<{username: string; password: string; domain: string}> { + askCredentials(username: string, domain: string): Promise<{username: string; password: string; domain: string}> { const dialogRef = this.dialog.open(CredentialsModalComponent, { data: { username, domain, }, }); - return dialogRef.afterClosed(); + return toPromise(dialogRef.afterClosed()); } } diff --git a/src/app/helpers/plugin.ts b/src/app/helpers/plugin.ts index bad9eef..8d88600 100644 --- a/src/app/helpers/plugin.ts +++ b/src/app/helpers/plugin.ts @@ -48,12 +48,12 @@ export class Plugin { ), 0, // Now UDS tries to check status - new Observable((observer) => { + new Promise((resolve) => { let readyTime = 0; const checker = () => { if (alert.componentInstance) { // Not closed dialog... - this.api.status(params[0], params[1]).subscribe( + this.api.status(params[0], params[1]).then( (data) => { if (data.status === 'ready') { if (!readyTime) { @@ -89,19 +89,16 @@ export class Plugin { alert.componentInstance.data.body = django.gettext( 'Machine ready, waiting for UDS Client' ); - observer.next(true); - observer.complete(); + resolve(true); } else if (data.status === 'running') { window.setTimeout(checker, this.delay); // Recheck after delay seconds } else { - observer.next(true); - observer.complete(); + resolve(true); notifyError(); } }, (error) => { - observer.next(true); - observer.complete(); + resolve(true); notifyError(error); } ); @@ -120,7 +117,7 @@ export class Plugin { }) ); - this.api.enabler(params[0], params[1]).subscribe( + this.api.enabler(params[0], params[1]).then( (data) => { if (data.error) { state = 'error'; @@ -162,15 +159,14 @@ export class Plugin { ), 0, // Now UDS tries to check status before closing dialog... - new Observable((observer) => { + new Promise((resolve) => { const checker = () => { if (alert.componentInstance) { // Not closed dialog... - this.api.transportUrl(url).subscribe( + this.api.transportUrl(url).then( (data) => { if (data.url) { - observer.next(true); - observer.complete(); // Notify window to close... + resolve(true); // Extract if credentials modal is requested let username = ''; let domain = ''; @@ -237,9 +233,9 @@ export class Plugin { if (askCredentials) { this.api.gui .askCredentials(username, domain) - .subscribe((result) => { + .then((result) => { // Update transport credentials - this.api.updateTransportTicket(ticket, scrambler,result.username, result.password, result.domain).subscribe( + this.api.updateTransportTicket(ticket, scrambler,result.username, result.password, result.domain).then( () => { openWindow(); } @@ -249,16 +245,14 @@ export class Plugin { openWindow(); // Open window } } else if (!data.running) { - observer.next(true); - observer.complete(); + resolve(true); notifyError(data.error); } else { window.setTimeout(checker, this.delay); // Recheck after 5 seconds } }, (error) => { - observer.next(true); - observer.complete(); + resolve(true); notifyError(error); } ); @@ -273,7 +267,7 @@ export class Plugin { text: string, info: string, waitTime: number, - checker: Observable = null + checker: Promise = null ) { return this.api.gui.alert( django.gettext('Launching service'), diff --git a/src/app/pages/services/services.component.ts b/src/app/pages/services/services.component.ts index 058292c..25503a8 100644 --- a/src/app/pages/services/services.component.ts +++ b/src/app/pages/services/services.component.ts @@ -84,7 +84,7 @@ export class ServicesComponent implements OnInit { // Obtain services list this.api .getServicesInformation() - .subscribe((result: JSONServicesInformation) => { + .then((result: JSONServicesInformation) => { this.servicesInformation = result; this.autorun(); diff --git a/src/app/uds-api.service-type.ts b/src/app/uds-api.service-type.ts index b74a644..6ed1248 100644 --- a/src/app/uds-api.service-type.ts +++ b/src/app/uds-api.service-type.ts @@ -19,22 +19,22 @@ export interface UDSApiServiceType { enabler( serviceId: string, transportId: string - ): Observable; + ): Promise; /* Service status */ status( serviceId: string, transportId: string - ): Observable; + ): Promise; /* Services resetter */ - action(action: string, serviceId: string): Observable; + action(action: string, serviceId: string): Promise; /* transport url */ - transportUrl(url: string): Observable; + transportUrl(url: string): Promise; /* Transport ticket credentials updater */ - updateTransportTicket(ticketId: string, scrambler: string, username: string, password: string, domain: string): Observable; + updateTransportTicket(ticketId: string, scrambler: string, username: string, password: string, domain: string): Promise; /* Go to admin dashboard */ gotoAdmin(): void; @@ -45,7 +45,7 @@ export interface UDSApiServiceType { /** * Gets services information */ - getServicesInformation(): Observable; + getServicesInformation(): Promise; /** * Executes custom javascript for service launch if it is available diff --git a/src/app/uds-api.service.ts b/src/app/uds-api.service.ts index fc23ee1..55cd805 100644 --- a/src/app/uds-api.service.ts +++ b/src/app/uds-api.service.ts @@ -2,7 +2,8 @@ import { Injectable } from '@angular/core'; import { HttpClient } from '@angular/common/http'; import { Router } from '@angular/router'; import { User, UDSConfig, Downloadable, Info } from './types/config'; -import { Observable } from 'rxjs'; +import { firstValueFrom, Observable } from 'rxjs'; +import { timeout } from 'rxjs/operators'; import { JSONServicesInformation, JSONEnabledService, @@ -23,7 +24,12 @@ declare const csrf: any; const DARK_THEME = 'dark-theme'; const LIGHT_THEME = 'light-theme'; +const TIMEOUT = 10000; +const toPromise = (observable: Observable, wait?: number): Promise => { + wait = wait || TIMEOUT; + return firstValueFrom(observable.pipe(timeout(wait))); +}; @Injectable() export class UDSApiService implements UDSApiServiceType { @@ -88,45 +94,50 @@ export class UDSApiService implements UDSApiServiceType { } /* Client enabler */ - enabler(serviceId: string, transportId: string) { + enabler(serviceId: string, transportId: string): Promise { const enabler = this.config.urls.enabler .replace('param1', serviceId) .replace('param2', transportId); - return this.http.get(enabler); + return toPromise(this.http.get(enabler)); } /* Check userService status */ - status( - serviceId: string, - transportId: string - ): Observable { + status(serviceId: string, transportId: string): Promise { const status = this.config.urls.status .replace('param1', serviceId) .replace('param2', transportId); - return this.http.get(status); + return toPromise(this.http.get(status)); } /* Services resetter */ - action(action: string, serviceId: string) { + action(action: string, serviceId: string): Promise { const actionURL = this.config.urls.action .replace('param1', serviceId) .replace('param2', action); - return this.http.get(actionURL); + return toPromise(this.http.get(actionURL)); } - transportUrl(url: string): Observable { - return this.http.get(url); + transportUrl(url: string): Promise { + return toPromise(this.http.get(url)); } - updateTransportTicket(ticketId: string, scrambler: string, username: string, password: string, domain: string): Observable { + updateTransportTicket( + ticketId: string, + scrambler: string, + username: string, + password: string, + domain: string + ): Promise { const url = this.config.urls.updateTransportTicket - .replace('param1', ticketId) - .replace('param2', scrambler); - return this.http.post(url, { - username, - password, - domain, - }); + .replace('param1', ticketId) + .replace('param2', scrambler); + return toPromise( + this.http.post(url, { + username, + password, + domain, + }) + ); } /* Images & static related */ @@ -149,15 +160,17 @@ export class UDSApiService implements UDSApiServiceType { /** * Gets services information */ - getServicesInformation(): Observable { - return this.http.get(this.config.urls.services); + getServicesInformation(): Promise { + return toPromise(this.http.get(this.config.urls.services)); } /** * Gets error string from a code */ getErrorInformation(errorCode: string): Observable { - return this.http.get(this.config.urls.error.replace('9999', errorCode)); + return this.http.get( + this.config.urls.error.replace('9999', errorCode) + ); } /** @@ -206,5 +219,4 @@ export class UDSApiService implements UDSApiServiceType { }); body.classList.add(dark ? DARK_THEME : LIGHT_THEME); } - } diff --git a/src/styles.scss b/src/styles.scss index d498764..2cba0e0 100644 --- a/src/styles.scss +++ b/src/styles.scss @@ -4,48 +4,53 @@ @include angular-material-theme($angular-default-theme); /* Cookie consent */ -@import "~cookieconsent/build/cookieconsent.min.css"; +@import '~cookieconsent/build/cookieconsent.min.css'; -html, body { - margin: 0; - font-family: 'Roboto', sans-serif; - height: 100%; +html, +body { + margin: 0; + font-family: 'Roboto', sans-serif; + height: 100%; } /* App loading logo */ .app-loading .logo { - width: 113px; - height: 120px; - background: url(); + width: 113px; + height: 120px; + background: url(); } /* Fix for vertical align on material icons */ i.material-icons { - vertical-align: middle !important; + vertical-align: middle !important; } /* colors for remmarks */ .highlight { - color: red; + color: red; } .mat-mdc-menu-content { - .mat-mdc-menu-item:hover { - background-color: #E8E8E8; - } + .mat-mdc-menu-item:hover { + background-color: #e8e8e8; } +} .mat-mdc-option:hover:not(.mat-mdc-option-disabled) { - background: #648cb0 !important; - color: white !important; + background: #648cb0 !important; + color: white !important; } - + .dark-theme { @include angular-material-color($angular-dark-theme); html, body { - background-color: $bg-dark; + background-color: $bg-dark; } } +/* Variables */ +:root { + --mdc-dialog-supporting-text-font: 'Roboto', 'Helvetica', 'Arial', sans-serif; +}