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" />