From 2d5229909f5c3ec94d526984d15d92b2c3be921d Mon Sep 17 00:00:00 2001
From: =?UTF-8?q?Adrian=20Or=C5=82=C3=B3w?= <adrian.orlow@fishbrain.com>
Date: Tue, 9 Apr 2024 18:34:47 +0200
Subject: [PATCH] feat: add tests for publications list modal

---
 ...PublicationsModalLayout.component.test.tsx | 140 ++++++++++++++++++
 .../PublicationsModalLayout.component.tsx     |   3 +-
 .../utils/getBasePublications.test.ts         |  49 ++++++
 .../mapBasePublicationToStandarized.test.ts   | 116 +++++++++++++++
 .../utils/mapBasePublicationToStandarized.ts  |   8 +-
 ...StandarizedPublicationsToCSVString.test.ts |  40 +++++
 src/models/mocks/publicationsResponseMock.ts  |   4 +-
 7 files changed, 355 insertions(+), 5 deletions(-)
 create mode 100644 src/components/FunctionalArea/Modal/PublicationsModal/PublicationsModalLayout/PublicationsModalLayout.component.test.tsx
 create mode 100644 src/components/FunctionalArea/Modal/PublicationsModal/PublicationsModalLayout/utils/getBasePublications.test.ts
 create mode 100644 src/components/FunctionalArea/Modal/PublicationsModal/PublicationsModalLayout/utils/mapBasePublicationToStandarized.test.ts
 create mode 100644 src/components/FunctionalArea/Modal/PublicationsModal/PublicationsModalLayout/utils/mapStandarizedPublicationsToCSVString.test.ts

diff --git a/src/components/FunctionalArea/Modal/PublicationsModal/PublicationsModalLayout/PublicationsModalLayout.component.test.tsx b/src/components/FunctionalArea/Modal/PublicationsModal/PublicationsModalLayout/PublicationsModalLayout.component.test.tsx
new file mode 100644
index 00000000..d44c61ba
--- /dev/null
+++ b/src/components/FunctionalArea/Modal/PublicationsModal/PublicationsModalLayout/PublicationsModalLayout.component.test.tsx
@@ -0,0 +1,140 @@
+/* eslint-disable react/no-children-prop */
+import { FIRST_ARRAY_ELEMENT, SECOND_ARRAY_ELEMENT, THIRD_ARRAY_ELEMENT } from '@/constants/common';
+import { DEFAULT_ERROR } from '@/constants/errors';
+import { bioEntityContentFixture } from '@/models/fixtures/bioEntityContentsFixture';
+import { MODELS_MOCK } from '@/models/mocks/modelsMock';
+import { PUBLICATIONS_DEFAULT_SEARCH_FIRST_10_MOCK } from '@/models/mocks/publicationsResponseMock';
+import { apiPath } from '@/redux/apiPath';
+import { downloadFileFromBlob } from '@/redux/export/export.utils';
+import { StoreType } from '@/redux/store';
+import { BioEntityContent, Publication } from '@/types/models';
+import { mockNetworkResponse } from '@/utils/mockNetworkResponse';
+import {
+  InitialStoreState,
+  getReduxWrapperWithStore,
+} from '@/utils/testing/getReduxWrapperWithStore';
+import { render, screen, waitFor } from '@testing-library/react';
+import { HttpStatusCode } from 'axios';
+import { fetchElementData } from '../utils/fetchElementData';
+import { PublicationsModalLayout } from './PublicationsModalLayout.component';
+
+const FIRST_MODEL_ID = MODELS_MOCK[FIRST_ARRAY_ELEMENT].idObject;
+const SECOND_MODEL_ID = MODELS_MOCK[SECOND_ARRAY_ELEMENT].idObject;
+const THIRD_MODEL_ID = MODELS_MOCK[THIRD_ARRAY_ELEMENT].idObject;
+
+const FIRST_ELEMENT_ID = 100;
+const SECOND_ELEMENT_ID = 200;
+const THIRD_ELEMENT_ID = 300;
+const FOURTH_ELEMENT_ID = 400;
+
+const BASE_PUBLICATION: Publication =
+  PUBLICATIONS_DEFAULT_SEARCH_FIRST_10_MOCK.data[FIRST_ARRAY_ELEMENT];
+const BASE_ELEMENT = BASE_PUBLICATION.elements[FIRST_ARRAY_ELEMENT];
+const PUBLICATION: Publication = {
+  ...BASE_PUBLICATION,
+  elements: [
+    {
+      ...BASE_ELEMENT,
+      id: FIRST_ELEMENT_ID,
+      modelId: FIRST_MODEL_ID,
+    },
+    {
+      ...BASE_ELEMENT,
+      id: SECOND_ELEMENT_ID,
+      modelId: SECOND_MODEL_ID,
+    },
+    {
+      ...BASE_ELEMENT,
+      id: THIRD_ELEMENT_ID,
+      modelId: THIRD_MODEL_ID, // model id duplicate
+    },
+    {
+      ...BASE_ELEMENT,
+      id: FOURTH_ELEMENT_ID,
+      modelId: THIRD_MODEL_ID, // model id duplicate
+    },
+  ],
+};
+
+const BIO_ENTITY_CONTENT = (id: number, elementId: string): BioEntityContent => ({
+  ...bioEntityContentFixture,
+  bioEntity: {
+    ...bioEntityContentFixture.bioEntity,
+    id,
+    elementId,
+  },
+});
+
+jest.mock('../../../../../redux/export/export.utils');
+jest.mock('../utils/fetchElementData');
+(fetchElementData as jest.Mock).mockImplementation(
+  (id: string): BioEntityContent =>
+    ({
+      [`${FIRST_ELEMENT_ID}`]: BIO_ENTITY_CONTENT(FIRST_ELEMENT_ID, 'mi100'),
+      [`${SECOND_ELEMENT_ID}`]: BIO_ENTITY_CONTENT(SECOND_ELEMENT_ID, 'ne200'),
+      [`${THIRD_ELEMENT_ID}`]: BIO_ENTITY_CONTENT(THIRD_ELEMENT_ID, 'r300'),
+      [`${FOURTH_ELEMENT_ID}`]: BIO_ENTITY_CONTENT(FOURTH_ELEMENT_ID, 'va400'),
+    })[id] as BioEntityContent,
+);
+
+const mockedAxiosClient = mockNetworkResponse();
+
+const renderComponent = (initialStore?: InitialStoreState): { store: StoreType } => {
+  const { Wrapper, store } = getReduxWrapperWithStore(initialStore);
+
+  return (
+    render(
+      <Wrapper>
+        <PublicationsModalLayout children={null} />
+      </Wrapper>,
+    ),
+    {
+      store,
+    }
+  );
+};
+
+describe('PublicationsModalLayout - component', () => {
+  const length = 1;
+  mockedAxiosClient
+    .onGet(apiPath.getPublications({ params: { length } }))
+    .reply(HttpStatusCode.Ok, {
+      ...PUBLICATIONS_DEFAULT_SEARCH_FIRST_10_MOCK,
+      data: [PUBLICATION],
+    });
+
+  it('should render download csv button', () => {
+    renderComponent();
+
+    expect(screen.getByTestId('download-csv-button')).toBeInTheDocument();
+  });
+
+  it('should run download file with valid content when download csv clicked', async () => {
+    renderComponent({
+      publications: {
+        data: {
+          ...PUBLICATIONS_DEFAULT_SEARCH_FIRST_10_MOCK,
+          data: [PUBLICATION],
+          filteredSize: length,
+        },
+        loading: 'succeeded',
+        error: DEFAULT_ERROR,
+        sortColumn: '',
+        sortOrder: 'asc',
+        searchValue: '',
+        selectedModelId: undefined,
+      },
+      models: { data: MODELS_MOCK, loading: 'idle', error: DEFAULT_ERROR },
+    });
+
+    const downloadButton = screen.getByTestId('download-csv-button');
+    downloadButton.click();
+
+    await waitFor(() => {
+      expect(downloadFileFromBlob).toHaveBeenCalledWith(
+        '"10049997","The glutamate receptor ion channels.","Dingledine R, Borges K, Bowie D, Traynelis SF.","Pharmacological reviews","1999","mi100,ne200,r300,va400","Core PD map,Histamine signaling,PRKN substrates"',
+        'publications.csv',
+      );
+    });
+  });
+});
diff --git a/src/components/FunctionalArea/Modal/PublicationsModal/PublicationsModalLayout/PublicationsModalLayout.component.tsx b/src/components/FunctionalArea/Modal/PublicationsModal/PublicationsModalLayout/PublicationsModalLayout.component.tsx
index a184a538..df12bd9a 100644
--- a/src/components/FunctionalArea/Modal/PublicationsModal/PublicationsModalLayout/PublicationsModalLayout.component.tsx
+++ b/src/components/FunctionalArea/Modal/PublicationsModal/PublicationsModalLayout/PublicationsModalLayout.component.tsx
@@ -12,7 +12,7 @@ import { MODAL_ROLE } from './PublicationsModalLayout.constants';
 import { useDownloadPublicationsAsCSVFile } from './utils/useDownloadPublicationsAsCSVFile';
 
 type ModalLayoutProps = {
-  children: React.ReactNode;
+  children: React.ReactNode | null;
 };
 
 export const PublicationsModalLayout = ({ children }: ModalLayoutProps): JSX.Element => {
@@ -39,6 +39,7 @@ export const PublicationsModalLayout = ({ children }: ModalLayoutProps): JSX.Ele
               <Button
                 onClick={downloadPublicationsAsCSVFile}
                 disabled={isLoading || !numberOfPublications}
+                data-testid="download-csv-button"
               >
                 {isLoading && (
                   <Image
diff --git a/src/components/FunctionalArea/Modal/PublicationsModal/PublicationsModalLayout/utils/getBasePublications.test.ts b/src/components/FunctionalArea/Modal/PublicationsModal/PublicationsModalLayout/utils/getBasePublications.test.ts
new file mode 100644
index 00000000..eb9a4b10
--- /dev/null
+++ b/src/components/FunctionalArea/Modal/PublicationsModal/PublicationsModalLayout/utils/getBasePublications.test.ts
@@ -0,0 +1,49 @@
+import { apiPath } from '@/redux/apiPath';
+import { mockNetworkResponse } from '@/utils/mockNetworkResponse';
+import { HttpStatusCode } from 'axios';
+import { PUBLICATIONS_DEFAULT_SEARCH_FIRST_10_MOCK } from '../../../../../../models/mocks/publicationsResponseMock';
+import { showToast } from '../../../../../../utils/showToast';
+import { getBasePublications } from './getBasePublications';
+
+const mockedAxiosClient = mockNetworkResponse();
+
+jest.mock('./../../../../../../utils/showToast');
+
+describe('getBasePublications - util', () => {
+  const length = 10;
+
+  it('should return valid data if provided', async () => {
+    mockedAxiosClient
+      .onGet(apiPath.getPublications({ params: { length } }))
+      .reply(HttpStatusCode.Ok, PUBLICATIONS_DEFAULT_SEARCH_FIRST_10_MOCK);
+
+    const result = await getBasePublications({ length });
+
+    expect(result).toStrictEqual(PUBLICATIONS_DEFAULT_SEARCH_FIRST_10_MOCK.data);
+  });
+
+  it('should return empty array if data structure is invalid', async () => {
+    mockedAxiosClient
+      .onGet(apiPath.getPublications({ params: { length } }))
+      .reply(HttpStatusCode.Ok, { randomObject: true });
+
+    const result = await getBasePublications({ length });
+
+    expect(result).toStrictEqual([]);
+  });
+
+  it('should return empty array and show toast error if http error', async () => {
+    mockedAxiosClient
+      .onGet(apiPath.getPublications({ params: { length } }))
+      .reply(HttpStatusCode.BadRequest);
+
+    const result = await getBasePublications({ length });
+
+    expect(result).toStrictEqual([]);
+    expect(showToast).toHaveBeenCalledWith({
+      message:
+        "Problem with fetching publications: The server couldn't understand your request. Please check your input and try again.",
+      type: 'error',
+    });
+  });
+});
diff --git a/src/components/FunctionalArea/Modal/PublicationsModal/PublicationsModalLayout/utils/mapBasePublicationToStandarized.test.ts b/src/components/FunctionalArea/Modal/PublicationsModal/PublicationsModalLayout/utils/mapBasePublicationToStandarized.test.ts
new file mode 100644
index 00000000..3a556eff
--- /dev/null
+++ b/src/components/FunctionalArea/Modal/PublicationsModal/PublicationsModalLayout/utils/mapBasePublicationToStandarized.test.ts
@@ -0,0 +1,116 @@
+/* eslint-disable no-magic-numbers */
+import { FIRST_ARRAY_ELEMENT } from '@/constants/common';
+import { bioEntityContentFixture } from '@/models/fixtures/bioEntityContentsFixture';
+import { PUBLICATIONS_DEFAULT_SEARCH_FIRST_10_MOCK } from '@/models/mocks/publicationsResponseMock';
+import { BioEntityContent, Publication } from '@/types/models';
+import { StandarizedPublication } from '@/types/publications';
+import { fetchElementData } from '../../utils/fetchElementData';
+import { mapBasePublicationToStandarized } from './mapBasePublicationToStandarized';
+
+const FIRST_MODEL_ID = 53;
+const SECOND_MODEL_ID = 63;
+const THIRD_MODEL_ID = 99;
+
+const FIRST_ELEMENT_ID = 100;
+const SECOND_ELEMENT_ID = 200;
+const THIRD_ELEMENT_ID = 300;
+const FOURTH_ELEMENT_ID = 400;
+
+const MODEL_NAME_ID_MAP: Record<number, string> = {
+  [FIRST_MODEL_ID]: 'first model',
+  [SECOND_MODEL_ID]: 'second model',
+  [THIRD_MODEL_ID]: 'third model',
+};
+
+const BASE_PUBLICATION: Publication =
+  PUBLICATIONS_DEFAULT_SEARCH_FIRST_10_MOCK.data[FIRST_ARRAY_ELEMENT];
+const BASE_ELEMENT = BASE_PUBLICATION.elements[FIRST_ARRAY_ELEMENT];
+const PUBLICATION: Publication = {
+  ...BASE_PUBLICATION,
+  elements: [
+    {
+      ...BASE_ELEMENT,
+      id: FIRST_ELEMENT_ID,
+      modelId: FIRST_MODEL_ID,
+    },
+    {
+      ...BASE_ELEMENT,
+      id: SECOND_ELEMENT_ID,
+      modelId: SECOND_MODEL_ID,
+    },
+    {
+      ...BASE_ELEMENT,
+      id: THIRD_ELEMENT_ID,
+      modelId: THIRD_MODEL_ID, // model id duplicate
+    },
+    {
+      ...BASE_ELEMENT,
+      id: FOURTH_ELEMENT_ID,
+      modelId: THIRD_MODEL_ID, // model id duplicate
+    },
+  ],
+};
+
+const BIO_ENTITY_CONTENT = (id: number, elementId: string): BioEntityContent => ({
+  ...bioEntityContentFixture,
+  bioEntity: {
+    ...bioEntityContentFixture.bioEntity,
+    id,
+    elementId,
+  },
+});
+
+jest.mock('../../utils/fetchElementData');
+(fetchElementData as jest.Mock).mockImplementation(
+  (id: string): BioEntityContent =>
+    ({
+      [`${FIRST_ELEMENT_ID}`]: BIO_ENTITY_CONTENT(FIRST_ELEMENT_ID, 'mi100'),
+      [`${SECOND_ELEMENT_ID}`]: BIO_ENTITY_CONTENT(SECOND_ELEMENT_ID, 'ne200'),
+      [`${THIRD_ELEMENT_ID}`]: BIO_ENTITY_CONTENT(THIRD_ELEMENT_ID, 'r300'),
+      [`${FOURTH_ELEMENT_ID}`]: BIO_ENTITY_CONTENT(FOURTH_ELEMENT_ID, 'va400'),
+    })[id] as BioEntityContent,
+);
+
+const getFuncResult = async (
+  publication: Publication,
+  modelNameIdMap: Record<number, string> = MODEL_NAME_ID_MAP,
+): Promise<StandarizedPublication> =>
+  mapBasePublicationToStandarized(publication, { modelNameIdMap });
+
+describe('mapBasePublicationToStandarized - util', () => {
+  it('should return valid pubmedId, journal, year and title', async () => {
+    const results = await getFuncResult(PUBLICATION);
+    const { pubmedId, journal, year, title } = PUBLICATION.publication.article;
+
+    expect(results.pubmedId).toBe(pubmedId);
+    expect(results.journal).toBe(journal);
+    expect(results.year).toBe(year.toString());
+    expect(results.title).toBe(title);
+  });
+
+  it('should return joined authors', async () => {
+    const results = await getFuncResult(PUBLICATION);
+    const { authors } = PUBLICATION.publication.article;
+
+    expect(results.authors).toBe(authors.join(','));
+  });
+
+  it('should return joined and unique model names if present', async () => {
+    const results = await getFuncResult(PUBLICATION);
+
+    expect(results.modelNames).toBe(['first model', 'second model', 'third model'].join(','));
+  });
+
+  it('should return empty model names string if model not existing', async () => {
+    const EMPTY_MODEL_ID_MAP = {};
+    const results = await getFuncResult(PUBLICATION, EMPTY_MODEL_ID_MAP);
+
+    expect(results.modelNames).toBe('');
+  });
+
+  it('should return joined elementsIds', async () => {
+    const results = await getFuncResult(PUBLICATION);
+
+    expect(results.elementsIds).toBe(['mi100', 'ne200', 'r300', 'va400'].join(','));
+  });
+});
diff --git a/src/components/FunctionalArea/Modal/PublicationsModal/PublicationsModalLayout/utils/mapBasePublicationToStandarized.ts b/src/components/FunctionalArea/Modal/PublicationsModal/PublicationsModalLayout/utils/mapBasePublicationToStandarized.ts
index ac4e9383..84b7d269 100644
--- a/src/components/FunctionalArea/Modal/PublicationsModal/PublicationsModalLayout/utils/mapBasePublicationToStandarized.ts
+++ b/src/components/FunctionalArea/Modal/PublicationsModal/PublicationsModalLayout/utils/mapBasePublicationToStandarized.ts
@@ -7,6 +7,8 @@ interface Options {
   modelNameIdMap: Record<number, string>;
 }
 
+const JOIN_SEPARATOR = ',';
+
 export const mapBasePublicationToStandarized = async (
   publication: Publication,
   options: Options,
@@ -28,8 +30,8 @@ export const mapBasePublicationToStandarized = async (
     journal,
     title,
     year: year ? `${year}` : '',
-    authors: authors.join(','),
-    modelNames: getUniqueArray(modelNames).join(','),
-    elementsIds: elementsIds.join(','),
+    authors: authors.join(JOIN_SEPARATOR),
+    modelNames: getUniqueArray(modelNames).join(JOIN_SEPARATOR),
+    elementsIds: elementsIds.join(JOIN_SEPARATOR),
   };
 };
diff --git a/src/components/FunctionalArea/Modal/PublicationsModal/PublicationsModalLayout/utils/mapStandarizedPublicationsToCSVString.test.ts b/src/components/FunctionalArea/Modal/PublicationsModal/PublicationsModalLayout/utils/mapStandarizedPublicationsToCSVString.test.ts
new file mode 100644
index 00000000..07588cfd
--- /dev/null
+++ b/src/components/FunctionalArea/Modal/PublicationsModal/PublicationsModalLayout/utils/mapStandarizedPublicationsToCSVString.test.ts
@@ -0,0 +1,40 @@
+import { StandarizedPublication } from '@/types/publications';
+import { mapStandarizedPublicationsToCSVString } from './mapStandarizedPublicationsToCSVString';
+
+const CASES: [StandarizedPublication[], string][] = [
+  [[], ''],
+  [
+    [
+      {
+        pubmedId: '',
+        year: '',
+        journal: '',
+        authors: '',
+        title: '',
+        modelNames: '',
+        elementsIds: '',
+      },
+    ],
+    '"","","","","","",""',
+  ],
+  [
+    [
+      {
+        authors: 'authors',
+        title: 'title',
+        journal: 'journal',
+        pubmedId: 'pubmedId',
+        modelNames: 'modelNames',
+        elementsIds: 'elementsIds',
+        year: 'year',
+      },
+    ],
+    '"pubmedId","title","authors","journal","year","elementsIds","modelNames"',
+  ],
+];
+
+describe('mapStandarizedPublicationsToCSVString - util', () => {
+  it.each(CASES)('should return valid string', (input, result) => {
+    expect(mapStandarizedPublicationsToCSVString(input)).toBe(result);
+  });
+});
diff --git a/src/models/mocks/publicationsResponseMock.ts b/src/models/mocks/publicationsResponseMock.ts
index 09021f4e..c165eb98 100644
--- a/src/models/mocks/publicationsResponseMock.ts
+++ b/src/models/mocks/publicationsResponseMock.ts
@@ -1,4 +1,6 @@
-export const PUBLICATIONS_DEFAULT_SEARCH_FIRST_10_MOCK = {
+import { PublicationsResponse } from '@/types/models';
+
+export const PUBLICATIONS_DEFAULT_SEARCH_FIRST_10_MOCK: PublicationsResponse = {
   data: [
     {
       elements: [
-- 
GitLab