| /* |
| * Licensed to the Apache Software Foundation (ASF) under one or more |
| * contributor license agreements. See the NOTICE file distributed with |
| * this work for additional information regarding copyright ownership. |
| * The ASF licenses this file to You under the Apache License, Version 2.0 |
| * (the "License"); you may not use this file except in compliance with |
| * the License. You may obtain a copy of the License at |
| * |
| * http://www.apache.org/licenses/LICENSE-2.0 |
| * |
| * Unless required by applicable law or agreed to in writing, software |
| * distributed under the License is distributed on an "AS IS" BASIS, |
| * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| * See the License for the specific language governing permissions and |
| * limitations under the License. |
| */ |
| package org.apache.camel.component.salesforce.internal.processor; |
| |
| import java.io.ByteArrayInputStream; |
| import java.io.ByteArrayOutputStream; |
| import java.io.IOException; |
| import java.io.InputStream; |
| import java.io.OutputStreamWriter; |
| import java.io.UnsupportedEncodingException; |
| import java.util.Map; |
| |
| import com.thoughtworks.xstream.XStream; |
| import com.thoughtworks.xstream.XStreamException; |
| import com.thoughtworks.xstream.mapper.CachingMapper; |
| import com.thoughtworks.xstream.mapper.CannotResolveClassException; |
| |
| import org.apache.camel.AsyncCallback; |
| import org.apache.camel.Exchange; |
| import org.apache.camel.Message; |
| import org.apache.camel.component.salesforce.SalesforceEndpoint; |
| import org.apache.camel.component.salesforce.api.SalesforceException; |
| import org.apache.camel.component.salesforce.api.dto.AbstractDTOBase; |
| import org.apache.camel.component.salesforce.api.dto.CreateSObjectResult; |
| import org.apache.camel.component.salesforce.api.dto.GlobalObjects; |
| import org.apache.camel.component.salesforce.api.dto.RestResources; |
| import org.apache.camel.component.salesforce.api.dto.SObjectBasicInfo; |
| import org.apache.camel.component.salesforce.api.dto.SObjectDescription; |
| import org.apache.camel.component.salesforce.api.dto.SearchResults; |
| import org.apache.camel.component.salesforce.api.dto.Versions; |
| import org.apache.camel.component.salesforce.api.dto.approval.ApprovalResult; |
| import org.apache.camel.component.salesforce.api.utils.XStreamUtils; |
| import org.eclipse.jetty.util.StringUtil; |
| |
| import static org.apache.camel.component.salesforce.SalesforceEndpointConfig.SOBJECT_NAME; |
| |
| public class XmlRestProcessor extends AbstractRestProcessor { |
| |
| // although XStream is generally thread safe, because of the way we use |
| // aliases |
| // for GET_BASIC_INFO and GET_DESCRIPTION, we need to use a ThreadLocal |
| // not very efficient when both JSON and XML are used together with a single |
| // Thread pool |
| // but this will do for now |
| private static ThreadLocal<XStream> xStream = new ThreadLocal<XStream>() { |
| @Override |
| protected XStream initialValue() { |
| return XStreamUtils.createXStream(); |
| } |
| }; |
| |
| private static final String RESPONSE_ALIAS = XmlRestProcessor.class.getName() + ".responseAlias"; |
| |
| public XmlRestProcessor(SalesforceEndpoint endpoint) throws SalesforceException { |
| super(endpoint); |
| |
| } |
| |
| @Override |
| protected void processRequest(Exchange exchange) throws SalesforceException { |
| |
| switch (operationName) { |
| case GET_VERSIONS: |
| exchange.setProperty(RESPONSE_CLASS, Versions.class); |
| break; |
| |
| case GET_RESOURCES: |
| exchange.setProperty(RESPONSE_CLASS, RestResources.class); |
| break; |
| |
| case GET_GLOBAL_OBJECTS: |
| // handle in built response types |
| exchange.setProperty(RESPONSE_CLASS, GlobalObjects.class); |
| break; |
| |
| case GET_BASIC_INFO: |
| // handle in built response types |
| exchange.setProperty(RESPONSE_CLASS, SObjectBasicInfo.class); |
| |
| // need to add alias for Salesforce XML that uses SObject name as |
| // root element |
| exchange.setProperty(RESPONSE_ALIAS, getParameter(SOBJECT_NAME, exchange, USE_BODY, NOT_OPTIONAL)); |
| break; |
| |
| case GET_DESCRIPTION: |
| // handle in built response types |
| exchange.setProperty(RESPONSE_CLASS, SObjectDescription.class); |
| |
| // need to add alias for Salesforce XML that uses SObject name as |
| // root element |
| exchange.setProperty(RESPONSE_ALIAS, getParameter(SOBJECT_NAME, exchange, USE_BODY, NOT_OPTIONAL)); |
| break; |
| |
| case GET_SOBJECT: |
| // need to add alias for Salesforce XML that uses SObject name as |
| // root element |
| exchange.setProperty(RESPONSE_ALIAS, getParameter(SOBJECT_NAME, exchange, IGNORE_BODY, NOT_OPTIONAL)); |
| break; |
| |
| case CREATE_SOBJECT: |
| // handle known response type |
| exchange.setProperty(RESPONSE_CLASS, CreateSObjectResult.class); |
| break; |
| |
| case GET_SOBJECT_WITH_ID: |
| // need to add alias for Salesforce XML that uses SObject name as |
| // root element |
| exchange.setProperty(RESPONSE_ALIAS, getParameter(SOBJECT_NAME, exchange, IGNORE_BODY, NOT_OPTIONAL)); |
| break; |
| |
| case UPSERT_SOBJECT: |
| // handle known response type |
| exchange.setProperty(RESPONSE_CLASS, CreateSObjectResult.class); |
| break; |
| |
| case QUERY: |
| case QUERY_ALL: |
| case QUERY_MORE: |
| // need to add alias for Salesforce XML that uses SObject name as |
| // root element |
| exchange.setProperty(RESPONSE_ALIAS, "QueryResult"); |
| break; |
| |
| case SEARCH: |
| // handle known response type |
| exchange.setProperty(RESPONSE_CLASS, SearchResults.class); |
| break; |
| |
| case APEX_CALL: |
| // need to add alias for Salesforce XML that uses SObject name as |
| // root element |
| exchange.setProperty(RESPONSE_ALIAS, "response"); |
| break; |
| |
| case APPROVAL: |
| exchange.setProperty(RESPONSE_CLASS, ApprovalResult.class); |
| break; |
| case APPROVALS: |
| throw new SalesforceException("Fetching of approvals (as of 18.11.2016) with XML format results in HTTP status 500." + " To fetch approvals please use JSON format.", |
| 0); |
| |
| default: |
| // ignore, some operations do not require alias or class exchange |
| // properties |
| } |
| } |
| |
| @Override |
| protected InputStream getRequestStream(Exchange exchange) throws SalesforceException { |
| try { |
| // get request stream from In message |
| Message in = exchange.getIn(); |
| InputStream request = in.getBody(InputStream.class); |
| if (request == null) { |
| AbstractDTOBase dto = in.getBody(AbstractDTOBase.class); |
| if (dto != null) { |
| // marshall the DTO |
| request = getRequestStream(in, dto); |
| } else { |
| // if all else fails, get body as String |
| final String body = in.getBody(String.class); |
| if (null == body) { |
| String msg = "Unsupported request message body " + (in.getBody() == null ? null : in.getBody().getClass()); |
| throw new SalesforceException(msg, null); |
| } else { |
| request = new ByteArrayInputStream(body.getBytes(StringUtil.__UTF8)); |
| } |
| } |
| } |
| return request; |
| } catch (XStreamException e) { |
| String msg = "Error marshaling request: " + e.getMessage(); |
| throw new SalesforceException(msg, e); |
| } catch (UnsupportedEncodingException e) { |
| String msg = "Error marshaling request: " + e.getMessage(); |
| throw new SalesforceException(msg, e); |
| } |
| } |
| |
| @Override |
| protected InputStream getRequestStream(final Message in, final Object object) throws SalesforceException { |
| final XStream localXStream = xStream.get(); |
| // first process annotations on the class, for things like alias, etc. |
| localXStream.processAnnotations(object.getClass()); |
| |
| final ByteArrayOutputStream out = new ByteArrayOutputStream(); |
| // make sure we write the XML with the right encoding |
| try { |
| localXStream.toXML(object, new OutputStreamWriter(out, StringUtil.__UTF8)); |
| } catch (UnsupportedEncodingException e) { |
| String msg = "Error marshaling request: " + e.getMessage(); |
| throw new SalesforceException(msg, e); |
| } |
| |
| return new ByteArrayInputStream(out.toByteArray()); |
| } |
| |
| @Override |
| protected void processResponse(final Exchange exchange, final InputStream responseEntity, final Map<String, String> headers, final SalesforceException exception, |
| final AsyncCallback callback) { |
| final XStream localXStream = xStream.get(); |
| try { |
| final Message out = exchange.getOut(); |
| final Message in = exchange.getIn(); |
| out.copyFromWithNewBody(in, null); |
| out.getHeaders().putAll(headers); |
| |
| if (exception != null) { |
| if (shouldReport(exception)) { |
| exchange.setException(exception); |
| } |
| } else if (responseEntity != null) { |
| // do we need to un-marshal a response |
| final Class<?> responseClass = exchange.getProperty(RESPONSE_CLASS, Class.class); |
| Object response; |
| if (!rawPayload && responseClass != null) { |
| // its ok to call this multiple times, as xstream ignores |
| // duplicate calls |
| localXStream.processAnnotations(responseClass); |
| final String responseAlias = exchange.getProperty(RESPONSE_ALIAS, String.class); |
| if (responseAlias != null) { |
| // extremely dirty, need to flush entire cache if its |
| // holding on to an old alias!!! |
| final CachingMapper mapper = (CachingMapper)localXStream.getMapper(); |
| try { |
| if (mapper.realClass(responseAlias) != responseClass) { |
| mapper.flushCache(); |
| } |
| } catch (CannotResolveClassException ignore) { |
| // recent XStream versions add a |
| // ClassNotFoundException to cache |
| mapper.flushCache(); |
| } |
| localXStream.alias(responseAlias, responseClass); |
| } |
| response = responseClass.newInstance(); |
| localXStream.fromXML(responseEntity, response); |
| } else { |
| // return the response as a stream, for getBlobField |
| response = responseEntity; |
| } |
| out.setBody(response); |
| } |
| } catch (XStreamException e) { |
| String msg = "Error parsing XML response: " + e.getMessage(); |
| exchange.setException(new SalesforceException(msg, e)); |
| } catch (Exception e) { |
| String msg = "Error creating XML response: " + e.getMessage(); |
| exchange.setException(new SalesforceException(msg, e)); |
| } finally { |
| // cleanup temporary exchange headers |
| exchange.removeProperty(RESPONSE_CLASS); |
| exchange.removeProperty(RESPONSE_ALIAS); |
| |
| // consume response entity |
| if (responseEntity != null) { |
| try { |
| responseEntity.close(); |
| } catch (IOException ignored) { |
| } |
| } |
| |
| // notify callback that exchange is done |
| callback.done(false); |
| } |
| } |
| |
| } |