/* * eXist Open Source Native XML Database * Copyright (C) 2001-04 The eXist Project * http://exist-db.org * * This program is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public License * as published by the Free Software Foundation; either version 2 * of the License, or (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. * * $Id$ */ package org.exist.xquery.value; import java.text.Collator; import java.util.GregorianCalendar; import javax.xml.datatype.DatatypeConstants; import javax.xml.datatype.XMLGregorianCalendar; import javax.xml.namespace.QName; import org.exist.xquery.Constants; import org.exist.xquery.XPathException; public class GMonthValue extends AbstractDateTimeValue { protected boolean addTrailingZ = false; public GMonthValue() throws XPathException { super(stripCalendar(TimeUtils.getInstance().newXMLGregorianCalendar(new GregorianCalendar()))); } public GMonthValue(XMLGregorianCalendar calendar) throws XPathException { super(stripCalendar((XMLGregorianCalendar) calendar.clone())); } public GMonthValue(String timeValue) throws XPathException { super(fixTimezone(timeValue)); timeValue = timeValue.trim(); if (timeValue.endsWith("Z")) addTrailingZ = true; if (timeValue.endsWith("-00:00")) addTrailingZ = true; if (timeValue.endsWith("+00:00")) addTrailingZ = true; if (addTrailingZ) this.calendar.setTimezone(0); try { if (calendar.getXMLSchemaType() != DatatypeConstants.GMONTH) throw new IllegalStateException(); } catch (IllegalStateException e) { throw new XPathException("xs:gMonth instance must not have year, month or day fields set"); } } private static XMLGregorianCalendar stripCalendar(XMLGregorianCalendar calendar) { calendar = (XMLGregorianCalendar) calendar.clone(); calendar.setYear(DatatypeConstants.FIELD_UNDEFINED); calendar.setDay(DatatypeConstants.FIELD_UNDEFINED); calendar.setHour(DatatypeConstants.FIELD_UNDEFINED); calendar.setMinute(DatatypeConstants.FIELD_UNDEFINED); calendar.setSecond(DatatypeConstants.FIELD_UNDEFINED); calendar.setMillisecond(DatatypeConstants.FIELD_UNDEFINED); return calendar; } public AtomicValue convertTo(int requiredType) throws XPathException { switch (requiredType) { case Type.GMONTH : case Type.ATOMIC : case Type.ITEM : return this; case Type.STRING : return new StringValue(getStringValue()); case Type.UNTYPED_ATOMIC : return new UntypedAtomicValue(getStringValue()); default : throw new XPathException( "Type error: cannot cast xs:gMonth to " + Type.getTypeName(requiredType)); } } protected AbstractDateTimeValue createSameKind(XMLGregorianCalendar cal) throws XPathException { return new GMonthValue(cal); } public int getType() { return Type.GMONTH; } protected QName getXMLSchemaType() { return DatatypeConstants.GMONTH; } /* public String getStringValue() throws XPathException { String r = super.getStringValue(); if (addTrailingZ) return r + "Z"; return r; } */ public ComputableValue minus(ComputableValue other) throws XPathException { throw new XPathException("Subtraction is not supported on values of type " + Type.getTypeName(getType())); } public int compareTo(Collator collator, AtomicValue other) throws XPathException { if (other.getType() == getType()) { if (!getTimezone().isEmpty()) { if (!((AbstractDateTimeValue) other).getTimezone().isEmpty()) { if (!((DayTimeDurationValue)getTimezone().itemAt(0)).compareTo(null, Constants.EQ, (DayTimeDurationValue)((AbstractDateTimeValue)other).getTimezone().itemAt(0))) return DatatypeConstants.LESSER; } else { if (!((DayTimeDurationValue)getTimezone().itemAt(0)).getStringValue().equals("PT0S")) return DatatypeConstants.LESSER; } } else { if (!((AbstractDateTimeValue)other).getTimezone().isEmpty()) { if (!((DayTimeDurationValue)((AbstractDateTimeValue)other).getTimezone().itemAt(0)).getStringValue().equals("PT0S")) return DatatypeConstants.LESSER; } } // filling in missing timezones with local timezone, should be total order as per XPath 2.0 10.4 int r = this.getImplicitCalendar().compare(((AbstractDateTimeValue) other).getImplicitCalendar()); //getImplicitCalendar().compare(((AbstractDateTimeValue) other).getImplicitCalendar()); if (r == DatatypeConstants.INDETERMINATE) throw new RuntimeException("indeterminate order between " + this + " and " + other); return r; } throw new XPathException( "Type error: cannot compare " + Type.getTypeName(getType()) + " to " + Type.getTypeName(other.getType())); } private static String fixTimezone(String value) { //TODO : should we imply a default "Z" here ? //TODO : should we raise an error on wrong TZ offsets (e.g. 60) ? int p = value.indexOf('Z'); if (p != Constants.STRING_NOT_FOUND) return value.substring(0, p); p = value.indexOf("-00:00"); if (p != Constants.STRING_NOT_FOUND) return value.substring(0, p); p = value.indexOf("+00:00"); if (p != Constants.STRING_NOT_FOUND) return value.substring(0, p); return value; } }