001/*
002 * #%L
003 * HAPI FHIR - Core Library
004 * %%
005 * Copyright (C) 2014 - 2025 Smile CDR, Inc.
006 * %%
007 * Licensed under the Apache License, Version 2.0 (the "License");
008 * you may not use this file except in compliance with the License.
009 * You may obtain a copy of the License at
010 *
011 *      http://www.apache.org/licenses/LICENSE-2.0
012 *
013 * Unless required by applicable law or agreed to in writing, software
014 * distributed under the License is distributed on an "AS IS" BASIS,
015 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
016 * See the License for the specific language governing permissions and
017 * limitations under the License.
018 * #L%
019 */
020package ca.uhn.fhir.util;
021
022import ca.uhn.fhir.i18n.Msg;
023import com.google.common.base.Preconditions;
024import org.apache.commons.lang3.StringUtils;
025import org.apache.commons.lang3.tuple.ImmutablePair;
026import org.apache.commons.lang3.tuple.Pair;
027
028import java.lang.ref.SoftReference;
029import java.text.ParseException;
030import java.text.ParsePosition;
031import java.text.SimpleDateFormat;
032import java.time.LocalDateTime;
033import java.time.Month;
034import java.time.YearMonth;
035import java.time.format.DateTimeFormatter;
036import java.time.temporal.ChronoField;
037import java.time.temporal.TemporalAccessor;
038import java.time.temporal.TemporalField;
039import java.util.Calendar;
040import java.util.Date;
041import java.util.HashMap;
042import java.util.Locale;
043import java.util.Map;
044import java.util.Objects;
045import java.util.Optional;
046import java.util.TimeZone;
047
048/**
049 * A utility class for parsing and formatting HTTP dates as used in cookies and
050 * other headers.  This class handles dates as defined by RFC 2616 section
051 * 3.3.1 as well as some other common non-standard formats.
052 * <p>
053 * This class is basically intended to be a high-performance workaround
054 * for the fact that Java SimpleDateFormat is kind of expensive to
055 * create and yet isn't thread safe.
056 * </p>
057 * <p>
058 * This class was adapted from the class with the same name from the Jetty
059 * project, licensed under the terms of the Apache Software License 2.0.
060 * </p>
061 */
062public final class DateUtils {
063
064        /**
065         * GMT TimeZone
066         */
067        public static final TimeZone GMT = TimeZone.getTimeZone("GMT");
068
069        /**
070         * Date format pattern used to parse HTTP date headers in RFC 1123 format.
071         */
072        @SuppressWarnings("WeakerAccess")
073        public static final String PATTERN_RFC1123 = "EEE, dd MMM yyyy HH:mm:ss zzz";
074
075        /**
076         * Date format pattern used to parse HTTP date headers in RFC 1036 format.
077         */
078        @SuppressWarnings("WeakerAccess")
079        public static final String PATTERN_RFC1036 = "EEE, dd-MMM-yy HH:mm:ss zzz";
080
081        /**
082         * Date format pattern used to parse HTTP date headers in ANSI C
083         * {@code asctime()} format.
084         */
085        @SuppressWarnings("WeakerAccess")
086        public static final String PATTERN_ASCTIME = "EEE MMM d HH:mm:ss yyyy";
087
088        private static final String PATTERN_INTEGER_DATE = "yyyyMMdd";
089
090        private static final String[] DEFAULT_PATTERNS = new String[] {PATTERN_RFC1123, PATTERN_RFC1036, PATTERN_ASCTIME};
091        private static final Date DEFAULT_TWO_DIGIT_YEAR_START;
092
093        static {
094                final Calendar calendar = Calendar.getInstance();
095                calendar.setTimeZone(GMT);
096                calendar.set(2000, Calendar.JANUARY, 1, 0, 0, 0);
097                calendar.set(Calendar.MILLISECOND, 0);
098                DEFAULT_TWO_DIGIT_YEAR_START = calendar.getTime();
099        }
100
101        /**
102         * This class should not be instantiated.
103         */
104        private DateUtils() {}
105
106        /**
107         * Calculate a LocalDateTime with any missing date/time data points defaulting to the earliest values (ex 0 for hour)
108         * from a TemporalAccessor or empty if it doesn't contain a year.
109         *
110         * @param theTemporalAccessor The TemporalAccessor containing date/time information
111         * @return A LocalDateTime or empty
112         */
113        public static Optional<LocalDateTime> extractLocalDateTimeForRangeStartOrEmpty(
114                        TemporalAccessor theTemporalAccessor) {
115                if (theTemporalAccessor.isSupported(ChronoField.YEAR)) {
116                        final int year = theTemporalAccessor.get(ChronoField.YEAR);
117                        final Month month = Month.of(getTimeUnitIfSupported(theTemporalAccessor, ChronoField.MONTH_OF_YEAR, 1));
118                        final int day = getTimeUnitIfSupported(theTemporalAccessor, ChronoField.DAY_OF_MONTH, 1);
119                        final int hour = getTimeUnitIfSupported(theTemporalAccessor, ChronoField.HOUR_OF_DAY, 0);
120                        final int minute = getTimeUnitIfSupported(theTemporalAccessor, ChronoField.MINUTE_OF_HOUR, 0);
121                        final int seconds = getTimeUnitIfSupported(theTemporalAccessor, ChronoField.SECOND_OF_MINUTE, 0);
122
123                        return Optional.of(LocalDateTime.of(year, month, day, hour, minute, seconds));
124                }
125
126                return Optional.empty();
127        }
128
129        /**
130         * Calculate a LocalDateTime with any missing date/time data points defaulting to the latest values (ex 23 for hour)
131         * from a TemporalAccessor or empty if it doesn't contain a year.
132         *
133         * @param theTemporalAccessor The TemporalAccessor containing date/time information
134         * @return A LocalDateTime or empty
135         */
136        public static Optional<LocalDateTime> extractLocalDateTimeForRangeEndOrEmpty(TemporalAccessor theTemporalAccessor) {
137                if (theTemporalAccessor.isSupported(ChronoField.YEAR)) {
138                        final int year = theTemporalAccessor.get(ChronoField.YEAR);
139                        final Month month = Month.of(getTimeUnitIfSupported(theTemporalAccessor, ChronoField.MONTH_OF_YEAR, 12));
140                        final int day = getTimeUnitIfSupported(
141                                        theTemporalAccessor,
142                                        ChronoField.DAY_OF_MONTH,
143                                        YearMonth.of(year, month).atEndOfMonth().getDayOfMonth());
144                        final int hour = getTimeUnitIfSupported(theTemporalAccessor, ChronoField.HOUR_OF_DAY, 23);
145                        final int minute = getTimeUnitIfSupported(theTemporalAccessor, ChronoField.MINUTE_OF_HOUR, 59);
146                        final int seconds = getTimeUnitIfSupported(theTemporalAccessor, ChronoField.SECOND_OF_MINUTE, 59);
147
148                        return Optional.of(LocalDateTime.of(year, month, day, hour, minute, seconds));
149                }
150
151                return Optional.empty();
152        }
153
154        /**
155         * With the provided DateTimeFormatter, parse a date time String or return empty if the String doesn't correspond
156         * to the formatter.
157         *
158         * @param theDateTimeString A date/time String in some date format
159         * @param theSupportedDateTimeFormatter The DateTimeFormatter we expect corresponds to the String
160         * @return The parsed TemporalAccessor or empty
161         */
162        public static Optional<TemporalAccessor> parseDateTimeStringIfValid(
163                        String theDateTimeString, DateTimeFormatter theSupportedDateTimeFormatter) {
164                Objects.requireNonNull(theSupportedDateTimeFormatter);
165                Preconditions.checkArgument(StringUtils.isNotBlank(theDateTimeString));
166
167                try {
168                        return Optional.of(theSupportedDateTimeFormatter.parse(theDateTimeString));
169                } catch (Exception exception) {
170                        return Optional.empty();
171                }
172        }
173
174        private static int getTimeUnitIfSupported(
175                        TemporalAccessor theTemporalAccessor, TemporalField theTemporalField, int theDefaultValue) {
176                return getTimeUnitIfSupportedOrEmpty(theTemporalAccessor, theTemporalField)
177                                .orElse(theDefaultValue);
178        }
179
180        private static Optional<Integer> getTimeUnitIfSupportedOrEmpty(
181                        TemporalAccessor theTemporalAccessor, TemporalField theTemporalField) {
182                if (theTemporalAccessor.isSupported(theTemporalField)) {
183                        return Optional.of(theTemporalAccessor.get(theTemporalField));
184                }
185
186                return Optional.empty();
187        }
188
189        /**
190         * A factory for {@link SimpleDateFormat}s. The instances are stored in a
191         * threadlocal way because SimpleDateFormat is not thread safe as noted in
192         * {@link SimpleDateFormat its javadoc}.
193         */
194        static final class DateFormatHolder {
195
196                private static final ThreadLocal<SoftReference<Map<String, SimpleDateFormat>>> THREADLOCAL_FORMATS =
197                                ThreadLocal.withInitial(() -> new SoftReference<>(new HashMap<>()));
198
199                /**
200                 * creates a {@link SimpleDateFormat} for the requested format string.
201                 *
202                 * @param pattern a non-{@code null} format String according to
203                 *                {@link SimpleDateFormat}. The format is not checked against
204                 *                {@code null} since all paths go through
205                 *                {@link DateUtils}.
206                 * @return the requested format. This simple DateFormat should not be used
207                 * to {@link SimpleDateFormat#applyPattern(String) apply} to a
208                 * different pattern.
209                 */
210                static SimpleDateFormat formatFor(final String pattern) {
211                        final SoftReference<Map<String, SimpleDateFormat>> ref = THREADLOCAL_FORMATS.get();
212                        Map<String, SimpleDateFormat> formats = ref.get();
213                        if (formats == null) {
214                                formats = new HashMap<>();
215                                THREADLOCAL_FORMATS.set(new SoftReference<>(formats));
216                        }
217
218                        SimpleDateFormat format = formats.get(pattern);
219                        if (format == null) {
220                                format = new SimpleDateFormat(pattern, Locale.US);
221                                format.setTimeZone(TimeZone.getTimeZone("GMT"));
222                                formats.put(pattern, format);
223                        }
224
225                        return format;
226                }
227        }
228
229        /**
230         * Parses a date value.  The formats used for parsing the date value are retrieved from
231         * the default http params.
232         *
233         * @param theDateValue the date value to parse
234         * @return the parsed date or null if input could not be parsed
235         */
236        public static Date parseDate(final String theDateValue) {
237                notNull(theDateValue, "Date value");
238                String v = theDateValue;
239                if (v.length() > 1 && v.startsWith("'") && v.endsWith("'")) {
240                        v = v.substring(1, v.length() - 1);
241                }
242
243                for (final String dateFormat : DEFAULT_PATTERNS) {
244                        final SimpleDateFormat dateParser = DateFormatHolder.formatFor(dateFormat);
245                        dateParser.set2DigitYearStart(DEFAULT_TWO_DIGIT_YEAR_START);
246                        final ParsePosition pos = new ParsePosition(0);
247                        final Date result = dateParser.parse(v, pos);
248                        if (pos.getIndex() != 0) {
249                                return result;
250                        }
251                }
252                return null;
253        }
254
255        public static Date getHighestInstantFromDate(Date theDateValue) {
256                Calendar sourceCal = Calendar.getInstance();
257                sourceCal.setTime(theDateValue);
258
259                Calendar cal = Calendar.getInstance(TimeZone.getTimeZone("GMT-12:00"));
260                copyDateAndTrundateTime(sourceCal, cal);
261                return cal.getTime();
262        }
263
264        public static Date getLowestInstantFromDate(Date theDateValue) {
265                Calendar sourceCal = Calendar.getInstance();
266                sourceCal.setTime(theDateValue);
267
268                Calendar cal = Calendar.getInstance(TimeZone.getTimeZone("GMT+14:00"));
269                copyDateAndTrundateTime(sourceCal, cal);
270                return cal.getTime();
271        }
272
273        private static void copyDateAndTrundateTime(Calendar theSourceCal, Calendar theCal) {
274                theCal.set(Calendar.YEAR, theSourceCal.get(Calendar.YEAR));
275                theCal.set(Calendar.MONTH, theSourceCal.get(Calendar.MONTH));
276                theCal.set(Calendar.DAY_OF_MONTH, theSourceCal.get(Calendar.DAY_OF_MONTH));
277                theCal.set(Calendar.HOUR_OF_DAY, 0);
278                theCal.set(Calendar.MINUTE, 0);
279                theCal.set(Calendar.SECOND, 0);
280                theCal.set(Calendar.MILLISECOND, 0);
281        }
282
283        public static int convertDateToDayInteger(final Date theDateValue) {
284                notNull(theDateValue, "Date value");
285                SimpleDateFormat format = new SimpleDateFormat(PATTERN_INTEGER_DATE);
286                String theDateString = format.format(theDateValue);
287                return Integer.parseInt(theDateString);
288        }
289
290        public static String convertDateToIso8601String(final Date theDateValue) {
291                notNull(theDateValue, "Date value");
292                SimpleDateFormat format = new SimpleDateFormat("yyyy-MM-dd'T'HH:mm:ss.SSSXXX");
293                return format.format(theDateValue);
294        }
295
296        /**
297         * Formats the given date according to the RFC 1123 pattern.
298         *
299         * @param date The date to format.
300         * @return An RFC 1123 formatted date string.
301         * @see #PATTERN_RFC1123
302         */
303        public static String formatDate(final Date date) {
304                notNull(date, "Date");
305                notNull(PATTERN_RFC1123, "Pattern");
306                final SimpleDateFormat formatter = DateFormatHolder.formatFor(PATTERN_RFC1123);
307                return formatter.format(date);
308        }
309
310        public static <T> T notNull(final T argument, final String name) {
311                if (argument == null) {
312                        throw new IllegalArgumentException(Msg.code(1783) + name + " may not be null");
313                }
314                return argument;
315        }
316
317        /**
318         * Convert an incomplete date e.g. 2020 or 2020-01 to a complete date with lower
319         * bound to the first day of the year/month, and upper bound to the last day of
320         * the year/month
321         *
322         *  e.g. 2020 to 2020-01-01 (left), 2020-12-31 (right)
323         *  2020-02 to 2020-02-01 (left), 2020-02-29 (right)
324         *
325         * @param theIncompleteDateStr 2020 or 2020-01
326         * @return a pair of complete date, left is lower bound, and right is upper bound
327         */
328        public static Pair<String, String> getCompletedDate(String theIncompleteDateStr) {
329
330                if (StringUtils.isBlank(theIncompleteDateStr)) return new ImmutablePair<>(null, null);
331
332                String lbStr, upStr;
333                // YYYY only, return the last day of the year
334                if (theIncompleteDateStr.length() == 4) {
335                        lbStr = theIncompleteDateStr + "-01-01"; // first day of the year
336                        upStr = theIncompleteDateStr + "-12-31"; // last day of the year
337                        return new ImmutablePair<>(lbStr, upStr);
338                }
339
340                // Not YYYY-MM, no change
341                if (theIncompleteDateStr.length() != 7) return new ImmutablePair<>(theIncompleteDateStr, theIncompleteDateStr);
342
343                // YYYY-MM Only
344                Date lb;
345                try {
346                        // first day of the month
347                        lb = new SimpleDateFormat("yyyy-MM-dd").parse(theIncompleteDateStr + "-01");
348                } catch (ParseException e) {
349                        return new ImmutablePair<>(theIncompleteDateStr, theIncompleteDateStr);
350                }
351
352                // last day of the month
353                Calendar calendar = Calendar.getInstance();
354                calendar.setTime(lb);
355
356                calendar.add(Calendar.MONTH, 1);
357                calendar.set(Calendar.DAY_OF_MONTH, 1);
358                calendar.add(Calendar.DATE, -1);
359
360                Date ub = calendar.getTime();
361
362                lbStr = new SimpleDateFormat("yyyy-MM-dd").format(lb);
363                upStr = new SimpleDateFormat("yyyy-MM-dd").format(ub);
364
365                return new ImmutablePair<>(lbStr, upStr);
366        }
367
368        public static Date getEndOfDay(Date theDate) {
369
370                Calendar cal = Calendar.getInstance();
371                cal.setTime(theDate);
372                cal.set(Calendar.HOUR_OF_DAY, cal.getMaximum(Calendar.HOUR_OF_DAY));
373                cal.set(Calendar.MINUTE, cal.getMaximum(Calendar.MINUTE));
374                cal.set(Calendar.SECOND, cal.getMaximum(Calendar.SECOND));
375                cal.set(Calendar.MILLISECOND, cal.getMaximum(Calendar.MILLISECOND));
376                return cal.getTime();
377        }
378}