/* * 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.axis2.databinding.utils; import org.apache.axiom.om.OMAbstractFactory; import org.apache.axiom.om.OMAttribute; import org.apache.axiom.om.OMElement; import org.apache.axiom.om.OMFactory; import org.apache.axiom.om.OMNode; import org.apache.axiom.om.OMText; import org.apache.axiom.om.OMXMLBuilderFactory; import org.apache.axiom.soap.SOAPBody; import org.apache.axiom.soap.SOAPEnvelope; import org.apache.axis2.AxisFault; import org.apache.axis2.databinding.typemapping.SimpleTypeMapper; import org.apache.axis2.engine.ObjectSupplier; import javax.xml.namespace.QName; import java.lang.reflect.ParameterizedType; import java.lang.reflect.Type; import java.util.HashMap; import java.util.Iterator; import java.util.Map; public class MultirefHelper { public static final String SOAP12_REF_ATTR = "ref"; public static final String SOAP11_REF_ATTR = "href"; private boolean filledTable; private OMElement parent; private HashMap objectmap = new HashMap(); private HashMap elementMap = new HashMap(); private HashMap omElementMap = new HashMap(); public MultirefHelper(OMElement parent) { this.parent = parent; } public Object getObject(String id) { return objectmap.get(id); } public OMElement getOMElement(String id) { return (OMElement)omElementMap.get(id); } public OMElement processOMElementRef(String id) throws AxisFault { if (!filledTable) { readallChildElements(); } OMElement val = (OMElement)elementMap.get(id); if (val == null) { throw new AxisFault("Invalid reference :" + id); } else { OMElement ele = processElementforRefs(val); OMElement cloneele = elementClone(ele); omElementMap.put(id, cloneele); return cloneele; } } public OMElement processElementforRefs(OMElement elemnts) throws AxisFault { Iterator itr = elemnts.getChildElements(); while (itr.hasNext()) { OMElement omElement = (OMElement)itr.next(); OMAttribute attri = processRefAtt(omElement); if (attri != null) { String ref = getAttvalue(attri); OMElement tempele = getOMElement(ref); if (tempele == null) { tempele = processOMElementRef(ref); } OMElement ele2 = elementClone(tempele); Iterator itrChild = ele2.getChildren(); while (itrChild.hasNext()) { Object obj = itrChild.next(); if (obj instanceof OMNode) { itrChild.remove(); omElement.addChild((OMNode)obj); } } } } return elemnts; } private OMElement elementClone(OMElement ele) { return OMXMLBuilderFactory.createStAXOMBuilder(ele.getXMLStreamReader()).getDocumentElement(); } public Object processRef(Class javatype, String id, ObjectSupplier objectSupplier) throws AxisFault { return processRef(javatype, null, id, objectSupplier); } public Object processRef(Class javatype, Type generictype, String id, ObjectSupplier objectSupplier) throws AxisFault { if (!filledTable) { readallChildElements(); } OMElement val = (OMElement)elementMap.get(id); if (val == null) { throw new AxisFault("Invalid reference :" + id); } else { if (SimpleTypeMapper.isSimpleType(javatype)) { /** * in this case OM element can not contains more child, that is no way to get * the value as an exp , * <refernce id="12"> * <value>foo</value> * </refernce> * the above one is not valid , that should always be like below * <refernce id="12">foo</refernce> */ Object valObj = SimpleTypeMapper.getSimpleTypeObject(javatype, val); objectmap.put(id, valObj); return valObj; } else if (generictype != null && SimpleTypeMapper.isCollection(javatype)) { return BeanUtil.processGenericCollection(val.getFirstElement(), generictype, this, objectSupplier); } else if (generictype != null && SimpleTypeMapper.isMap(javatype)) { Type[] parameterArgTypes = {Object.class, Object.class}; if (generictype instanceof ParameterizedType) { ParameterizedType aType = (ParameterizedType) generictype; parameterArgTypes = aType.getActualTypeArguments(); } return BeanUtil.processGenericsMapElement(parameterArgTypes, val.getFirstElement(), this, val.getChildren(), objectSupplier, generictype); } else { Object obj = BeanUtil.deserialize(javatype, val, this, objectSupplier); objectmap.put(id, obj); return obj; } } } private void readallChildElements() { Iterator childs = parent.getChildElements(); while (childs.hasNext()) { OMElement omElement = (OMElement)childs.next(); OMAttribute id = omElement.getAttribute(new QName("id")); if (id != null) { childs.remove(); elementMap.put(id.getAttributeValue(), omElement); } } filledTable = true; } public static String getAttvalue(OMAttribute omatribute) { String ref; ref = omatribute.getAttributeValue(); if (ref != null) { if (ref.charAt(0) == '#') { ref = ref.substring(1); } } return ref; } public static OMAttribute processRefAtt(OMElement omElement) { OMAttribute omatribute = omElement.getAttribute(new QName(SOAP11_REF_ATTR)); if (omatribute == null) { omatribute = omElement.getAttribute(new QName(SOAP12_REF_ATTR)); } return omatribute; } public void clean() { elementMap.clear(); objectmap.clear(); } /** * this method is used to process the href attributes which may comes with the incomming soap mesaage * <soap:body> * <operation> * <arg1 href="#obj1"/> * </operation> * <multiref id="obj1"> * <name>the real argument</name> * <color>blue</color> * </multiref> * </soap:body> * here we assume first child of the soap body has the main object structure and others contain the * multiref parts. * Soap spec says that those multiref parts must be top level elements. * * @param soapEnvelope */ public static void processHrefAttributes(SOAPEnvelope soapEnvelope) throws AxisFault { // first populate the multiref parts to a hash table. SOAPBody soapBody = soapEnvelope.getBody(); // first build the whole tree soapBody.build(); OMElement omElement = null; OMAttribute idAttribute = null; Map idAndOMElementMap = new HashMap(); for (Iterator iter = soapBody.getChildElements(); iter.hasNext();) { omElement = (OMElement) iter.next(); // the attribute id is an unqualified attribute idAttribute = omElement.getAttribute(new QName(null, "id")); if (idAttribute != null) { // for the first element there may not have an id idAndOMElementMap.put(idAttribute.getAttributeValue(), omElement); } } // start processing from the first child processHrefAttributes(idAndOMElementMap, soapBody.getFirstElement(), OMAbstractFactory.getOMFactory()); } public static void processHrefAttributes(Map idAndOMElementMap, OMElement elementToProcess, OMFactory omFactory) throws AxisFault { // first check whether this element has an href value. // href is also an unqualifed attribute OMAttribute hrefAttribute = elementToProcess.getAttribute(new QName(null, "href")); if (hrefAttribute != null) { // i.e this has an href attribute String hrefAttributeValue = hrefAttribute.getAttributeValue(); if (!hrefAttributeValue.startsWith("#")) { throw new AxisFault("In valid href ==> " + hrefAttributeValue + " does not starts with #"); } else { OMElement referedOMElement = (OMElement) idAndOMElementMap.get(hrefAttributeValue.substring(1)); if (referedOMElement == null) { throw new AxisFault("In valid href ==> " + hrefAttributeValue + " can not find" + "the matching element"); } else { // now we have to remove the hrefAttribute and add all the child elements to the // element being proccesed elementToProcess.removeAttribute(hrefAttribute); OMElement clonedReferenceElement = getClonedOMElement(referedOMElement, omFactory); OMNode omNode = null; for (Iterator iter = clonedReferenceElement.getChildren(); iter.hasNext();) { omNode = (OMNode) iter.next(); iter.remove(); elementToProcess.addChild(omNode); } // add attributes OMAttribute omAttribute = null; for (Iterator iter = clonedReferenceElement.getAllAttributes(); iter.hasNext();) { omAttribute = (OMAttribute) iter.next(); // we do not have to populate the id attribute if (!omAttribute.getLocalName().equals("id")) { elementToProcess.addAttribute(omAttribute); } } } } } // call recursively to proces all elements OMElement childOMElement = null; for (Iterator iter = elementToProcess.getChildElements(); iter.hasNext();) { childOMElement = (OMElement) iter.next(); processHrefAttributes(idAndOMElementMap, childOMElement, omFactory); } } /** * returns an cloned om element for this OMElement * * @param omElement * @return cloned omElement */ public static OMElement getClonedOMElement(OMElement omElement, OMFactory omFactory) throws AxisFault { OMElement newOMElement = omFactory.createOMElement(omElement.getQName()); // copying attributes OMAttribute omAttribute = null; OMAttribute newOMAttribute = null; for (Iterator iter = omElement.getAllAttributes(); iter.hasNext();) { omAttribute = (OMAttribute) iter.next(); if (!omAttribute.getAttributeValue().equals("id")) { newOMAttribute = omFactory.createOMAttribute( omAttribute.getLocalName(), omAttribute.getNamespace(), omAttribute.getAttributeValue()); newOMElement.addAttribute(newOMAttribute); } } OMNode omNode = null; OMText omText = null; for (Iterator iter = omElement.getChildren(); iter.hasNext();) { omNode = (OMNode) iter.next(); if (omNode instanceof OMText) { omText = (OMText) omNode; newOMElement.addChild(omFactory.createOMText(omText.getText())); } else if (omNode instanceof OMElement) { newOMElement.addChild(getClonedOMElement((OMElement) omNode, omFactory)); } else { throw new AxisFault("Unknown child element type ==> " + omNode.getClass().getName()); } } return newOMElement; } }