2022-01-20 21:56:17 +00:00
|
|
|
/*****************************************************************************
|
2024-01-09 21:31:51 +00:00
|
|
|
* Open MCT, Copyright (c) 2014-2024, United States Government
|
2022-01-20 21:56:17 +00:00
|
|
|
* as represented by the Administrator of the National Aeronautics and Space
|
|
|
|
* Administration. All rights reserved.
|
|
|
|
*
|
|
|
|
* Open MCT is 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.
|
|
|
|
*
|
|
|
|
* Open MCT includes source code licensed under additional open source
|
|
|
|
* licenses. See the Open Source Licenses file (LICENSES.md) included with
|
|
|
|
* this source code distribution or the Licensing information page available
|
|
|
|
* at runtime from the About dialog for additional information.
|
|
|
|
*****************************************************************************/
|
|
|
|
|
2024-07-31 17:46:16 +00:00
|
|
|
import { EventEmitter } from 'eventemitter3';
|
2022-05-25 21:09:08 +00:00
|
|
|
import { v4 as uuid } from 'uuid';
|
2023-08-31 20:40:00 +00:00
|
|
|
|
2024-01-02 15:24:22 +00:00
|
|
|
import createExampleUser from './exampleUserCreator.js';
|
2022-01-20 21:56:17 +00:00
|
|
|
|
2023-05-18 21:54:46 +00:00
|
|
|
const STATUSES = [
|
|
|
|
{
|
|
|
|
key: 'NO_STATUS',
|
|
|
|
label: 'Not set',
|
|
|
|
iconClass: 'icon-question-mark',
|
|
|
|
iconClassPoll: 'icon-status-poll-question-mark'
|
|
|
|
},
|
|
|
|
{
|
|
|
|
key: 'GO',
|
|
|
|
label: 'Go',
|
|
|
|
iconClass: 'icon-check',
|
|
|
|
iconClassPoll: 'icon-status-poll-question-mark',
|
|
|
|
statusClass: 's-status-ok',
|
|
|
|
statusBgColor: '#33cc33',
|
|
|
|
statusFgColor: '#000'
|
|
|
|
},
|
|
|
|
{
|
|
|
|
key: 'MAYBE',
|
|
|
|
label: 'Maybe',
|
|
|
|
iconClass: 'icon-alert-triangle',
|
|
|
|
iconClassPoll: 'icon-status-poll-question-mark',
|
|
|
|
statusClass: 's-status-warning',
|
|
|
|
statusBgColor: '#ffb66c',
|
|
|
|
statusFgColor: '#000'
|
|
|
|
},
|
|
|
|
{
|
|
|
|
key: 'NO_GO',
|
|
|
|
label: 'No go',
|
|
|
|
iconClass: 'icon-circle-slash',
|
|
|
|
iconClassPoll: 'icon-status-poll-question-mark',
|
|
|
|
statusClass: 's-status-error',
|
|
|
|
statusBgColor: '#9900cc',
|
|
|
|
statusFgColor: '#fff'
|
|
|
|
}
|
|
|
|
];
|
2024-02-06 21:44:01 +00:00
|
|
|
|
|
|
|
const MISSION_STATUSES = [
|
|
|
|
{
|
|
|
|
key: 0,
|
|
|
|
label: 'NO GO'
|
|
|
|
},
|
|
|
|
{
|
|
|
|
key: 1,
|
|
|
|
label: 'GO'
|
|
|
|
}
|
|
|
|
];
|
2022-06-02 20:46:13 +00:00
|
|
|
/**
|
|
|
|
* @implements {StatusUserProvider}
|
|
|
|
*/
|
2022-01-20 21:56:17 +00:00
|
|
|
export default class ExampleUserProvider extends EventEmitter {
|
2024-02-06 21:44:01 +00:00
|
|
|
#actionToStatusMap;
|
2023-07-14 19:10:58 +00:00
|
|
|
constructor(
|
|
|
|
openmct,
|
|
|
|
{ statusRoles } = {
|
|
|
|
statusRoles: []
|
|
|
|
}
|
|
|
|
) {
|
2023-05-18 21:54:46 +00:00
|
|
|
super();
|
2022-06-02 20:46:13 +00:00
|
|
|
|
2023-05-18 21:54:46 +00:00
|
|
|
this.openmct = openmct;
|
2024-02-06 21:44:01 +00:00
|
|
|
this.#actionToStatusMap = {
|
|
|
|
Imagery: MISSION_STATUSES[0],
|
|
|
|
Commanding: MISSION_STATUSES[0],
|
|
|
|
Driving: MISSION_STATUSES[0]
|
|
|
|
};
|
2023-05-18 21:54:46 +00:00
|
|
|
this.user = undefined;
|
|
|
|
this.loggedIn = false;
|
|
|
|
this.autoLoginUser = undefined;
|
2023-07-14 19:10:58 +00:00
|
|
|
this.statusRoleValues = statusRoles.map((role) => ({
|
|
|
|
role: role,
|
|
|
|
status: STATUSES[0]
|
|
|
|
}));
|
2023-05-18 21:54:46 +00:00
|
|
|
this.pollQuestion = undefined;
|
2023-07-14 19:10:58 +00:00
|
|
|
this.statusRoles = statusRoles;
|
2022-06-02 20:46:13 +00:00
|
|
|
|
2023-05-18 21:54:46 +00:00
|
|
|
this.ExampleUser = createExampleUser(this.openmct.user.User);
|
|
|
|
this.loginPromise = undefined;
|
|
|
|
}
|
2022-06-02 20:46:13 +00:00
|
|
|
|
2023-05-18 21:54:46 +00:00
|
|
|
isLoggedIn() {
|
|
|
|
return this.loggedIn;
|
|
|
|
}
|
2022-06-02 20:46:13 +00:00
|
|
|
|
2023-05-18 21:54:46 +00:00
|
|
|
autoLogin(username) {
|
|
|
|
this.autoLoginUser = username;
|
|
|
|
}
|
2022-06-02 20:46:13 +00:00
|
|
|
|
2023-05-18 21:54:46 +00:00
|
|
|
getCurrentUser() {
|
|
|
|
if (!this.loginPromise) {
|
|
|
|
this.loginPromise = this._login().then(() => this.user);
|
2022-06-02 20:46:13 +00:00
|
|
|
}
|
|
|
|
|
2023-05-18 21:54:46 +00:00
|
|
|
return this.loginPromise;
|
|
|
|
}
|
|
|
|
|
2023-07-14 19:10:58 +00:00
|
|
|
canProvideStatusForRole(role) {
|
|
|
|
return this.statusRoles.includes(role);
|
2023-05-18 21:54:46 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
canSetPollQuestion() {
|
|
|
|
return Promise.resolve(true);
|
|
|
|
}
|
2024-02-02 22:50:16 +00:00
|
|
|
|
|
|
|
canSetMissionStatus() {
|
2024-02-06 21:44:01 +00:00
|
|
|
return Promise.resolve(true);
|
2024-02-02 22:50:16 +00:00
|
|
|
}
|
|
|
|
|
2023-05-18 21:54:46 +00:00
|
|
|
hasRole(roleId) {
|
|
|
|
if (!this.loggedIn) {
|
|
|
|
Promise.resolve(undefined);
|
|
|
|
}
|
|
|
|
|
|
|
|
return Promise.resolve(this.user.getRoles().includes(roleId));
|
|
|
|
}
|
|
|
|
|
2023-07-14 19:10:58 +00:00
|
|
|
getPossibleRoles() {
|
|
|
|
return this.user.getRoles();
|
2023-05-18 21:54:46 +00:00
|
|
|
}
|
|
|
|
|
2024-02-06 21:44:01 +00:00
|
|
|
getPossibleMissionActions() {
|
|
|
|
return Promise.resolve(Object.keys(this.#actionToStatusMap));
|
|
|
|
}
|
|
|
|
|
|
|
|
getPossibleMissionActionStatuses() {
|
|
|
|
return Promise.resolve(MISSION_STATUSES);
|
|
|
|
}
|
|
|
|
|
|
|
|
getStatusForMissionAction(action) {
|
|
|
|
return Promise.resolve(this.#actionToStatusMap[action]);
|
|
|
|
}
|
|
|
|
|
|
|
|
setStatusForMissionAction(action, status) {
|
|
|
|
this.#actionToStatusMap[action] = status;
|
|
|
|
this.emit('missionStatusChange', {
|
|
|
|
action,
|
|
|
|
status
|
|
|
|
});
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2023-05-18 21:54:46 +00:00
|
|
|
getAllStatusRoles() {
|
2023-07-14 19:10:58 +00:00
|
|
|
return Promise.resolve(this.statusRoles);
|
2023-05-18 21:54:46 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
getStatusForRole(role) {
|
2023-07-14 19:10:58 +00:00
|
|
|
const statusForRole = this.statusRoleValues.find((statusRole) => statusRole.role === role);
|
|
|
|
|
|
|
|
return Promise.resolve(statusForRole?.status);
|
2023-05-18 21:54:46 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
async getDefaultStatusForRole(role) {
|
|
|
|
const allRoles = await this.getPossibleStatuses();
|
|
|
|
|
|
|
|
return allRoles?.[0];
|
|
|
|
}
|
|
|
|
|
|
|
|
setStatusForRole(role, status) {
|
|
|
|
status.timestamp = Date.now();
|
2023-07-14 19:10:58 +00:00
|
|
|
const matchingIndex = this.statusRoleValues.findIndex((statusRole) => statusRole.role === role);
|
|
|
|
this.statusRoleValues[matchingIndex].status = status;
|
2023-05-18 21:54:46 +00:00
|
|
|
this.emit('statusChange', {
|
|
|
|
role,
|
|
|
|
status
|
|
|
|
});
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
// eslint-disable-next-line require-await
|
|
|
|
async getPollQuestion() {
|
|
|
|
if (this.pollQuestion) {
|
|
|
|
return this.pollQuestion;
|
|
|
|
} else {
|
|
|
|
return undefined;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
setPollQuestion(pollQuestion) {
|
|
|
|
if (!pollQuestion) {
|
|
|
|
// If the poll question is undefined, set it to a blank string.
|
|
|
|
// This behavior better reflects how other telemetry systems
|
|
|
|
// deal with undefined poll questions.
|
|
|
|
pollQuestion = '';
|
|
|
|
}
|
|
|
|
|
|
|
|
this.pollQuestion = {
|
|
|
|
question: pollQuestion,
|
|
|
|
timestamp: Date.now()
|
|
|
|
};
|
|
|
|
this.emit('pollQuestionChange', this.pollQuestion);
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
getPossibleStatuses() {
|
|
|
|
return Promise.resolve(STATUSES);
|
|
|
|
}
|
|
|
|
|
|
|
|
_login() {
|
|
|
|
const id = uuid();
|
|
|
|
|
|
|
|
// for testing purposes, this will skip the form, this wouldn't be used in
|
|
|
|
// a normal authentication process
|
|
|
|
if (this.autoLoginUser) {
|
2023-07-14 19:10:58 +00:00
|
|
|
this.user = new this.ExampleUser(id, this.autoLoginUser, ['flight', 'driver', 'observer']);
|
2023-05-18 21:54:46 +00:00
|
|
|
this.loggedIn = true;
|
2022-06-02 20:46:13 +00:00
|
|
|
|
2023-05-18 21:54:46 +00:00
|
|
|
return Promise.resolve();
|
2022-06-02 20:46:13 +00:00
|
|
|
}
|
|
|
|
|
2023-05-18 21:54:46 +00:00
|
|
|
const formStructure = {
|
|
|
|
title: 'Login',
|
|
|
|
sections: [
|
|
|
|
{
|
|
|
|
rows: [
|
|
|
|
{
|
|
|
|
key: 'username',
|
|
|
|
control: 'textfield',
|
|
|
|
name: 'Username',
|
|
|
|
pattern: '\\S+',
|
|
|
|
required: true,
|
|
|
|
cssClass: 'l-input-lg',
|
|
|
|
value: ''
|
|
|
|
}
|
|
|
|
]
|
2023-01-20 02:56:46 +00:00
|
|
|
}
|
2023-05-18 21:54:46 +00:00
|
|
|
],
|
|
|
|
buttons: {
|
|
|
|
submit: {
|
|
|
|
label: 'Login'
|
2022-01-20 21:56:17 +00:00
|
|
|
}
|
2023-05-18 21:54:46 +00:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
return this.openmct.forms.showForm(formStructure).then(
|
|
|
|
(info) => {
|
|
|
|
this.user = new this.ExampleUser(id, info.username, ['example-role']);
|
|
|
|
this.loggedIn = true;
|
|
|
|
},
|
|
|
|
() => {
|
|
|
|
// user canceled, setting a default username
|
|
|
|
this.user = new this.ExampleUser(id, 'Pat', ['example-role']);
|
|
|
|
this.loggedIn = true;
|
|
|
|
}
|
|
|
|
);
|
|
|
|
}
|
2022-01-20 21:56:17 +00:00
|
|
|
}
|
2022-06-02 20:46:13 +00:00
|
|
|
/**
|
|
|
|
* @typedef {import('@/api/user/StatusUserProvider').default} StatusUserProvider
|
|
|
|
*/
|