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.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
*/
public void enableHttpClientCompression() {
httpClient.enableCompression();
}
/**
* Disables content compression support on this environment's HttpClient
* @deprecated use {@link HttpClientSetup} to configure http client
*/
public void disableHttpClientCompression() {
httpClient.disableCompression();
}
/**
* Disables SSL certificate verification on this environment's HttpClient
* @deprecated use {@link HttpClientSetup} to configure http client
*/
public void disableHttpClientSSLVerification() {
httpClient.disableSSLVerification();
}
/**
* Enables SSL certificate verification on this environment's HttpClient
* @deprecated use {@link HttpClientSetup} to configure http client
*/
public void enableHttpClientSSLVerification() {
httpClient.enableSSLVerification();
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy