adapted /measurements to use sample_id
This commit is contained in:
		| @@ -54,7 +54,6 @@ tags: | |||||||
|   - name: / |   - name: / | ||||||
|   - name: /sample |   - name: /sample | ||||||
|   - name: /material |   - name: /material | ||||||
|   - name: /condition |  | ||||||
|   - name: /measurement |   - name: /measurement | ||||||
|   - name: /template |   - name: /template | ||||||
|   - name: /model |   - name: /model | ||||||
|   | |||||||
| @@ -133,7 +133,7 @@ Measurement: | |||||||
|   allOf: |   allOf: | ||||||
|     - $ref: 'api.yaml#/components/schemas/_Id' |     - $ref: 'api.yaml#/components/schemas/_Id' | ||||||
|   properties: |   properties: | ||||||
|     condition_id: |     sample_id: | ||||||
|       $ref: 'api.yaml#/components/schemas/Id' |       $ref: 'api.yaml#/components/schemas/Id' | ||||||
|     values: |     values: | ||||||
|       type: object |       type: object | ||||||
|   | |||||||
| @@ -12,6 +12,7 @@ import db from './db'; | |||||||
| // TODO: condition values not needed on initial add | // TODO: condition values not needed on initial add | ||||||
| // TODO: add multiple samples at once | // TODO: add multiple samples at once | ||||||
| // TODO: coverage | // TODO: coverage | ||||||
|  | // TODO: think about the display of deleted/new samples and validation in data and UI | ||||||
|  |  | ||||||
| // tell if server is running in debug or production environment | // tell if server is running in debug or production environment | ||||||
| console.info(process.env.NODE_ENV === 'production' ? '===== PRODUCTION =====' : process.env.NODE_ENV === 'test' ? '' :'===== DEVELOPMENT ====='); | console.info(process.env.NODE_ENV === 'production' ? '===== PRODUCTION =====' : process.env.NODE_ENV === 'test' ? '' :'===== DEVELOPMENT ====='); | ||||||
|   | |||||||
| @@ -9,6 +9,6 @@ const MeasurementSchema = new mongoose.Schema({ | |||||||
|   values: mongoose.Schema.Types.Mixed, |   values: mongoose.Schema.Types.Mixed, | ||||||
|   measurement_template: {type: mongoose.Schema.Types.ObjectId, ref: MeasurementTemplateModel}, |   measurement_template: {type: mongoose.Schema.Types.ObjectId, ref: MeasurementTemplateModel}, | ||||||
|   status: Number |   status: Number | ||||||
| }); | }, {minimize: false}); | ||||||
|  |  | ||||||
| export default mongoose.model('measurement', MeasurementSchema); | export default mongoose.model('measurement', MeasurementSchema); | ||||||
| @@ -3,7 +3,7 @@ import MeasurementModel from '../models/measurement'; | |||||||
| import TestHelper from "../test/helper"; | import TestHelper from "../test/helper"; | ||||||
| import globals from '../globals'; | import globals from '../globals'; | ||||||
|  |  | ||||||
| // TODO: allow empty values | // TODO: restore measurements for m/a | ||||||
|  |  | ||||||
|  |  | ||||||
| describe('/measurement', () => { | describe('/measurement', () => { | ||||||
| @@ -19,7 +19,7 @@ describe('/measurement', () => { | |||||||
|         url: '/measurement/800000000000000000000001', |         url: '/measurement/800000000000000000000001', | ||||||
|         auth: {basic: 'janedoe'}, |         auth: {basic: 'janedoe'}, | ||||||
|         httpStatus: 200, |         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'} |         res: {_id: '800000000000000000000001', sample_id: '400000000000000000000001', 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 => { |     it('returns the measurement for an API key', done => { | ||||||
| @@ -28,7 +28,24 @@ describe('/measurement', () => { | |||||||
|         url: '/measurement/800000000000000000000001', |         url: '/measurement/800000000000000000000001', | ||||||
|         auth: {key: 'janedoe'}, |         auth: {key: 'janedoe'}, | ||||||
|         httpStatus: 200, |         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'} |         res: {_id: '800000000000000000000001', sample_id: '400000000000000000000001', values: {dpt: [[3997.12558,98.00555],[3995.08519,98.03253],[3993.04480,98.02657]]}, measurement_template: '300000000000000000000001'} | ||||||
|  |       }); | ||||||
|  |     }); | ||||||
|  |     it('returns deleted measurements for a maintain/admin user', done => { | ||||||
|  |       TestHelper.request(server, done, { | ||||||
|  |         method: 'get', | ||||||
|  |         url: '/measurement/800000000000000000000004', | ||||||
|  |         auth: {basic: 'admin'}, | ||||||
|  |         httpStatus: 200, | ||||||
|  |         res: {_id: '800000000000000000000004', sample_id: '400000000000000000000003', values: {val1: 1}, measurement_template: '300000000000000000000003'} | ||||||
|  |       }); | ||||||
|  |     }); | ||||||
|  |     it('rejects requests for deleted measurements from a write user', done => { | ||||||
|  |       TestHelper.request(server, done, { | ||||||
|  |         method: 'get', | ||||||
|  |         url: '/measurement/800000000000000000000004', | ||||||
|  |         auth: {basic: 'janedoe'}, | ||||||
|  |         httpStatus: 403 | ||||||
|       }); |       }); | ||||||
|     }); |     }); | ||||||
|     it('rejects an invalid id', done => { |     it('rejects an invalid id', done => { | ||||||
| @@ -64,7 +81,7 @@ describe('/measurement', () => { | |||||||
|         auth: {basic: 'janedoe'}, |         auth: {basic: 'janedoe'}, | ||||||
|         httpStatus: 200, |         httpStatus: 200, | ||||||
|         req: {}, |         req: {}, | ||||||
|         res: {_id: '800000000000000000000001', condition_id: '700000000000000000000001', values: {dpt: [[3997.12558,98.00555],[3995.08519,98.03253],[3993.04480,98.02657]]}, measurement_template: '300000000000000000000001'} |         res: {_id: '800000000000000000000001', sample_id: '400000000000000000000001', values: {dpt: [[3997.12558,98.00555],[3995.08519,98.03253],[3993.04480,98.02657]]}, measurement_template: '300000000000000000000001'} | ||||||
|       }); |       }); | ||||||
|     }); |     }); | ||||||
|     it('keeps unchanged values', done => { |     it('keeps unchanged values', done => { | ||||||
| @@ -76,7 +93,7 @@ describe('/measurement', () => { | |||||||
|         req: {values: {dpt: [[3997.12558,98.00555],[3995.08519,98.03253],[3993.04480,98.02657]]}} |         req: {values: {dpt: [[3997.12558,98.00555],[3995.08519,98.03253],[3993.04480,98.02657]]}} | ||||||
|       }).end((err, res) => { |       }).end((err, res) => { | ||||||
|         if (err) return done(err); |         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'}); |         should(res.body).be.eql({_id: '800000000000000000000001', sample_id: '400000000000000000000001', 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) => { |         MeasurementModel.findById('800000000000000000000001').lean().exec((err, data: any) => { | ||||||
|           if (err) return done(err); |           if (err) return done(err); | ||||||
|           should(data).have.property('status',globals.status.validated); |           should(data).have.property('status',globals.status.validated); | ||||||
| @@ -93,7 +110,7 @@ describe('/measurement', () => { | |||||||
|         req: {values: {'weight %': 0.5}} |         req: {values: {'weight %': 0.5}} | ||||||
|       }).end((err, res) => { |       }).end((err, res) => { | ||||||
|         if (err) return done(err); |         if (err) return done(err); | ||||||
|         should(res.body).be.eql({_id: '800000000000000000000002', condition_id: '700000000000000000000002', values: {'weight %': 0.5, 'standard deviation': 0.2}, measurement_template: '300000000000000000000002'}); |         should(res.body).be.eql({_id: '800000000000000000000002', sample_id: '400000000000000000000002', values: {'weight %': 0.5, 'standard deviation': 0.2}, measurement_template: '300000000000000000000002'}); | ||||||
|         MeasurementModel.findById('800000000000000000000002').lean().exec((err, data: any) => { |         MeasurementModel.findById('800000000000000000000002').lean().exec((err, data: any) => { | ||||||
|           if (err) return done(err); |           if (err) return done(err); | ||||||
|           should(data).have.property('status',globals.status.validated); |           should(data).have.property('status',globals.status.validated); | ||||||
| @@ -110,10 +127,10 @@ describe('/measurement', () => { | |||||||
|         req: {values: {dpt: [[1,2],[3,4],[5,6]]}} |         req: {values: {dpt: [[1,2],[3,4],[5,6]]}} | ||||||
|       }).end((err, res) => { |       }).end((err, res) => { | ||||||
|         if (err) return done(err); |         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'}); |         should(res.body).be.eql({_id: '800000000000000000000001', sample_id: '400000000000000000000001', values: {dpt: [[1,2],[3,4],[5,6]]}, measurement_template: '300000000000000000000001'}); | ||||||
|         MeasurementModel.findById('800000000000000000000001').lean().exec((err, data: any) => { |         MeasurementModel.findById('800000000000000000000001').lean().exec((err, data: any) => { | ||||||
|           should(data).have.only.keys('_id', 'condition_id', 'values', 'measurement_template', 'status', '__v'); |           should(data).have.only.keys('_id', 'sample_id', 'values', 'measurement_template', 'status', '__v'); | ||||||
|           should(data.condition_id.toString()).be.eql('700000000000000000000001'); |           should(data.sample_id.toString()).be.eql('400000000000000000000001'); | ||||||
|           should(data.measurement_template.toString()).be.eql('300000000000000000000001'); |           should(data.measurement_template.toString()).be.eql('300000000000000000000001'); | ||||||
|           should(data).have.property('status',globals.status.new); |           should(data).have.property('status',globals.status.new); | ||||||
|           should(data).have.property('values'); |           should(data).have.property('values'); | ||||||
| @@ -129,7 +146,17 @@ describe('/measurement', () => { | |||||||
|         auth: {basic: 'janedoe'}, |         auth: {basic: 'janedoe'}, | ||||||
|         httpStatus: 200, |         httpStatus: 200, | ||||||
|         req: {values: {'weight %': 0.9}}, |         req: {values: {'weight %': 0.9}}, | ||||||
|         res: {_id: '800000000000000000000002', condition_id: '700000000000000000000002', values: {'weight %': 0.9, 'standard deviation': 0.2}, measurement_template: '300000000000000000000002'} |         res: {_id: '800000000000000000000002', sample_id: '400000000000000000000002', values: {'weight %': 0.9, 'standard deviation': 0.2}, measurement_template: '300000000000000000000002'} | ||||||
|  |       }); | ||||||
|  |     }); | ||||||
|  |     it('allows keeping empty values empty', done => { | ||||||
|  |       TestHelper.request(server, done, { | ||||||
|  |         method: 'put', | ||||||
|  |         url: '/measurement/800000000000000000000005', | ||||||
|  |         auth: {basic: 'janedoe'}, | ||||||
|  |         httpStatus: 200, | ||||||
|  |         req: {values: {'weight %': 0.9}}, | ||||||
|  |         res: {_id: '800000000000000000000005', sample_id: '400000000000000000000002', values: {'weight %': 0.9, 'standard deviation': null}, measurement_template: '300000000000000000000002'} | ||||||
|       }); |       }); | ||||||
|     }); |     }); | ||||||
|     it('rejects not specified values', done => { |     it('rejects not specified values', done => { | ||||||
| @@ -149,7 +176,7 @@ describe('/measurement', () => { | |||||||
|         auth: {basic: 'admin'}, |         auth: {basic: 'admin'}, | ||||||
|         httpStatus: 400, |         httpStatus: 400, | ||||||
|         req: {values: {val1: 4}}, |         req: {values: {val1: 4}}, | ||||||
|         res: {status: 'Invalid body format', details: '"val1" must be one of [1, 2, 3]'} |         res: {status: 'Invalid body format', details: '"val1" must be one of [1, 2, 3, null]'} | ||||||
|       }); |       }); | ||||||
|     }); |     }); | ||||||
|     it('rejects a value below minimum range', done => { |     it('rejects a value below minimum range', done => { | ||||||
| @@ -182,6 +209,16 @@ describe('/measurement', () => { | |||||||
|         res: {status: 'Invalid body format', details: '"measurement_template" is not allowed'} |         res: {status: 'Invalid body format', details: '"measurement_template" is not allowed'} | ||||||
|       }); |       }); | ||||||
|     }); |     }); | ||||||
|  |     it('rejects a new sample id', done => { | ||||||
|  |       TestHelper.request(server, done, { | ||||||
|  |         method: 'put', | ||||||
|  |         url: '/measurement/800000000000000000000002', | ||||||
|  |         auth: {basic: 'janedoe'}, | ||||||
|  |         httpStatus: 400, | ||||||
|  |         req: {values: {'weight %': 0.9, 'standard deviation': 0.3}, sample_id: '400000000000000000000002'}, | ||||||
|  |         res: {status: 'Invalid body format', details: '"sample_id" is not allowed'} | ||||||
|  |       }); | ||||||
|  |     }); | ||||||
|     it('rejects editing a measurement for a write user who did not create this measurement', done => { |     it('rejects editing a measurement for a write user who did not create this measurement', done => { | ||||||
|       TestHelper.request(server, done, { |       TestHelper.request(server, done, { | ||||||
|         method: 'put', |         method: 'put', | ||||||
| @@ -198,7 +235,7 @@ describe('/measurement', () => { | |||||||
|         auth: {basic: 'admin'}, |         auth: {basic: 'admin'}, | ||||||
|         httpStatus: 200, |         httpStatus: 200, | ||||||
|         req: {values: {'weight %': 0.9, 'standard deviation': 0.3}}, |         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'} |         res: {_id: '800000000000000000000002', sample_id: '400000000000000000000002', values: {'weight %': 0.9, 'standard deviation': 0.3}, measurement_template: '300000000000000000000002'} | ||||||
|       }); |       }); | ||||||
|     }); |     }); | ||||||
|     it('rejects an invalid id', done => { |     it('rejects an invalid id', done => { | ||||||
| @@ -327,12 +364,12 @@ describe('/measurement', () => { | |||||||
|         url: '/measurement/new', |         url: '/measurement/new', | ||||||
|         auth: {basic: 'janedoe'}, |         auth: {basic: 'janedoe'}, | ||||||
|         httpStatus: 200, |         httpStatus: 200, | ||||||
|         req: {condition_id: '700000000000000000000001', values: {'weight %': 0.8, 'standard deviation': 0.1}, measurement_template: '300000000000000000000002'} |         req: {sample_id: '400000000000000000000001', values: {'weight %': 0.8, 'standard deviation': 0.1}, measurement_template: '300000000000000000000002'} | ||||||
|       }).end((err, res) => { |       }).end((err, res) => { | ||||||
|         if (err) return done(err); |         if (err) return done(err); | ||||||
|         should(res.body).have.only.keys('_id', 'condition_id', 'values', 'measurement_template'); |         should(res.body).have.only.keys('_id', 'sample_id', 'values', 'measurement_template'); | ||||||
|         should(res.body).have.property('_id').be.type('string'); |         should(res.body).have.property('_id').be.type('string'); | ||||||
|         should(res.body).have.property('condition_id', '700000000000000000000001'); |         should(res.body).have.property('sample_id', '400000000000000000000001'); | ||||||
|         should(res.body).have.property('measurement_template', '300000000000000000000002'); |         should(res.body).have.property('measurement_template', '300000000000000000000002'); | ||||||
|         should(res.body).have.property('values'); |         should(res.body).have.property('values'); | ||||||
|         should(res.body.values).have.property('weight %', 0.8); |         should(res.body.values).have.property('weight %', 0.8); | ||||||
| @@ -346,13 +383,13 @@ describe('/measurement', () => { | |||||||
|         url: '/measurement/new', |         url: '/measurement/new', | ||||||
|         auth: {basic: 'janedoe'}, |         auth: {basic: 'janedoe'}, | ||||||
|         httpStatus: 200, |         httpStatus: 200, | ||||||
|         req: {condition_id: '700000000000000000000001', values: {'weight %': 0.8, 'standard deviation': 0.1}, measurement_template: '300000000000000000000002'} |         req: {sample_id: '400000000000000000000001', values: {'weight %': 0.8, 'standard deviation': 0.1}, measurement_template: '300000000000000000000002'} | ||||||
|       }).end((err, res) => { |       }).end((err, res) => { | ||||||
|         if (err) return done(err); |         if (err) return done(err); | ||||||
|         MeasurementModel.findById(res.body._id).lean().exec((err, data: any) => { |         MeasurementModel.findById(res.body._id).lean().exec((err, data: any) => { | ||||||
|           if (err) return done(err); |           if (err) return done(err); | ||||||
|           should(data).have.only.keys('_id', 'condition_id', 'values', 'measurement_template', 'status', '__v'); |           should(data).have.only.keys('_id', 'sample_id', 'values', 'measurement_template', 'status', '__v'); | ||||||
|           should(data.condition_id.toString()).be.eql('700000000000000000000001'); |           should(data.sample_id.toString()).be.eql('400000000000000000000001'); | ||||||
|           should(data.measurement_template.toString()).be.eql('300000000000000000000002'); |           should(data.measurement_template.toString()).be.eql('300000000000000000000002'); | ||||||
|           should(data).have.property('status', 0); |           should(data).have.property('status', 0); | ||||||
|           should(data).have.property('values'); |           should(data).have.property('values'); | ||||||
| @@ -362,24 +399,24 @@ describe('/measurement', () => { | |||||||
|         }); |         }); | ||||||
|       }); |       }); | ||||||
|     }); |     }); | ||||||
|     it('rejects an invalid condition id', done => { |     it('rejects an invalid sample id', done => { | ||||||
|       TestHelper.request(server, done, { |       TestHelper.request(server, done, { | ||||||
|         method: 'post', |         method: 'post', | ||||||
|         url: '/measurement/new', |         url: '/measurement/new', | ||||||
|         auth: {basic: 'janedoe'}, |         auth: {basic: 'janedoe'}, | ||||||
|         httpStatus: 400, |         httpStatus: 400, | ||||||
|         req: {condition_id: '700000000000h00000000001', values: {'weight %': 0.8, 'standard deviation': 0.1}, measurement_template: '300000000000000000000002'}, |         req: {sample_id: '400000000000h00000000001', 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}/'} |         res: {status: 'Invalid body format', details: '"sample_id" with value "400000000000h00000000001" fails to match the required pattern: /[0-9a-f]{24}/'} | ||||||
|       }); |       }); | ||||||
|     }); |     }); | ||||||
|     it('rejects a condition id not available', done => { |     it('rejects a sample id not available', done => { | ||||||
|       TestHelper.request(server, done, { |       TestHelper.request(server, done, { | ||||||
|         method: 'post', |         method: 'post', | ||||||
|         url: '/measurement/new', |         url: '/measurement/new', | ||||||
|         auth: {basic: 'janedoe'}, |         auth: {basic: 'janedoe'}, | ||||||
|         httpStatus: 400, |         httpStatus: 400, | ||||||
|         req: {condition_id: '000000000000000000000001', values: {'weight %': 0.8, 'standard deviation': 0.1}, measurement_template: '300000000000000000000002'}, |         req: {sample_id: '000000000000000000000001', values: {'weight %': 0.8, 'standard deviation': 0.1}, measurement_template: '300000000000000000000002'}, | ||||||
|         res: {status: 'Condition id not available'} |         res: {status: 'Sample id not available'} | ||||||
|       }); |       }); | ||||||
|     }); |     }); | ||||||
|     it('rejects an invalid measurement_template id', done => { |     it('rejects an invalid measurement_template id', done => { | ||||||
| @@ -388,7 +425,7 @@ describe('/measurement', () => { | |||||||
|         url: '/measurement/new', |         url: '/measurement/new', | ||||||
|         auth: {basic: 'janedoe'}, |         auth: {basic: 'janedoe'}, | ||||||
|         httpStatus: 400, |         httpStatus: 400, | ||||||
|         req: {condition_id: '700000000000000000000001', values: {'weight %': 0.8, 'standard deviation': 0.1}, measurement_template: '30000000000h000000000002'}, |         req: {sample_id: '400000000000000000000001', 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}/'} |         res: {status: 'Invalid body format', details: '"measurement_template" with value "30000000000h000000000002" fails to match the required pattern: /[0-9a-f]{24}/'} | ||||||
|       }); |       }); | ||||||
|     }); |     }); | ||||||
| @@ -398,7 +435,7 @@ describe('/measurement', () => { | |||||||
|         url: '/measurement/new', |         url: '/measurement/new', | ||||||
|         auth: {basic: 'janedoe'}, |         auth: {basic: 'janedoe'}, | ||||||
|         httpStatus: 400, |         httpStatus: 400, | ||||||
|         req: {condition_id: '700000000000000000000001', values: {'weight %': 0.8, 'standard deviation': 0.1}, measurement_template: '000000000000000000000002'}, |         req: {sample_id: '400000000000000000000001', values: {'weight %': 0.8, 'standard deviation': 0.1}, measurement_template: '000000000000000000000002'}, | ||||||
|         res: {status: 'Measurement template not available'} |         res: {status: 'Measurement template not available'} | ||||||
|       }); |       }); | ||||||
|     }); |     }); | ||||||
| @@ -408,18 +445,27 @@ describe('/measurement', () => { | |||||||
|         url: '/measurement/new', |         url: '/measurement/new', | ||||||
|         auth: {basic: 'janedoe'}, |         auth: {basic: 'janedoe'}, | ||||||
|         httpStatus: 400, |         httpStatus: 400, | ||||||
|         req: {condition_id: '700000000000000000000001', values: {'weight %': 0.8, 'standard deviation': 0.1, xx: 44}, measurement_template: '300000000000000000000002'}, |         req: {sample_id: '400000000000000000000001', values: {'weight %': 0.8, 'standard deviation': 0.1, xx: 44}, measurement_template: '300000000000000000000002'}, | ||||||
|         res: {status: 'Invalid body format', details: '"xx" is not allowed'} |         res: {status: 'Invalid body format', details: '"xx" is not allowed'} | ||||||
|       }); |       }); | ||||||
|     }); |     }); | ||||||
|     it('rejects missing values', done => { |     it('accepts missing values', done => { | ||||||
|       TestHelper.request(server, done, { |       TestHelper.request(server, done, { | ||||||
|         method: 'post', |         method: 'post', | ||||||
|         url: '/measurement/new', |         url: '/measurement/new', | ||||||
|         auth: {basic: 'janedoe'}, |         auth: {basic: 'janedoe'}, | ||||||
|         httpStatus: 400, |         httpStatus: 200, | ||||||
|         req: {condition_id: '700000000000000000000001', values: {'weight %': 0.8}, measurement_template: '300000000000000000000002'}, |         req: {sample_id: '400000000000000000000001', values: {'weight %': 0.8}, measurement_template: '300000000000000000000002'} | ||||||
|         res: {status: 'Invalid body format', details: '"standard deviation" is required'} |       }).end((err, res) => { | ||||||
|  |         if (err) return done(err); | ||||||
|  |         should(res.body).have.only.keys('_id', 'sample_id', 'values', 'measurement_template'); | ||||||
|  |         should(res.body).have.property('_id').be.type('string'); | ||||||
|  |         should(res.body).have.property('sample_id', '400000000000000000000001'); | ||||||
|  |         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', null); | ||||||
|  |         done(); | ||||||
|       }); |       }); | ||||||
|     }); |     }); | ||||||
|     it('rejects a value not in the value range', done => { |     it('rejects a value not in the value range', done => { | ||||||
| @@ -428,8 +474,8 @@ describe('/measurement', () => { | |||||||
|         url: '/measurement/new', |         url: '/measurement/new', | ||||||
|         auth: {basic: 'janedoe'}, |         auth: {basic: 'janedoe'}, | ||||||
|         httpStatus: 400, |         httpStatus: 400, | ||||||
|         req: {condition_id: '700000000000000000000001', values: {val1: 4}, measurement_template: '300000000000000000000003'}, |         req: {sample_id: '400000000000000000000001', values: {val1: 4}, measurement_template: '300000000000000000000003'}, | ||||||
|         res: {status: 'Invalid body format', details: '"val1" must be one of [1, 2, 3]'} |         res: {status: 'Invalid body format', details: '"val1" must be one of [1, 2, 3, null]'} | ||||||
|       }); |       }); | ||||||
|     }); |     }); | ||||||
|     it('rejects a value below minimum range', done => { |     it('rejects a value below minimum range', done => { | ||||||
| @@ -438,7 +484,7 @@ describe('/measurement', () => { | |||||||
|         url: '/measurement/new', |         url: '/measurement/new', | ||||||
|         auth: {basic: 'janedoe'}, |         auth: {basic: 'janedoe'}, | ||||||
|         httpStatus: 400, |         httpStatus: 400, | ||||||
|         req: {condition_id: '700000000000000000000001', values: {'weight %': -1, 'standard deviation': 0.1}, measurement_template: '300000000000000000000002'}, |         req: {sample_id: '400000000000000000000001', 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'} |         res: {status: 'Invalid body format', details: '"weight %" must be larger than or equal to 0'} | ||||||
|       }); |       }); | ||||||
|     }); |     }); | ||||||
| @@ -448,18 +494,18 @@ describe('/measurement', () => { | |||||||
|         url: '/measurement/new', |         url: '/measurement/new', | ||||||
|         auth: {basic: 'janedoe'}, |         auth: {basic: 'janedoe'}, | ||||||
|         httpStatus: 400, |         httpStatus: 400, | ||||||
|         req: {condition_id: '700000000000000000000001', values: {'weight %': 0.8, 'standard deviation': 2}, measurement_template: '300000000000000000000002'}, |         req: {sample_id: '400000000000000000000001', 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'} |         res: {status: 'Invalid body format', details: '"standard deviation" must be less than or equal to 0.5'} | ||||||
|       }); |       }); | ||||||
|     }); |     }); | ||||||
|     it('rejects a missing condition id', done => { |     it('rejects a missing sample id', done => { | ||||||
|       TestHelper.request(server, done, { |       TestHelper.request(server, done, { | ||||||
|         method: 'post', |         method: 'post', | ||||||
|         url: '/measurement/new', |         url: '/measurement/new', | ||||||
|         auth: {basic: 'janedoe'}, |         auth: {basic: 'janedoe'}, | ||||||
|         httpStatus: 400, |         httpStatus: 400, | ||||||
|         req: {values: {'weight %': 0.8, 'standard deviation': 0.1}, measurement_template: '300000000000000000000002'}, |         req: {values: {'weight %': 0.8, 'standard deviation': 0.1}, measurement_template: '300000000000000000000002'}, | ||||||
|         res: {status: 'Invalid body format', details: '"condition_id" is required'} |         res: {status: 'Invalid body format', details: '"sample_id" is required'} | ||||||
|       }); |       }); | ||||||
|     }); |     }); | ||||||
|     it('rejects a missing measurement_template', done => { |     it('rejects a missing measurement_template', done => { | ||||||
| @@ -468,7 +514,7 @@ describe('/measurement', () => { | |||||||
|         url: '/measurement/new', |         url: '/measurement/new', | ||||||
|         auth: {basic: 'janedoe'}, |         auth: {basic: 'janedoe'}, | ||||||
|         httpStatus: 400, |         httpStatus: 400, | ||||||
|         req: {condition_id: '700000000000000000000001', values: {'weight %': 0.8, 'standard deviation': 0.1}}, |         req: {sample_id: '400000000000000000000001', values: {'weight %': 0.8, 'standard deviation': 0.1}}, | ||||||
|         res: {status: 'Invalid body format', details: '"measurement_template" is required'} |         res: {status: 'Invalid body format', details: '"measurement_template" is required'} | ||||||
|       }); |       }); | ||||||
|     }); |     }); | ||||||
| @@ -478,7 +524,7 @@ describe('/measurement', () => { | |||||||
|         url: '/measurement/new', |         url: '/measurement/new', | ||||||
|         auth: {basic: 'janedoe'}, |         auth: {basic: 'janedoe'}, | ||||||
|         httpStatus: 403, |         httpStatus: 403, | ||||||
|         req: {condition_id: '700000000000000000000003', values: {'weight %': 0.8, 'standard deviation': 0.1}, measurement_template: '300000000000000000000002'} |         req: {sample_id: '400000000000000000000003', 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 => { |     it('accepts adding a measurement to the sample of another user for a maintain/admin user', done => { | ||||||
| @@ -487,12 +533,12 @@ describe('/measurement', () => { | |||||||
|         url: '/measurement/new', |         url: '/measurement/new', | ||||||
|         auth: {basic: 'admin'}, |         auth: {basic: 'admin'}, | ||||||
|         httpStatus: 200, |         httpStatus: 200, | ||||||
|         req: {condition_id: '700000000000000000000001', values: {'weight %': 0.8, 'standard deviation': 0.1}, measurement_template: '300000000000000000000002'} |         req: {sample_id: '400000000000000000000001', values: {'weight %': 0.8, 'standard deviation': 0.1}, measurement_template: '300000000000000000000002'} | ||||||
|       }).end((err, res) => { |       }).end((err, res) => { | ||||||
|         if (err) return done(err); |         if (err) return done(err); | ||||||
|         should(res.body).have.only.keys('_id', 'condition_id', 'values', 'measurement_template'); |         should(res.body).have.only.keys('_id', 'sample_id', 'values', 'measurement_template'); | ||||||
|         should(res.body).have.property('_id').be.type('string'); |         should(res.body).have.property('_id').be.type('string'); | ||||||
|         should(res.body).have.property('condition_id', '700000000000000000000001'); |         should(res.body).have.property('sample_id', '400000000000000000000001'); | ||||||
|         should(res.body).have.property('measurement_template', '300000000000000000000002'); |         should(res.body).have.property('measurement_template', '300000000000000000000002'); | ||||||
|         should(res.body).have.property('values'); |         should(res.body).have.property('values'); | ||||||
|         should(res.body.values).have.property('weight %', 0.8); |         should(res.body.values).have.property('weight %', 0.8); | ||||||
| @@ -506,7 +552,7 @@ describe('/measurement', () => { | |||||||
|         url: '/measurement/new', |         url: '/measurement/new', | ||||||
|         auth: {key: 'janedoe'}, |         auth: {key: 'janedoe'}, | ||||||
|         httpStatus: 401, |         httpStatus: 401, | ||||||
|         req: {condition_id: '700000000000000000000001', values: {'weight %': 0.8, 'standard deviation': 0.1}, measurement_template: '300000000000000000000002'} |         req: {sample_id: '400000000000000000000001', values: {'weight %': 0.8, 'standard deviation': 0.1}, measurement_template: '300000000000000000000002'} | ||||||
|       }); |       }); | ||||||
|     }); |     }); | ||||||
|     it('rejects requests from a read user', done => { |     it('rejects requests from a read user', done => { | ||||||
| @@ -515,7 +561,7 @@ describe('/measurement', () => { | |||||||
|         url: '/measurement/new', |         url: '/measurement/new', | ||||||
|         auth: {basic: 'user'}, |         auth: {basic: 'user'}, | ||||||
|         httpStatus: 403, |         httpStatus: 403, | ||||||
|         req: {condition_id: '700000000000000000000001', values: {'weight %': 0.8, 'standard deviation': 0.1}, measurement_template: '300000000000000000000002'} |         req: {sample_id: '400000000000000000000001', values: {'weight %': 0.8, 'standard deviation': 0.1}, measurement_template: '300000000000000000000002'} | ||||||
|       }); |       }); | ||||||
|     }); |     }); | ||||||
|     it('rejects unauthorized requests', done => { |     it('rejects unauthorized requests', done => { | ||||||
| @@ -523,7 +569,7 @@ describe('/measurement', () => { | |||||||
|         method: 'post', |         method: 'post', | ||||||
|         url: '/measurement/new', |         url: '/measurement/new', | ||||||
|         httpStatus: 401, |         httpStatus: 401, | ||||||
|         req: {condition_id: '700000000000000000000001', values: {'weight %': 0.8, 'standard deviation': 0.1}, measurement_template: '300000000000000000000002'} |         req: {sample_id: '400000000000000000000001', values: {'weight %': 0.8, 'standard deviation': 0.1}, measurement_template: '300000000000000000000002'} | ||||||
|       }); |       }); | ||||||
|     }); |     }); | ||||||
|   }); |   }); | ||||||
|   | |||||||
| @@ -3,6 +3,7 @@ import _ from 'lodash'; | |||||||
|  |  | ||||||
| import MeasurementModel from '../models/measurement'; | import MeasurementModel from '../models/measurement'; | ||||||
| import MeasurementTemplateModel from '../models/measurement_template'; | import MeasurementTemplateModel from '../models/measurement_template'; | ||||||
|  | import SampleModel from '../models/sample'; | ||||||
| import MeasurementValidate from './validate/measurement'; | import MeasurementValidate from './validate/measurement'; | ||||||
| import IdValidate from './validate/id'; | import IdValidate from './validate/id'; | ||||||
| import res400 from './validate/res400'; | import res400 from './validate/res400'; | ||||||
| @@ -15,11 +16,12 @@ const router = express.Router(); | |||||||
| router.get('/measurement/' + IdValidate.parameter(), (req, res, next) => { | router.get('/measurement/' + IdValidate.parameter(), (req, res, next) => { | ||||||
|   if (!req.auth(res, ['read', 'write', 'maintain', 'dev', 'admin'], 'all')) return; |   if (!req.auth(res, ['read', 'write', 'maintain', 'dev', 'admin'], 'all')) return; | ||||||
|  |  | ||||||
|   MeasurementModel.findById(req.params.id).lean().exec((err, data) => { |   MeasurementModel.findById(req.params.id).lean().exec((err, data: any) => { | ||||||
|     if (err) return next(err); |     if (err) return next(err); | ||||||
|     if (!data) { |     if (!data) { | ||||||
|       return res.status(404).json({status: 'Not found'}); |       return res.status(404).json({status: 'Not found'}); | ||||||
|     } |     } | ||||||
|  |     if (data.status ===globals.status.deleted && !req.auth(res, ['maintain', 'admin'], 'all')) return;  // deleted measurements only available for maintain/admin | ||||||
|  |  | ||||||
|     res.json(MeasurementValidate.output(data)); |     res.json(MeasurementValidate.output(data)); | ||||||
|   }); |   }); | ||||||
| @@ -37,13 +39,13 @@ router.put('/measurement/' + IdValidate.parameter(), async (req, res, next) => { | |||||||
|     res.status(404).json({status: 'Not found'}); |     res.status(404).json({status: 'Not found'}); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   // add properties needed for conditionIdCheck |   // add properties needed for sampleIdCheck | ||||||
|   measurement.measurement_template = data.measurement_template; |   measurement.measurement_template = data.measurement_template; | ||||||
|   measurement.condition_id = data.condition_id; |   measurement.sample_id = data.sample_id; | ||||||
|   if (!await conditionIdCheck(measurement, req, res, next)) return; |   if (!await sampleIdCheck(measurement, req, res, next)) return; | ||||||
|  |  | ||||||
|   // check for changes |   // check for changes | ||||||
|   if (measurement.values) { |   if (measurement.values) {  // fill not changed values from database | ||||||
|     measurement.values = _.assign({}, data.values, measurement.values); |     measurement.values = _.assign({}, data.values, measurement.values); | ||||||
|     if (!_.isEqual(measurement.values, data.values)) { |     if (!_.isEqual(measurement.values, data.values)) { | ||||||
|       measurement.status = globals.status.new;  // set status to new |       measurement.status = globals.status.new;  // set status to new | ||||||
| @@ -53,6 +55,7 @@ router.put('/measurement/' + IdValidate.parameter(), async (req, res, next) => { | |||||||
|   if (!await templateCheck(measurement, 'change', res, next)) return; |   if (!await templateCheck(measurement, 'change', res, next)) return; | ||||||
|   await MeasurementModel.findByIdAndUpdate(req.params.id, measurement, {new: true}).lean().exec((err, data) => { |   await MeasurementModel.findByIdAndUpdate(req.params.id, measurement, {new: true}).lean().exec((err, data) => { | ||||||
|     if (err) return next(err); |     if (err) return next(err); | ||||||
|  |     console.log(data); | ||||||
|     res.json(MeasurementValidate.output(data)); |     res.json(MeasurementValidate.output(data)); | ||||||
|   }); |   }); | ||||||
| }); | }); | ||||||
| @@ -65,7 +68,7 @@ router.delete('/measurement/' + IdValidate.parameter(), (req, res, next) => { | |||||||
|     if (!data) { |     if (!data) { | ||||||
|       res.status(404).json({status: 'Not found'}); |       res.status(404).json({status: 'Not found'}); | ||||||
|     } |     } | ||||||
|     if (!await conditionIdCheck(data, req, res, next)) return; |     if (!await sampleIdCheck(data, req, res, next)) return; | ||||||
|     await MeasurementModel.findByIdAndUpdate(req.params.id, {status:globals.status.deleted}).lean().exec(err => { |     await MeasurementModel.findByIdAndUpdate(req.params.id, {status:globals.status.deleted}).lean().exec(err => { | ||||||
|       if (err) return next(err); |       if (err) return next(err); | ||||||
|       res.json({status: 'OK'}); |       res.json({status: 'OK'}); | ||||||
| @@ -79,12 +82,14 @@ router.post('/measurement/new', async (req, res, next) => { | |||||||
|   const {error, value: measurement} = MeasurementValidate.input(req.body, 'new'); |   const {error, value: measurement} = MeasurementValidate.input(req.body, 'new'); | ||||||
|   if (error) return res400(error, res); |   if (error) return res400(error, res); | ||||||
|  |  | ||||||
|   if (!await conditionIdCheck(measurement, req, res, next)) return; |   if (!await sampleIdCheck(measurement, req, res, next)) return; | ||||||
|   if (!await templateCheck(measurement, 'new', res, next)) return; |   measurement.values = await templateCheck(measurement, 'new', res, next); | ||||||
|  |   if (!measurement.values) return; | ||||||
|  |  | ||||||
|   measurement.status = 0; |   measurement.status = 0; | ||||||
|   await new MeasurementModel(measurement).save((err, data) => { |   await new MeasurementModel(measurement).save((err, data) => { | ||||||
|     if (err) return next(err); |     if (err) return next(err); | ||||||
|  |     console.log(data); | ||||||
|     res.json(MeasurementValidate.output(data.toObject())); |     res.json(MeasurementValidate.output(data.toObject())); | ||||||
|   }); |   }); | ||||||
| }); | }); | ||||||
| @@ -93,25 +98,38 @@ router.post('/measurement/new', async (req, res, next) => { | |||||||
| module.exports = router; | module.exports = router; | ||||||
|  |  | ||||||
|  |  | ||||||
| async function conditionIdCheck (measurement, req, res, next) {  // validate condition_id, returns false if invalid  // TODO | async function sampleIdCheck (measurement, req, res, next) {  // validate sample_id, returns false if invalid or user has no access for this sample | ||||||
|   // const sampleData = await ConditionModel.findById(measurement.condition_id).populate('sample_id').lean().exec().catch(err => {next(err); return false;}) as any; |   const sampleData = await SampleModel.findById(measurement.sample_id).lean().exec().catch(err => {next(err); return false;}) as any; | ||||||
|   // if (!sampleData) {  // sample_id not found |   if (!sampleData) {  // sample_id not found | ||||||
|   //   res.status(400).json({status: 'Condition id not available'}); |     res.status(400).json({status: 'Sample id not available'}); | ||||||
|     return false |     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 |   if (sampleData.user_id.toString() !== req.authDetails.id && !req.auth(res, ['maintain', 'admin'], 'basic')) return false;  // sample does not belong to user | ||||||
|   // return true; |   return true; | ||||||
| } | } | ||||||
|  |  | ||||||
| async function templateCheck (measurement, param, res, next) {  // validate measurement_template and values, param for new/change | async function templateCheck (measurement, param, res, next) {  // validate measurement_template and values, returns values, true if values are {} or false if invalid, param for 'new'/'change' | ||||||
|   const templateData = await MeasurementTemplateModel.findById(measurement.measurement_template).lean().exec().catch(err => {next(err); return false;}) as any; |   const templateData = await MeasurementTemplateModel.findById(measurement.measurement_template).lean().exec().catch(err => {next(err); return false;}) as any; | ||||||
|   if (!templateData) {  // template not found |   if (!templateData) {  // template not found | ||||||
|     res.status(400).json({status: 'Measurement template not available'}); |     res.status(400).json({status: 'Measurement template not available'}); | ||||||
|     return false |     return false | ||||||
|   } |   } | ||||||
|  |  | ||||||
|  |   // fill not given values for new measurements | ||||||
|  |   if (param === 'new') { | ||||||
|  |     if (Object.keys(measurement.values).length === 0) { | ||||||
|  |       res.status(400).json({status: 'At least one value is required'}); | ||||||
|  |       return false | ||||||
|  |     } | ||||||
|  |     const fillValues = {};  // initialize not given values with null | ||||||
|  |     templateData.parameters.forEach(parameter => { | ||||||
|  |       fillValues[parameter.name] = null; | ||||||
|  |     }); | ||||||
|  |     measurement.values = _.assign({}, fillValues, measurement.values); | ||||||
|  |   } | ||||||
|  |  | ||||||
|   // validate values |   // validate values | ||||||
|   const {error, value: ignore} = ParametersValidate.input(measurement.values, templateData.parameters, param); |   const {error, value} = ParametersValidate.input(measurement.values, templateData.parameters, 'null'); | ||||||
|   if (error) {res400(error, res); return false;} |   if (error) {res400(error, res); return false;} | ||||||
|   return true; |   return value || true; | ||||||
| } | } | ||||||
| @@ -288,7 +288,7 @@ describe('/user', () => { | |||||||
|         auth: {basic: 'admin'}, |         auth: {basic: 'admin'}, | ||||||
|         httpStatus: 400, |         httpStatus: 400, | ||||||
|         req: {pass: 'password'}, |         req: {pass: 'password'}, | ||||||
|         res: {status: 'Invalid body format', details: '"pass" with value "password" fails to match the required pattern: /^(?=.*[0-9])(?=.*[a-z])(?=.*[A-Z])(?=.*[!"#%&\'()*+,-.\\/:;<=>?@[\\]^_`{|}~])(?=\\S+$).{8,}$/'} |         res: {status: 'Invalid body format', details: '"pass" with value "password" fails to match the required pattern: /^(?=.*[0-9])(?=.*[a-z])(?=.*[A-Z])(?=.*[!"#%&\'()*+,-.\\/:;<=>?@[\\]^_`{|}~])(?=\\S+$)[a-zA-Z0-9!"#%&\'()*+,\\-.\\/:;<=>?@[\\]^_`{|}~]{8,}$/'} | ||||||
|       }); |       }); | ||||||
|     }); |     }); | ||||||
|     it('rejects requests from non-admins for another user', done => { |     it('rejects requests from non-admins for another user', done => { | ||||||
| @@ -546,7 +546,7 @@ describe('/user', () => { | |||||||
|         auth: {basic: 'admin'}, |         auth: {basic: 'admin'}, | ||||||
|         httpStatus: 400, |         httpStatus: 400, | ||||||
|         req: {email: 'john.doe@bosch.com', name: 'johndoe', pass: 'password', level: 'read', location: 'Rng', device_name: 'Alpha II'}, |         req: {email: 'john.doe@bosch.com', name: 'johndoe', pass: 'password', level: 'read', location: 'Rng', device_name: 'Alpha II'}, | ||||||
|         res: {status: 'Invalid body format', details: '"pass" with value "password" fails to match the required pattern: /^(?=.*[0-9])(?=.*[a-z])(?=.*[A-Z])(?=.*[!"#%&\'()*+,-.\\/:;<=>?@[\\]^_`{|}~])(?=\\S+$).{8,}$/'} |         res: {status: 'Invalid body format', details: '"pass" with value "password" fails to match the required pattern: /^(?=.*[0-9])(?=.*[a-z])(?=.*[A-Z])(?=.*[!"#%&\'()*+,-.\\/:;<=>?@[\\]^_`{|}~])(?=\\S+$)[a-zA-Z0-9!"#%&\'()*+,\\-.\\/:;<=>?@[\\]^_`{|}~]{8,}$/'} | ||||||
|       }); |       }); | ||||||
|     }); |     }); | ||||||
|     it('rejects requests from non-admins', done => { |     it('rejects requests from non-admins', done => { | ||||||
|   | |||||||
| @@ -12,13 +12,14 @@ export default class MeasurementValidate { | |||||||
|           Joi.boolean(), |           Joi.boolean(), | ||||||
|           Joi.array() |           Joi.array() | ||||||
|         ) |         ) | ||||||
|  |         .allow(null) | ||||||
|       ) |       ) | ||||||
|   }; |   }; | ||||||
|  |  | ||||||
|   static input (data, param) {  // validate input, set param to 'new' to make all attributes required |   static input (data, param) {  // validate input, set param to 'new' to make all attributes required | ||||||
|     if (param === 'new') { |     if (param === 'new') { | ||||||
|       return Joi.object({ |       return Joi.object({ | ||||||
|         condition_id: IdValidate.get().required(), |         sample_id: IdValidate.get().required(), | ||||||
|         values: this.measurement.values.required(), |         values: this.measurement.values.required(), | ||||||
|         measurement_template: IdValidate.get().required() |         measurement_template: IdValidate.get().required() | ||||||
|       }).validate(data); |       }).validate(data); | ||||||
| @@ -37,7 +38,7 @@ export default class MeasurementValidate { | |||||||
|     data = IdValidate.stringify(data); |     data = IdValidate.stringify(data); | ||||||
|     const {value, error} = Joi.object({ |     const {value, error} = Joi.object({ | ||||||
|       _id: IdValidate.get(), |       _id: IdValidate.get(), | ||||||
|       condition_id: IdValidate.get(), |       sample_id: IdValidate.get(), | ||||||
|       values: this.measurement.values, |       values: this.measurement.values, | ||||||
|       measurement_template: IdValidate.get() |       measurement_template: IdValidate.get() | ||||||
|     }).validate(data, {stripUnknown: true}); |     }).validate(data, {stripUnknown: true}); | ||||||
|   | |||||||
| @@ -1,7 +1,7 @@ | |||||||
| import Joi from '@hapi/joi'; | import Joi from '@hapi/joi'; | ||||||
|  |  | ||||||
| export default class ParametersValidate { | export default class ParametersValidate { | ||||||
|   static input (data, parameters, param) {  // data to validate, parameters from template, param: 'new', 'change' |   static input (data, parameters, param) {  // data to validate, parameters from template, param: 'new', 'change', 'null'(null values are allowed) | ||||||
|     let joiObject = {}; |     let joiObject = {}; | ||||||
|     parameters.forEach(parameter => { |     parameters.forEach(parameter => { | ||||||
|       if (parameter.range.hasOwnProperty('values')) {  // append right validation method according to parameter |       if (parameter.range.hasOwnProperty('values')) {  // append right validation method according to parameter | ||||||
| @@ -39,6 +39,9 @@ export default class ParametersValidate { | |||||||
|       if (param === 'new') { |       if (param === 'new') { | ||||||
|         joiObject[parameter.name] = joiObject[parameter.name].required() |         joiObject[parameter.name] = joiObject[parameter.name].required() | ||||||
|       } |       } | ||||||
|  |       else if (param === 'null') { | ||||||
|  |         joiObject[parameter.name] = joiObject[parameter.name].allow(null) | ||||||
|  |       } | ||||||
|     }); |     }); | ||||||
|     return Joi.object(joiObject).validate(data); |     return Joi.object(joiObject).validate(data); | ||||||
|   } |   } | ||||||
|   | |||||||
| @@ -10,7 +10,6 @@ export default class TemplateValidate { | |||||||
|       .min(1), |       .min(1), | ||||||
|  |  | ||||||
|     parameters: Joi.array() |     parameters: Joi.array() | ||||||
|       .min(1) |  | ||||||
|       .items( |       .items( | ||||||
|         Joi.object({ |         Joi.object({ | ||||||
|           name: Joi.string() |           name: Joi.string() | ||||||
|   | |||||||
| @@ -305,6 +305,27 @@ | |||||||
|         "status": 0, |         "status": 0, | ||||||
|         "measurement_template": {"$oid":"300000000000000000000003"}, |         "measurement_template": {"$oid":"300000000000000000000003"}, | ||||||
|         "__v": 0 |         "__v": 0 | ||||||
|  |       }, | ||||||
|  |       { | ||||||
|  |         "_id": {"$oid":"800000000000000000000004"}, | ||||||
|  |         "sample_id": {"$oid":"400000000000000000000003"}, | ||||||
|  |         "values": { | ||||||
|  |           "val1": 1 | ||||||
|  |         }, | ||||||
|  |         "status": -1, | ||||||
|  |         "measurement_template": {"$oid":"300000000000000000000003"}, | ||||||
|  |         "__v": 0 | ||||||
|  |       }, | ||||||
|  |       { | ||||||
|  |         "_id": {"$oid":"800000000000000000000005"}, | ||||||
|  |         "sample_id": {"$oid":"400000000000000000000002"}, | ||||||
|  |         "values": { | ||||||
|  |           "weight %": 0.5, | ||||||
|  |           "standard deviation":null | ||||||
|  |         }, | ||||||
|  |         "status": 10, | ||||||
|  |         "measurement_template": {"$oid":"300000000000000000000002"}, | ||||||
|  |         "__v": 0 | ||||||
|       } |       } | ||||||
|     ], |     ], | ||||||
|     "condition_templates": [ |     "condition_templates": [ | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user
	 VLE2FE
					VLE2FE