diff --git a/src/app/api-connector/facility.service.ts b/src/app/api-connector/facility.service.ts index 4e8216fe07..b4a03d766f 100644 --- a/src/app/api-connector/facility.service.ts +++ b/src/app/api-connector/facility.service.ts @@ -24,10 +24,10 @@ export class FacilityService { } /** - * Get all available computecenters. - * - * @returns - */ + * Get all available computecenters. + * + * @returns + */ getComputeCenters(): Observable { return this.http.get(`${ApiSettings.getApiBaseURL()}computecenters/`, { withCredentials: true, @@ -35,8 +35,8 @@ export class FacilityService { } /** - * Sets support e-mail addresses for computecenter. - */ + * Sets support e-mail addresses for computecenter. + */ setSupportMails(facilityId: string, supportMails: string): Observable { const params: HttpParams = new HttpParams().set('mails', supportMails); @@ -47,8 +47,8 @@ export class FacilityService { } /** - * Get support e-mail addresses for computecenter. - */ + * Get support e-mail addresses for computecenter. + */ getSupportMails(facilityId: string): Observable { return this.http.get(`${ApiSettings.getApiBaseURL()}computecenters/${facilityId}/supportMails/`, { withCredentials: true, @@ -146,11 +146,11 @@ export class FacilityService { } /** - * Sets the newsID of the facility news which contains the motd for facility with the corresponding facility ID. - * - * @param facilityID facility id of the facility to set the id - * @param newsId the id of the news containing the motd - */ + * Sets the newsID of the facility news which contains the motd for facility with the corresponding facility ID. + * + * @param facilityID facility id of the facility to set the id + * @param newsId the id of the news containing the motd + */ setMOTDForFacility(facilityID: string, newsId: string): Observable { const httpParams: HttpParams = new HttpParams().set('facilityID', facilityID).set('newsID', newsId); @@ -160,10 +160,10 @@ export class FacilityService { } /** - * Get all facility, where the current user is manager. - * - * @returns - */ + * Get all facility, where the current user is manager. + * + * @returns + */ getManagerFacilities(): Observable { return this.http.get(`${ApiSettings.getApiBaseURL()}facilityManagers/current/facilities/`, { withCredentials: true, @@ -178,12 +178,12 @@ export class FacilityService { } /** - * Get allowed groups from a facility with a specific status. - * - * @param facility - * @param status - * @returns - */ + * Get allowed groups from a facility with a specific status. + * + * @param facility + * @param status + * @returns + */ getFacilityAllowedGroupsWithDetailsAndSpecificStatus( facility: number | string, status: number, @@ -199,16 +199,18 @@ export class FacilityService { } /** - * Gets FacilityGroups by the elixirId of the member. - * - * @param facility the facility - * @param elixir_id the id of the member - */ - getFacilityGroupsByMemberElixirId(facility: number | string, elixir_id: string): Observable { + * Gets FacilityGroups by the elixirId of the member. + * + * @param facility the facility + * @param elixir_id the id of the member + */ + getFacilityGroupsByMemberElixirId(facility: number | string, elixir_id: string, isPi: boolean, isAdmin: boolean, isMember: boolean): Observable { return this.http .get(`${ApiSettings.getApiBaseURL()}computecenters/${facility}/projects/filter/`, { withCredentials: true, - params: { elixir_id: elixir_id.toString() }, + params: { + elixir_id, isPi, isAdmin, isMember, + }, }) .pipe( map((applications: Application[]): Application[] => applications.map((application: Application): Application => new Application(application))), @@ -216,11 +218,11 @@ export class FacilityService { } /** - * Get all resources assigned to a facility. - * - * @param facility id of the facility - * @returns - */ + * Get all resources assigned to a facility. + * + * @param facility id of the facility + * @returns + */ getFacilityResources(facility: number | string): Observable { return this.http.get(`${ApiSettings.getApiBaseURL()}computecenters/${facility}/projects/resources/`, { withCredentials: true, @@ -228,11 +230,11 @@ export class FacilityService { } /** - * Gets all facility applications which are waiting for conirmation. - * - * @param facility - * @returns - */ + * Gets all facility applications which are waiting for conirmation. + * + * @param facility + * @returns + */ getFacilityApplicationsWaitingForConfirmation(facility: number | string): Observable { return this.http.get(`${ApiSettings.getApiBaseURL()}computecenters/${facility}/applications/`, { withCredentials: true, @@ -240,11 +242,11 @@ export class FacilityService { } /** - * Gets all facility applications history. - * - * @param facility - * @returns - */ + * Gets all facility applications history. + * + * @param facility + * @returns + */ getFacilityApplicationsHistory(facility: number | string): Observable { return this.http.get( `${ApiSettings.getApiBaseURL()}computecenters/${facility}/applications_history/`, @@ -255,11 +257,11 @@ export class FacilityService { } /** - * Get application for facility by id. - * - * @param facility self-speaking - * @param id self-speaking - */ + * Get application for facility by id. + * + * @param facility self-speaking + * @param id self-speaking + */ getFacilityApplicationById(facility: number | string, id: string): Observable { return this.http.get( `${ApiSettings.getApiBaseURL()}computecenters/${facility}/applications/${id}/detail/`, @@ -270,11 +272,11 @@ export class FacilityService { } /** - * Gets all volumes from a specific facility. - * - * @param facility - * @returns - */ + * Gets all volumes from a specific facility. + * + * @param facility + * @returns + */ getFacilityVolumes(facility: number | string, items_per_page: number, current_page: number): Observable { const params: HttpParams = new HttpParams() .set('items_per_page', items_per_page.toString()) @@ -289,13 +291,13 @@ export class FacilityService { } /** - * Gets all volumes from a specific facility. - * - * @param facility - * @param currentPage - * @param snapsPerSite - * @returns - */ + * Gets all volumes from a specific facility. + * + * @param facility + * @param currentPage + * @param snapsPerSite + * @returns + */ getFacilitySnapshots( facility: number | string, currentPage: number, @@ -318,11 +320,11 @@ export class FacilityService { } /** - * Gets all facility modification applications which are waiting for conirmation. - * - * @param facility - * @returns - */ + * Gets all facility modification applications which are waiting for conirmation. + * + * @param facility + * @returns + */ getFacilityModificationApplicationsWaitingForConfirmation(facility: number | string): Observable { return this.http.get( `${ApiSettings.getApiBaseURL()}computecenters/${facility}/modification_applications/`, @@ -333,12 +335,12 @@ export class FacilityService { } /** - * Approves an facility application. - * - * @param facility - * @param application_id - * @returns - */ + * Approves an facility application. + * + * @param facility + * @param application_id + * @returns + */ approveFacilityApplication(facility: number | string, application_id: number | string): Observable { const params: HttpParams = new HttpParams().set('action', 'approve'); @@ -383,12 +385,12 @@ export class FacilityService { } /** - * Deletes an RamFactor. - * - * @param facility - * @param resource_machine_id - * @returns - */ + * Deletes an RamFactor. + * + * @param facility + * @param resource_machine_id + * @returns + */ deleteResourceMachine( facility: number | string, resource_machine_id: number | string, @@ -514,11 +516,11 @@ export class FacilityService { } /** - * Updates the CoreFactor. - * - * @param facility - * @param factor - */ + * Updates the CoreFactor. + * + * @param facility + * @param factor + */ updateVolumeStorageFactor(facility: number | string, factor: VolumeStorageFactor): Observable { const params: HttpParams = new HttpParams().set('factor', JSON.stringify(factor)); @@ -533,11 +535,11 @@ export class FacilityService { } /** - * Updates the CoreFactor. - * - * @param facility - * @param factor - */ + * Updates the CoreFactor. + * + * @param facility + * @param factor + */ updateObjectStorageFactor(facility: number | string, factor: ObjectStorageFactor): Observable { const params: HttpParams = new HttpParams().set('factor', JSON.stringify(factor)); @@ -552,12 +554,12 @@ export class FacilityService { } /** - * Deletes an CoreFactor. - * - * @param facility - * @param factor_id - * @returns - */ + * Deletes an CoreFactor. + * + * @param facility + * @param factor_id + * @returns + */ deleteVolumeStorageFactor(facility: number | string, factor_id: number | string): Observable { return this.http.delete( `${ApiSettings.getApiBaseURL()}computecenters/${facility}/resources/volumeStorageFactors/${factor_id}/`, @@ -568,12 +570,12 @@ export class FacilityService { } /** - * Deletes an CoreFactor. - * - * @param facility - * @param factor_id - * @returns - */ + * Deletes an CoreFactor. + * + * @param facility + * @param factor_id + * @returns + */ deleteObjectStorageFactor(facility: number | string, factor_id: number | string): Observable { return this.http.delete( `${ApiSettings.getApiBaseURL()}computecenters/${facility}/resources/objectStorageFactors/${factor_id}/`, @@ -645,11 +647,11 @@ export class FacilityService { } /** - * Get CoreFactors from a facility. - * - * @param facility - * @returns - */ + * Get CoreFactors from a facility. + * + * @param facility + * @returns + */ getVolumeStorageFactors(facility: number | string): Observable { return this.http.get( `${ApiSettings.getApiBaseURL()}computecenters/${facility}/resources/volumeStorageFactors/`, @@ -660,11 +662,11 @@ export class FacilityService { } /** - * Get CoreFactors from a facility. - * - * @param facility - * @returns - */ + * Get CoreFactors from a facility. + * + * @param facility + * @returns + */ getObjectStorageFactors(facility: number | string): Observable { return this.http.get( `${ApiSettings.getApiBaseURL()}computecenters/${facility}/resources/objectStorageFactors/`, @@ -675,12 +677,12 @@ export class FacilityService { } /** - * Declines an application for the facility - * - * @param facility - * @param application_id - * @returns - */ + * Declines an application for the facility + * + * @param facility + * @param application_id + * @returns + */ declineFacilityApplication(facility: number, application_id: number): Observable { const params: HttpParams = new HttpParams().set('action', 'decline'); @@ -695,18 +697,18 @@ export class FacilityService { } /** - * Sends an email to all members of the facility. - * - * @param facility facility that should be contacted - * @param subject email subject - * @param message email message - * @param project_type which users to email - * @param reply reply address - * @param sendNews boolean if news should be send - * @param alternative_news_text an alternative news text - * @param news_tags additional tags - * @returns - */ + * Sends an email to all members of the facility. + * + * @param facility facility that should be contacted + * @param subject email subject + * @param message email message + * @param project_type which users to email + * @param reply reply address + * @param sendNews boolean if news should be send + * @param alternative_news_text an alternative news text + * @param news_tags additional tags + * @returns + */ sendMailToFacility( facility: string, subject: string, @@ -734,12 +736,12 @@ export class FacilityService { } /** - * Get Members of a project with emails. - * - * @param groupid id of the group - * @param facility id of the facility - * @returns - */ + * Get Members of a project with emails. + * + * @param groupid id of the group + * @param facility id of the facility + * @returns + */ getFacilityGroupRichMembers(groupid: number | string, facility: number | string): Observable { return this.http.get( `${ApiSettings.getApiBaseURL()}computecenters/${facility}/projects/${groupid}/members/`, diff --git a/src/app/api-connector/vo.service.ts b/src/app/api-connector/vo.service.ts index 533931d2b9..f472afce24 100644 --- a/src/app/api-connector/vo.service.ts +++ b/src/app/api-connector/vo.service.ts @@ -14,7 +14,8 @@ import { MaintenanceTimeFrame } from '../vo_manager/maintenance/maintenanceTimeF */ @Injectable() export class VoService { - constructor(private http: HttpClient) {} + constructor(private http: HttpClient) { + } getTsvInformation(): Observable { return this.http.get(`${ApiSettings.getApiBaseURL()}voManagers/tsv_information/`, { @@ -78,6 +79,19 @@ export class VoService { ); } + getGroupsByMemberElixirId(elixir_id: string, isPi: boolean, isAdmin: boolean, isMember: boolean): Observable { + return this.http + .get(`${ApiSettings.getApiBaseURL()}vo/projects/filter/`, { + withCredentials: true, + params: { + elixir_id, isPi, isAdmin, isMember, + }, + }) + .pipe( + map((applications: Application[]): Application[] => applications.map((application: Application): Application => new Application(application))), + ); + } + getProjectStatus(groupid: number | string): Observable { return this.http.get(`${ApiSettings.getApiBaseURL()}vo/projects/${groupid}/status/`, { withCredentials: true, @@ -189,11 +203,11 @@ export class VoService { } /** - * Get members of a project with emails. - * - * @param groupid id of the group - * @returns - */ + * Get members of a project with emails. + * + * @param groupid id of the group + * @returns + */ getVoGroupRichMembers(groupid: number | string): Observable { return this.http.get(`${ApiSettings.getApiBaseURL()}vo/projects/${groupid}/members/`, { withCredentials: true, diff --git a/src/app/facility_manager/facilityprojectsoverview.component.html b/src/app/facility_manager/facilityprojectsoverview.component.html index b733c50afb..4145c0989c 100644 --- a/src/app/facility_manager/facilityprojectsoverview.component.html +++ b/src/app/facility_manager/facilityprojectsoverview.component.html @@ -1,679 +1,766 @@
-
- -
-
- -
+
+ +
+
+ +
-
- +
+ - + -
-
-
- -
-
- -
-
- - -
-
- -
-
- -
-
-
-
Projects
+
+
+
+ +
+
+ +
+
+ + +
+
+ +
+
+ +
+
+
+
Projects
-
-
- - +
+
+ + -
- +
+ - + - -
+ +
- -
+ +
-
-
- -
- -
- Loading... -
-
-
-
-
-
-
- -
+ + + + + } @else { + + + + + + } + + +
+
+
+
+ +
+ +
+ + + + + + + diff --git a/src/app/facility_manager/facilityprojectsoverview.component.ts b/src/app/facility_manager/facilityprojectsoverview.component.ts index 7047290e56..fd7b9d25e0 100644 --- a/src/app/facility_manager/facilityprojectsoverview.component.ts +++ b/src/app/facility_manager/facilityprojectsoverview.component.ts @@ -32,123 +32,130 @@ import { CsvMailTemplateModel } from '../shared/classes/csvMailTemplate.model'; providers: [FacilityService, UserService, GroupService, ApiSettings, NewsService, ProjectSortService], }) export class FacilityProjectsOverviewComponent extends AbstractBaseClass implements OnInit { - @Input() voRegistrationLink: string = environment.voRegistrationLink; - - title: string = 'Projects Overview'; - filter: string; - - membersLoaded: boolean = false; - public memberFilter: string = ''; - filteredMembers: object[] = []; - selectedMember: object[] = []; - facility_members: object[] = []; - - filterChanged: Subject = new Subject(); - isLoaded: boolean = false; - projects: Application[] = []; - show_openstack_projects: boolean = true; - show_simple_vm_projects: boolean = true; - details_loaded: boolean = false; - selectedEmailProjects: Application[] = []; - bsModalRef: BsModalRef; - - /** - * Approved group status. - * - * @type {number} - */ - STATUS_APPROVED: number = 2; - - selectedProjectType: string = 'ALL'; - - // modal variables for User list - public selectedProjectForSearch: Application; - public usersModalProjectMembers: ProjectMember[] = []; - allFacilityMembers: object[] = []; - public usersModalProjectID: number; - public usersModalProjectName: string; - public selectedProject: Application; - public userSearchValue: string; - - public emailSubject: string; - public emailText: string; - public emailStatus: number = 0; - public emailReply: string = ''; - public sendNews: boolean; - public alternative_emailText: string = ''; - public news_tags: string[] = []; - FILTER_DEBOUNCE_TIME: number = 500; - - public managerFacilities: [string, number][] = []; - public selectedFacility: [string, number]; - projects_filtered: Application[] = []; - facilitySupportMails: string = ''; - supportMailEditing: boolean = false; - PREDEFINED_TAGS: string[] = ['downtime', 'openstack', 'simplevm', 'maintenance', 'update']; - - @ViewChildren(NgbdSortableHeaderDirective) headers: QueryList; - - applictions$: Observable; - total$: Observable; - - constructor( - private groupService: GroupService, - private facilityService: FacilityService, - private newsService: NewsService, - public sortProjectService: ProjectSortService, - private modalService: BsModalService, - private emailService: EmailService, - ) { - super(); - } - - switchShowSimpleVmProjects(): void { - this.show_simple_vm_projects = !this.show_simple_vm_projects; - } - - switchOpenStackVmProjects(): void { - this.show_openstack_projects = !this.show_openstack_projects; - } - - setEmailSubject(): void { - switch (this.selectedProjectType) { - case 'ALL': - this.emailSubject = `[${this.selectedFacility['Facility']}]`; - break; - case 'OVP': - this.emailSubject = `[${this.selectedFacility['Facility']}: OpenStack]`; - break; - case 'SVP': - this.emailSubject = `[${this.selectedFacility['Facility']}: SimpleVm]`; - break; - case 'USER': - this.emailSubject = `[${this.selectedFacility['Facility']}: Specific Members]`; - break; - default: - // eslint-disable-next-line no-case-declarations - const pro: Application = this.projects.find( - (project: Application): boolean => project.project_application_perun_id.toString() === this.selectedProjectType.toString(), - ); - if (pro) { - this.emailSubject = `[${this.selectedFacility['Facility']}: ${pro.project_application_shortname}]`; - } else { + @Input() voRegistrationLink: string = environment.voRegistrationLink; + + title: string = 'Projects Overview'; + filter: string; + + membersLoaded: boolean = false; + public memberFilter: string = ''; + filteredMembers: object[] = []; + selectedMember: object[] = []; + facility_members: object[] = []; + + filterChanged: Subject = new Subject(); + isLoaded: boolean = false; + projects: Application[] = []; + projectsCopy: Application[] = []; + show_openstack_projects: boolean = true; + show_simple_vm_projects: boolean = true; + details_loaded: boolean = false; + selectedEmailProjects: Application[] = []; + bsModalRef: BsModalRef; + userElixirSearchPI: boolean = true; + userElixirSearchAdmin: boolean = true; + userElixirSearchMember: boolean = true; + userElixirIdFilter: string; + + projectsLoaded: boolean = false; + + /** + * Approved group status. + * + * @type {number} + */ + STATUS_APPROVED: number = 2; + + selectedProjectType: string = 'ALL'; + + // modal variables for User list + public selectedProjectForSearch: Application; + public usersModalProjectMembers: ProjectMember[] = []; + allFacilityMembers: object[] = []; + public usersModalProjectID: number; + public usersModalProjectName: string; + public selectedProject: Application; + public userSearchValue: string; + + public emailSubject: string; + public emailText: string; + public emailStatus: number = 0; + public emailReply: string = ''; + public sendNews: boolean; + public alternative_emailText: string = ''; + public news_tags: string[] = []; + FILTER_DEBOUNCE_TIME: number = 500; + + public managerFacilities: [string, number][] = []; + public selectedFacility: [string, number]; + projects_filtered: Application[] = []; + facilitySupportMails: string = ''; + supportMailEditing: boolean = false; + PREDEFINED_TAGS: string[] = ['downtime', 'openstack', 'simplevm', 'maintenance', 'update']; + + @ViewChildren(NgbdSortableHeaderDirective) headers: QueryList; + + applictions$: Observable; + total$: Observable; + + constructor( + private groupService: GroupService, + private facilityService: FacilityService, + private newsService: NewsService, + public sortProjectService: ProjectSortService, + private modalService: BsModalService, + private emailService: EmailService, + ) { + super(); + } + + switchShowSimpleVmProjects(): void { + this.show_simple_vm_projects = !this.show_simple_vm_projects; + } + + switchOpenStackVmProjects(): void { + this.show_openstack_projects = !this.show_openstack_projects; + } + + setEmailSubject(): void { + switch (this.selectedProjectType) { + case 'ALL': this.emailSubject = `[${this.selectedFacility['Facility']}]`; - } - break; + break; + case 'OVP': + this.emailSubject = `[${this.selectedFacility['Facility']}: OpenStack]`; + break; + case 'SVP': + this.emailSubject = `[${this.selectedFacility['Facility']}: SimpleVm]`; + break; + case 'USER': + this.emailSubject = `[${this.selectedFacility['Facility']}: Specific Members]`; + break; + default: + // eslint-disable-next-line no-case-declarations + const pro: Application = this.projects.find( + (project: Application): boolean => project.project_application_perun_id.toString() === this.selectedProjectType.toString(), + ); + if (pro) { + this.emailSubject = `[${this.selectedFacility['Facility']}: ${pro.project_application_shortname}]`; + } else { + this.emailSubject = `[${this.selectedFacility['Facility']}]`; + } + break; + } } - } - ngOnInit(): void { - this.facilityService.getManagerFacilities().subscribe((result: any): void => { - this.managerFacilities = result; - this.selectedFacility = this.managerFacilities[0]; - this.emailSubject = `[${this.selectedFacility['Facility']}]`; - this.getFacilityProjects(this.managerFacilities[0]['FacilityId']); - this.title = `${this.title}:${this.selectedFacility['Facility']}`; - }); - this.sendNews = true; + ngOnInit(): void { + this.facilityService.getManagerFacilities().subscribe((result: any): void => { + this.managerFacilities = result; + this.selectedFacility = this.managerFacilities[0]; + this.emailSubject = `[${this.selectedFacility['Facility']}]`; + this.getFacilityProjects(this.managerFacilities[0]['FacilityId']); + this.title = `${this.title}:${this.selectedFacility['Facility']}`; + }); + this.sendNews = true; - /** needs refactoring in case we introduce tags to wagtail + /** needs refactoring in case we introduce tags to wagtail * this.newsService.getAvailableTagsFromWordPress().subscribe((tags: WordPressTag[]): void => { if (!(('code' in tags) && tags['code'] === 'wp-die')) { if (tags) { @@ -156,365 +163,383 @@ export class FacilityProjectsOverviewComponent extends AbstractBaseClass impleme } } }); * */ - } - - onCsvFileSelected(event): void { - const inputElement = event.target as HTMLInputElement; - if (inputElement.files && inputElement.files.length > 0) { - this.emailService.sendCsvTemplate(inputElement.files[0]).subscribe( - (csvTemplate: CsvMailTemplateModel) => { - this.openProjectMailsModal(inputElement.files[0], csvTemplate); - }, - (error: CsvMailTemplateModel) => { - console.log(error['error']); - this.openProjectMailsModal(inputElement.files[0], error['error']); - }, - ); } - } - onSort({ column, direction }: SortEvent) { - // resetting other headers - this.headers.forEach(header => { - if (header.appSortable !== column) { - header.direction = ''; - } - }); - - this.sortProjectService.sortColumn = column; - this.sortProjectService.sortDirection = direction; - } - - searchForUserInFacility(searchString: string): void { - this.facilityService.getFilteredMembersOfFacility(searchString); - } - - filterMembers(bare_searchString: string): void { - this.filteredMembers = []; - const searchString: string = bare_searchString.toLowerCase(); - - this.allFacilityMembers.forEach((member: object): void => { - if ( - member['elixirId'].toLowerCase().includes(searchString) - || member['email'].toLowerCase().includes(searchString) - || member['firstName'].toLowerCase().includes(searchString) - || member['lastName'].toLowerCase().includes(searchString) - ) { - this.filteredMembers.push(member); + onCsvFileSelected(event): void { + const inputElement = event.target as HTMLInputElement; + if (inputElement.files && inputElement.files.length > 0) { + this.emailService.sendCsvTemplate(inputElement.files[0]).subscribe( + (csvTemplate: CsvMailTemplateModel) => { + this.openProjectMailsModal(inputElement.files[0], csvTemplate); + }, + (error: CsvMailTemplateModel) => { + console.log(error['error']); + this.openProjectMailsModal(inputElement.files[0], error['error']); + }, + ); } - }); - } - - getProjectsByMemberElixirId(): void { - // tslint:disable-next-line:max-line-length - this.facilityService - .getFacilityGroupsByMemberElixirId(this.managerFacilities[0]['FacilityId'], this.filter) - .subscribe((applications: Application[]): void => { - this.projects_filtered = []; - for (const group of applications) { - if (group.project_application_lifetime > 0) { - group.lifetime_reached = this.lifeTimeReached(group.lifetime_days, group.DaysRunning); - } - this.projects_filtered.push(group); + } + + onSort({ column, direction }: SortEvent) { + // resetting other headers + this.headers.forEach(header => { + if (header.appSortable !== column) { + header.direction = ''; } }); - } - - /** - * Gets projects and sets email subject prefix when selected facility changes. - */ - onChangeSelectedFacility(): void { - this.isLoaded = false; - this.getFacilityProjects(this.selectedFacility['FacilityId']); - this.emailSubject = `[${this.selectedFacility['Facility']}]`; - } - - showMembersModal(application: Application): void { - const initialState = { - projectId: application.project_application_perun_id, - projectName: application.project_application_shortname, - facilityId: this.selectedFacility['FacilityId'], - }; - - this.bsModalRef = this.modalService.show(MembersListModalComponent, { initialState, class: 'modal-lg' }); - } - - getProjectLifetime(): void { - this.details_loaded = true; - } - - selectAllFilteredProjects(): void { - this.selectedEmailProjects = []; - - // get all the applications - this.applictions$.pipe(take(1)).subscribe(applications => { - // set the selected state of all projects to true - applications.forEach(application => { - application.is_project_selected = true; - this.toggleSelectedEmailApplication(application, application.is_project_selected); - }); - }); - } - - unselectAll(): void { - this.sortProjectService.applications.forEach((pr: Application) => { - pr.is_project_selected = false; - this.toggleSelectedEmailApplication(pr, pr.is_project_selected); - }); - // this.selectedEmailProjects = []; // clear the selectedEmailProjects list - } - - unselectAllFilteredProjects(): void { - // get all the applications - this.applictions$.pipe(take(1)).subscribe(applications => { - // set the selected state of all projects to false - applications.forEach(application => { - application.is_project_selected = false; - this.toggleSelectedEmailApplication(application, application.is_project_selected); - }); - }); - } - toggleSelectedEmailApplication(application: Application, isChecked: boolean): void { - const index = this.selectedEmailProjects.indexOf(application); + this.sortProjectService.sortColumn = column; + this.sortProjectService.sortDirection = direction; + } + + searchForUserInFacility(searchString: string): void { + this.facilityService.getFilteredMembersOfFacility(searchString); + } - if (isChecked) { - // checkbox was checked - if (index === -1) { - // application is not in the list, so add it - this.selectedEmailProjects.push(application); + filterMembers(bare_searchString: string): void { + this.filteredMembers = []; + const searchString: string = bare_searchString.toLowerCase(); + + this.allFacilityMembers.forEach((member: object): void => { + if ( + member['elixirId'].toLowerCase().includes(searchString) + || member['email'].toLowerCase().includes(searchString) + || member['firstName'].toLowerCase().includes(searchString) + || member['lastName'].toLowerCase().includes(searchString) + ) { + this.filteredMembers.push(member); + } + }); + } + + getProjectsByMemberElixirId(): void { + // tslint:disable-next-line:max-line-length + this.userElixirIdFilter = this.userElixirIdFilter.trim(); + if (this.userElixirIdFilter && this.userElixirIdFilter.includes('@elixir-europe.org')) { + this.projectsLoaded = false; + + this.facilityService + .getFacilityGroupsByMemberElixirId(this.selectedFacility['FacilityId'], this.userElixirIdFilter, this.userElixirSearchPI, this.userElixirSearchAdmin, this.userElixirSearchMember) + .subscribe((applications: Application[]): void => { + this.projects = applications; + for (const group of applications) { + if (group.project_application_lifetime > 0) { + group.lifetime_reached = this.lifeTimeReached(group.lifetime_days, group.DaysRunning); + } + + } + this.sortProjectService.applications = this.projects; + this.applictions$ = this.sortProjectService.applications$; + this.total$ = this.sortProjectService.total$; + this.projectsLoaded = true; + + }); + } else { + this.sortProjectService.applications = this.projectsCopy; + this.applictions$ = this.sortProjectService.applications$; + this.total$ = this.sortProjectService.total$; + this.projectsLoaded = true; } - } else { - // checkbox was unchecked - // application is in the list, so remove it - this.selectedEmailProjects.splice(index, 1); } - } - - /** - * Returns the name of the project with the id of the selectedProjectType - */ - getProjectNameBySelectedProjectTypeAsId(): string { - const id: string = this.selectedProjectType; - if (!id) { - return 'NOT_FOUND'; + + /** + * Gets projects and sets email subject prefix when selected facility changes. + */ + onChangeSelectedFacility(): void { + this.isLoaded = false; + this.getFacilityProjects(this.selectedFacility['FacilityId']); + this.emailSubject = `[${this.selectedFacility['Facility']}]`; } - const project: Application = this.projects.find( - (element: Application): boolean => element.project_application_perun_id.toString() === id.toString(), - ); - if (project) { - return project.project_application_shortname; + + showMembersModal(application: Application): void { + const initialState = { + projectId: application.project_application_perun_id, + projectName: application.project_application_shortname, + facilityId: this.selectedFacility['FacilityId'], + }; + + this.bsModalRef = this.modalService.show(MembersListModalComponent, { initialState, class: 'modal-lg' }); } - return 'NOT_FOUND'; - } + getProjectLifetime(): void { + this.details_loaded = true; + } - getFacilityProjects(facility: string): void { - this.projects = []; + selectAllFilteredProjects(): void { + this.selectedEmailProjects = []; - // tslint:disable-next-line:max-line-length - this.facilityService - .getFacilityAllowedGroupsWithDetailsAndSpecificStatus(facility, this.STATUS_APPROVED) - .subscribe((applications: Application[]): void => { - for (const group of applications) { - if (group.project_application_lifetime > 0) { - group.lifetime_reached = this.lifeTimeReached(group.lifetime_days, group.DaysRunning); - } - this.projects.push(group); - } - this.sortProjectService.applications = this.projects; - this.applictions$ = this.sortProjectService.applications$; - this.total$ = this.sortProjectService.total$; + // get all the applications + this.applictions$.pipe(take(1)).subscribe(applications => { + // set the selected state of all projects to true + applications.forEach(application => { + application.is_project_selected = true; + this.toggleSelectedEmailApplication(application, application.is_project_selected); + }); + }); + } - this.isLoaded = true; + unselectAll(): void { + this.sortProjectService.applications.forEach((pr: Application) => { + pr.is_project_selected = false; + this.toggleSelectedEmailApplication(pr, pr.is_project_selected); }); - this.facilityService.getAllMembersOfFacility(facility, this.STATUS_APPROVED).subscribe( - (result: any[]): void => { - this.membersLoaded = true; - this.allFacilityMembers = result; - }, - (error: any): void => { - console.log(error); - this.membersLoaded = false; - }, - ); - } - - /** - * Adds or deletes tags from the list of tags to add to the news when the corresponding checkbox gets clicked. - * - * @param tag the tag which gets added/deleted. - */ - - /** - * Sends an email to users and also posts it as a news in WordPress via newsManager if selected. - * - * @param facility the facility of the users which shall be informed - * @param subject the subject as a string - * @param message the message as a string - * @param reply the reply-address - * @param send boolean if it should be sent to WordPress - * @param alternative_news_text the news text for WordPress, in case it shall be different from the original text - * @param selectedMember the specific member the mail is sent to in case one specific member is chosen - */ - sendMailToFacility( - facility: string, - subject: string, - message: string, - reply?: string, - send?: any, - alternative_news_text?: string, - ): void { - this.emailStatus = 0; - if (this.selectedProjectType === 'USER') { - const tempMailList: string[] = []; - // tslint:disable-next-line:no-for-each-push - this.selectedMember.forEach((member: object): void => { - tempMailList.push(member['email']); + // this.selectedEmailProjects = []; // clear the selectedEmailProjects list + } + + unselectAllFilteredProjects(): void { + // get all the applications + this.applictions$.pipe(take(1)).subscribe(applications => { + // set the selected state of all projects to false + applications.forEach(application => { + application.is_project_selected = false; + this.toggleSelectedEmailApplication(application, application.is_project_selected); + }); }); - this.selectedProjectType = tempMailList.join(','); } - if (reply) { - reply = reply.trim(); + + toggleSelectedEmailApplication(application: Application, isChecked: boolean): void { + const index = this.selectedEmailProjects.indexOf(application); + + if (isChecked) { + // checkbox was checked + if (index === -1) { + // application is not in the list, so add it + this.selectedEmailProjects.push(application); + } + } else { + // checkbox was unchecked + // application is in the list, so remove it + this.selectedEmailProjects.splice(index, 1); + } } - this.facilityService - .sendMailToFacility( - facility, - encodeURIComponent(subject), - encodeURIComponent(message), - this.selectedProjectType, - encodeURIComponent(reply), - send, - encodeURIComponent(alternative_news_text), - this.news_tags.join(), - ) - .subscribe( - (result: any): void => { - if (result.status === 201) { - this.emailStatus = 1; - } else { - this.emailStatus = 2; + /** + * Returns the name of the project with the id of the selectedProjectType + */ + getProjectNameBySelectedProjectTypeAsId(): string { + const id: string = this.selectedProjectType; + if (!id) { + return 'NOT_FOUND'; + } + const project: Application = this.projects.find( + (element: Application): boolean => element.project_application_perun_id.toString() === id.toString(), + ); + if (project) { + return project.project_application_shortname; + } + + return 'NOT_FOUND'; + } + + getFacilityProjects(facility: string): void { + this.projects = []; + this.projectsLoaded = false; + + // tslint:disable-next-line:max-line-length + this.facilityService + .getFacilityAllowedGroupsWithDetailsAndSpecificStatus(facility, this.STATUS_APPROVED) + .subscribe((applications: Application[]): void => { + for (const group of applications) { + if (group.project_application_lifetime > 0) { + group.lifetime_reached = this.lifeTimeReached(group.lifetime_days, group.DaysRunning); + } + this.projects.push(group); } + this.projectsCopy = this.projects; + this.sortProjectService.applications = this.projects; + this.applictions$ = this.sortProjectService.applications$; + this.total$ = this.sortProjectService.total$; + this.projectsLoaded = true; + + this.isLoaded = true; + }); + this.facilityService.getAllMembersOfFacility(facility, this.STATUS_APPROVED).subscribe( + (result: any[]): void => { + this.membersLoaded = true; + this.allFacilityMembers = result; }, - (): void => { - this.emailStatus = 2; - }, - (): void => { - this.filteredMembers = []; - this.selectedProjectType = 'ALL'; - this.emailReply = ''; - this.selectedMember = []; - this.memberFilter = ''; + (error: any): void => { + console.log(error); + this.membersLoaded = false; }, ); - } + } - /** - * Sets the member selected in the mail modal as the member to send the mail to. - * - * @param member the selected member - */ + /** + * Adds or deletes tags from the list of tags to add to the news when the corresponding checkbox gets clicked. + * + * @param tag the tag which gets added/deleted. + */ + + /** + * Sends an email to users and also posts it as a news in WordPress via newsManager if selected. + * + * @param facility the facility of the users which shall be informed + * @param subject the subject as a string + * @param message the message as a string + * @param reply the reply-address + * @param send boolean if it should be sent to WordPress + * @param alternative_news_text the news text for WordPress, in case it shall be different from the original text + * @param selectedMember the specific member the mail is sent to in case one specific member is chosen + */ + sendMailToFacility( + facility: string, + subject: string, + message: string, + reply?: string, + send?: any, + alternative_news_text?: string, + ): void { + this.emailStatus = 0; + if (this.selectedProjectType === 'USER') { + const tempMailList: string[] = []; + // tslint:disable-next-line:no-for-each-push + this.selectedMember.forEach((member: object): void => { + tempMailList.push(member['email']); + }); + this.selectedProjectType = tempMailList.join(','); + } + if (reply) { + reply = reply.trim(); + } - setSelectedUserForMail(member: object): void { - if (!this.selectedMember.includes(member)) { - this.selectedMember.push(member); + this.facilityService + .sendMailToFacility( + facility, + encodeURIComponent(subject), + encodeURIComponent(message), + this.selectedProjectType, + encodeURIComponent(reply), + send, + encodeURIComponent(alternative_news_text), + this.news_tags.join(), + ) + .subscribe( + (result: any): void => { + if (result.status === 201) { + this.emailStatus = 1; + } else { + this.emailStatus = 2; + } + }, + (): void => { + this.emailStatus = 2; + }, + (): void => { + this.filteredMembers = []; + this.selectedProjectType = 'ALL'; + this.emailReply = ''; + this.selectedMember = []; + this.memberFilter = ''; + }, + ); } - } - removeSelectedUserForMail(member: object): void { - const index: number = this.selectedMember.indexOf(member); - if (index > -1) { - this.selectedMember.splice(index, 1); + /** + * Sets the member selected in the mail modal as the member to send the mail to. + * + * @param member the selected member + */ + + setSelectedUserForMail(member: object): void { + if (!this.selectedMember.includes(member)) { + this.selectedMember.push(member); + } } - } - - getMembersOfTheProject(projectid: number, projectname: string): void { - this.facilityService - .getFacilityGroupRichMembers(projectid, this.selectedFacility['FacilityId']) - .subscribe((members: ProjectMember[]): void => { - this.usersModalProjectID = projectid; - this.usersModalProjectName = projectname; - this.usersModalProjectMembers = members; - }); - } - - public resetEmailModal(): void { - this.selectedProjectType = 'ALL'; - this.emailSubject = `[${this.selectedFacility['Facility']}]`; - this.emailText = null; - this.emailReply = null; - this.emailStatus = 0; - this.sendNews = true; - this.alternative_emailText = ''; - this.news_tags = []; - this.selectedMember = []; - } - - getFacilitySupportMails(): void { - this.facilityService.getSupportMails(this.selectedFacility['FacilityId']).subscribe((result: any) => { - this.facilitySupportMails = result['body']; - if (this.facilitySupportMails === '' || this.facilitySupportMails === null) { - this.facilitySupportMails = 'example@mail1.com, example@mail2.com'; + + removeSelectedUserForMail(member: object): void { + const index: number = this.selectedMember.indexOf(member); + if (index > -1) { + this.selectedMember.splice(index, 1); } - }); - } + } - openProjectMailsModal(csvFile: File = null, csvTemplate: CsvMailTemplateModel = null): void { - let initialState = {}; + getMembersOfTheProject(projectid: number, projectname: string): void { + this.facilityService + .getFacilityGroupRichMembers(projectid, this.selectedFacility['FacilityId']) + .subscribe((members: ProjectMember[]): void => { + this.usersModalProjectID = projectid; + this.usersModalProjectName = projectname; + this.usersModalProjectMembers = members; + }); + } - if (csvFile) { - initialState = { - selectedProjects: csvTemplate.valid_projects, - csvFile, - csvMailTemplate: csvTemplate, - }; - } else { - initialState = { selectedProjects: this.selectedEmailProjects }; + public resetEmailModal(): void { + this.selectedProjectType = 'ALL'; + this.emailSubject = `[${this.selectedFacility['Facility']}]`; + this.emailText = null; + this.emailReply = null; + this.emailStatus = 0; + this.sendNews = true; + this.alternative_emailText = ''; + this.news_tags = []; + this.selectedMember = []; } - this.bsModalRef = this.modalService.show(ProjectEmailModalComponent, { initialState, class: 'modal-lg' }); - this.bsModalRef.content.event.subscribe((sent_successfully: boolean) => { - if (sent_successfully) { - const initialStateNotification = { - notificationModalTitle: 'Success', - notificationModalType: 'success', - notificationModalMessage: 'Mails were successfully sent', - }; - this.modalService.show(NotificationModalComponent, { initialState: initialStateNotification }); - } else { - const initialStateNotification = { - notificationModalTitle: 'Failed', - notificationModalType: 'danger', - notificationModalMessage: 'Failed to send mails!', + getFacilitySupportMails(): void { + this.facilityService.getSupportMails(this.selectedFacility['FacilityId']).subscribe((result: any) => { + this.facilitySupportMails = result['body']; + if (this.facilitySupportMails === '' || this.facilitySupportMails === null) { + this.facilitySupportMails = 'example@mail1.com, example@mail2.com'; + } + }); + } + + openProjectMailsModal(csvFile: File = null, csvTemplate: CsvMailTemplateModel = null): void { + let initialState = {}; + + if (csvFile) { + initialState = { + selectedProjects: csvTemplate.valid_projects, + csvFile, + csvMailTemplate: csvTemplate, }; - this.modalService.show(NotificationModalComponent, { initialState: initialStateNotification }); - } - }); - } - - setFacilitySupportMails(supportMails: string): void { - const facilityId = this.selectedFacility['FacilityId']; - this.facilityService.setSupportMails(facilityId, supportMails).subscribe((result: any): void => { - if (result.ok) { - this.updateNotificationModal( - 'Facility support mails changed', - 'You successfully changed the facility support mails.', - true, - 'success', - ); } else { - this.updateNotificationModal( - "Couldn't change facility support mails", - 'An error occurred while trying to change the facility support mails.', - true, - 'danger', - ); + initialState = { selectedProjects: this.selectedEmailProjects }; } - }); - } + this.bsModalRef = this.modalService.show(ProjectEmailModalComponent, { initialState, class: 'modal-lg' }); + this.bsModalRef.content.event.subscribe((sent_successfully: boolean) => { + if (sent_successfully) { + const initialStateNotification = { + notificationModalTitle: 'Success', + notificationModalType: 'success', + notificationModalMessage: 'Mails were successfully sent', + }; + + this.modalService.show(NotificationModalComponent, { initialState: initialStateNotification }); + } else { + const initialStateNotification = { + notificationModalTitle: 'Failed', + notificationModalType: 'danger', + notificationModalMessage: 'Failed to send mails!', + }; + this.modalService.show(NotificationModalComponent, { initialState: initialStateNotification }); + } + }); + } - toggleSupportMailEditing(): void { - this.supportMailEditing = !this.supportMailEditing; - } + setFacilitySupportMails(supportMails: string): void { + const facilityId = this.selectedFacility['FacilityId']; + this.facilityService.setSupportMails(facilityId, supportMails).subscribe((result: any): void => { + if (result.ok) { + this.updateNotificationModal( + 'Facility support mails changed', + 'You successfully changed the facility support mails.', + true, + 'success', + ); + } else { + this.updateNotificationModal( + 'Couldn\'t change facility support mails', + 'An error occurred while trying to change the facility support mails.', + true, + 'danger', + ); + } + }); + } + + toggleSupportMailEditing(): void { + this.supportMailEditing = !this.supportMailEditing; + } } diff --git a/src/app/vo_manager/VoOverviewComponent.ts b/src/app/vo_manager/VoOverviewComponent.ts index ba38963185..b3d73c0a50 100644 --- a/src/app/vo_manager/VoOverviewComponent.ts +++ b/src/app/vo_manager/VoOverviewComponent.ts @@ -41,621 +41,653 @@ export class VoOverviewComponent extends AbstractBaseClass implements OnInit, On public emailText: string; public emailStatus: number = 0; - @ViewChild('notificationModal') notificationModal: ModalDirective; - public emailHeader: string; - public emailVerify: string; - public emailType: number; - public emailAdminsOnly: boolean = false; - public expiredTemplated: boolean = false; - - public removalDate: Date = new Date(); - public selectedProject: Application; - selectedEmailProjects: Application[] = []; - computecenters: ComputecenterComponent[] = []; - bsModalRef: BsModalRef; - subscription: Subscription = new Subscription(); - protected readonly ConfirmationActions = ConfirmationActions; - - show_openstack_projects: boolean = true; - show_simple_vm_projects: boolean = true; - show_simple_vm: boolean = true; - show_openstack: boolean = true; - - tsvTaskRunning: boolean = true; - numberOfTsvs: number = 0; - checkTSVTimer: ReturnType; - checkTSVTimeout: number = 10000; - - selectedProjectType: string = 'ALL'; - selectedFacility: string | number = 'ALL'; - - public newsletterSubscriptionCounter: number; - isLoaded: boolean = false; - member_id: number; - projects: Application[] = []; - projects_filtered: Application[] = []; - - // modal variables for User list - public usersModalProjectMembers: ProjectMember[] = []; - public usersModalProjectID: number; - public usersModalProjectName: string; - public managerFacilities: [string, number][]; - - projectMailTemplates: string[] = []; - @ViewChildren(NgbdSortableHeaderDirective) headers: QueryList; - - applictions$: Observable; - total$: Observable; - - // public selectedFacility: [string, number]; - - constructor( - private fullLayout: FullLayoutComponent, - private sanitizer: DomSanitizer, - private voService: VoService, - private groupservice: GroupService, - private facilityService: FacilityService, - public sortProjectService: ProjectSortService, - private modalService: BsModalService, - private emailService: EmailService, - ) { - super(); - } - - ngOnInit(): void { - this.getVoProjects(); - this.getComputeCenters(); - this.voService.getNewsletterSubscriptionCounter().subscribe((result: IResponseTemplate): void => { - this.newsletterSubscriptionCounter = result.value as number; - }); - this.getTSVInformation(); - } - - ngOnDestroy() { - this.subscription.unsubscribe(); - } - - onCsvFileSelected(event): void { - const inputElement = event.target as HTMLInputElement; - if (inputElement.files && inputElement.files.length > 0) { - this.emailService.sendCsvTemplate(inputElement.files[0]).subscribe( - (csvTemplate: CsvMailTemplateModel) => { - this.openProjectMailsModal(inputElement.files[0], csvTemplate); - }, - (error: CsvMailTemplateModel) => { - console.log(error['error']); - this.openProjectMailsModal(inputElement.files[0], error['error']); - }, - ); + @ViewChild('notificationModal') notificationModal: ModalDirective; + public emailHeader: string; + public emailVerify: string; + public emailType: number; + public emailAdminsOnly: boolean = false; + public expiredTemplated: boolean = false; + + public removalDate: Date = new Date(); + public selectedProject: Application; + selectedEmailProjects: Application[] = []; + computecenters: ComputecenterComponent[] = []; + bsModalRef: BsModalRef; + subscription: Subscription = new Subscription(); + protected readonly ConfirmationActions = ConfirmationActions; + userElixirSearchPI: boolean = true; + userElixirSearchAdmin: boolean = true; + userElixirSearchMember: boolean = true; + projectsLoaded: boolean = false; + show_openstack_projects: boolean = true; + show_simple_vm_projects: boolean = true; + show_simple_vm: boolean = true; + show_openstack: boolean = true; + + tsvTaskRunning: boolean = false; + numberOfTsvs: number = 0; + checkTSVTimer: ReturnType; + checkTSVTimeout: number = 10000; + projectsCopy: Application[] = []; + + selectedProjectType: string = 'ALL'; + selectedFacility: string | number = 'ALL'; + userElixirIdFilter: string; + + public newsletterSubscriptionCounter: number; + member_id: number; + projects: Application[] = []; + + // modal variables for User list + public usersModalProjectMembers: ProjectMember[] = []; + public usersModalProjectID: number; + public usersModalProjectName: string; + public managerFacilities: [string, number][]; + + projectMailTemplates: string[] = []; + @ViewChildren(NgbdSortableHeaderDirective) headers: QueryList; + + applictions$: Observable; + total$: Observable; + + // public selectedFacility: [string, number]; + + constructor( + private fullLayout: FullLayoutComponent, + private sanitizer: DomSanitizer, + private voService: VoService, + private groupservice: GroupService, + private facilityService: FacilityService, + public sortProjectService: ProjectSortService, + private modalService: BsModalService, + private emailService: EmailService, + ) { + super(); } - } - getTSVInformation(timeout: number = this.checkTSVTimeout): void { - this.stopCheckTSVTimer(); - this.subscription.add( - this.voService.getTsvInformation().subscribe( - (result: any): void => { - this.tsvTaskRunning = result[0]; - this.numberOfTsvs = result[1]; - if (result[0] !== true) { - this.stopCheckTSVTimer(); - } else { + ngOnInit(): void { + this.getVoProjects(); + this.getComputeCenters(); + this.voService.getNewsletterSubscriptionCounter().subscribe((result: IResponseTemplate): void => { + this.newsletterSubscriptionCounter = result.value as number; + }); + this.getTSVInformation(); + } + + ngOnDestroy() { + this.subscription.unsubscribe(); + } + + onCsvFileSelected(event): void { + const inputElement = event.target as HTMLInputElement; + if (inputElement.files && inputElement.files.length > 0) { + this.emailService.sendCsvTemplate(inputElement.files[0]).subscribe( + (csvTemplate: CsvMailTemplateModel) => { + this.openProjectMailsModal(inputElement.files[0], csvTemplate); + }, + (error: CsvMailTemplateModel) => { + console.log(error['error']); + this.openProjectMailsModal(inputElement.files[0], error['error']); + }, + ); + } + } + + getTSVInformation(timeout: number = this.checkTSVTimeout): void { + this.stopCheckTSVTimer(); + this.subscription.add( + this.voService.getTsvInformation().subscribe( + (result: any): void => { + this.tsvTaskRunning = result[0]; + this.numberOfTsvs = result[1]; + if (result[0] !== true) { + this.stopCheckTSVTimer(); + } else { + this.checkTSVTimer = setTimeout((): void => { + this.getTSVInformation(); + }, timeout); + } + }, + () => { + this.tsvTaskRunning = true; + this.numberOfTsvs = 0; this.checkTSVTimer = setTimeout((): void => { this.getTSVInformation(); }, timeout); - } - }, - () => { - this.tsvTaskRunning = true; - this.numberOfTsvs = 0; - this.checkTSVTimer = setTimeout((): void => { - this.getTSVInformation(); - }, timeout); - }, - ), - ); - } + }, + ), + ); + } - stopCheckTSVTimer(): void { - if (this.checkTSVTimer) { - clearTimeout(this.checkTSVTimer); + stopCheckTSVTimer(): void { + if (this.checkTSVTimer) { + clearTimeout(this.checkTSVTimer); + } } - } - selectAllFilteredProjects(): void { - this.selectedEmailProjects = []; + selectAllFilteredProjects(): void { + this.selectedEmailProjects = []; - // get all the applications - this.applictions$.pipe(take(1)).subscribe(applications => { - // set the selected state of all projects to true - applications.forEach(application => { - application.is_project_selected = true; - this.toggleSelectedEmailApplication(application, application.is_project_selected); + // get all the applications + this.applictions$.pipe(take(1)).subscribe(applications => { + // set the selected state of all projects to true + applications.forEach(application => { + application.is_project_selected = true; + this.toggleSelectedEmailApplication(application, application.is_project_selected); + }); }); - }); - } - - showConfirmationModal(application: Application, action: ConfirmationActions): void { - const initialState = { application, action }; - console.log(initialState); - - this.bsModalRef = this.modalService.show(ConfirmationModalComponent, { initialState, class: 'modal-lg' }); - this.subscribeToBsModalRef(); - } - - showMembersModal(application: Application): void { - const initialState = { - projectId: application.project_application_perun_id, - projectName: application.project_application_shortname, - }; - - this.bsModalRef = this.modalService.show(MembersListModalComponent, { initialState, class: 'modal-lg' }); - } - - subscribeToBsModalRef(): void { - this.subscription.add( - this.bsModalRef.content.event.subscribe((result: any) => { - let action = null; - if ('action' in result) { - action = result['action']; - } + } - if (ConfirmationActions.ENABLE_APPLICATION === action) { - this.enableProject(result['application']); - } - if (ConfirmationActions.DISABLE_APPLICATION === action) { - this.disableProject(result['application']); - } - }), - ); - } - - unselectAll(): void { - this.sortProjectService.applications.forEach((pr: Application) => { - pr.is_project_selected = false; - this.toggleSelectedEmailApplication(pr, pr.is_project_selected); - }); - // this.selectedEmailProjects = []; // clear the selectedEmailProjects list - } - - unselectAllFilteredProjects(): void { - // get all the applications - this.applictions$.pipe(take(1)).subscribe(applications => { - // set the selected state of all projects to false - applications.forEach(application => { - application.is_project_selected = false; - this.toggleSelectedEmailApplication(application, application.is_project_selected); - }); - }); - } + showConfirmationModal(application: Application, action: ConfirmationActions): void { + const initialState = { application, action }; + console.log(initialState); - toggleSelectedEmailApplication(application: Application, isChecked: boolean): void { - const index = this.selectedEmailProjects.indexOf(application); + this.bsModalRef = this.modalService.show(ConfirmationModalComponent, { initialState, class: 'modal-lg' }); + this.subscribeToBsModalRef(); + } - if (isChecked) { - // checkbox was checked - if (index === -1) { - // application is not in the list, so add it - this.selectedEmailProjects.push(application); - } - } else { - // checkbox was unchecked - // application is in the list, so remove it - this.selectedEmailProjects.splice(index, 1); + showMembersModal(application: Application): void { + const initialState = { + projectId: application.project_application_perun_id, + projectName: application.project_application_shortname, + }; + + this.bsModalRef = this.modalService.show(MembersListModalComponent, { initialState, class: 'modal-lg' }); } - } - openProjectMailsModal(csvFile: File = null, csvTemplate: CsvMailTemplateModel = null): void { - let initialState = {}; + subscribeToBsModalRef(): void { + this.subscription.add( + this.bsModalRef.content.event.subscribe((result: any) => { + let action = null; + if ('action' in result) { + action = result['action']; + } - if (csvFile) { - initialState = { - selectedProjects: csvTemplate.valid_projects, - csvFile, - csvMailTemplate: csvTemplate, - }; - } else { - initialState = { selectedProjects: this.selectedEmailProjects }; + if (ConfirmationActions.ENABLE_APPLICATION === action) { + this.enableProject(result['application']); + } + if (ConfirmationActions.DISABLE_APPLICATION === action) { + this.disableProject(result['application']); + } + }), + ); } - this.bsModalRef = this.modalService.show(ProjectEmailModalComponent, { initialState, class: 'modal-lg' }); - this.bsModalRef.content.event.subscribe((sent_successfully: boolean) => { - if (sent_successfully) { - this.updateNotificationModal('Success', 'Mails were successfully sent', true, 'success'); + unselectAll(): void { + this.sortProjectService.applications.forEach((pr: Application) => { + pr.is_project_selected = false; + this.toggleSelectedEmailApplication(pr, pr.is_project_selected); + }); + // this.selectedEmailProjects = []; // clear the selectedEmailProjects list + } + + unselectAllFilteredProjects(): void { + // get all the applications + this.applictions$.pipe(take(1)).subscribe(applications => { + // set the selected state of all projects to false + applications.forEach(application => { + application.is_project_selected = false; + this.toggleSelectedEmailApplication(application, application.is_project_selected); + }); + }); + } + + toggleSelectedEmailApplication(application: Application, isChecked: boolean): void { + const index = this.selectedEmailProjects.indexOf(application); + + if (isChecked) { + // checkbox was checked + if (index === -1) { + // application is not in the list, so add it + this.selectedEmailProjects.push(application); + } } else { - this.updateNotificationModal('Failed', 'Failed to send mails!', true, 'danger'); + // checkbox was unchecked + // application is in the list, so remove it + this.selectedEmailProjects.splice(index, 1); } - this.notificationModal.show(); - }); - } - - disableProject(project: Application): void { - this.voService.setDisabledProject(project.project_application_perun_id).subscribe((upd_app: Application) => { - const idx = this.projects.indexOf(project); - this.projects[idx] = upd_app; - this.sortProjectService.applications = this.projects; - }); - } - - enableProject(project: Application): void { - this.voService.unsetDisabledProject(project.project_application_perun_id).subscribe((upd_app: Application) => { - const idx = this.projects.indexOf(project); - this.projects[idx] = upd_app; - this.sortProjectService.applications = this.projects; - }); - } - - onSort({ column, direction }: SortEvent) { - // resetting other headers - this.headers.forEach(header => { - if (header.appSortable !== column) { - header.direction = ''; - } - }); - - this.sortProjectService.sortColumn = column; - this.sortProjectService.sortDirection = direction; - } - - getApplicationInfos(): void { - this.voService.getVoProjectResourcesTimeframes().subscribe(); - - this.voService.getVoProjectCounter().subscribe(); - this.voService.getVoProjectDates().subscribe(); - } - - sendEmail(subject: string, message: string, reply?: string): void { - if (reply) { - reply = reply.trim(); - } - switch (this.emailType) { - case 0: { - this.sendMailToVo( - subject, - message, - this.selectedFacility.toString(), - this.selectedProjectType, - this.emailAdminsOnly, - this.expiredTemplated, - this.removalDate, - reply, - ); - break; - } - case 1: { - this.sendNewsletterToVo(subject, message, this.selectedProjectType, this.emailAdminsOnly, reply); - break; + } + + openProjectMailsModal(csvFile: File = null, csvTemplate: CsvMailTemplateModel = null): void { + let initialState = {}; + + if (csvFile) { + initialState = { + selectedProjects: csvTemplate.valid_projects, + csvFile, + csvMailTemplate: csvTemplate, + }; + } else { + initialState = { selectedProjects: this.selectedEmailProjects }; } - default: - } - } - - sendTestBug(): void { - this.voService.sendTestError().subscribe(); - } - - sendNewsletterToVo( - subject: string, - message: string, - selectedProjectType: string, - adminsOnly: boolean, - reply?: string, - ): void { - this.voService - .sendNewsletterToVo( - encodeURIComponent(subject), - encodeURIComponent(message), - selectedProjectType, - adminsOnly, - encodeURIComponent(reply), - ) - .subscribe((result: IResponseTemplate): void => { - if ((result.value as boolean) === true) { - this.emailStatus = 1; + + this.bsModalRef = this.modalService.show(ProjectEmailModalComponent, { initialState, class: 'modal-lg' }); + this.bsModalRef.content.event.subscribe((sent_successfully: boolean) => { + if (sent_successfully) { + this.updateNotificationModal('Success', 'Mails were successfully sent', true, 'success'); } else { - this.emailStatus = 2; + this.updateNotificationModal('Failed', 'Failed to send mails!', true, 'danger'); } + this.notificationModal.show(); }); - } - - sendMailToVo( - subject: string, - message: string, - facility: string, - type: string, - adminsOnly: boolean, - expiredTemplate: boolean, - removalDate: Date, - reply?: string, - ): void { - this.voService - .sendMailToVo( - encodeURIComponent(subject), - encodeURIComponent(message), - facility, - type, - adminsOnly, - expiredTemplate, - removalDate, - encodeURIComponent(reply), - ) - .subscribe((result: IResponseTemplate): void => { - if ((result.value as boolean) === true) { - this.emailStatus = 1; - } else { - this.emailStatus = 2; + } + + disableProject(project: Application): void { + this.voService.setDisabledProject(project.project_application_perun_id).subscribe((upd_app: Application) => { + const idx = this.projects.indexOf(project); + this.projects[idx] = upd_app; + this.sortProjectService.applications = this.projects; + }); + } + + getProjectsByMemberElixirId(): void { + // tslint:disable-next-line:max-line-length + this.userElixirIdFilter = this.userElixirIdFilter.trim(); + if (this.userElixirIdFilter && this.userElixirIdFilter.includes('@elixir-europe.org')) { + this.projectsLoaded = false; + + this.voService + .getGroupsByMemberElixirId(this.userElixirIdFilter, this.userElixirSearchPI, this.userElixirSearchAdmin, this.userElixirSearchMember) + .subscribe((applications: Application[]): void => { + this.projects = applications; + for (const group of applications) { + if (group.project_application_lifetime > 0) { + group.lifetime_reached = this.lifeTimeReached(group.lifetime_days, group.DaysRunning); + } + + } + this.sortProjectService.applications = this.projects; + this.applictions$ = this.sortProjectService.applications$; + this.total$ = this.sortProjectService.total$; + this.projectsLoaded = true; + + }); + } else { + this.sortProjectService.applications = this.projectsCopy; + this.applictions$ = this.sortProjectService.applications$; + this.total$ = this.sortProjectService.total$; + this.projectsLoaded = true; + } + } + + enableProject(project: Application): void { + this.voService.unsetDisabledProject(project.project_application_perun_id).subscribe((upd_app: Application) => { + const idx = this.projects.indexOf(project); + this.projects[idx] = upd_app; + this.sortProjectService.applications = this.projects; + }); + } + + onSort({ column, direction }: SortEvent) { + // resetting other headers + this.headers.forEach(header => { + if (header.appSortable !== column) { + header.direction = ''; } - this.selectedProjectType = 'ALL'; - this.selectedFacility = 'ALL'; }); - } - - dayChanged(date: { year: number; month: number; day: number }): void { - this.removalDate.setDate(date.day); - this.removalDate.setMonth(date.month - 1); - this.removalDate.setFullYear(date.year); - } - - setEmailType(type: number): void { - this.emailType = type; - switch (this.emailType) { - case 0: { - this.emailHeader = 'Send email to selected members of the VO'; - break; + + this.sortProjectService.sortColumn = column; + this.sortProjectService.sortDirection = direction; + } + + getApplicationInfos(): void { + this.voService.getVoProjectResourcesTimeframes().subscribe(); + + this.voService.getVoProjectCounter().subscribe(); + this.voService.getVoProjectDates().subscribe(); + } + + sendEmail(subject: string, message: string, reply?: string): void { + if (reply) { + reply = reply.trim(); } - case 1: { - this.emailHeader = 'Send newsletter to VO'; - break; + switch (this.emailType) { + case 0: { + this.sendMailToVo( + subject, + message, + this.selectedFacility.toString(), + this.selectedProjectType, + this.emailAdminsOnly, + this.expiredTemplated, + this.removalDate, + reply, + ); + break; + } + case 1: { + this.sendNewsletterToVo(subject, message, this.selectedProjectType, this.emailAdminsOnly, reply); + break; + } + default: } - default: } - this.emailVerify = 'Are you sure you want to send this newsletter to all members of the de.NBI VO?'; - } - getFacilityName(): string { - if (this.selectedFacility === 'ALL') { - return 'of the de.NBI VO'; - } else { - const temp_cc = this.computecenters.find(cc => cc.FacilityId === this.selectedFacility); - if (temp_cc === undefined) { + sendTestBug(): void { + this.voService.sendTestError().subscribe(); + } + + sendNewsletterToVo( + subject: string, + message: string, + selectedProjectType: string, + adminsOnly: boolean, + reply?: string, + ): void { + this.voService + .sendNewsletterToVo( + encodeURIComponent(subject), + encodeURIComponent(message), + selectedProjectType, + adminsOnly, + encodeURIComponent(reply), + ) + .subscribe((result: IResponseTemplate): void => { + if ((result.value as boolean) === true) { + this.emailStatus = 1; + } else { + this.emailStatus = 2; + } + }); + } + + sendMailToVo( + subject: string, + message: string, + facility: string, + type: string, + adminsOnly: boolean, + expiredTemplate: boolean, + removalDate: Date, + reply?: string, + ): void { + this.voService + .sendMailToVo( + encodeURIComponent(subject), + encodeURIComponent(message), + facility, + type, + adminsOnly, + expiredTemplate, + removalDate, + encodeURIComponent(reply), + ) + .subscribe((result: IResponseTemplate): void => { + if ((result.value as boolean) === true) { + this.emailStatus = 1; + } else { + this.emailStatus = 2; + } + this.selectedProjectType = 'ALL'; + this.selectedFacility = 'ALL'; + }); + } + + dayChanged(date: { year: number; month: number; day: number }): void { + this.removalDate.setDate(date.day); + this.removalDate.setMonth(date.month - 1); + this.removalDate.setFullYear(date.year); + } + + setEmailType(type: number): void { + this.emailType = type; + switch (this.emailType) { + case 0: { + this.emailHeader = 'Send email to selected members of the VO'; + break; + } + case 1: { + this.emailHeader = 'Send newsletter to VO'; + break; + } + default: + } + this.emailVerify = 'Are you sure you want to send this newsletter to all members of the de.NBI VO?'; + } + + getFacilityName(): string { + if (this.selectedFacility === 'ALL') { return 'of the de.NBI VO'; } else { - return `of the facility "${temp_cc.Name}"`; + const temp_cc = this.computecenters.find(cc => cc.FacilityId === this.selectedFacility); + if (temp_cc === undefined) { + return 'of the de.NBI VO'; + } else { + return `of the facility "${temp_cc.Name}"`; + } } } - } - - getMailConfinementByProjectType(): string { - switch (this.selectedProjectType) { - case 'ALL_GM': - return 'of all active projects'; - case 'EXP': - return 'of all expired projects'; - case 'SVP': - return 'of all SimpleVM projects'; - case 'OVP': - return 'of all OpenStack projects'; - case 'WSH': - return 'of all Workshops'; - default: - return ''; - } - } - - adjustVerifyText(): void { - switch (this.emailType) { - case 0: { - this.emailVerify = `Are you sure you want to send this email to all ${ - this.emailAdminsOnly ? ' group administrators' : 'members' - } ${this.getMailConfinementByProjectType()} ${this.getFacilityName()} ?`; - break; - } - case 1: { - this.emailVerify = `Are you sure you want to send this newsletter to all members ${this.getMailConfinementByProjectType()} ${this.getFacilityName()} ?`; - break; + + getMailConfinementByProjectType(): string { + switch (this.selectedProjectType) { + case 'ALL_GM': + return 'of all active projects'; + case 'EXP': + return 'of all expired projects'; + case 'SVP': + return 'of all SimpleVM projects'; + case 'OVP': + return 'of all OpenStack projects'; + case 'WSH': + return 'of all Workshops'; + default: + return ''; } - default: - this.emailVerify = 'Are you sure you want to send this?'; - } - if (this.selectedProjectType !== 'EXP') { - this.expiredTemplated = false; } - } - getVoProjects(): void { - this.projects = []; - this.voService.getAllGroupsWithDetails().subscribe((applications: Application[]): void => { - for (const application of applications) { - if (application.project_application_lifetime > 0) { - application.lifetime_reached = this.lifeTimeReached(application.lifetime_days, application.DaysRunning); + adjustVerifyText(): void { + switch (this.emailType) { + case 0: { + this.emailVerify = `Are you sure you want to send this email to all ${ + this.emailAdminsOnly ? ' group administrators' : 'members' + } ${this.getMailConfinementByProjectType()} ${this.getFacilityName()} ?`; + break; } - this.projects.push(application); + case 1: { + this.emailVerify = `Are you sure you want to send this newsletter to all members ${this.getMailConfinementByProjectType()} ${this.getFacilityName()} ?`; + break; + } + default: + this.emailVerify = 'Are you sure you want to send this?'; } - this.sortProjectService.applications = this.projects; - this.applictions$ = this.sortProjectService.applications$; - this.total$ = this.sortProjectService.total$; - - this.isLoaded = true; - }); - } - - resetEmailModal(): void { - this.emailHeader = null; - this.emailSubject = null; - this.emailText = null; - this.emailType = null; - this.emailVerify = null; - this.emailReply = ''; - this.emailStatus = 0; - this.emailAdminsOnly = false; - } - - public resetNotificationModal(): void { - this.notificationModalTitle = 'Notification'; - this.notificationModalMessage = 'Please wait...'; - this.notificationModalIsClosable = false; - this.notificationModalType = 'info'; - } - - /** - * Get all computecenters. - */ - getComputeCenters(): void { - this.facilityService.getComputeCenters().subscribe((result: any): void => { - for (const cc of result) { - const compute_center: ComputecenterComponent = new ComputecenterComponent( - cc['compute_center_facility_id'], - cc['compute_center_name'], - cc['compute_center_login'], - cc['compute_center_support_mail'], - ); - this.computecenters.push(compute_center); + if (this.selectedProjectType !== 'EXP') { + this.expiredTemplated = false; } - }); - } - - /** - * Bugfix not scrollable site after closing modal - */ - removeModalOpen(): void { - document.body.classList.remove('modal-open'); - } - - public terminateProject(): void { - this.voService.terminateProject(this.selectedProject.project_application_perun_id).subscribe( - (): void => { - const indexAll: number = this.projects.indexOf(this.selectedProject, 0); - if (!this.selectedProject.project_application_openstack_project) { - this.projects.splice(indexAll, 1); - this.sortProjectService.applications = this.projects; - } else { - this.getProjectStatus(this.projects[indexAll]); + } + + getVoProjects(): void { + this.projects = []; + this.voService.getAllGroupsWithDetails().subscribe((applications: Application[]): void => { + for (const application of applications) { + if (application.project_application_lifetime > 0) { + application.lifetime_reached = this.lifeTimeReached(application.lifetime_days, application.DaysRunning); + } + this.projects.push(application); } - this.fullLayout.getGroupsEnumeration(); - if (this.selectedProject.project_application_openstack_project) { - this.updateNotificationModal( - 'Success', - 'The request to terminate the project was forwarded to the facility manager.', - true, - 'success', + this.sortProjectService.applications = this.projects; + this.applictions$ = this.sortProjectService.applications$; + this.total$ = this.sortProjectService.total$; + this.projectsLoaded = true; + }); + } + + resetEmailModal(): void { + this.emailHeader = null; + this.emailSubject = null; + this.emailText = null; + this.emailType = null; + this.emailVerify = null; + this.emailReply = ''; + this.emailStatus = 0; + this.emailAdminsOnly = false; + } + + public resetNotificationModal(): void { + this.notificationModalTitle = 'Notification'; + this.notificationModalMessage = 'Please wait...'; + this.notificationModalIsClosable = false; + this.notificationModalType = 'info'; + } + + /** + * Get all computecenters. + */ + getComputeCenters(): void { + this.facilityService.getComputeCenters().subscribe((result: any): void => { + for (const cc of result) { + const compute_center: ComputecenterComponent = new ComputecenterComponent( + cc['compute_center_facility_id'], + cc['compute_center_name'], + cc['compute_center_login'], + cc['compute_center_support_mail'], ); - } else { - this.updateNotificationModal('Success', 'The project was terminated.', true, 'success'); + this.computecenters.push(compute_center); } - }, - (error: any): void => { - if (error['status'] === 409) { + }); + } + + /** + * Bugfix not scrollable site after closing modal + */ + removeModalOpen(): void { + document.body.classList.remove('modal-open'); + } + + public terminateProject(): void { + this.voService.terminateProject(this.selectedProject.project_application_perun_id).subscribe( + (): void => { + const indexAll: number = this.projects.indexOf(this.selectedProject, 0); + if (!this.selectedProject.project_application_openstack_project) { + this.projects.splice(indexAll, 1); + this.sortProjectService.applications = this.projects; + } else { + this.getProjectStatus(this.projects[indexAll]); + } + this.fullLayout.getGroupsEnumeration(); + if (this.selectedProject.project_application_openstack_project) { + this.updateNotificationModal( + 'Success', + 'The request to terminate the project was forwarded to the facility manager.', + true, + 'success', + ); + } else { + this.updateNotificationModal('Success', 'The project was terminated.', true, 'success'); + } + }, + (error: any): void => { + if (error['status'] === 409) { + this.updateNotificationModal( + 'Failed', + `The project could not be terminated. Reason: ${error['error']['reason']} for ${error['error']['openstackid']}`, + true, + 'danger', + ); + } else { + this.updateNotificationModal('Failed', 'The project could not be terminated.', true, 'danger'); + } + }, + ); + } + + getProjectStatus(project: Application): void { + this.voService.getProjectStatus(project.project_application_perun_id).subscribe((res: any): void => { + project.project_application_statuses = res['status']; + }); + } + + suspendProject(project: Application): void { + this.voService.removeResourceFromGroup(project.project_application_perun_id).subscribe( + (): void => { + this.updateNotificationModal('Success', 'The project got suspended successfully', true, 'success'); + this.getProjectStatus(project); + project.project_application_compute_center = null; + }, + (): void => { + this.updateNotificationModal('Failed', 'The status change was not successful.', true, 'danger'); + }, + ); + } + + resumeProject(project: Application): void { + this.voService.resumeProject(project.project_application_perun_id).subscribe( + (): void => { + this.updateNotificationModal('Success', 'The project got resumed successfully', true, 'success'); + this.getProjectStatus(project); + }, + (): void => { + this.updateNotificationModal('Failed', 'The status change was not successful.', true, 'danger'); + }, + ); + } + + declineTermination(project: Application): void { + this.voService.declineTermination(project.project_application_perun_id).subscribe( + (): void => { + this.updateNotificationModal('Success', 'The termination was successfully declined', true, 'success'); + const indexAll: number = this.projects.indexOf(project, 0); + this.getProjectStatus(this.projects[indexAll]); + }, + (): void => { + this.updateNotificationModal('Failed', 'The status change was not successful.', true, 'danger'); + }, + ); + } + + setProtected(project: Application, set: boolean): void { + this.voService.setProtected(project.project_application_perun_id, set).subscribe( + (result: any): void => { this.updateNotificationModal( - 'Failed', - `The project could not be terminated. Reason: ${error['error']['reason']} for ${error['error']['openstackid']}`, + 'Success', + result['result'] === 'set' + ? 'The project was successfully set as protected.' + : 'The status "Protected" was removed successfully', true, - 'danger', + 'success', ); - } else { - this.updateNotificationModal('Failed', 'The project could not be terminated.', true, 'danger'); - } - }, - ); - } - - getProjectStatus(project: Application): void { - this.voService.getProjectStatus(project.project_application_perun_id).subscribe((res: any): void => { - project.project_application_statuses = res['status']; - }); - } - - suspendProject(project: Application): void { - this.voService.removeResourceFromGroup(project.project_application_perun_id).subscribe( - (): void => { - this.updateNotificationModal('Success', 'The project got suspended successfully', true, 'success'); - this.getProjectStatus(project); - project.project_application_compute_center = null; - }, - (): void => { - this.updateNotificationModal('Failed', 'The status change was not successful.', true, 'danger'); - }, - ); - } - - resumeProject(project: Application): void { - this.voService.resumeProject(project.project_application_perun_id).subscribe( - (): void => { - this.updateNotificationModal('Success', 'The project got resumed successfully', true, 'success'); - this.getProjectStatus(project); - }, - (): void => { - this.updateNotificationModal('Failed', 'The status change was not successful.', true, 'danger'); - }, - ); - } - - declineTermination(project: Application): void { - this.voService.declineTermination(project.project_application_perun_id).subscribe( - (): void => { - this.updateNotificationModal('Success', 'The termination was successfully declined', true, 'success'); - const indexAll: number = this.projects.indexOf(project, 0); - this.getProjectStatus(this.projects[indexAll]); - }, - (): void => { - this.updateNotificationModal('Failed', 'The status change was not successful.', true, 'danger'); - }, - ); - } - - setProtected(project: Application, set: boolean): void { - this.voService.setProtected(project.project_application_perun_id, set).subscribe( - (result: any): void => { - this.updateNotificationModal( - 'Success', - result['result'] === 'set' - ? 'The project was successfully set as protected.' - : 'The status "Protected" was removed successfully', - true, - 'success', - ); - const indexAll: number = this.projects.indexOf(project, 0); - this.getProjectStatus(this.projects[indexAll]); - }, - (error: any): void => { - if (error['status'] === 500) { - this.updateNotificationModal('Failed', 'The status change was not successful.', true, 'danger'); - } - }, - ); - } - - getMembersOfTheProject(projectid: number, projectname: string): void { - this.voService.getVoGroupRichMembers(projectid).subscribe((members: ProjectMember[]): void => { - this.usersModalProjectID = projectid; - this.usersModalProjectName = projectname; - this.usersModalProjectMembers = members; - }); - } - - showMembersOfTheProject(projectid: number, projectname: string): void { - this.getMembersOfTheProject(projectid, projectname); - } - - initiateTsvExport(): void { - this.tsvTaskRunning = true; - this.voService.getAllProjectsForTsvExport().subscribe((): void => { - this.getTSVInformation(); - }); - } - - downloadCurrentTSV(): void { - this.voService.downloadProjectsTsv().subscribe( - (result): void => { - const blobn = new Blob([result], { - type: 'text/tsv', - }); + const indexAll: number = this.projects.indexOf(project, 0); + this.getProjectStatus(this.projects[indexAll]); + }, + (error: any): void => { + if (error['status'] === 500) { + this.updateNotificationModal('Failed', 'The status change was not successful.', true, 'danger'); + } + }, + ); + } - const dateTime = new Date(); - FileSaver.saveAs(blobn, `projects-${dateTime.getDate()}-${dateTime.getMonth()}-${dateTime.getFullYear()}.tsv`); - }, - (err: any) => { - console.log(`No such file found! - ${err.toString()}`); - }, - ); - } + getMembersOfTheProject(projectid: number, projectname: string): void { + this.voService.getVoGroupRichMembers(projectid).subscribe((members: ProjectMember[]): void => { + this.usersModalProjectID = projectid; + this.usersModalProjectName = projectname; + this.usersModalProjectMembers = members; + }); + } + + showMembersOfTheProject(projectid: number, projectname: string): void { + this.getMembersOfTheProject(projectid, projectname); + } + + initiateTsvExport(): void { + this.tsvTaskRunning = true; + this.voService.getAllProjectsForTsvExport().subscribe((): void => { + this.getTSVInformation(); + }); + } + + downloadCurrentTSV(): void { + this.voService.downloadProjectsTsv().subscribe( + (result): void => { + const blobn = new Blob([result], { + type: 'text/tsv', + }); + + const dateTime = new Date(); + FileSaver.saveAs(blobn, `projects-${dateTime.getDate()}-${dateTime.getMonth()}-${dateTime.getFullYear()}.tsv`); + }, + (err: any) => { + console.log(`No such file found! - ${err.toString()}`); + }, + ); + } } diff --git a/src/app/vo_manager/voOverview.component.html b/src/app/vo_manager/voOverview.component.html index e6e264d005..cae2288c8c 100644 --- a/src/app/vo_manager/voOverview.component.html +++ b/src/app/vo_manager/voOverview.component.html @@ -1,1161 +1,1268 @@ -
+
+
+ -
-
- +
+ -
- + - + + + + - - - - + - +
+
Projects
-
-
Projects
+
+
+ -
-
- +
+ -
- + - + +
- -
+ +
- -
+
+
+ +
+ +
+ Loading... +
+
+ + +
+
+
+
+ +
+
+
+
+ +
+ -
-
- -
- -
- Loading... -
-
- - -
-
-
-
- -
-
-
+ Loading... -
-
-
+ + +
+
+ -
-
- +
+
+ -
-
- +
+
+ -
-
- +
+
+ -
- + +
+ -
- +
+
+ +
-
-
-
- +
+
+ -
-
- +
+
+ -
-
- - - - - + [checked]="sortProjectService.filterStatusList | inList: application_states.WAIT_FOR_TERMINATION_FM" + /> + {{ + application_states[application_states.WAIT_FOR_TERMINATION_FM] + }} + + + +
SelectType
+ + + + - + - - - - - - - + + + + + + + - - - - - - - - - - - + + + + + + @if (projectsLoaded) { - - + + + + + + + + - - - + + + + - + - - - -
SelectTypeFacilityFacilityProject IDProject NameDetailsStatusRamCoresGPUsProject ID + Project Name + DetailsStatusRam + Cores + GPUs + CreditsActions
- - - - - - - > - - - - Credits + Actions
- - +
+ + + + + + + > + + + + + + {{ project_states[project_states.ACTIVE] }} + *ngIf="project | hasstatusinlist: project_states.ACTIVE" + style="margin-left: 5px" + class="badge bg-success" + >{{ project_states[project_states.ACTIVE] }} - {{ application_states[application_states.DISABLED] }} + {{ application_states[application_states.DISABLED] }} - {{ lifetime_states[lifetime_states.EXPIRED] }} + style="margin-left: 5px" + class="badge bg-danger" + >{{ lifetime_states[lifetime_states.EXPIRED] }} - {{ lifetime_states[lifetime_states.EXPIRES_SOON] }} - {{ project_states[project_states.SUSPENDED] }} + {{ lifetime_states[lifetime_states.EXPIRES_SOON] }} + {{ project_states[project_states.SUSPENDED] }} - WAIT FOR CONFIRMATION + WAIT FOR CONFIRMATION - TERMINATION REQUESTED + TERMINATION REQUESTED - WAIT FOR TERMINATION BY FM + WAIT FOR TERMINATION BY FM - PROTECTED + PROTECTED - - - - > - - - > - - + + + > + + + > + + - > - - / - > - - + > + + / + > + + -
- +
+
+ - - - + + + - + - - - - - -
-
-
-
-
-
- -
-
+ + + + + +
+ + + } @else { + + + + + + } + + +
+
+
+
+ +
+
- + - - + > + Yes + + +
+ + + + + + - - + + - - + + - + - - + + + + - + - - + + - - + +