Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

BHC-2098: Refactor Functionality Used to Display Points on Map #907

Merged
merged 17 commits into from
Jan 18, 2023
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
14 changes: 7 additions & 7 deletions api/package-lock.json

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

88 changes: 88 additions & 0 deletions api/src/paths/dwc/metadata.test.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,88 @@
import chai, { expect } from 'chai';
import { GeoJsonProperties } from 'geojson';
import { describe } from 'mocha';
import sinon from 'sinon';
import sinonChai from 'sinon-chai';
import * as db from '../../database/db';
import { HTTPError } from '../../errors/http-error';
import { OccurrenceService } from '../../services/occurrence-service';
import { getMockDBConnection } from '../../__mocks__/db';
import * as metadata from './metadata';

chai.use(sinonChai);

describe('getSpatialMetadataBySubmissionSpatialComponentIds', () => {
const dbConnectionObj = getMockDBConnection();

const sampleReq = {
keycloak_token: {},
body: {
occurrence_submission_id: null
}
} as any;

let actualResult: any = null;

const sampleRes = {
status: () => {
return {
json: (result: any) => {
actualResult = result;
}
};
}
};

afterEach(() => {
sinon.restore();
});

it('should throw an error when failed to get metadata', async () => {
sinon.stub(db, 'getDBConnection').returns({
...dbConnectionObj,
systemUserId: () => {
return 20;
}
});
const expectedError = new Error('cannot process request');
sinon
.stub(OccurrenceService.prototype, 'findSpatialMetadataBySubmissionSpatialComponentIds')
.rejects(expectedError);

try {
const result = metadata.getSpatialMetadataBySubmissionSpatialComponentIds();

await result(
{ ...sampleReq, query: { submissionSpatialComponentIds: [1] } },
(null as unknown) as any,
(null as unknown) as any
);
expect.fail();
} catch (actualError) {
expect((actualError as HTTPError).message).to.equal(expectedError.message);
}
});

it('should return the occurrences view data on success', async () => {
sinon.stub(db, 'getDBConnection').returns({
...dbConnectionObj,
systemUserId: () => {
return 20;
}
});

sinon
.stub(OccurrenceService.prototype, 'findSpatialMetadataBySubmissionSpatialComponentIds')
.resolves([({ id: 1 } as unknown) as GeoJsonProperties]);

const result = metadata.getSpatialMetadataBySubmissionSpatialComponentIds();

await result(
{ ...sampleReq, query: { submissionSpatialComponentIds: [1] } },
sampleRes as any,
(null as unknown) as any
);

expect(actualResult).to.be.eql([{ id: 1 }]);
});
});
100 changes: 100 additions & 0 deletions api/src/paths/dwc/metadata.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,100 @@
import { RequestHandler } from 'express';
import { Operation } from 'express-openapi';
import { getDBConnection } from '../../database/db';
import { OccurrenceService } from '../../services/occurrence-service';
import { getLogger } from '../../utils/logger';

const defaultLog = getLogger('paths/dwc/metadata');

export const GET: Operation = [getSpatialMetadataBySubmissionSpatialComponentIds()];

GET.apiDoc = {
description: 'Retrieves spatial component metadata based on submission spatial component id',
tags: ['spatial'],
security: [
{
Bearer: []
}
],
parameters: [
{
description: 'spatial component submission ids',
in: 'query',
name: 'submissionSpatialComponentIds',
schema: {
type: 'array',
items: {
type: 'number',
minimum: 1
}
},
required: true
}
],
responses: {
200: {
description: 'Spatial metadata response object.',
content: {
'application/json': {
schema: {
type: 'array',
items: {
type: 'object'
}
}
}
}
},
400: {
$ref: '#/components/responses/400'
},
401: {
$ref: '#/components/responses/401'
},
403: {
$ref: '#/components/responses/401'
},
409: {
$ref: '#/components/responses/409'
},
500: {
$ref: '#/components/responses/500'
},
default: {
$ref: '#/components/responses/default'
}
}
};

/**
* Retrieves dataset metadata from Elastic Search.
*
* @returns {RequestHandler}
*/
export function getSpatialMetadataBySubmissionSpatialComponentIds(): RequestHandler {
return async (req, res) => {
const submissionSpatialComponentIds = ((req.query.submissionSpatialComponentIds || []) as string[]).map(Number);

const connection = getDBConnection(req['keycloak_token']);

try {
await connection.open();

const occurrenceService = new OccurrenceService(connection);

const response = await occurrenceService.findSpatialMetadataBySubmissionSpatialComponentIds(
submissionSpatialComponentIds
);

await connection.commit();

res.status(200).json(response);
} catch (error) {
defaultLog.error({ label: 'getSpatialMetadataBySubmissionSpatialComponentIds', message: 'error', error });
await connection.rollback();
throw error;
} finally {
connection.release();
}
};
}
Original file line number Diff line number Diff line change
Expand Up @@ -2,18 +2,14 @@ import chai, { expect } from 'chai';
import { describe } from 'mocha';
import sinon from 'sinon';
import sinonChai from 'sinon-chai';
import SQL from 'sql-template-strings';
import * as db from '../../../../../../../../database/db';
import { HTTPError } from '../../../../../../../../errors/http-error';
import survey_queries from '../../../../../../../../queries/survey';
import { OccurrenceService } from '../../../../../../../../services/occurrence-service';
import { getMockDBConnection } from '../../../../../../../../__mocks__/db';
import * as delete_submission from './delete';

chai.use(sinonChai);

describe('deleteOccurrenceSubmission', () => {
const dbConnectionObj = getMockDBConnection();

const sampleReq = {
keycloak_token: {},
params: {
Expand All @@ -39,119 +35,31 @@ describe('deleteOccurrenceSubmission', () => {
sinon.restore();
});

it('should throw a 400 error when no projectId is provided', async () => {
it('should return false if no rows were deleted', async () => {
const dbConnectionObj = getMockDBConnection();
sinon.stub(db, 'getDBConnection').returns(dbConnectionObj);

try {
const result = delete_submission.deleteOccurrenceSubmission();
await result(
{ ...sampleReq, params: { ...sampleReq.params, projectId: null } },
(null as unknown) as any,
(null as unknown) as any
);
expect.fail();
} catch (actualError) {
expect((actualError as HTTPError).status).to.equal(400);
expect((actualError as HTTPError).message).to.equal('Missing required path param `projectId`');
}
});

it('should throw a 400 error when no surveyId is provided', async () => {
sinon.stub(db, 'getDBConnection').returns(dbConnectionObj);

try {
const result = delete_submission.deleteOccurrenceSubmission();
await result(
{ ...sampleReq, params: { ...sampleReq.params, surveyId: null } },
(null as unknown) as any,
(null as unknown) as any
);
expect.fail();
} catch (actualError) {
expect((actualError as HTTPError).status).to.equal(400);
expect((actualError as HTTPError).message).to.equal('Missing required path param `surveyId`');
}
});

it('should throw a 400 error when no submissionId is provided', async () => {
sinon.stub(db, 'getDBConnection').returns(dbConnectionObj);

try {
const result = delete_submission.deleteOccurrenceSubmission();
await result(
{ ...sampleReq, params: { ...sampleReq.params, submissionId: null } },
(null as unknown) as any,
(null as unknown) as any
);
expect.fail();
} catch (actualError) {
expect((actualError as HTTPError).status).to.equal(400);
expect((actualError as HTTPError).message).to.equal('Missing required path param `submissionId`');
}
});

it('should throw a 400 error when no sql statement returned for deleteOccurrenceSubmissionSQL', async () => {
sinon.stub(db, 'getDBConnection').returns({
...dbConnectionObj,
systemUserId: () => {
return 20;
}
});

sinon.stub(survey_queries, 'deleteOccurrenceSubmissionSQL').returns(null);

try {
const result = delete_submission.deleteOccurrenceSubmission();

await result(sampleReq, (null as unknown) as any, (null as unknown) as any);
expect.fail();
} catch (actualError) {
expect((actualError as HTTPError).status).to.equal(400);
expect((actualError as HTTPError).message).to.equal('Failed to build SQL delete statement');
}
});

it('should return null when no rowCount', async () => {
const mockQuery = sinon.stub();

mockQuery.resolves({ rowCount: null });

sinon.stub(db, 'getDBConnection').returns({
...dbConnectionObj,
systemUserId: () => {
return 20;
},
query: mockQuery
});

sinon.stub(survey_queries, 'deleteOccurrenceSubmissionSQL').returns(SQL`something`);
sinon.stub(OccurrenceService.prototype, 'deleteOccurrenceSubmission').resolves([]);

const result = delete_submission.deleteOccurrenceSubmission();

await result(sampleReq, sampleRes as any, (null as unknown) as any);

expect(actualResult).to.equal(null);
expect(actualResult).to.equal(false);
});

it('should return rowCount on success', async () => {
const mockQuery = sinon.stub();

mockQuery.resolves({ rowCount: 1 });

sinon.stub(db, 'getDBConnection').returns({
...dbConnectionObj,
systemUserId: () => {
return 20;
},
query: mockQuery
});
it('should return true if occurrence submission was deleted', async () => {
const dbConnectionObj = getMockDBConnection();
sinon.stub(db, 'getDBConnection').returns(dbConnectionObj);

sinon.stub(survey_queries, 'deleteOccurrenceSubmissionSQL').returns(SQL`something`);
sinon
.stub(OccurrenceService.prototype, 'deleteOccurrenceSubmission')
.resolves([{ submission_spatial_component_id: 1 }]);

const result = delete_submission.deleteOccurrenceSubmission();

await result(sampleReq, sampleRes as any, (null as unknown) as any);

expect(actualResult).to.equal(1);
expect(actualResult).to.equal(true);
});
});
Loading