From f44a66190d6323d5dda2471e446745fc9527996c Mon Sep 17 00:00:00 2001 From: Hendrik de Graaf Date: Thu, 8 Jul 2021 21:54:27 +0200 Subject: [PATCH 1/6] fix: add period type to datasets --- src/app-context/app-provider.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/app-context/app-provider.js b/src/app-context/app-provider.js index fa2a6abd..3a8b12c1 100644 --- a/src/app-context/app-provider.js +++ b/src/app-context/app-provider.js @@ -23,7 +23,7 @@ const query = { 'displayName', 'dataApprovalLevels', 'periodType', - 'dataSets[id,displayName]', + 'dataSets[id,displayName,periodType]', ], }, }, From a45ce5d0cbee08b3f9edf1cd13d96f1b16bb6c26 Mon Sep 17 00:00:00 2001 From: Hendrik de Graaf Date: Thu, 8 Jul 2021 21:54:54 +0200 Subject: [PATCH 2/6] fix: add helper for period conversion --- src/top-bar/period-select/fixed-periods.js | 36 ++++++++ .../period-select/fixed-periods.test.js | 86 +++++++++++++++++++ 2 files changed, 122 insertions(+) diff --git a/src/top-bar/period-select/fixed-periods.js b/src/top-bar/period-select/fixed-periods.js index 95137d26..8fc1afc6 100644 --- a/src/top-bar/period-select/fixed-periods.js +++ b/src/top-bar/period-select/fixed-periods.js @@ -738,3 +738,39 @@ export const parsePeriodId = (id, allowedTypes) => { periodType, } } + +export const getFixedPeriodsForTypeAndDateRange = ( + type, + startDate, + endDate +) => { + // Allow dates and date-strings + startDate = startDate instanceof Date ? startDate : new Date(startDate) + endDate = endDate instanceof Date ? endDate : new Date(endDate) + + let year = endDate.getFullYear() + let startDateReached = false + const convertedPeriods = [] + + while (!startDateReached) { + getFixedPeriodsByTypeAndYear(type, year) + .reverse() + .forEach(period => { + const isAfterPeriodStart = + new Date(period.startDate) >= startDate + const isBeforePeriodEnd = new Date(period.endDate) <= endDate + + if (!isAfterPeriodStart) { + startDateReached = true + } + + if (isAfterPeriodStart && isBeforePeriodEnd) { + convertedPeriods.push(period) + } + }) + + --year + } + + return convertedPeriods.reverse() +} diff --git a/src/top-bar/period-select/fixed-periods.test.js b/src/top-bar/period-select/fixed-periods.test.js index d9536fd4..2231bd56 100644 --- a/src/top-bar/period-select/fixed-periods.test.js +++ b/src/top-bar/period-select/fixed-periods.test.js @@ -1,4 +1,5 @@ import { + getFixedPeriodsForTypeAndDateRange, getFixedPeriodType, getFixedPeriodTypes, parsePeriodId, @@ -781,4 +782,89 @@ describe('fixedPeriods utils', () => { expect(period).toMatchSnapshot() }) }) + + describe('getFixedPeriodsForTypeAndDateRange', () => { + it('should return an empty array if the period exceeds the date range', () => { + expect( + getFixedPeriodsForTypeAndDateRange( + 'Yearly', + '2021-04-08', + '2021-06-06' + ) + ).toEqual([]) + }) + it('should return the expected result even when the date range spans multiple years', () => { + expect( + getFixedPeriodsForTypeAndDateRange( + 'Monthly', + '2020-08-08', + '2021-06-06' + ) + ).toEqual([ + { + displayName: 'September 2020', + endDate: '2020-09-30', + id: '202009', + iso: '202009', + startDate: '2020-09-01', + }, + { + displayName: 'October 2020', + endDate: '2020-10-31', + id: '202010', + iso: '202010', + startDate: '2020-10-01', + }, + { + displayName: 'November 2020', + endDate: '2020-11-30', + id: '202011', + iso: '202011', + startDate: '2020-11-01', + }, + { + displayName: 'December 2020', + endDate: '2020-12-31', + id: '202012', + iso: '202012', + startDate: '2020-12-01', + }, + { + displayName: 'January 2021', + endDate: '2021-01-31', + id: '202101', + iso: '202101', + startDate: '2021-01-01', + }, + { + displayName: 'February 2021', + endDate: '2021-02-28', + id: '202102', + iso: '202102', + startDate: '2021-02-01', + }, + { + displayName: 'March 2021', + endDate: '2021-03-31', + id: '202103', + iso: '202103', + startDate: '2021-03-01', + }, + { + displayName: 'April 2021', + endDate: '2021-04-30', + id: '202104', + iso: '202104', + startDate: '2021-04-01', + }, + { + displayName: 'May 2021', + endDate: '2021-05-31', + id: '202105', + iso: '202105', + startDate: '2021-05-01', + }, + ]) + }) + }) }) From 7fb6cac05fca0980682e1ec285c1d9ef51cd62ff Mon Sep 17 00:00:00 2001 From: Hendrik de Graaf Date: Thu, 8 Jul 2021 22:46:46 +0200 Subject: [PATCH 3/6] fix: improve period conversion logic --- src/top-bar/period-select/fixed-periods.js | 12 ++--- .../period-select/fixed-periods.test.js | 45 +++++++++++++++++++ 2 files changed, 52 insertions(+), 5 deletions(-) diff --git a/src/top-bar/period-select/fixed-periods.js b/src/top-bar/period-select/fixed-periods.js index 8fc1afc6..b8120e50 100644 --- a/src/top-bar/period-select/fixed-periods.js +++ b/src/top-bar/period-select/fixed-periods.js @@ -756,15 +756,17 @@ export const getFixedPeriodsForTypeAndDateRange = ( getFixedPeriodsByTypeAndYear(type, year) .reverse() .forEach(period => { - const isAfterPeriodStart = - new Date(period.startDate) >= startDate - const isBeforePeriodEnd = new Date(period.endDate) <= endDate + const periodEnd = new Date(period.endDate) + // exclude periods that end before start date + const endsAfterPeriodStart = periodEnd >= startDate + // exclude periods that end after end date + const endsBeforePeriodEnd = periodEnd <= endDate - if (!isAfterPeriodStart) { + if (!endsAfterPeriodStart) { startDateReached = true } - if (isAfterPeriodStart && isBeforePeriodEnd) { + if (endsAfterPeriodStart && endsBeforePeriodEnd) { convertedPeriods.push(period) } }) diff --git a/src/top-bar/period-select/fixed-periods.test.js b/src/top-bar/period-select/fixed-periods.test.js index 2231bd56..c4d26fb1 100644 --- a/src/top-bar/period-select/fixed-periods.test.js +++ b/src/top-bar/period-select/fixed-periods.test.js @@ -793,6 +793,44 @@ describe('fixedPeriods utils', () => { ) ).toEqual([]) }) + it('should return the expected result even when the date range spans multiple years', () => { + expect( + getFixedPeriodsForTypeAndDateRange( + 'Weekly', + '2021-04-01', + '2021-04-30' + ) + ).toEqual([ + { + displayName: 'Week 13 - 2021-03-29 - 2021-04-04', + endDate: '2021-04-04', + id: '2021W13', + iso: '2021W13', + startDate: '2021-03-29', + }, + { + displayName: 'Week 14 - 2021-04-05 - 2021-04-11', + endDate: '2021-04-11', + id: '2021W14', + iso: '2021W14', + startDate: '2021-04-05', + }, + { + displayName: 'Week 15 - 2021-04-12 - 2021-04-18', + endDate: '2021-04-18', + id: '2021W15', + iso: '2021W15', + startDate: '2021-04-12', + }, + { + displayName: 'Week 16 - 2021-04-19 - 2021-04-25', + endDate: '2021-04-25', + id: '2021W16', + iso: '2021W16', + startDate: '2021-04-19', + }, + ]) + }) it('should return the expected result even when the date range spans multiple years', () => { expect( getFixedPeriodsForTypeAndDateRange( @@ -801,6 +839,13 @@ describe('fixedPeriods utils', () => { '2021-06-06' ) ).toEqual([ + { + displayName: 'August 2020', + endDate: '2020-08-31', + id: '202008', + iso: '202008', + startDate: '2020-08-01', + }, { displayName: 'September 2020', endDate: '2020-09-30', From b7a351cd7f618f8ae3b35011a0dcf3a00f689474 Mon Sep 17 00:00:00 2001 From: Jan-Gerke Salomon Date: Tue, 13 Jul 2021 12:15:16 +0200 Subject: [PATCH 4/6] feat(fixed periods): add getLastSubPeriodForTypeAndPeriod & helpers --- src/top-bar/period-select/fixed-periods.js | 101 ++++++-- .../period-select/fixed-periods.test.js | 231 ++++++++++++++++++ 2 files changed, 314 insertions(+), 18 deletions(-) diff --git a/src/top-bar/period-select/fixed-periods.js b/src/top-bar/period-select/fixed-periods.js index b8120e50..a9f3bf70 100644 --- a/src/top-bar/period-select/fixed-periods.js +++ b/src/top-bar/period-select/fixed-periods.js @@ -13,24 +13,24 @@ import i18n from '@dhis2/d2-i18n' */ // Period types -const DAILY = 'Daily' -const WEEKLY = 'Weekly' -const WEEKLY_WEDNESDAY = 'WeeklyWednesday' -const WEEKLY_THURSDAY = 'WeeklyThursday' -const WEEKLY_SATURDAY = 'WeeklySaturday' -const WEEKLY_SUNDAY = 'WeeklySunday' -const BI_WEEKLY = 'BiWeekly' -const MONTHLY = 'Monthly' -const BI_MONTHLY = 'BiMonthly' -const QUARTERLY = 'Quarterly' -const SIX_MONTHLY = 'SixMonthly' -const SIX_MONTHLY_APRIL = 'SixMonthlyApril' -// const SIX_MONTHLY_NOV = 'SixMonthlyNov' -const YEARLY = 'Yearly' -const FINANCIAL_APRIL = 'FinancialApril' -const FINANCIAL_JULY = 'FinancialJuly' -const FINANCIAL_OCT = 'FinancialOct' -const FINANCIAL_NOV = 'FinancialNov' +export const DAILY = 'Daily' +export const WEEKLY = 'Weekly' +export const WEEKLY_WEDNESDAY = 'WeeklyWednesday' +export const WEEKLY_THURSDAY = 'WeeklyThursday' +export const WEEKLY_SATURDAY = 'WeeklySaturday' +export const WEEKLY_SUNDAY = 'WeeklySunday' +export const BI_WEEKLY = 'BiWeekly' +export const MONTHLY = 'Monthly' +export const BI_MONTHLY = 'BiMonthly' +export const QUARTERLY = 'Quarterly' +export const SIX_MONTHLY = 'SixMonthly' +export const SIX_MONTHLY_APRIL = 'SixMonthlyApril' +// export const SIX_MONTHLY_NOV = 'SixMonthlyNov' +export const YEARLY = 'Yearly' +export const FINANCIAL_APRIL = 'FinancialApril' +export const FINANCIAL_JULY = 'FinancialJuly' +export const FINANCIAL_OCT = 'FinancialOct' +export const FINANCIAL_NOV = 'FinancialNov' const getMonthName = key => { const monthNames = [ @@ -776,3 +776,68 @@ export const getFixedPeriodsForTypeAndDateRange = ( return convertedPeriods.reverse() } + +const FIXED_PERIODS_BY_LENGTH = [ + [DAILY], + [WEEKLY, WEEKLY_WEDNESDAY, WEEKLY_THURSDAY, WEEKLY_SATURDAY, WEEKLY_SUNDAY], + [BI_WEEKLY], + [MONTHLY], + [BI_MONTHLY], + [QUARTERLY], + [SIX_MONTHLY, SIX_MONTHLY_APRIL], + [YEARLY, FINANCIAL_APRIL, FINANCIAL_JULY, FINANCIAL_OCT, FINANCIAL_NOV], +] + +export const PERIOD_GREATER = 1 +export const PERIOD_EQUAL = 0 +export const PERIOD_SHORTER = -1 + +/* + * IF left < right THEN return 1 + * IF left > right THEN return -1 + * IF left = right THEN return 0 + * + * Why right > left = 1 and not the other way? + * When partially applying this function, it makes sense this way: + * + * const compareWithWeekly = partial(compareFixedPeriodLength, 'WEEKLY') + * const isGreater = compareWithWeekly('YEARLY') === PERIOD_GREATER + * -> isGreater is true + * + * @param {string} right + * @param {string} left + * @returns {Int} + */ +export const compareFixedPeriodLength = (left, right) => { + const leftIndex = FIXED_PERIODS_BY_LENGTH.findIndex(types => types.includes(left)) + const rightIndex = FIXED_PERIODS_BY_LENGTH.findIndex(types => types.includes(right)) + + if (leftIndex === rightIndex) return PERIOD_EQUAL + return leftIndex > rightIndex ? PERIOD_SHORTER : PERIOD_GREATER +} + +export const getLastSubPeriodForTypeAndPeriod = (type, period) => { + const fullPeriod = parsePeriodId(period.id) + + if (!fullPeriod) { + throw new Error( + `Can't parse period with id "${period.id}"` + ) + } + + const { type: periodType } = parsePeriodId(period.id).periodType + if (compareFixedPeriodLength(type, periodType) !== PERIOD_GREATER) { + throw new Error( + `The sub-period type "${type}" is not shorter that the period's type ("${periodType}")` + ) + } + + // .slice(-1) returns an array with the last item + const [lastSubPeriod] = getFixedPeriodsForTypeAndDateRange( + type, + period.startDate, + period.endDate + ).slice(-1) + + return lastSubPeriod +} diff --git a/src/top-bar/period-select/fixed-periods.test.js b/src/top-bar/period-select/fixed-periods.test.js index c4d26fb1..187d058d 100644 --- a/src/top-bar/period-select/fixed-periods.test.js +++ b/src/top-bar/period-select/fixed-periods.test.js @@ -1,7 +1,30 @@ import { + PERIOD_GREATER, + PERIOD_EQUAL, + PERIOD_SHORTER, + DAILY, + WEEKLY, + WEEKLY_WEDNESDAY, + WEEKLY_THURSDAY, + WEEKLY_SATURDAY, + WEEKLY_SUNDAY, + BI_WEEKLY, + MONTHLY, + BI_MONTHLY, + QUARTERLY, + SIX_MONTHLY, + SIX_MONTHLY_APRIL, + YEARLY, + FINANCIAL_APRIL, + FINANCIAL_JULY, + FINANCIAL_OCT, + FINANCIAL_NOV, + compareFixedPeriodLength, getFixedPeriodsForTypeAndDateRange, + getFixedPeriodsByTypeAndYear, getFixedPeriodType, getFixedPeriodTypes, + getLastSubPeriodForTypeAndPeriod, parsePeriodId, getYearOffsetFromNow, } from './fixed-periods.js' @@ -912,4 +935,212 @@ describe('fixedPeriods utils', () => { ]) }) }) + + describe('compareFixedPeriodLength', () => { + const notEqual = [ + [DAILY, WEEKLY], + [DAILY, WEEKLY_WEDNESDAY], + [DAILY, WEEKLY_THURSDAY], + [DAILY, WEEKLY_SATURDAY], + [DAILY, WEEKLY_SUNDAY], + [DAILY, BI_WEEKLY], + [DAILY, MONTHLY], + [DAILY, BI_MONTHLY], + [DAILY, QUARTERLY], + [DAILY, SIX_MONTHLY], + [DAILY, SIX_MONTHLY_APRIL], + [DAILY, YEARLY], + [DAILY, FINANCIAL_APRIL], + [DAILY, FINANCIAL_JULY], + [DAILY, FINANCIAL_OCT], + [DAILY, FINANCIAL_NOV], + [WEEKLY, BI_WEEKLY], + [WEEKLY, MONTHLY], + [WEEKLY, BI_MONTHLY], + [WEEKLY, QUARTERLY], + [WEEKLY, SIX_MONTHLY], + [WEEKLY, SIX_MONTHLY_APRIL], + [WEEKLY, YEARLY], + [WEEKLY, FINANCIAL_APRIL], + [WEEKLY, FINANCIAL_JULY], + [WEEKLY, FINANCIAL_OCT], + [WEEKLY, FINANCIAL_NOV], + [WEEKLY_WEDNESDAY, BI_WEEKLY], + [WEEKLY_WEDNESDAY, MONTHLY], + [WEEKLY_WEDNESDAY, BI_MONTHLY], + [WEEKLY_WEDNESDAY, QUARTERLY], + [WEEKLY_WEDNESDAY, SIX_MONTHLY], + [WEEKLY_WEDNESDAY, SIX_MONTHLY_APRIL], + [WEEKLY_WEDNESDAY, YEARLY], + [WEEKLY_WEDNESDAY, FINANCIAL_APRIL], + [WEEKLY_WEDNESDAY, FINANCIAL_JULY], + [WEEKLY_WEDNESDAY, FINANCIAL_OCT], + [WEEKLY_WEDNESDAY, FINANCIAL_NOV], + [WEEKLY_THURSDAY, BI_WEEKLY], + [WEEKLY_THURSDAY, MONTHLY], + [WEEKLY_THURSDAY, BI_MONTHLY], + [WEEKLY_THURSDAY, QUARTERLY], + [WEEKLY_THURSDAY, SIX_MONTHLY], + [WEEKLY_THURSDAY, SIX_MONTHLY_APRIL], + [WEEKLY_THURSDAY, YEARLY], + [WEEKLY_THURSDAY, FINANCIAL_APRIL], + [WEEKLY_THURSDAY, FINANCIAL_JULY], + [WEEKLY_THURSDAY, FINANCIAL_OCT], + [WEEKLY_THURSDAY, FINANCIAL_NOV], + [WEEKLY_SATURDAY, BI_WEEKLY], + [WEEKLY_SATURDAY, MONTHLY], + [WEEKLY_SATURDAY, BI_MONTHLY], + [WEEKLY_SATURDAY, QUARTERLY], + [WEEKLY_SATURDAY, SIX_MONTHLY], + [WEEKLY_SATURDAY, SIX_MONTHLY_APRIL], + [WEEKLY_SATURDAY, YEARLY], + [WEEKLY_SATURDAY, FINANCIAL_APRIL], + [WEEKLY_SATURDAY, FINANCIAL_JULY], + [WEEKLY_SATURDAY, FINANCIAL_OCT], + [WEEKLY_SATURDAY, FINANCIAL_NOV], + [WEEKLY_SUNDAY, BI_WEEKLY], + [WEEKLY_SUNDAY, MONTHLY], + [WEEKLY_SUNDAY, BI_MONTHLY], + [WEEKLY_SUNDAY, QUARTERLY], + [WEEKLY_SUNDAY, SIX_MONTHLY], + [WEEKLY_SUNDAY, SIX_MONTHLY_APRIL], + [WEEKLY_SUNDAY, YEARLY], + [WEEKLY_SUNDAY, FINANCIAL_APRIL], + [WEEKLY_SUNDAY, FINANCIAL_JULY], + [WEEKLY_SUNDAY, FINANCIAL_OCT], + [WEEKLY_SUNDAY, FINANCIAL_NOV], + [BI_WEEKLY, MONTHLY], + [BI_WEEKLY, BI_MONTHLY], + [BI_WEEKLY, QUARTERLY], + [BI_WEEKLY, SIX_MONTHLY], + [BI_WEEKLY, SIX_MONTHLY_APRIL], + [BI_WEEKLY, YEARLY], + [BI_WEEKLY, FINANCIAL_APRIL], + [BI_WEEKLY, FINANCIAL_JULY], + [BI_WEEKLY, FINANCIAL_OCT], + [BI_WEEKLY, FINANCIAL_NOV], + [MONTHLY, BI_MONTHLY], + [MONTHLY, QUARTERLY], + [MONTHLY, SIX_MONTHLY], + [MONTHLY, SIX_MONTHLY_APRIL], + [MONTHLY, YEARLY], + [MONTHLY, FINANCIAL_APRIL], + [MONTHLY, FINANCIAL_JULY], + [MONTHLY, FINANCIAL_OCT], + [MONTHLY, FINANCIAL_NOV], + [BI_MONTHLY, QUARTERLY], + [BI_MONTHLY, SIX_MONTHLY], + [BI_MONTHLY, SIX_MONTHLY_APRIL], + [BI_MONTHLY, YEARLY], + [BI_MONTHLY, FINANCIAL_APRIL], + [BI_MONTHLY, FINANCIAL_JULY], + [BI_MONTHLY, FINANCIAL_OCT], + [BI_MONTHLY, FINANCIAL_NOV], + [SIX_MONTHLY, YEARLY], + [SIX_MONTHLY, FINANCIAL_APRIL], + [SIX_MONTHLY, FINANCIAL_JULY], + [SIX_MONTHLY, FINANCIAL_OCT], + [SIX_MONTHLY, FINANCIAL_NOV], + [SIX_MONTHLY_APRIL, YEARLY], + [SIX_MONTHLY_APRIL, FINANCIAL_APRIL], + [SIX_MONTHLY_APRIL, FINANCIAL_JULY], + [SIX_MONTHLY_APRIL, FINANCIAL_OCT], + [SIX_MONTHLY_APRIL, FINANCIAL_NOV], + ] + + it('should be shorter', () => { + notEqual.forEach(([shorter, greater]) => { + const result = compareFixedPeriodLength(greater, shorter) + expect(result).toBe(PERIOD_SHORTER) + }) + }) + + it('should be greater', () => { + notEqual.forEach(([shorter, greater]) => { + const result = compareFixedPeriodLength(shorter, greater) + expect(result).toBe(PERIOD_GREATER) + }) + }) + + const equal = [ + [DAILY, DAILY], + [WEEKLY, WEEKLY], + [WEEKLY, WEEKLY_WEDNESDAY], + [WEEKLY, WEEKLY_THURSDAY], + [WEEKLY, WEEKLY_SATURDAY], + [WEEKLY, WEEKLY_SUNDAY], + [WEEKLY_WEDNESDAY, WEEKLY_WEDNESDAY], + [WEEKLY_WEDNESDAY, WEEKLY_THURSDAY], + [WEEKLY_WEDNESDAY, WEEKLY_SATURDAY], + [WEEKLY_WEDNESDAY, WEEKLY_SUNDAY], + [WEEKLY_THURSDAY, WEEKLY_THURSDAY], + [WEEKLY_THURSDAY, WEEKLY_SATURDAY], + [WEEKLY_THURSDAY, WEEKLY_SUNDAY], + [WEEKLY_SATURDAY, WEEKLY_SATURDAY], + [WEEKLY_SATURDAY, WEEKLY_SUNDAY], + [WEEKLY_SUNDAY, WEEKLY_SUNDAY], + [BI_WEEKLY, BI_WEEKLY], + [MONTHLY, MONTHLY], + [BI_MONTHLY, BI_MONTHLY], + [QUARTERLY, QUARTERLY], + [SIX_MONTHLY, SIX_MONTHLY], + [SIX_MONTHLY, SIX_MONTHLY_APRIL], + [SIX_MONTHLY_APRIL, SIX_MONTHLY_APRIL], + [YEARLY, YEARLY], + [YEARLY, FINANCIAL_APRIL], + [YEARLY, FINANCIAL_JULY], + [YEARLY, FINANCIAL_OCT], + [YEARLY, FINANCIAL_NOV], + [FINANCIAL_APRIL, FINANCIAL_APRIL], + [FINANCIAL_APRIL, FINANCIAL_JULY], + [FINANCIAL_APRIL, FINANCIAL_OCT], + [FINANCIAL_APRIL, FINANCIAL_NOV], + [FINANCIAL_JULY, FINANCIAL_JULY], + [FINANCIAL_JULY, FINANCIAL_OCT], + [FINANCIAL_JULY, FINANCIAL_NOV], + [FINANCIAL_OCT, FINANCIAL_OCT], + [FINANCIAL_OCT, FINANCIAL_NOV], + [FINANCIAL_NOV, FINANCIAL_NOV], + ] + + it('should equal', () => { + equal.forEach(([left, right]) => { + const result = compareFixedPeriodLength(left, right) + expect(result).toBe(PERIOD_EQUAL) + }) + }) + }) + + describe('getLastSubPeriodForTypeAndPeriod', () => { + it("should throw an error if the period's type is equal to the type", () => { + expect(() => getLastSubPeriodForTypeAndPeriod(DAILY, DAILY)).toThrow() + }) + + it("should throw an error if the period's type is shorter than the type", () => { + expect(() => getLastSubPeriodForTypeAndPeriod(WEEKLY, DAILY)).toThrow() + }) + + it('should return the last weekly period of the year 2021', () => { + const yearPeriod = { + endDate: '2021-12-31', + startDate: '2021-01-01', + displayName: '2021', + iso: '2021', + id: '2021' + } + + const lastWeeklyPeriod = getLastSubPeriodForTypeAndPeriod( + WEEKLY, + yearPeriod + ) + + expect(lastWeeklyPeriod).toEqual({ + startDate: '2021-12-20', + iso: '2021W51', + id: '2021W51', + endDate: '2021-12-26', + displayName: 'Week 51 - 2021-12-20 - 2021-12-26' + }) + }) + }) }) From d9421abe996224bd877ca3f87d2694887c6f9742 Mon Sep 17 00:00:00 2001 From: Jan-Gerke Salomon Date: Wed, 14 Jul 2021 11:11:11 +0200 Subject: [PATCH 5/6] feat: add utils to compare greater period end dates with shoter periods --- package.json | 3 +- src/top-bar/period-select/fixed-periods.js | 62 +++++--- .../period-select/fixed-periods.test.js | 135 ++++++++++++++---- yarn.lock | 5 + 4 files changed, 157 insertions(+), 48 deletions(-) diff --git a/package.json b/package.json index b4120ab5..833c0e91 100644 --- a/package.json +++ b/package.json @@ -18,7 +18,8 @@ "@testing-library/react-hooks": "^7.0.1", "enzyme": "^3.11.0", "enzyme-adapter-react-16": "^1.15.6", - "identity-obj-proxy": "^3.0.0" + "identity-obj-proxy": "^3.0.0", + "mockdate": "^3.0.5" }, "dependencies": { "@dhis2/app-runtime": "^2.8.0", diff --git a/src/top-bar/period-select/fixed-periods.js b/src/top-bar/period-select/fixed-periods.js index a9f3bf70..b21457b4 100644 --- a/src/top-bar/period-select/fixed-periods.js +++ b/src/top-bar/period-select/fixed-periods.js @@ -809,35 +809,55 @@ export const PERIOD_SHORTER = -1 * @returns {Int} */ export const compareFixedPeriodLength = (left, right) => { - const leftIndex = FIXED_PERIODS_BY_LENGTH.findIndex(types => types.includes(left)) - const rightIndex = FIXED_PERIODS_BY_LENGTH.findIndex(types => types.includes(right)) + const leftIndex = FIXED_PERIODS_BY_LENGTH.findIndex(types => + types.includes(left) + ) + const rightIndex = FIXED_PERIODS_BY_LENGTH.findIndex(types => + types.includes(right) + ) if (leftIndex === rightIndex) return PERIOD_EQUAL return leftIndex > rightIndex ? PERIOD_SHORTER : PERIOD_GREATER } -export const getLastSubPeriodForTypeAndPeriod = (type, period) => { - const fullPeriod = parsePeriodId(period.id) +export const getCurrentPeriodForType = type => { + const currentDate = new Date() + let year = currentDate.getFullYear() - if (!fullPeriod) { - throw new Error( - `Can't parse period with id "${period.id}"` - ) - } + // cover this and the next years as that + // should cover all existing period types + for (let i = 0; i < 2; ++i) { + const periods = getFixedPeriodsByTypeAndYear(type, year).reverse() + + for (const period of periods) { + const periodStart = new Date(period.startDate) + const periodEnd = new Date(period.endDate) + const endsBeforePeriodEnd = currentDate <= periodEnd + const startsAfterPeriodStart = currentDate >= periodStart - const { type: periodType } = parsePeriodId(period.id).periodType - if (compareFixedPeriodLength(type, periodType) !== PERIOD_GREATER) { - throw new Error( - `The sub-period type "${type}" is not shorter that the period's type ("${periodType}")` - ) + if (endsBeforePeriodEnd && startsAfterPeriodStart) { + return period + } + } + + ++year } - // .slice(-1) returns an array with the last item - const [lastSubPeriod] = getFixedPeriodsForTypeAndDateRange( - type, - period.startDate, - period.endDate - ).slice(-1) + return null +} - return lastSubPeriod +export const isGreaterPeriodTypeEndDateWithinShorterPeriod = ( + greaterPeriodType, + shorterPeriod +) => { + const greaterPeriod = getCurrentPeriodForType(greaterPeriodType) + const greaterPeriodEndDate = new Date(greaterPeriod.endDate) + const shorterPeriodEndDate = new Date(shorterPeriod.endDate) + const shorterPeriodStartDate = new Date(shorterPeriod.startDate) + const greaterEndsAfterShorterStarts = + greaterPeriodEndDate >= shorterPeriodStartDate + const greaterEndsBeforeShorterEnds = + greaterPeriodEndDate <= shorterPeriodEndDate + + return greaterEndsAfterShorterStarts && greaterEndsBeforeShorterEnds } diff --git a/src/top-bar/period-select/fixed-periods.test.js b/src/top-bar/period-select/fixed-periods.test.js index 187d058d..a2fc3d98 100644 --- a/src/top-bar/period-select/fixed-periods.test.js +++ b/src/top-bar/period-select/fixed-periods.test.js @@ -1,3 +1,4 @@ +import MockDate from 'mockdate' import { PERIOD_GREATER, PERIOD_EQUAL, @@ -20,13 +21,13 @@ import { FINANCIAL_OCT, FINANCIAL_NOV, compareFixedPeriodLength, - getFixedPeriodsForTypeAndDateRange, - getFixedPeriodsByTypeAndYear, + getCurrentPeriodForType, getFixedPeriodType, getFixedPeriodTypes, - getLastSubPeriodForTypeAndPeriod, - parsePeriodId, + getFixedPeriodsForTypeAndDateRange, getYearOffsetFromNow, + isGreaterPeriodTypeEndDateWithinShorterPeriod, + parsePeriodId, } from './fixed-periods.js' describe('fixedPeriods utils', () => { @@ -1111,36 +1112,118 @@ describe('fixedPeriods utils', () => { }) }) - describe('getLastSubPeriodForTypeAndPeriod', () => { - it("should throw an error if the period's type is equal to the type", () => { - expect(() => getLastSubPeriodForTypeAndPeriod(DAILY, DAILY)).toThrow() + describe('getCurrentPeriodForType', () => { + afterEach(() => { + MockDate.reset() }) - it("should throw an error if the period's type is shorter than the type", () => { - expect(() => getLastSubPeriodForTypeAndPeriod(WEEKLY, DAILY)).toThrow() - }) + it('should return the current period that ends in the current year', () => { + MockDate.set(new Date('2021-10-01').getTime()) - it('should return the last weekly period of the year 2021', () => { - const yearPeriod = { + const periodType = SIX_MONTHLY + const actual = getCurrentPeriodForType(periodType) + const expected = { + startDate: '2021-07-01', endDate: '2021-12-31', - startDate: '2021-01-01', - displayName: '2021', - iso: '2021', - id: '2021' + displayName: 'July - December 2021', + iso: '2021S2', + id: '2021S2', } - const lastWeeklyPeriod = getLastSubPeriodForTypeAndPeriod( - WEEKLY, - yearPeriod - ) + expect(actual).toEqual(expected) + }) + + it('should return the current period that ends in the next year', () => { + MockDate.set(new Date('2021-10-01').getTime()) + + const periodType = SIX_MONTHLY_APRIL + const actual = getCurrentPeriodForType(periodType) + const expected = { + startDate: '2021-10-01', + endDate: '2022-03-31', + displayName: 'October 2021 - March 2022', + iso: '2021AprilS2', + id: '2021AprilS2', + } + + expect(actual).toEqual(expected) + }) + + describe('yearly - edge case', () => { + it('should return the current period that ends in the current year', () => { + MockDate.set(new Date('2021-10-01').getTime()) + + const periodType = YEARLY + const actual = getCurrentPeriodForType(periodType) + const expected = { + endDate: '2021-12-31', + startDate: '2021-01-01', + displayName: '2021', + iso: '2021', + id: '2021', + } + + expect(actual).toEqual(expected) + }) + + it('should return the current period that ends in the next year', () => { + MockDate.set(new Date('2021-10-01').getTime()) - expect(lastWeeklyPeriod).toEqual({ - startDate: '2021-12-20', - iso: '2021W51', - id: '2021W51', - endDate: '2021-12-26', - displayName: 'Week 51 - 2021-12-20 - 2021-12-26' + const periodType = FINANCIAL_APRIL + const actual = getCurrentPeriodForType(periodType) + const expected = { + endDate: '2022-03-31', + startDate: '2021-04-01', + displayName: 'April 2021 - March 2022', + id: '2021April', + } + + expect(actual).toEqual(expected) }) }) }) + + describe('isGreaterPeriodTypeEndDateWithinShorterPeriod', () => { + it('should return true when the short period spans over the greater periods end date', () => { + MockDate.set(new Date('2021-10-01').getTime()) + + const greaterPeriodType = YEARLY + const shorterPeriod = parsePeriodId('2021Q4') + const expected = true + const actual = isGreaterPeriodTypeEndDateWithinShorterPeriod( + greaterPeriodType, + shorterPeriod + ) + + expect(actual).toBe(expected) + }) + + it('should return false when the short period spans over the greater periods end date', () => { + MockDate.set(new Date('2021-10-01').getTime()) + + const greaterPeriodType = YEARLY + const shorterPeriod = parsePeriodId('2021Q3') + const expected = false + const actual = isGreaterPeriodTypeEndDateWithinShorterPeriod( + greaterPeriodType, + shorterPeriod + ) + + expect(actual).toBe(expected) + }) + + it('should return true when a short period that ends in the following year spans over the greater periods end date', () => { + MockDate.set(new Date('2021-10-01').getTime()) + + const greaterPeriodType = YEARLY + const shorterPeriod = parsePeriodId('2021W52') + const expected = true + const actual = isGreaterPeriodTypeEndDateWithinShorterPeriod( + greaterPeriodType, + shorterPeriod + ) + + expect(actual).toBe(expected) + }) + }) }) diff --git a/yarn.lock b/yarn.lock index 2c72a810..e13f6fc3 100644 --- a/yarn.lock +++ b/yarn.lock @@ -10859,6 +10859,11 @@ mkdirp@^1.0.3, mkdirp@^1.0.4: resolved "https://registry.yarnpkg.com/mkdirp/-/mkdirp-1.0.4.tgz#3eb5ed62622756d79a5f0e2a221dfebad75c2f7e" integrity sha512-vVqVZQyf3WLx2Shd0qJ9xuvqgAyKPLAiqITEtqW0oIUjzo3PePDd6fW9iFz30ef7Ysp/oiWqbhszeGWW2T6Gzw== +mockdate@^3.0.5: + version "3.0.5" + resolved "https://registry.yarnpkg.com/mockdate/-/mockdate-3.0.5.tgz#789be686deb3149e7df2b663d2bc4392bc3284fb" + integrity sha512-iniQP4rj1FhBdBYS/+eQv7j1tadJ9lJtdzgOpvsOHng/GbcDh2Fhdeq+ZRldrPYdXvCyfFUmFeEwEGXZB5I/AQ== + moment@^2.24.0, moment@^2.29.1: version "2.29.1" resolved "https://registry.yarnpkg.com/moment/-/moment-2.29.1.tgz#b2be769fa31940be9eeea6469c075e35006fa3d3" From 422f72d300b6b235cfda06512699613d9ae2291d Mon Sep 17 00:00:00 2001 From: "@dhis2-bot" Date: Wed, 14 Jul 2021 12:16:10 +0000 Subject: [PATCH 6/6] chore(release): cut 1.7.0 [skip ci] # [1.7.0](https://github.com/dhis2/approval-app/compare/v1.6.1...v1.7.0) (2021-07-14) ### Bug Fixes * add helper for period conversion ([a45ce5d](https://github.com/dhis2/approval-app/commit/a45ce5d0cbee08b3f9edf1cd13d96f1b16bb6c26)) * add period type to datasets ([f44a661](https://github.com/dhis2/approval-app/commit/f44a66190d6323d5dda2471e446745fc9527996c)) * improve period conversion logic ([7fb6cac](https://github.com/dhis2/approval-app/commit/7fb6cac05fca0980682e1ec285c1d9ef51cd62ff)) ### Features * add utils to compare greater period end dates with shoter periods ([d9421ab](https://github.com/dhis2/approval-app/commit/d9421abe996224bd877ca3f87d2694887c6f9742)) * **fixed periods:** add getLastSubPeriodForTypeAndPeriod & helpers ([b7a351c](https://github.com/dhis2/approval-app/commit/b7a351cd7f618f8ae3b35011a0dcf3a00f689474)) --- CHANGELOG.md | 15 +++++++++++++++ package.json | 2 +- 2 files changed, 16 insertions(+), 1 deletion(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index 4285e9bf..8201dc2a 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -1,3 +1,18 @@ +# [1.7.0](https://github.com/dhis2/approval-app/compare/v1.6.1...v1.7.0) (2021-07-14) + + +### Bug Fixes + +* add helper for period conversion ([a45ce5d](https://github.com/dhis2/approval-app/commit/a45ce5d0cbee08b3f9edf1cd13d96f1b16bb6c26)) +* add period type to datasets ([f44a661](https://github.com/dhis2/approval-app/commit/f44a66190d6323d5dda2471e446745fc9527996c)) +* improve period conversion logic ([7fb6cac](https://github.com/dhis2/approval-app/commit/7fb6cac05fca0980682e1ec285c1d9ef51cd62ff)) + + +### Features + +* add utils to compare greater period end dates with shoter periods ([d9421ab](https://github.com/dhis2/approval-app/commit/d9421abe996224bd877ca3f87d2694887c6f9742)) +* **fixed periods:** add getLastSubPeriodForTypeAndPeriod & helpers ([b7a351c](https://github.com/dhis2/approval-app/commit/b7a351cd7f618f8ae3b35011a0dcf3a00f689474)) + ## [1.6.1](https://github.com/dhis2/approval-app/compare/v1.6.0...v1.6.1) (2021-07-08) diff --git a/package.json b/package.json index 833c0e91..7eee8cd6 100644 --- a/package.json +++ b/package.json @@ -1,6 +1,6 @@ { "name": "approval-app", - "version": "1.6.1", + "version": "1.7.0", "description": "", "license": "BSD-3-Clause", "private": true,