Commit 44b0a1100f19225fef098785db92d78af5ef75e1

Authored by Vladyslav_Prykhodko
1 parent 3b55ebe5

Add new device credentials MQTT_BASIC

... ... @@ -58,6 +58,37 @@
58 58 {{ 'device.rsa-key-required' | translate }}
59 59 </mat-error>
60 60 </mat-form-field>
  61 + <section *ngIf="deviceCredentialsFormGroup.get('credentialsType').value === deviceCredentialsType.MQTT_BASIC" formGroupName="credentialsBasic">
  62 + <mat-form-field class="mat-block">
  63 + <mat-label translate>device.client-id</mat-label>
  64 + <input matInput formControlName="clientId">
  65 + <mat-error *ngIf="deviceCredentialsFormGroup.get('credentialsBasic.clientId').hasError('pattern')">
  66 + {{ 'device.client-id-pattern' | translate }}
  67 + </mat-error>
  68 + </mat-form-field>
  69 + <mat-form-field class="mat-block">
  70 + <mat-label translate>device.user-name</mat-label>
  71 + <input matInput formControlName="userName" [required]="!!deviceCredentialsFormGroup.get('credentialsBasic.password').value">
  72 + <mat-error *ngIf="deviceCredentialsFormGroup.get('credentialsBasic.userName').hasError('required')">
  73 + {{ 'device.user-name-required' | translate }}
  74 + </mat-error>
  75 + </mat-form-field>
  76 + <mat-form-field class="mat-block">
  77 + <mat-label translate>device.password</mat-label>
  78 + <input matInput formControlName="password"
  79 + autocomplete="new-password"
  80 + (ngModelChange)="passwordChanged()"
  81 + [type]="hidePassword ? 'password' : 'text'">
  82 + <button mat-icon-button matSuffix type="button"
  83 + (click)="hidePassword = !hidePassword"
  84 + [attr.aria-pressed]="hidePassword">
  85 + <mat-icon>{{hidePassword ? 'visibility_off' : 'visibility'}}</mat-icon>
  86 + </button>
  87 + </mat-form-field>
  88 + <mat-error *ngIf="deviceCredentialsFormGroup.get('credentialsBasic').hasError('atLeastOne')">
  89 + {{ 'device.client-id-or-user-name-necessary' | translate }}
  90 + </mat-error>
  91 + </section>
61 92 </fieldset>
62 93 </div>
63 94 <div mat-dialog-actions fxLayoutAlign="end center">
... ...
... ... @@ -19,9 +19,23 @@ import { ErrorStateMatcher } from '@angular/material/core';
19 19 import { MAT_DIALOG_DATA, MatDialogRef } from '@angular/material/dialog';
20 20 import { Store } from '@ngrx/store';
21 21 import { AppState } from '@core/core.state';
22   -import { FormBuilder, FormControl, FormGroup, FormGroupDirective, NgForm, Validators } from '@angular/forms';
  22 +import {
  23 + FormBuilder,
  24 + FormControl,
  25 + FormGroup,
  26 + FormGroupDirective,
  27 + NgForm,
  28 + ValidationErrors,
  29 + ValidatorFn,
  30 + Validators
  31 +} from '@angular/forms';
23 32 import { DeviceService } from '@core/http/device.service';
24   -import { credentialTypeNames, DeviceCredentials, DeviceCredentialsType } from '@shared/models/device.models';
  33 +import {
  34 + credentialTypeNames,
  35 + DeviceCredentialMQTTBasic,
  36 + DeviceCredentials,
  37 + DeviceCredentialsType
  38 +} from '@shared/models/device.models';
25 39 import { DialogComponent } from '@shared/components/dialog.component';
26 40 import { Router } from '@angular/router';
27 41
... ... @@ -53,6 +67,8 @@ export class DeviceCredentialsDialogComponent extends
53 67
54 68 credentialTypeNamesMap = credentialTypeNames;
55 69
  70 + hidePassword = true;
  71 +
56 72 constructor(protected store: Store<AppState>,
57 73 protected router: Router,
58 74 @Inject(MAT_DIALOG_DATA) public data: DeviceCredentialsDialogData,
... ... @@ -69,7 +85,12 @@ export class DeviceCredentialsDialogComponent extends
69 85 this.deviceCredentialsFormGroup = this.fb.group({
70 86 credentialsType: [DeviceCredentialsType.ACCESS_TOKEN],
71 87 credentialsId: [''],
72   - credentialsValue: ['']
  88 + credentialsValue: [''],
  89 + credentialsBasic: this.fb.group({
  90 + clientId: ['', [Validators.pattern(/^[A-Za-z0-9]+$/)]],
  91 + userName: [''],
  92 + password: ['']
  93 + }, {validators: this.atLeastOne(Validators.required, ['clientId', 'userName'])})
73 94 });
74 95 if (this.isReadOnly) {
75 96 this.deviceCredentialsFormGroup.disable({emitEvent: false});
... ... @@ -89,10 +110,17 @@ export class DeviceCredentialsDialogComponent extends
89 110 this.deviceService.getDeviceCredentials(this.data.deviceId).subscribe(
90 111 (deviceCredentials) => {
91 112 this.deviceCredentials = deviceCredentials;
  113 + let credentialsValue = deviceCredentials.credentialsValue;
  114 + let credentialsBasic = {clientId: null, userName: null, password: null};
  115 + if (deviceCredentials.credentialsType === DeviceCredentialsType.MQTT_BASIC) {
  116 + credentialsValue = null;
  117 + credentialsBasic = deviceCredentials.credentialsValue as DeviceCredentialMQTTBasic;
  118 + }
92 119 this.deviceCredentialsFormGroup.patchValue({
93 120 credentialsType: deviceCredentials.credentialsType,
94 121 credentialsId: deviceCredentials.credentialsId,
95   - credentialsValue: deviceCredentials.credentialsValue
  122 + credentialsValue,
  123 + credentialsBasic
96 124 });
97 125 this.updateValidators();
98 126 }
... ... @@ -100,12 +128,16 @@ export class DeviceCredentialsDialogComponent extends
100 128 }
101 129
102 130 credentialsTypeChanged(): void {
103   - this.deviceCredentialsFormGroup.patchValue(
104   - {credentialsId: null, credentialsValue: null}, {emitEvent: true});
  131 + this.deviceCredentialsFormGroup.patchValue({
  132 + credentialsId: null,
  133 + credentialsValue: null,
  134 + credentialsBasic: {clientId: '', userName: '', password: ''}
  135 + }, {emitEvent: true});
105 136 this.updateValidators();
106 137 }
107 138
108 139 updateValidators(): void {
  140 + this.hidePassword = true;
109 141 const crendetialsType = this.deviceCredentialsFormGroup.get('credentialsType').value as DeviceCredentialsType;
110 142 switch (crendetialsType) {
111 143 case DeviceCredentialsType.ACCESS_TOKEN:
... ... @@ -113,27 +145,66 @@ export class DeviceCredentialsDialogComponent extends
113 145 this.deviceCredentialsFormGroup.get('credentialsId').updateValueAndValidity();
114 146 this.deviceCredentialsFormGroup.get('credentialsValue').setValidators([]);
115 147 this.deviceCredentialsFormGroup.get('credentialsValue').updateValueAndValidity();
  148 + this.deviceCredentialsFormGroup.get('credentialsBasic').disable();
116 149 break;
117 150 case DeviceCredentialsType.X509_CERTIFICATE:
118 151 this.deviceCredentialsFormGroup.get('credentialsValue').setValidators([Validators.required]);
119 152 this.deviceCredentialsFormGroup.get('credentialsValue').updateValueAndValidity();
120 153 this.deviceCredentialsFormGroup.get('credentialsId').setValidators([]);
121 154 this.deviceCredentialsFormGroup.get('credentialsId').updateValueAndValidity();
  155 + this.deviceCredentialsFormGroup.get('credentialsBasic').disable();
122 156 break;
  157 + case DeviceCredentialsType.MQTT_BASIC:
  158 + this.deviceCredentialsFormGroup.get('credentialsBasic').enable();
  159 + this.deviceCredentialsFormGroup.get('credentialsBasic').updateValueAndValidity();
  160 + this.deviceCredentialsFormGroup.get('credentialsId').setValidators([]);
  161 + this.deviceCredentialsFormGroup.get('credentialsId').updateValueAndValidity();
  162 + this.deviceCredentialsFormGroup.get('credentialsValue').setValidators([]);
  163 + this.deviceCredentialsFormGroup.get('credentialsValue').updateValueAndValidity();
123 164 }
124 165 }
125 166
  167 + private atLeastOne(validator: ValidatorFn, controls: string[] = null) {
  168 + return (group: FormGroup): ValidationErrors | null => {
  169 + if (!controls) {
  170 + controls = Object.keys(group.controls);
  171 + }
  172 + const hasAtLeastOne = group?.controls && controls.some(k => !validator(group.controls[k]));
  173 +
  174 + return hasAtLeastOne ? null : {atLeastOne: true};
  175 + };
  176 + }
  177 +
126 178 cancel(): void {
127 179 this.dialogRef.close(null);
128 180 }
129 181
130 182 save(): void {
131 183 this.submitted = true;
132   - this.deviceCredentials = {...this.deviceCredentials, ...this.deviceCredentialsFormGroup.value};
  184 + const deviceCredentialsValue = this.deviceCredentialsFormGroup.value;
  185 + if (deviceCredentialsValue.credentialsType === DeviceCredentialsType.MQTT_BASIC) {
  186 + deviceCredentialsValue.credentialsValue = deviceCredentialsValue.credentialsBasic;
  187 + }
  188 + delete deviceCredentialsValue.credentialsBasic;
  189 + this.deviceCredentials = {...this.deviceCredentials, ...deviceCredentialsValue};
133 190 this.deviceService.saveDeviceCredentials(this.deviceCredentials).subscribe(
134 191 (deviceCredentials) => {
135 192 this.dialogRef.close(deviceCredentials);
136 193 }
137 194 );
138 195 }
  196 +
  197 + passwordChanged() {
  198 + const value = this.deviceCredentialsFormGroup.get('credentialsBasic.password').value;
  199 + if (value !== '') {
  200 + this.deviceCredentialsFormGroup.get('credentialsBasic.userName').setValidators([Validators.required]);
  201 + if (this.deviceCredentialsFormGroup.get('credentialsBasic.userName').untouched) {
  202 + this.deviceCredentialsFormGroup.get('credentialsBasic.userName').markAsTouched({onlySelf: true});
  203 + }
  204 + this.deviceCredentialsFormGroup.get('credentialsBasic.userName').updateValueAndValidity();
  205 + } else {
  206 + this.deviceCredentialsFormGroup.get('credentialsBasic.userName').setValidators([]);
  207 + this.deviceCredentialsFormGroup.get('credentialsBasic.userName').updateValueAndValidity();
  208 + }
  209 + }
139 210 }
... ...
... ... @@ -292,13 +292,15 @@ export interface DeviceInfo extends Device {
292 292
293 293 export enum DeviceCredentialsType {
294 294 ACCESS_TOKEN = 'ACCESS_TOKEN',
295   - X509_CERTIFICATE = 'X509_CERTIFICATE'
  295 + X509_CERTIFICATE = 'X509_CERTIFICATE',
  296 + MQTT_BASIC = 'MQTT_BASIC'
296 297 }
297 298
298 299 export const credentialTypeNames = new Map<DeviceCredentialsType, string>(
299 300 [
300 301 [DeviceCredentialsType.ACCESS_TOKEN, 'Access token'],
301   - [DeviceCredentialsType.X509_CERTIFICATE, 'X.509 Certificate'],
  302 + [DeviceCredentialsType.X509_CERTIFICATE, 'MQTT X.509'],
  303 + [DeviceCredentialsType.MQTT_BASIC, 'MQTT Basic']
302 304 ]
303 305 );
304 306
... ... @@ -306,7 +308,13 @@ export interface DeviceCredentials extends BaseData<DeviceCredentialsId> {
306 308 deviceId: DeviceId;
307 309 credentialsType: DeviceCredentialsType;
308 310 credentialsId: string;
309   - credentialsValue: string;
  311 + credentialsValue: string | DeviceCredentialMQTTBasic;
  312 +}
  313 +
  314 +export interface DeviceCredentialMQTTBasic {
  315 + clientId: string;
  316 + userName: string;
  317 + password: string;
310 318 }
311 319
312 320 export interface DeviceSearchQuery extends EntitySearchQuery {
... ...
... ... @@ -718,6 +718,12 @@
718 718 "access-token-invalid": "Access token length must be from 1 to 20 characters.",
719 719 "rsa-key": "RSA public key",
720 720 "rsa-key-required": "RSA public key is required.",
  721 + "client-id": "Client ID",
  722 + "client-id-pattern": "Contains invalid character.",
  723 + "user-name": "User Name",
  724 + "user-name-required": "User Name is required.",
  725 + "client-id-or-user-name-necessary": "Client ID and/or User Name are necessary",
  726 + "password": "Password",
721 727 "secret": "Secret",
722 728 "secret-required": "Secret is required.",
723 729 "device-type": "Device type",
... ...