1 /* 2 * Copyright (c) 2012, 2013, Oracle and/or its affiliates. All rights reserved. 3 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. 4 * 5 * This code is free software; you can redistribute it and/or modify it 6 * under the terms of the GNU General Public License version 2 only, as 7 * published by the Free Software Foundation. Oracle designates this 8 * particular file as subject to the "Classpath" exception as provided 9 * by Oracle in the LICENSE file that accompanied this code. 10 * 11 * This code is distributed in the hope that it will be useful, but WITHOUT 12 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or 13 * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License 14 * version 2 for more details (a copy is included in the LICENSE file that 15 * accompanied this code). 16 * 17 * You should have received a copy of the GNU General Public License version 18 * 2 along with this work; if not, write to the Free Software Foundation, 19 * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. 20 * 21 * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA 22 * or visit www.oracle.com if you need additional information or have any 23 * questions. 24 */ 25 26 /* 27 * Copyright (c) 2012, Stephen Colebourne & Michael Nascimento Santos 28 * 29 * All rights reserved. 30 * 31 * Redistribution and use in source and binary forms, with or without 32 * modification, are permitted provided that the following conditions are met: 33 * 34 * * Redistributions of source code must retain the above copyright notice, 35 * this list of conditions and the following disclaimer. 36 * 37 * * Redistributions in binary form must reproduce the above copyright notice, 38 * this list of conditions and the following disclaimer in the documentation 39 * and/or other materials provided with the distribution. 40 * 41 * * Neither the name of JSR-310 nor the names of its contributors 42 * may be used to endorse or promote products derived from this software 43 * without specific prior written permission. 44 * 45 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS 46 * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT 47 * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR 48 * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR 49 * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, 50 * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, 51 * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR 52 * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF 53 * LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING 54 * NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS 55 * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. 56 */ 57 58 package java.time.chrono; 59 60 import static java.time.temporal.ChronoField.EPOCH_DAY; 61 62 import java.io.File; 63 import java.io.FileInputStream; 64 import java.io.IOException; 65 import java.io.InputStream; 66 import java.io.Serializable; 67 import java.security.AccessController; 68 import java.security.PrivilegedActionException; 69 import java.time.Clock; 70 import java.time.DateTimeException; 71 import java.time.Instant; 72 import java.time.LocalDate; 73 import java.time.ZoneId; 74 import java.time.temporal.ChronoField; 75 import java.time.temporal.TemporalAccessor; 76 import java.time.temporal.ValueRange; 77 import java.util.Arrays; 78 import java.util.HashMap; 79 import java.util.List; 80 import java.util.Map; 81 import java.util.Objects; 82 import java.util.Properties; 83 84 import sun.util.logging.PlatformLogger; 85 86 /** 87 * The Hijrah calendar is a lunar calendar supporting Islamic calendars. 88 * <p> 89 * The HijrahChronology follows the rules of the Hijrah calendar system. The Hijrah 90 * calendar has several variants based on differences in when the new moon is 91 * determined to have occurred and where the observation is made. 92 * In some variants the length of each month is 93 * computed algorithmically from the astronomical data for the moon and earth and 94 * in others the length of the month is determined by an authorized sighting 95 * of the new moon. For the algorithmically based calendars the calendar 96 * can project into the future. 97 * For sighting based calendars only historical data from past 98 * sightings is available. 99 * <p> 100 * The length of each month is 29 or 30 days. 101 * Ordinary years have 354 days; leap years have 355 days. 102 * 103 * <p> 104 * CLDR and LDML identify variants: 105 * <table cellpadding="2" summary="Variants of Hijrah Calendars"> 106 * <thead> 107 * <tr class="tableSubHeadingColor"> 108 * <th class="colFirst" align="left" >Chronology ID</th> 109 * <th class="colFirst" align="left" >Calendar Type</th> 110 * <th class="colFirst" align="left" >Locale extension, see {@link java.util.Locale}</th> 111 * <th class="colLast" align="left" >Description</th> 112 * </tr> 113 * </thead> 114 * <tbody> 115 * <tr class="altColor"> 116 * <td>Hijrah-umalqura</td> 117 * <td>islamic-umalqura</td> 118 * <td>ca-islamic-cv-umalqura</td> 119 * <td>Islamic - Umm Al-Qura calendar of Saudi Arabia</td> 120 * </tr> 121 * </tbody> 122 * </table> 123 * <p>Additional variants may be available through {@link Chronology#getAvailableChronologies()}. 124 * 125 * <p>Example</p> 126 * <p> 127 * Selecting the chronology from the locale uses {@link Chronology#ofLocale} 128 * to find the Chronology based on Locale supported BCP 47 extension mechanism 129 * to request a specific calendar ("ca") and variant ("cv"). For example, 130 * </p> 131 * <pre> 132 * Locale locale = Locale.forLanguageTag("en-US-u-ca-islamic-cv-umalqura"); 133 * Chronology chrono = Chronology.ofLocale(locale); 134 * </pre> 135 * 136 * <h3>Specification for implementors</h3> 137 * This class is immutable and thread-safe. 138 * <h3>Implementation Note for Hijrah Calendar Variant Configuration</h3> 139 * Each Hijrah variant is configured individually. Each variant is defined by a 140 * property resource that defines the {@code ID}, the {@code calendar type}, 141 * the start of the calendar, the alignment with the 142 * ISO calendar, and the length of each month for a range of years. 143 * The variants are identified in the {@code calendars.properties} file. 144 * The new properties are prefixed with {@code "calendars.hijrah."}: 145 * <table cellpadding="2" border="0" summary="Configuration of Hijrah Calendar Variants"> 146 * <thead> 147 * <tr class="tableSubHeadingColor"> 148 * <th class="colFirst" align="left">Property Name</th> 149 * <th class="colFirst" align="left">Property value</th> 150 * <th class="colLast" align="left">Description </th> 151 * </tr> 152 * </thead> 153 * <tbody> 154 * <tr class="altColor"> 155 * <td>calendars.hijrah.{ID}</td> 156 * <td>The property resource defining the {@code {ID}} variant</td> 157 * <td>The property resource is located with the {@code calendars.properties} file</td> 158 * </tr> 159 * <tr class="rowColor"> 160 * <td>calendars.hijrah.{ID}.type</td> 161 * <td>The calendar type</td> 162 * <td>LDML defines the calendar type names</td> 163 * </tr> 164 * </tbody> 165 * </table> 166 * <p> 167 * The Hijrah property resource is a set of properties that describe the calendar. 168 * The syntax is defined by {@code java.util.Properties#load(Reader)}. 169 * <table cellpadding="2" summary="Configuration of Hijrah Calendar"> 170 * <thead> 171 * <tr class="tableSubHeadingColor"> 172 * <th class="colFirst" align="left" > Property Name</th> 173 * <th class="colFirst" align="left" > Property value</th> 174 * <th class="colLast" align="left" > Description </th> 175 * </tr> 176 * </thead> 177 * <tbody> 178 * <tr class="altColor"> 179 * <td>id</td> 180 * <td>Chronology Id, for example, "Hijrah-umalqura"</td> 181 * <td>The Id of the calendar in common usage</td> 182 * </tr> 183 * <tr class="rowColor"> 184 * <td>type</td> 185 * <td>Calendar type, for example, "islamic-umalqura"</td> 186 * <td>LDML defines the calendar types</td> 187 * </tr> 188 * <tr class="altColor"> 189 * <td>version</td> 190 * <td>Version, for example: "1.8.0_1"</td> 191 * <td>The version of the Hijrah variant data</td> 192 * </tr> 193 * <tr class="rowColor"> 194 * <td>iso-start</td> 195 * <td>ISO start date, formatted as {@code yyyy-MM-dd}, for example: "1900-04-30"</td> 196 * <td>The ISO date of the first day of the minimum Hijrah year.</td> 197 * </tr> 198 * <tr class="altColor"> 199 * <td>yyyy - a numeric 4 digit year, for example "1434"</td> 200 * <td>The value is a sequence of 12 month lengths, 201 * for example: "29 30 29 30 29 30 30 30 29 30 29 29"</td> 202 * <td>The lengths of the 12 months of the year separated by whitespace. 203 * A numeric year property must be present for every year without any gaps. 204 * The month lengths must be between 29-32 inclusive. 205 * </td> 206 * </tr> 207 * </tbody> 208 * </table> 209 * 210 * @since 1.8 211 */ 212 public final class HijrahChronology extends Chronology implements Serializable { 213 214 /** 215 * The Hijrah Calendar id. 216 */ 217 private final String typeId; 218 /** 219 * The Hijrah calendarType. 220 */ 221 private transient final String calendarType; 222 /** 223 * Serialization version. 224 */ 225 private static final long serialVersionUID = 3127340209035924785L; 226 /** 227 * Singleton instance of the Islamic Umm Al-Qura calendar of Saudi Arabia. 228 * Other Hijrah chronology variants may be available from 229 * {@link Chronology#getAvailableChronologies}. 230 */ 231 public static final HijrahChronology INSTANCE; 232 /** 233 * Array of epoch days indexed by Hijrah Epoch month. 234 * Computed by {@link #loadCalendarData}. 235 */ 236 private transient int[] hijrahEpochMonthStartDays; 237 /** 238 * The minimum epoch day of this Hijrah calendar. 239 * Computed by {@link #loadCalendarData}. 240 */ 241 private transient int minEpochDay; 242 /** 243 * The maximum epoch day for which calendar data is available. 244 * Computed by {@link #loadCalendarData}. 245 */ 246 private transient int maxEpochDay; 247 /** 248 * The minimum epoch month. 249 * Computed by {@link #loadCalendarData}. 250 */ 251 private transient int hijrahStartEpochMonth; 252 /** 253 * The minimum length of a month. 254 * Computed by {@link #createEpochMonths}. 255 */ 256 private transient int minMonthLength; 257 /** 258 * The maximum length of a month. 259 * Computed by {@link #createEpochMonths}. 260 */ 261 private transient int maxMonthLength; 262 /** 263 * The minimum length of a year in days. 264 * Computed by {@link #createEpochMonths}. 265 */ 266 private transient int minYearLength; 267 /** 268 * The maximum length of a year in days. 269 * Computed by {@link #createEpochMonths}. 270 */ 271 private transient int maxYearLength; 272 /** 273 * A reference to the properties stored in 274 * ${java.home}/lib/calendars.properties 275 */ 276 private transient final static Properties calendarProperties; 277 278 /** 279 * Prefix of property names for Hijrah calendar variants. 280 */ 281 private static final String PROP_PREFIX = "calendar.hijrah."; 282 /** 283 * Suffix of property names containing the calendar type of a variant. 284 */ 285 private static final String PROP_TYPE_SUFFIX = ".type"; 286 287 /** 288 * Name data. 289 */ 290 static { 291 try { 292 calendarProperties = sun.util.calendar.BaseCalendar.getCalendarProperties(); 293 } catch (IOException ioe) { 294 throw new InternalError("Can't initialize lib/calendars.properties", ioe); 295 } 296 297 try { 298 INSTANCE = new HijrahChronology("Hijrah-umalqura"); 299 // Register it by its aliases 300 Chronology.registerChrono(INSTANCE, "Hijrah"); 301 Chronology.registerChrono(INSTANCE, "islamic"); 302 303 } catch (Exception ex) { 304 // Absence of Hijrah calendar is fatal to initializing this class. 305 PlatformLogger logger = PlatformLogger.getLogger("java.time.chrono"); 306 logger.severe("Unable to initialize Hijrah calendar: Hijrah-umalqura", ex); 307 throw new RuntimeException("Unable to initialize Hijrah-umalqura calendar", ex.getCause()); 308 } 309 registerVariants(); 310 } 311 312 /** 313 * For each Hijrah variant listed, create the HijrahChronology and register it. 314 * Exceptions during initialization are logged but otherwise ignored. 315 */ 316 private static void registerVariants() { 317 for (String name : calendarProperties.stringPropertyNames()) { 318 if (name.startsWith(PROP_PREFIX)) { 319 String id = name.substring(PROP_PREFIX.length()); 320 if (id.indexOf('.') >= 0) { 321 continue; // no name or not a simple name of a calendar 322 } 323 if (id.equals(INSTANCE.getId())) { 324 continue; // do not duplicate the default 325 } 326 try { 327 // Create and register the variant 328 HijrahChronology chrono = new HijrahChronology(id); 329 Chronology.registerChrono(chrono); 330 } catch (Exception ex) { 331 // Log error and continue 332 PlatformLogger logger = PlatformLogger.getLogger("java.time.chrono"); 333 logger.severe("Unable to initialize Hijrah calendar: " + id, ex); 334 } 335 } 336 } 337 } 338 339 /** 340 * Create a HijrahChronology for the named variant. 341 * The resource and calendar type are retrieved from properties 342 * in the {@code calendars.properties}. 343 * The property names are {@code "calendar.hijrah." + id} 344 * and {@code "calendar.hijrah." + id + ".type"} 345 * @param id the id of the calendar 346 * @throws Exception if the resource can not be accessed or 347 * the format is invalid 348 */ 349 private HijrahChronology(String id) throws Exception { 350 if (id.isEmpty()) { 351 throw new IllegalArgumentException("calendar id is empty"); 352 } 353 this.typeId = id; 354 this.calendarType = calendarProperties.getProperty(PROP_PREFIX + id + PROP_TYPE_SUFFIX); 355 356 try { 357 String resource = calendarProperties.getProperty(PROP_PREFIX + id); 358 Objects.requireNonNull(resource, "Resource missing for calendar"); 359 loadCalendarData(resource); 360 } catch (Exception ex) { 361 throw new Exception("Unable to initialize HijrahCalendar: " + id, ex); 362 } 363 } 364 365 //----------------------------------------------------------------------- 366 /** 367 * Gets the ID of the chronology. 368 * <p> 369 * The ID uniquely identifies the {@code Chronology}. It can be used to 370 * lookup the {@code Chronology} using {@link #of(String)}. 371 * 372 * @return the chronology ID, non-null 373 * @see #getCalendarType() 374 */ 375 @Override 376 public String getId() { 377 return typeId; 378 } 379 380 /** 381 * Gets the calendar type of the Islamic calendar. 382 * <p> 383 * The calendar type is an identifier defined by the 384 * <em>Unicode Locale Data Markup Language (LDML)</em> specification. 385 * It can be used to lookup the {@code Chronology} using {@link #of(String)}. 386 * 387 * @return the calendar system type; non-null if the calendar has 388 * a standard type, otherwise null 389 * @see #getId() 390 */ 391 @Override 392 public String getCalendarType() { 393 return calendarType; 394 } 395 396 //----------------------------------------------------------------------- 397 /** 398 * Obtains a local date in Hijrah calendar system from the 399 * era, year-of-era, month-of-year and day-of-month fields. 400 * 401 * @param era the Hijrah era, not null 402 * @param yearOfEra the year-of-era 403 * @param month the month-of-year 404 * @param dayOfMonth the day-of-month 405 * @return the Hijrah local date, not null 406 * @throws DateTimeException if unable to create the date 407 * @throws ClassCastException if the {@code era} is not a {@code HijrahEra} 408 */ 409 @Override 410 public HijrahDate date(Era era, int yearOfEra, int month, int dayOfMonth) { 411 return date(prolepticYear(era, yearOfEra), month, dayOfMonth); 412 } 413 414 /** 415 * Obtains a local date in Hijrah calendar system from the 416 * proleptic-year, month-of-year and day-of-month fields. 417 * 418 * @param prolepticYear the proleptic-year 419 * @param month the month-of-year 420 * @param dayOfMonth the day-of-month 421 * @return the Hijrah local date, not null 422 * @throws DateTimeException if unable to create the date 423 */ 424 @Override 425 public HijrahDate date(int prolepticYear, int month, int dayOfMonth) { 426 return HijrahDate.of(this, prolepticYear, month, dayOfMonth); 427 } 428 429 /** 430 * Obtains a local date in Hijrah calendar system from the 431 * era, year-of-era and day-of-year fields. 432 * 433 * @param era the Hijrah era, not null 434 * @param yearOfEra the year-of-era 435 * @param dayOfYear the day-of-year 436 * @return the Hijrah local date, not null 437 * @throws DateTimeException if unable to create the date 438 * @throws ClassCastException if the {@code era} is not a {@code HijrahEra} 439 */ 440 @Override 441 public HijrahDate dateYearDay(Era era, int yearOfEra, int dayOfYear) { 442 return dateYearDay(prolepticYear(era, yearOfEra), dayOfYear); 443 } 444 445 /** 446 * Obtains a local date in Hijrah calendar system from the 447 * proleptic-year and day-of-year fields. 448 * 449 * @param prolepticYear the proleptic-year 450 * @param dayOfYear the day-of-year 451 * @return the Hijrah local date, not null 452 * @throws DateTimeException if unable to create the date 453 */ 454 @Override 455 public HijrahDate dateYearDay(int prolepticYear, int dayOfYear) { 456 return HijrahDate.of(this, prolepticYear, 1, 1).plusDays(dayOfYear - 1); // TODO better 457 } 458 459 /** 460 * Obtains a local date in the Hijrah calendar system from the epoch-day. 461 * 462 * @param epochDay the epoch day 463 * @return the Hijrah local date, not null 464 * @throws DateTimeException if unable to create the date 465 */ 466 @Override // override with covariant return type 467 public HijrahDate dateEpochDay(long epochDay) { 468 return HijrahDate.ofEpochDay(this, epochDay); 469 } 470 471 @Override 472 public HijrahDate dateNow() { 473 return dateNow(Clock.systemDefaultZone()); 474 } 475 476 @Override 477 public HijrahDate dateNow(ZoneId zone) { 478 return dateNow(Clock.system(zone)); 479 } 480 481 @Override 482 public HijrahDate dateNow(Clock clock) { 483 return date(LocalDate.now(clock)); 484 } 485 486 @Override 487 public HijrahDate date(TemporalAccessor temporal) { 488 if (temporal instanceof HijrahDate) { 489 return (HijrahDate) temporal; 490 } 491 return HijrahDate.ofEpochDay(this, temporal.getLong(EPOCH_DAY)); 492 } 493 494 @Override 495 public ChronoLocalDateTime<HijrahDate> localDateTime(TemporalAccessor temporal) { 496 return (ChronoLocalDateTime<HijrahDate>) super.localDateTime(temporal); 497 } 498 499 @Override 500 public ChronoZonedDateTime<HijrahDate> zonedDateTime(TemporalAccessor temporal) { 501 return (ChronoZonedDateTime<HijrahDate>) super.zonedDateTime(temporal); 502 } 503 504 @Override 505 public ChronoZonedDateTime<HijrahDate> zonedDateTime(Instant instant, ZoneId zone) { 506 return (ChronoZonedDateTime<HijrahDate>) super.zonedDateTime(instant, zone); 507 } 508 509 //----------------------------------------------------------------------- 510 @Override 511 public boolean isLeapYear(long prolepticYear) { 512 int epochMonth = yearToEpochMonth((int) prolepticYear); 513 if (epochMonth < 0 || epochMonth > maxEpochDay) { 514 throw new DateTimeException("Hijrah date out of range"); 515 } 516 int len = getYearLength((int) prolepticYear); 517 return (len > 354); 518 } 519 520 @Override 521 public int prolepticYear(Era era, int yearOfEra) { 522 if (era instanceof HijrahEra == false) { 523 throw new ClassCastException("Era must be HijrahEra"); 524 } 525 return yearOfEra; 526 } 527 528 @Override 529 public Era eraOf(int eraValue) { 530 switch (eraValue) { 531 case 1: 532 return HijrahEra.AH; 533 default: 534 throw new DateTimeException("invalid Hijrah era"); 535 } 536 } 537 538 @Override 539 public List<Era> eras() { 540 return Arrays.<Era>asList(HijrahEra.values()); 541 } 542 543 //----------------------------------------------------------------------- 544 @Override 545 public ValueRange range(ChronoField field) { 546 if (field instanceof ChronoField) { 547 ChronoField f = field; 548 switch (f) { 549 case DAY_OF_MONTH: 550 return ValueRange.of(1, 1, getMinimumMonthLength(), getMaximumMonthLength()); 551 case DAY_OF_YEAR: 552 return ValueRange.of(1, getMaximumDayOfYear()); 553 case ALIGNED_WEEK_OF_MONTH: 554 return ValueRange.of(1, 5); 555 case YEAR: 556 case YEAR_OF_ERA: 557 return ValueRange.of(getMinimumYear(), getMaximumYear()); 558 default: 559 return field.range(); 560 } 561 } 562 return field.range(); 563 } 564 565 /** 566 * Check the validity of a year. 567 * 568 * @param prolepticYear the year to check 569 */ 570 int checkValidYear(long prolepticYear) { 571 if (prolepticYear < getMinimumYear() || prolepticYear > getMaximumYear()) { 572 throw new DateTimeException("Invalid Hijrah year: " + prolepticYear); 573 } 574 return (int) prolepticYear; 575 } 576 577 void checkValidDayOfYear(int dayOfYear) { 578 if (dayOfYear < 1 || dayOfYear > getMaximumDayOfYear()) { 579 throw new DateTimeException("Invalid Hijrah day of year: " + dayOfYear); 580 } 581 } 582 583 void checkValidMonth(int month) { 584 if (month < 1 || month > 12) { 585 throw new DateTimeException("Invalid Hijrah month: " + month); 586 } 587 } 588 589 //----------------------------------------------------------------------- 590 /** 591 * Returns an array containing the Hijrah year, month and day 592 * computed from the epoch day. 593 * 594 * @param epochDay the EpochDay 595 * @return int[0] = YEAR, int[1] = MONTH, int[2] = DATE 596 */ 597 int[] getHijrahDateInfo(int epochDay) { 598 if (epochDay < minEpochDay || epochDay >= maxEpochDay) { 599 throw new DateTimeException("Hijrah date out of range"); 600 } 601 602 int epochMonth = epochDayToEpochMonth(epochDay); 603 int year = epochMonthToYear(epochMonth); 604 int month = epochMonthToMonth(epochMonth); 605 int day1 = epochMonthToEpochDay(epochMonth); 606 int date = epochDay - day1; // epochDay - dayOfEpoch(year, month); 607 608 int dateInfo[] = new int[3]; 609 dateInfo[0] = year; 610 dateInfo[1] = month + 1; // change to 1-based. 611 dateInfo[2] = date + 1; // change to 1-based. 612 return dateInfo; 613 } 614 615 /** 616 * Return the epoch day computed from Hijrah year, month, and day. 617 * 618 * @param prolepticYear the year to represent, 0-origin 619 * @param monthOfYear the month-of-year to represent, 1-origin 620 * @param dayOfMonth the day-of-month to represent, 1-origin 621 * @return the epoch day 622 */ 623 long getEpochDay(int prolepticYear, int monthOfYear, int dayOfMonth) { 624 checkValidMonth(monthOfYear); 625 int epochMonth = yearToEpochMonth(prolepticYear) + (monthOfYear - 1); 626 if (epochMonth < 0 || epochMonth >= hijrahEpochMonthStartDays.length) { 627 throw new DateTimeException("Invalid Hijrah date, year: " + 628 prolepticYear + ", month: " + monthOfYear); 629 } 630 if (dayOfMonth < 1 || dayOfMonth > getMonthLength(prolepticYear, monthOfYear)) { 631 throw new DateTimeException("Invalid Hijrah day of month: " + dayOfMonth); 632 } 633 return epochMonthToEpochDay(epochMonth) + (dayOfMonth - 1); 634 } 635 636 /** 637 * Returns day of year for the year and month. 638 * 639 * @param prolepticYear a proleptic year 640 * @param month a month, 1-origin 641 * @return the day of year, 1-origin 642 */ 643 int getDayOfYear(int prolepticYear, int month) { 644 return yearMonthToDayOfYear(prolepticYear, (month - 1)); 645 } 646 647 /** 648 * Returns month length for the year and month. 649 * 650 * @param prolepticYear a proleptic year 651 * @param monthOfYear a month, 1-origin. 652 * @return the length of the month 653 */ 654 int getMonthLength(int prolepticYear, int monthOfYear) { 655 int epochMonth = yearToEpochMonth(prolepticYear) + (monthOfYear - 1); 656 if (epochMonth < 0 || epochMonth >= hijrahEpochMonthStartDays.length) { 657 throw new DateTimeException("Invalid Hijrah date, year: " + 658 prolepticYear + ", month: " + monthOfYear); 659 } 660 return epochMonthLength(epochMonth); 661 } 662 663 /** 664 * Returns year length. 665 * Note: The 12th month must exist in the data. 666 * 667 * @param prolepticYear a proleptic year 668 * @return year length in days 669 */ 670 int getYearLength(int prolepticYear) { 671 return yearMonthToDayOfYear(prolepticYear, 12); 672 } 673 674 /** 675 * Return the minimum supported Hijrah year. 676 * 677 * @return the minimum 678 */ 679 int getMinimumYear() { 680 return epochMonthToYear(0); 681 } 682 683 /** 684 * Return the maximum supported Hijrah ear. 685 * 686 * @return the minimum 687 */ 688 int getMaximumYear() { 689 return epochMonthToYear(hijrahEpochMonthStartDays.length - 1) - 1; 690 } 691 692 /** 693 * Returns maximum day-of-month. 694 * 695 * @return maximum day-of-month 696 */ 697 int getMaximumMonthLength() { 698 return maxMonthLength; 699 } 700 701 /** 702 * Returns smallest maximum day-of-month. 703 * 704 * @return smallest maximum day-of-month 705 */ 706 int getMinimumMonthLength() { 707 return minMonthLength; 708 } 709 710 /** 711 * Returns maximum day-of-year. 712 * 713 * @return maximum day-of-year 714 */ 715 int getMaximumDayOfYear() { 716 return maxYearLength; 717 } 718 719 /** 720 * Returns smallest maximum day-of-year. 721 * 722 * @return smallest maximum day-of-year 723 */ 724 int getSmallestMaximumDayOfYear() { 725 return minYearLength; 726 } 727 728 /** 729 * Returns the epochMonth found by locating the epochDay in the table. The 730 * epochMonth is the index in the table 731 * 732 * @param epochDay 733 * @return The index of the element of the start of the month containing the 734 * epochDay. 735 */ 736 private int epochDayToEpochMonth(int epochDay) { 737 // binary search 738 int ndx = Arrays.binarySearch(hijrahEpochMonthStartDays, epochDay); 739 if (ndx < 0) { 740 ndx = -ndx - 2; 741 } 742 return ndx; 743 } 744 745 /** 746 * Returns the year computed from the epochMonth 747 * 748 * @param epochMonth the epochMonth 749 * @return the Hijrah Year 750 */ 751 private int epochMonthToYear(int epochMonth) { 752 return (epochMonth + hijrahStartEpochMonth) / 12; 753 } 754 755 /** 756 * Returns the epochMonth for the Hijrah Year. 757 * 758 * @param year the HijrahYear 759 * @return the epochMonth for the beginning of the year. 760 */ 761 private int yearToEpochMonth(int year) { 762 return (year * 12) - hijrahStartEpochMonth; 763 } 764 765 /** 766 * Returns the Hijrah month from the epochMonth. 767 * 768 * @param epochMonth the epochMonth 769 * @return the month of the Hijrah Year 770 */ 771 private int epochMonthToMonth(int epochMonth) { 772 return (epochMonth + hijrahStartEpochMonth) % 12; 773 } 774 775 /** 776 * Returns the epochDay for the start of the epochMonth. 777 * 778 * @param epochMonth the epochMonth 779 * @return the epochDay for the start of the epochMonth. 780 */ 781 private int epochMonthToEpochDay(int epochMonth) { 782 return hijrahEpochMonthStartDays[epochMonth]; 783 784 } 785 786 /** 787 * Returns the day of year for the requested HijrahYear and month. 788 * 789 * @param prolepticYear the Hijrah year 790 * @param month the Hijrah month 791 * @return the day of year for the start of the month of the year 792 */ 793 private int yearMonthToDayOfYear(int prolepticYear, int month) { 794 int epochMonthFirst = yearToEpochMonth(prolepticYear); 795 return epochMonthToEpochDay(epochMonthFirst + month) 796 - epochMonthToEpochDay(epochMonthFirst); 797 } 798 799 /** 800 * Returns the length of the epochMonth. It is computed from the start of 801 * the following month minus the start of the requested month. 802 * 803 * @param epochMonth the epochMonth; assumed to be within range 804 * @return the length in days of the epochMonth 805 */ 806 private int epochMonthLength(int epochMonth) { 807 // The very last entry in the epochMonth table is not the start of a month 808 return hijrahEpochMonthStartDays[epochMonth + 1] 809 - hijrahEpochMonthStartDays[epochMonth]; 810 } 811 812 //----------------------------------------------------------------------- 813 private static final String KEY_ID = "id"; 814 private static final String KEY_TYPE = "type"; 815 private static final String KEY_VERSION = "version"; 816 private static final String KEY_ISO_START = "iso-start"; 817 818 /** 819 * Return the configuration properties from the resource. 820 * <p> 821 * The default location of the variant configuration resource is: 822 * <pre> 823 * "$java.home/lib/" + resource-name 824 * </pre> 825 * 826 * @param resource the name of the calendar property resource 827 * @return a Properties containing the properties read from the resource. 828 * @throws Exception if access to the property resource fails 829 */ 830 private static Properties readConfigProperties(final String resource) throws Exception { 831 try { 832 return AccessController 833 .doPrivileged((java.security.PrivilegedExceptionAction<Properties>) 834 () -> { 835 String libDir = System.getProperty("java.home") + File.separator + "lib"; 836 File file = new File(libDir, resource); 837 Properties props = new Properties(); 838 try (InputStream is = new FileInputStream(file)) { 839 props.load(is); 840 } 841 return props; 842 }); 843 } catch (PrivilegedActionException pax) { 844 throw pax.getException(); 845 } 846 } 847 848 /** 849 * Loads and processes the Hijrah calendar properties file. 850 * The starting Hijrah date and the corresponding ISO date are 851 * extracted and used to calculate the epochDate offset. 852 * The version number is identified and ignored. 853 * Everything else is the data for a year with containing the length of each 854 * of 12 months. 855 * 856 * @param resourceName containing the properties defining the calendar, not null 857 * @throws IllegalArgumentException if any of the values are malformed 858 * @throws NumberFormatException if numbers, including properties that should 859 * be years are invalid 860 * @throws IOException if access to the property resource fails. 861 */ 862 private void loadCalendarData(String resourceName) throws Exception { 863 Properties props = readConfigProperties(resourceName); 864 865 Map<Integer, int[]> years = new HashMap<>(); 866 int minYear = Integer.MAX_VALUE; 867 int maxYear = Integer.MIN_VALUE; 868 String id = null; 869 String type = null; 870 String version = null; 871 int isoStart = 0; 872 for (Map.Entry<Object, Object> entry : props.entrySet()) { 873 String key = (String) entry.getKey(); 874 switch (key) { 875 case KEY_ID: 876 id = (String)entry.getValue(); 877 break; 878 case KEY_TYPE: 879 type = (String)entry.getValue(); 880 break; 881 case KEY_VERSION: 882 version = (String)entry.getValue(); 883 break; 884 case KEY_ISO_START: { 885 int[] ymd = parseYMD((String) entry.getValue()); 886 isoStart = (int) LocalDate.of(ymd[0], ymd[1], ymd[2]).toEpochDay(); 887 break; 888 } 889 default: 890 try { 891 // Everything else is either a year or invalid 892 int year = Integer.valueOf(key); 893 int[] months = parseMonths((String) entry.getValue()); 894 years.put(year, months); 895 maxYear = Math.max(maxYear, year); 896 minYear = Math.min(minYear, year); 897 } catch (NumberFormatException nfe) { 898 throw new IllegalArgumentException("bad key: " + key); 899 } 900 } 901 } 902 903 if (!getId().equals(id)) { 904 throw new IllegalArgumentException("Configuration is for a different calendar: " + id); 905 } 906 if (!getCalendarType().equals(type)) { 907 throw new IllegalArgumentException("Configuration is for a different calendar type: " + type); 908 } 909 if (version == null || version.isEmpty()) { 910 throw new IllegalArgumentException("Configuration does not contain a version"); 911 } 912 if (isoStart == 0) { 913 throw new IllegalArgumentException("Configuration does not contain a ISO start date"); 914 } 915 916 // Now create and validate the array of epochDays indexed by epochMonth 917 hijrahStartEpochMonth = minYear * 12; 918 minEpochDay = isoStart; 919 hijrahEpochMonthStartDays = createEpochMonths(minEpochDay, minYear, maxYear, years); 920 maxEpochDay = hijrahEpochMonthStartDays[hijrahEpochMonthStartDays.length - 1]; 921 922 // Compute the min and max year length in days. 923 for (int year = minYear; year < maxYear; year++) { 924 int length = getYearLength(year); 925 minYearLength = Math.min(minYearLength, length); 926 maxYearLength = Math.max(maxYearLength, length); 927 } 928 } 929 930 /** 931 * Converts the map of year to month lengths ranging from minYear to maxYear 932 * into a linear contiguous array of epochDays. The index is the hijrahMonth 933 * computed from year and month and offset by minYear. The value of each 934 * entry is the epochDay corresponding to the first day of the month. 935 * 936 * @param minYear The minimum year for which data is provided 937 * @param maxYear The maximum year for which data is provided 938 * @param years a Map of year to the array of 12 month lengths 939 * @return array of epochDays for each month from min to max 940 */ 941 private int[] createEpochMonths(int epochDay, int minYear, int maxYear, Map<Integer, int[]> years) { 942 // Compute the size for the array of dates 943 int numMonths = (maxYear - minYear + 1) * 12 + 1; 944 945 // Initialize the running epochDay as the corresponding ISO Epoch day 946 int epochMonth = 0; // index into array of epochMonths 947 int[] epochMonths = new int[numMonths]; 948 minMonthLength = Integer.MAX_VALUE; 949 maxMonthLength = Integer.MIN_VALUE; 950 951 // Only whole years are valid, any zero's in the array are illegal 952 for (int year = minYear; year <= maxYear; year++) { 953 int[] months = years.get(year);// must not be gaps 954 for (int month = 0; month < 12; month++) { 955 int length = months[month]; 956 epochMonths[epochMonth++] = epochDay; 957 958 if (length < 29 || length > 32) { 959 throw new IllegalArgumentException("Invalid month length in year: " + minYear); 960 } 961 epochDay += length; 962 minMonthLength = Math.min(minMonthLength, length); 963 maxMonthLength = Math.max(maxMonthLength, length); 964 } 965 } 966 967 // Insert the final epochDay 968 epochMonths[epochMonth++] = epochDay; 969 970 if (epochMonth != epochMonths.length) { 971 throw new IllegalStateException("Did not fill epochMonths exactly: ndx = " + epochMonth 972 + " should be " + epochMonths.length); 973 } 974 975 return epochMonths; 976 } 977 978 /** 979 * Parses the 12 months lengths from a property value for a specific year. 980 * 981 * @param line the value of a year property 982 * @return an array of int[12] containing the 12 month lengths 983 * @throws IllegalArgumentException if the number of months is not 12 984 * @throws NumberFormatException if the 12 tokens are not numbers 985 */ 986 private int[] parseMonths(String line) { 987 int[] months = new int[12]; 988 String[] numbers = line.split("\\s"); 989 if (numbers.length != 12) { 990 throw new IllegalArgumentException("wrong number of months on line: " + Arrays.toString(numbers) + "; count: " + numbers.length); 991 } 992 for (int i = 0; i < 12; i++) { 993 try { 994 months[i] = Integer.valueOf(numbers[i]); 995 } catch (NumberFormatException nfe) { 996 throw new IllegalArgumentException("bad key: " + numbers[i]); 997 } 998 } 999 return months; 1000 } 1001 1002 /** 1003 * Parse yyyy-MM-dd into a 3 element array [yyyy, mm, dd]. 1004 * 1005 * @param string the input string 1006 * @return the 3 element array with year, month, day 1007 */ 1008 private int[] parseYMD(String string) { 1009 // yyyy-MM-dd 1010 string = string.trim(); 1011 try { 1012 if (string.charAt(4) != '-' || string.charAt(7) != '-') { 1013 throw new IllegalArgumentException("date must be yyyy-MM-dd"); 1014 } 1015 int[] ymd = new int[3]; 1016 ymd[0] = Integer.valueOf(string.substring(0, 4)); 1017 ymd[1] = Integer.valueOf(string.substring(5, 7)); 1018 ymd[2] = Integer.valueOf(string.substring(8, 10)); 1019 return ymd; 1020 } catch (NumberFormatException ex) { 1021 throw new IllegalArgumentException("date must be yyyy-MM-dd", ex); 1022 } 1023 } 1024 }