2020-04-29 12:10:27 +02:00
|
|
|
import express from 'express';
|
2020-05-12 17:15:36 +02:00
|
|
|
import _ from 'lodash';
|
2020-04-29 12:10:27 +02:00
|
|
|
|
|
|
|
import MaterialValidate from './validate/material';
|
|
|
|
import MaterialModel from '../models/material'
|
2020-05-08 09:58:12 +02:00
|
|
|
import SampleModel from '../models/sample';
|
2020-05-28 17:05:23 +02:00
|
|
|
import MaterialGroupsModel from '../models/material_groups';
|
|
|
|
import MaterialSuppliersModel from '../models/material_suppliers';
|
2020-04-29 12:10:27 +02:00
|
|
|
import IdValidate from './validate/id';
|
2020-05-07 21:55:29 +02:00
|
|
|
import res400 from './validate/res400';
|
2020-05-08 09:58:12 +02:00
|
|
|
import mongoose from 'mongoose';
|
2020-05-27 14:31:17 +02:00
|
|
|
import globals from '../globals';
|
2020-04-29 12:10:27 +02:00
|
|
|
|
2020-05-13 17:28:18 +02:00
|
|
|
|
2020-04-29 12:10:27 +02:00
|
|
|
|
|
|
|
const router = express.Router();
|
|
|
|
|
|
|
|
router.get('/materials', (req, res, next) => {
|
|
|
|
if (!req.auth(res, ['read', 'write', 'maintain', 'dev', 'admin'], 'all')) return;
|
|
|
|
|
2020-05-28 17:05:23 +02:00
|
|
|
MaterialModel.find({status:globals.status.validated}).populate('group_id').populate('supplier_id').lean().exec((err, data) => {
|
2020-05-06 14:39:04 +02:00
|
|
|
if (err) return next(err);
|
2020-05-28 17:05:23 +02:00
|
|
|
console.log(data);
|
|
|
|
data.forEach((material: any) => { // map group and supplier
|
|
|
|
material.group = material.group_id.name;
|
|
|
|
material.supplier = material.supplier_id.name;
|
|
|
|
});
|
2020-05-12 17:15:36 +02:00
|
|
|
res.json(_.compact(data.map(e => MaterialValidate.output(e)))); // validate all and filter null values from validation errors
|
2020-04-29 12:10:27 +02:00
|
|
|
});
|
2020-05-18 10:43:26 +02:00
|
|
|
});
|
|
|
|
|
2020-05-28 17:05:23 +02:00
|
|
|
router.get('/materials/:state(new|deleted)', (req, res, next) => {
|
2020-05-18 10:43:26 +02:00
|
|
|
if (!req.auth(res, ['maintain', 'admin'], 'basic')) return;
|
|
|
|
|
2020-05-28 17:05:23 +02:00
|
|
|
MaterialModel.find({status: globals.status[req.params.state]}).populate('group_id').populate('supplier_id').lean().exec((err, data) => {
|
2020-05-18 10:43:26 +02:00
|
|
|
if (err) return next(err);
|
2020-05-28 17:05:23 +02:00
|
|
|
data.forEach((material: any) => { // map group and supplier
|
|
|
|
material.group = material.group_id.name;
|
|
|
|
material.supplier = material.supplier_id.name;
|
|
|
|
});
|
2020-05-18 10:43:26 +02:00
|
|
|
res.json(_.compact(data.map(e => MaterialValidate.output(e)))); // validate all and filter null values from validation errors
|
|
|
|
});
|
2020-04-29 12:10:27 +02:00
|
|
|
});
|
|
|
|
|
|
|
|
router.get('/material/' + IdValidate.parameter(), (req, res, next) => {
|
|
|
|
if (!req.auth(res, ['read', 'write', 'maintain', 'dev', 'admin'], 'all')) return;
|
|
|
|
|
2020-05-28 17:05:23 +02:00
|
|
|
MaterialModel.findById(req.params.id).populate('group_id').populate('supplier_id').lean().exec((err, data: any) => {
|
2020-05-06 14:39:04 +02:00
|
|
|
if (err) return next(err);
|
2020-05-28 14:11:19 +02:00
|
|
|
|
|
|
|
if (!data) {
|
|
|
|
return res.status(404).json({status: 'Not found'});
|
2020-04-29 12:10:27 +02:00
|
|
|
}
|
2020-05-28 17:05:23 +02:00
|
|
|
|
|
|
|
data.group = data.group_id.name;
|
|
|
|
data.supplier = data.supplier_id.name;
|
2020-05-28 14:11:19 +02:00
|
|
|
if (data.status === globals.status.deleted && !req.auth(res, ['maintain', 'admin'], 'all')) return; // deleted materials only available for maintain/admin
|
|
|
|
res.json(MaterialValidate.output(data));
|
2020-04-29 12:10:27 +02:00
|
|
|
});
|
|
|
|
});
|
|
|
|
|
2020-04-29 16:09:31 +02:00
|
|
|
router.put('/material/' + IdValidate.parameter(), (req, res, next) => {
|
|
|
|
if (!req.auth(res, ['write', 'maintain', 'dev', 'admin'], 'basic')) return;
|
|
|
|
|
|
|
|
const {error, value: material} = MaterialValidate.input(req.body, 'change');
|
2020-05-07 21:55:29 +02:00
|
|
|
if (error) return res400(error, res);
|
2020-04-29 16:09:31 +02:00
|
|
|
|
2020-05-13 14:18:15 +02:00
|
|
|
MaterialModel.findById(req.params.id).lean().exec(async (err, materialData: any) => {
|
|
|
|
if (!materialData) {
|
|
|
|
return res.status(404).json({status: 'Not found'});
|
|
|
|
}
|
2020-05-28 14:11:19 +02:00
|
|
|
if (materialData.status === globals.status.deleted) {
|
|
|
|
return res.status(403).json({status: 'Forbidden'});
|
|
|
|
}
|
2020-05-13 14:18:15 +02:00
|
|
|
if (material.hasOwnProperty('name') && material.name !== materialData.name) {
|
|
|
|
if (!await nameCheck(material, res, next)) return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// check for changes
|
|
|
|
if (!_.isEqual(_.pick(IdValidate.stringify(materialData), _.keys(material)), material)) {
|
2020-05-27 14:31:17 +02:00
|
|
|
material.status = globals.status.new; // set status to new
|
2020-05-13 14:18:15 +02:00
|
|
|
}
|
2020-04-29 16:09:31 +02:00
|
|
|
|
2020-05-13 14:18:15 +02:00
|
|
|
await MaterialModel.findByIdAndUpdate(req.params.id, material, {new: true}).lean().exec((err, data) => {
|
2020-05-06 14:39:04 +02:00
|
|
|
if (err) return next(err);
|
2020-05-13 14:18:15 +02:00
|
|
|
res.json(MaterialValidate.output(data));
|
2020-04-29 16:09:31 +02:00
|
|
|
});
|
2020-05-13 14:18:15 +02:00
|
|
|
});
|
2020-04-29 16:09:31 +02:00
|
|
|
});
|
|
|
|
|
|
|
|
router.delete('/material/' + IdValidate.parameter(), (req, res, next) => {
|
|
|
|
if (!req.auth(res, ['write', 'maintain', 'dev', 'admin'], 'basic')) return;
|
|
|
|
|
2020-05-08 09:58:12 +02:00
|
|
|
// check if there are still samples referencing this material
|
|
|
|
SampleModel.find({'material_id': new mongoose.Types.ObjectId(req.params.id)}).lean().exec((err, data) => {
|
2020-05-06 14:39:04 +02:00
|
|
|
if (err) return next(err);
|
2020-05-08 09:58:12 +02:00
|
|
|
if (data.length) {
|
|
|
|
return res.status(400).json({status: 'Material still in use'});
|
2020-04-29 16:09:31 +02:00
|
|
|
}
|
2020-05-27 14:31:17 +02:00
|
|
|
MaterialModel.findByIdAndUpdate(req.params.id, {status:globals.status.deleted}).lean().exec((err, data) => {
|
2020-05-08 09:58:12 +02:00
|
|
|
if (err) return next(err);
|
|
|
|
if (data) {
|
2020-05-08 15:12:36 +02:00
|
|
|
res.json({status: 'OK'});
|
2020-05-08 09:58:12 +02:00
|
|
|
}
|
|
|
|
else {
|
|
|
|
res.status(404).json({status: 'Not found'});
|
|
|
|
}
|
|
|
|
});
|
2020-04-29 16:09:31 +02:00
|
|
|
});
|
|
|
|
});
|
|
|
|
|
2020-05-28 14:54:52 +02:00
|
|
|
router.put('/material/restore/' + IdValidate.parameter(), (req, res, next) => {
|
|
|
|
if (!req.auth(res, ['maintain', 'admin'], 'basic')) return;
|
|
|
|
|
|
|
|
MaterialModel.findByIdAndUpdate(req.params.id, {status: globals.status.new}).lean().exec((err, data) => {
|
|
|
|
if (err) return next(err);
|
|
|
|
|
|
|
|
if (!data) {
|
|
|
|
return res.status(404).json({status: 'Not found'});
|
|
|
|
}
|
|
|
|
res.json({status: 'OK'});
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
2020-05-28 17:05:23 +02:00
|
|
|
router.post('/material/new', async (req, res, next) => { // TODO: check supplier and group, also for PUT and DELETE
|
2020-04-29 12:10:27 +02:00
|
|
|
if (!req.auth(res, ['write', 'maintain', 'dev', 'admin'], 'basic')) return;
|
|
|
|
|
|
|
|
const {error, value: material} = MaterialValidate.input(req.body, 'new');
|
2020-05-07 21:55:29 +02:00
|
|
|
if (error) return res400(error, res);
|
2020-04-29 12:10:27 +02:00
|
|
|
|
2020-05-13 14:18:15 +02:00
|
|
|
if (!await nameCheck(material, res, next)) return;
|
2020-04-29 12:10:27 +02:00
|
|
|
|
2020-05-27 14:31:17 +02:00
|
|
|
material.status = globals.status.new; // set status to new
|
2020-05-13 14:18:15 +02:00
|
|
|
await new MaterialModel(material).save((err, data) => {
|
|
|
|
if (err) return next(err);
|
|
|
|
res.json(MaterialValidate.output(data.toObject()));
|
2020-04-29 16:09:31 +02:00
|
|
|
});
|
|
|
|
});
|
2020-04-29 12:10:27 +02:00
|
|
|
|
2020-05-28 17:05:23 +02:00
|
|
|
router.get('/material/groups', (req, res, next) => {
|
|
|
|
if (!req.auth(res, ['read', 'write', 'maintain', 'dev', 'admin'], 'all')) return;
|
|
|
|
|
|
|
|
MaterialGroupsModel.find().lean().exec((err, data: any) => {
|
|
|
|
if (err) return next(err);
|
|
|
|
|
|
|
|
res.json(_.compact(data.map(e => MaterialValidate.outputGroups(e.name)))); // validate all and filter null values from validation errors
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
router.get('/material/suppliers', (req, res, next) => {
|
|
|
|
if (!req.auth(res, ['read', 'write', 'maintain', 'dev', 'admin'], 'all')) return;
|
|
|
|
|
|
|
|
MaterialSuppliersModel.find().lean().exec((err, data: any) => {
|
|
|
|
if (err) return next(err);
|
|
|
|
|
|
|
|
res.json(_.compact(data.map(e => MaterialValidate.outputSuppliers(e.name)))); // validate all and filter null values from validation errors
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
2020-04-29 12:10:27 +02:00
|
|
|
|
2020-05-13 14:18:15 +02:00
|
|
|
module.exports = router;
|
|
|
|
|
|
|
|
|
|
|
|
async function nameCheck (material, res, next) { // check if name was already taken
|
2020-05-18 14:47:22 +02:00
|
|
|
const materialData = await MaterialModel.findOne({name: material.name}).lean().exec().catch(err => next(err)) as any;
|
2020-05-14 15:36:47 +02:00
|
|
|
if (materialData instanceof Error) return false;
|
2020-05-13 14:18:15 +02:00
|
|
|
if (materialData) { // could not find material_id
|
|
|
|
res.status(400).json({status: 'Material name already taken'});
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|