All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.nimbusds.oauth2.sdk.util.date.DateWithTimeZoneOffset Maven / Gradle / Ivy

/*
 * oauth2-oidc-sdk
 *
 * Copyright 2012-2016, Connect2id Ltd and contributors.
 *
 * Licensed under the Apache License, Version 2.0 (the "License"); you may not use
 * this file except in compliance with the License. You may obtain a copy of the
 * License at
 *
 *    http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software distributed
 * under the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR
 * CONDITIONS OF ANY KIND, either express or implied. See the License for the
 * specific language governing permissions and limitations under the License.
 */

package com.nimbusds.oauth2.sdk.util.date;


import java.text.SimpleDateFormat;
import java.util.Date;
import java.util.Objects;
import java.util.TimeZone;
import java.util.regex.Matcher;
import java.util.regex.Pattern;

import com.nimbusds.jwt.util.DateUtils;
import com.nimbusds.oauth2.sdk.ParseException;


/**
 * Date with optional timezone offset. Supports basic ISO 8601 formatting and
 * parsing.
 */
public class DateWithTimeZoneOffset {
	
	
	/**
	 * The date.
	 */
	private final Date date;
	
	
	/**
	 * The time zone offset in minutes relative to UTC.
	 */
	private final int tzOffsetMinutes;
	
	
	/**
	 * {@code true} if the date is in UTC.
	 */
	private final boolean isUTC;
	
	
	/**
	 * Creates a new date in UTC, to be {@link #toISO8601String() output}
	 * with {@code Z} timezone designation.
	 *
	 * @param date The date. Must not be {@code null}.
	 */
	public DateWithTimeZoneOffset(final Date date) {
		if (date == null) {
			throw new IllegalArgumentException("The date must not be null");
		}
		this.date = date;
		tzOffsetMinutes = 0;
		isUTC = true;
	}
	
	
	/**
	 * Creates a new date with timezone offset.
	 *
	 * @param date            The date. Must not be {@code null}.
	 * @param tzOffsetMinutes The time zone offset in minutes relative to
	 *                        UTC, zero if none. Must be less than
	 *                        {@code +/- 12 x 60}.
	 */
	public DateWithTimeZoneOffset(final Date date, final int tzOffsetMinutes) {
		if (date == null) {
			throw new IllegalArgumentException("The date must not be null");
		}
		this.date = date;
		if (tzOffsetMinutes >= 12*60 || tzOffsetMinutes <= -12*60) {
			throw new IllegalArgumentException("The time zone offset must be less than +/- 12 x 60 minutes");
		}
		this.tzOffsetMinutes = tzOffsetMinutes;
		isUTC = false;
	}
	
	
	/**
	 * Creates a new date with timezone offset.
	 *
	 * @param date The date. Must not be {@code null}.
	 * @param tz   The time zone to determine the time zone offset.
	 */
	public DateWithTimeZoneOffset(final Date date, final TimeZone tz) {
		this(date, tz.getOffset(date.getTime()) / 60_000);
	}
	
	
	/**
	 * Returns the date.
	 *
	 * @return The date.
	 */
	public Date getDate() {
		return date;
	}
	
	
	/**
	 * Returns {@code true} if the date is in UTC.
	 *
	 * @return {@code true} if the date is in UTC, else the time zone
	 *         {@link #getTimeZoneOffsetMinutes offset} applies.
	 */
	public boolean isUTC() {
		return isUTC;
	}
	
	
	/**
	 * Returns the time zone offset in minutes relative to UTC.
	 *
	 * @return The time zone offset in minutes relative to UTC, zero if
	 *         none.
	 */
	public int getTimeZoneOffsetMinutes() {
		return tzOffsetMinutes;
	}
	
	
	/**
	 * Returns an ISO 8601 representation in
	 * {@code YYYY-MM-DDThh:mm:ssZ} or {@code YYYY-MM-DDThh:mm:ss±hh:mm}
	 * format
	 *
	 * 

Example: {@code 2019-11-01T18:19:43+03:00} * * @return The ISO 8601 representation. */ public String toISO8601String() { SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd'T'HH:mm:ss"); // Hack to format date/time with TZ offset TimeZone tz = TimeZone.getTimeZone("UTC"); sdf.setTimeZone(tz); long localTimeSeconds = DateUtils.toSecondsSinceEpoch(date); localTimeSeconds = localTimeSeconds + (tzOffsetMinutes * 60L); String out = sdf.format(DateUtils.fromSecondsSinceEpoch(localTimeSeconds)); if (isUTC()) { return out + "Z"; } // Append TZ offset int tzOffsetWholeHours = tzOffsetMinutes / 60; int tzOffsetRemainderMinutes = tzOffsetMinutes - (tzOffsetWholeHours * 60); if (tzOffsetMinutes == 0) { return out + "+00:00"; } if (tzOffsetWholeHours > 0) { out += "+" + (tzOffsetWholeHours < 10 ? "0" : "") + Math.abs(tzOffsetWholeHours); } else if (tzOffsetWholeHours < 0) { out += "-" + (tzOffsetWholeHours > -10 ? "0" : "") + Math.abs(tzOffsetWholeHours); } else { if (tzOffsetMinutes > 0) { out += "+00"; } else { out += "-00"; } } out += ":"; if (tzOffsetRemainderMinutes > 0) { out += (tzOffsetRemainderMinutes < 10 ? "0" : "") + tzOffsetRemainderMinutes; } else if (tzOffsetRemainderMinutes < 0) { out += (tzOffsetRemainderMinutes > -10 ? "0" : "") + Math.abs(tzOffsetRemainderMinutes); } else { out += "00"; } return out; } @Override public String toString() { return toISO8601String(); } @Override public boolean equals(Object o) { if (this == o) return true; if (!(o instanceof DateWithTimeZoneOffset)) return false; DateWithTimeZoneOffset that = (DateWithTimeZoneOffset) o; return tzOffsetMinutes == that.tzOffsetMinutes && getDate().equals(that.getDate()); } @Override public int hashCode() { return Objects.hash(getDate(), tzOffsetMinutes); } /** * Parses an ISO 8601 representation in * {@code YYYY-MM-DDThh:mm:ss±hh:mm} format. * *

Example: {@code 2019-11-01T18:19:43+03:00} * * @param s The string to parse. * * @return The date with timezone offset. * * @throws ParseException If parsing failed. */ public static DateWithTimeZoneOffset parseISO8601String(final String s) throws ParseException { String stringToParse = s; if (Pattern.compile(".*[\\+\\-][\\d]{2}$").matcher(s).matches()) { // append minutes to hour resolution offset TZ stringToParse += ":00"; } Matcher m = Pattern.compile("(.*[\\+\\-][\\d]{2})(\\d{2})$").matcher(stringToParse); if (m.matches()) { // insert colon between hh and mm offset stringToParse = m.group(1) + ":" + m.group(2); } m = Pattern.compile("(.*\\d{2}:\\d{2}:\\d{2})([\\+\\-Z].*)$").matcher(stringToParse); if (m.matches()) { // insert zero milliseconds stringToParse = m.group(1) + ".000" + m.group(2); } int colonCount = stringToParse.length() - stringToParse.replace(":", "").length(); Date date; try { if (colonCount == 1) { date = new SimpleDateFormat("yyyy-MM-dd'T'HH:mmXXX").parse(stringToParse); } else { date = new SimpleDateFormat("yyyy-MM-dd'T'HH:mm:ss.SSSXXX").parse(stringToParse); } } catch (java.text.ParseException e) { throw new ParseException(e.getMessage()); } if (stringToParse.trim().endsWith("Z") || stringToParse.trim().endsWith("z")) { return new DateWithTimeZoneOffset(date); // UTC } int tzOffsetMinutes; try { // E.g. +03:00 String offsetSpec = stringToParse.substring("2019-11-01T06:19:43.000".length()); int hoursOffset = Integer.parseInt(offsetSpec.substring(0, 3)); int minutesOffset = Integer.parseInt(offsetSpec.substring(4)); if (offsetSpec.startsWith("+")) { tzOffsetMinutes = hoursOffset * 60 + minutesOffset; } else { // E.g. -03:00, -00:30 tzOffsetMinutes = hoursOffset * 60 - minutesOffset; } } catch (Exception e) { throw new ParseException("Unexpected timezone offset: " + s); } return new DateWithTimeZoneOffset(date, tzOffsetMinutes); } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy