2017-02-09 14 views
3

Я генерирую классы JAXB из XSD для SOAP WebService Я создаю клиент для (сгенерированный с помощью jaxws-maven-plugin v2.4.1, wsimport goal).JAXB: информация о конкретном типе (xsi: type) отсутствует в marshalled XML

У меня возникла проблема, когда JAXB не добавляет xsi: type-Information в узел, который имеет абстрактный тип при сортировке моих объектов. Теперь WebService (по-моему) жалуется, что я пытаюсь передать его элементам без указания того, какой тип они («Определение типа не может быть абстрактным для элемента ...»).

Вот упрощенный пример, который демонстрирует мой вопрос:

абстрактной схемы: (abstract.xsd)

<?xml version="1.0" encoding="UTF-8"?> 
<xsd:schema targetNamespace="http://www.example.com/namespace_abstract" 
      elementFormDefault="qualified" 
      attributeFormDefault="qualified" 
      xmlns:xsd="http://www.w3.org/2001/XMLSchema" 
      xmlns="http://www.example.com/namespace_abstract"> 
    <xsd:complexType name="ElementType" abstract="true"> 
     <xsd:simpleContent> 
      <xsd:extension base="xsd:string"/> 
     </xsd:simpleContent> 
    </xsd:complexType> 
</xsd:schema> 

Бетон Тип схемы: (concrete.xsd)

<?xml version="1.0" encoding="UTF-8"?> 
<xsd:schema targetNamespace="http://www.example.com/namespace_concrete" 
      elementFormDefault="qualified" 
      attributeFormDefault="qualified" 
      xmlns:xsd="http://www.w3.org/2001/XMLSchema" 
      xmlns="http://www.example.com/namespace_concrete" 
      xmlns:abstr="http://www.example.com/namespace_abstract"> 
    <xsd:import namespace="http://www.example.com/namespace_abstract" schemaLocation="abstract.xsd"/> 

    <!-- Concrete type --> 
    <xsd:complexType name="ElementTypeExtension"> 
     <xsd:simpleContent> 
      <xsd:restriction base="abstr:ElementType"> 
       <xsd:enumeration value="one"/> 
       <xsd:enumeration value="two"/> 
       <xsd:enumeration value="three"/> 
      </xsd:restriction> 
     </xsd:simpleContent> 
    </xsd:complexType> 

    <!-- Type that has a field of the abstract type --> 
    <xsd:complexType name="ExtensionTypeContainer"> 
     <xsd:sequence> 
      <xsd:element name="type" type="abstr:ElementType"/> 
     </xsd:sequence> 
    </xsd:complexType> 
</xsd:schema> 

Тест:

import com.example.namespace_concrete.*; 

import javax.xml.bind.*; 
import javax.xml.bind.annotation.XmlRootElement; 

public class MarshallingTest { 
    public static void main(String[] args) throws JAXBException { 
     ElementTypeExtension elementTypeExtension = new ElementTypeExtension(); 
     elementTypeExtension.setValue("one"); 
     ExtensionTypeContainer extensionTypeContainer = new ExtensionTypeContainer(); 
     extensionTypeContainer.setType(elementTypeExtension); 

     XmlRoot xmlRoot = new XmlRoot(); 
     xmlRoot.setContainer(extensionTypeContainer); 

     Marshaller marshaller = JAXBContext.newInstance(XmlRoot.class, ExtensionTypeContainer.class).createMarshaller(); 
     marshaller.setProperty(Marshaller.JAXB_FORMATTED_OUTPUT, true); 
     marshaller.marshal(xmlRoot, System.out); 
    } 

    @XmlRootElement 
    static class XmlRoot { 
     private ExtensionTypeContainer container; 
     public ExtensionTypeContainer getContainer() { return container; } 
     public void setContainer(ExtensionTypeContainer container) { this.container = container; } 
    } 
} 

Выход:

<?xml version="1.0" encoding="UTF-8" standalone="yes"?> 
<xmlRoot xmlns:ns2="http://www.example.com/namespace_concrete"> 
    <container> 
     <ns2:type>one</ns2:type> 
    </container> 
</xmlRoot> 

Недостающую часть является xsi:type="ns2:ElementTypeExtension" на ns2:type узел, оставляя определение типа неоднозначным (конечно, в моем примере есть только один конкретный тип, но все же).

Я нашел способ, чтобы генерировать xsi:type путем модифицирования сгенерированный исходный код абстрактного класса (удален JAXB-Javadoc для удобства чтения):

package com.example.namespace_abstract; 

import javax.xml.bind.annotation.*; 
import com.example.namespace_concrete.ElementTypeExtension; 

@XmlAccessorType(XmlAccessType.FIELD) 
@XmlType(name = "ElementType", propOrder = { "value" }) 
@XmlSeeAlso({ ElementTypeExtension.class }) 
public abstract class ElementType { 
    @XmlValue 
    protected String value; 
    public String getValue() { return value; } 
    public void setValue(String value) { this.value = value; } 
} 

После того, как я удалить @XmlValue -Annotation из value поле xsi:type -Информация присутствует в маршаллированном XML. Исходный код генерируется из XSD, хотя это не совсем вариант.

Может кто-нибудь, пожалуйста, дать мне представление о том, как получить xsi:type? Модификация xsd была бы вариантом.

+0

Я запускал xjc против ваших исходных схем, и ваш пример работал на меня. Сгенерированный класс «ElementType» такой же, как и в вашем вопросе. – teppic

+0

Спасибо за тестирование, похоже, проблема связана с ошибкой [this] (https://java.net/jira/browse/JAXB-890) в JAXB. Обновление моей версии JAXB до версии 2.2.11 было исправлено. – kwj

ответ