From acd6f31c2b68cae4c2e8954e66ac7a804fa911d5 Mon Sep 17 00:00:00 2001 From: cosa65 Date: Wed, 13 Apr 2022 16:46:53 -0300 Subject: [PATCH 1/3] Add samples patch endpoint --- .../controllers/experimentController.js | 4 +- src/api.v2/controllers/sampleController.js | 17 ++++++ src/api.v2/routes/sample.js | 5 ++ src/specs/api.v2.yaml | 52 +++++++++++++++++-- ...tPatch.v2.yaml => PatchExperiment.v2.yaml} | 2 +- 5 files changed, 74 insertions(+), 6 deletions(-) rename src/specs/models/experiment-bodies/{ExperimentPatch.v2.yaml => PatchExperiment.v2.yaml} (89%) diff --git a/src/api.v2/controllers/experimentController.js b/src/api.v2/controllers/experimentController.js index 8941c7f3e..0840661ed 100644 --- a/src/api.v2/controllers/experimentController.js +++ b/src/api.v2/controllers/experimentController.js @@ -49,13 +49,13 @@ const createExperiment = async (req, res) => { const patchExperiment = async (req, res) => { const { params: { experimentId }, body } = req; - logger.log(`Updating experiment ${experimentId}`); + logger.log(`Patching experiment ${experimentId}`); const snakeCasedKeysToPatch = _.mapKeys(body, (_value, key) => _.snakeCase(key)); await new Experiment().update(experimentId, snakeCasedKeysToPatch); - logger.log(`Finished updating experiment ${experimentId}`); + logger.log(`Finished patching experiment ${experimentId}`); res.json(OK()); }; diff --git a/src/api.v2/controllers/sampleController.js b/src/api.v2/controllers/sampleController.js index 03521f2be..3e1bbfefb 100644 --- a/src/api.v2/controllers/sampleController.js +++ b/src/api.v2/controllers/sampleController.js @@ -1,3 +1,5 @@ +const _ = require('lodash'); + const Sample = require('../model/Sample'); const Experiment = require('../model/Experiment'); const MetadataTrack = require('../model/MetadataTrack'); @@ -36,6 +38,20 @@ const createSample = async (req, res) => { res.json(OK()); }; +const patchSample = async (req, res) => { + const { params: { experimentId, sampleId }, body } = req; + + logger.log(`Patching sample ${sampleId} in experiment ${experimentId}`); + + const snakeCasedKeysToPatch = _.mapKeys(body, (_value, key) => _.snakeCase(key)); + + await new Sample().update(sampleId, snakeCasedKeysToPatch); + + logger.log(`Finished patching sample ${sampleId} in experiment ${experimentId}`); + + res.json(OK()); +}; + const deleteSample = async (req, res) => { const { params: { experimentId, sampleId } } = req; @@ -51,5 +67,6 @@ const deleteSample = async (req, res) => { module.exports = { createSample, + patchSample, deleteSample, }; diff --git a/src/api.v2/routes/sample.js b/src/api.v2/routes/sample.js index 9f09cba17..872b8bf6e 100644 --- a/src/api.v2/routes/sample.js +++ b/src/api.v2/routes/sample.js @@ -1,5 +1,6 @@ const { createSample, + patchSample, deleteSample, } = require('../controllers/sampleController'); @@ -10,6 +11,10 @@ module.exports = { expressAuthorizationMiddleware, (req, res, next) => createSample(req, res).catch(next), ], + 'sample#patchSample': [ + expressAuthorizationMiddleware, + (req, res, next) => patchSample(req, res).catch(next), + ], 'sample#deleteSample': [ expressAuthorizationMiddleware, (req, res, next) => deleteSample(req, res).catch(next), diff --git a/src/specs/api.v2.yaml b/src/specs/api.v2.yaml index 231ecc704..3d1959720 100644 --- a/src/specs/api.v2.yaml +++ b/src/specs/api.v2.yaml @@ -168,7 +168,7 @@ paths: content: application/json: schema: - $ref: '#/components/schemas/ExperimentPatch' + $ref: '#/components/schemas/PatchExperiment' responses: '200': description: Create experiment @@ -242,6 +242,52 @@ paths: application/json: schema: $ref: '#/components/schemas/HTTPError' + patch: + summary: Patch sample + operationId: patchSample + x-eov-operation-id: sample#patchSample + x-eov-operation-handler: routes/sample + requestBody: + content: + application/json: + schema: + type: object + properties: + name: + type: string + additionalProperties: false + responses: + '200': + description: Success + content: + application/json: + schema: + $ref: '#/components/schemas/HTTPSuccess' + '400': + description: Bad Request + content: + application/json: + schema: + $ref: '#/components/schemas/HTTPError' + '401': + description: The request lacks authentication credentials. + content: + application/json: + schema: + $ref: '#/components/schemas/HTTPError' + '403': + description: Forbidden request for this user. + content: + application/json: + schema: + $ref: '#/components/schemas/HTTPError' + '404': + description: Not found error. + content: + application/json: + schema: + $ref: '#/components/schemas/HTTPError' + delete: summary: Delete sample operationId: deleteSample @@ -336,8 +382,8 @@ components: $ref: './models/experiment-bodies/CreateExperiment.v2.yaml' ExperimentInfo: $ref: './models/experiment-bodies/ExperimentInfo.v2.yaml' - ExperimentPatch: - $ref: './models/experiment-bodies/ExperimentPatch.v2.yaml' + PatchExperiment: + $ref: './models/experiment-bodies/PatchExperiment.v2.yaml' GetAllExperiments: $ref: './models/experiment-bodies/GetAllExperiments.v2.yaml' CreateSample: diff --git a/src/specs/models/experiment-bodies/ExperimentPatch.v2.yaml b/src/specs/models/experiment-bodies/PatchExperiment.v2.yaml similarity index 89% rename from src/specs/models/experiment-bodies/ExperimentPatch.v2.yaml rename to src/specs/models/experiment-bodies/PatchExperiment.v2.yaml index 7ebbd8520..a99124eb1 100644 --- a/src/specs/models/experiment-bodies/ExperimentPatch.v2.yaml +++ b/src/specs/models/experiment-bodies/PatchExperiment.v2.yaml @@ -1,4 +1,4 @@ -title: Experiment Patch +title: Patch Experiment description: The properties of an experiment that can be patched type: object properties: From dc38b6cd702cb5f30ddfb143525f7eb43a113a12 Mon Sep 17 00:00:00 2001 From: cosa65 Date: Mon, 18 Apr 2022 14:54:01 -0300 Subject: [PATCH 2/3] Adapt basicModel call for sample patch --- src/api.v2/controllers/sampleController.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/api.v2/controllers/sampleController.js b/src/api.v2/controllers/sampleController.js index 3e1bbfefb..45fab570a 100644 --- a/src/api.v2/controllers/sampleController.js +++ b/src/api.v2/controllers/sampleController.js @@ -45,7 +45,7 @@ const patchSample = async (req, res) => { const snakeCasedKeysToPatch = _.mapKeys(body, (_value, key) => _.snakeCase(key)); - await new Sample().update(sampleId, snakeCasedKeysToPatch); + await new Sample().updateById(sampleId, snakeCasedKeysToPatch); logger.log(`Finished patching sample ${sampleId} in experiment ${experimentId}`); From 483ee59886d5267b89760bd7b25b09e191bd582c Mon Sep 17 00:00:00 2001 From: cosa65 Date: Mon, 18 Apr 2022 15:40:38 -0300 Subject: [PATCH 3/3] Add tests --- src/api.v2/controllers/sampleController.js | 2 + .../controllers/sampleController.test.js | 18 ++++++++ tests/api.v2/routes/sample.test.js | 46 +++++++++++++++++-- 3 files changed, 61 insertions(+), 5 deletions(-) diff --git a/src/api.v2/controllers/sampleController.js b/src/api.v2/controllers/sampleController.js index 45fab570a..e2898a612 100644 --- a/src/api.v2/controllers/sampleController.js +++ b/src/api.v2/controllers/sampleController.js @@ -55,6 +55,8 @@ const patchSample = async (req, res) => { const deleteSample = async (req, res) => { const { params: { experimentId, sampleId } } = req; + logger.log(`Deleting sample ${sampleId} from experiment ${experimentId}`); + await sqlClient.get().transaction(async (trx) => { await new Sample(trx).destroy(sampleId); await new Experiment(trx).deleteSample(experimentId, sampleId); diff --git a/tests/api.v2/controllers/sampleController.test.js b/tests/api.v2/controllers/sampleController.test.js index d8b9db81f..76782ff86 100644 --- a/tests/api.v2/controllers/sampleController.test.js +++ b/tests/api.v2/controllers/sampleController.test.js @@ -123,4 +123,22 @@ describe('sampleController', () => { expect(mockRes.json).not.toHaveBeenCalled(); }); + + it('patchSample works correctly', async () => { + const mockSampleNewName = 'theNewName'; + const mockReq = { + params: { experimentId: mockExperimentId, sampleId: mockSampleId }, + body: { name: mockSampleNewName }, + }; + + sampleInstance.updateById.mockImplementationOnce(() => Promise.resolve()); + + await sampleController.patchSample(mockReq, mockRes); + + expect(sampleInstance.updateById).toHaveBeenCalledWith( + mockSampleId, { name: mockSampleNewName }, + ); + + expect(mockRes.json).toHaveBeenCalledWith(OK()); + }); }); diff --git a/tests/api.v2/routes/sample.test.js b/tests/api.v2/routes/sample.test.js index 11c118f0a..46fc32691 100644 --- a/tests/api.v2/routes/sample.test.js +++ b/tests/api.v2/routes/sample.test.js @@ -10,6 +10,7 @@ const sampleController = require('../../../src/api.v2/controllers/sampleControll jest.mock('../../../src/api.v2/controllers/sampleController', () => ({ createSample: jest.fn(), deleteSample: jest.fn(), + patchSample: jest.fn(), })); jest.mock('../../../src/api.v2/middlewares/authMiddlewares'); @@ -89,13 +90,8 @@ describe('tests for experiment route', () => { return Promise.resolve(); }); - const invalidExperimentData = { - description: 'experimentDescription', - }; - request(app) .delete(`/v2/experiments/${experimentId}/samples/${sampleId}`) - .send(invalidExperimentData) .expect(200) .end((err) => { if (err) { @@ -106,4 +102,44 @@ describe('tests for experiment route', () => { return done(); }); }); + + it('Patching a sample works', async (done) => { + sampleController.patchSample.mockImplementationOnce((req, res) => { + res.json(OK()); + return Promise.resolve(); + }); + + request(app) + .patch(`/v2/experiments/${experimentId}/samples/${sampleId}`) + .send({ name: 'newSampleName' }) + .expect(200) + .end((err) => { + if (err) { + return done(err); + } + // there is no point testing for the values of the response body + // - if something is wrong, the schema validator will catch it + return done(); + }); + }); + + it('Patching a sample fails if requestBody is invalid', async (done) => { + sampleController.patchSample.mockImplementationOnce((req, res) => { + res.json(OK()); + return Promise.resolve(); + }); + + request(app) + .patch(`/v2/experiments/${experimentId}/samples/${sampleId}`) + .send({ aName: 'newSampleName' }) + .expect(400) + .end((err) => { + if (err) { + return done(err); + } + // there is no point testing for the values of the response body + // - if something is wrong, the schema validator will catch it + return done(); + }); + }); });