blob: 6619bfddbbaba0a3c177696b0ce33c7fe2dbdbc4 [file] [log] [blame]
/*
Copyright 2020 TATA ELXSI
Licensed under the Apache License, Version 2.0 (the 'License');
you may not use this file except in compliance with the License.
You may obtain a copy of the License at
http://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.
Author: KUMARAN M (kumaran.m@tataelxsi.co.in), RAJESH S (rajesh.s@tataelxsi.co.in), BARATH KUMAR R (barath.r@tataelxsi.co.in)
*/
/**
* @file Vim Account Component.
*/
import { HttpHeaders } from '@angular/common/http';
import { Component, ElementRef, Injector, OnInit, ViewChild } from '@angular/core';
import { AbstractControl, FormBuilder, FormGroup, Validators } from '@angular/forms';
import { ActivatedRoute, Router } from '@angular/router';
import { NgbModal, NgbModalRef } from '@ng-bootstrap/ng-bootstrap';
import { TranslateService } from '@ngx-translate/core';
import { NotifierService } from 'angular-notifier';
import 'codemirror/addon/dialog/dialog';
import 'codemirror/addon/display/autorefresh';
import 'codemirror/addon/display/fullscreen';
import 'codemirror/addon/edit/closebrackets';
import 'codemirror/addon/edit/matchbrackets';
import 'codemirror/addon/fold/brace-fold';
import 'codemirror/addon/fold/foldcode';
import 'codemirror/addon/fold/foldgutter';
import 'codemirror/addon/search/search';
import 'codemirror/addon/search/searchcursor';
import 'codemirror/keymap/sublime';
import 'codemirror/lib/codemirror';
import 'codemirror/mode/javascript/javascript';
import 'codemirror/mode/markdown/markdown';
import 'codemirror/mode/yaml/yaml';
import {
APIURLHEADER, CONFIGCONSTANT, ERRORDATA, MODALCLOSERESPONSEDATA, TYPEAWS, TYPEAZURE, TYPEOPENSTACK, TYPEOPENVIMNEBULA, TYPEOTERS,
TYPESECTION, TYPEVMWARE, VIM_TYPES
} from 'CommonModel';
import { environment } from 'environment';
import * as jsyaml from 'js-yaml';
import { RestService } from 'RestService';
import { SharedService, isNullOrUndefined } from 'SharedService';
import { VimAccountDetails } from 'VimAccountModel';
import { WarningComponent } from 'WarningComponent';
/**
* Creating component
* @Component takes NewVimaccountComponent.html as template url
*/
@Component({
selector: 'app-new-vimaccount',
templateUrl: './NewVimaccountComponent.html',
styleUrls: ['./NewVimaccountComponent.scss']
})
/** Exporting a class @exports NewVimaccountComponent */
export class NewVimaccountComponent implements OnInit {
/** To inject services @public */
public injector: Injector;
/** FormGroup vim New Account added to the form @ html @public */
public vimNewAccountForm: FormGroup;
/** Supported Vim type for the dropdown */
public vimType: TYPESECTION[];
/** Supported Vim type for the dropdown */
public selectedVimType: string;
/** Form submission Add */
public submitted: boolean = false;
/** Showing more details of collapase */
public isCollapsed: boolean = false;
/** Check the Projects loading results @public */
public isLocationLoadingResults: boolean = false;
/** Give the message for the loading @public */
public message: string = 'PLEASEWAIT';
/** Set the check value @public */
public check: boolean = false;
/** Handle the formate Change @public */
public defaults: {} = {
'text/x-yaml': ''
};
/** To Set Mode @public */
public mode: string = 'text/x-yaml';
/** To Set Mode @public */
public modeDefault: string = 'yaml';
/** options @public */
public options: {} = {
// eslint-disable-next-line no-invalid-this
mode: this.modeDefault,
showCursorWhenSelecting: true,
autofocus: false,
autoRefresh: true,
lineNumbers: true,
lineWrapping: true,
foldGutter: true,
gutters: ['CodeMirror-linenumbers', 'CodeMirror-foldgutter'],
autoCloseBrackets: true,
matchBrackets: true,
theme: 'neat',
keyMap: 'sublime'
};
/** Data @public */
public data: string = '';
/** contains vim ID @public */
public vimID: string;
/** Element ref for fileInput @public */
@ViewChild('fileInput', { static: true }) public fileInput: ElementRef;
/** Element ref for fileInput @public */
@ViewChild('fileInputLabel', { static: true }) public fileInputLabel: ElementRef;
/** Contains all methods related to shared @private */
public sharedService: SharedService;
/** Check key for the Edit form @public */
public checkFormKeys: string[] =
[
'name',
'vim_type',
'vim_tenant_name',
'description',
'vim_url',
'schema_type',
'vim_user',
'vim_password',
'locationName',
'latitude',
'longitude',
'config'
];
/** Contains config details in edit @public */
public config: {};
/** Contains latitude value @public */
public latitude: string;
/** Contains longitude value @public */
public longitude: string;
/** Contains location value @public */
public locationName: string;
/** Contains VIMAccount Details @private */
private details: VimAccountDetails;
/** Contains config with location @private */
private configLocation: string;
/** Check for config length @private */
// eslint-disable-next-line @typescript-eslint/no-magic-numbers
private configLength: number = 3;
/** Contains config length from get api @private */
private getConfigLength: number;
/** Contains config when update @private */
private updateConfig: object;
/** Contains config length when update @private */
private updateConfigLength: number;
/** Instance of the rest service @private */
private restService: RestService;
/** Holds the instance of router class @private */
private router: Router;
/** Controls the header form @private */
private headers: HttpHeaders;
/** FormBuilder instance added to the formBuilder @private */
private formBuilder: FormBuilder;
/** Notifier service to popup notification @private */
private notifierService: NotifierService;
/** Contains tranlsate instance @private */
private translateService: TranslateService;
/** Holds teh instance of AuthService class of type AuthService @private */
private activatedRoute: ActivatedRoute;
/** Instance of the modal service @private */
private modalService: NgbModal;
constructor(injector: Injector) {
this.injector = injector;
this.restService = this.injector.get(RestService);
this.formBuilder = this.injector.get(FormBuilder);
this.router = this.injector.get(Router);
this.notifierService = this.injector.get(NotifierService);
this.translateService = this.injector.get(TranslateService);
this.sharedService = this.injector.get(SharedService);
this.activatedRoute = this.injector.get(ActivatedRoute);
this.modalService = this.injector.get(NgbModal);
}
/** convenience getter for easy access to form fields */
get f(): FormGroup['controls'] { return this.vimNewAccountForm.controls; }
/**
* Lifecyle Hooks the trigger before component is instantiate
*/
public ngOnInit(): void {
this.vimID = this.activatedRoute.snapshot.paramMap.get('id');
this.vimType = VIM_TYPES;
this.headers = new HttpHeaders({
Accept: 'application/json',
'Content-Type': 'application/json',
'Cache-Control': 'no-cache, no-store, must-revalidate, max-age=0'
});
this.initializeForm();
if (!isNullOrUndefined(this.vimID)) {
this.getVIMDetails(this.vimID);
}
}
/** VIM Initialize Forms @public */
public initializeForm(): void {
this.vimNewAccountForm = this.formBuilder.group({
name: [null, Validators.required],
vim_type: [null, Validators.required],
vim_tenant_name: [null, Validators.required],
description: [null],
vim_url: [null, [Validators.required, Validators.pattern(this.sharedService.REGX_URL_PATTERN)]],
schema_type: [''],
vim_user: [null, Validators.required],
vim_password: [null, Validators.required],
locationName: [''],
latitude: ['', Validators.pattern(this.sharedService.REGX_LAT_PATTERN)],
longitude: ['', Validators.pattern(this.sharedService.REGX_LONG_PATTERN)],
config: this.paramsBuilder()
});
}
/** Generate params for config @public */
public paramsBuilder(): FormGroup {
return this.formBuilder.group({
location: [null]
});
}
/** Fetching the vim details from get api @protected */
private getVIMDetails(id: string): void {
this.isLocationLoadingResults = true;
this.restService.getResource(environment.VIMACCOUNTS_URL + '/' + id).subscribe((vimAccountsData: VimAccountDetails) => {
this.details = vimAccountsData;
if (!isNullOrUndefined(this.details.config.location)) {
this.configLocation = this.details.config.location;
if (this.configLocation.indexOf(',') !== -1) {
this.locationName = this.configLocation.split(',')[0];
this.latitude = this.configLocation.split(',')[1];
this.longitude = this.configLocation.split(',')[2];
}
}
delete this.details.config.location;
this.getConfigLength = Object.keys(this.details.config).length;
this.getFormControl('schema_type').disable();
this.getFormControl('vim_url').disable();
this.getFormControl('vim_type').disable();
this.config = { ...this.details.config };
this.details.vim_password = '';
this.setEditValue(this.details, this.checkFormKeys);
this.isLocationLoadingResults = false;
}, (error: ERRORDATA) => {
this.restService.handleError(error, 'get');
this.isLocationLoadingResults = false;
});
}
/** Set the value for the Edit Section @public */
public setEditValue(formValues: VimAccountDetails, checkKey: string[]): void {
Object.keys(formValues).forEach((name: string): void => {
if (checkKey.includes(name)) {
if (name === 'config') {
this.loadConfig();
this.getFormControl('locationName').patchValue(this.locationName);
this.getFormControl('latitude').patchValue(this.latitude);
this.getFormControl('longitude').patchValue(this.longitude);
}
else {
// eslint-disable-next-line security/detect-object-injection
this.getFormControl(name).setValue(formValues[name], { onlySelf: true });
this.getFormControl(name).updateValueAndValidity();
}
}
});
}
/** On modal submit newVimAccountSubmit will called @public */
public newVimAccountSubmit(): void {
this.submitted = true;
if (!this.vimNewAccountForm.invalid) {
this.isLocationLoadingResults = true;
this.sharedService.cleanForm(this.vimNewAccountForm, 'vim');
if (!isNullOrUndefined(this.data) && this.data !== '') {
Object.assign(this.vimNewAccountForm.value.config, jsyaml.load(this.data.toString(), { json: true }));
} else {
Object.keys(this.vimNewAccountForm.value.config).forEach((res: string): void => {
if (res !== 'location') {
// eslint-disable-next-line @typescript-eslint/no-dynamic-delete, security/detect-object-injection
delete this.vimNewAccountForm.value.config[res];
}
});
}
if (!isNullOrUndefined(this.vimNewAccountForm.value.latitude) && !isNullOrUndefined(this.vimNewAccountForm.value.longitude)) {
this.vimNewAccountForm.value.config.location = this.vimNewAccountForm.value.locationName + ',' +
this.vimNewAccountForm.value.longitude + ',' +
this.vimNewAccountForm.value.latitude;
}
if (isNullOrUndefined(this.vimNewAccountForm.value.config.location)) {
delete this.vimNewAccountForm.value.config.location;
}
Object.keys(this.vimNewAccountForm.value.config).forEach((res: string): void => {
// eslint-disable-next-line security/detect-object-injection
if (isNullOrUndefined(this.vimNewAccountForm.value.config[res]) || this.vimNewAccountForm.value.config[res] === '') {
// eslint-disable-next-line @typescript-eslint/no-dynamic-delete, security/detect-object-injection
delete this.vimNewAccountForm.value.config[res];
}
});
delete this.vimNewAccountForm.value.config.location;
if (!isNullOrUndefined(this.data)) {
this.updateConfig = jsyaml.load(this.data, { json: true });
if (!isNullOrUndefined(this.updateConfig)) {
this.updateConfigLength = Object.keys(this.updateConfig).length;
}
}
if (this.updateConfig === undefined) {
this.notifierService.notify('warning', this.translateService.instant('PAGE.VIMDETAILS.VIMDELETE'));
this.isLocationLoadingResults = false;
} else if (this.getConfigLength > this.updateConfigLength) {
this.notifierService.notify('warning', this.translateService.instant('PAGE.VIMDETAILS.VIMEMPTY'));
this.isLocationLoadingResults = false;
}
if (!isNullOrUndefined(this.vimID) && ((this.getConfigLength <= this.updateConfigLength))) {
this.editVIM();
} else if (isNullOrUndefined(this.vimID)) {
this.createNewVIM();
}
}
}
/** Create a new VIM Account @public */
public createNewVIM(): void {
const apiURLHeader: APIURLHEADER = {
url: environment.VIMACCOUNTS_URL,
httpOptions: { headers: this.headers }
};
delete this.vimNewAccountForm.value.locationName;
delete this.vimNewAccountForm.value.latitude;
delete this.vimNewAccountForm.value.longitude;
this.restService.postResource(apiURLHeader, this.vimNewAccountForm.value)
.subscribe((result: { id: string }): void => {
this.notifierService.notify('success', this.translateService.instant('PAGE.VIM.CREATEDSUCCESSFULLY'));
this.isLocationLoadingResults = false;
this.router.navigate(['vim/info/' + result.id]).catch((): void => {
// Error Cached;
});
}, (error: ERRORDATA): void => {
this.restService.handleError(error, 'post');
this.isLocationLoadingResults = false;
});
}
/** Create a edit VIM Account @public */
public editVIM(): void {
const apiURLHeader: APIURLHEADER = {
url: environment.VIMACCOUNTS_URL + '/' + this.vimID,
httpOptions: { headers: this.headers }
};
delete this.vimNewAccountForm.value.locationName;
delete this.vimNewAccountForm.value.latitude;
delete this.vimNewAccountForm.value.longitude;
this.restService.patchResource(apiURLHeader, this.vimNewAccountForm.value)
.subscribe((result: { id: string }): void => {
this.notifierService.notify('success', this.translateService.instant('PAGE.VIM.UPDATEDSUCCESSFULLY'));
this.isLocationLoadingResults = false;
this.router.navigate(['vim/info/' + this.vimID]).catch((): void => {
// Error Cached;
});
}, (error: ERRORDATA): void => {
this.restService.handleError(error, 'post');
this.isLocationLoadingResults = false;
});
}
/** HandleChange function @public */
public handleChange($event: string): void {
this.data = $event;
}
/** Routing to VIM Account Details Page @public */
public onVimAccountBack(): void {
this.router.navigate(['vim/details']).catch((): void => {
// Error Cached
});
}
/** Drag and drop feature and fetchind the details of files @private */
public filesDropped(files: FileList): void {
if (files && files.length === 1) {
this.sharedService.getFileString(files, 'yaml').then((fileContent: string): void => {
const getJson: string = jsyaml.load(fileContent, { json: true });
this.defaults['text/x-yaml'] = fileContent;
this.data = fileContent;
}).catch((err: string): void => {
if (err === 'typeError') {
this.notifierService.notify('error', this.translateService.instant('YAMLFILETYPEERRROR'));
} else {
this.notifierService.notify('error', this.translateService.instant('ERROR'));
}
this.fileInputLabel.nativeElement.innerText = this.translateService.instant('CHOOSEFILE');
this.fileInput.nativeElement.value = null;
});
} else if (files && files.length > 1) {
this.notifierService.notify('error', this.translateService.instant('DROPFILESVALIDATION'));
}
this.fileInputLabel.nativeElement.innerText = files[0].name;
this.fileInput.nativeElement.value = null;
}
/** Check data is empty or not to load config @public */
public checkData(): void {
if (this.data !== '' && this.data.length > this.configLength) {
// eslint-disable-next-line security/detect-non-literal-fs-filename
const modalRef: NgbModalRef = this.modalService.open(WarningComponent, { backdrop: 'static' });
modalRef.componentInstance.heading = this.translateService.instant('PAGE.VIMDETAILS.VIMHEADER');
modalRef.componentInstance.confirmationMessage = this.translateService.instant('PAGE.VIMDETAILS.VIMCONTENT');
modalRef.componentInstance.submitMessage = this.translateService.instant('PAGE.VIMDETAILS.VIMSUBMIT');
modalRef.result.then((result: MODALCLOSERESPONSEDATA): void => {
if (result.message === CONFIGCONSTANT.done) {
this.loadSampleConfig();
}
}).catch((): void => {
// Catch Navigation Error
});
} else if (this.data.length < this.configLength || this.data === '') {
this.loadSampleConfig();
}
}
/** Load sample config based on VIM type @public */
public loadSampleConfig(): void {
if (this.selectedVimType === 'openstack') {
this.defaults['text/x-yaml'] = jsyaml.dump(TYPEOPENSTACK);
this.data = JSON.stringify(TYPEOPENSTACK, null, '\t');
} else if (this.selectedVimType === 'aws') {
this.defaults['text/x-yaml'] = jsyaml.dump(TYPEAWS);
this.data = JSON.stringify(TYPEAWS, null, '\t');
} else if (this.selectedVimType === 'vmware') {
this.defaults['text/x-yaml'] = jsyaml.dump(TYPEVMWARE);
this.data = JSON.stringify(TYPEVMWARE, null, '\t');
} else if (this.selectedVimType === 'openvim' || this.selectedVimType === 'opennebula') {
this.defaults['text/x-yaml'] = jsyaml.dump(TYPEOPENVIMNEBULA);
this.data = JSON.stringify(TYPEOPENVIMNEBULA, null, '\t');
} else if (this.selectedVimType === 'azure' || this.selectedVimType === 'opennebula') {
this.defaults['text/x-yaml'] = jsyaml.dump(TYPEAZURE);
this.data = JSON.stringify(TYPEAZURE, null, '\t');
} else {
this.defaults['text/x-yaml'] = jsyaml.dump(TYPEOTERS);
this.data = JSON.stringify(TYPEOTERS, null, '\t');
}
}
/** Load sample config based on VIM type in edit @public */
public loadConfig(): void {
this.clearConfig();
if (this.details.vim_type === 'openstack') {
this.defaults['text/x-yaml'] = jsyaml.dump(this.config);
this.data = JSON.stringify(this.config, null, '\t');
} else if (this.details.vim_type === 'aws') {
this.defaults['text/x-yaml'] = jsyaml.dump(this.config);
this.data = JSON.stringify(this.config, null, '\t');
} else if (this.details.vim_type === 'vmware') {
this.defaults['text/x-yaml'] = jsyaml.dump(this.config);
this.data = JSON.stringify(this.config, null, '\t');
} else if (this.details.vim_type === 'openvim' || this.details.vim_type === 'opennebula') {
this.defaults['text/x-yaml'] = jsyaml.dump(this.config);
this.data = JSON.stringify(this.config, null, '\t');
} else if (this.details.vim_type === 'azure' || this.details.vim_type === 'opennebula') {
this.defaults['text/x-yaml'] = jsyaml.dump(this.config);
this.data = JSON.stringify(this.config, null, '\t');
} else {
this.defaults['text/x-yaml'] = jsyaml.dump(this.config);
this.data = JSON.stringify(this.config, null, '\t');
}
}
/** Clear config parameters @public */
public clearConfig(): void {
this.check = true;
if (this.data !== '' && this.data.length > this.configLength) {
// eslint-disable-next-line security/detect-non-literal-fs-filename
const modalRef: NgbModalRef = this.modalService.open(WarningComponent, { backdrop: 'static' });
modalRef.componentInstance.heading = this.translateService.instant('PAGE.VIMDETAILS.VIMHEADER');
modalRef.componentInstance.confirmationMessage = this.translateService.instant('PAGE.VIMDETAILS.CLEARCONTENT');
modalRef.componentInstance.submitMessage = this.translateService.instant('PAGE.VIMDETAILS.VIMSUBMIT');
modalRef.result.then((result: MODALCLOSERESPONSEDATA): void => {
if (result.message === CONFIGCONSTANT.done) {
this.defaults['text/x-yaml'] = '';
this.data = '';
this.fileInput.nativeElement.value = null;
}
}).catch((): void => {
// Catch Navigation Error
});
} else {
this.defaults['text/x-yaml'] = '';
this.data = '';
this.fileInput.nativeElement.value = null;
}
}
/** Used to get the AbstractControl of controlName passed @private */
private getFormControl(controlName: string): AbstractControl {
// eslint-disable-next-line security/detect-object-injection
return this.vimNewAccountForm.controls[controlName];
}
}