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

net.sf.saxon.value.SaxonDuration 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.value;

import net.sf.saxon.functions.AccessorFn;
import net.sf.saxon.lib.NamespaceConstant;
import net.sf.saxon.trans.XPathException;

import javax.xml.datatype.DatatypeConstants;
import javax.xml.datatype.Duration;
import javax.xml.namespace.QName;
import java.math.BigDecimal;
import java.math.BigInteger;
import java.util.Calendar;

/**
 * Saxon implementation of the JAXP class javax.xml.datatype.Duration. This is currently used only by the XQJ
 * interface for XQuery: the normal representation of a duration in Saxon is the class {@link DurationValue}.
 * 

* The JAXP specification for this class defines it in terms of XML Schema 1.0 semantics. This defines a structure * with six independent components (year, month, day, hour, minute, second). This implementation is more aligned * to the XPath 2.0 semantics of the data type, which essentially defines duration as an integer number of months plus * a decimal number of seconds. */ public class SaxonDuration extends Duration { private DurationValue duration; /** * Create a SaxonDuration that wraps a supplied DurationValue * * @param duration the value to be wrapped. */ public SaxonDuration(DurationValue duration) { this.duration = duration; } /** * Get the underlying DurationValue * * @return the underlying DurationValue */ public DurationValue getDurationValue() { return duration; } /** * Get the type of this duration, as one of the values xs:duration, xs:dayTimeDuration, or * xs:yearMonthDuration. (Note that the XML Schema namespace URI is used, whereas the current * implementation of the superclass uses a provisional URI allocated in a 2003 W3C working draft) * * @return the type of this duration, as one of the values xs:duration, xs:dayTimeDuration, or * xs:yearMonthDuration */ public QName getXMLSchemaType() { if (duration instanceof DayTimeDurationValue) { return new QName(NamespaceConstant.SCHEMA, "dayTimeDuration"); } else if (duration instanceof YearMonthDurationValue) { return new QName(NamespaceConstant.SCHEMA, "yearMonthDuration"); } else { return new QName(NamespaceConstant.SCHEMA, "duration"); } } /** * Returns the sign of this duration in -1,0, or 1. * * @return -1 if this duration is negative, 0 if the duration is zero, * and 1 if the duration is positive. */ public int getSign() { return duration.signum(); } /** * Gets the value of a field. *

* Fields of a duration object may contain arbitrary large value. * Therefore this method is designed to return a {@link Number} object. *

* In case of YEARS, MONTHS, DAYS, HOURS, and MINUTES, the returned * number will be a non-negative integer. In case of seconds, * the returned number may be a non-negative decimal value. *

* The Saxon implementation of duration uses normalized values. This means * that the YEARS and DAYS fields may be arbitrarily large, but other * components will be limited in size: for example MINUTES will never * exceed 60 and MONTHS will never exceed 12. * * @param field one of the six Field constants (YEARS, MONTHS, DAYS, HOURS, * MINUTES, or SECONDS.) * @return If the specified field is present, this method returns * a non-null non-negative {@link Number} object that * represents its value. If it is not present, return null. * For YEARS, MONTHS, DAYS, HOURS, and MINUTES, this method * returns a {@link java.math.BigInteger} object. For SECONDS, this * method returns a {@link java.math.BigDecimal}. * @throws NullPointerException If the field is null. */ public Number getField(DatatypeConstants.Field field) { try { if (field == DatatypeConstants.YEARS) { return BigInteger.valueOf(((Int64Value) duration.getComponent(AccessorFn.Component.YEAR)).longValue()); } else if (field == DatatypeConstants.MONTHS) { return BigInteger.valueOf(((Int64Value) duration.getComponent(AccessorFn.Component.MONTH)).longValue()); } else if (field == DatatypeConstants.DAYS) { return BigInteger.valueOf(((Int64Value) duration.getComponent(AccessorFn.Component.DAY)).longValue()); } else if (field == DatatypeConstants.HOURS) { return BigInteger.valueOf(((Int64Value) duration.getComponent(AccessorFn.Component.HOURS)).longValue()); } else if (field == DatatypeConstants.MINUTES) { return BigInteger.valueOf(((Int64Value) duration.getComponent(AccessorFn.Component.MINUTES)).longValue()); } else if (field == DatatypeConstants.SECONDS) { return (((DecimalValue) duration.getComponent(AccessorFn.Component.SECONDS)).getDecimalValue()); } else { throw new IllegalArgumentException("Invalid field"); } } catch (XPathException e) { throw new AssertionError("Component extraction on duration failed"); } } /** * Checks if a field is set. In this implementation, all fields are always set. * * @param field one of the six Field constants (YEARS, MONTHS, DAYS, HOURS, * MINUTES, or SECONDS.) * @return This implementation always returns true. */ public boolean isSet(DatatypeConstants.Field field) { return true; } /** *

Computes a new duration whose value is this+rhs.

*

This implementation follows the XPath semantics. This means that the operation will fail * if the duration is not a yearMonthDuration or a dayTimeDuration. * * @param rhs Duration to add to this Duration * @return non-null valid Duration object. * @throws NullPointerException If the rhs parameter is null. * @throws IllegalStateException If the durations are not both dayTimeDurations, or * both yearMonthDurations. * @see #subtract(javax.xml.datatype.Duration) */ public Duration add(Duration rhs) { try { return new SaxonDuration(duration.add(((SaxonDuration) rhs).duration)); } catch (XPathException e) { throw new IllegalStateException(e.getMessage()); } } /** *

Computes a new duration whose value is this-rhs.

*

This implementation follows the XPath semantics. This means that the operation will fail * if the duration is not a yearMonthDuration or a dayTimeDuration. * * @param rhs Duration to subtract from this Duration * @return non-null valid Duration object. * @throws NullPointerException If the rhs parameter is null. * @throws IllegalStateException If the durations are not both dayTimeDurations, or * both yearMonthDurations. * @see #add(javax.xml.datatype.Duration) */ public Duration subtract(Duration rhs) { try { return new SaxonDuration(duration.subtract(((SaxonDuration) rhs).duration)); } catch (XPathException e) { throw new IllegalStateException(e.getMessage()); } } /** * Adds this duration to a {@link java.util.Calendar} object. *

*

* Calls {@link java.util.Calendar#add(int, int)} in the * order of YEARS, MONTHS, DAYS, HOURS, MINUTES, SECONDS, and MILLISECONDS * if those fields are present. Because the {@link java.util.Calendar} class * uses int to hold values, there are cases where this method * won't work correctly (for example if values of fields * exceed the range of int.) *

*

*

* Also, since this duration class is a Gregorian duration, this * method will not work correctly if the given {@link java.util.Calendar} * object is based on some other calendar systems. *

*

*

* Any fractional parts of this Duration object * beyond milliseconds will be simply ignored. For example, if * this duration is "P1.23456S", then 1 is added to SECONDS, * 234 is added to MILLISECONDS, and the rest will be unused. *

*

*

* Note that because {@link java.util.Calendar#add(int, int)} is using * int, Duration with values beyond the * range of int in its fields * will cause overflow/underflow to the given {@link java.util.Calendar}. * {@link javax.xml.datatype.XMLGregorianCalendar#add(javax.xml.datatype.Duration)} provides the same * basic operation as this method while avoiding * the overflow/underflow issues. * * @param calendar A calendar object whose value will be modified. * @throws NullPointerException if the calendar parameter is null. */ public void addTo(Calendar calendar) { int sign = getSign(); if (sign == 0) { return; } try { calendar.add(getYears() * sign, Calendar.YEAR); calendar.add(getMonths() * sign, Calendar.MONTH); calendar.add(getDays() * sign, Calendar.DAY_OF_MONTH); calendar.add(getHours() * sign, Calendar.HOUR_OF_DAY); calendar.add(getMinutes() * sign, Calendar.MINUTE); calendar.add((int) ((Int64Value) duration.getComponent(AccessorFn.Component.WHOLE_SECONDS)).longValue() * sign, Calendar.SECOND); calendar.add((int) ((Int64Value) duration.getComponent(AccessorFn.Component.MICROSECONDS)).longValue() * sign / 1000, Calendar.MILLISECOND); } catch (XPathException e) { throw new IllegalStateException(e.getMessage()); } } /** * Computes a new duration whose value is factor times * longer than the value of this duration. *

* This implementation follows the XPath semantics. This means that it is defined * only on yearMonthDuration and dayTimeDuration. Other cases produce an IllegalStateException. * * @param factor to multiply by * @return returns a non-null valid Duration object * @throws IllegalStateException if operation produces fraction in * the months field. * @throws NullPointerException if the factor parameter is * null. */ public Duration multiply(BigDecimal factor) { try { return new SaxonDuration(duration.multiply(factor.doubleValue())); } catch (XPathException e) { throw new IllegalStateException(e.getMessage()); } } /** * Returns a new Duration object whose * value is -this. *

*

* Since the Duration class is immutable, this method * doesn't change the value of this object. It simply computes * a new Duration object and returns it. * * @return always return a non-null valid Duration object. */ public Duration negate() { return new SaxonDuration(duration.negate()); } /** *

Converts the years and months fields into the days field * by using a specific time instant as the reference point.

*

* This implementation does not support this method * * @param startTimeInstant Calendar reference point. * @return Duration of years and months of this Duration as days. * @throws NullPointerException If the startTimeInstant parameter is null. * @throws UnsupportedOperationException Always thrown by this implementation. */ public Duration normalizeWith(Calendar startTimeInstant) { throw new UnsupportedOperationException(); } /** *

Partial order relation comparison with this Duration instance.

*

This implementation follows the XPath semantics. This means that the result is defined only * for dayTimeDuration and yearMonthDuration values, and the result is never indeterminate. *

*

Return:

*
    *
  • {@link javax.xml.datatype.DatatypeConstants#LESSER} * if this Duration is shorter than duration parameter
  • *
  • {@link javax.xml.datatype.DatatypeConstants#EQUAL} * if this Duration is equal to duration parameter
  • *
  • {@link javax.xml.datatype.DatatypeConstants#GREATER} * if this Duration is longer than duration parameter
  • *
  • {@link javax.xml.datatype.DatatypeConstants#INDETERMINATE} * if a conclusive partial order relation cannot be determined
  • *
* * @param rhs duration to compare * @return the relationship between this Durationand duration parameter as * {@link javax.xml.datatype.DatatypeConstants#LESSER}, * {@link javax.xml.datatype.DatatypeConstants#EQUAL}, * {@link javax.xml.datatype.DatatypeConstants#GREATER} * or {@link javax.xml.datatype.DatatypeConstants#INDETERMINATE}. * @throws UnsupportedOperationException If the underlying implementation * cannot reasonably process the request, e.g. W3C XML Schema allows for * arbitrarily large/small/precise values, the request may be beyond the * implementations capability. * @throws NullPointerException if duration is null. * @throws IllegalArgumentException if the operands are not dayTimeDuration or yearMonthDuration values. * @see #isShorterThan(javax.xml.datatype.Duration) * @see #isLongerThan(javax.xml.datatype.Duration) */ public int compare(/*@NotNull*/ Duration rhs) { if (!(rhs instanceof SaxonDuration)) { throw new IllegalArgumentException("Supplied duration is not a SaxonDuration"); } Comparable c0 = duration.getSchemaComparable(); Comparable c1 = ((SaxonDuration) rhs).duration.getSchemaComparable(); return c0.compareTo(c1); } /** * Returns a hash code consistent with the definition of the equals method. * * @see Object#hashCode() */ public int hashCode() { return duration.hashCode(); } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy