com.jaspersoft.ireport.designer.data.fieldsproviders.olap.OLAPQueryExecuter.java Source code

Java tutorial

Introduction

Here is the source code for com.jaspersoft.ireport.designer.data.fieldsproviders.olap.OLAPQueryExecuter.java

Source

/*
 * iReport - Visual Designer for JasperReports.
 * Copyright (C) 2002 - 2009 Jaspersoft Corporation. All rights reserved.
 * http://www.jaspersoft.com
 *
 * Unless you have purchased a commercial license agreement from Jaspersoft,
 * the following license terms apply:
 *
 * This program is part of iReport.
 *
 * iReport is free software: you can redistribute it and/or modify
 * it under the terms of the GNU Affero General Public License as published by
 * the Free Software Foundation, either version 3 of the License, or
 * (at your option) any later version.
 *
 * iReport is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
 * GNU Affero General Public License for more details.
 *
 * You should have received a copy of the GNU Affero General Public License
 * along with iReport. If not, see <http://www.gnu.org/licenses/>.
 */
package com.jaspersoft.ireport.designer.data.fieldsproviders.olap;

import bsh.Interpreter;
import com.jaspersoft.ireport.designer.IReportConnection;
import com.jaspersoft.ireport.designer.IReportManager;
import com.jaspersoft.ireport.designer.connection.JRXMLADataSourceConnection;
import com.jaspersoft.ireport.designer.connection.MondrianConnection;
import com.jaspersoft.ireport.designer.data.ReportQueryDialog;
import com.jaspersoft.ireport.designer.utils.Misc;
import java.net.MalformedURLException;
import java.net.URL;
import java.util.HashMap;
import java.util.Iterator;
import java.util.List;
import java.util.Map;
import java.util.Set;
import java.util.regex.Matcher;
import java.util.regex.Pattern;
import javax.xml.soap.MessageFactory;
import javax.xml.soap.MimeHeaders;
import javax.xml.soap.Name;
import javax.xml.soap.SOAPBody;
import javax.xml.soap.SOAPConnection;
import javax.xml.soap.SOAPConnectionFactory;
import javax.xml.soap.SOAPElement;
import javax.xml.soap.SOAPEnvelope;
import javax.xml.soap.SOAPException;
import javax.xml.soap.SOAPFactory;
import javax.xml.soap.SOAPFault;
import javax.xml.soap.SOAPMessage;
import javax.xml.soap.SOAPPart;
import mondrian.olap.Connection;
import mondrian.olap.Hierarchy;
import mondrian.olap.Level;
import mondrian.olap.Member;
import mondrian.olap.Query;
import mondrian.olap.QueryAxis;
import net.sf.jasperreports.engine.JRException;
import net.sf.jasperreports.engine.JRRuntimeException;
import net.sf.jasperreports.engine.design.JRDesignParameter;
import net.sf.jasperreports.olap.result.JROlapResult;
import net.sf.jasperreports.olap.xmla.JRXmlaCell;
import net.sf.jasperreports.olap.xmla.JRXmlaHierarchy;
import net.sf.jasperreports.olap.xmla.JRXmlaMember;
import net.sf.jasperreports.olap.xmla.JRXmlaMemberTuple;
import net.sf.jasperreports.olap.xmla.JRXmlaResult;
import net.sf.jasperreports.olap.xmla.JRXmlaResultAxis;
import org.apache.commons.logging.Log;
import org.apache.commons.logging.LogFactory;

/**
 *
 * @author gtoffoli
 */
public class OLAPQueryExecuter {

    private static String soapMessageFactoryClass = null;
    private static String soapConnectionFactoryClass = null;

    public static void setAxisSOAPClientConfig() {
        try {
            if (soapMessageFactoryClass == null) {
                soapMessageFactoryClass = System.getProperty("javax.xml.soap.MessageFactory");
                if (soapMessageFactoryClass == null) {
                    soapMessageFactoryClass = MessageFactory.newInstance().getClass().getName();
                }
            }
        } catch (SOAPException ex) {
        }

        try {
            if (soapConnectionFactoryClass == null) {
                soapConnectionFactoryClass = System.getProperty("javax.xml.soap.SOAPConnectionFactory");
                if (soapConnectionFactoryClass == null) {
                    soapConnectionFactoryClass = SOAPConnectionFactory.newInstance().getClass().getName();
                }
            }
        } catch (SOAPException ex) {
        }

        System.setProperty("javax.xml.soap.MessageFactory", "org.apache.axis.soap.MessageFactoryImpl");
        System.setProperty("javax.xml.soap.SOAPConnectionFactory",
                "org.apache.axis.soap.SOAPConnectionFactoryImpl");
    }

    public static void restoreSOAPClientConfig() {
        if (soapMessageFactoryClass != null) {
            System.setProperty("javax.xml.soap.MessageFactory", soapMessageFactoryClass);
        }

        if (soapConnectionFactoryClass != null) {
            System.setProperty("javax.xml.soap.SOAPConnectionFactory", soapConnectionFactoryClass);
        }

    }

    private static final Log log = LogFactory.getLog(OLAPQueryExecuter.class);

    private static final String SLICER_AXIS_NAME = "SlicerAxis";
    private static final String MDD_URI = "urn:schemas-microsoft-com:xml-analysis:mddataset";
    private static final String XMLA_URI = "urn:schemas-microsoft-com:xml-analysis";

    private static final Pattern LEVEL_UNIQUE_NAME_PATTERN = Pattern.compile("\\[[^\\]]+\\]\\.\\[([^\\]]+)\\]");
    private static final int LEVEL_UNIQUE_NAME_PATTERN_NAME_GROUP = 1;

    private SOAPFactory sf;
    private SOAPConnection connection;
    private JROlapResult xmlaResult;

    private Interpreter interpreter = null;
    private List reportParameters = null;
    private String queryString = "";
    private HashMap queryParameters = new HashMap();

    /** Creates a new instance of HQLFieldsReader */
    public OLAPQueryExecuter(String queryStr, List reportParameters) {

        this.setQueryString(queryStr);
        this.setReportParameters(reportParameters);

    }

    public void prepareQuery() throws Exception {
        Iterator iterParams = getReportParameters().iterator();

        while (iterParams.hasNext()) {

            JRDesignParameter param = (JRDesignParameter) iterParams.next();
            String parameterName = param.getName();

            if (queryString.indexOf("$P!{" + parameterName + "}") > 0) {
                String expStr = (param.getDefaultValueExpression() == null) ? ""
                        : param.getDefaultValueExpression().getText();
                Object paramVal = ReportQueryDialog.recursiveInterpreter(getInterpreter(), expStr,
                        getReportParameters());

                if (paramVal == null) {
                    paramVal = "";
                }

                queryString = Misc.string_replace("" + paramVal, "$P!{" + parameterName + "}", queryString);
            } else if (getQueryString().indexOf("$P{" + parameterName + "}") > 0) {
                String expStr = (param.getDefaultValueExpression() == null) ? ""
                        : param.getDefaultValueExpression().getText();
                Object paramVal = ReportQueryDialog.recursiveInterpreter(getInterpreter(), expStr,
                        getReportParameters());

                if (paramVal == null) {
                    paramVal = "";
                }

                queryString = Misc.string_replace("" + paramVal, "$P!{" + parameterName + "}", queryString);
            }
        }
    }

    private Interpreter prepareExpressionEvaluator() throws bsh.EvalError {

        Interpreter interpreter1 = new Interpreter();
        interpreter1.setClassLoader(Thread.currentThread().getContextClassLoader());
        return interpreter1;
    }

    public Interpreter getInterpreter() {

        if (interpreter == null) {
            try {
                interpreter = prepareExpressionEvaluator();
            } catch (Exception ex) {

            }
        }
        return interpreter;
    }

    public void setInterpreter(Interpreter interpreter) {
        this.interpreter = interpreter;
    }

    public List getReportParameters() {
        return reportParameters;
    }

    public void setReportParameters(List reportParameters) {
        this.reportParameters = reportParameters;
    }

    public String getQueryString() {
        return queryString;
    }

    public void setQueryString(String queryString) {
        this.queryString = queryString;
    }

    public HashMap getQueryParameters() {
        return queryParameters;
    }

    public void setQueryParameters(HashMap queryParameters) {
        this.queryParameters = queryParameters;
    }

    public JROlapResult createOlapResult() throws Exception {
        prepareQuery();

        try {
            IReportConnection conn = IReportManager.getInstance().getDefaultConnection();
            if (conn instanceof MondrianConnection) {
                Connection mconn = ((MondrianConnection) conn).getMondrianConnection();
                if (mconn == null) {
                    throw new Exception("The supplied mondrian.olap.Connection object is null.");
                }

                Query query = mconn.parseQuery(queryString);

                xmlaResult = new IROlapResult();

                // create result from a Mondrian Query...
                parseQuery(query);
                return xmlaResult;
                //Result result = mconn.execute(query);
                //return new JRMondrianResult(result);
            } else if (conn instanceof JRXMLADataSourceConnection) {
                JRXMLADataSourceConnection xmlaConnection = ((JRXMLADataSourceConnection) conn);
                xmlaResult = new JRXmlaResult();

                try {
                    setAxisSOAPClientConfig();

                    this.sf = SOAPFactory.newInstance();
                    this.connection = createSOAPConnection();
                    SOAPMessage queryMessage = createQueryMessage(xmlaConnection);

                    queryMessage.writeTo(System.out);
                    System.out.println();

                    URL soapURL = new URL(getSoapUrl(xmlaConnection));
                    System.out.println("URL: " + soapURL);

                    System.out.flush();

                    SOAPMessage resultMessage = executeQuery(queryMessage, soapURL);

                    parseResult(resultMessage);
                } catch (MalformedURLException e) {
                    log.error(e);
                    throw new JRRuntimeException(e);
                } catch (SOAPException e) {
                    log.error(e);
                    throw new JRRuntimeException(e);
                } finally {
                    restoreSOAPClientConfig();
                }

                return xmlaResult;
            } else {
                throw new Exception(
                        "The supplied Connection is not an OLAP connection. An XML/A or Mondrian connection is required.");
            }

        } catch (Exception ex) {
            ex.printStackTrace();
            throw ex;
        } finally {

        }
    }

    protected void parseQuery(Query query) {

        QueryAxis[] axes = query.getAxes();
        for (QueryAxis axis : axes) {
            IROlapResultAxis jrAxis = new IROlapResultAxis();
            ((IROlapResult) xmlaResult).addAxis(jrAxis);

            Hierarchy[] hierarchies = query.getMdxHierarchiesOnAxis(axis.getAxisOrdinal());
            for (Hierarchy hierarchy : hierarchies) {
                boolean addMeasures = false;
                IROlapHierarchy jrHierarchy = new IROlapHierarchy(hierarchy.getName());
                if (hierarchy.getName().equals("Measures")) {
                    addMeasures = true;
                }
                jrHierarchy.setHierarchyUniqueName(hierarchy.getUniqueName());
                Level[] levels = hierarchy.getLevels();
                for (Level level : levels) {
                    IROlapHierarchyLevel jrLevel = new IROlapHierarchyLevel(level.getName(), level.getDepth());

                    jrHierarchy.addHierarchyLevel(jrLevel);
                }

                jrAxis.addHierarchy(jrHierarchy);

                // Add axis tubles for measures...
                if (addMeasures) {
                    Set<Member> memberSet = query.getMeasuresMembers();
                    IROlapMemberTuple tuple = new IROlapMemberTuple();

                    Iterator iter = memberSet.iterator();
                    while (iter.hasNext()) {
                        Member m = (Member) iter.next();
                        JRXmlaMember member = new JRXmlaMember(m.getName(), m.getUniqueName(),
                                m.getDimension().getName(), m.getLevel().getName(), m.getDepth());
                        tuple.addMember(member);

                    }

                    jrAxis.addTuple(tuple);
                }
            }

        }

    }

    protected String getSoapUrl(JRXMLADataSourceConnection xmlaConnection) throws MalformedURLException {
        String soapUrl;
        String xmlaUrl = xmlaConnection.getUrl();
        String user = xmlaConnection.getUsername();
        if (user == null || user.length() == 0) {
            soapUrl = xmlaUrl;
        } else {
            URL url = new URL(xmlaUrl);
            soapUrl = url.getProtocol() + "://" + user;

            String password = xmlaConnection.getPassword();
            if (password != null && password.length() > 0) {
                soapUrl += ":" + password;
            }

            soapUrl += "@" + url.getHost() + ":" + url.getPort() + url.getPath();
        }
        return soapUrl;
    }

    public boolean cancelQuery() throws JRException {
        return false;
    }

    public void close() {
        if (connection != null) {
            try {
                connection.close();
            } catch (SOAPException e) {
                log.error(e);
                throw new JRRuntimeException(e);
            }
            connection = null;
        }
    }

    protected SOAPConnection createSOAPConnection() {
        try {
            SOAPConnectionFactory scf = SOAPConnectionFactory.newInstance();
            SOAPConnection soapConnection = scf.createConnection();
            return soapConnection;
        } catch (UnsupportedOperationException e) {
            log.error(e);
            throw new JRRuntimeException(e);
        } catch (SOAPException e) {
            log.error(e);
            throw new JRRuntimeException(e);
        }
    }

    protected SOAPMessage createQueryMessage(JRXMLADataSourceConnection xmlaConnection) {
        String queryStr = getQueryString();

        if (log.isDebugEnabled()) {
            log.debug("MDX query: " + queryStr);
        }

        try {
            // Force the use of Axis as message factory...

            MessageFactory mf = MessageFactory.newInstance();

            SOAPMessage message = mf.createMessage();

            MimeHeaders mh = message.getMimeHeaders();
            mh.setHeader("SOAPAction", "\"urn:schemas-microsoft-com:xml-analysis:Execute\"");
            //mh.setHeader("Content-Type", "text/xml; charset=utf-8");

            SOAPPart soapPart = message.getSOAPPart();
            SOAPEnvelope envelope = soapPart.getEnvelope();
            SOAPBody body = envelope.getBody();
            Name nEx = envelope.createName("Execute", "", XMLA_URI);

            SOAPElement eEx = body.addChildElement(nEx);

            // add the parameters

            // COMMAND parameter
            // <Command>
            // <Statement>queryStr</Statement>
            // </Command>
            Name nCom = envelope.createName("Command", "", XMLA_URI);
            SOAPElement eCommand = eEx.addChildElement(nCom);
            Name nSta = envelope.createName("Statement", "", XMLA_URI);
            SOAPElement eStatement = eCommand.addChildElement(nSta);
            eStatement.addTextNode(queryStr);

            // <Properties>
            // <PropertyList>
            // <DataSourceInfo>dataSource</DataSourceInfo>
            // <Catalog>catalog</Catalog>
            // <Format>Multidimensional</Format>
            // <AxisFormat>TupleFormat</AxisFormat>
            // </PropertyList>
            // </Properties>
            Map paraList = new HashMap();
            String datasource = xmlaConnection.getDatasource();
            paraList.put("DataSourceInfo", datasource);
            String catalog = xmlaConnection.getCatalog();
            paraList.put("Catalog", catalog);
            paraList.put("Format", "Multidimensional");
            paraList.put("AxisFormat", "TupleFormat");
            addParameterList(envelope, eEx, "Properties", "PropertyList", paraList);
            message.saveChanges();

            if (log.isDebugEnabled()) {
                log.debug("XML/A query message: " + message.toString());
            }

            return message;
        } catch (SOAPException e) {
            log.error(e);
            throw new JRRuntimeException(e);
        }
    }

    protected void addParameterList(SOAPEnvelope envelope, SOAPElement eParent, String typeName, String listName,
            Map params) throws SOAPException {
        Name nPara = envelope.createName(typeName, "", XMLA_URI);
        SOAPElement eType = eParent.addChildElement(nPara);
        nPara = envelope.createName(listName, "", XMLA_URI);
        SOAPElement eList = eType.addChildElement(nPara);
        if (params == null)
            return;

        Iterator it = params.keySet().iterator();
        while (it.hasNext()) {
            String tag = (String) it.next();
            String value = (String) params.get(tag);
            nPara = envelope.createName(tag, "", XMLA_URI);
            SOAPElement eTag = eList.addChildElement(nPara);
            eTag.addTextNode(value);
        }
    }

    /**
     * Sends the SOAP Message over the connection and returns the
     * Result-SOAP-Message
     *
     * @return Reply-Message
     */
    protected SOAPMessage executeQuery(SOAPMessage message, URL url) {
        try {

            SOAPMessage soapResult = connection.call(message, url);
            return soapResult;
        } catch (SOAPException e) {
            log.error("Message-Call failed.", e);
            throw new JRRuntimeException(e);
        }
    }

    /**
     * Parses the result-Message into this class's structure
     *
     * @param reply
     *            The reply-Message from the Server
     */
    protected void parseResult(SOAPMessage reply) throws SOAPException, JRRuntimeException {
        SOAPPart soapPart = reply.getSOAPPart();
        SOAPEnvelope soapEnvelope = soapPart.getEnvelope();
        SOAPBody soapBody = soapEnvelope.getBody();
        SOAPElement eElement = null;

        if (log.isDebugEnabled()) {
            log.debug("XML/A result envelope: " + soapEnvelope.toString());
        }

        SOAPFault fault = soapBody.getFault();
        if (fault != null) {
            handleResultFault(fault);
        }

        Name eName = soapEnvelope.createName("ExecuteResponse", "", XMLA_URI);

        // Get the ExecuteResponse-Node
        Iterator responseElements = soapBody.getChildElements(eName);
        if (responseElements.hasNext()) {
            Object eObj = responseElements.next();
            if (eObj == null) {
                log.error("ExecuteResponse Element is null.");
                throw new JRRuntimeException("ExecuteResponse Element is null.");
            }
            eElement = (SOAPElement) eObj;
        } else {
            log.error("Could not retrieve ExecuteResponse Element.");
            throw new JRRuntimeException("Could not retrieve ExecuteResponse Element.");
        }

        // Get the return-Node
        Name rName = soapEnvelope.createName("return", "", XMLA_URI);
        Iterator returnElements = eElement.getChildElements(rName);
        SOAPElement returnElement = null;
        if (returnElements.hasNext()) {
            Object eObj = returnElements.next();
            if (eObj == null) {
                log.error("return Element is null.");
                throw new JRRuntimeException("return Element is null.");
            }
            returnElement = (SOAPElement) eObj;
        } else {
            // Should be old-Microsoft XMLA-SDK. Try without m-prefix
            Name rName2 = soapEnvelope.createName("return", "", "");
            returnElements = eElement.getChildElements(rName2);
            if (returnElements.hasNext()) {
                Object eObj = returnElements.next();
                if (eObj == null) {
                    log.error("return Element is null.");
                    throw new JRRuntimeException("return Element is null.");
                }
                returnElement = (SOAPElement) eObj;
            } else {
                log.error("Could not retrieve return Element.");
                throw new JRRuntimeException("Could not retrieve return Element.");
            }
        }

        // Get the root-Node
        Name rootName = soapEnvelope.createName("root", "", MDD_URI);
        SOAPElement rootElement = null;
        Iterator rootElements = returnElement.getChildElements(rootName);
        if (rootElements.hasNext()) {
            Object eObj = rootElements.next();
            if (eObj == null) {
                log.error("root Element is null.");
                throw new JRRuntimeException("root Element is null.");
            }
            rootElement = (SOAPElement) eObj;
        } else {
            log.error("Could not retrieve root Element.");
            throw new JRRuntimeException("Could not retrieve root Element.");
        }
        // Get the OlapInfo-Node
        Name olapInfoName = soapEnvelope.createName("OlapInfo", "", MDD_URI);
        SOAPElement olapInfoElement = null;
        Iterator olapInfoElements = rootElement.getChildElements(olapInfoName);
        if (olapInfoElements.hasNext()) {
            Object eObj = olapInfoElements.next();
            if (eObj == null) {
                log.error("OlapInfo Element is null.");
                throw new JRRuntimeException("OlapInfo Element is null.");
            }
            olapInfoElement = (SOAPElement) eObj;
        } else {
            log.error("Could not retrieve OlapInfo Element.");
            throw new JRRuntimeException("Could not retrieve OlapInfo Element.");
        }

        parseOLAPInfoElement(olapInfoElement);

        // Get the Axes Element
        Name axesName = soapEnvelope.createName("Axes", "", MDD_URI);
        SOAPElement axesElement = null;
        Iterator axesElements = rootElement.getChildElements(axesName);
        if (axesElements.hasNext()) {
            Object eObj = axesElements.next();
            if (eObj == null) {
                log.error("Axes Element is null");
                throw new JRRuntimeException("Axes Element is null");
            }
            axesElement = (SOAPElement) eObj;
        } else {
            log.error("Could not retrieve Axes Element.");
            throw new JRRuntimeException("Could not retrieve Axes Element.");
        }

        parseAxesElement(axesElement);

        // Get the CellData Element
        Name cellDataName = soapEnvelope.createName("CellData", "", MDD_URI);
        SOAPElement cellDataElement = null;
        Iterator cellDataElements = rootElement.getChildElements(cellDataName);
        if (cellDataElements.hasNext()) {
            Object eObj = cellDataElements.next();
            if (eObj == null) {
                log.error("CellData element is null");
                throw new JRRuntimeException("CellData element is null");
            }
            cellDataElement = (SOAPElement) eObj;
        } else {
            log.error("Could not retrieve CellData Element.");
            throw new JRRuntimeException("Could not retrieve CellData Element.");
        }
        parseCellDataElement(cellDataElement);
    }

    protected void handleResultFault(SOAPFault fault) {
        StringBuffer errorMsg = new StringBuffer();
        errorMsg.append("XML/A fault: ");

        String faultString = fault.getFaultString();
        if (faultString != null) {
            errorMsg.append(faultString);
            errorMsg.append("; ");
        }

        String faultActor = fault.getFaultActor();
        if (faultActor != null) {
            errorMsg.append("Actor: ");
            errorMsg.append(faultActor);
            errorMsg.append("; ");
        }

        String faultCode = fault.getFaultCode();
        if (faultCode != null) {
            errorMsg.append("Code: ");
            errorMsg.append(faultCode);
            errorMsg.append("; ");
        }

        throw new JRRuntimeException(errorMsg.toString());
    }

    protected void parseOLAPInfoElement(SOAPElement olapInfoElement) throws SOAPException {
        // CubeInfo-Element is not needed

        // Get the AxesInfo-Node
        Name axesInfoName = sf.createName("AxesInfo", "", MDD_URI);
        SOAPElement axesElement = null;
        Iterator axesInfoElements = olapInfoElement.getChildElements(axesInfoName);
        if (axesInfoElements.hasNext()) {
            Object axesObj = axesInfoElements.next();
            if (axesObj == null) {
                log.error("AxisInfo Element is null.");
                throw new JRRuntimeException("AxisInfo Element is null.");
            }
            axesElement = (SOAPElement) axesObj;
        } else {
            log.error("Could not retrieve AxesInfo Element.");
            throw new JRRuntimeException("Could not retrieve AxesInfo Element.");
        }

        parseAxesInfoElement(axesElement);

        // CellInfo is not needed
    }

    protected void parseAxesInfoElement(SOAPElement axesInfoElement) throws SOAPException {
        // Cycle over AxisInfo-Elements
        Name axisInfoName = sf.createName("AxisInfo", "", MDD_URI);
        Iterator itAxis = axesInfoElement.getChildElements(axisInfoName);
        while (itAxis.hasNext()) {
            SOAPElement axisElement = (SOAPElement) itAxis.next();
            Name name = sf.createName("name");
            String axisName = axisElement.getAttributeValue(name);
            if (axisName.equals(SLICER_AXIS_NAME)) {
                continue;
            }

            JRXmlaResultAxis axis = new JRXmlaResultAxis(axisName);
            ((JRXmlaResult) xmlaResult).addAxis(axis);

            // retrieve the hierarchies by <HierarchyInfo>
            name = sf.createName("HierarchyInfo", "", MDD_URI);
            Iterator itHierInfo = axisElement.getChildElements(name);
            while (itHierInfo.hasNext()) {
                SOAPElement eHierInfo = (SOAPElement) itHierInfo.next();
                handleHierInfo(axis, eHierInfo);
            }
        }
    }

    protected void parseAxesElement(SOAPElement axesElement) throws SOAPException {
        // Cycle over Axis-Elements
        Name aName = sf.createName("Axis", "", MDD_URI);
        Iterator itAxis = axesElement.getChildElements(aName);
        while (itAxis.hasNext()) {
            SOAPElement axisElement = (SOAPElement) itAxis.next();
            Name name = sf.createName("name");
            String axisName = axisElement.getAttributeValue(name);

            if (axisName.equals(SLICER_AXIS_NAME)) {
                continue;
            }

            // LookUp for the Axis
            JRXmlaResultAxis axis = ((JRXmlaResult) xmlaResult).getAxisByName(axisName);

            // retrieve the tuples by <Tuples>
            name = sf.createName("Tuples", "", MDD_URI);
            Iterator itTuples = axisElement.getChildElements(name);
            if (itTuples.hasNext()) {
                SOAPElement eTuples = (SOAPElement) itTuples.next();
                handleTuplesElement(axis, eTuples);
            }
        }
    }

    protected void parseCellDataElement(SOAPElement cellDataElement) throws SOAPException {
        Name name = sf.createName("Cell", "", MDD_URI);
        Iterator itCells = cellDataElement.getChildElements(name);
        while (itCells.hasNext()) {
            SOAPElement cellElement = (SOAPElement) itCells.next();

            Name errorName = sf.createName("Error", "", MDD_URI);
            Iterator errorElems = cellElement.getChildElements(errorName);
            if (errorElems.hasNext()) {
                handleCellErrors(errorElems);
            }

            Name ordinalName = sf.createName("CellOrdinal");
            String cellOrdinal = cellElement.getAttributeValue(ordinalName);

            Object value = null;
            Iterator valueElements = cellElement.getChildElements(sf.createName("Value", "", MDD_URI));
            if (valueElements.hasNext()) {
                SOAPElement valueElement = (SOAPElement) valueElements.next();
                String valueType = valueElement.getAttribute("xsi:type");
                if (valueType.equals("xsd:int"))
                    value = new Long(valueElement.getValue());
                else if (valueType.equals("xsd:double"))
                    value = new Double(valueElement.getValue());
                else if (valueType.equals("xsd:decimal"))
                    value = new Double(valueElement.getValue());
                else
                    value = valueElement.getValue();
            }

            String fmtValue = "";
            Iterator fmtValueElements = cellElement.getChildElements(sf.createName("FmtValue", "", MDD_URI));
            if (fmtValueElements.hasNext()) {
                SOAPElement fmtValueElement = ((SOAPElement) fmtValueElements.next());
                fmtValue = fmtValueElement.getValue();
            }

            int pos = Integer.parseInt(cellOrdinal);
            JRXmlaCell cell = new JRXmlaCell(value, fmtValue);
            ((JRXmlaResult) xmlaResult).setCell(cell, pos);
        }
    }

    protected void handleCellErrors(Iterator errorElems) throws SOAPException {
        SOAPElement errorElem = (SOAPElement) errorElems.next();

        StringBuffer errorMsg = new StringBuffer();
        errorMsg.append("Cell error: ");

        Iterator descriptionElems = errorElem.getChildElements(sf.createName("Description", "", MDD_URI));
        if (descriptionElems.hasNext()) {
            SOAPElement descrElem = (SOAPElement) descriptionElems.next();
            errorMsg.append(descrElem.getValue());
            errorMsg.append("; ");
        }

        Iterator sourceElems = errorElem.getChildElements(sf.createName("Source", "", MDD_URI));
        if (sourceElems.hasNext()) {
            SOAPElement sourceElem = (SOAPElement) sourceElems.next();
            errorMsg.append("Source: ");
            errorMsg.append(sourceElem.getValue());
            errorMsg.append("; ");
        }

        Iterator codeElems = errorElem.getChildElements(sf.createName("ErrorCode", "", MDD_URI));
        if (codeElems.hasNext()) {
            SOAPElement codeElem = (SOAPElement) codeElems.next();
            errorMsg.append("Code: ");
            errorMsg.append(codeElem.getValue());
            errorMsg.append("; ");
        }

        throw new JRRuntimeException(errorMsg.toString());
    }

    protected void handleHierInfo(JRXmlaResultAxis axis, SOAPElement hierInfoElement) throws SOAPException {
        Name name = sf.createName("name");
        String dimName = hierInfoElement.getAttributeValue(name); // Get the Dimension Name

        JRXmlaHierarchy hier = new JRXmlaHierarchy(dimName);
        axis.addHierarchy(hier);
    }

    protected void handleTuplesElement(JRXmlaResultAxis axis, SOAPElement tuplesElement) throws SOAPException {
        Name tName = sf.createName("Tuple", "", MDD_URI);
        for (Iterator itTuple = tuplesElement.getChildElements(tName); itTuple.hasNext();) {
            SOAPElement eTuple = (SOAPElement) itTuple.next();
            handleTupleElement(axis, eTuple);
        }
    }

    protected void handleTupleElement(JRXmlaResultAxis axis, SOAPElement tupleElement) throws SOAPException {
        JRXmlaMemberTuple tuple = new JRXmlaMemberTuple(axis.getHierarchiesOnAxis().length);

        Name memName = sf.createName("Member", "", MDD_URI);
        Iterator itMember = tupleElement.getChildElements(memName);
        int memNum = 0;
        while (itMember.hasNext()) {
            SOAPElement memElement = (SOAPElement) itMember.next();

            Name name = sf.createName("Hierarchy", "", "");
            String hierName = memElement.getAttributeValue(name);

            String uName = "";
            Iterator uNameElements = memElement.getChildElements(sf.createName("UName", "", MDD_URI));
            if (uNameElements.hasNext())
                uName = ((SOAPElement) uNameElements.next()).getValue();

            String caption = "";
            Iterator captionElements = memElement.getChildElements(sf.createName("Caption", "", MDD_URI));
            if (captionElements.hasNext())
                caption = ((SOAPElement) captionElements.next()).getValue();

            String lName = "";
            Iterator lNameElements = memElement.getChildElements(sf.createName("LName", "", MDD_URI));
            if (lNameElements.hasNext()) {
                String levelUniqueName = ((SOAPElement) lNameElements.next()).getValue();
                Matcher matcher = LEVEL_UNIQUE_NAME_PATTERN.matcher(levelUniqueName);
                if (matcher.matches()) {
                    lName = matcher.group(LEVEL_UNIQUE_NAME_PATTERN_NAME_GROUP);
                }
            }

            int lNum = 0;
            Iterator lNumElements = memElement.getChildElements(sf.createName("LNum", "", MDD_URI));
            if (lNumElements.hasNext())
                lNum = Integer.parseInt(((SOAPElement) lNumElements.next()).getValue());

            JRXmlaMember member = new JRXmlaMember(caption, uName, hierName, lName, lNum);
            tuple.setMember(memNum++, member);
        }

        axis.addTuple(tuple);
    }

}