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

nl.hsac.fitnesse.fixture.Environment Maven / Gradle / Ivy

package nl.hsac.fitnesse.fixture;

import fit.exception.FitFailureException;
import fitnesse.ContextConfigurator;
import freemarker.template.Configuration;
import freemarker.template.DefaultObjectWrapperBuilder;
import freemarker.template.Template;
import nl.hsac.fitnesse.fixture.slim.HttpClientSetup;
import nl.hsac.fitnesse.fixture.util.DatesHelper;
import nl.hsac.fitnesse.fixture.util.Formatter;
import nl.hsac.fitnesse.fixture.util.FreeMarkerHelper;
import nl.hsac.fitnesse.fixture.util.HtmlCleaner;
import nl.hsac.fitnesse.fixture.util.HttpClient;
import nl.hsac.fitnesse.fixture.util.HttpResponse;
import nl.hsac.fitnesse.fixture.util.JsonFormatter;
import nl.hsac.fitnesse.fixture.util.JsonHelper;
import nl.hsac.fitnesse.fixture.util.JsonPathHelper;
import nl.hsac.fitnesse.fixture.util.LineEndingHelper;
import nl.hsac.fitnesse.fixture.util.MapHelper;
import nl.hsac.fitnesse.fixture.util.NamespaceContextImpl;
import nl.hsac.fitnesse.fixture.util.ProgramHelper;
import nl.hsac.fitnesse.fixture.util.ProgramResponse;
import nl.hsac.fitnesse.fixture.util.PropertiesHelper;
import nl.hsac.fitnesse.fixture.util.ReflectionHelper;
import nl.hsac.fitnesse.fixture.util.SecretMasker;
import nl.hsac.fitnesse.fixture.util.TextFormatter;
import nl.hsac.fitnesse.fixture.util.TimeoutHelper;
import nl.hsac.fitnesse.fixture.util.XMLFormatter;
import nl.hsac.fitnesse.fixture.util.XPathHelper;
import nl.hsac.fitnesse.fixture.util.XmlHttpResponse;
import nl.hsac.fitnesse.fixture.util.ZipHelper;
import nl.hsac.fitnesse.fixture.util.selenium.CookieConverter;
import nl.hsac.fitnesse.fixture.util.selenium.SeleniumHelper;
import nl.hsac.fitnesse.fixture.util.selenium.driverfactory.DriverManager;
import org.apache.commons.text.StringEscapeUtils;
import org.apache.http.client.CookieStore;
import org.apache.http.impl.client.BasicCookieStore;
import org.openqa.selenium.Cookie;

import java.io.File;
import java.util.Map;
import java.util.Set;
import java.util.concurrent.ConcurrentHashMap;

/**
 * Holds overall environment settings. Expected to be set up before actual tests
 * are performed.
 */
public class Environment {
    private final static Environment INSTANCE = new Environment();
    private String fitNesseDir = ".";
    private String fitNesseRoot = ContextConfigurator.DEFAULT_ROOT;
    private Configuration freemarkerConfig;
    private FreeMarkerHelper fmHelper;
    private ConcurrentHashMap templateCache;
    private ConcurrentHashMap symbols;
    private HttpClient httpClient;
    private long nextSequenceNr = System.currentTimeMillis();
    private NamespaceContextImpl nsContext;
    private XPathHelper xPathHelper;
    private TextFormatter textFormatter;
    private XMLFormatter xmlFormatter;
    private JsonPathHelper jsonPathHelper;
    private JsonFormatter jsonFormatter;
    private JsonHelper jsonHelper;
    private HtmlCleaner htmlCleaner;
    private TimeoutHelper timeoutHelper = new TimeoutHelper();
    private ProgramHelper programHelper;
    private DatesHelper datesHelper = new DatesHelper();
    private DriverManager driverManager;
    private CookieConverter cookieConverter;
    private MapHelper mapHelper = new MapHelper();
    private ReflectionHelper reflectionHelper = new ReflectionHelper();
    private SecretMasker secretMasker = new SecretMasker();
    private LineEndingHelper lineEndingHelper = new LineEndingHelper();
    private PropertiesHelper propertiesHelper = new PropertiesHelper();
    private ZipHelper zipHelper = new ZipHelper();

    private Environment() {
        Configuration cfg = new Configuration(Configuration.VERSION_2_3_23);
        // Specify the data source where the template files come from.
        cfg.setClassForTemplateLoading(getClass(), "/templates/");
        DefaultObjectWrapperBuilder builder = new DefaultObjectWrapperBuilder(Configuration.VERSION_2_3_23);
        builder.setExposeFields(true);
        cfg.setObjectWrapper(builder.build());
        freemarkerConfig = cfg;

        fmHelper = new FreeMarkerHelper();
        templateCache = new ConcurrentHashMap();

        symbols = new ConcurrentHashMap();

        textFormatter = new TextFormatter();

        xmlFormatter = new XMLFormatter();
        nsContext = new NamespaceContextImpl();
        fillNamespaceContext();
        xPathHelper = new XPathHelper();

        jsonPathHelper = new JsonPathHelper();
        jsonFormatter = new JsonFormatter();
        jsonHelper = new JsonHelper();

        htmlCleaner = new HtmlCleaner();

        httpClient = new HttpClient();

        programHelper = new ProgramHelper();
        programHelper.setTimeoutHelper(timeoutHelper);
        configDatesHelper();

        driverManager = new DriverManager();
        cookieConverter = new CookieConverter();
    }

    /**
     * Fills namespace context with default namespaces.
     */
    private void fillNamespaceContext() {
        // SOAP
        registerNamespace("env", "http://schemas.xmlsoap.org/soap/envelope/");
        registerNamespace("xsi", "http://www.w3.org/2001/XMLSchema-instance");
    }

    /**
     * @return singleton instance.
     */
    public static Environment getInstance() {
        return INSTANCE;
    }

    /**
     * @return new instance of class.
     * @throws RuntimeException if no instance could be created.
     */
    public  T createInstance(Class clazz) {
        try {
            return clazz.getDeclaredConstructor().newInstance();
        } catch (Exception e) {
            throw new RuntimeException("Unable to create instance of: " + clazz.getName(), e);
        }
    }

    /**
     * Stores key/value to be used.
     * @param key
     * @param value
     */
    public void setSymbol(String key, String value) {
        if (value == null) {
            symbols.remove(key);
        } else {
            symbols.put(key, value);
        }
    }

    /**
     * Retrieves value previously stored.
     * @param key
     * @return value stored for key.
     */
    public String getSymbol(String key) {
        return symbols.get(key);
    }

    /**
     * @return next sequence nr
     */
    public long getNextNr() {
        return nextSequenceNr++;
    }

    /**
     * Only to be used in unit tests.
     * @param value next number to provide
     */
    public void setNextNr(long value) {
        nextSequenceNr = value;
    }

    /**
     * Gets symbol value, or throws exception if no symbol by that key exists.
     * @param key symbol's key.
     * @return symbol's value.
     */
    public String getRequiredSymbol(String key) {
        String result = null;
        Object symbol = getSymbol(key);
        if (symbol == null) {
            throw new FitFailureException("No Symbol defined with key: " + key);
        } else {
            result = symbol.toString();
        }
        return result;
    }

    /**
     * @return FreeMarker configuration to use.
     */
    public Configuration getConfiguration() {
        return freemarkerConfig;
    }

    /**
     * @param name name of template to get
     * @return template by that name
     */
    public Template getTemplate(String name) {
        Template result;
        if (!templateCache.containsKey(name)) {
            Template t = fmHelper.getTemplate(getConfiguration(), name);
            result = templateCache.putIfAbsent(name, t);
            if (result == null) {
                result = t;
            }
        } else {
            result = templateCache.get(name);
        }
        return result;
    }

    /**
     * @param templateName name of template to apply
     * @param model model to supply to template
     * @return result of template
     */
    public String processTemplate(String templateName, Object model) {
        Template t = getTemplate(templateName);
        return fmHelper.processTemplate(t, model);
    }

    /**
     * Performs POST to supplied url of result of applying template with model.
     * All namespaces registered in this environment will be registered with result.
     * @param url url to post to.
     * @param templateName name of template to use.
     * @param model model for template.
     * @param result result to populate with response.
     */
    public void callService(String url, String templateName, Object model, XmlHttpResponse result) {
        callService(url, templateName, model, result, null);
    }

    /**
     * Performs POST to supplied url of result of applying template with model.
     * All namespaces registered in this environment will be registered with result.
     * @param url url to post to.
     * @param templateName name of template to use.
     * @param model model for template.
     * @param result result to populate with response.
     * @param headers headers to add.
     */
    public void callService(String url, String templateName, Object model, XmlHttpResponse result, Map headers) {
        doHttpPost(url, templateName, model, result, headers, XmlHttpResponse.CONTENT_TYPE_XML_TEXT_UTF8);
        setContext(result);
    }

    /**
     * Performs POST to supplied url of result of applying template with model.
     * @param url url to post to.
     * @param templateName name of template to use.
     * @param model model for template.
     * @param result result to populate with response.
     */
    public void doHttpPost(String url, String templateName, Object model, HttpResponse result) {
        doHttpPost(url, templateName, model, result, null, XmlHttpResponse.CONTENT_TYPE_XML_TEXT_UTF8);
    }

    /**
     * Performs POST to supplied url of result of applying template with model.
     * @param url url to post to.
     * @param templateName name of template to use.
     * @param model model for template.
     * @param result result to populate with response.
     * @param headers headers to add.
     * @param contentType contentType for request.
     */
    public void doHttpPost(String url, String templateName, Object model, HttpResponse result, Map headers, String contentType) {
        String request = processTemplate(templateName, model);
        result.setRequest(request);
        doHttpPost(url, result, headers, contentType);
    }

    /**
     * Performs POST to supplied url of result's request.
     * @param url url to post to.
     * @param result result containing request, its response will be filled.
     * @param headers headers to add.
     * @param contentType contentType for request.
     */
    public void doHttpPost(String url, HttpResponse result, Map headers, String contentType) {
        httpClient.post(url, result, headers, contentType);
    }

    /**
     * Performs POST to supplied url of a file as binary data.
     * @param url url to post to.
     * @param result result containing request, its response will be filled.
     * @param headers headers to add.
     * @param partName partName for file
     * @param file file containing binary data to post.
     */
    public void doHttpFilePost(String url, HttpResponse result, Map headers, String partName, File file) {
        httpClient.post(url, result, headers, partName, file);
    }

    /**
     * Performs PUT to supplied url of a file as binary data.
     * @param url url to post to.
     * @param result result containing request, its response will be filled.
     * @param headers headers to add.
     * @param partName partName for file
     * @param file file containing binary data to post.
     */
    public void doHttpFilePut(String url, HttpResponse result, Map headers, String partName, File file) {
        httpClient.put(url, result, headers, partName, file);
    }

    /**
     * Performs PUT to supplied url of result of applying template with model.
     * @param url url to put to.
     * @param templateName name of template to use.
     * @param model model for template.
     * @param result result to populate with response.
     */
    public void doHttpPut(String url, String templateName, Object model, HttpResponse result) {
        doHttpPut(url, templateName, model, result, null, XmlHttpResponse.CONTENT_TYPE_XML_TEXT_UTF8);
    }

    /**
     * Performs PUT to supplied url of result of applying template with model.
     * @param url url to put to.
     * @param templateName name of template to use.
     * @param model model for template.
     * @param result result to populate with response.
     * @param headers headers to add.
     * @param contentType contentType for request.
     */
    public void doHttpPut(String url, String templateName, Object model, HttpResponse result, Map headers, String contentType) {
        String request = processTemplate(templateName, model);
        result.setRequest(request);
        doHttpPut(url, result, headers, contentType);
    }

    /**
     * Performs PUT to supplied url of result's request.
     * @param url url to put to.
     * @param result result containing request, its response will be filled.
     * @param headers headers to add.
     * @param contentType contentType for request.
     */
    public void doHttpPut(String url, HttpResponse result, Map headers, String contentType) {
        httpClient.put(url, result, headers, contentType);
    }

    /**
     * GETs content from URL.
     * @param url url to get from.
     * @param headers headers to add
     * @return response.
     */
    public HttpResponse doHttpGet(String url, Map headers, boolean followRedirect) {
        HttpResponse response = new HttpResponse();
        doGet(url, response, headers, followRedirect);
        return response;
    }

    /**
     * GETs XML content from URL.
     * @param url url to get from.
     * @return response.
     */
    public XmlHttpResponse doHttpGetXml(String url) {
        XmlHttpResponse response = new XmlHttpResponse();
        doGet(url, response);
        setContext(response);
        return response;
    }

    /**
     * GETs content from URL.
     * @param url url to get from.
     * @param response response to store url and response value in.
     * @param headers http headers to add.
     */
    public void doGet(String url, HttpResponse response, Map headers, boolean followRedirect) {
        response.setRequest(url);
        httpClient.get(url, response, headers, followRedirect);
    }

    /**
     * GETs content from URL.
     * @param url url to get from.
     * @param response response to store url and response value in.
     */
    public void doGet(String url, HttpResponse response, Map headers) {
        doGet(url, response, headers, true);
    }

    /**
     * GETs content from URL.
     * @param url url to get from.
     * @param response response to store url and response value in.
     */
    public void doGet(String url, HttpResponse response) {
        doGet(url, response, null, true);
    }

    /**
     * GETs content from URL.
     * @param url url to get from.
     * @param response response to store url and response value in.
     */
    public void doGet(String url, HttpResponse response, boolean followRedirect) {
        doGet(url, response, null, followRedirect);
    }

    /**
     * HEADs content from URL.
     * @param url url to get from.
     * @param response response to store url and response value in.
     * @param headers http headers to add.
     */
    public void doHead(String url, HttpResponse response, Map headers) {
        response.setRequest(url);
        httpClient.head(url, response, headers);
    }

    /**
     * DELETEs content at URL.
     * @param url url to send delete to.
     * @param response response to store url and response value in.
     * @param headers http headers to add.
     */
    public void doDelete(String url, HttpResponse response, Map headers) {
        response.setRequest(url);
        httpClient.delete(url, response, headers);
    }

    /**
     * Performs DELETE to supplied url of result of applying template with model.
     * @param url url to delete.
     * @param templateName name of template to use.
     * @param model model for template.
     * @param result result to populate with response.
     * @param headers headers to add.
     * @param contentType contentType for request.
     */
    public void doDelete(String url, String templateName, Object model, HttpResponse result, Map headers, String contentType) {
        String request = processTemplate(templateName, model);
        result.setRequest(request);
        doDelete(url, result, headers, contentType);
    }

    /**
     * Performs DELETE to supplied url of result's request.
     * @param url url to delete.
     * @param result result containing request, its response will be filled.
     * @param headers headers to add.
     * @param contentType contentType for request.
     */
    public void doDelete(String url, HttpResponse result, Map headers, String contentType) {
        httpClient.delete(url, result, headers, contentType);
    }

    /**
     * Performs PATCH to supplied url of result of applying template with model.
     * @param url url to patch.
     * @param templateName name of template to use.
     * @param model model for template.
     * @param result result to populate with response.
     * @param headers headers to add.
     * @param contentType contentType for request.
     */
    public void doHttpPatch(String url, String templateName, Object model, HttpResponse result, Map headers, String contentType) {
        String request = processTemplate(templateName, model);
        result.setRequest(request);
        doHttpPatch(url, result, headers, contentType);
    }

    /**
     * Performs PATCH to supplied url of result's request.
     * @param url url to patch.
     * @param result result containing request, its response will be filled.
     * @param headers headers to add.
     * @param contentType contentType for request.
     */
    public void doHttpPatch(String url, HttpResponse result, Map headers, String contentType) {
        httpClient.patch(url, result, headers, contentType);
    }

    /**
     * @return client to use for HTTP calls.
     */
    public HttpClient getHttpClient() {
        return httpClient;
    }

    public void setContext(XmlHttpResponse response) {
        response.setNamespaceContext(getNamespaceContext());
        response.setXPathHelper(getXPathHelper());
    }

    /**
     * Adds new mapping of prefix to uri for XPath naming resolution.
     * @param prefix prefix that will be used
     * @param uri uri that prefix should refer to.
     */
    public void registerNamespace(String prefix, String uri) {
        nsContext.add(prefix, uri);
    }

    /**
     * @return namespace context for XPath evaluation
     */
    public NamespaceContextImpl getNamespaceContext() {
        return nsContext;
    }

    /**
     * @return XPath helper to use.
     */
    public XPathHelper getXPathHelper() {
        return xPathHelper;
    }

    /**
     * Formats supplied text string for display as-is (including whitespace etc) in FitNesse page.
     * @param content string to format.
     * @return HTML formatted version of content
     */
    public String getHtml(String content) {
        return getHtml(textFormatter, content);
    }

    /**
     * Formats supplied XML string for display in FitNesse page.
     * @param xmlString XML to format.
     * @return HTML formatted version of xmlString
     */
    public String getHtmlForXml(String xmlString) {
        return getHtml(xmlFormatter, xmlString);
    }

    /**
     * Formats supplied Json string for display in FitNesse page.
     * @param jsonString json to format.
     * @return HTML formatted version of jsonString
     */
    public String getHtmlForJson(String jsonString) {
        return getHtml(jsonFormatter, jsonString);
    }

    /**
     * Formats supplied value for display as pre-formatted text in FitNesse page.
     * @param formatter formatter to use to generate pre-formatted text.
     * @param value value to format.
     * @return HTML formatted version of value.
     */
    public String getHtml(Formatter formatter, String value) {
        String result = null;
        if (value != null) {
            if ("".equals(value)) {
                result = "";
            } else {
                String formattedResponse = formatter.format(value);
                result = "
" + StringEscapeUtils.escapeHtml4(formattedResponse) + "
"; } } return result; } /** * Creates exception that will display nicely in a columnFixture. * @param msg message for exception * @param responseText XML received, which will be shown in wiki table. * @throws FitFailureException always */ public static void handleErrorResponse(String msg, String responseText) { String responseHtml; Environment instance = getInstance(); try { responseHtml = instance.getHtmlForXml(responseText); } catch (Exception e) { responseHtml = instance.getHtml(value -> value, responseText); } throw new FitFailureException(msg + responseHtml); } /** * @return helper to clean wiki values provided to fixtures. */ public HtmlCleaner getHtmlCleaner() { return htmlCleaner; } /** * Invokes an external program, waits for it to complete, * and returns the result. * @param timeout maximum time (in milliseconds) to wait. * @param directory working directory for program * (may be null if not important). * @param command program to start. * @param arguments arguments for program. * @return response from program. */ public ProgramResponse invokeProgram(int timeout, String directory, String command, String... arguments) { ProgramResponse result = new ProgramResponse(); result.setDirectory(directory); result.setCommand(command); result.setArguments(arguments); invokeProgram(timeout, result); return result; } /** * Invokes an external program, waits for it to complete, * and returns the result. * @param timeout maximum time (in milliseconds) to wait. * @param result conatiner for which program to call and its response. */ public void invokeProgram(int timeout, ProgramResponse result) { programHelper.execute(result, timeout); } private void configDatesHelper() { datesHelper.setDayPattern("%s_dag"); datesHelper.setMonthPattern("%s_maand"); datesHelper.setYearPattern("%s_jaar"); } /** * @return datesHelper to use. */ public DatesHelper getDatesHelper() { return datesHelper; } /** * @return seleniumHelper to use. */ public SeleniumHelper getSeleniumHelper() { return driverManager.getSeleniumHelper(); } public DriverManager getSeleniumDriverManager() { return driverManager; } public void setSeleniumDriverManager(DriverManager driverManager) { this.driverManager = driverManager; } public String getFitNesseDir() { return fitNesseDir; } public void setFitNesseDir(String fitNesseDir) { this.fitNesseDir = fitNesseDir; } /** * @return directory containing FitNesse's root. */ public String getFitNesseRootDir() { return fitNesseRoot; } /** * @return directory containing FitNesse's files section. */ public String getFitNesseFilesSectionDir() { return new File(fitNesseRoot, "files").getAbsolutePath(); } /** * @param fitNesseRoot directory containing FitNesse's root. */ public void setFitNesseRoot(String fitNesseRoot) { File root = new File(fitNesseRoot); if (!root.exists() || !root.isDirectory()) { throw new IllegalArgumentException("value for fitNesseRoot must be an existing directory"); } this.fitNesseRoot = fitNesseRoot; } /** * Converts a file path into a relative wiki path, if the path is insides the wiki's 'files' section. * @param filePath path to file. * @return relative URL pointing to the file (so a hyperlink to it can be created). */ public String getWikiUrl(String filePath) { String wikiUrl = null; String filesDir = getFitNesseFilesSectionDir(); if (filePath.startsWith(filesDir)) { String relativeFile = filePath.substring(filesDir.length()); relativeFile = relativeFile.replace('\\', '/'); wikiUrl = "files" + relativeFile; } return wikiUrl; } /** * Gets absolute path from wiki url, if file exists. * @param wikiUrl a relative path that can be used in wiki page, or any file path. * @return absolute path to the target of the url, if such a file exists; null if the target does not exist. */ public String getFilePathFromWikiUrl(String wikiUrl) { String url = getHtmlCleaner().getUrl(wikiUrl); File file; if (url.startsWith("files/") || url.startsWith("http://files/")) { String prefix = url.startsWith("files/") ? "files" : "http://files"; String relativeFile = url.substring(prefix.length()); relativeFile = relativeFile.replace('/', File.separatorChar); String pathname = getFitNesseFilesSectionDir() + relativeFile; file = new File(pathname); } else { file = new File(url); } return file.exists() ? file.getAbsolutePath() : url; } /** * @return default (global) map helper. */ public MapHelper getMapHelper() { return mapHelper; } /** * Sets the default MapHelper. * @param aMapHelper map helper to use. */ public void setMapHelper(MapHelper aMapHelper) { mapHelper = aMapHelper; } /** * @return XML formatter used. */ public XMLFormatter getXmlFormatter() { return xmlFormatter; } /** * @return json path helper used. */ public JsonPathHelper getJsonPathHelper() { return jsonPathHelper; } /** * @return JSON helper/formatter used. */ public JsonHelper getJsonHelper() { return jsonHelper; } public ReflectionHelper getReflectionHelper() { return reflectionHelper; } /** * Adds Selenium cookies to response's cookie store. * @param response response to which cookies must be added. */ public void addSeleniumCookies(HttpResponse response) { CookieStore cookieStore = ensureResponseHasCookieStore(response); CookieConverter converter = getCookieConverter(); Set browserCookies = getSeleniumHelper().getCookies(); converter.copySeleniumCookies(browserCookies, cookieStore); } protected CookieStore ensureResponseHasCookieStore(HttpResponse response) { CookieStore cookieStore = response.getCookieStore(); if (cookieStore == null) { cookieStore = new BasicCookieStore(); response.setCookieStore(cookieStore); } return cookieStore; } public CookieConverter getCookieConverter() { return cookieConverter; } public void setCookieConverter(CookieConverter cookieConverter) { this.cookieConverter = cookieConverter; } public SecretMasker getSecretMasker() { return secretMasker; } public void setSecretMasker(SecretMasker secretMasker) { this.secretMasker = secretMasker; } public LineEndingHelper getLineEndingHelper() { return lineEndingHelper; } public void setLineEndingHelper(LineEndingHelper lineEndingHelper) { this.lineEndingHelper = lineEndingHelper; } public PropertiesHelper getPropertiesHelper() { return propertiesHelper; } public void setPropertiesHelper(PropertiesHelper propertiesHelper) { this.propertiesHelper = propertiesHelper; } public ZipHelper getZipHelper() { return zipHelper; } public void setZipHelper(ZipHelper zipHelper) { this.zipHelper = zipHelper; } /** * Enables content compression support on this environment's HttpClient * @deprecated use {@link HttpClientSetup} to configure http client */ @Deprecated public void enableHttpClientCompression() { httpClient.enableCompression(); } /** * Disables content compression support on this environment's HttpClient * @deprecated use {@link HttpClientSetup} to configure http client */ @Deprecated public void disableHttpClientCompression() { httpClient.disableCompression(); } /** * Disables SSL certificate verification on this environment's HttpClient * @deprecated use {@link HttpClientSetup} to configure http client */ @Deprecated public void disableHttpClientSSLVerification() { httpClient.disableSSLVerification(); } /** * Enables SSL certificate verification on this environment's HttpClient * @deprecated use {@link HttpClientSetup} to configure http client */ @Deprecated public void enableHttpClientSSLVerification() { httpClient.enableSSLVerification(); } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy