8225641: Calendar.roll(int field) does not work correctly for WEEK_OF_YEAR
Reviewed-by: naoto
This commit is contained in:
parent
3399fbf9fa
commit
a324fa2639
@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (c) 1996, 2022, Oracle and/or its affiliates. All rights reserved.
|
* Copyright (c) 1996, 2023, Oracle and/or its affiliates. All rights reserved.
|
||||||
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
||||||
*
|
*
|
||||||
* This code is free software; you can redistribute it and/or modify it
|
* This code is free software; you can redistribute it and/or modify it
|
||||||
@ -1307,7 +1307,15 @@ public class GregorianCalendar extends Calendar {
|
|||||||
woy = min;
|
woy = min;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
set(field, getRolledValue(woy, amount, min, max));
|
int newWeekOfYear = getRolledValue(woy, amount, min, max);
|
||||||
|
// Final check to ensure that the first week has the
|
||||||
|
// current DAY_OF_WEEK. Only make a check for
|
||||||
|
// rolling up into week 1, as the existing checks
|
||||||
|
// sufficiently handle rolling down into week 1.
|
||||||
|
if (newWeekOfYear == 1 && isInvalidWeek1() && amount > 0) {
|
||||||
|
newWeekOfYear++;
|
||||||
|
}
|
||||||
|
set(field, newWeekOfYear);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -2972,6 +2980,54 @@ public class GregorianCalendar extends Calendar {
|
|||||||
return normalizedYear == cutoverYear;
|
return normalizedYear == cutoverYear;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* {@return {@code true} if the first week of the current year is minimum
|
||||||
|
* and the {@code DAY_OF_WEEK} does not exist in that week}
|
||||||
|
*
|
||||||
|
* This method is used to check the validity of a {@code WEEK_OF_YEAR} and
|
||||||
|
* {@code DAY_OF_WEEK} combo when WEEK_OF_YEAR is rolled to a value of 1.
|
||||||
|
* This prevents other methods from calling complete() with an invalid combo.
|
||||||
|
*/
|
||||||
|
private boolean isInvalidWeek1() {
|
||||||
|
// Calculate the DAY_OF_WEEK for Jan 1 of the current YEAR
|
||||||
|
long jan1Fd = gcal.getFixedDate(internalGet(YEAR), 1, 1, null);
|
||||||
|
int jan1Dow = BaseCalendar.getDayOfWeekFromFixedDate(jan1Fd);
|
||||||
|
// Calculate how many days are in the first week
|
||||||
|
int daysInFirstWeek;
|
||||||
|
if (getFirstDayOfWeek() <= jan1Dow) {
|
||||||
|
// Add wrap around days
|
||||||
|
daysInFirstWeek = 7 - jan1Dow + getFirstDayOfWeek();
|
||||||
|
} else {
|
||||||
|
daysInFirstWeek = getFirstDayOfWeek() - jan1Dow;
|
||||||
|
}
|
||||||
|
// Calculate the end day of the first week
|
||||||
|
int endDow = getFirstDayOfWeek() - 1 == 0
|
||||||
|
? 7 : getFirstDayOfWeek() - 1;
|
||||||
|
// If the week is a valid minimum, check if the DAY_OF_WEEK does not exist
|
||||||
|
return daysInFirstWeek >= getMinimalDaysInFirstWeek() &&
|
||||||
|
!dayInMinWeek(internalGet(DAY_OF_WEEK), jan1Dow, endDow);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Given the first day and last day of a week, this method determines
|
||||||
|
* if the specified day exists in the minimum week.
|
||||||
|
* This method expects all parameters to be passed in as DAY_OF_WEEK values.
|
||||||
|
* For example, dayInMinWeek(4, 6, 3) returns false since Wednesday
|
||||||
|
* is not between the minimum week given by [Friday, Saturday,
|
||||||
|
* Sunday, Monday, Tuesday].
|
||||||
|
*/
|
||||||
|
private boolean dayInMinWeek (int day, int startDay, int endDay) {
|
||||||
|
if (endDay >= startDay) {
|
||||||
|
// dayInMinWeek(6, 3, 5), check that 6 is
|
||||||
|
// between 3 4 5
|
||||||
|
return (day >= startDay && day <= endDay);
|
||||||
|
} else {
|
||||||
|
// dayInMinWeek(4, 6, 3), check that 4 is
|
||||||
|
// between 6 7 1 2 3
|
||||||
|
return (day >= startDay || day <= endDay);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Returns the fixed date of the first day of the year (usually
|
* Returns the fixed date of the first day of the year (usually
|
||||||
* January 1) before the specified date.
|
* January 1) before the specified date.
|
||||||
|
130
test/jdk/java/util/Calendar/RollFromLastToFirstWeek.java
Normal file
130
test/jdk/java/util/Calendar/RollFromLastToFirstWeek.java
Normal file
@ -0,0 +1,130 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (c) 2023, Oracle and/or its affiliates. All rights reserved.
|
||||||
|
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
||||||
|
*
|
||||||
|
* This code is free software; you can redistribute it and/or modify it
|
||||||
|
* under the terms of the GNU General Public License version 2 only, as
|
||||||
|
* published by the Free Software Foundation.
|
||||||
|
*
|
||||||
|
* This code is distributed in the hope that it will be useful, but WITHOUT
|
||||||
|
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
|
||||||
|
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
|
||||||
|
* version 2 for more details (a copy is included in the LICENSE file that
|
||||||
|
* accompanied this code).
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License version
|
||||||
|
* 2 along with this work; if not, write to the Free Software Foundation,
|
||||||
|
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
|
||||||
|
*
|
||||||
|
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
|
||||||
|
* or visit www.oracle.com if you need additional information or have any
|
||||||
|
* questions.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/*
|
||||||
|
* @test
|
||||||
|
* @bug 8225641
|
||||||
|
* @summary Test the behavior of GregorianCalendar.roll(WEEK_OF_YEAR)
|
||||||
|
* when the last week is rolled into the first week of the same year
|
||||||
|
* @run junit RollFromLastToFirstWeek
|
||||||
|
*/
|
||||||
|
|
||||||
|
|
||||||
|
import java.util.*;
|
||||||
|
import java.util.stream.Stream;
|
||||||
|
import static java.util.Calendar.*;
|
||||||
|
import static org.junit.jupiter.api.Assertions.fail;
|
||||||
|
import org.junit.jupiter.params.ParameterizedTest;
|
||||||
|
import org.junit.jupiter.params.provider.MethodSource;
|
||||||
|
import org.junit.jupiter.params.provider.Arguments;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Test to validate the behavior of GregorianCalendar.roll(WEEK_OF_YEAR, +1)
|
||||||
|
* when rolling from the last week of a year into the first week of the same year.
|
||||||
|
* This only test the implementation of the Gregorian Calendar roll.
|
||||||
|
*
|
||||||
|
* Rolling from the last week of a year into the first week of the same year
|
||||||
|
* could cause a WEEK_OF_YEAR with a non-existent DAY_OF_WEEK combination.
|
||||||
|
* The associated fix ensures that a final check is made, so that the first
|
||||||
|
* week is incremented to prevent this.
|
||||||
|
*/
|
||||||
|
public class RollFromLastToFirstWeek {
|
||||||
|
private static final Builder GREGORIAN_BUILDER = new Builder()
|
||||||
|
.setCalendarType("gregory");
|
||||||
|
|
||||||
|
@ParameterizedTest
|
||||||
|
@MethodSource("rollUpCalProvider")
|
||||||
|
public void rollUpTest(Calendar calendar, String[] validDates){
|
||||||
|
if (calendar instanceof GregorianCalendar) {
|
||||||
|
testRoll(calendar, validDates);
|
||||||
|
} else {
|
||||||
|
fail(String.format("Calendar is not Gregorian: %s", calendar));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private void testRoll(Calendar calendar, String[] validDates) {
|
||||||
|
String originalDate = longDateString(calendar);
|
||||||
|
calendar.roll(Calendar.WEEK_OF_YEAR, 1);
|
||||||
|
String rolledDate = longDateString(calendar);
|
||||||
|
if (!Arrays.asList(validDates).contains(rolledDate)) {
|
||||||
|
fail(String.format("""
|
||||||
|
{$$$ Failed: Rolled: "%s" by 1 week, where the first day of the week
|
||||||
|
is: %s with a minimum week length of: %s and was expecting one of: "%s", but got: "%s"},
|
||||||
|
""", originalDate, calendar.getFirstDayOfWeek(),
|
||||||
|
calendar.getMinimalDaysInFirstWeek(), Arrays.toString(validDates), rolledDate));
|
||||||
|
} else {
|
||||||
|
System.out.printf("""
|
||||||
|
{$$$ Passed: Rolled: "%s" by 1 week where the first day of the week
|
||||||
|
is: %s with a minimum week length of: %s and successfully got: "%s"},
|
||||||
|
""", originalDate, calendar.getFirstDayOfWeek(),
|
||||||
|
calendar.getMinimalDaysInFirstWeek(), rolledDate);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// This implicitly tests the Iso8601 calendar as
|
||||||
|
// MinWeek = 4 and FirstDayOfWeek = Monday is included in the provider
|
||||||
|
private static Stream<Arguments> rollUpCalProvider() {
|
||||||
|
ArrayList<Arguments> calList = new ArrayList<Arguments>();
|
||||||
|
// Week 1, Week 2 are all potential dates to roll into
|
||||||
|
// Depends on first day of week / min days in week
|
||||||
|
String[][] validDates = {
|
||||||
|
{"Wednesday, 2 January 2019", "Wednesday, 9 January 2019"},
|
||||||
|
{"Thursday, 3 January 2019" , "Thursday, 10 January 2019"},
|
||||||
|
{"Friday, 4 January 2019" , "Friday, 11 January 2019"},
|
||||||
|
{"Saturday, 5 January 2019" , "Saturday, 12 January 2019"},
|
||||||
|
{"Sunday, 6 January 2019" , "Sunday, 13 January 2019"},
|
||||||
|
{"Monday, 7 January 2019" , "Monday, 14 January 2019"},
|
||||||
|
{"Tuesday, 1 January 2019" , "Tuesday, 8 January 2019"}
|
||||||
|
};
|
||||||
|
int date = 0;
|
||||||
|
// Test all days at the end of the year that roll into week 1
|
||||||
|
for (int dayOfMonth = 25; dayOfMonth <= 31; dayOfMonth++) {
|
||||||
|
for (int weekLength = 1; weekLength <= 7; weekLength++) {
|
||||||
|
// Sunday .. Monday -> Saturday
|
||||||
|
for (int firstDay = SUNDAY; firstDay <= SATURDAY; firstDay++) {
|
||||||
|
calList.add(Arguments.of(buildCalendar(firstDay, weekLength,
|
||||||
|
dayOfMonth, DECEMBER, 2019), validDates[date]));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
date++;
|
||||||
|
}
|
||||||
|
return calList.stream();
|
||||||
|
}
|
||||||
|
|
||||||
|
private static Calendar buildCalendar(int firstDayOfWeek,
|
||||||
|
int minimumWeekLength, int dayOfMonth,
|
||||||
|
int month, int year) {
|
||||||
|
return GREGORIAN_BUILDER
|
||||||
|
.setWeekDefinition(firstDayOfWeek, minimumWeekLength)
|
||||||
|
.setDate(year, month, dayOfMonth)
|
||||||
|
.build();
|
||||||
|
}
|
||||||
|
|
||||||
|
private static String longDateString(Calendar calendar) {
|
||||||
|
return String.format("%s, %s %s %s",
|
||||||
|
calendar.getDisplayName(Calendar.DAY_OF_WEEK, Calendar.LONG, Locale.ENGLISH),
|
||||||
|
calendar.get(Calendar.DAY_OF_MONTH),
|
||||||
|
calendar.getDisplayName(Calendar.MONTH, Calendar.LONG, Locale.ENGLISH),
|
||||||
|
calendar.get(YEAR));
|
||||||
|
}
|
||||||
|
}
|
Loading…
Reference in New Issue
Block a user