diff --git a/src/utils/helper.js b/src/utils/helper.js index 853274c24..71c80067a 100644 --- a/src/utils/helper.js +++ b/src/utils/helper.js @@ -188,22 +188,53 @@ const uploadFile = ({ }) } -const downloadWindowTitle = i18n.t('Loading exported data') -const downloadWindowHtml = ` -
-

${downloadWindowTitle}

-
-` +const propsToInclude = [ + 'program', + 'ouMode', + 'orgUnits', + 'orgUnit', + 'trackedEntityType', +] +const getParamsForFileName = (url) => { + const params = [...new URL(url).searchParams.entries()] + return params + .filter(([prop]) => propsToInclude.includes(prop)) + .map(([prop, val]) => `${prop}_${val}`) + .join('__') +} // call stub function if available -const locationAssign = (url) => { +const locationAssign = (relativeUrl) => { if (window.locationAssign) { - window.locationAssign(url) + window.locationAssign(relativeUrl) } else { - const downloadWindow = window.open(url, '_blank') + try { + const url = relativeUrl.startsWith('..') + ? new URL(relativeUrl, document.baseURI).href + : relativeUrl + + const requestParams = getParamsForFileName(url) + + const urlFilePart = new URL(url).pathname.split('/').pop() + const [, file, extension] = urlFilePart.match(/(^[^.]+)(\..+$)/) - downloadWindow.document.title = downloadWindowTitle - downloadWindow.document.body.innerHTML = downloadWindowHtml // does not work in Chrome + const timeStamp = new Intl.DateTimeFormat('en-CA').format( + new Date() + ) // en-CA is yyyy-MM-dd + + const downloadedFileName = `${file}_${timeStamp}_${requestParams}${extension}` + + const link = document.createElement('a') + link.href = url + link.download = downloadedFileName + link.target = '_blank' + link.click() + + return link + } catch (err) { + console.error(err) + window.open(relativeUrl, '_blank') + } } } diff --git a/src/utils/helper.test.js b/src/utils/helper.test.js new file mode 100644 index 000000000..440f2d9fe --- /dev/null +++ b/src/utils/helper.test.js @@ -0,0 +1,85 @@ +import { locationAssign } from './helper.js' + +describe('locationAssign', () => { + beforeEach(() => { + jest.useFakeTimers('modern') + jest.setSystemTime(new Date(2024, 2, 12)) + }) + + afterEach(() => { + jest.useRealTimers() + }) + + it('should create a file name based on the params', () => { + const url = + 'https://debug.dhis2.org/dev/api/tracker/trackedEntities.json?ouMode=CAPTURE&format=json&includeDeleted=false&dataElementIdScheme=UID&eventIdScheme=UID&orgUnitIdScheme=UID&idScheme=UID&attachment=trackedEntities.json&paging=false&totalPages=false&program=lxAQ7Zs9VYR' + const link = locationAssign(url) + expect(link.download).toEqual( + 'trackedEntities_2024-03-12_ouMode_CAPTURE__program_lxAQ7Zs9VYR.json' + ) + }) + it('should create url with orgUnits', () => { + const url = + 'https://debug.dhis2.org/dev/api/tracker/trackedEntities.json?ouMode=SELECTED&includeDeleted=false&dataElementIdScheme=UID&eventIdScheme=UID&orgUnitIdScheme=UID&idScheme=UID&attachment=trackedEntities.json&paging=false&totalPages=false&orgUnits=O6uvpzGd5pu,fdc6uOvgoji&program=kla3mAPgvCH' + const link = locationAssign(url) + expect(link.download).toEqual( + 'trackedEntities_2024-03-12_ouMode_SELECTED__orgUnits_O6uvpzGd5pu,fdc6uOvgoji__program_kla3mAPgvCH.json' + ) + }) + it('should create url with tracked entities', () => { + const url = + 'https://debug.dhis2.org/dev/api/tracker/trackedEntities.json?ouMode=SELECTED&includeDeleted=false&dataElementIdScheme=UID&eventIdScheme=UID&orgUnitIdScheme=UID&idScheme=UID&attachment=trackedEntities.json&paging=false&totalPages=false&orgUnits=ImspTQPwCqd&trackedEntityType=bVkFYAvoUCP' + const link = locationAssign(url) + expect(link.download).toEqual( + 'trackedEntities_2024-03-12_ouMode_SELECTED__orgUnits_ImspTQPwCqd__trackedEntityType_bVkFYAvoUCP.json' + ) + }) + it('should create url with CSV', () => { + const url = + 'https://debug.dhis2.org/dev/api/tracker/trackedEntities.csv?ouMode=SELECTED&includeDeleted=false&dataElementIdScheme=UID&eventIdScheme=UID&orgUnitIdScheme=UID&idScheme=UID&attachment=trackedEntities.csv&paging=false&totalPages=false&orgUnits=ImspTQPwCqd&program=lxAQ7Zs9VYR' + const link = locationAssign(url) + expect(link.download).toEqual( + 'trackedEntities_2024-03-12_ouMode_SELECTED__orgUnits_ImspTQPwCqd__program_lxAQ7Zs9VYR.csv' + ) + }) + + it('should create url with events zip', () => { + const url = + 'https://debug.dhis2.org/dev/api/tracker/events.json.zip?links=false&paging=false&totalPages=false&orgUnit=fwH9ipvXde9&program=VBqh0ynB2wv&includeDeleted=false&dataElementIdScheme=UID&orgUnitIdScheme=UID&idScheme=UID&attachment=events.json.zip&occurredAfter=2023-12-12&occurredBefore=2024-03-12&ouMode=CHILDREN&format=json' + const link = locationAssign(url) + expect(link.download).toEqual( + 'events_2024-03-12_orgUnit_fwH9ipvXde9__program_VBqh0ynB2wv__ouMode_CHILDREN.json.zip' + ) + }) + + it('should create url with events gzip', () => { + const url = + 'https://debug.dhis2.org/dev/api/tracker/events.json.gz?links=false&paging=false&totalPages=false&orgUnit=ImspTQPwCqd&program=lxAQ7Zs9VYR&includeDeleted=false&dataElementIdScheme=UID&orgUnitIdScheme=UID&idScheme=UID&attachment=events.json.gz&occurredAfter=2023-12-12&occurredBefore=2024-03-12&ouMode=SELECTED&format=json' + const link = locationAssign(url) + expect(link.download).toEqual( + 'events_2024-03-12_orgUnit_ImspTQPwCqd__program_lxAQ7Zs9VYR__ouMode_SELECTED.json.gz' + ) + }) + it('should work with relative URLs when bundled in DHIS2', () => { + Object.defineProperty(global.document, 'baseURI', { + value: 'http://localhost:8080/dhis-web-import-export/index.html#/export/tei', + }) + const url = + '../api/tracker/trackedEntities.json?ouMode=SELECTED&includeDeleted=false&dataElementIdScheme=UID&eventIdScheme=UID&orgUnitIdScheme=UID&idScheme=UID&attachment=trackedEntities.json&paging=false&totalPages=false&orgUnits=ImspTQPwCqd&program=lxAQ7Zs9VYR' + const link = locationAssign(url) + expect(link.download).toEqual( + 'trackedEntities_2024-03-12_ouMode_SELECTED__orgUnits_ImspTQPwCqd__program_lxAQ7Zs9VYR.json' + ) + }) + it('should work with relative URLs when bundled in DHIS2 for zip', () => { + Object.defineProperty(global.document, 'baseURI', { + value: 'http://localhost:8080/dhis-web-import-export/index.html#/export/tei', + }) + const url = + '../api/tracker/events.json.zip?links=false&paging=false&totalPages=false&orgUnit=ImspTQPwCqd&program=lxAQ7Zs9VYR&includeDeleted=false&dataElementIdScheme=UID&orgUnitIdScheme=UID&idScheme=UID&attachment=events.json.zip&occurredAfter=2023-12-12&occurredBefore=2024-03-12&ouMode=SELECTED&format=json' + const link = locationAssign(url) + expect(link.download).toEqual( + 'trackedEntities_2024-03-12_ouMode_SELECTED__orgUnits_ImspTQPwCqd__program_lxAQ7Zs9VYR.json' + ) + }) +})