SAXParseException: значение не является допустимым значением для "даты"

У меня есть дерево объектов POJO, которое представляет XML-схему. Это было создано со следующим jaxb муравей сценарий.

Я хочу проверить корневой POJO и его дочерние объекты по схеме для отсутствующих атрибутов.

Мой код выглядит следующим образом: (блок try/catch опущен, вдохновлен SO-вопросом. Как проверить схему в JAXB 2.0 без сортировки?)

public boolean validateAgainstSchema(Pojo pojo)
{
        JAXBContext jc;
        jc = JAXBContext.newInstance(Pojo.class);
        SchemaFactory sf = SchemaFactory.newInstance(XMLConstants.W3C_XML_SCHEMA_NS_URI);
        Schema schema = sf.newSchema(new ClassPathResource("schema.xsd").getFile());

        Marshaller marshaller = jc.createMarshaller();
        marshaller.setSchema(schema);
        marshaller.marshal(schema, new DefaultHandler());
        return true;
}

Один из моих атрибутов (pojo.childEntity.someAttribute) это date

XSD

<xsd:attribute name="some_date" use="required">
  <xsd:simpleType>
    <xsd:restriction base="xsd:date" />
  </xsd:simpleType>
</xsd:attribute>

Джава

@XmlAttribute(name = "someDate", required = true)
protected XMLGregorianCalendar someDate;

Это становится заполненным от java.util.Date объект из другого POJO (который сопоставлен с Hibernate).

private static final XMLGregorianCalendar dateToCalendar(Date date)
{
    if (date == null)
        return null;
    try
    {
        GregorianCalendar c = new GregorianCalendar();
        c.setTime(date);

        return DatatypeFactory.newInstance()
                .newXMLGregorianCalendar(c);
    }
    catch (DatatypeConfigurationException e)
    {
        e.printStackTrace();
        return null;
    }

}

Исключение составляет:

javax.xml.bind.MarshalException
 - with linked exception:
[org.xml.sax.SAXParseException: cvc-datatype-valid.1.2.1: '2001-05-11T00:00:00.000+02:00' is not a valid value for 'date'.]

Похоже, что JAXB пытается установить дату и время для поля, которое должно содержать только дату, а XMLGregorianCalendard - это просто контейнер даты и времени.

Вопрос в том, что вызывает ошибку? Как исправить?

2 ответа

Решение

По умолчанию вывод для XMLGregorianCalendar свойство будет основано на том, как вы его создадите. Если вы заполните часть времени, то часть времени будет выведена в XML. Вы можете позвонить getXMLSchemaType() метод, чтобы увидеть, что такое соответствующее представление XML:

Вы можете использовать @XmlSchemaType аннотация для переопределения представления.

Модель Java (Root)

Ниже находится объект с 3 XMLGregorianCalendar поля. На 3-м я буду использовать @XmlSchemaType аннотация для указания типа схемы.

import javax.xml.bind.annotation.*;
import javax.xml.datatype.XMLGregorianCalendar;

@XmlRootElement
@XmlAccessorType(XmlAccessType.FIELD)
public class Root {

    protected XMLGregorianCalendar default1;

    protected XMLGregorianCalendar default2;

    @XmlSchemaType(name="date")
    protected XMLGregorianCalendar schemaTypeDate;

}

Демонстрационный код

В демонстрационном коде ниже мы создадим 2 экземпляра XMLGregorianCalendar, Один будет иметь тип схемы date другой dateTime, По умолчанию это представление XML, используемое при сортировке в XML. На schemaTypeDate поле мы будем использовать @XmlSchemaType аннотация для переопределения по умолчанию.

import javax.xml.bind.*;
import javax.xml.datatype.*;

public class Demo {

    public static void main(String[] args) throws Exception {
        JAXBContext jc = JAXBContext.newInstance(Root.class);

        DatatypeFactory df = DatatypeFactory.newInstance();
        XMLGregorianCalendar date  = df.newXMLGregorianCalendar("2013-07-03");
        XMLGregorianCalendar dateTime = df.newXMLGregorianCalendar("1999-12-31T23:59:00");

        Root root = new Root();
        root.default1 = date;
        root.default2 = dateTime;
        root.schemaTypeDate = dateTime;

        Marshaller marshaller = jc.createMarshaller();
        marshaller.setProperty(Marshaller.JAXB_FORMATTED_OUTPUT, true);
        marshaller.marshal(root, System.out);
    }

}

Выход

<?xml version="1.0" encoding="UTF-8" standalone="yes"?>
<root>
    <default1>2013-07-03</default1>
    <default2>1999-12-31T23:59:00</default2>
    <schemaTypeDate>1999-12-31</schemaTypeDate>
</root>

ОБНОВИТЬ

Хорошо, так как у меня есть loooooooooooooooooooooooooooooooooots XmlGregorianCalendars, есть ли способ сказать XJC добавить атрибут xmlSchemaType ко всем XGCs?

JAXB сделает это за вас, когда тип схемы является одним из встроенных в типы XML-схем, т.е. xsd:date или же xsd:dateTime, но не тогда, когда вы расширили один из этих типов.

Схема XML (schema.xsd)

<?xml version="1.0" encoding="UTF-8"?>
<schema xmlns="http://www.w3.org/2001/XMLSchema">
    <complexType name="root">
        <sequence>
            <element name="dateElement" type="date" />
            <element name="dateTimeElement" type="dateTime" />
            <element name="extendedDateElement">
                <simpleType>
                    <restriction base="date" />
                </simpleType>
            </element>
        </sequence>
        <attribute name="dateAttribute" type="date" />
        <attribute name="dateTimeAttribute" type="dateTime" />
        <attribute name="extendedDateAttribute">
            <simpleType>
                <restriction base="date" />
            </simpleType>
        </attribute>
    </complexType>
</schema>

корень

@XmlAccessorType(XmlAccessType.FIELD)
@XmlType(name = "root", propOrder = {
    "dateElement",
    "dateTimeElement",
    "extendedDateElement"
})
public class Root {

    @XmlElement(required = true)
    @XmlSchemaType(name = "date")
    protected XMLGregorianCalendar dateElement;

    @XmlElement(required = true)
    @XmlSchemaType(name = "dateTime")
    protected XMLGregorianCalendar dateTimeElement;

    @XmlElement(required = true)
    protected XMLGregorianCalendar extendedDateElement;

    @XmlAttribute(name = "dateAttribute")
    @XmlSchemaType(name = "date")
    protected XMLGregorianCalendar dateAttribute;

    @XmlAttribute(name = "dateTimeAttribute")
    @XmlSchemaType(name = "dateTime")
    protected XMLGregorianCalendar dateTimeAttribute;

    @XmlAttribute(name = "extendedDateAttribute")
    protected XMLGregorianCalendar extendedDateAttribute;

}

Если дата "2001-05-11T00:00:00.000+02:00", используйте

<xsd:restriction base="xsd:dateTime" />
Другие вопросы по тегам