Исключение из-за нехватки памяти при получении большого файла с помощью веб-службы SOAP

Я пытаюсь прочитать большой файл, передаваемый через веб-сервис на основе SOAP. Сбой клиентского кода с исключением "Недостаточно памяти". Я использую MTOM для отправки двоичных данных, надеясь, что смогу позаботиться о передаче и чтении большого файла. Размер файла составляет 750 МБ. Я использую Apache CXF. Вот реализация конечных точек веб-сервисов.


@MTOM
@WebService(endpointInterface = "com.test.contentservice.service.IContentService")
@BindingType(value=javax.xml.ws.soap.SOAPBinding.SOAP11HTTP_MTOM_BINDING)
public class ContentServiceImpl implements IContentService {

@Override
    public ContentResponse getFile(ContentRequest req) {
        ContentResponse res = new ContentResponse();
        try {
            File file = this.contentManager.getFile(req);
            DataSource source = new FileDataSource(file);
            DataHandler dataHandler = new DataHandler(source);
            res.setFileData(dataHandler);
            res.setFileName(file.getName());
        } catch (Exception ex) {
        }
        return res;
    }

Весенняя запись


<jaxws:endpoint id="contentService" implementor="#contentServiceImpl"
        address="/contentservice">
        <jaxws:dataBinding>
            <bean class="org.apache.cxf.jaxb.JAXBDataBinding">
                <property name="marshallerProperties">
                    <map>
                        <entry>
                            <key>
                                <value>jaxb.encoding</value>
                            </key>
                            <value>UTF-8</value>
                        </entry>
                    </map>
                </property>
            </bean>
        </jaxws:dataBinding>
        <jaxws:properties>
            <entry key="mtom-enabled" value="true" />
        </jaxws:properties>
    </jaxws:endpoint>

Как вы можете заметить, "mtom-enabled" включен.

Код клиента генерируется с помощью инструмента cxf wsdl2java. Вот пример кода клиента


ContentResponse res =  new ContentResponse();
        try{
            res = getRegisterPort().getFile(req);
            DataHandler dataHandler = res.getFileData();
            if(dataHandler!=null){
                    final InputStream in = dataHandler.getInputStream(); 
                    byte[] bytes = IOUtils.toByteArray(in);
            }
        }catch (Exception ex) {
            LOGGER.error("Error in invoking getContent service",ex);
        }

Вот объект ответа


@XmlAccessorType( XmlAccessType.FIELD )
public class ContentResponse extends ContentServiceResponseBase {

    private String content;
    private String source;
    private String fileName;
    @XmlMimeType("application/octet-stream")
    private DataHandler fileData;
// Getter / Setter ...
}

Я устанавливаю mtom на уровне кода клиента

final BindingProvider bpAdmin = (BindingProvider) port;
        bpAdmin.getRequestContext().put(BindingProvider.ENDPOINT_ADDRESS_PROPERTY, curlUrl);
        SOAPBinding binding = (SOAPBinding)bpAdmin.getBinding();
        binding.setMTOMEnabled(true);

JVM запись выглядит следующим образом:


java -Xms64m -Xmx6144m 

И исключение:

Exception in thread "taskExecutor-12" java.lang.OutOfMemoryError: Java heap space
       at java.util.Arrays.copyOf(Arrays.java:2882)
       at java.lang.AbstractStringBuilder.expandCapacity(AbstractStringBuilder.java:100)
       at java.lang.AbstractStringBuilder.append(AbstractStringBuilder.java:515)
       at java.lang.StringBuilder.append(StringBuilder.java:189)
       at com.sun.xml.internal.bind.v2.runtime.unmarshaller.StAXStreamConnector.handleCharacters(StAXStreamConnector.java:312)
       at com.sun.xml.internal.bind.v2.runtime.unmarshaller.StAXStreamConnector.bridge(StAXStreamConnector.java:176)
       at com.sun.xml.internal.bind.v2.runtime.unmarshaller.UnmarshallerImpl.unmarshal0(UnmarshallerImpl.java:349)
       at com.sun.xml.internal.bind.v2.runtime.BridgeImpl.unmarshal(BridgeImpl.java:109)
       at com.sun.xml.internal.bind.api.Bridge.unmarshal(Bridge.java:222)
       at com.sun.xml.internal.ws.client.sei.ResponseBuilder$DocLit.readResponse(ResponseBuilder.java:514)
       at com.sun.xml.internal.ws.client.sei.SyncMethodHandler.invoke(SyncMethodHandler.java:110)
       at com.sun.xml.internal.ws.client.sei.SyncMethodHandler.invoke(SyncMethodHandler.java:78)
       at com.sun.xml.internal.ws.client.sei.SEIStub.invoke(SEIStub.java:107)
       at $Proxy33.getFile(Unknown Source)

Не уверен, что мне здесь не хватает, любой указатель будет высоко оценен.

  • Спасибо

1 ответ

ОБНОВИТЬ

Не могли бы вы в этом случае попробовать StreamingDataHandler при отправке данных с вызовом readOnce на сервере?

Ссылка, которую вы можете адаптировать, только для части обработчика потоков: http://docs.oracle.com/cd/E14571_01/web.1111/e13734/mtom.htm


Проблема возникает в IOUtils.toByteArray?

Если нет, попытайтесь сузить операцию, в которой выдается исключение, и обновите свой вопрос.

Другие вопросы по тегам