Skip to content
Snippets Groups Projects
XMLDateTime.cpp 38.4 KiB
Newer Older
PeiYong Zhang's avatar
PeiYong Zhang committed
/*
 * The Apache Software License, Version 1.1
 *
 * Copyright (c) 2001 The Apache Software Foundation.  All rights
 * reserved.
 *
 * Redistribution and use in source and binary forms, with or without
 * modification, are permitted provided that the following conditions
 * are met:
 *
 * 1. Redistributions of source code must retain the above copyright
 *    notice, this list of conditions and the following disclaimer.
 *
 * 2. Redistributions in binary form must reproduce the above copyright
 *    notice, this list of conditions and the following disclaimer in
 *    the documentation and/or other materials provided with the
 *    distribution.
 *
 * 3. The end-user documentation included with the redistribution,
 *    if any, must include the following acknowledgment:
 *       "This product includes software developed by the
 *        Apache Software Foundation (http://www.apache.org/)."
 *    Alternately, this acknowledgment may appear in the software itself,
 *    if and wherever such third-party acknowledgments normally appear.
 *
 * 4. The names "Xerces" and "Apache Software Foundation" must
 *    not be used to endorse or promote products derived from this
 *    software without prior written permission. For written
 *    permission, please contact apache\@apache.org.
 *
 * 5. Products derived from this software may not be called "Apache",
 *    nor may "Apache" appear in their name, without prior written
 *    permission of the Apache Software Foundation.
 *
 * THIS SOFTWARE IS PROVIDED ``AS IS'' AND ANY EXPRESSED OR IMPLIED
 * WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
 * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
 * DISCLAIMED.  IN NO EVENT SHALL THE APACHE SOFTWARE FOUNDATION OR
 * ITS CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
 * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
 * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF
 * USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
 * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
 * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT
 * OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
 * SUCH DAMAGE.
 * ====================================================================
 *
 * This software consists of voluntary contributions made by many
 * individuals on behalf of the Apache Software Foundation, and was
 * originally based on software copyright (c) 2001, International
 * Business Machines, Inc., http://www.ibm.com .  For more information
 * on the Apache Software Foundation, please see
 * <http://www.apache.org/>.
 */

/*
 * $Id$
 * $Log$
 * Revision 1.10  2003/05/15 19:07:46  knoaman
 * Partial implementation of the configurable memory manager.
 *
 * Revision 1.9  2003/05/15 16:32:19  gareth
 * We did not allow dateTimes with a timezone due to the last seconds fix.
 *
PeiYong Zhang's avatar
PeiYong Zhang committed
 * Revision 1.8  2003/03/23 22:54:49  peiyongz
 * invalid second values
 *
 * Revision 1.7  2003/02/22 22:49:09  peiyongz
 * Schema Errata E2-45 24:00:00 allowed
 *
 * Revision 1.6  2003/02/02 23:54:43  peiyongz
 * getFormattedString() added to return original and converted value.
 *
 * Revision 1.5  2003/01/30 21:55:22  tng
 * Performance: create getRawData which is similar to toString but return the internal data directly, user is not required to delete the returned memory.
 *
 * Revision 1.4  2002/11/28 20:39:27  peiyongz
 * Schema Errata: E2-23 seconds part shall have at least one digit after the dot
 *                           if it appears.
 *
 * Revision 1.3  2002/11/06 22:22:21  peiyongz
 * Schema-Errata: E2-12: gMonth
 *
Tinny Ng's avatar
Tinny Ng committed
 * Revision 1.2  2002/11/04 15:22:05  tng
 * C++ Namespace Support.
 *
 * Revision 1.1.1.1  2002/02/01 22:22:14  peiyongz
 * sane_include
PeiYong Zhang's avatar
PeiYong Zhang committed
 *
 * Revision 1.4  2001/11/14 22:04:03  peiyongz
 * Patch to apply check on Year and more rigorous on other fields as well.
 *
 * Revision 1.3  2001/11/12 20:36:54  peiyongz
 * SchemaDateTimeException defined
 *
 * Revision 1.2  2001/11/09 20:41:45  peiyongz
 * Fix: compilation error on Solaris and AIX.
 *
 * Revision 1.1  2001/11/07 19:16:03  peiyongz
 * DateTime Port
 *
 *
 */

// ---------------------------------------------------------------------------
//  Includes
// ---------------------------------------------------------------------------
#include <stdlib.h>
#include <xercesc/util/XMLDateTime.hpp>
#include <xercesc/util/XMLString.hpp>
#include <xercesc/util/XMLUni.hpp>
#include <xercesc/util/Janitor.hpp>

Tinny Ng's avatar
Tinny Ng committed
XERCES_CPP_NAMESPACE_BEGIN

PeiYong Zhang's avatar
PeiYong Zhang committed
//
// constants used to process raw data (fBuffer)
//
// [-]{CCYY-MM-DD}'T'{HH:MM:SS.MS}['Z']
//                                [{+|-}hh:mm']
Tinny Ng's avatar
Tinny Ng committed
//
PeiYong Zhang's avatar
PeiYong Zhang committed

static const XMLCh DURATION_STARTER     = chLatin_P;              // 'P'
static const XMLCh DURATION_Y           = chLatin_Y;              // 'Y'
static const XMLCh DURATION_M           = chLatin_M;              // 'M'
static const XMLCh DURATION_D           = chLatin_D;              // 'D'
static const XMLCh DURATION_H           = chLatin_H;              // 'H'
static const XMLCh DURATION_S           = chLatin_S;              // 'S'

static const XMLCh DATE_SEPARATOR       = chDash;                 // '-'
static const XMLCh TIME_SEPARATOR       = chColon;                // ':'
static const XMLCh TIMEZONE_SEPARATOR   = chColon;                // ':'
static const XMLCh DATETIME_SEPARATOR   = chLatin_T;              // 'T'
static const XMLCh MILISECOND_SEPARATOR = chPeriod;               // '.'

static const XMLCh UTC_STD_CHAR         = chLatin_Z;              // 'Z'
static const XMLCh UTC_POS_CHAR         = chPlus;                 // '+'
static const XMLCh UTC_NEG_CHAR         = chDash;                 // '-'

static const XMLCh UTC_SET[]            = {UTC_STD_CHAR           //"Z+-"
                                         , UTC_POS_CHAR
                                         , UTC_NEG_CHAR
                                         , chNull};

static const int YMD_MIN_SIZE    = 10;   // CCYY-MM-DD
static const int YMONTH_MIN_SIZE = 7;    // CCYY_MM
static const int TIME_MIN_SIZE   = 8;    // hh:mm:ss
static const int TIMEZONE_SIZE   = 5;    // hh:mm
static const int DAY_SIZE        = 5;    // ---DD
//static const int MONTH_SIZE      = 6;    // --MM--
PeiYong Zhang's avatar
PeiYong Zhang committed
static const int MONTHDAY_SIZE   = 7;    // --MM-DD
Tinny Ng's avatar
Tinny Ng committed
static const int NOT_FOUND       = -1;
PeiYong Zhang's avatar
PeiYong Zhang committed

Tinny Ng's avatar
Tinny Ng committed
//define constants to be used in assigning default values for
PeiYong Zhang's avatar
PeiYong Zhang committed
//all date/time excluding duration
static const int YEAR_DEFAULT  = 2000;
static const int MONTH_DEFAULT = 01;
static const int DAY_DEFAULT   = 15;

Tinny Ng's avatar
Tinny Ng committed
// order-relation on duration is a partial order. The dates below are used to
PeiYong Zhang's avatar
PeiYong Zhang committed
// for comparison of 2 durations, based on the fact that
// duration x and y is x<=y iff s+x<=s+y
// see 3.2.6 duration W3C schema datatype specs
//
Tinny Ng's avatar
Tinny Ng committed
// the dates are in format: {CCYY,MM,DD, H, S, M, MS, timezone}
static const int DATETIMES[][XMLDateTime::TOTAL_SIZE] =
PeiYong Zhang's avatar
PeiYong Zhang committed
{
Tinny Ng's avatar
Tinny Ng committed
    {1696, 9, 1, 0, 0, 0, 0, XMLDateTime::UTC_STD},
PeiYong Zhang's avatar
PeiYong Zhang committed
	{1697, 2, 1, 0, 0, 0, 0, XMLDateTime::UTC_STD},
	{1903, 3, 1, 0, 0, 0, 0, XMLDateTime::UTC_STD},
	{1903, 7, 1, 0, 0, 0, 0, XMLDateTime::UTC_STD}
};

// ---------------------------------------------------------------------------
//  local methods
// ---------------------------------------------------------------------------
static inline int fQuotient(int a, int b)
{
    div_t div_result = div(a, b);
    return div_result.quot;
}

Tinny Ng's avatar
Tinny Ng committed
static inline int fQuotient(int temp, int low, int high)
PeiYong Zhang's avatar
PeiYong Zhang committed
{
    return fQuotient(temp - low, high - low);
}

Tinny Ng's avatar
Tinny Ng committed
static inline int mod(int a, int b, int quotient)
PeiYong Zhang's avatar
PeiYong Zhang committed
{
	return (a - quotient*b) ;
}

Tinny Ng's avatar
Tinny Ng committed
static inline int modulo (int temp, int low, int high)
PeiYong Zhang's avatar
PeiYong Zhang committed
{
Tinny Ng's avatar
Tinny Ng committed
    //modulo(a - low, high - low) + low
PeiYong Zhang's avatar
PeiYong Zhang committed
    int a = temp - low;
    int b = high - low;
    return (mod (a, b, fQuotient(a, b)) + low) ;
}

static inline bool isLeapYear(int year)
{
Tinny Ng's avatar
Tinny Ng committed
    return((year%4 == 0) && ((year%100 != 0) || (year%400 == 0)));
PeiYong Zhang's avatar
PeiYong Zhang committed
}

Tinny Ng's avatar
Tinny Ng committed
static int maxDayInMonthFor(int year, int month)
PeiYong Zhang's avatar
PeiYong Zhang committed
{

Tinny Ng's avatar
Tinny Ng committed
    if ( month == 4 || month == 6 || month == 9 || month == 11 )
PeiYong Zhang's avatar
PeiYong Zhang committed
    {
        return 30;
    }
Tinny Ng's avatar
Tinny Ng committed
    else if ( month==2 )
PeiYong Zhang's avatar
PeiYong Zhang committed
    {
Tinny Ng's avatar
Tinny Ng committed
        if ( isLeapYear(year) )
PeiYong Zhang's avatar
PeiYong Zhang committed
            return 29;
Tinny Ng's avatar
Tinny Ng committed
        else
PeiYong Zhang's avatar
PeiYong Zhang committed
            return 28;
    }
    else
    {
        return 31;
    }

}

// ---------------------------------------------------------------------------
//  static methods : for duration
// ---------------------------------------------------------------------------
/**
 * Compares 2 given durations. (refer to W3C Schema Datatypes "3.2.6 duration")
 *
 * 3.2.6.2 Order relation on duration
 *
Tinny Ng's avatar
Tinny Ng committed
 *     In general, the order-relation on duration is a partial order since there is no
 *  determinate relationship between certain durations such as one month (P1M) and 30 days (P30D).
 *  The order-relation of two duration values x and y is x < y iff s+x < s+y for each qualified
 *  dateTime s in the list below.
PeiYong Zhang's avatar
PeiYong Zhang committed
 *
 *     These values for s cause the greatest deviations in the addition of dateTimes and durations
Tinny Ng's avatar
Tinny Ng committed
 *
PeiYong Zhang's avatar
PeiYong Zhang committed
 **/
int XMLDateTime::compare(const XMLDateTime* const pDate1
                       , const XMLDateTime* const pDate2
                       , bool  strict)
{
    //REVISIT: this is unoptimazed vs of comparing 2 durations
    //         Algorithm is described in 3.2.6.2 W3C Schema Datatype specs
    //

    int resultA, resultB = INDETERMINATE;

    //try and see if the objects are equal
    if ( (resultA = compareOrder(pDate1, pDate2)) == EQUAL)
        return EQUAL;

    //long comparison algorithm is required
    XMLDateTime tempA, *pTempA = &tempA;
    XMLDateTime tempB, *pTempB = &tempB;

    addDuration(pTempA, pDate1, 0);
    addDuration(pTempB, pDate2, 0);
    resultA = compareOrder(pTempA, pTempB);
Tinny Ng's avatar
Tinny Ng committed
    if ( resultA == INDETERMINATE )
PeiYong Zhang's avatar
PeiYong Zhang committed
        return INDETERMINATE;

    addDuration(pTempA, pDate1, 1);
    addDuration(pTempB, pDate2, 1);
    resultB = compareOrder(pTempA, pTempB);
    resultA = compareResult(resultA, resultB, strict);
Tinny Ng's avatar
Tinny Ng committed
    if ( resultA == INDETERMINATE )
PeiYong Zhang's avatar
PeiYong Zhang committed
        return INDETERMINATE;

    addDuration(pTempA, pDate1, 2);
    addDuration(pTempB, pDate2, 2);
    resultB = compareOrder(pTempA, pTempB);
    resultA = compareResult(resultA, resultB, strict);
    if ( resultA == INDETERMINATE )
        return INDETERMINATE;

    addDuration(pTempA, pDate1, 3);
    addDuration(pTempB, pDate2, 3);
    resultB = compareOrder(pTempA, pTempB);
    resultA = compareResult(resultA, resultB, strict);

    return resultA;

}

//
// Form a new XMLDateTime with duration and baseDate array
// Note: C++        Java
//       fNewDate   duration
//       fDuration  date
//

Tinny Ng's avatar
Tinny Ng committed
void XMLDateTime::addDuration(XMLDateTime*             fNewDate
PeiYong Zhang's avatar
PeiYong Zhang committed
                            , const XMLDateTime* const fDuration
                            , int index)

{

    //REVISIT: some code could be shared between normalize() and this method,
    //         however is it worth moving it? The structures are different...
    //

    fNewDate->reset();
    //add months (may be modified additionaly below)
    int temp = DATETIMES[index][Month] + fDuration->fValue[Month];
    fNewDate->fValue[Month] = modulo(temp, 1, 13);
    int carry = fQuotient(temp, 1, 13);

    //add years (may be modified additionaly below)
    fNewDate->fValue[CentYear] = DATETIMES[index][CentYear] + fDuration->fValue[CentYear] + carry;

    //add seconds
    temp = DATETIMES[index][Second] + fDuration->fValue[Second];
    carry = fQuotient (temp, 60);
    fNewDate->fValue[Second] =  mod(temp, 60, carry);
		
Tinny Ng's avatar
Tinny Ng committed
    //add minutes
    temp = DATETIMES[index][Minute] + fDuration->fValue[Minute] + carry;
    carry = fQuotient(temp, 60);
    fNewDate->fValue[Minute] = mod(temp, 60, carry);
PeiYong Zhang's avatar
PeiYong Zhang committed

    //add hours
    temp = DATETIMES[index][Hour] + fDuration->fValue[Hour] + carry;
    carry = fQuotient(temp, 24);
    fNewDate->fValue[Hour] = mod(temp, 24, carry);
		
    fNewDate->fValue[Day] = DATETIMES[index][Day] + fDuration->fValue[Day] + carry;

Tinny Ng's avatar
Tinny Ng committed
    while ( true )
PeiYong Zhang's avatar
PeiYong Zhang committed
    {
        temp = maxDayInMonthFor(fNewDate->fValue[CentYear], fNewDate->fValue[Month]);
Tinny Ng's avatar
Tinny Ng committed
        if ( fNewDate->fValue[Day] < 1 )
PeiYong Zhang's avatar
PeiYong Zhang committed
        { //original fNewDate was negative
            fNewDate->fValue[Day] += maxDayInMonthFor(fNewDate->fValue[CentYear], fNewDate->fValue[Month]-1);
            carry = -1;
        }
Tinny Ng's avatar
Tinny Ng committed
        else if ( fNewDate->fValue[Day] > temp )
PeiYong Zhang's avatar
PeiYong Zhang committed
        {
            fNewDate->fValue[Day] -= temp;
            carry = 1;
        }
Tinny Ng's avatar
Tinny Ng committed
        else
PeiYong Zhang's avatar
PeiYong Zhang committed
        {
            break;
        }

        temp = fNewDate->fValue[Month] + carry;
        fNewDate->fValue[Month] = modulo(temp, 1, 13);
        fNewDate->fValue[CentYear] += fQuotient(temp, 1, 13);
    }

    //fNewDate->fValue[utc] = UTC_STD_CHAR;
    fNewDate->fValue[utc] = UTC_STD;
}

int XMLDateTime::compareResult(short resultA
                             , short resultB
                             , bool strict)
{

Tinny Ng's avatar
Tinny Ng committed
    if ( resultB == INDETERMINATE )
PeiYong Zhang's avatar
PeiYong Zhang committed
    {
        return INDETERMINATE;
    }
Tinny Ng's avatar
Tinny Ng committed
    else if ( (resultA != resultB) &&
              strict                )
PeiYong Zhang's avatar
PeiYong Zhang committed
    {
        return INDETERMINATE;
    }
Tinny Ng's avatar
Tinny Ng committed
    else if ( (resultA != resultB) &&
              !strict               )
PeiYong Zhang's avatar
PeiYong Zhang committed
    {
Tinny Ng's avatar
Tinny Ng committed
        if ( (resultA != EQUAL) &&
             (resultB != EQUAL)  )
PeiYong Zhang's avatar
PeiYong Zhang committed
        {
            return INDETERMINATE;
        }
Tinny Ng's avatar
Tinny Ng committed
        else
PeiYong Zhang's avatar
PeiYong Zhang committed
        {
            return (resultA != EQUAL)? resultA : resultB;
        }
    }

    return resultA;
	
}

// ---------------------------------------------------------------------------
//  static methods : for others
// ---------------------------------------------------------------------------
int XMLDateTime::compare(const XMLDateTime* const pDate1
                       , const XMLDateTime* const pDate2)
{

    if (pDate1->fValue[utc] == pDate2->fValue[utc])
    {
Tinny Ng's avatar
Tinny Ng committed
        return XMLDateTime::compareOrder(pDate1, pDate2);
PeiYong Zhang's avatar
PeiYong Zhang committed
    }

    short c1, c2;

Tinny Ng's avatar
Tinny Ng committed
    if ( pDate1->isNormalized())
PeiYong Zhang's avatar
PeiYong Zhang committed
    {
        c1 = compareResult(pDate1, pDate2, false, UTC_POS);
        c2 = compareResult(pDate1, pDate2, false, UTC_NEG);
        return getRetVal(c1, c2);
    }
Tinny Ng's avatar
Tinny Ng committed
    else if ( pDate2->isNormalized())
PeiYong Zhang's avatar
PeiYong Zhang committed
    {
        c1 = compareResult(pDate1, pDate2, true, UTC_POS);
        c2 = compareResult(pDate1, pDate2, true, UTC_NEG);
        return getRetVal(c1, c2);
    }

    return INDETERMINATE;	
}

int XMLDateTime::compareResult(const XMLDateTime* const pDate1
                             , const XMLDateTime* const pDate2
                             , bool  set2Left
                             , int   utc_type)
{
    XMLDateTime tmpDate = (set2Left ? *pDate1 : *pDate2);

    tmpDate.fTimeZone[hh] = 14;
    tmpDate.fTimeZone[mm] = 0;
    tmpDate.fValue[utc] = utc_type;
    tmpDate.normalize();

    return (set2Left? XMLDateTime::compareOrder(&tmpDate, pDate2) :
                      XMLDateTime::compareOrder(pDate1, &tmpDate));
}

int XMLDateTime::compareOrder(const XMLDateTime* const lValue
                            , const XMLDateTime* const rValue)
Tinny Ng's avatar
Tinny Ng committed
{
PeiYong Zhang's avatar
PeiYong Zhang committed
    //
Tinny Ng's avatar
Tinny Ng committed
    // If any of the them is not normalized() yet,
PeiYong Zhang's avatar
PeiYong Zhang committed
    // we need to do something here.
    //
    XMLDateTime lTemp = *lValue;
    XMLDateTime rTemp = *rValue;

    lTemp.normalize();
    rTemp.normalize();

Tinny Ng's avatar
Tinny Ng committed
    for ( int i = 0 ; i < TOTAL_SIZE; i++ )
PeiYong Zhang's avatar
PeiYong Zhang committed
    {
Tinny Ng's avatar
Tinny Ng committed
        if ( lTemp.fValue[i] < rTemp.fValue[i] )
PeiYong Zhang's avatar
PeiYong Zhang committed
        {
            return LESS_THAN;
        }
Tinny Ng's avatar
Tinny Ng committed
        else if ( lTemp.fValue[i] > rTemp.fValue[i] )
PeiYong Zhang's avatar
PeiYong Zhang committed
        {
            return GREATER_THAN;
        }
    }

    return EQUAL;
}

// ---------------------------------------------------------------------------
//  ctor and dtor
// ---------------------------------------------------------------------------
XMLDateTime::~XMLDateTime()
{
    if (fBuffer)
        fMemoryManager->deallocate(fBuffer);//delete[] fBuffer;
PeiYong Zhang's avatar
PeiYong Zhang committed
}

XMLDateTime::XMLDateTime()
: fBuffer(0)
, fMemoryManager(XMLPlatformUtils::fgMemoryManager)
PeiYong Zhang's avatar
PeiYong Zhang committed
{
    reset();
}

XMLDateTime::XMLDateTime(const XMLCh* const aString)
: fBuffer(0)
, fMemoryManager(XMLPlatformUtils::fgMemoryManager)
PeiYong Zhang's avatar
PeiYong Zhang committed
{
    setBuffer(aString);
}

// -----------------------------------------------------------------------
// Copy ctor and Assignment operators
// -----------------------------------------------------------------------

XMLDateTime::XMLDateTime(const XMLDateTime &toCopy)
: fBuffer(0)
, fMemoryManager(XMLPlatformUtils::fgMemoryManager)
PeiYong Zhang's avatar
PeiYong Zhang committed
{
    copy(toCopy);
}

XMLDateTime& XMLDateTime::operator=(const XMLDateTime& rhs)
{
    if (this == &rhs)
        return *this;

    copy(rhs);
    return *this;
}

// -----------------------------------------------------------------------
// Implementation of Abstract Interface
// -----------------------------------------------------------------------

//
// We may simply return the handle to fBuffer, but
Tinny Ng's avatar
Tinny Ng committed
// for the sake of consistency, we return a duplicated copy
PeiYong Zhang's avatar
PeiYong Zhang committed
// and the caller is responsible for the release of the buffer
// just like any other things in the XMLNumber family.
//
XMLCh*  XMLDateTime::toString() const
{
    assertBuffer();

    // Return data using global operator new
PeiYong Zhang's avatar
PeiYong Zhang committed
    XMLCh* retBuf = XMLString::replicate(fBuffer);
    return retBuf;
}

//
// We may simply return the handle to fBuffer
//
XMLCh*  XMLDateTime::getRawData() const
{
    assertBuffer();    
    return fBuffer;
}


const XMLCh*  XMLDateTime::getFormattedString() const
{
    return getRawData();
}

PeiYong Zhang's avatar
PeiYong Zhang committed
int XMLDateTime::getSign() const
{
    return 0;
}

// ---------------------------------------------------------------------------
//  Parsers
// ---------------------------------------------------------------------------

//
// [-]{CCYY-MM-DD}'T'{HH:MM:SS.MS}[TimeZone]
//
void XMLDateTime::parseDateTime()
{
    initParser();
    getDate();

    //fStart is supposed to point to 'T'
    if (fBuffer[fStart++] != DATETIME_SEPARATOR)
        ThrowXML1(SchemaDateTimeException
                , XMLExcepts::DateTime_dt_missingT
                , fBuffer);

    getTime();
    validateDateTime();
    normalize();
}

//
// [-]{CCYY-MM-DD}[TimeZone]
//
void XMLDateTime::parseDate()
{
    initParser();
    getDate();
    parseTimeZone();
    validateDateTime();
    normalize();
}

void XMLDateTime::parseTime()
{
    initParser();

    // time initialize to default values
    fValue[CentYear]= YEAR_DEFAULT;
    fValue[Month]   = MONTH_DEFAULT;
    fValue[Day]     = DAY_DEFAULT;

    getTime();

    validateDateTime();
    normalize();
}

//
// {---DD}[TimeZone]
//  01234
//
void XMLDateTime::parseDay()
{
    initParser();

Tinny Ng's avatar
Tinny Ng committed
    if (fBuffer[0] != DATE_SEPARATOR ||
        fBuffer[1] != DATE_SEPARATOR ||
        fBuffer[2] != DATE_SEPARATOR  )
PeiYong Zhang's avatar
PeiYong Zhang committed
    {
        ThrowXML1(SchemaDateTimeException
                , XMLExcepts::DateTime_gDay_invalid
                , fBuffer);
    }

Tinny Ng's avatar
Tinny Ng committed
    //initialize values
PeiYong Zhang's avatar
PeiYong Zhang committed
    fValue[CentYear] = YEAR_DEFAULT;
Tinny Ng's avatar
Tinny Ng committed
    fValue[Month]    = MONTH_DEFAULT;
PeiYong Zhang's avatar
PeiYong Zhang committed
    fValue[Day]      = parseInt(fStart+3, fStart+5);

Tinny Ng's avatar
Tinny Ng committed
    if ( DAY_SIZE < fEnd )
PeiYong Zhang's avatar
PeiYong Zhang committed
    {
        int sign = findUTCSign(DAY_SIZE);
Tinny Ng's avatar
Tinny Ng committed
        if ( sign < 0 )
PeiYong Zhang's avatar
PeiYong Zhang committed
        {
            ThrowXML1(SchemaDateTimeException
                    , XMLExcepts::DateTime_gDay_invalid
                    , fBuffer);
        }
Tinny Ng's avatar
Tinny Ng committed
        else
PeiYong Zhang's avatar
PeiYong Zhang committed
        {
            getTimeZone(sign);
        }
    }

    validateDateTime();
    normalize();
}

//
// {--MM--}[TimeZone]
// {--MM}[TimeZone]
PeiYong Zhang's avatar
PeiYong Zhang committed
//  012345
//
void XMLDateTime::parseMonth()
{
    initParser();

Tinny Ng's avatar
Tinny Ng committed
    if (fBuffer[0] != DATE_SEPARATOR ||
        fBuffer[1] != DATE_SEPARATOR  )
PeiYong Zhang's avatar
PeiYong Zhang committed
    {
        ThrowXML1(SchemaDateTimeException
                , XMLExcepts::DateTime_gMth_invalid
                , fBuffer);
    }

    //set constants
    fValue[CentYear] = YEAR_DEFAULT;
    fValue[Day]      = DAY_DEFAULT;
    fValue[Month]    = parseInt(2, 4);

    // REVISIT: allow both --MM and --MM-- now. 
    // need to remove the following lines to disallow --MM-- 
    // when the errata is officially in the rec. 
    fStart = 4;
    if ( fEnd >= fStart+2 && fBuffer[fStart] == DATE_SEPARATOR && fBuffer[fStart+1] == DATE_SEPARATOR ) 
    { 
        fStart += 2; 
    } 

    //
    // parse TimeZone if any
    //
    if ( fStart < fEnd )
PeiYong Zhang's avatar
PeiYong Zhang committed
    {
        int sign = findUTCSign(fStart);
Tinny Ng's avatar
Tinny Ng committed
        if ( sign < 0 )
PeiYong Zhang's avatar
PeiYong Zhang committed
        {
            ThrowXML1(SchemaDateTimeException
                    , XMLExcepts::DateTime_gMth_invalid
                    , fBuffer);
        }
Tinny Ng's avatar
Tinny Ng committed
        else
PeiYong Zhang's avatar
PeiYong Zhang committed
        {
            getTimeZone(sign);
        }
    }

    validateDateTime();
    normalize();
}

//
//[-]{CCYY}[TimeZone]
// 0  1234
//
void XMLDateTime::parseYear()
{
    initParser();

    // skip the first '-' and search for timezone
    //
    int sign = findUTCSign((fBuffer[0] == chDash) ? 1 : 0);

Tinny Ng's avatar
Tinny Ng committed
    if (sign == NOT_FOUND)
PeiYong Zhang's avatar
PeiYong Zhang committed
    {
        fValue[CentYear] = parseIntYear(fEnd);
    }
Tinny Ng's avatar
Tinny Ng committed
    else
PeiYong Zhang's avatar
PeiYong Zhang committed
    {
        fValue[CentYear] = parseIntYear(sign);
        getTimeZone(sign);
    }

Tinny Ng's avatar
Tinny Ng committed
    //initialize values
PeiYong Zhang's avatar
PeiYong Zhang committed
    fValue[Month] = MONTH_DEFAULT;
    fValue[Day]   = DAY_DEFAULT;   //java is 1

    validateDateTime();
    normalize();
}

//
//{--MM-DD}[TimeZone]
// 0123456
//
void XMLDateTime::parseMonthDay()
{
    initParser();

Tinny Ng's avatar
Tinny Ng committed
    if (fBuffer[0] != DATE_SEPARATOR ||
        fBuffer[1] != DATE_SEPARATOR ||
PeiYong Zhang's avatar
PeiYong Zhang committed
        fBuffer[4] != DATE_SEPARATOR )
    {
        ThrowXML1(SchemaDateTimeException
                , XMLExcepts::DateTime_gMthDay_invalid
                , fBuffer);
    }


Tinny Ng's avatar
Tinny Ng committed
    //initialize
PeiYong Zhang's avatar
PeiYong Zhang committed
    fValue[CentYear] = YEAR_DEFAULT;
    fValue[Month]    = parseInt(2, 4);	
    fValue[Day]      = parseInt(5, 7);

Tinny Ng's avatar
Tinny Ng committed
    if ( MONTHDAY_SIZE < fEnd )
PeiYong Zhang's avatar
PeiYong Zhang committed
    {
        int sign = findUTCSign(MONTHDAY_SIZE);
Tinny Ng's avatar
Tinny Ng committed
        if ( sign<0 )
PeiYong Zhang's avatar
PeiYong Zhang committed
        {
            ThrowXML1(SchemaDateTimeException
                    , XMLExcepts::DateTime_gMthDay_invalid
                    , fBuffer);
        }
Tinny Ng's avatar
Tinny Ng committed
        else
PeiYong Zhang's avatar
PeiYong Zhang committed
        {
            getTimeZone(sign);
        }
    }

    validateDateTime();
    normalize();
}

void XMLDateTime::parseYearMonth()
{
    initParser();

    // get date
    getYearMonth();
    fValue[Day] = DAY_DEFAULT;
    parseTimeZone();

    validateDateTime();
    normalize();
}

//
Tinny Ng's avatar
Tinny Ng committed
//PnYn MnDTnH nMnS: -P1Y2M3DT10H30M
PeiYong Zhang's avatar
PeiYong Zhang committed
//
// [-]{'P'{[n'Y'][n'M'][n'D']['T'][n'H'][n'M'][n'S']}}
//
//  Note: the n above shall be >= 0
//        if no time element found, 'T' shall be absent
//
void XMLDateTime::parseDuration()
{
    initParser();

    // must start with '-' or 'P'
    //
    XMLCh c = fBuffer[fStart++];
Tinny Ng's avatar
Tinny Ng committed
    if ( (c != DURATION_STARTER) &&
         (c != chDash)            )
PeiYong Zhang's avatar
PeiYong Zhang committed
    {
        ThrowXML1(SchemaDateTimeException
                , XMLExcepts::DateTime_dur_Start_dashP
                , fBuffer);
    }

    // 'P' must ALWAYS be present in either case
Tinny Ng's avatar
Tinny Ng committed
    if ( (c == chDash) &&
PeiYong Zhang's avatar
PeiYong Zhang committed
         (fBuffer[fStart++]!= DURATION_STARTER ))
    {
        ThrowXML1(SchemaDateTimeException
                , XMLExcepts::DateTime_dur_noP
                , fBuffer);
    }

    // java code
    //date[utc]=(c=='-')?'-':0;
Tinny Ng's avatar
Tinny Ng committed
    //fValue[utc] = UTC_STD;
PeiYong Zhang's avatar
PeiYong Zhang committed
    fValue[utc] = (fBuffer[0] == chDash? UTC_NEG : UTC_STD);

    int negate = ( fBuffer[0] == chDash ? -1 : 1);

Tinny Ng's avatar
Tinny Ng committed
    //
PeiYong Zhang's avatar
PeiYong Zhang committed
    // No negative value is allowed after 'P'
    //
    // eg P-1234, invalid
    //
    if (indexOf(fStart, fEnd, chDash) != NOT_FOUND)
    {
        ThrowXML1(SchemaDateTimeException
                , XMLExcepts::DateTime_dur_DashNotFirst
                , fBuffer);
    }

    //at least one number and designator must be seen after P
    bool designator = false;

Tinny Ng's avatar
Tinny Ng committed
    int endDate = indexOf(fStart, fEnd, DATETIME_SEPARATOR);
    if ( endDate == NOT_FOUND )
PeiYong Zhang's avatar
PeiYong Zhang committed
    {
        endDate = fEnd;  // 'T' absent
    }

Tinny Ng's avatar
Tinny Ng committed
    //find 'Y'
PeiYong Zhang's avatar
PeiYong Zhang committed
    int end = indexOf(fStart, endDate, DURATION_Y);
Tinny Ng's avatar
Tinny Ng committed
    if ( end != NOT_FOUND )
PeiYong Zhang's avatar
PeiYong Zhang committed
    {
        //scan year
        fValue[CentYear] = negate * parseInt(fStart, end);
        fStart = end+1;
        designator = true;
    }

    end = indexOf(fStart, endDate, DURATION_M);
Tinny Ng's avatar
Tinny Ng committed
    if ( end != NOT_FOUND )
PeiYong Zhang's avatar
PeiYong Zhang committed
    {
        //scan month
        fValue[Month] = negate * parseInt(fStart, end);
        fStart = end+1;
        designator = true;
    }

    end = indexOf(fStart, endDate, DURATION_D);
Tinny Ng's avatar
Tinny Ng committed
    if ( end != NOT_FOUND )
PeiYong Zhang's avatar
PeiYong Zhang committed
    {
        //scan day
        fValue[Day] = negate * parseInt(fStart,end);
        fStart = end+1;
        designator = true;
    }

    if ( (fEnd == endDate) &&   // 'T' absent
         (fStart != fEnd)   )   // something after Day
    {
        ThrowXML1(SchemaDateTimeException
                , XMLExcepts::DateTime_dur_inv_b4T
                , fBuffer);
    }

    if ( fEnd != endDate ) // 'T' present
    {
        //scan hours, minutes, seconds
Tinny Ng's avatar
Tinny Ng committed
        //
PeiYong Zhang's avatar
PeiYong Zhang committed

        // skip 'T' first
        end = indexOf(++fStart, fEnd, DURATION_H);
Tinny Ng's avatar
Tinny Ng committed
        if ( end != NOT_FOUND )
PeiYong Zhang's avatar
PeiYong Zhang committed
        {
            //scan hours
            fValue[Hour] = negate * parseInt(fStart, end);
            fStart = end+1;
            designator = true;
        }

        end = indexOf(fStart, fEnd, DURATION_M);
Tinny Ng's avatar
Tinny Ng committed
        if ( end != NOT_FOUND )
PeiYong Zhang's avatar
PeiYong Zhang committed
        {
            //scan min
            fValue[Minute] = negate * parseInt(fStart, end);
            fStart = end+1;
            designator = true;
        }

        end = indexOf(fStart, fEnd, DURATION_S);
Tinny Ng's avatar
Tinny Ng committed
        if ( end != NOT_FOUND )
PeiYong Zhang's avatar
PeiYong Zhang committed
        {
            //scan seconds
            int mlsec = indexOf (fStart, end, MILISECOND_SEPARATOR);

            /***
             * Schema Errata: E2-23
             * at least one digit must follow the decimal point if it appears. 
             * That is, the value of the seconds component must conform 
             * to the following pattern: [0-9]+(.[0-9]+)? 
             */
Tinny Ng's avatar
Tinny Ng committed
            if ( mlsec != NOT_FOUND )
PeiYong Zhang's avatar
PeiYong Zhang committed
            {
                /***
                 * make usure there is something after the '.' and before the end.
                 */
                if ( mlsec+1 == end )
                {
                    ThrowXML1(SchemaDateTimeException
                            , XMLExcepts::DateTime_dur_inv_seconds
                            ,fBuffer);
                }

PeiYong Zhang's avatar
PeiYong Zhang committed
                fValue[Second]     = negate * parseInt(fStart, mlsec);
                fValue[MiliSecond] = negate * parseInt(mlsec+1, end);
            }
Tinny Ng's avatar
Tinny Ng committed
            else
PeiYong Zhang's avatar
PeiYong Zhang committed
            {
                fValue[Second] = negate * parseInt(fStart,end);
            }
Tinny Ng's avatar
Tinny Ng committed

PeiYong Zhang's avatar
PeiYong Zhang committed
            fStart = end+1;
            designator = true;
        }

        // no additional data should appear after last item
        // P1Y1M1DT is illigal value as well
Tinny Ng's avatar
Tinny Ng committed
        if ( (fStart != fEnd) ||
              fBuffer[--fStart] == DATETIME_SEPARATOR )
PeiYong Zhang's avatar
PeiYong Zhang committed
        {
            ThrowXML1(SchemaDateTimeException
                    , XMLExcepts::DateTime_dur_NoTimeAfterT
                    ,fBuffer);
        }
    }

Tinny Ng's avatar
Tinny Ng committed
    if ( !designator )
PeiYong Zhang's avatar
PeiYong Zhang committed
    {
        ThrowXML1(SchemaDateTimeException
                , XMLExcepts::DateTime_dur_NoElementAtAll
                , fBuffer);
    }

}

// ---------------------------------------------------------------------------
//  Scanners
// ---------------------------------------------------------------------------

//
// [-]{CCYY-MM-DD}
//
// Note: CCYY could be more than 4 digits
//       Assuming fStart point to the beginning of the Date Section
//       fStart updated to point to the position right AFTER the second 'D'
//       Since the lenght of CCYY might be variable, we can't check format upfront
//
void XMLDateTime::getDate()
{

    // Ensure enough chars in buffer
    if ( (fStart+YMD_MIN_SIZE) > fEnd)
        ThrowXML1(SchemaDateTimeException
                , XMLExcepts::DateTime_date_incomplete
                , fBuffer);

Tinny Ng's avatar
Tinny Ng committed
    getYearMonth();    // Scan YearMonth and
                       // fStart point to the next '-'
PeiYong Zhang's avatar
PeiYong Zhang committed

Tinny Ng's avatar
Tinny Ng committed
    if (fBuffer[fStart++] != DATE_SEPARATOR)
PeiYong Zhang's avatar
PeiYong Zhang committed
    {
        ThrowXML1(SchemaDateTimeException
                , XMLExcepts::DateTime_date_invalid
                , fBuffer);
        //("CCYY-MM must be followed by '-' sign");
    }

    fValue[Day] = parseInt(fStart, fStart+2);
    fStart += 2 ;  //fStart points right after the Day

    return;
}

//
// hh:mm:ss[.msssss]['Z']
// hh:mm:ss[.msssss][['+'|'-']hh:mm]
// 012345678
//
// Note: Assuming fStart point to the beginning of the Time Section
//       fStart updated to point to the position right AFTER the second 's'
//                                                  or ms if any
//