net.sf.saxon.option.exslt.Date Maven / Gradle / Ivy
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
// Copyright (c) 2015 Saxonica Limited.
// This Source Code Form is subject to the terms of the Mozilla Public License, v. 2.0.
// If a copy of the MPL was not distributed with this file, You can obtain one at http://mozilla.org/MPL/2.0/.
// This Source Code Form is "Incompatible With Secondary Licenses", as defined by the Mozilla Public License, v. 2.0.
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
package net.sf.saxon.option.exslt;
import net.sf.saxon.expr.XPathContext;
import net.sf.saxon.functions.AccessorFn;
import net.sf.saxon.lib.ConversionRules;
import net.sf.saxon.om.Item;
import net.sf.saxon.om.SequenceIterator;
import net.sf.saxon.trans.XPathException;
import net.sf.saxon.type.BuiltInAtomicType;
import net.sf.saxon.type.ConversionResult;
import net.sf.saxon.type.Converter;
import net.sf.saxon.type.ValidationFailure;
import net.sf.saxon.value.*;
import java.math.BigDecimal;
import java.util.Calendar;
import java.util.GregorianCalendar;
/**
* This class implements extension functions in the
* http://exslt.org/dates-and-times namespace.
*/
public final class Date {
/**
* Private constructor to disallow instantiation
*/
private Date() {
}
/**
* The date:date-time function returns the current date and time as a date/time string.
* The date/time string that's returned must be a string in the format defined as the
* lexical representation of xs:dateTime in [3.2.7 dateTime] of [XML Schema Part 2: Datatypes].
* The date/time format is basically CCYY-MM-DDThh:mm:ss+hh:mm.
* The date/time string format must include a time zone, either a Z to indicate
* Coordinated Universal Time or a + or - followed by the difference between the
* difference from UTC represented as hh:mm.
*
* @param context the XPath dynamic context
* @return the current date and time as a date/time string
* @throws XPathException if the context does not allow a date and time to be obtained
*/
public static String dateTime(XPathContext context) throws XPathException {
return context.getCurrentDateTime().getStringValue();
}
/**
* The date:date function returns the date specified in the date/time string given
* as the argument. If no argument is given, then the current local date/time, as
* returned by date:date-time is used as a default argument.
* The date/time string that's returned must be a string in the format defined as the
* lexical representation of xs:dateTime in
* [3.2.7 dateTime] of
* [XML Schema Part 2: Datatypes].
* If the argument is not in either of these formats, date:date returns an empty string ('').
* The date/time format is basically CCYY-MM-DDThh:mm:ss, although implementers should consult
* [XML Schema Part 2: Datatypes] and
* [ISO 8601] for details.
* The date is returned as a string with a lexical representation as defined for xs:date in
* [3.2.9 date] of [XML Schema Part 2: Datatypes]. The date format is basically CCYY-MM-DD,
* although implementers should consult [XML Schema Part 2: Datatypes] and [ISO 8601] for details.
* If no argument is given or the argument date/time specifies a time zone, then the date string
* format must include a time zone, either a Z to indicate Coordinated Universal Time or a + or -
* followed by the difference between the difference from UTC represented as hh:mm. If an argument
* is specified and it does not specify a time zone, then the date string format must not include
* a time zone.
*
* @param context the XPath dynamic context
* @param datetimeIn the date and time as a date/time string
* @return the date part of the dateTime supplied, or "" if invalid
*/
public static String date(XPathContext context, String datetimeIn) {
ConversionRules rules = context.getConfiguration().getConversionRules();
datetimeIn = nn(datetimeIn);
if (datetimeIn.indexOf('T') >= 0) {
ConversionResult cr = DateTimeValue.makeDateTimeValue(datetimeIn, rules);
if (cr instanceof ValidationFailure) {
return "";
} else {
return ((DateTimeValue) cr).toDateValue().getStringValue();
}
} else {
ConversionResult cr = DateValue.makeDateValue(datetimeIn, rules);
if (cr instanceof ValidationFailure) {
return "";
} else {
return ((AtomicValue) cr).getStringValue();
}
}
}
/**
* The date:date function returns the current date.
*
* @param context the XPath dynamic context
* @return the current date as a string
* @throws XPathException if the context does not allow the date and time to be established
*/
public static String date(XPathContext context) throws XPathException {
return date(context, dateTime(context));
}
/**
* The date:time function returns the time specified in the date/time string given as the
* argument.
*
* @param context the XPath dynamic context
* @param dateTime must start with [+|-]CCYY-MM-DDThh:mm:ss
* @return the time part of the string, or "" if invalid
*/
public static String time(XPathContext context, String dateTime) {
dateTime = nn(dateTime);
if (dateTime.indexOf('T') >= 0) {
ConversionResult cr = DateTimeValue.makeDateTimeValue(dateTime, context.getConfiguration().getConversionRules());
if (cr instanceof ValidationFailure) {
return "";
} else {
return ((DateTimeValue) cr).toTimeValue().getStringValue();
}
} else {
ConversionResult cr = TimeValue.makeTimeValue(dateTime);
if (cr instanceof ValidationFailure) {
return "";
} else {
return ((AtomicValue) cr).getStringValue();
}
}
}
/**
* The date:time function returns the current time.
*
* @param context the XPath dynamic context
* @return the current time as a string
* @throws XPathException if the context does not allow a date/time to be established
*/
public static String time(XPathContext context) throws XPathException {
return time(context, dateTime(context));
}
/**
* The date:year function returns the year of a date as a number. If no
* argument is given, then the current local date/time, as returned by
* date:date-time is used as a default argument.
* The date/time string specified as the first argument must be a right-truncated
* string in the format defined as the lexical representation of xs:dateTime in one
* of the formats defined in
* [XML Schema Part 2: Datatypes].
* The permitted formats are as follows:
* xs:dateTime (CCYY-MM-DDThh:mm:ss)
* xs:date (CCYY-MM-DD)
* xs:gYearMonth (CCYY-MM)
* xs:gYear (CCYY)
* If the date/time string is not in one of these formats, then NaN is returned.
*
Note: although not specifically permitted in the EXSLT specification, the Saxon
* implementation also allows the input value to contain a timezone
*
* @param context the XPath dynamic context
* @param datetimeIn the supplied date/time in ISO format
* @return the year component of the supplied date time, or NaN if invalid
*/
public static double year(XPathContext context, String datetimeIn) {
datetimeIn = nn(datetimeIn);
try {
ConversionResult cr = CalendarValue.makeCalendarValue(datetimeIn, context.getConfiguration().getConversionRules());
if (cr instanceof ValidationFailure) {
return Double.NaN;
}
if (cr instanceof GMonthValue || cr instanceof GMonthDayValue ||
cr instanceof GDayValue || cr instanceof TimeValue) {
return Double.NaN;
}
AtomicValue year = ((CalendarValue) cr).getComponent(AccessorFn.Component.YEAR);
return ((NumericValue) year).getDoubleValue();
} catch (XPathException e) {
return Double.NaN;
}
}
/**
* The date:year function returns the current year.
*
* @param context the XPath dynamic context
* @return the current year as a double
* @throws XPathException if the context does not allow a date/time to be established
*/
public static double year(XPathContext context) throws XPathException {
return year(context, dateTime(context));
}
/**
* Return true if the year specified in the date/time string
* given as the argument is a leap year.
*
* @param context the XPath dynamic context
* @param dateTime a dateTime as a string
* @return true if the year is a leap year (false if not, or if input is invalid)
*/
public static boolean leapYear(XPathContext context, String dateTime) {
dateTime = nn(dateTime);
double year = year(context, dateTime);
if (Double.isNaN(year)) {
return false;
}
int y = (int) year;
return (y % 4 == 0) && !((y % 100 == 0) && !(y % 400 == 0));
}
/**
* Returns true if the current year is a leap year
*
* @param context the XPath dynamic context
* @return true if the current year is a leap year (false if not, or if input is invalid)
* @throws XPathException if the context does not allow a date/time to be established
*/
public static boolean leapYear(XPathContext context) throws XPathException {
return leapYear(context, dateTime(context));
}
/**
* Return the month number from a date.
* The date must start with either "CCYY-MM" or "--MM"
*
* @param context the XPath dynamic context
* @param dateTime a dateTime as a string
* @return the month extracted from the dateTime
*/
public static double monthInYear(XPathContext context, String dateTime) {
dateTime = nn(dateTime);
try {
ConversionResult cr = CalendarValue.makeCalendarValue(dateTime, context.getConfiguration().getConversionRules());
if (cr instanceof ValidationFailure) {
return Double.NaN;
}
if (cr instanceof GYearValue ||
cr instanceof GDayValue || cr instanceof TimeValue) {
return Double.NaN;
}
AtomicValue month = ((CalendarValue) cr).getComponent(AccessorFn.Component.MONTH);
return ((NumericValue) month).getDoubleValue();
} catch (XPathException e) {
return Double.NaN;
}
}
/**
* Return the month number from the current date.
*
* @param context the XPath dynamic context
* @return the current month number
* @throws XPathException if the context does not allow a date/time to be established
*/
public static double monthInYear(XPathContext context) throws XPathException {
return monthInYear(context, dateTime(context));
}
/**
* Return the month name from a date.
* The date must start with either "CCYY-MM" or "--MM"
*
* @param context the XPath dynamic context
* @param date the date/time as a string
* @return the English month name, for example "January", "February"
*/
public static String monthName(XPathContext context, String date) {
date = nn(date);
String[] months = {"January", "February", "March", "April", "May", "June",
"July", "August", "September", "October", "November", "December"};
double m = monthInYear(context, date);
if (Double.isNaN(m)) {
return "";
}
return months[(int) m - 1];
}
/**
* Return the month name from the current date.
*
* @param context the XPath dynamic context
* @return the English month name, for example "January", "February"
* @throws XPathException if the context does not allow a date/time to be established
*/
public static String monthName(XPathContext context) throws XPathException {
return monthName(context, dateTime(context));
}
/**
* Return the month abbreviation from a date.
*
* @param context the XPath dynamic context
* @param date The date must start with either "CCYY-MM" or "--MM"
* @return the English month abbreviation, for example "Jan", "Feb"
*/
public static String monthAbbreviation(XPathContext context, String date) {
date = nn(date);
String[] months = {"Jan", "Feb", "Mar", "Apr", "May", "Jun",
"Jul", "Aug", "Sep", "Oct", "Nov", "Dec"};
double m = monthInYear(context, date);
if (Double.isNaN(m)) {
return "";
}
return months[(int) m - 1];
}
/**
* Return the month abbreviation from the current date.
*
* @param context the XPath dynamic context
* @return the English month abbreviation, for example "Jan", "Feb"
* @throws XPathException if the context does not allow a date/time to be established
*/
public static String monthAbbreviation(XPathContext context) throws XPathException {
return monthAbbreviation(context, dateTime(context));
}
/**
* Return the ISO week number of a specified date within the year
* (Note, this returns the ISO week number: the result in EXSLT is underspecified)
*
* @param context the XPath dynamic context
* @param dateTime the current date starting CCYY-MM-DD
* @return the ISO week number
*/
public static double weekInYear(XPathContext context, String dateTime) {
dateTime = nn(dateTime);
int dayInYear = (int) dayInYear(context, dateTime);
String firstJan = dateTime.substring(0, 4) + "-01-01";
int jan1day = ((int) dayInWeek(context, firstJan) + 5) % 7;
int daysInFirstWeek = jan1day == 0 ? 0 : 7 - jan1day;
int rawWeek = (dayInYear - daysInFirstWeek + 6) / 7;
if (daysInFirstWeek >= 4) {
return rawWeek + 1;
} else {
if (rawWeek > 0) {
return rawWeek;
} else {
// week number should be 52 or 53: same as 31 Dec in previous year
int lastYear = Integer.parseInt(dateTime.substring(0, 4)) - 1;
String dec31 = lastYear + "-12-31";
// assumes year > 999
return weekInYear(context, dec31);
}
}
}
/**
* Return the ISO week number of the current date
* (Note, this returns the ISO week number: the result in EXSLT is underspecified)
*
* @param context the XPath dynamic context
* @return the ISO week number
* @throws XPathException if the context does not allow a date/time to be established
*/
public static double weekInYear(XPathContext context) throws XPathException {
return weekInYear(context, dateTime(context));
}
/**
* Return the week number of a specified date within the month
* (Note, this function is underspecified in EXSLT)
*
* @param context the XPath dynamic context
* @param dateTime the date starting CCYY-MM-DD
* @return the week number within the month
*/
public static double weekInMonth(XPathContext context, String dateTime) {
dateTime = nn(dateTime);
return (double) (int) ((dayInMonth(context, dateTime) - 1) / 7 + 1);
}
/**
* Return the ISO week number of the current date within the month
*
* @param context the XPath dynamic context
* @return the week number within the month
* @throws XPathException if the context does not allow a date/time to be established
*/
public static double weekInMonth(XPathContext context) throws XPathException {
return weekInMonth(context, dateTime(context));
}
/**
* Return the day number of a specified date within the year
*
* @param context the XPath dynamic context
* @param dateTime the date starting with CCYY-MM-DD
* @return the day number within the year, as a double
*/
public static double dayInYear(XPathContext context, String dateTime) {
dateTime = nn(dateTime);
int month = (int) monthInYear(context, dateTime);
int day = (int) dayInMonth(context, dateTime);
int[] prev = {0,
31,
31 + 28,
31 + 28 + 31,
31 + 28 + 31 + 30,
31 + 28 + 31 + 30 + 31,
31 + 28 + 31 + 30 + 31 + 30,
31 + 28 + 31 + 30 + 31 + 30 + 31,
31 + 28 + 31 + 30 + 31 + 30 + 31 + 31,
31 + 28 + 31 + 30 + 31 + 30 + 31 + 31 + 30,
31 + 28 + 31 + 30 + 31 + 30 + 31 + 31 + 30 + 31,
31 + 28 + 31 + 30 + 31 + 30 + 31 + 31 + 30 + 31 + 30,
31 + 28 + 31 + 30 + 31 + 30 + 31 + 31 + 30 + 31 + 30 + 31};
int leap = month > 2 && leapYear(context, dateTime) ? 1 : 0;
return prev[month - 1] + leap + day;
}
/**
* Return the day number of the current date within the year
*
* @param context the XPath dynamic context
* @return the day number within the year, as a double
* @throws XPathException if the context does not allow a date/time to be established
*/
public static double dayInYear(XPathContext context) throws XPathException {
return dayInYear(context, dateTime(context));
}
/**
* Return the day number of a specified date within the month
*
* @param context the XPath dynamic context
* @param dateTime must start with CCYY-MM-DD, or --MM-DD, or ---DD
* @return the day number within the month, as a double
*/
public static double dayInMonth(XPathContext context, String dateTime) {
dateTime = nn(dateTime);
try {
ConversionResult cr = CalendarValue.makeCalendarValue(dateTime, context.getConfiguration().getConversionRules());
if (cr instanceof ValidationFailure) {
return Double.NaN;
}
if (cr instanceof GYearValue || cr instanceof GYearMonthValue ||
cr instanceof GMonthValue || cr instanceof TimeValue) {
return Double.NaN;
}
AtomicValue day = ((CalendarValue) cr).getComponent(AccessorFn.Component.DAY);
return ((NumericValue) day).getDoubleValue();
} catch (XPathException e) {
return Double.NaN;
}
}
/**
* Return the day number of the current date within the month
*
* @param context the XPath dynamic context
* @return the current day number, as a double
* @throws XPathException if the context does not allow a date/time to be established
*/
public static double dayInMonth(XPathContext context) throws XPathException {
return dayInMonth(context, dateTime(context));
}
/**
* Return the day-of-the-week in a month of a date as a number
* (for example 3 for the 3rd Tuesday in May).
*
* @param context the XPath dynamic context
* @param dateTime must start with CCYY-MM-DD
* @return the the day-of-the-week in a month of a date as a number
* (for example 3 for the 3rd Tuesday in May).
*/
public static double dayOfWeekInMonth(XPathContext context, String dateTime) {
dateTime = nn(dateTime);
double dd = dayInMonth(context, dateTime);
if (Double.isNaN(dd)) {
return dd;
}
return ((int) dd - 1) / 7 + 1;
}
/**
* Return the day-of-the-week in a month of the current date as a number
* (for example 3 for the 3rd Tuesday in May).
*
* @param context the XPath dynamic context
* @return the the day-of-the-week in a month of the current date as a number
* (for example 3 for the 3rd Tuesday in May).
* @throws XPathException if the context does not allow a date/time to be established
*/
public static double dayOfWeekInMonth(XPathContext context) throws XPathException {
return dayOfWeekInMonth(context, dateTime(context));
}
/**
* Return the day of the week given in a date as a number.
* The numbering of days of the week starts at 1 for Sunday, 2 for Monday
* and so on up to 7 for Saturday.
*
* @param context the XPath dynamic context
* @param dateTime must start with CCYY-MM-DD
* @return the day of the week as a number
*/
public static double dayInWeek(XPathContext context, String dateTime) {
dateTime = nn(dateTime);
double yy = year(context, dateTime);
double mm = monthInYear(context, dateTime);
double dd = dayInMonth(context, dateTime);
if (Double.isNaN(yy) || Double.isNaN(mm) || Double.isNaN(dd)) {
return Double.NaN;
}
GregorianCalendar calDate =
new GregorianCalendar(
(int) yy,
(int) mm - 1,
(int) dd);
calDate.setFirstDayOfWeek(Calendar.SUNDAY);
return calDate.get(Calendar.DAY_OF_WEEK);
}
/**
* Return the day of the week in the current date as a number.
* The numbering of days of the week starts at 1 for Sunday, 2 for Monday
* and so on up to 7 for Saturday.
*
* @param context the XPath dynamic context
* @return the day of the week as a number
* @throws XPathException if the context does not allow a date/time to be established
*/
public static double dayInWeek(XPathContext context) throws XPathException {
return dayInWeek(context, dateTime(context));
}
/**
* Return the day of the week given in a date as an English day name:
* one of 'Sunday', 'Monday', 'Tuesday', 'Wednesday', 'Thursday' or 'Friday'.
*
* @param context the XPath dynamic context
* @param dateTime must start with CCYY-MM-DD
* @return the English name of the day of the week
*/
public static String dayName(XPathContext context, String dateTime) {
dateTime = nn(dateTime);
String[] days = {"Sunday", "Monday", "Tuesday", "Wednesday", "Thursday", "Friday",
"Saturday"};
double d = dayInWeek(context, dateTime);
if (Double.isNaN(d)) {
return "";
}
return days[(int) d - 1];
}
/**
* Return the day of the week given in the current date as an English day name:
* one of 'Sunday', 'Monday', 'Tuesday', 'Wednesday', 'Thursday' or 'Friday'.
*
* @param context the XPath dynamic context
* @return the English name of the day of the week
* @throws XPathException if the context does not allow a date/time to be established
*/
public static String dayName(XPathContext context) throws XPathException {
return dayName(context, dateTime(context));
}
/**
* Return the day of the week given in a date as an English day abbreviation:
* one of 'Sun', 'Mon', 'Tue', 'Wed', 'Thu', 'Fri', or 'Sat'.
*
* @param context the XPath dynamic context
* @param dateTime must start with CCYY-MM-DD
* @return the English day abbreviation
*/
public static String dayAbbreviation(XPathContext context, String dateTime) {
dateTime = nn(dateTime);
String[] days = {"Sun", "Mon", "Tue", "Wed", "Thu", "Fri", "Sat"};
double d = dayInWeek(context, dateTime);
if (Double.isNaN(d)) {
return "";
}
return days[(int) d - 1];
}
/**
* Return the day of the week given in the current date as an English day abbreviation:
* one of 'Sun', 'Mon', 'Tue', 'Wed', 'Thu', 'Fri', or 'Sat'.
*
* @param context the XPath dynamic context
* @return the English day abbreviation
* @throws XPathException if the context does not allow a date/time to be established
*/
public static String dayAbbreviation(XPathContext context) throws XPathException {
return dayAbbreviation(context, dateTime(context));
}
/**
* Return the hour of the day in the specified date or date/time
*
* @param context the XPath dynamic context
* @param dateTime must start with CCYY-MM-DDThh:mm:ss or hh:mm:ss
* @return the hour
*/
public static double hourInDay(XPathContext context, String dateTime) {
dateTime = nn(dateTime);
try {
ConversionResult cr = CalendarValue.makeCalendarValue(dateTime, context.getConfiguration().getConversionRules());
if (cr instanceof ValidationFailure) {
return Double.NaN;
}
if (!(cr instanceof DateTimeValue || cr instanceof TimeValue)) {
return Double.NaN;
}
AtomicValue hour = ((CalendarValue) cr).getComponent(AccessorFn.Component.HOURS);
return ((NumericValue) hour).getDoubleValue();
} catch (XPathException e) {
return Double.NaN;
}
}
/**
* Return the current hour of the day
*
* @param context the XPath dynamic context
* @return the hour
* @throws XPathException if the context does not allow a date/time to be established
*/
public static double hourInDay(XPathContext context) throws XPathException {
return hourInDay(context, dateTime(context));
}
/**
* Return the minute of the hour in the specified date or date/time
*
* @param context the XPath dynamic context
* @param dateTime must start with CCYY-MM-DDThh:mm:ss or hh:mm:ss
* @return the minute
*/
public static double minuteInHour(XPathContext context, String dateTime) {
dateTime = nn(dateTime);
try {
ConversionResult cr = CalendarValue.makeCalendarValue(dateTime, context.getConfiguration().getConversionRules());
if (cr instanceof ValidationFailure) {
return Double.NaN;
}
if (!(cr instanceof DateTimeValue || cr instanceof TimeValue)) {
return Double.NaN;
}
AtomicValue minute = ((CalendarValue) cr).getComponent(AccessorFn.Component.MINUTES);
return ((NumericValue) minute).getDoubleValue();
} catch (XPathException e) {
return Double.NaN;
}
}
/**
* Return the current minute of the hour
*
* @param context the XPath dynamic context
* @return the minute
* @throws XPathException if the context does not allow a date/time to be established
*/
public static double minuteInHour(XPathContext context) throws XPathException {
return minuteInHour(context, dateTime(context));
}
/**
* Return the second of the minute in the specified date or date/time
*
* @param context the XPath dynamic context
* @param dateTime must start with CCYY-MM-DDThh:mm:ss or hh:mm:ss
* @return the second
*/
public static double secondInMinute(XPathContext context, String dateTime) {
dateTime = nn(dateTime);
try {
ConversionResult cr = CalendarValue.makeCalendarValue(dateTime, context.getConfiguration().getConversionRules());
if (cr instanceof ValidationFailure) {
return Double.NaN;
}
if (!(cr instanceof DateTimeValue || cr instanceof TimeValue)) {
return Double.NaN;
}
AtomicValue second = ((CalendarValue) cr).getComponent(AccessorFn.Component.SECONDS);
return ((NumericValue) second).getDoubleValue();
} catch (XPathException e) {
return Double.NaN;
}
}
/**
* Return the current second of the minute
*
* @param context the XPath dynamic context
* @return the second
* @throws XPathException if the context does not allow a date/time to be established
*/
public static double secondInMinute(XPathContext context) throws XPathException {
return secondInMinute(context, dateTime(context));
}
/**
* The date:add function returns the date/time resulting from adding a duration to a date/time.
* The first argument must be right-truncated date/time strings in one of the formats defined in
* [XML Schema Part 2: Datatypes].
* The permitted formats are as follows:
* xs:dateTime (CCYY-MM-DDThh:mm:ss)
* xs:date (CCYY-MM-DD)
* xs:gYearMonth (CCYY-MM)
* xs:gYear (CCYY)
* The second argument is a string in the format defined for xs:duration in [3.2.6 duration] of
* [XML Schema Part 2: Datatypes].
* The return value is a right-truncated date/time strings in one of the formats defined in
* [XML Schema Part 2: Datatypes] and listed above.
* This value is calculated using the algorithm described in
* [Appendix E Adding durations to dateTimes] of [XML Schema Part 2: Datatypes].
*
* @param context the XPath dynamic context
* @param datetimeIn the supplied date/time in ISO date/time format
* @param durationIn the supplied duration in ISO duration format
* @return the result of the addition, as a string representing a dateTimeValue
* @throws XPathException if the supplied date or duration is invalid
*/
public static String add(XPathContext context, String datetimeIn, String durationIn) throws XPathException {
datetimeIn = nn(datetimeIn);
durationIn = nn(durationIn);
ConversionResult cr0 = CalendarValue.makeCalendarValue(datetimeIn, context.getConfiguration().getConversionRules());
if (cr0 instanceof ValidationFailure) {
return "";
}
CalendarValue cv0 = (CalendarValue) cr0;
if (specificity(cv0) < 0) {
return "";
}
DateTimeValue v0 = cv0.toDateTime();
ConversionResult cr1 = DurationValue.makeDuration(durationIn);
if (cr1 instanceof ValidationFailure) {
return "";
}
DurationValue v1 = (DurationValue) cr1;
YearMonthDurationValue v1m =
(YearMonthDurationValue) Converter.DURATION_TO_YEAR_MONTH_DURATION.convert(v1);
DayTimeDurationValue v1s =
(DayTimeDurationValue) Converter.DURATION_TO_DAY_TIME_DURATION.convert(v1);
DateTimeValue sum = v0.add(v1m).add(v1s);
return Converter.convert(sum, cv0.getPrimitiveType(), context.getConfiguration().getConversionRules()).getStringValue();
}
/**
* The date:sum function adds a set of durations together.
* The string values of the nodes in the node set passed as an argument
* are interpreted as durations and added together as if using the date:add-duration
* function. The string values of the nodes in the node set passed as the argument
* must be in the format defined for xs:duration in [3.2.6 duration] of
* [XML Schema Part 2: Datatypes]. If any of the string values of these nodes
* are not in this format, or if the node set is empty, the function returns an
* empty string (''). The result is a string in the format defined for xs:duration
* in [3.2.6 duration] of [XML Schema Part 2: Datatypes].
* The durations can be summed by summing the numbers given for each of the components in the durations.
*
*
* @param durations a sequence which when atomized and converted to strings using the string()
* function yields a sequence of strings representing the input durations in ISO format
* @return a string representation of the total duration, or "" if any input is invalid or if the
* input is an empty sequence
* @throws XPathException if the input sequence cannot be evaluated
*/
public static String sum(SequenceIterator durations) throws XPathException {
DurationValue tot = (DurationValue) DurationValue.makeDuration("PT0S");
while (true) {
Item it = durations.next();
if (it == null) {
break;
}
ConversionResult cr = DurationValue.makeDuration(it.getStringValueCS());
if (cr instanceof ValidationFailure) {
return "";
}
tot = addDurationValues(tot, (DurationValue) cr);
if (tot == null) {
return "";
}
}
return tot.getStringValue();
}
/**
* The date:add-duration function returns the duration resulting from adding two durations together.
* Both arguments are strings in the format defined for xs:duration in [3.2.6 duration] of
* [XML Schema Part 2: Datatypes]. If either argument is not in this format, the function returns
* an empty string (''). The return value is a string in the format defined for xs:duration in
* [3.2.6 duration] of [XML Schema Part 2: Datatypes]. The durations can usually be added by
* summing the numbers given for each of the components in the durations. However, if the durations
* are differently signed, then this sometimes results in durations that are impossible to express
* in this syntax (e.g. 'P1M' + '-P1D'). In these cases, the function returns an empty string ('').
*
*
* @param duration0 the first operand, a string in the format of an ISO duration
* @param duration1 the second operand, a string in the format of an ISO duration
* @return a string representation of the total duration, or "" if the input is invalid or the
* total is inexpressible
*/
public String addDuration(String duration0, String duration1) {
duration0 = nn(duration0);
duration1 = nn(duration1);
ConversionResult dv0 = DurationValue.makeDuration(duration0);
ConversionResult dv1 = DurationValue.makeDuration(duration1);
if (dv0 instanceof ValidationFailure || dv1 instanceof ValidationFailure) {
return "";
}
DurationValue result = addDurationValues((DurationValue) dv0, (DurationValue) dv1);
return result == null ? "" : result.getStringValue();
}
/**
* Add two duration values, according to the rules for the add-duration() and sum() functions
*
*
* @param dv0 the first duration
* @param dv1 the second duration
* @return the sum of the two durations, or null if they are not summable (e.g. P1Y + -P1D)
*/
/*@Nullable*/
private static DurationValue addDurationValues(DurationValue dv0, DurationValue dv1) {
YearMonthDurationValue dv0m =
(YearMonthDurationValue) Converter.DURATION_TO_YEAR_MONTH_DURATION.convert(dv0);
DayTimeDurationValue dv0s =
(DayTimeDurationValue) Converter.DURATION_TO_DAY_TIME_DURATION.convert(dv0);
YearMonthDurationValue dv1m = (YearMonthDurationValue) Converter.DURATION_TO_YEAR_MONTH_DURATION.convert(dv1);
DayTimeDurationValue dv1s =
(DayTimeDurationValue) Converter.DURATION_TO_DAY_TIME_DURATION.convert(dv1);
int months = dv0m.getLengthInMonths() + dv1m.getLengthInMonths();
long micros = dv0s.getLengthInMicroseconds() + dv1s.getLengthInMicroseconds();
if (Integer.signum(months) * Long.signum(micros) < 0) {
return null;
}
boolean positive = months >= 0 && micros >= 0;
if (!positive) {
months = -months;
micros = -micros;
}
return new DurationValue(positive, 0, months,
0, 0, 0, (int) (micros / 1000000), (int) (micros % 1000000), BuiltInAtomicType.DURATION);
}
/**
* The date:difference function returns the duration between the first date and the second date.
* If the first date occurs before the second date, then the result is a positive duration;
* if it occurs after the second date, the result is a negative duration.
* The two dates must both be right-truncated date/time strings in one of the formats defined in
* [XML Schema Part 2: Datatypes].
* The date/time with the most specific format (i.e. the least truncation) is converted into the
* same format as the date with the most specific format (i.e. the most truncation).
* The permitted formats are as follows, from most specific to least specific:
* xs:dateTime (CCYY-MM-DDThh:mm:ss)
* xs:date (CCYY-MM-DD)
* xs:gYearMonth (CCYY-MM)
* xs:gYear (CCYY)
*
* If either of the arguments is not in one of these formats, date:difference
* returns the empty string ('').
* The difference between the date/times is returned as a string in the format defined for
* xs:duration in [3.2.6 duration] of [XML Schema Part 2: Datatypes].
* If the date/time string with the least specific format is in either xs:gYearMonth or xs:gYear format,
* then the number of days, hours, minutes and seconds in the duration string must be equal to zero.
* (The format of the string will be PnYnM.)
* The number of months specified in the duration must be less than 12.
* Otherwise, the number of years and months in the duration string must be equal to zero.
* (The format of the string will be PnDTnHnMnS.)
* The number of seconds specified in the duration string must be less than 60;
* the number of minutes must be less than 60; the number of hours must be less than 24.
*
* @param context the XPath dynamic context
* @param dateLeftIn the first operand, a string in the format of an ISO dateTime, date, yearMonth, or year
* @param dateRightIn the second operand, a string in the format of an ISO dateTime, date, yearMonth, or year
* @return a string representation of the difference as an ISO duration, or "" if the input is invalid or the
* difference is inexpressible
*/
public static String difference(XPathContext context, String dateLeftIn, String dateRightIn) {
try {
dateLeftIn = nn(dateLeftIn);
dateRightIn = nn(dateRightIn);
final ConversionRules rules = context.getConfiguration().getConversionRules();
ConversionResult op0 = CalendarValue.makeCalendarValue(dateLeftIn, rules);
ConversionResult op1 = CalendarValue.makeCalendarValue(dateRightIn, rules);
if (op0 instanceof ValidationFailure || op1 instanceof ValidationFailure) {
return "";
}
CalendarValue v0 = (CalendarValue) op0;
CalendarValue v1 = (CalendarValue) op1;
int s0 = specificity(v0);
int s1 = specificity(v1);
if (s0 < 0 || s1 < 0) {
return "";
}
if (s0 < s1) {
v1 = (CalendarValue) Converter.convert(v1, v0.getPrimitiveType(), rules);
} else if (s1 < s0) {
v0 = (CalendarValue) Converter.convert(v0, v1.getPrimitiveType(), rules);
}
if (v0 instanceof GYearValue) {
int y0 = ((GYearValue) v0).getYear();
int y1 = ((GYearValue) v1).getYear();
return YearMonthDurationValue.fromMonths(12 * (y1 - y0)).getStringValue();
} else if (v0 instanceof GYearMonthValue) {
int y0 = ((GYearMonthValue) v0).getYear();
int y1 = ((GYearMonthValue) v1).getYear();
int m0 = ((GYearMonthValue) v0).getMonth();
int m1 = ((GYearMonthValue) v1).getMonth();
return YearMonthDurationValue.fromMonths(12 * (y1 - y0) + (m1 - m0)).getStringValue();
} else {
DateTimeValue dt0 = v0.toDateTime();
DateTimeValue dt1 = v1.toDateTime();
return dt1.subtract(dt0, context).getStringValue();
}
} catch (XPathException e) {
return "";
}
}
/**
* Rank the right-truncated calendar types in order of specificity
*
* @param val a value of a given calendar type
* @return the specificity of the type: 0 for least specific, 4 for most specific, -1 if this
* is not a right-truncated type
*/
private static int specificity(CalendarValue val) {
if (val instanceof GYearValue) {
return 0;
} else if (val instanceof GYearMonthValue) {
return 1;
} else if (val instanceof DateValue) {
return 2;
} else if (val instanceof DateTimeValue) {
return 3;
} else {
return -1;
}
}
/**
* The date:duration function returns a duration string representing the number of seconds
* specified by the argument string. If no argument is given, then the result of calling
* date:seconds without any arguments is used as a default argument. The duration is returned
* as a string in the format defined for xs:duration in [3.2.6 duration] of
* [XML Schema Part 2: Datatypes]. The number of years and months in the duration string
* must be equal to zero. (The format of the string will be PnDTnHnMnS.)
* The number of seconds specified in the duration string must be less than 60;
* the number of minutes must be less than 60; the number of hours must be less than 24.
* If the argument is Infinity, -Infinity or NaN, then date:duration returns an empty string ('').
*
* @param seconds the input number of seconds
* @return the duration as a string in ISO format
*/
public static String duration(double seconds) {
DayTimeDurationValue v = DayTimeDurationValue.fromSeconds(new BigDecimal(seconds));
return v.getStringValue();
}
/**
* Return the number of seconds since 1 Jan 1970
*
* @param context the XPath dynamic context
* @return the number of seconds since 1 Jan 1970 (the "epoch" according to Java and Unix)
* @throws XPathException if the context does not allow a date/time to be established
*/
public static double seconds(XPathContext context) throws XPathException {
DateTimeValue now = context.getCurrentDateTime();
DurationValue diff = now.subtract(DateTimeValue.EPOCH, context);
return diff.getLengthInSeconds();
}
/**
* The date:seconds function returns the number of seconds specified by the argument string.
* If no argument is given, then the current local date/time, as returned by date:date-time
* is used as a default argument.
* The argument string may be in one of the following formats:
* 1. A right-truncated date/time string in one of the formats defined in
* [XML Schema Part 2: Datatypes].
* In these cases, the difference between the date/time string and 1970-01-01T00:00:00Z
* is calculated as with date:difference and the result is converted to seconds with
* date:seconds.
* The legal formats are as follows:
* xs:dateTime (CCYY-MM-DDThh:mm:ss)
* xs:date (CCYY-MM-DD)
* xs:gYearMonth (CCYY-MM)
* xs:gYear (CCYY)
* 2. A duration specified in days, hours, minutes and seconds in the format defined for
* xs:duration in
* [3.2.6 duration]
* of
* [XML Schema Part 2: Datatypes].
* The number of years and months in the duration string must both be equal to zero:
* either P0Y0M120D or P120D are permitted, but P3M is not.
* If the argument to date:seconds is defined as a duration, the number returned is the
* result of converting the duration to seconds by assuming that
* 1 day = 24 hours, 1 hour = 60 minutes and 1 minute = 60 seconds.
* The permitted duration format is basically PnDTnHnMnS, although implementers should
* consult
* [XML Schema Part 2: Datatypes]
* and
* [ISO 8601]
* for details.
* If the argument is not in any of these formats, date:seconds returns NaN.
*
* @param context the XPath dynamic context
* @param datetimeIn the input dateTime, date, yearMonth, or year as an ISO string
* @return the number of seconds since 1 Jan 1970 (the "epoch" according to Java and Unix)
*/
public static double seconds(XPathContext context, String datetimeIn) {
try {
datetimeIn = nn(datetimeIn);
ConversionResult cr = CalendarValue.makeCalendarValue(datetimeIn, context.getConfiguration().getConversionRules());
if (cr instanceof DateTimeValue || cr instanceof DateValue ||
cr instanceof GYearValue || cr instanceof GYearMonthValue) {
DateTimeValue dateTime = ((CalendarValue) cr).toDateTime();
DayTimeDurationValue diff = dateTime.subtract(DateTimeValue.EPOCH, context);
return diff.getLengthInSeconds();
}
cr = DurationValue.makeDuration(datetimeIn);
if (cr instanceof DurationValue) {
DurationValue duration = (DurationValue) cr;
if (duration.getYears() != 0 || duration.getMonths() != 0) {
return Double.NaN;
} else {
return duration.getLengthInSeconds();
}
} else {
return Double.NaN;
}
} catch (XPathException e) {
return Double.NaN;
}
}
/**
* Treat null as zero-length string
*
* @param in input string
* @return zero-length string if input is null, else the input string unchanged.
*/
private static String nn(String in) {
return in == null ? "" : in;
}
}