diff --git a/SAS/TMSS/frontend/tmss_webapp/src/routes/Cycle/view.js b/SAS/TMSS/frontend/tmss_webapp/src/routes/Cycle/view.js index 61f385ed7b7e6360a1a0775ec6880e4bc8d91392..34eef2047443a069cb60beecfe610a022de8c901 100644 --- a/SAS/TMSS/frontend/tmss_webapp/src/routes/Cycle/view.js +++ b/SAS/TMSS/frontend/tmss_webapp/src/routes/Cycle/view.js @@ -144,11 +144,9 @@ export class CycleView extends Component { <div className="col-lg-3 col-md-3 col-sm-12"> <h5 data-testid="project-list">Projects</h5> </div> - <div className="col-lg-12 col-md-12 "> - <ProjectList cycle={this.state.cycle.name}/> - </div> </div> </div> + <ProjectList cycle={this.state.cycle.name}/> </div> </React.Fragment> } diff --git a/SAS/TMSS/frontend/tmss_webapp/src/routes/Project/view.js b/SAS/TMSS/frontend/tmss_webapp/src/routes/Project/view.js index 9498a81b929948705bc5c2eb31ae36d6750916b9..9f1c5e83cdd4e748a335f7bf4735c68fb004be41 100644 --- a/SAS/TMSS/frontend/tmss_webapp/src/routes/Project/view.js +++ b/SAS/TMSS/frontend/tmss_webapp/src/routes/Project/view.js @@ -186,11 +186,9 @@ export class ProjectView extends Component { <div className="col-lg-3 col-md-3 col-sm-12"> <h5 data-testid="resource_alloc">Scheduling Unit - List</h5> </div> - <div className="col-lg-12 col-md-12 "> - <SchedulingUnitList project={this.state.project.name}/> - </div> </div> </div> + <SchedulingUnitList project={this.state.project.name}/> </div> </React.Fragment> } diff --git a/SAS/TMSS/frontend/tmss_webapp/src/routes/Scheduling/ViewSchedulingUnit.js b/SAS/TMSS/frontend/tmss_webapp/src/routes/Scheduling/ViewSchedulingUnit.js index 3a443824d92d79c5ca8433a461fe12ef4efaebaa..23a829a506f8f89f56a9af3aee8319cf1ae2c9ee 100644 --- a/SAS/TMSS/frontend/tmss_webapp/src/routes/Scheduling/ViewSchedulingUnit.js +++ b/SAS/TMSS/frontend/tmss_webapp/src/routes/Scheduling/ViewSchedulingUnit.js @@ -122,7 +122,9 @@ class ViewSchedulingUnit extends Component{ </div> </div> */} <PageHeader location={this.props.location} title={'Scheduling Unit - Details'} - actions={[{icon: 'fa-window-close',title:'Click to Close Scheduling Unit View', props : { pathname: '/schedulingunit'}}]}/> + actions={[{icon: 'fa-edit',title:'Click to Edit Scheduling Unit View', type:'link', + props : { pathname: `/schedulingunit/edit/${this.props.match.params.id}` }}, + {icon: 'fa-window-close',title:'Click to Close Scheduling Unit View', props : { pathname: '/schedulingunit'}}]}/> { this.state.isLoading ? <AppLoader/> :this.state.scheduleunit && <> <div className="main-content"> diff --git a/SAS/TMSS/frontend/tmss_webapp/src/routes/Scheduling/edit.js b/SAS/TMSS/frontend/tmss_webapp/src/routes/Scheduling/edit.js new file mode 100644 index 0000000000000000000000000000000000000000..8ba417b69773952d7cd64e70c470e581aad19f8a --- /dev/null +++ b/SAS/TMSS/frontend/tmss_webapp/src/routes/Scheduling/edit.js @@ -0,0 +1,392 @@ +import React, {Component} from 'react'; +import { Redirect } from 'react-router-dom'; +import _ from 'lodash'; +import $RefParser from "@apidevtools/json-schema-ref-parser"; + +import {InputText} from 'primereact/inputtext'; +import {InputTextarea} from 'primereact/inputtextarea'; +import {Dropdown} from 'primereact/dropdown'; +import { Button } from 'primereact/button'; +import {Growl} from 'primereact/components/growl/Growl'; + + +import AppLoader from '../../layout/components/AppLoader'; +import PageHeader from '../../layout/components/PageHeader'; +import Jeditor from '../../components/JSONEditor/JEditor'; + +import ProjectService from '../../services/project.service'; +import ScheduleService from '../../services/schedule.service'; +import TaskService from '../../services/task.service'; +import UIConstants from '../../utils/ui.constants'; + +/** + * Compoenent to edit scheduling unit draft + */ +export class EditSchedulingUnit extends Component { + constructor(props) { + super(props); + this.state = { + isLoading: true, + dialog: { header: '', detail: ''}, + redirect: null, + errors: [], + schedulingSets: [], + schedulingUnit: { + }, + projectDisabled: (props.match?(props.match.params.project? true:false):false), + observStrategy: {}, + paramsSchema: null, + validEditor: false, + validFields: {}, + observStrategyVisible: false + } + this.projects = []; + this.schedulingSets = []; + this.observStrategies = []; + this.taskTemplates = []; + this.tooltipOptions = UIConstants.tooltipOptions; + this.nameInput = React.createRef(); + this.formRules = { + name: {required: true, message: "Name can not be empty"}, + description: {required: true, message: "Description can not be empty"}, + }; + + this.setEditorOutput = this.setEditorOutput.bind(this); + this.changeStrategy = this.changeStrategy.bind(this); + this.setSchedUnitParams = this.setSchedUnitParams.bind(this); + this.validateForm = this.validateForm.bind(this); + this.validateEditor = this.validateEditor.bind(this); + this.setEditorFunction = this.setEditorFunction.bind(this); + this.saveSchedulingUnit = this.saveSchedulingUnit.bind(this); + this.cancelCreate = this.cancelCreate.bind(this); + + } + + + + /** + * Function called when observation strategy template is changed. + * It generates the JSON schema for JSON editor and defult vales for the parameters to be captured + * @param {number} strategyId + */ + async changeStrategy (strategyId) { + let tasksToUpdate = {}; + const observStrategy = _.find(this.observStrategies, {'id': strategyId}); + const tasks = observStrategy.template.tasks; + let paramsOutput = {}; + let schema = { type: 'object', additionalProperties: false, + properties: {}, definitions:{} + }; + for (const taskName in tasks) { + const task = tasks[taskName]; + const taskDraft = this.state.taskDrafts.find(taskD => taskD.name === taskName); + if (taskDraft) { + task.specifications_doc = taskDraft.specifications_doc; + } + //Resolve task from the strategy template + const $taskRefs = await $RefParser.resolve(task); + + // Identify the task specification template of every task in the strategy template + const taskTemplate = _.find(this.taskTemplates, {'name': task['specifications_template']}); + schema['$id'] = taskTemplate.schema['$id']; + schema['$schema'] = taskTemplate.schema['$schema']; + observStrategy.template.parameters.forEach(async(param, index) => { + if (param.refs[0].indexOf(`/tasks/${taskName}`) > 0) { + tasksToUpdate[taskName] = taskName; + // Resolve the identified template + const $templateRefs = await $RefParser.resolve(taskTemplate); + let property = { }; + let tempProperty = null; + // Get the property type from the template and create new property in the schema for the parameters + try { + tempProperty = $templateRefs.get(param.refs[0].replace(`#/tasks/${taskName}/specifications_doc`, '#/schema/properties')) + } catch(error) { + const taskPaths = param.refs[0].split("/"); + tempProperty = _.cloneDeep(taskTemplate.schema.properties[taskPaths[4]]); + if (tempProperty.type === 'array') { + tempProperty = tempProperty.items.properties[taskPaths[6]]; + } + property = tempProperty; + } + property.title = param.name; + property.default = $taskRefs.get(param.refs[0].replace(`#/tasks/${taskName}`, '#')); + paramsOutput[`param_${index}`] = property.default; + schema.properties[`param_${index}`] = property; + // Set property defintions taken from the task template in new schema + for (const definitionName in taskTemplate.schema.definitions) { + schema.definitions[definitionName] = taskTemplate.schema.definitions[definitionName]; + } + } + }); + } + this.setState({observStrategy: observStrategy, paramsSchema: schema, paramsOutput: paramsOutput, tasksToUpdate: tasksToUpdate}); + + // Function called to clear the JSON Editor fields and reload with new schema + if (this.state.editorFunction) { + this.state.editorFunction(); + } + } + + componentDidMount() { + const promises = [ ProjectService.getProjectList(), + ScheduleService.getSchedulingSets(), + ScheduleService.getObservationStrategies(), + TaskService.getTaskTemplates(), + ScheduleService.getSchedulingUnitDraftById(this.props.match.params.id), + ScheduleService.getTasksDraftBySchedulingUnitId(this.props.match.params.id) + ]; + Promise.all(promises).then(responses => { + this.projects = responses[0]; + this.schedulingSets = responses[1]; + this.observStrategies = responses[2]; + this.taskTemplates = responses[3]; + responses[4].project = this.schedulingSets.find(i => i.id === responses[4].scheduling_set_id).project_id; + this.setState({ schedulingUnit: responses[4], taskDrafts: responses[5].data.results, + observStrategyVisible: responses[4].observation_strategy_template_id?true:false }); + if (responses[4].observation_strategy_template_id) { + this.changeStrategy(responses[4].observation_strategy_template_id); + } + if (this.state.schedulingUnit.project) { + const projectSchedSets = _.filter(this.schedulingSets, {'project_id': this.state.schedulingUnit.project}); + this.setState({isLoading: false, schedulingSets: projectSchedSets}); + } else { + this.setState({isLoading: false}); + } + }); + } + + /** + * This is the callback method to be passed to the JSON editor. + * JEditor will call this function when there is change in the editor. + * @param {Object} jsonOutput + * @param {Array} errors + */ + setEditorOutput(jsonOutput, errors) { + this.paramsOutput = jsonOutput; + this.validEditor = errors.length === 0; + this.setState({ paramsOutput: jsonOutput, + validEditor: errors.length === 0, + validForm: this.validateForm()}); + } + + /** + * This function is mainly added for Unit Tests. If this function is removed Unit Tests will fail. + */ + validateEditor() { + return this.validEditor?true:false; + } + + /** + * Function to set form values to the SU object + * @param {string} key + * @param {object} value + */ + setSchedUnitParams(key, value) { + let schedulingUnit = this.state.schedulingUnit; + schedulingUnit[key] = value; + this.setState({schedulingUnit: schedulingUnit, validForm: this.validateForm(key), validEditor: this.validateEditor()}); + this.validateEditor(); + } + + /** + * JEditor's function that to be called when parent wants to trigger change in the JSON Editor + * @param {Function} editorFunction + */ + setEditorFunction(editorFunction) { + this.setState({editorFunction: editorFunction}); + } + + /** + * Validation function to validate the form or field based on the form rules. + * If no argument passed for fieldName, validates all fields in the form. + * @param {string} fieldName + */ + validateForm(fieldName) { + let validForm = false; + let errors = this.state.errors; + let validFields = this.state.validFields; + if (fieldName) { + delete errors[fieldName]; + delete validFields[fieldName]; + if (this.formRules[fieldName]) { + const rule = this.formRules[fieldName]; + const fieldValue = this.state.schedulingUnit[fieldName]; + if (rule.required) { + if (!fieldValue) { + errors[fieldName] = rule.message?rule.message:`${fieldName} is required`; + } else { + validFields[fieldName] = true; + } + } + } + } else { + errors = {}; + validFields = {}; + for (const fieldName in this.formRules) { + const rule = this.formRules[fieldName]; + const fieldValue = this.state.schedulingUnit[fieldName]; + if (rule.required) { + if (!fieldValue) { + errors[fieldName] = rule.message?rule.message:`${fieldName} is required`; + } else { + validFields[fieldName] = true; + } + } + } + } + this.setState({errors: errors, validFields: validFields}); + if (Object.keys(validFields).length === Object.keys(this.formRules).length) { + validForm = true; + } + return validForm; + } + + /** + * Function to create Scheduling unit + */ + async saveSchedulingUnit() { + if (this.state.schedulingUnit.observation_strategy_template_id) { + let observStrategy = _.cloneDeep(this.state.observStrategy); + const $refs = await $RefParser.resolve(observStrategy.template); + observStrategy.template.parameters.forEach(async(param, index) => { + $refs.set(observStrategy.template.parameters[index]['refs'][0], this.state.paramsOutput['param_' + index]); + }); + const schedulingUnit = await ScheduleService.updateSUDraftFromObservStrategy(observStrategy, this.state.schedulingUnit, this.state.taskDrafts, this.state.tasksToUpdate); + if (schedulingUnit) { + // this.growl.show({severity: 'success', summary: 'Success', detail: 'Scheduling Unit and tasks edited successfully!'}); + this.props.history.push({ + pathname: `/schedulingunit/view/draft/${this.props.match.params.id}`, + }); + } else { + this.growl.show({severity: 'error', summary: 'Error Occured', detail: 'Unable to update Scheduling Unit/Tasks'}); + } + } else { + this.growl.show({severity: 'error', summary: 'Error Occured', detail: 'Template Missing.'}); + } + } + + /** + * Cancel SU creation and redirect + */ + cancelCreate() { + this.setState({redirect: `/schedulingunit/view/draft/${this.props.match.params.id}`}) + } + + render() { + if (this.state.redirect) { + return <Redirect to={ {pathname: this.state.redirect} }></Redirect> + } + + const schema = this.state.paramsSchema; + + let jeditor = null; + if (schema) { + jeditor = React.createElement(Jeditor, {title: "Task Parameters", + schema: schema, + initValue: this.state.paramsOutput, + callback: this.setEditorOutput, + parentFunction: this.setEditorFunction + }); + } + return ( + <React.Fragment> + <Growl ref={el => (this.growl = el)} /> + <PageHeader location={this.props.location} title={'Scheduling Unit - Edit'} + actions={[{icon: 'fa-window-close',title:'Click to Close Scheduling Unit View', props : { pathname: `/schedulingunit/view/draft/${this.props.match.params.id}`}}]}/> + { this.state.isLoading ? <AppLoader /> : + <> + <div> + <div className="p-fluid"> + <div className="p-field p-grid"> + <label htmlFor="schedUnitName" className="col-lg-2 col-md-2 col-sm-12">Name <span style={{color:'red'}}>*</span></label> + <div className="col-lg-3 col-md-3 col-sm-12"> + <InputText className={this.state.errors.name ?'input-error':''} id="schedUnitName" data-testid="name" + tooltip="Enter name of the Scheduling Unit" tooltipOptions={this.tooltipOptions} maxLength="128" + ref={input => {this.nameInput = input;}} + value={this.state.schedulingUnit.name} autoFocus + onChange={(e) => this.setSchedUnitParams('name', e.target.value)} + onBlur={(e) => this.setSchedUnitParams('name', e.target.value)}/> + <label className={this.state.errors.name?"error":"info"}> + {this.state.errors.name ? this.state.errors.name : "Max 128 characters"} + </label> + </div> + <div className="col-lg-1 col-md-1 col-sm-12"></div> + <label htmlFor="description" className="col-lg-2 col-md-2 col-sm-12">Description <span style={{color:'red'}}>*</span></label> + <div className="col-lg-3 col-md-3 col-sm-12"> + <InputTextarea className={this.state.errors.description ?'input-error':''} rows={3} cols={30} + tooltip="Longer description of the scheduling unit" tooltipOptions={this.tooltipOptions} maxLength="128" + data-testid="description" value={this.state.schedulingUnit.description} + onChange={(e) => this.setSchedUnitParams('description', e.target.value)} + onBlur={(e) => this.setSchedUnitParams('description', e.target.value)}/> + <label className={this.state.errors.description ?"error":"info"}> + {this.state.errors.description ? this.state.errors.description : "Max 255 characters"} + </label> + </div> + </div> + <div className="p-field p-grid"> + <label htmlFor="project" className="col-lg-2 col-md-2 col-sm-12">Project </label> + <div className="col-lg-3 col-md-3 col-sm-12" data-testid="project" > + <Dropdown inputId="project" optionLabel="name" optionValue="name" + tooltip="Project" tooltipOptions={this.tooltipOptions} + value={this.state.schedulingUnit.project} disabled={this.state.schedulingUnit.project?true:false} + options={this.projects} + placeholder="Select Project" /> + </div> + <div className="col-lg-1 col-md-1 col-sm-12"></div> + <label htmlFor="schedSet" className="col-lg-2 col-md-2 col-sm-12">Scheduling Set </label> + <div className="col-lg-3 col-md-3 col-sm-12"> + <Dropdown data-testid="schedSet" id="schedSet" optionLabel="name" optionValue="id" + tooltip="Scheduling set of the project" tooltipOptions={this.tooltipOptions} + value={this.state.schedulingUnit.scheduling_set_id} + options={this.state.schedulingSets} + disabled={this.state.schedulingUnit.scheduling_set_id?true:false} + placeholder="Select Scheduling Set" /> + </div> + </div> + <div className="p-field p-grid"> + { this.state.observStrategyVisible && + <> + <label htmlFor="observStrategy" className="col-lg-2 col-md-2 col-sm-12">Observation Strategy </label> + <div className="col-lg-3 col-md-3 col-sm-12" data-testid="observStrategy" > + <Dropdown inputId="observStrategy" optionLabel="name" optionValue="id" + tooltip="Observation Strategy Template to be used to create the Scheduling Unit and Tasks" tooltipOptions={this.tooltipOptions} + value={this.state.schedulingUnit.observation_strategy_template_id} + disabled={this.state.schedulingUnit.observation_strategy_template_id?true:false} + options={this.observStrategies} + onChange={(e) => {this.changeStrategy(e.value)}} + placeholder="Select Strategy" /> + </div> + </> + } + <div className="col-lg-1 col-md-1 col-sm-12"></div> + </div> + + </div> + <div className="p-fluid"> + <div className="p-grid"> + <div className="p-col-12"> + {this.state.paramsSchema?jeditor:""} + </div> + </div> + </div> + + <div className="p-grid p-justify-start"> + <div className="p-col-1"> + <Button label="Save" className="p-button-primary" icon="pi pi-check" onClick={this.saveSchedulingUnit} + disabled={!this.state.validEditor || !this.state.validForm} data-testid="save-btn" /> + </div> + <div className="p-col-1"> + <Button label="Cancel" className="p-button-danger" icon="pi pi-times" onClick={this.cancelCreate} /> + </div> + </div> + </div> + + </> + } + + </React.Fragment> + ); + } +} + +export default EditSchedulingUnit; \ No newline at end of file diff --git a/SAS/TMSS/frontend/tmss_webapp/src/routes/index.js b/SAS/TMSS/frontend/tmss_webapp/src/routes/index.js index 63e56d5b57ec44c2f90ac1daf48703f3084c3f23..6a637d5586126d3d29e3081bba0618c9c8556a05 100644 --- a/SAS/TMSS/frontend/tmss_webapp/src/routes/index.js +++ b/SAS/TMSS/frontend/tmss_webapp/src/routes/index.js @@ -12,6 +12,7 @@ import {Scheduling} from './Scheduling'; import {TaskEdit, TaskView, DataProduct} from './Task'; import ViewSchedulingUnit from './Scheduling/ViewSchedulingUnit' import SchedulingUnitCreate from './Scheduling/create'; +import EditSchedulingUnit from './Scheduling/edit'; import { CycleList, CycleCreate, CycleView, CycleEdit } from './Cycle'; export const routes = [ @@ -58,6 +59,11 @@ export const routes = [ component: ViewSchedulingUnit, name: 'Scheduling View', title: 'Scheduling Unit - Details' + },{ + path: "/schedulingunit/edit/:id", + component: EditSchedulingUnit, + name: 'Scheduling Edit', + title: 'Scheduling Unit - Edit' },{ path: "/schedulingunit/view/:type/:id", component: ViewSchedulingUnit, diff --git a/SAS/TMSS/frontend/tmss_webapp/src/services/schedule.service.js b/SAS/TMSS/frontend/tmss_webapp/src/services/schedule.service.js index f9479f4977a3f190cb0e1f7e221f3110068d959f..30a2c0db09bf506a44b47f156ba4f8b26ec2c3f6 100644 --- a/SAS/TMSS/frontend/tmss_webapp/src/services/schedule.service.js +++ b/SAS/TMSS/frontend/tmss_webapp/src/services/schedule.service.js @@ -217,12 +217,32 @@ const ScheduleService = { return null; }; }, + + updateSUDraftFromObservStrategy: async function(observStrategy, schedulingUnit,tasks,tasksToUpdate) { + try { + delete schedulingUnit['duration']; + schedulingUnit = await this.updateSchedulingUnitDraft(schedulingUnit); + for (const taskToUpdate in tasksToUpdate) { + let task = tasks.find(task => { return task.name === taskToUpdate}); + task.specifications_doc = observStrategy.template.tasks[taskToUpdate].specifications_doc; + delete task['duration']; + delete task['relative_start_time']; + delete task['relative_stop_time']; + task = await TaskService.updateTask('draft', task); + } + return schedulingUnit; + } catch(error) { + console.error(error); + return null; + }; + }, updateSchedulingUnitDraft: async function(schedulingUnit) { try { + console.log(schedulingUnit); const suUpdateResponse = await axios.put(`/api/scheduling_unit_draft/${schedulingUnit.id}/`, schedulingUnit); return suUpdateResponse.data; } catch(error) { - console.error(error); + console.error("Mistake",error); return null } }, @@ -282,4 +302,5 @@ const ScheduleService = { } } -export default ScheduleService; \ No newline at end of file + +export default ScheduleService;