All Downloads are FREE. Search and download functionalities are using the official Maven repository.

nl.hsac.fitnesse.fixture.util.XmlHttpResponse Maven / Gradle / Ivy

package nl.hsac.fitnesse.fixture.util;

import fit.exception.FitFailureException;
import nl.hsac.fitnesse.fixture.Environment;

import javax.xml.namespace.NamespaceContext;
import java.util.List;
import java.util.Map;
import java.util.Map.Entry;


/**
 * Wrapper around XML HTTP response (and request).
 */
public class XmlHttpResponse extends HttpResponse {
    /** Content type for requests. */
    public final static String CONTENT_TYPE_XML_TEXT_UTF8 = "text/xml; charset=UTF-8";

    private NamespaceContext namespaceContext;
    private XPathHelper xPathHelper;

    @Override
    public void validResponse() {
        super.validResponse();

        String response = getResponse();
        if (response != null) {
            boolean ableToXPath = false;
            try {
                String faultCode = getRawXPath(response, "/env:Envelope/env:Body/env:Fault/faultcode");
                ableToXPath = true;
                if (faultCode != null) {
                    Environment.handleErrorResponse("SOAP fault received: ", response);
                }
            } catch (FitFailureException e) {
                if (ableToXPath) {
                    throw e;
                } else {
                    Environment.handleErrorResponse("Unable to check for SOAP fault, is the result XML? Response was:", response);
                }
            }
        }
    }

    /**
     * @param xPathExpr expression to apply to response.
     * @param params values to put inside expression before evaluation
     * @return result of xpath expression.
     * @throws RuntimeException if no valid response was available or XPath could not be evaluated.
     */
    public String getXPath(String xPathExpr, Object... params) {
        validResponse();

        return getRawXPath(xPathExpr, params);
    }

    /**
     * Gets XPath value without checking whether response is valid.
     * @param xPathExpr expression to apply to response.
     * @param params values to put inside expression before evaluation
     * @return result of xpath expression.
     */
    public String getRawXPath(String xPathExpr, Object... params) {
        return getRawXPath(getResponse(), xPathExpr, params);
    }

    protected String getRawXPath(String soapResponse, String xPathExpr, Object... params) {
        String expr = String.format(xPathExpr, params);
        String xPathValue = getXPathHelper().getXPath(namespaceContext, soapResponse, expr);
        if ("".equals(xPathValue)) {
            xPathValue = null;
        }
        return xPathValue;
    }

    /**
     * @param xPathExpr expression to apply to response.
     * @param params values to put inside expression before evaluation
     * @return result of xpath expression.
     * @throws RuntimeException if no valid response was available or XPath could not be evaluated.
     */
    public Double getXPathDouble(String xPathExpr, Object... params) {
        Double result = null;
        String xPathValue = getXPath(xPathExpr, params);
        if (xPathValue != null) {
            result = Double.valueOf(xPathValue);
        }
        return result;
    }

    /**
     * @param xPathExpr expression to apply to response.
     * @param params values to put inside expression before evaluation
     * @return result of xpath expression.
     * @throws RuntimeException if no valid response was available or XPath could not be evaluated.
     */
    public Integer getXPathInt(String xPathExpr, Object... params) {
        Integer result = null;
        String xPathValue = getXPath(xPathExpr, params);
        if (xPathValue != null) {
            result = Integer.valueOf(xPathValue);
        }
        return result;
    }

    /**
     * @param xPathExpr expression to apply to response.
     * @param params values to put inside expression before evaluation
     * @return result of xpath expression.
     * @throws RuntimeException if no valid response was available or XPath could not be evaluated.
     */
    public Double getXPathAmount(String xPathExpr, Object... params) {
        Double result = getXPathDouble(xPathExpr, params);
        if (result != null) {
            result = Math.round(result * 100) / 100d;
        }
        return result;
    }

    /**
     * @param xPathExpr expression to apply to response.
     * @param params values to put inside expression before evaluation
     * @return result of xpath expression, without timezone part.
     * @throws RuntimeException if no valid response was available or XPath could not be evaluated.
     */
    public String getXPathDate(String xPathExpr, Object... params) {
        String result = getXPath(xPathExpr, params);
        if (result != null) {
            // take only date part (i.e. YYYY-MM-DD)
            result = result.substring(0, 10);
        }
        return result;
    }

    /**
     * Checks whether input values are present at correct locations in the response.
     * @param values keyName -> value, input parameters supplied to get request.
     * @param expressionsToCheck xpath -> keyName, each xpath in this map is expected
     *        to evaluate to the value present in values for that key (i.e. values[keyName])
     * @return OK if all xpath expressions evaluated to the correct value. A description
     *        of the mismatches otherwise.
     */
    public XPathCheckResult checkXPaths(Map values, Map expressionsToCheck) {
        XPathCheckResult result;
        String content = getResponse();
        if (content == null) {
            result = new XPathCheckResult();
            result.setMismatchDetail("NOK: no response available.");
        } else {
            validResponse();
            result = checkRawXPaths(content, expressionsToCheck, values);
        }

        return result;
    }

    protected XPathCheckResult checkRawXPaths(String content, Map expressionsToCheck, Map values) {
        XPathCheckResult result = new XPathCheckResult();
        for (Entry exprEntry : expressionsToCheck.entrySet()) {
            String xpath = exprEntry.getKey();
            String keyName = exprEntry.getValue();
            Object value = null;
            String valueStr = null;
            if (keyName.contains(".") && !values.containsKey(keyName)) {
                value = getNestedValue(values, keyName);
            } else {
                value = values.get(keyName);
            }
            valueStr = String.valueOf(value);
            String xpathValue = String.valueOf(getRawXPath(content, xpath)).trim();
            if (!valueStr.equals(xpathValue) && !equalsDates(valueStr, xpathValue)
                    && !equalsAmounts(valueStr, xpathValue)
                    && !("".equals(valueStr) && "null".equals(xpathValue))) {
                result.addMisMatch(keyName, valueStr, xpathValue);
            }
        }

        return result;
    }

    private Object getNestedValue(Map values, String keyName) {
        String[] keyPath = keyName.split("\\.");
        String parentKey = keyPath[0];
        String valueKey = keyPath[1];
        @SuppressWarnings("unchecked")
        Map firstValue = (Map) values.get(parentKey);
        return firstValue.get(valueKey);
    }

    private boolean equalsDates(String aValue, String aXpathValue) {
        boolean result = false;
        if (aValue.length() == 10 && aXpathValue.startsWith(aValue)) {
            String timeZone = aXpathValue.substring(10);
            if (timeZone.matches("[+-]\\d\\d:\\d\\d")) {
                result = true;
            }
        }
        return result;
    }

    private boolean equalsAmounts(String aValue, String aXpathValue) {
        boolean result = false;
        try {
            Double value = Double.parseDouble(aValue);
            Double xpath = Double.parseDouble(aXpathValue);
            result = value.equals(xpath);
        } catch (Exception e) {
            // ignore
        }
        return result;
    }

    /**
     * Gets the value xsi:type attribute will have for provided type.
     * @param type sub type wanted.
     * @return type name including namespace prefix as present in response.
     */
    public String getXsiTypeValue(String type) {
        return type;
    }

    /**
     * @param xPathExpr expression to apply to response.
     * @param params values to put inside expression before evaluation
     * @return all results of xpath expression.
     * @throws RuntimeException if no valid response was available or XPath could not be evaluated.
     */
    public List getAllXPath(String xPathExpr, Object... params) {
        validResponse();
        return getRawAllXPath(response, xPathExpr, params);
    }

    protected List getRawAllXPath(String soapResponse, String xPathExpr, Object... params) {
        String expr = String.format(xPathExpr, params);
        return getXPathHelper().getAllXPath(namespaceContext, soapResponse, expr);
    }

    /**
     * @param aNamespaceContext the namespaceContext to set
     */
    public void setNamespaceContext(NamespaceContext aNamespaceContext) {
        namespaceContext = aNamespaceContext;
    }

    public void setXPathHelper(XPathHelper xPathHelper) {
        this.xPathHelper = xPathHelper;
    }

    public XPathHelper getXPathHelper() {
        return xPathHelper;
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy