Merge pull request #7 in ~VLE2FE/dfop-api from measurement to develop
* commit '5bce7a1e988003a9b88bf9b5e6fdac801832b8f8': DELETE method PUT method added status to POST method GET and POST method for measurement
This commit is contained in:
		@@ -27,6 +27,13 @@ info:
 | 
				
			|||||||
      <li>no whitespace</li>
 | 
					      <li>no whitespace</li>
 | 
				
			||||||
      <li>at least 8 characters</li>
 | 
					      <li>at least 8 characters</li>
 | 
				
			||||||
    </ul>
 | 
					    </ul>
 | 
				
			||||||
 | 
					  x-doc: |
 | 
				
			||||||
 | 
					    status:
 | 
				
			||||||
 | 
					    <ul>
 | 
				
			||||||
 | 
					      <li>-10: deleted</li>
 | 
				
			||||||
 | 
					      <li>0: newly added/changed</li>
 | 
				
			||||||
 | 
					      <li>10: validated</li>
 | 
				
			||||||
 | 
					    </ul>
 | 
				
			||||||
# TODO: Link to new documentation page
 | 
					# TODO: Link to new documentation page
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -22,8 +22,9 @@
 | 
				
			|||||||
      500:
 | 
					      500:
 | 
				
			||||||
        $ref: 'api.yaml#/components/responses/500'
 | 
					        $ref: 'api.yaml#/components/responses/500'
 | 
				
			||||||
  put:
 | 
					  put:
 | 
				
			||||||
    summary: TODO change measurement
 | 
					    summary: change measurement
 | 
				
			||||||
    description: 'Auth: basic, levels: write, maintain, dev, admin'
 | 
					    description: 'Auth: basic, levels: write, maintain, dev, admin'
 | 
				
			||||||
 | 
					    x-doc: status is reset to 0 on any changes
 | 
				
			||||||
    tags:
 | 
					    tags:
 | 
				
			||||||
      - /measurement
 | 
					      - /measurement
 | 
				
			||||||
    security:
 | 
					    security:
 | 
				
			||||||
@@ -33,7 +34,9 @@
 | 
				
			|||||||
      content:
 | 
					      content:
 | 
				
			||||||
        application/json:
 | 
					        application/json:
 | 
				
			||||||
          schema:
 | 
					          schema:
 | 
				
			||||||
            $ref: 'api.yaml#/components/schemas/Measurement'
 | 
					            properties:
 | 
				
			||||||
 | 
					              values:
 | 
				
			||||||
 | 
					                type: object
 | 
				
			||||||
    responses:
 | 
					    responses:
 | 
				
			||||||
      200:
 | 
					      200:
 | 
				
			||||||
        description: measurement details
 | 
					        description: measurement details
 | 
				
			||||||
@@ -52,7 +55,7 @@
 | 
				
			|||||||
      500:
 | 
					      500:
 | 
				
			||||||
        $ref: 'api.yaml#/components/responses/500'
 | 
					        $ref: 'api.yaml#/components/responses/500'
 | 
				
			||||||
  delete:
 | 
					  delete:
 | 
				
			||||||
    summary: TODO delete measurement
 | 
					    summary: delete measurement
 | 
				
			||||||
    description: 'Auth: basic, levels: write, maintain, dev, admin'
 | 
					    description: 'Auth: basic, levels: write, maintain, dev, admin'
 | 
				
			||||||
    tags:
 | 
					    tags:
 | 
				
			||||||
      - /measurement
 | 
					      - /measurement
 | 
				
			||||||
@@ -74,7 +77,7 @@
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
/measurement/new:
 | 
					/measurement/new:
 | 
				
			||||||
  post:
 | 
					  post:
 | 
				
			||||||
    summary: TODO add measurement
 | 
					    summary: add measurement
 | 
				
			||||||
    description: 'Auth: basic, levels: write, maintain, dev, admin'
 | 
					    description: 'Auth: basic, levels: write, maintain, dev, admin'
 | 
				
			||||||
    x-doc: 'Adds status: 0 automatically'
 | 
					    x-doc: 'Adds status: 0 automatically'
 | 
				
			||||||
    tags:
 | 
					    tags:
 | 
				
			||||||
 
 | 
				
			|||||||
							
								
								
									
										9
									
								
								package-lock.json
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										9
									
								
								package-lock.json
									
									
									
										generated
									
									
									
								
							@@ -137,6 +137,12 @@
 | 
				
			|||||||
        "@types/range-parser": "*"
 | 
					        "@types/range-parser": "*"
 | 
				
			||||||
      }
 | 
					      }
 | 
				
			||||||
    },
 | 
					    },
 | 
				
			||||||
 | 
					    "@types/lodash": {
 | 
				
			||||||
 | 
					      "version": "4.14.150",
 | 
				
			||||||
 | 
					      "resolved": "https://registry.npmjs.org/@types/lodash/-/lodash-4.14.150.tgz",
 | 
				
			||||||
 | 
					      "integrity": "sha512-kMNLM5JBcasgYscD9x/Gvr6lTAv2NVgsKtet/hm93qMyf/D1pt+7jeEZklKJKxMVmXjxbRVQQGfqDSfipYCO6w==",
 | 
				
			||||||
 | 
					      "dev": true
 | 
				
			||||||
 | 
					    },
 | 
				
			||||||
    "@types/mime": {
 | 
					    "@types/mime": {
 | 
				
			||||||
      "version": "2.0.1",
 | 
					      "version": "2.0.1",
 | 
				
			||||||
      "resolved": "https://registry.npmjs.org/@types/mime/-/mime-2.0.1.tgz",
 | 
					      "resolved": "https://registry.npmjs.org/@types/mime/-/mime-2.0.1.tgz",
 | 
				
			||||||
@@ -1345,8 +1351,7 @@
 | 
				
			|||||||
    "lodash": {
 | 
					    "lodash": {
 | 
				
			||||||
      "version": "4.17.15",
 | 
					      "version": "4.17.15",
 | 
				
			||||||
      "resolved": "https://registry.npmjs.org/lodash/-/lodash-4.17.15.tgz",
 | 
					      "resolved": "https://registry.npmjs.org/lodash/-/lodash-4.17.15.tgz",
 | 
				
			||||||
      "integrity": "sha512-8xOcRHvCjnocdS5cpwXQXVzmmh5e5+saE2QGoeQmbKmRS6J3VQppPOIt0MnmE+4xlZoumy0GPG0D0MVIQbNA1A==",
 | 
					      "integrity": "sha512-8xOcRHvCjnocdS5cpwXQXVzmmh5e5+saE2QGoeQmbKmRS6J3VQppPOIt0MnmE+4xlZoumy0GPG0D0MVIQbNA1A=="
 | 
				
			||||||
      "dev": true
 | 
					 | 
				
			||||||
    },
 | 
					    },
 | 
				
			||||||
    "log-symbols": {
 | 
					    "log-symbols": {
 | 
				
			||||||
      "version": "3.0.0",
 | 
					      "version": "3.0.0",
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -32,6 +32,7 @@
 | 
				
			|||||||
    "content-filter": "^1.1.2",
 | 
					    "content-filter": "^1.1.2",
 | 
				
			||||||
    "express": "^4.17.1",
 | 
					    "express": "^4.17.1",
 | 
				
			||||||
    "json-schema": "^0.2.5",
 | 
					    "json-schema": "^0.2.5",
 | 
				
			||||||
 | 
					    "lodash": "^4.17.15",
 | 
				
			||||||
    "mongo-sanitize": "^1.1.0",
 | 
					    "mongo-sanitize": "^1.1.0",
 | 
				
			||||||
    "mongoose": "^5.8.7",
 | 
					    "mongoose": "^5.8.7",
 | 
				
			||||||
    "nodemon": "^2.0.3",
 | 
					    "nodemon": "^2.0.3",
 | 
				
			||||||
@@ -40,6 +41,7 @@
 | 
				
			|||||||
    "typescript": "^3.7.4"
 | 
					    "typescript": "^3.7.4"
 | 
				
			||||||
  },
 | 
					  },
 | 
				
			||||||
  "devDependencies": {
 | 
					  "devDependencies": {
 | 
				
			||||||
 | 
					    "@types/lodash": "^4.14.150",
 | 
				
			||||||
    "mocha": "^7.1.2",
 | 
					    "mocha": "^7.1.2",
 | 
				
			||||||
    "should": "^13.2.3",
 | 
					    "should": "^13.2.3",
 | 
				
			||||||
    "supertest": "^4.0.2"
 | 
					    "supertest": "^4.0.2"
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -49,6 +49,7 @@ app.use('/', require('./routes/material'));
 | 
				
			|||||||
app.use('/', require('./routes/template'));
 | 
					app.use('/', require('./routes/template'));
 | 
				
			||||||
app.use('/', require('./routes/user'));
 | 
					app.use('/', require('./routes/user'));
 | 
				
			||||||
app.use('/', require('./routes/condition'));
 | 
					app.use('/', require('./routes/condition'));
 | 
				
			||||||
 | 
					app.use('/', require('./routes/measurement'));
 | 
				
			||||||
 | 
					
 | 
				
			||||||
// static files
 | 
					// static files
 | 
				
			||||||
app.use('/static', express.static('static'));
 | 
					app.use('/static', express.static('static'));
 | 
				
			||||||
 
 | 
				
			|||||||
							
								
								
									
										12
									
								
								src/models/measurement.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										12
									
								
								src/models/measurement.ts
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,12 @@
 | 
				
			|||||||
 | 
					import mongoose from 'mongoose';
 | 
				
			||||||
 | 
					import ConditionModel from './condition';
 | 
				
			||||||
 | 
					import MeasurementTemplateModel from './measurement_template';
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					const MeasurementSchema = new mongoose.Schema({
 | 
				
			||||||
 | 
					  condition_id: {type: mongoose.Schema.Types.ObjectId, ref: ConditionModel},
 | 
				
			||||||
 | 
					  values: mongoose.Schema.Types.Mixed,
 | 
				
			||||||
 | 
					  status: Number,
 | 
				
			||||||
 | 
					  measurement_template: {type: mongoose.Schema.Types.ObjectId, ref: MeasurementTemplateModel}
 | 
				
			||||||
 | 
					});
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					export default mongoose.model('measurement', MeasurementSchema);
 | 
				
			||||||
@@ -1,7 +1,7 @@
 | 
				
			|||||||
import should from 'should/as-function';
 | 
					import should from 'should/as-function';
 | 
				
			||||||
import ConditionModel from '../models/condition';
 | 
					import ConditionModel from '../models/condition';
 | 
				
			||||||
import TestHelper from "../test/helper";
 | 
					import TestHelper from "../test/helper";
 | 
				
			||||||
 | 
					// TODO: status
 | 
				
			||||||
 | 
					
 | 
				
			||||||
describe('/condition', () => {
 | 
					describe('/condition', () => {
 | 
				
			||||||
  let server;
 | 
					  let server;
 | 
				
			||||||
@@ -184,7 +184,7 @@ describe('/condition', () => {
 | 
				
			|||||||
        req: {parameters: {material: 'hot air', weeks: 10}}
 | 
					        req: {parameters: {material: 'hot air', weeks: 10}}
 | 
				
			||||||
      });
 | 
					      });
 | 
				
			||||||
    });
 | 
					    });
 | 
				
			||||||
    it('rejects requests form a read user', done => {
 | 
					    it('rejects requests from a read user', done => {
 | 
				
			||||||
      TestHelper.request(server, done, {
 | 
					      TestHelper.request(server, done, {
 | 
				
			||||||
        method: 'put',
 | 
					        method: 'put',
 | 
				
			||||||
        url: '/condition/700000000000000000000001',
 | 
					        url: '/condition/700000000000000000000001',
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -48,8 +48,7 @@ router.put('/condition/' + IdValidate.parameter(), async (req, res, next) => {
 | 
				
			|||||||
  }
 | 
					  }
 | 
				
			||||||
  if (!await treatmentCheck(condition, 'change', res, next)) return;
 | 
					  if (!await treatmentCheck(condition, 'change', res, next)) return;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  console.log(condition);
 | 
					  await ConditionModel.findByIdAndUpdate(req.params.id, condition, {new: true}).lean().exec((err, data) => {
 | 
				
			||||||
  ConditionModel.findByIdAndUpdate(req.params.id, condition, {new: true}).lean().exec((err, data) => {
 | 
					 | 
				
			||||||
    if (err) return next(err);
 | 
					    if (err) return next(err);
 | 
				
			||||||
    res.json(ConditionValidate.output(data));
 | 
					    res.json(ConditionValidate.output(data));
 | 
				
			||||||
  });
 | 
					  });
 | 
				
			||||||
@@ -64,7 +63,7 @@ router.delete('/condition/' + IdValidate.parameter(), (req, res, next) => {
 | 
				
			|||||||
      res.status(404).json({status: 'Not found'});
 | 
					      res.status(404).json({status: 'Not found'});
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
    if (!await sampleIdCheck(data, req, res, next)) return;
 | 
					    if (!await sampleIdCheck(data, req, res, next)) return;
 | 
				
			||||||
    ConditionModel.findByIdAndDelete(req.params.id).lean().exec(async err => {
 | 
					    await ConditionModel.findByIdAndDelete(req.params.id).lean().exec(async err => {
 | 
				
			||||||
      if (err) return next(err);
 | 
					      if (err) return next(err);
 | 
				
			||||||
      res.json({status: 'OK'});
 | 
					      res.json({status: 'OK'});
 | 
				
			||||||
    });
 | 
					    });
 | 
				
			||||||
@@ -81,7 +80,7 @@ router.post('/condition/new', async (req, res, next) => {
 | 
				
			|||||||
  if (!await numberCheck(condition, res, next)) return;
 | 
					  if (!await numberCheck(condition, res, next)) return;
 | 
				
			||||||
  if (!await treatmentCheck(condition, 'new', res, next)) return;
 | 
					  if (!await treatmentCheck(condition, 'new', res, next)) return;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  new ConditionModel(condition).save((err, data) => {
 | 
					  await new ConditionModel(condition).save((err, data) => {
 | 
				
			||||||
    if (err) return next(err);
 | 
					    if (err) return next(err);
 | 
				
			||||||
    res.json(ConditionValidate.output(data.toObject()));
 | 
					    res.json(ConditionValidate.output(data.toObject()));
 | 
				
			||||||
  });
 | 
					  });
 | 
				
			||||||
@@ -113,7 +112,7 @@ async function numberCheck (condition, res, next) {  // validate number, returns
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
async function treatmentCheck (condition, param, res, next) {
 | 
					async function treatmentCheck (condition, param, res, next) {
 | 
				
			||||||
  const treatmentData = await TreatmentTemplateModel.findById(condition.treatment_template).lean().exec().catch(err => {next(err); return false;}) as any;
 | 
					  const treatmentData = await TreatmentTemplateModel.findById(condition.treatment_template).lean().exec().catch(err => {next(err); return false;}) as any;
 | 
				
			||||||
  if (!treatmentData) {  // sample_id not found
 | 
					  if (!treatmentData) {  // template not found
 | 
				
			||||||
    res.status(400).json({status: 'Treatment template not available'});
 | 
					    res.status(400).json({status: 'Treatment template not available'});
 | 
				
			||||||
    return false
 | 
					    return false
 | 
				
			||||||
  }
 | 
					  }
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -1,7 +1,7 @@
 | 
				
			|||||||
import should from 'should/as-function';
 | 
					import should from 'should/as-function';
 | 
				
			||||||
import MaterialModel from '../models/material';
 | 
					import MaterialModel from '../models/material';
 | 
				
			||||||
import TestHelper from "../test/helper";
 | 
					import TestHelper from "../test/helper";
 | 
				
			||||||
 | 
					// TODO: status
 | 
				
			||||||
 | 
					
 | 
				
			||||||
describe('/material', () => {
 | 
					describe('/material', () => {
 | 
				
			||||||
  let server;
 | 
					  let server;
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -1,4 +1,5 @@
 | 
				
			|||||||
import express from 'express';
 | 
					import express from 'express';
 | 
				
			||||||
 | 
					import _ from 'lodash';
 | 
				
			||||||
 | 
					
 | 
				
			||||||
import MaterialValidate from './validate/material';
 | 
					import MaterialValidate from './validate/material';
 | 
				
			||||||
import MaterialModel from '../models/material'
 | 
					import MaterialModel from '../models/material'
 | 
				
			||||||
@@ -15,7 +16,7 @@ router.get('/materials', (req, res, next) => {
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
  MaterialModel.find({}).lean().exec((err, data) => {
 | 
					  MaterialModel.find({}).lean().exec((err, data) => {
 | 
				
			||||||
    if (err) return next(err);
 | 
					    if (err) return next(err);
 | 
				
			||||||
    res.json(data.map(e => MaterialValidate.output(e)).filter(e => e !== null));  // validate all and filter null values from validation errors
 | 
					    res.json(_.compact(data.map(e => MaterialValidate.output(e))));  // validate all and filter null values from validation errors
 | 
				
			||||||
  });
 | 
					  });
 | 
				
			||||||
});
 | 
					});
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 
 | 
				
			|||||||
							
								
								
									
										508
									
								
								src/routes/measurement.spec.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										508
									
								
								src/routes/measurement.spec.ts
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,508 @@
 | 
				
			|||||||
 | 
					import should from 'should/as-function';
 | 
				
			||||||
 | 
					import MeasurementModel from '../models/measurement';
 | 
				
			||||||
 | 
					import TestHelper from "../test/helper";
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					describe('/measurement', () => {
 | 
				
			||||||
 | 
					  let server;
 | 
				
			||||||
 | 
					  before(done => TestHelper.before(done));
 | 
				
			||||||
 | 
					  beforeEach(done => server = TestHelper.beforeEach(server, done));
 | 
				
			||||||
 | 
					  afterEach(done => TestHelper.afterEach(server, done));
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  describe('GET /mesurement/{id}', () => {
 | 
				
			||||||
 | 
					    it('returns the right measurement', done => {
 | 
				
			||||||
 | 
					      TestHelper.request(server, done, {
 | 
				
			||||||
 | 
					        method: 'get',
 | 
				
			||||||
 | 
					        url: '/measurement/800000000000000000000001',
 | 
				
			||||||
 | 
					        auth: {basic: 'janedoe'},
 | 
				
			||||||
 | 
					        httpStatus: 200,
 | 
				
			||||||
 | 
					        res: {_id: '800000000000000000000001', condition_id: '700000000000000000000001', values: {dpt: [[3997.12558,98.00555],[3995.08519,98.03253],[3993.04480,98.02657]]}, measurement_template: '300000000000000000000001'}
 | 
				
			||||||
 | 
					      });
 | 
				
			||||||
 | 
					    });
 | 
				
			||||||
 | 
					    it('returns the measurement for an API key', done => {
 | 
				
			||||||
 | 
					      TestHelper.request(server, done, {
 | 
				
			||||||
 | 
					        method: 'get',
 | 
				
			||||||
 | 
					        url: '/measurement/800000000000000000000001',
 | 
				
			||||||
 | 
					        auth: {key: 'janedoe'},
 | 
				
			||||||
 | 
					        httpStatus: 200,
 | 
				
			||||||
 | 
					        res: {_id: '800000000000000000000001', condition_id: '700000000000000000000001', values: {dpt: [[3997.12558,98.00555],[3995.08519,98.03253],[3993.04480,98.02657]]}, measurement_template: '300000000000000000000001'}
 | 
				
			||||||
 | 
					      });
 | 
				
			||||||
 | 
					    });
 | 
				
			||||||
 | 
					    it('rejects an invalid id', done => {
 | 
				
			||||||
 | 
					      TestHelper.request(server, done, {
 | 
				
			||||||
 | 
					        method: 'get',
 | 
				
			||||||
 | 
					        url: '/measurement/8000000000h0000000000001',
 | 
				
			||||||
 | 
					        auth: {basic: 'janedoe'},
 | 
				
			||||||
 | 
					        httpStatus: 404
 | 
				
			||||||
 | 
					      });
 | 
				
			||||||
 | 
					    });
 | 
				
			||||||
 | 
					    it('rejects an unknown id', done => {
 | 
				
			||||||
 | 
					      TestHelper.request(server, done, {
 | 
				
			||||||
 | 
					        method: 'get',
 | 
				
			||||||
 | 
					        url: '/measurement/000000000000000000000001',
 | 
				
			||||||
 | 
					        auth: {basic: 'janedoe'},
 | 
				
			||||||
 | 
					        httpStatus: 404
 | 
				
			||||||
 | 
					      });
 | 
				
			||||||
 | 
					    });
 | 
				
			||||||
 | 
					    it('rejects unauthorized requests', done => {
 | 
				
			||||||
 | 
					      TestHelper.request(server, done, {
 | 
				
			||||||
 | 
					        method: 'get',
 | 
				
			||||||
 | 
					        url: '/measurement/800000000000000000000001',
 | 
				
			||||||
 | 
					        httpStatus: 401
 | 
				
			||||||
 | 
					      });
 | 
				
			||||||
 | 
					    });
 | 
				
			||||||
 | 
					  });
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  describe('PUT /measurement/{id}', () => {
 | 
				
			||||||
 | 
					    it('returns the right measurement', done => {
 | 
				
			||||||
 | 
					      TestHelper.request(server, done, {
 | 
				
			||||||
 | 
					        method: 'put',
 | 
				
			||||||
 | 
					        url: '/measurement/800000000000000000000001',
 | 
				
			||||||
 | 
					        auth: {basic: 'janedoe'},
 | 
				
			||||||
 | 
					        httpStatus: 200,
 | 
				
			||||||
 | 
					        req: {},
 | 
				
			||||||
 | 
					        res: {_id: '800000000000000000000001', condition_id: '700000000000000000000001', values: {dpt: [[3997.12558,98.00555],[3995.08519,98.03253],[3993.04480,98.02657]]}, measurement_template: '300000000000000000000001'}
 | 
				
			||||||
 | 
					      });
 | 
				
			||||||
 | 
					    });
 | 
				
			||||||
 | 
					    it('keeps unchanged values', done => {
 | 
				
			||||||
 | 
					      TestHelper.request(server, done, {
 | 
				
			||||||
 | 
					        method: 'put',
 | 
				
			||||||
 | 
					        url: '/measurement/800000000000000000000001',
 | 
				
			||||||
 | 
					        auth: {basic: 'janedoe'},
 | 
				
			||||||
 | 
					        httpStatus: 200,
 | 
				
			||||||
 | 
					        req: {values: {dpt: [[3997.12558,98.00555],[3995.08519,98.03253],[3993.04480,98.02657]]}}
 | 
				
			||||||
 | 
					      }).end((err, res) => {
 | 
				
			||||||
 | 
					        if (err) return done(err);
 | 
				
			||||||
 | 
					        should(res.body).be.eql({_id: '800000000000000000000001', condition_id: '700000000000000000000001', values: {dpt: [[3997.12558,98.00555],[3995.08519,98.03253],[3993.04480,98.02657]]}, measurement_template: '300000000000000000000001'});
 | 
				
			||||||
 | 
					        MeasurementModel.findById('800000000000000000000001').lean().exec((err, data: any) => {
 | 
				
			||||||
 | 
					          should(data).have.property('status', 10);
 | 
				
			||||||
 | 
					          done();
 | 
				
			||||||
 | 
					        });
 | 
				
			||||||
 | 
					      });
 | 
				
			||||||
 | 
					    });
 | 
				
			||||||
 | 
					    it('changes the given values', done => {
 | 
				
			||||||
 | 
					      TestHelper.request(server, done, {
 | 
				
			||||||
 | 
					        method: 'put',
 | 
				
			||||||
 | 
					        url: '/measurement/800000000000000000000001',
 | 
				
			||||||
 | 
					        auth: {basic: 'janedoe'},
 | 
				
			||||||
 | 
					        httpStatus: 200,
 | 
				
			||||||
 | 
					        req: {values: {dpt: [[1,2],[3,4],[5,6]]}}
 | 
				
			||||||
 | 
					      }).end((err, res) => {
 | 
				
			||||||
 | 
					        if (err) return done(err);
 | 
				
			||||||
 | 
					        should(res.body).be.eql({_id: '800000000000000000000001', condition_id: '700000000000000000000001', values: {dpt: [[1,2],[3,4],[5,6]]}, measurement_template: '300000000000000000000001'});
 | 
				
			||||||
 | 
					        MeasurementModel.findById('800000000000000000000001').lean().exec((err, data: any) => {
 | 
				
			||||||
 | 
					          should(data).have.only.keys('_id', 'condition_id', 'values', 'measurement_template', 'status', '__v');
 | 
				
			||||||
 | 
					          should(data.condition_id.toString()).be.eql('700000000000000000000001');
 | 
				
			||||||
 | 
					          should(data.measurement_template.toString()).be.eql('300000000000000000000001');
 | 
				
			||||||
 | 
					          should(data).have.property('status', 0);
 | 
				
			||||||
 | 
					          should(data).have.property('values');
 | 
				
			||||||
 | 
					          should(data.values).have.property('dpt', [[1,2],[3,4],[5,6]]);
 | 
				
			||||||
 | 
					          done();
 | 
				
			||||||
 | 
					        });
 | 
				
			||||||
 | 
					      });
 | 
				
			||||||
 | 
					    });
 | 
				
			||||||
 | 
					    it('allows changing only one value', done => {
 | 
				
			||||||
 | 
					      TestHelper.request(server, done, {
 | 
				
			||||||
 | 
					        method: 'put',
 | 
				
			||||||
 | 
					        url: '/measurement/800000000000000000000002',
 | 
				
			||||||
 | 
					        auth: {basic: 'janedoe'},
 | 
				
			||||||
 | 
					        httpStatus: 200,
 | 
				
			||||||
 | 
					        req: {values: {'weight %': 0.9}},
 | 
				
			||||||
 | 
					        res: {_id: '800000000000000000000002', condition_id: '700000000000000000000002', values: {'weight %': 0.9, 'standard deviation': 0.2}, measurement_template: '300000000000000000000002'}
 | 
				
			||||||
 | 
					      });
 | 
				
			||||||
 | 
					    });
 | 
				
			||||||
 | 
					    it('rejects not specified values', done => {
 | 
				
			||||||
 | 
					      TestHelper.request(server, done, {
 | 
				
			||||||
 | 
					        method: 'put',
 | 
				
			||||||
 | 
					        url: '/measurement/800000000000000000000002',
 | 
				
			||||||
 | 
					        auth: {basic: 'janedoe'},
 | 
				
			||||||
 | 
					        httpStatus: 400,
 | 
				
			||||||
 | 
					        req: {values: {'weight %': 0.9, 'standard deviation': 0.3, xx: 44}},
 | 
				
			||||||
 | 
					        res: {status: 'Invalid body format', details: '"xx" is not allowed'}
 | 
				
			||||||
 | 
					      });
 | 
				
			||||||
 | 
					    });
 | 
				
			||||||
 | 
					    it('rejects a value not in the value range', done => {
 | 
				
			||||||
 | 
					      TestHelper.request(server, done, {
 | 
				
			||||||
 | 
					        method: 'put',
 | 
				
			||||||
 | 
					        url: '/measurement/800000000000000000000003',
 | 
				
			||||||
 | 
					        auth: {basic: 'admin'},
 | 
				
			||||||
 | 
					        httpStatus: 400,
 | 
				
			||||||
 | 
					        req: {values: {val1: 4}},
 | 
				
			||||||
 | 
					        res: {status: 'Invalid body format', details: '"val1" must be one of [1, 2, 3]'}
 | 
				
			||||||
 | 
					      });
 | 
				
			||||||
 | 
					    });
 | 
				
			||||||
 | 
					    it('rejects a value below minimum range', done => {
 | 
				
			||||||
 | 
					      TestHelper.request(server, done, {
 | 
				
			||||||
 | 
					        method: 'put',
 | 
				
			||||||
 | 
					        url: '/measurement/800000000000000000000002',
 | 
				
			||||||
 | 
					        auth: {basic: 'janedoe'},
 | 
				
			||||||
 | 
					        httpStatus: 400,
 | 
				
			||||||
 | 
					        req: {values: {'weight %': -1, 'standard deviation': 0.3}},
 | 
				
			||||||
 | 
					        res: {status: 'Invalid body format', details: '"weight %" must be larger than or equal to 0'}
 | 
				
			||||||
 | 
					      });
 | 
				
			||||||
 | 
					    });
 | 
				
			||||||
 | 
					    it('rejects a value above maximum range', done => {
 | 
				
			||||||
 | 
					      TestHelper.request(server, done, {
 | 
				
			||||||
 | 
					        method: 'put',
 | 
				
			||||||
 | 
					        url: '/measurement/800000000000000000000002',
 | 
				
			||||||
 | 
					        auth: {basic: 'janedoe'},
 | 
				
			||||||
 | 
					        httpStatus: 400,
 | 
				
			||||||
 | 
					        req: {values: {'weight %': 0.9, 'standard deviation': 3}},
 | 
				
			||||||
 | 
					        res: {status: 'Invalid body format', details: '"standard deviation" must be less than or equal to 0.5'}
 | 
				
			||||||
 | 
					      });
 | 
				
			||||||
 | 
					    });
 | 
				
			||||||
 | 
					    it('rejects a new measurement template', done => {
 | 
				
			||||||
 | 
					      TestHelper.request(server, done, {
 | 
				
			||||||
 | 
					        method: 'put',
 | 
				
			||||||
 | 
					        url: '/measurement/800000000000000000000002',
 | 
				
			||||||
 | 
					        auth: {basic: 'janedoe'},
 | 
				
			||||||
 | 
					        httpStatus: 400,
 | 
				
			||||||
 | 
					        req: {values: {'weight %': 0.9, 'standard deviation': 0.3}, measurement_template: '300000000000000000000001'},
 | 
				
			||||||
 | 
					        res: {status: 'Invalid body format', details: '"measurement_template" is not allowed'}
 | 
				
			||||||
 | 
					      });
 | 
				
			||||||
 | 
					    });
 | 
				
			||||||
 | 
					    it('rejects editing a measurement for a write user who did not create this measurement', done => {
 | 
				
			||||||
 | 
					      TestHelper.request(server, done, {
 | 
				
			||||||
 | 
					        method: 'put',
 | 
				
			||||||
 | 
					        url: '/measurement/800000000000000000000003',
 | 
				
			||||||
 | 
					        auth: {basic: 'janedoe'},
 | 
				
			||||||
 | 
					        httpStatus: 403,
 | 
				
			||||||
 | 
					        req: {values: {val1: 2}}
 | 
				
			||||||
 | 
					      });
 | 
				
			||||||
 | 
					    });
 | 
				
			||||||
 | 
					    it('accepts editing a measurement of another user for a maintain/admin user', done => {
 | 
				
			||||||
 | 
					      TestHelper.request(server, done, {
 | 
				
			||||||
 | 
					        method: 'put',
 | 
				
			||||||
 | 
					        url: '/measurement/800000000000000000000002',
 | 
				
			||||||
 | 
					        auth: {basic: 'admin'},
 | 
				
			||||||
 | 
					        httpStatus: 200,
 | 
				
			||||||
 | 
					        req: {values: {'weight %': 0.9, 'standard deviation': 0.3}},
 | 
				
			||||||
 | 
					        res: {_id: '800000000000000000000002', condition_id: '700000000000000000000002', values: {'weight %': 0.9, 'standard deviation': 0.3}, measurement_template: '300000000000000000000002'}
 | 
				
			||||||
 | 
					      });
 | 
				
			||||||
 | 
					    });
 | 
				
			||||||
 | 
					    it('rejects an invalid id', done => {
 | 
				
			||||||
 | 
					      TestHelper.request(server, done, {
 | 
				
			||||||
 | 
					        method: 'put',
 | 
				
			||||||
 | 
					        url: '/measurement/800000000h00000000000002',
 | 
				
			||||||
 | 
					        auth: {basic: 'janedoe'},
 | 
				
			||||||
 | 
					        httpStatus: 404
 | 
				
			||||||
 | 
					      });
 | 
				
			||||||
 | 
					    });
 | 
				
			||||||
 | 
					    it('rejects an unknown id', done => {
 | 
				
			||||||
 | 
					      TestHelper.request(server, done, {
 | 
				
			||||||
 | 
					        method: 'put',
 | 
				
			||||||
 | 
					        url: '/measurement/000000000000000000000002',
 | 
				
			||||||
 | 
					        auth: {basic: 'janedoe'},
 | 
				
			||||||
 | 
					        httpStatus: 404
 | 
				
			||||||
 | 
					      });
 | 
				
			||||||
 | 
					    });
 | 
				
			||||||
 | 
					    it('rejects an API key', done => {
 | 
				
			||||||
 | 
					      TestHelper.request(server, done, {
 | 
				
			||||||
 | 
					        method: 'put',
 | 
				
			||||||
 | 
					        url: '/measurement/800000000000000000000002',
 | 
				
			||||||
 | 
					        auth: {key: 'janedoe'},
 | 
				
			||||||
 | 
					        httpStatus: 401,
 | 
				
			||||||
 | 
					        req: {values: {'weight %': 0.9, 'standard deviation': 0.3}},
 | 
				
			||||||
 | 
					      });
 | 
				
			||||||
 | 
					    });
 | 
				
			||||||
 | 
					    it('rejects requests from a read user', done => {
 | 
				
			||||||
 | 
					      TestHelper.request(server, done, {
 | 
				
			||||||
 | 
					        method: 'put',
 | 
				
			||||||
 | 
					        url: '/measurement/800000000000000000000002',
 | 
				
			||||||
 | 
					        auth: {basic: 'user'},
 | 
				
			||||||
 | 
					        httpStatus: 403,
 | 
				
			||||||
 | 
					        req: {values: {'weight %': 0.9, 'standard deviation': 0.3}},
 | 
				
			||||||
 | 
					      });
 | 
				
			||||||
 | 
					    });
 | 
				
			||||||
 | 
					    it('rejects unauthorized requests', done => {
 | 
				
			||||||
 | 
					      TestHelper.request(server, done, {
 | 
				
			||||||
 | 
					        method: 'put',
 | 
				
			||||||
 | 
					        url: '/measurement/800000000000000000000002',
 | 
				
			||||||
 | 
					        httpStatus: 401,
 | 
				
			||||||
 | 
					        req: {values: {'weight %': 0.9, 'standard deviation': 0.3}},
 | 
				
			||||||
 | 
					      });
 | 
				
			||||||
 | 
					    });
 | 
				
			||||||
 | 
					  });
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  describe('DELETE /measurement/{id}', () => {
 | 
				
			||||||
 | 
					    it('sets the status to deleted', done => {
 | 
				
			||||||
 | 
					      TestHelper.request(server, done, {
 | 
				
			||||||
 | 
					        method: 'delete',
 | 
				
			||||||
 | 
					        url: '/measurement/800000000000000000000001',
 | 
				
			||||||
 | 
					        auth: {basic: 'janedoe'},
 | 
				
			||||||
 | 
					        httpStatus: 200,
 | 
				
			||||||
 | 
					      }).end((err, res) => {
 | 
				
			||||||
 | 
					        if (err) return done(err);
 | 
				
			||||||
 | 
					        should(res.body).be.eql({status: 'OK'});
 | 
				
			||||||
 | 
					        MeasurementModel.findById('800000000000000000000001').lean().exec((err, data) => {
 | 
				
			||||||
 | 
					          if (err) return done(err);
 | 
				
			||||||
 | 
					          should(data).have.property('status', -1);
 | 
				
			||||||
 | 
					          done();
 | 
				
			||||||
 | 
					        });
 | 
				
			||||||
 | 
					      });
 | 
				
			||||||
 | 
					    });
 | 
				
			||||||
 | 
					    it('rejects an API key', done => {
 | 
				
			||||||
 | 
					      TestHelper.request(server, done, {
 | 
				
			||||||
 | 
					        method: 'delete',
 | 
				
			||||||
 | 
					        url: '/measurement/800000000000000000000001',
 | 
				
			||||||
 | 
					        auth: {key: 'janedoe'},
 | 
				
			||||||
 | 
					        httpStatus: 401,
 | 
				
			||||||
 | 
					      });
 | 
				
			||||||
 | 
					    });
 | 
				
			||||||
 | 
					    it('rejects requests from a read user', done => {
 | 
				
			||||||
 | 
					      TestHelper.request(server, done, {
 | 
				
			||||||
 | 
					        method: 'delete',
 | 
				
			||||||
 | 
					        url: '/measurement/800000000000000000000001',
 | 
				
			||||||
 | 
					        auth: {basic: 'user'},
 | 
				
			||||||
 | 
					        httpStatus: 403,
 | 
				
			||||||
 | 
					      });
 | 
				
			||||||
 | 
					    });
 | 
				
			||||||
 | 
					    it('rejects deleting a measurement for a write user who did not create this measurement', done => {
 | 
				
			||||||
 | 
					      TestHelper.request(server, done, {
 | 
				
			||||||
 | 
					        method: 'delete',
 | 
				
			||||||
 | 
					        url: '/measurement/800000000000000000000003',
 | 
				
			||||||
 | 
					        auth: {basic: 'janedoe'},
 | 
				
			||||||
 | 
					        httpStatus: 403,
 | 
				
			||||||
 | 
					      });
 | 
				
			||||||
 | 
					    });
 | 
				
			||||||
 | 
					    it('accepts deleting a measurement of another user for a maintain/admin user', done => {
 | 
				
			||||||
 | 
					      TestHelper.request(server, done, {
 | 
				
			||||||
 | 
					        method: 'delete',
 | 
				
			||||||
 | 
					        url: '/measurement/800000000000000000000001',
 | 
				
			||||||
 | 
					        auth: {basic: 'admin'},
 | 
				
			||||||
 | 
					        httpStatus: 200,
 | 
				
			||||||
 | 
					        res: {status: 'OK'}
 | 
				
			||||||
 | 
					      });
 | 
				
			||||||
 | 
					    });
 | 
				
			||||||
 | 
					    it('rejects an invalid id', done => {
 | 
				
			||||||
 | 
					      TestHelper.request(server, done, {
 | 
				
			||||||
 | 
					        method: 'delete',
 | 
				
			||||||
 | 
					        url: '/measurement/800000000h00000000000001',
 | 
				
			||||||
 | 
					        auth: {basic: 'janedoe'},
 | 
				
			||||||
 | 
					        httpStatus: 404,
 | 
				
			||||||
 | 
					      });
 | 
				
			||||||
 | 
					    });
 | 
				
			||||||
 | 
					    it('rejects an unknown id', done => {
 | 
				
			||||||
 | 
					      TestHelper.request(server, done, {
 | 
				
			||||||
 | 
					        method: 'delete',
 | 
				
			||||||
 | 
					        url: '/measurement/000000000000000000000001',
 | 
				
			||||||
 | 
					        auth: {basic: 'janedoe'},
 | 
				
			||||||
 | 
					        httpStatus: 404,
 | 
				
			||||||
 | 
					      });
 | 
				
			||||||
 | 
					    });
 | 
				
			||||||
 | 
					    it('rejects unauthorized requests', done => {
 | 
				
			||||||
 | 
					      TestHelper.request(server, done, {
 | 
				
			||||||
 | 
					        method: 'delete',
 | 
				
			||||||
 | 
					        url: '/measurement/800000000000000000000001',
 | 
				
			||||||
 | 
					        httpStatus: 401,
 | 
				
			||||||
 | 
					      });
 | 
				
			||||||
 | 
					    });
 | 
				
			||||||
 | 
					  });
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  describe('POST /measurement/new', () => {
 | 
				
			||||||
 | 
					    it('returns the right measurement', done => {
 | 
				
			||||||
 | 
					      TestHelper.request(server, done, {
 | 
				
			||||||
 | 
					        method: 'post',
 | 
				
			||||||
 | 
					        url: '/measurement/new',
 | 
				
			||||||
 | 
					        auth: {basic: 'janedoe'},
 | 
				
			||||||
 | 
					        httpStatus: 200,
 | 
				
			||||||
 | 
					        req: {condition_id: '700000000000000000000001', values: {'weight %': 0.8, 'standard deviation': 0.1}, measurement_template: '300000000000000000000002'}
 | 
				
			||||||
 | 
					      }).end((err, res) => {
 | 
				
			||||||
 | 
					        if (err) return done(err);
 | 
				
			||||||
 | 
					        should(res.body).have.only.keys('_id', 'condition_id', 'values', 'measurement_template');
 | 
				
			||||||
 | 
					        should(res.body).have.property('_id').be.type('string');
 | 
				
			||||||
 | 
					        should(res.body).have.property('condition_id', '700000000000000000000001');
 | 
				
			||||||
 | 
					        should(res.body).have.property('measurement_template', '300000000000000000000002');
 | 
				
			||||||
 | 
					        should(res.body).have.property('values');
 | 
				
			||||||
 | 
					        should(res.body.values).have.property('weight %', 0.8);
 | 
				
			||||||
 | 
					        should(res.body.values).have.property('standard deviation', 0.1);
 | 
				
			||||||
 | 
					        done();
 | 
				
			||||||
 | 
					      });
 | 
				
			||||||
 | 
					    });
 | 
				
			||||||
 | 
					    it('stores the measurement', done => {
 | 
				
			||||||
 | 
					      TestHelper.request(server, done, {
 | 
				
			||||||
 | 
					        method: 'post',
 | 
				
			||||||
 | 
					        url: '/measurement/new',
 | 
				
			||||||
 | 
					        auth: {basic: 'janedoe'},
 | 
				
			||||||
 | 
					        httpStatus: 200,
 | 
				
			||||||
 | 
					        req: {condition_id: '700000000000000000000001', values: {'weight %': 0.8, 'standard deviation': 0.1}, measurement_template: '300000000000000000000002'}
 | 
				
			||||||
 | 
					      }).end((err, res) => {
 | 
				
			||||||
 | 
					        if (err) return done(err);
 | 
				
			||||||
 | 
					        MeasurementModel.findById(res.body._id).lean().exec((err, data: any) => {
 | 
				
			||||||
 | 
					          if (err) return done(err);
 | 
				
			||||||
 | 
					          should(data).have.only.keys('_id', 'condition_id', 'values', 'measurement_template', 'status', '__v');
 | 
				
			||||||
 | 
					          should(data.condition_id.toString()).be.eql('700000000000000000000001');
 | 
				
			||||||
 | 
					          should(data.measurement_template.toString()).be.eql('300000000000000000000002');
 | 
				
			||||||
 | 
					          should(data).have.property('status', 0);
 | 
				
			||||||
 | 
					          should(data).have.property('values');
 | 
				
			||||||
 | 
					          should(data.values).have.property('weight %', 0.8);
 | 
				
			||||||
 | 
					          should(data.values).have.property('standard deviation', 0.1);
 | 
				
			||||||
 | 
					          done();
 | 
				
			||||||
 | 
					        });
 | 
				
			||||||
 | 
					      });
 | 
				
			||||||
 | 
					    });
 | 
				
			||||||
 | 
					    it('rejects an invalid condition id', done => {
 | 
				
			||||||
 | 
					      TestHelper.request(server, done, {
 | 
				
			||||||
 | 
					        method: 'post',
 | 
				
			||||||
 | 
					        url: '/measurement/new',
 | 
				
			||||||
 | 
					        auth: {basic: 'janedoe'},
 | 
				
			||||||
 | 
					        httpStatus: 400,
 | 
				
			||||||
 | 
					        req: {condition_id: '700000000000h00000000001', values: {'weight %': 0.8, 'standard deviation': 0.1}, measurement_template: '300000000000000000000002'},
 | 
				
			||||||
 | 
					        res: {status: 'Invalid body format', details: '"condition_id" with value "700000000000h00000000001" fails to match the required pattern: /[0-9a-f]{24}/'}
 | 
				
			||||||
 | 
					      });
 | 
				
			||||||
 | 
					    });
 | 
				
			||||||
 | 
					    it('rejects a condition id not available', done => {
 | 
				
			||||||
 | 
					      TestHelper.request(server, done, {
 | 
				
			||||||
 | 
					        method: 'post',
 | 
				
			||||||
 | 
					        url: '/measurement/new',
 | 
				
			||||||
 | 
					        auth: {basic: 'janedoe'},
 | 
				
			||||||
 | 
					        httpStatus: 400,
 | 
				
			||||||
 | 
					        req: {condition_id: '000000000000000000000001', values: {'weight %': 0.8, 'standard deviation': 0.1}, measurement_template: '300000000000000000000002'},
 | 
				
			||||||
 | 
					        res: {status: 'Condition id not available'}
 | 
				
			||||||
 | 
					      });
 | 
				
			||||||
 | 
					    });
 | 
				
			||||||
 | 
					    it('rejects an invalid measurement_template id', done => {
 | 
				
			||||||
 | 
					      TestHelper.request(server, done, {
 | 
				
			||||||
 | 
					        method: 'post',
 | 
				
			||||||
 | 
					        url: '/measurement/new',
 | 
				
			||||||
 | 
					        auth: {basic: 'janedoe'},
 | 
				
			||||||
 | 
					        httpStatus: 400,
 | 
				
			||||||
 | 
					        req: {condition_id: '700000000000000000000001', values: {'weight %': 0.8, 'standard deviation': 0.1}, measurement_template: '30000000000h000000000002'},
 | 
				
			||||||
 | 
					        res: {status: 'Invalid body format', details: '"measurement_template" with value "30000000000h000000000002" fails to match the required pattern: /[0-9a-f]{24}/'}
 | 
				
			||||||
 | 
					      });
 | 
				
			||||||
 | 
					    });
 | 
				
			||||||
 | 
					    it('rejects a measurement_template not available', done => {
 | 
				
			||||||
 | 
					      TestHelper.request(server, done, {
 | 
				
			||||||
 | 
					        method: 'post',
 | 
				
			||||||
 | 
					        url: '/measurement/new',
 | 
				
			||||||
 | 
					        auth: {basic: 'janedoe'},
 | 
				
			||||||
 | 
					        httpStatus: 400,
 | 
				
			||||||
 | 
					        req: {condition_id: '700000000000000000000001', values: {'weight %': 0.8, 'standard deviation': 0.1}, measurement_template: '000000000000000000000002'},
 | 
				
			||||||
 | 
					        res: {status: 'Measurement template not available'}
 | 
				
			||||||
 | 
					      });
 | 
				
			||||||
 | 
					    });
 | 
				
			||||||
 | 
					    it('rejects not specified values', done => {
 | 
				
			||||||
 | 
					      TestHelper.request(server, done, {
 | 
				
			||||||
 | 
					        method: 'post',
 | 
				
			||||||
 | 
					        url: '/measurement/new',
 | 
				
			||||||
 | 
					        auth: {basic: 'janedoe'},
 | 
				
			||||||
 | 
					        httpStatus: 400,
 | 
				
			||||||
 | 
					        req: {condition_id: '700000000000000000000001', values: {'weight %': 0.8, 'standard deviation': 0.1, xx: 44}, measurement_template: '300000000000000000000002'},
 | 
				
			||||||
 | 
					        res: {status: 'Invalid body format', details: '"xx" is not allowed'}
 | 
				
			||||||
 | 
					      });
 | 
				
			||||||
 | 
					    });
 | 
				
			||||||
 | 
					    it('rejects missing values', done => {
 | 
				
			||||||
 | 
					      TestHelper.request(server, done, {
 | 
				
			||||||
 | 
					        method: 'post',
 | 
				
			||||||
 | 
					        url: '/measurement/new',
 | 
				
			||||||
 | 
					        auth: {basic: 'janedoe'},
 | 
				
			||||||
 | 
					        httpStatus: 400,
 | 
				
			||||||
 | 
					        req: {condition_id: '700000000000000000000001', values: {'weight %': 0.8}, measurement_template: '300000000000000000000002'},
 | 
				
			||||||
 | 
					        res: {status: 'Invalid body format', details: '"standard deviation" is required'}
 | 
				
			||||||
 | 
					      });
 | 
				
			||||||
 | 
					    });
 | 
				
			||||||
 | 
					    it('rejects a value not in the value range', done => {
 | 
				
			||||||
 | 
					      TestHelper.request(server, done, {
 | 
				
			||||||
 | 
					        method: 'post',
 | 
				
			||||||
 | 
					        url: '/measurement/new',
 | 
				
			||||||
 | 
					        auth: {basic: 'janedoe'},
 | 
				
			||||||
 | 
					        httpStatus: 400,
 | 
				
			||||||
 | 
					        req: {condition_id: '700000000000000000000001', values: {val1: 4}, measurement_template: '300000000000000000000003'},
 | 
				
			||||||
 | 
					        res: {status: 'Invalid body format', details: '"val1" must be one of [1, 2, 3]'}
 | 
				
			||||||
 | 
					      });
 | 
				
			||||||
 | 
					    });
 | 
				
			||||||
 | 
					    it('rejects a value below minimum range', done => {
 | 
				
			||||||
 | 
					      TestHelper.request(server, done, {
 | 
				
			||||||
 | 
					        method: 'post',
 | 
				
			||||||
 | 
					        url: '/measurement/new',
 | 
				
			||||||
 | 
					        auth: {basic: 'janedoe'},
 | 
				
			||||||
 | 
					        httpStatus: 400,
 | 
				
			||||||
 | 
					        req: {condition_id: '700000000000000000000001', values: {'weight %': -1, 'standard deviation': 0.1}, measurement_template: '300000000000000000000002'},
 | 
				
			||||||
 | 
					        res: {status: 'Invalid body format', details: '"weight %" must be larger than or equal to 0'}
 | 
				
			||||||
 | 
					      });
 | 
				
			||||||
 | 
					    });
 | 
				
			||||||
 | 
					    it('rejects a value above maximum range', done => {
 | 
				
			||||||
 | 
					      TestHelper.request(server, done, {
 | 
				
			||||||
 | 
					        method: 'post',
 | 
				
			||||||
 | 
					        url: '/measurement/new',
 | 
				
			||||||
 | 
					        auth: {basic: 'janedoe'},
 | 
				
			||||||
 | 
					        httpStatus: 400,
 | 
				
			||||||
 | 
					        req: {condition_id: '700000000000000000000001', values: {'weight %': 0.8, 'standard deviation': 2}, measurement_template: '300000000000000000000002'},
 | 
				
			||||||
 | 
					        res: {status: 'Invalid body format', details: '"standard deviation" must be less than or equal to 0.5'}
 | 
				
			||||||
 | 
					      });
 | 
				
			||||||
 | 
					    });
 | 
				
			||||||
 | 
					    it('rejects a missing condition id', done => {
 | 
				
			||||||
 | 
					      TestHelper.request(server, done, {
 | 
				
			||||||
 | 
					        method: 'post',
 | 
				
			||||||
 | 
					        url: '/measurement/new',
 | 
				
			||||||
 | 
					        auth: {basic: 'janedoe'},
 | 
				
			||||||
 | 
					        httpStatus: 400,
 | 
				
			||||||
 | 
					        req: {values: {'weight %': 0.8, 'standard deviation': 0.1}, measurement_template: '300000000000000000000002'},
 | 
				
			||||||
 | 
					        res: {status: 'Invalid body format', details: '"condition_id" is required'}
 | 
				
			||||||
 | 
					      });
 | 
				
			||||||
 | 
					    });
 | 
				
			||||||
 | 
					    it('rejects a missing measurement_template', done => {
 | 
				
			||||||
 | 
					      TestHelper.request(server, done, {
 | 
				
			||||||
 | 
					        method: 'post',
 | 
				
			||||||
 | 
					        url: '/measurement/new',
 | 
				
			||||||
 | 
					        auth: {basic: 'janedoe'},
 | 
				
			||||||
 | 
					        httpStatus: 400,
 | 
				
			||||||
 | 
					        req: {condition_id: '700000000000000000000001', values: {'weight %': 0.8, 'standard deviation': 0.1}},
 | 
				
			||||||
 | 
					        res: {status: 'Invalid body format', details: '"measurement_template" is required'}
 | 
				
			||||||
 | 
					      });
 | 
				
			||||||
 | 
					    });
 | 
				
			||||||
 | 
					    it('rejects adding a measurement to the sample of another user for a write user', done => {
 | 
				
			||||||
 | 
					      TestHelper.request(server, done, {
 | 
				
			||||||
 | 
					        method: 'post',
 | 
				
			||||||
 | 
					        url: '/measurement/new',
 | 
				
			||||||
 | 
					        auth: {basic: 'janedoe'},
 | 
				
			||||||
 | 
					        httpStatus: 403,
 | 
				
			||||||
 | 
					        req: {condition_id: '700000000000000000000003', values: {'weight %': 0.8, 'standard deviation': 0.1}, measurement_template: '300000000000000000000002'}
 | 
				
			||||||
 | 
					      });
 | 
				
			||||||
 | 
					    });
 | 
				
			||||||
 | 
					    it('accepts adding a measurement to the sample of another user for a maintain/admin user', done => {
 | 
				
			||||||
 | 
					      TestHelper.request(server, done, {
 | 
				
			||||||
 | 
					        method: 'post',
 | 
				
			||||||
 | 
					        url: '/measurement/new',
 | 
				
			||||||
 | 
					        auth: {basic: 'admin'},
 | 
				
			||||||
 | 
					        httpStatus: 200,
 | 
				
			||||||
 | 
					        req: {condition_id: '700000000000000000000001', values: {'weight %': 0.8, 'standard deviation': 0.1}, measurement_template: '300000000000000000000002'}
 | 
				
			||||||
 | 
					      }).end((err, res) => {
 | 
				
			||||||
 | 
					        if (err) return done(err);
 | 
				
			||||||
 | 
					        should(res.body).have.only.keys('_id', 'condition_id', 'values', 'measurement_template');
 | 
				
			||||||
 | 
					        should(res.body).have.property('_id').be.type('string');
 | 
				
			||||||
 | 
					        should(res.body).have.property('condition_id', '700000000000000000000001');
 | 
				
			||||||
 | 
					        should(res.body).have.property('measurement_template', '300000000000000000000002');
 | 
				
			||||||
 | 
					        should(res.body).have.property('values');
 | 
				
			||||||
 | 
					        should(res.body.values).have.property('weight %', 0.8);
 | 
				
			||||||
 | 
					        should(res.body.values).have.property('standard deviation', 0.1);
 | 
				
			||||||
 | 
					        done();
 | 
				
			||||||
 | 
					      });
 | 
				
			||||||
 | 
					    });
 | 
				
			||||||
 | 
					    it('rejects an API key', done => {
 | 
				
			||||||
 | 
					      TestHelper.request(server, done, {
 | 
				
			||||||
 | 
					        method: 'post',
 | 
				
			||||||
 | 
					        url: '/measurement/new',
 | 
				
			||||||
 | 
					        auth: {key: 'janedoe'},
 | 
				
			||||||
 | 
					        httpStatus: 401,
 | 
				
			||||||
 | 
					        req: {condition_id: '700000000000000000000001', values: {'weight %': 0.8, 'standard deviation': 0.1}, measurement_template: '300000000000000000000002'}
 | 
				
			||||||
 | 
					      });
 | 
				
			||||||
 | 
					    });
 | 
				
			||||||
 | 
					    it('rejects requests from a read user', done => {
 | 
				
			||||||
 | 
					      TestHelper.request(server, done, {
 | 
				
			||||||
 | 
					        method: 'post',
 | 
				
			||||||
 | 
					        url: '/measurement/new',
 | 
				
			||||||
 | 
					        auth: {basic: 'user'},
 | 
				
			||||||
 | 
					        httpStatus: 403,
 | 
				
			||||||
 | 
					        req: {condition_id: '700000000000000000000001', values: {'weight %': 0.8, 'standard deviation': 0.1}, measurement_template: '300000000000000000000002'}
 | 
				
			||||||
 | 
					      });
 | 
				
			||||||
 | 
					    });
 | 
				
			||||||
 | 
					    it('rejects unauthorized requests', done => {
 | 
				
			||||||
 | 
					      TestHelper.request(server, done, {
 | 
				
			||||||
 | 
					        method: 'post',
 | 
				
			||||||
 | 
					        url: '/measurement/new',
 | 
				
			||||||
 | 
					        httpStatus: 401,
 | 
				
			||||||
 | 
					        req: {condition_id: '700000000000000000000001', values: {'weight %': 0.8, 'standard deviation': 0.1}, measurement_template: '300000000000000000000002'}
 | 
				
			||||||
 | 
					      });
 | 
				
			||||||
 | 
					    });
 | 
				
			||||||
 | 
					  });
 | 
				
			||||||
 | 
					});
 | 
				
			||||||
							
								
								
									
										116
									
								
								src/routes/measurement.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										116
									
								
								src/routes/measurement.ts
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,116 @@
 | 
				
			|||||||
 | 
					import express from 'express';
 | 
				
			||||||
 | 
					import _ from 'lodash';
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import MeasurementModel from '../models/measurement';
 | 
				
			||||||
 | 
					import ConditionModel from '../models/condition';
 | 
				
			||||||
 | 
					import MeasurementTemplateModel from '../models/measurement_template';
 | 
				
			||||||
 | 
					import MeasurementValidate from './validate/measurement';
 | 
				
			||||||
 | 
					import IdValidate from './validate/id';
 | 
				
			||||||
 | 
					import res400 from './validate/res400';
 | 
				
			||||||
 | 
					import ParametersValidate from './validate/parameters';
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					const router = express.Router();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					router.get('/measurement/' + IdValidate.parameter(), (req, res, next) => {
 | 
				
			||||||
 | 
					  if (!req.auth(res, ['read', 'write', 'maintain', 'dev', 'admin'], 'all')) return;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  MeasurementModel.findById(req.params.id).lean().exec((err, data) => {
 | 
				
			||||||
 | 
					    if (err) return next(err);
 | 
				
			||||||
 | 
					    if (!data) {
 | 
				
			||||||
 | 
					      return res.status(404).json({status: 'Not found'});
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    res.json(MeasurementValidate.output(data));
 | 
				
			||||||
 | 
					  });
 | 
				
			||||||
 | 
					});
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					router.put('/measurement/' + IdValidate.parameter(), async (req, res, next) => {
 | 
				
			||||||
 | 
					  if (!req.auth(res, ['write', 'maintain', 'dev', 'admin'], 'basic')) return;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  const {error, value: measurement} = MeasurementValidate.input(req.body, 'change');
 | 
				
			||||||
 | 
					  if (error) return res400(error, res);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  const data = await MeasurementModel.findById(req.params.id).lean().exec().catch(err => {next(err);}) as any;
 | 
				
			||||||
 | 
					  if (data instanceof Error) {
 | 
				
			||||||
 | 
					    return;
 | 
				
			||||||
 | 
					  }
 | 
				
			||||||
 | 
					  if (!data) {
 | 
				
			||||||
 | 
					    res.status(404).json({status: 'Not found'});
 | 
				
			||||||
 | 
					  }
 | 
				
			||||||
 | 
					  // add properties needed for conditionIdCheck
 | 
				
			||||||
 | 
					  measurement.measurement_template = data.measurement_template;
 | 
				
			||||||
 | 
					  measurement.condition_id = data.condition_id;
 | 
				
			||||||
 | 
					  if (measurement.hasOwnProperty('values') && !_.isEqual(measurement.values, data.values)) {
 | 
				
			||||||
 | 
					    measurement.status = 0;
 | 
				
			||||||
 | 
					  }
 | 
				
			||||||
 | 
					  if (!await conditionIdCheck(measurement, req, res, next)) return;
 | 
				
			||||||
 | 
					  if (measurement.values) {
 | 
				
			||||||
 | 
					    measurement.values = Object.assign(data.values, measurement.values);
 | 
				
			||||||
 | 
					  }
 | 
				
			||||||
 | 
					  if (!await templateCheck(measurement, 'change', res, next)) return;
 | 
				
			||||||
 | 
					  await MeasurementModel.findByIdAndUpdate(req.params.id, measurement, {new: true}).lean().exec((err, data) => {
 | 
				
			||||||
 | 
					    if (err) return next(err);
 | 
				
			||||||
 | 
					    res.json(MeasurementValidate.output(data));
 | 
				
			||||||
 | 
					  });
 | 
				
			||||||
 | 
					});
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					router.delete('/measurement/' + IdValidate.parameter(), (req, res, next) => {
 | 
				
			||||||
 | 
					  if (!req.auth(res, ['write', 'maintain', 'dev', 'admin'], 'basic')) return;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  MeasurementModel.findById(req.params.id).lean().exec(async (err, data) => {
 | 
				
			||||||
 | 
					    if (err) return next(err);
 | 
				
			||||||
 | 
					    if (!data) {
 | 
				
			||||||
 | 
					      res.status(404).json({status: 'Not found'});
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					    if (!await conditionIdCheck(data, req, res, next)) return;
 | 
				
			||||||
 | 
					    await MeasurementModel.findByIdAndUpdate(req.params.id, {status: -1}).lean().exec(async err => {
 | 
				
			||||||
 | 
					      if (err) return next(err);
 | 
				
			||||||
 | 
					      res.json({status: 'OK'});
 | 
				
			||||||
 | 
					    });
 | 
				
			||||||
 | 
					  });
 | 
				
			||||||
 | 
					});
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					router.post('/measurement/new', async (req, res, next) => {
 | 
				
			||||||
 | 
					  if (!req.auth(res, ['write', 'maintain', 'dev', 'admin'], 'basic')) return;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  const {error, value: measurement} = MeasurementValidate.input(req.body, 'new');
 | 
				
			||||||
 | 
					  if (error) return res400(error, res);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  if (!await conditionIdCheck(measurement, req, res, next)) return;
 | 
				
			||||||
 | 
					  if (!await templateCheck(measurement, 'new', res, next)) return;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  measurement.status = 0;
 | 
				
			||||||
 | 
					  await new MeasurementModel(measurement).save((err, data) => {
 | 
				
			||||||
 | 
					    if (err) return next(err);
 | 
				
			||||||
 | 
					    res.json(MeasurementValidate.output(data.toObject()));
 | 
				
			||||||
 | 
					  });
 | 
				
			||||||
 | 
					});
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					module.exports = router;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					async function conditionIdCheck (measurement, req, res, next) {  // validate condition_id, returns false if invalid
 | 
				
			||||||
 | 
					  const sampleData = await ConditionModel.findById(measurement.condition_id).populate('sample_id').lean().exec().catch(err => {next(err); return false;}) as any;
 | 
				
			||||||
 | 
					  if (!sampleData) {  // sample_id not found
 | 
				
			||||||
 | 
					    res.status(400).json({status: 'Condition id not available'});
 | 
				
			||||||
 | 
					    return false
 | 
				
			||||||
 | 
					  }
 | 
				
			||||||
 | 
					  if (sampleData.sample_id.user_id.toString() !== req.authDetails.id && !req.auth(res, ['maintain', 'admin'], 'basic')) return false;  // sample does not belong to user
 | 
				
			||||||
 | 
					  return true;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					async function templateCheck (measurement, param, res, next) {  // validate measurement_template and values
 | 
				
			||||||
 | 
					  const templateData = await MeasurementTemplateModel.findById(measurement.measurement_template).lean().exec().catch(err => {next(err); return false;}) as any;
 | 
				
			||||||
 | 
					  if (!templateData) {  // template not found
 | 
				
			||||||
 | 
					    res.status(400).json({status: 'Measurement template not available'});
 | 
				
			||||||
 | 
					    return false
 | 
				
			||||||
 | 
					  }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  // validate values
 | 
				
			||||||
 | 
					  const {error, value: ignore} = ParametersValidate.input(measurement.values, templateData.parameters, param);
 | 
				
			||||||
 | 
					  console.log(error);
 | 
				
			||||||
 | 
					  if (error) {res400(error, res); return false;}
 | 
				
			||||||
 | 
					  return true;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
@@ -4,6 +4,8 @@ import NoteModel from '../models/note';
 | 
				
			|||||||
import NoteFieldModel from '../models/note_field';
 | 
					import NoteFieldModel from '../models/note_field';
 | 
				
			||||||
import TestHelper from "../test/helper";
 | 
					import TestHelper from "../test/helper";
 | 
				
			||||||
// TODO: generate sample number
 | 
					// TODO: generate sample number
 | 
				
			||||||
 | 
					// TODO: think again which parameters are required at POST
 | 
				
			||||||
 | 
					// TODO: status
 | 
				
			||||||
 | 
					
 | 
				
			||||||
describe('/sample', () => {
 | 
					describe('/sample', () => {
 | 
				
			||||||
  let server;
 | 
					  let server;
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -1,4 +1,5 @@
 | 
				
			|||||||
import express from 'express';
 | 
					import express from 'express';
 | 
				
			||||||
 | 
					import _ from 'lodash';
 | 
				
			||||||
 | 
					
 | 
				
			||||||
import SampleValidate from './validate/sample';
 | 
					import SampleValidate from './validate/sample';
 | 
				
			||||||
import NoteFieldValidate from './validate/note_field';
 | 
					import NoteFieldValidate from './validate/note_field';
 | 
				
			||||||
@@ -17,7 +18,7 @@ router.get('/samples', (req, res, next) => {
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
  SampleModel.find({}).lean().exec((err, data) => {
 | 
					  SampleModel.find({}).lean().exec((err, data) => {
 | 
				
			||||||
    if (err) return next(err);
 | 
					    if (err) return next(err);
 | 
				
			||||||
    res.json(data.map(e => SampleValidate.output(e)).filter(e => e !== null));  // validate all and filter null values from validation errors
 | 
					    res.json(_.compact(data.map(e => SampleValidate.output(e))));  // validate all and filter null values from validation errors
 | 
				
			||||||
  })
 | 
					  })
 | 
				
			||||||
});
 | 
					});
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -141,7 +142,7 @@ router.get('/sample/notes/fields', (req, res, next) => {
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
  NoteFieldModel.find({}).lean().exec((err, data) => {
 | 
					  NoteFieldModel.find({}).lean().exec((err, data) => {
 | 
				
			||||||
    if (err) return next(err);
 | 
					    if (err) return next(err);
 | 
				
			||||||
    res.json(data.map(e => NoteFieldValidate.output(e)).filter(e => e !== null));  // validate all and filter null values from validation errors
 | 
					    res.json(_.compact(data.map(e => NoteFieldValidate.output(e))));  // validate all and filter null values from validation errors
 | 
				
			||||||
  })
 | 
					  })
 | 
				
			||||||
});
 | 
					});
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -1,4 +1,5 @@
 | 
				
			|||||||
import express from 'express';
 | 
					import express from 'express';
 | 
				
			||||||
 | 
					import _ from 'lodash';
 | 
				
			||||||
 | 
					
 | 
				
			||||||
import TemplateValidate from './validate/template';
 | 
					import TemplateValidate from './validate/template';
 | 
				
			||||||
import TemplateTreatmentModel from '../models/treatment_template';
 | 
					import TemplateTreatmentModel from '../models/treatment_template';
 | 
				
			||||||
@@ -14,7 +15,7 @@ router.get('/template/:collection(measurements|treatments)', (req, res, next) =>
 | 
				
			|||||||
  (req.params.collection === 'treatments' ? TemplateTreatmentModel : TemplateMeasurementModel)
 | 
					  (req.params.collection === 'treatments' ? TemplateTreatmentModel : TemplateMeasurementModel)
 | 
				
			||||||
    .find({}).lean().exec((err, data) => {
 | 
					    .find({}).lean().exec((err, data) => {
 | 
				
			||||||
     if (err) next (err);
 | 
					     if (err) next (err);
 | 
				
			||||||
     res.json(data.map(e => TemplateValidate.output(e)).filter(e => e !== null));  // validate all and filter null values from validation errors
 | 
					     res.json(_.compact(data.map(e => TemplateValidate.output(e))));  // validate all and filter null values from validation errors
 | 
				
			||||||
  });
 | 
					  });
 | 
				
			||||||
});
 | 
					});
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -1,6 +1,7 @@
 | 
				
			|||||||
import express from 'express';
 | 
					import express from 'express';
 | 
				
			||||||
import mongoose from 'mongoose';
 | 
					import mongoose from 'mongoose';
 | 
				
			||||||
import bcrypt from 'bcryptjs';
 | 
					import bcrypt from 'bcryptjs';
 | 
				
			||||||
 | 
					import _ from 'lodash';
 | 
				
			||||||
 | 
					
 | 
				
			||||||
import UserValidate from './validate/user';
 | 
					import UserValidate from './validate/user';
 | 
				
			||||||
import UserModel from '../models/user';
 | 
					import UserModel from '../models/user';
 | 
				
			||||||
@@ -14,7 +15,7 @@ router.get('/users', (req, res) => {
 | 
				
			|||||||
  if (!req.auth(res, ['admin'], 'basic')) return;
 | 
					  if (!req.auth(res, ['admin'], 'basic')) return;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  UserModel.find({}).lean().exec(  (err, data:any) => {
 | 
					  UserModel.find({}).lean().exec(  (err, data:any) => {
 | 
				
			||||||
    res.json(data.map(e => UserValidate.output(e)).filter(e => e !== null));  // validate all and filter null values from validation errors
 | 
					    res.json(_.compact(data.map(e => UserValidate.output(e))));  // validate all and filter null values from validation errors
 | 
				
			||||||
  });
 | 
					  });
 | 
				
			||||||
});
 | 
					});
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -4,8 +4,6 @@ import IdValidate from './id';
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
export default class ConditionValidate {
 | 
					export default class ConditionValidate {
 | 
				
			||||||
  private static condition = {
 | 
					  private static condition = {
 | 
				
			||||||
    sample_id: IdValidate.get(),
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    number: Joi.string()
 | 
					    number: Joi.string()
 | 
				
			||||||
      .max(128),
 | 
					      .max(128),
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -14,20 +12,19 @@ export default class ConditionValidate {
 | 
				
			|||||||
        .try(
 | 
					        .try(
 | 
				
			||||||
          Joi.string().max(128),
 | 
					          Joi.string().max(128),
 | 
				
			||||||
          Joi.number(),
 | 
					          Joi.number(),
 | 
				
			||||||
          Joi.boolean()
 | 
					          Joi.boolean(),
 | 
				
			||||||
 | 
					          Joi.array()
 | 
				
			||||||
 | 
					        )
 | 
				
			||||||
      )
 | 
					      )
 | 
				
			||||||
      ),
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    treatment_template: IdValidate.get()
 | 
					 | 
				
			||||||
  }
 | 
					  }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  static input (data, param) {
 | 
					  static input (data, param) {
 | 
				
			||||||
    if (param === 'new') {
 | 
					    if (param === 'new') {
 | 
				
			||||||
      return Joi.object({
 | 
					      return Joi.object({
 | 
				
			||||||
        sample_id: this.condition.sample_id.required(),
 | 
					        sample_id: IdValidate.get().required(),
 | 
				
			||||||
        number: this.condition.number.required(),
 | 
					        number: this.condition.number.required(),
 | 
				
			||||||
        parameters: this.condition.parameters.required(),
 | 
					        parameters: this.condition.parameters.required(),
 | 
				
			||||||
        treatment_template: this.condition.treatment_template.required()
 | 
					        treatment_template: IdValidate.get().required()
 | 
				
			||||||
      }).validate(data);
 | 
					      }).validate(data);
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
    else if (param === 'change') {
 | 
					    else if (param === 'change') {
 | 
				
			||||||
@@ -45,10 +42,10 @@ export default class ConditionValidate {
 | 
				
			|||||||
    data = IdValidate.stringify(data);
 | 
					    data = IdValidate.stringify(data);
 | 
				
			||||||
    const {value, error} = Joi.object({
 | 
					    const {value, error} = Joi.object({
 | 
				
			||||||
      _id: IdValidate.get(),
 | 
					      _id: IdValidate.get(),
 | 
				
			||||||
      sample_id: this.condition.sample_id,
 | 
					      sample_id: IdValidate.get(),
 | 
				
			||||||
      number: this.condition.number,
 | 
					      number: this.condition.number,
 | 
				
			||||||
      parameters: this.condition.parameters,
 | 
					      parameters: this.condition.parameters,
 | 
				
			||||||
      treatment_template: this.condition.treatment_template
 | 
					      treatment_template: IdValidate.get()
 | 
				
			||||||
    }).validate(data, {stripUnknown: true});
 | 
					    }).validate(data, {stripUnknown: true});
 | 
				
			||||||
    return error !== undefined? null : value;
 | 
					    return error !== undefined? null : value;
 | 
				
			||||||
  }
 | 
					  }
 | 
				
			||||||
 
 | 
				
			|||||||
							
								
								
									
										46
									
								
								src/routes/validate/measurement.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										46
									
								
								src/routes/validate/measurement.ts
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,46 @@
 | 
				
			|||||||
 | 
					import Joi from '@hapi/joi';
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import IdValidate from './id';
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					export default class MeasurementValidate {
 | 
				
			||||||
 | 
					  private static measurement = {
 | 
				
			||||||
 | 
					    values: Joi.object()
 | 
				
			||||||
 | 
					      .pattern(/.*/, Joi.alternatives()
 | 
				
			||||||
 | 
					        .try(
 | 
				
			||||||
 | 
					          Joi.string().max(128),
 | 
				
			||||||
 | 
					          Joi.number(),
 | 
				
			||||||
 | 
					          Joi.boolean(),
 | 
				
			||||||
 | 
					          Joi.array()
 | 
				
			||||||
 | 
					        )
 | 
				
			||||||
 | 
					      )
 | 
				
			||||||
 | 
					  };
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  static input (data, param) {
 | 
				
			||||||
 | 
					    if (param === 'new') {
 | 
				
			||||||
 | 
					      return Joi.object({
 | 
				
			||||||
 | 
					        condition_id: IdValidate.get().required(),
 | 
				
			||||||
 | 
					        values: this.measurement.values.required(),
 | 
				
			||||||
 | 
					        measurement_template: IdValidate.get().required()
 | 
				
			||||||
 | 
					      }).validate(data);
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					    else if (param === 'change') {
 | 
				
			||||||
 | 
					      return Joi.object({
 | 
				
			||||||
 | 
					        values: this.measurement.values
 | 
				
			||||||
 | 
					      }).validate(data);
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					    else {
 | 
				
			||||||
 | 
					      return{error: 'No parameter specified!', value: {}};
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					  }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  static output (data) {
 | 
				
			||||||
 | 
					    data = IdValidate.stringify(data);
 | 
				
			||||||
 | 
					    const {value, error} = Joi.object({
 | 
				
			||||||
 | 
					      _id: IdValidate.get(),
 | 
				
			||||||
 | 
					      condition_id: IdValidate.get(),
 | 
				
			||||||
 | 
					      values: this.measurement.values,
 | 
				
			||||||
 | 
					      measurement_template: IdValidate.get()
 | 
				
			||||||
 | 
					    }).validate(data, {stripUnknown: true});
 | 
				
			||||||
 | 
					    return error !== undefined? null : value;
 | 
				
			||||||
 | 
					  }
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
@@ -6,7 +6,7 @@ export default class ParametersValidate {
 | 
				
			|||||||
    parameters.forEach(parameter => {
 | 
					    parameters.forEach(parameter => {
 | 
				
			||||||
      if (parameter.range.hasOwnProperty('values')) {
 | 
					      if (parameter.range.hasOwnProperty('values')) {
 | 
				
			||||||
        joiObject[parameter.name] = Joi.alternatives()
 | 
					        joiObject[parameter.name] = Joi.alternatives()
 | 
				
			||||||
          .try(Joi.string(), Joi.number(), Joi.boolean())
 | 
					          .try(Joi.string().max(128), Joi.number(), Joi.boolean())
 | 
				
			||||||
          .valid(...parameter.range.values);
 | 
					          .valid(...parameter.range.values);
 | 
				
			||||||
      }
 | 
					      }
 | 
				
			||||||
      else if (parameter.range.hasOwnProperty('min') && parameter.range.hasOwnProperty('max')) {
 | 
					      else if (parameter.range.hasOwnProperty('min') && parameter.range.hasOwnProperty('max')) {
 | 
				
			||||||
@@ -22,9 +22,19 @@ export default class ParametersValidate {
 | 
				
			|||||||
        joiObject[parameter.name] = Joi.number()
 | 
					        joiObject[parameter.name] = Joi.number()
 | 
				
			||||||
          .max(parameter.range.max);
 | 
					          .max(parameter.range.max);
 | 
				
			||||||
      }
 | 
					      }
 | 
				
			||||||
 | 
					      else if (parameter.range.hasOwnProperty('type')) {
 | 
				
			||||||
 | 
					        switch (parameter.range.type) {
 | 
				
			||||||
 | 
					          case 'array':
 | 
				
			||||||
 | 
					            joiObject[parameter.name] = Joi.array();
 | 
				
			||||||
 | 
					            break;
 | 
				
			||||||
 | 
					          default:
 | 
				
			||||||
 | 
					            joiObject[parameter.name] = Joi.string().max(128);
 | 
				
			||||||
 | 
					            break;
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					      }
 | 
				
			||||||
      else {
 | 
					      else {
 | 
				
			||||||
        joiObject[parameter.name] = Joi.alternatives()
 | 
					        joiObject[parameter.name] = Joi.alternatives()
 | 
				
			||||||
          .try(Joi.string(), Joi.number(), Joi.boolean());
 | 
					          .try(Joi.string().max(128), Joi.number(), Joi.boolean());
 | 
				
			||||||
      }
 | 
					      }
 | 
				
			||||||
      if (param === 'new') {
 | 
					      if (param === 'new') {
 | 
				
			||||||
        joiObject[parameter.name] = joiObject[parameter.name].required()
 | 
					        joiObject[parameter.name] = joiObject[parameter.name].required()
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -229,6 +229,43 @@
 | 
				
			|||||||
        "__v": 0
 | 
					        "__v": 0
 | 
				
			||||||
      }
 | 
					      }
 | 
				
			||||||
    ],
 | 
					    ],
 | 
				
			||||||
 | 
					    "measurements": [
 | 
				
			||||||
 | 
					      {
 | 
				
			||||||
 | 
					        "_id": {"$oid":"800000000000000000000001"},
 | 
				
			||||||
 | 
					        "condition_id": {"$oid":"700000000000000000000001"},
 | 
				
			||||||
 | 
					        "values": {
 | 
				
			||||||
 | 
					          "dpt": [
 | 
				
			||||||
 | 
					            [3997.12558,98.00555],
 | 
				
			||||||
 | 
					            [3995.08519,98.03253],
 | 
				
			||||||
 | 
					            [3993.04480,98.02657]
 | 
				
			||||||
 | 
					          ]
 | 
				
			||||||
 | 
					        },
 | 
				
			||||||
 | 
					        "status": 10,
 | 
				
			||||||
 | 
					        "measurement_template": {"$oid":"300000000000000000000001"},
 | 
				
			||||||
 | 
					        "__v": 0
 | 
				
			||||||
 | 
					      },
 | 
				
			||||||
 | 
					      {
 | 
				
			||||||
 | 
					        "_id": {"$oid":"800000000000000000000002"},
 | 
				
			||||||
 | 
					        "condition_id": {"$oid":"700000000000000000000002"},
 | 
				
			||||||
 | 
					        "values": {
 | 
				
			||||||
 | 
					          "weight %": 0.5,
 | 
				
			||||||
 | 
					          "standard deviation": 0.2
 | 
				
			||||||
 | 
					        },
 | 
				
			||||||
 | 
					        "status": 10,
 | 
				
			||||||
 | 
					        "measurement_template": {"$oid":"300000000000000000000002"},
 | 
				
			||||||
 | 
					        "__v": 0
 | 
				
			||||||
 | 
					      },
 | 
				
			||||||
 | 
					      {
 | 
				
			||||||
 | 
					        "_id": {"$oid":"800000000000000000000003"},
 | 
				
			||||||
 | 
					        "condition_id": {"$oid":"700000000000000000000003"},
 | 
				
			||||||
 | 
					        "values": {
 | 
				
			||||||
 | 
					          "val1": 1
 | 
				
			||||||
 | 
					        },
 | 
				
			||||||
 | 
					        "status": 0,
 | 
				
			||||||
 | 
					        "measurement_template": {"$oid":"300000000000000000000003"},
 | 
				
			||||||
 | 
					        "__v": 0
 | 
				
			||||||
 | 
					      }
 | 
				
			||||||
 | 
					    ],
 | 
				
			||||||
    "treatment_templates": [
 | 
					    "treatment_templates": [
 | 
				
			||||||
      {
 | 
					      {
 | 
				
			||||||
        "_id": {"$oid":"200000000000000000000001"},
 | 
					        "_id": {"$oid":"200000000000000000000001"},
 | 
				
			||||||
@@ -272,7 +309,9 @@
 | 
				
			|||||||
        "parameters": [
 | 
					        "parameters": [
 | 
				
			||||||
          {
 | 
					          {
 | 
				
			||||||
            "name": "dpt",
 | 
					            "name": "dpt",
 | 
				
			||||||
            "range": {}
 | 
					            "range": {
 | 
				
			||||||
 | 
					              "type": "array"
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
          }
 | 
					          }
 | 
				
			||||||
        ],
 | 
					        ],
 | 
				
			||||||
        "__v": 0
 | 
					        "__v": 0
 | 
				
			||||||
@@ -297,6 +336,19 @@
 | 
				
			|||||||
          }
 | 
					          }
 | 
				
			||||||
        ],
 | 
					        ],
 | 
				
			||||||
        "__v": 0
 | 
					        "__v": 0
 | 
				
			||||||
 | 
					      },
 | 
				
			||||||
 | 
					      {
 | 
				
			||||||
 | 
					        "_id": {"$oid":"300000000000000000000003"},
 | 
				
			||||||
 | 
					        "name": "mt 3",
 | 
				
			||||||
 | 
					        "parameters": [
 | 
				
			||||||
 | 
					          {
 | 
				
			||||||
 | 
					            "name": "val1",
 | 
				
			||||||
 | 
					            "range": {
 | 
				
			||||||
 | 
					              "values": [1,2,3]
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					          }
 | 
				
			||||||
 | 
					        ],
 | 
				
			||||||
 | 
					        "__v": 0
 | 
				
			||||||
      }
 | 
					      }
 | 
				
			||||||
    ],
 | 
					    ],
 | 
				
			||||||
    "users": [
 | 
					    "users": [
 | 
				
			||||||
 
 | 
				
			|||||||
		Reference in New Issue
	
	Block a user