org.cloudfoundry.identity.api.web.ServerRunning.java Source code

Java tutorial

Introduction

Here is the source code for org.cloudfoundry.identity.api.web.ServerRunning.java

Source

/*******************************************************************************
 *     Cloud Foundry 
 *     Copyright (c) [2009-2016] Pivotal Software, Inc. All Rights Reserved.
 *
 *     This product is licensed to you under the Apache License, Version 2.0 (the "License").
 *     You may not use this product except in compliance with the License.
 *
 *     This product includes a number of subcomponents with
 *     separate copyright notices and license terms. Your use of these
 *     subcomponents is subject to the terms and conditions of the
 *     subcomponent's license, as noted in the LICENSE file.
 *******************************************************************************/
package org.cloudfoundry.identity.api.web;

import java.io.IOException;
import java.io.UnsupportedEncodingException;
import java.net.HttpURLConnection;
import java.net.URI;
import java.net.URISyntaxException;
import java.util.HashMap;
import java.util.Map;

import org.apache.commons.logging.Log;
import org.apache.commons.logging.LogFactory;
import org.cloudfoundry.identity.uaa.test.UrlHelper;
import org.junit.Assume;
import org.junit.internal.AssumptionViolatedException;
import org.junit.rules.TestWatchman;
import org.junit.runners.model.FrameworkMethod;
import org.junit.runners.model.Statement;
import org.springframework.http.HttpEntity;
import org.springframework.http.HttpHeaders;
import org.springframework.http.HttpMethod;
import org.springframework.http.HttpStatus;
import org.springframework.http.MediaType;
import org.springframework.http.ResponseEntity;
import org.springframework.http.client.ClientHttpRequest;
import org.springframework.http.client.ClientHttpResponse;
import org.springframework.http.client.SimpleClientHttpRequestFactory;
import org.springframework.security.oauth2.client.test.RestTemplateHolder;
import org.springframework.util.LinkedMultiValueMap;
import org.springframework.util.MultiValueMap;
import org.springframework.web.client.RequestCallback;
import org.springframework.web.client.ResponseErrorHandler;
import org.springframework.web.client.ResponseExtractor;
import org.springframework.web.client.RestClientException;
import org.springframework.web.client.RestOperations;
import org.springframework.web.client.RestTemplate;
import org.springframework.web.util.UriTemplate;
import org.springframework.web.util.UriUtils;

/**
 * <p>
 * A rule that prevents integration tests from failing if the server application
 * is not running or not accessible. If the server is not running in the
 * background all the tests here will simply be skipped because of a violated
 * assumption (showing as successful). Usage:
 * </p>
 * 
 * <pre>
 * &#064;Rule public static ServerRunning brokerIsRunning = ServerRunning.isRunning();
 * 
 * &#064;Test public void testSendAndReceive() throws Exception { // ... test using server etc. }
 * </pre>
 * <p>
 * The rule can be declared as static so that it only has to check once for all
 * tests in the enclosing test case, but there isn't a lot of overhead in making
 * it non-static.
 * </p>
 * 
 * @see Assume
 * @see AssumptionViolatedException
 * 
 * @author Dave Syer
 * 
 */
public class ServerRunning extends TestWatchman implements RestTemplateHolder, UrlHelper {

    private static Log logger = LogFactory.getLog(ServerRunning.class);

    // Static so that we only test once on failure: speeds up test suite
    private static Map<Integer, Boolean> serverOnline = new HashMap<Integer, Boolean>();

    // Static so that we only test once on failure
    private static Map<Integer, Boolean> serverOffline = new HashMap<Integer, Boolean>();

    private final boolean assumeOnline;

    private static int DEFAULT_PORT = 8080;

    private static int DEFAULT_UAA_PORT = 8080;

    private static String DEFAULT_HOST = "localhost";

    private static final String DEFAULT_AUTH_SERVER_ROOT = "/uaa";

    private String authServerRoot = DEFAULT_AUTH_SERVER_ROOT;

    private int port;

    private int uaaPort;

    private String hostName = DEFAULT_HOST;

    private RestOperations client;

    /**
     * @return a new rule that assumes an existing running broker
     */
    public static ServerRunning isRunning() {
        return new ServerRunning(true);
    }

    /**
     * @return a new rule that assumes there is no existing broker
     */
    public static ServerRunning isNotRunning() {
        return new ServerRunning(false);
    }

    private ServerRunning(boolean assumeOnline) {
        this.assumeOnline = assumeOnline;
        setPort(DEFAULT_PORT);
        setUaaPort(DEFAULT_UAA_PORT);
    }

    public void setUaaPort(int uaaPort) {
        this.uaaPort = uaaPort;
    }

    /**
     * @param port the port to set
     */
    public void setPort(int port) {
        this.port = port;
        if (!serverOffline.containsKey(port)) {
            serverOffline.put(port, true);
        }
        if (!serverOnline.containsKey(port)) {
            serverOnline.put(port, true);
        }
        client = createRestTemplate();
    }

    /**
     * @param hostName the hostName to set
     */
    public void setHostName(String hostName) {
        this.hostName = hostName;
    }

    @Override
    public Statement apply(Statement base, FrameworkMethod method, Object target) {

        // Check at the beginning, so this can be used as a static field
        if (assumeOnline) {
            Assume.assumeTrue(serverOnline.get(port));
        } else {
            Assume.assumeTrue(serverOffline.get(port));
        }

        RestTemplate client = new RestTemplate();
        boolean online = false;
        try {
            client.getForEntity(new UriTemplate(getUrl("/uaa/login", uaaPort)).toString(), String.class);
            client.getForEntity(new UriTemplate(getUrl("/api/index.html")).toString(), String.class);
            online = true;
            logger.debug("Basic connectivity test passed");
        } catch (RestClientException e) {
            logger.warn(String.format(
                    "Not executing tests because basic connectivity test failed for hostName=%s, port=%d", hostName,
                    port), e);
            if (assumeOnline) {
                Assume.assumeNoException(e);
            }
        } finally {
            if (online) {
                serverOffline.put(port, false);
                if (!assumeOnline) {
                    Assume.assumeTrue(serverOffline.get(port));
                }

            } else {
                serverOnline.put(port, false);
            }
        }

        return super.apply(base, method, target);

    }

    @Override
    public String getBaseUrl() {
        return "http://" + hostName + ":" + port;
    }

    @Override
    public String getAccessTokenUri() {
        return getUrl(authServerRoot + "/oauth/token");
    }

    @Override
    public String getAuthorizationUri() {
        return getUrl(authServerRoot + "/oauth/authorize");
    }

    @Override
    public String getClientsUri() {
        return getUrl(authServerRoot + "/oauth/clients");
    }

    @Override
    public String getUsersUri() {
        return getUrl(authServerRoot + "/Users");
    }

    @Override
    public String getUserUri() {
        return getUrl(authServerRoot + "/Users");
    }

    @Override
    public String getUrl(String path) {
        return getUrl(path, port);
    }

    public String getUrl(String path, int port) {
        if (path.startsWith("http:")) {
            return path;
        }
        if (!path.startsWith("/")) {
            path = "/" + path;
        }
        return "http://" + hostName + ":" + port + path;
    }

    public String getHostHeader() {
        return hostName + ":" + port;
    }

    @SuppressWarnings("rawtypes")
    public ResponseEntity<Map> postForMap(String path, MultiValueMap<String, String> formData,
            HttpHeaders headers) {
        if (headers.getContentType() == null) {
            headers.setContentType(MediaType.APPLICATION_FORM_URLENCODED);
        }
        return client.exchange(getUrl(path), HttpMethod.POST,
                new HttpEntity<MultiValueMap<String, String>>(formData, headers), Map.class);
    }

    public ResponseEntity<String> postForString(String path, MultiValueMap<String, String> formData,
            HttpHeaders headers) {
        if (headers.getContentType() == null) {
            headers.setContentType(MediaType.APPLICATION_FORM_URLENCODED);
        }
        return client.exchange(getUrl(path), HttpMethod.POST,
                new HttpEntity<MultiValueMap<String, String>>(formData, headers), String.class);
    }

    public ResponseEntity<String> postForString(String path, MultiValueMap<String, String> formData) {
        return postForString(path, formData, new HttpHeaders());
    }

    public ResponseEntity<String> getForString(String path) {
        return getForString(path, new HttpHeaders());
    }

    public ResponseEntity<String> getForString(String path, HttpHeaders headers) {
        HttpEntity<Void> request = new HttpEntity<Void>((Void) null, headers);
        return client.exchange(getUrl(path), HttpMethod.GET, request, String.class);
    }

    public HttpStatus getStatusCode(String path, final HttpHeaders headers) {
        RequestCallback requestCallback = new NullRequestCallback();
        if (headers != null) {
            requestCallback = new RequestCallback() {
                @Override
                public void doWithRequest(ClientHttpRequest request) throws IOException {
                    request.getHeaders().putAll(headers);
                }
            };
        }
        return client.execute(getUrl(path), HttpMethod.GET, requestCallback,
                new ResponseExtractor<ResponseEntity<String>>() {
                    @Override
                    public ResponseEntity<String> extractData(ClientHttpResponse response) throws IOException {
                        return new ResponseEntity<String>(response.getStatusCode());
                    }
                }).getStatusCode();
    }

    public HttpStatus getStatusCode(String path) {
        return getStatusCode(getUrl(path), null);
    }

    @Override
    public void setRestTemplate(RestOperations restTemplate) {
        client = restTemplate;
    }

    @Override
    public RestOperations getRestTemplate() {
        return client;
    }

    public RestOperations createRestTemplate() {
        RestTemplate client = new RestTemplate();
        client.setRequestFactory(new SimpleClientHttpRequestFactory() {
            @Override
            protected void prepareConnection(HttpURLConnection connection, String httpMethod) throws IOException {
                super.prepareConnection(connection, httpMethod);
                connection.setInstanceFollowRedirects(false);
            }
        });
        client.setErrorHandler(new ResponseErrorHandler() {
            // Pass errors through in response entity for status code analysis
            @Override
            public boolean hasError(ClientHttpResponse response) throws IOException {
                return false;
            }

            @Override
            public void handleError(ClientHttpResponse response) throws IOException {
            }
        });
        return client;
    }

    public UriBuilder buildUri(String url) {
        return UriBuilder.fromUri(url.startsWith("http:") ? url : getUrl(url));
    }

    private static final class NullRequestCallback implements RequestCallback {
        @Override
        public void doWithRequest(ClientHttpRequest request) throws IOException {
        }
    }

    public static class UriBuilder {

        private final String url;

        private MultiValueMap<String, String> params = new LinkedMultiValueMap<String, String>();

        public UriBuilder(String url) {
            this.url = url;
        }

        public static UriBuilder fromUri(String url) {
            return new UriBuilder(url);
        }

        public UriBuilder queryParam(String key, String value) {
            params.add(key, value);
            return this;
        }

        public URI build() {
            StringBuilder builder = new StringBuilder(url);
            try {
                if (!params.isEmpty()) {
                    builder.append("?");
                    boolean first = true;
                    for (String key : params.keySet()) {
                        if (!first) {
                            builder.append("&");
                        } else {
                            first = false;
                        }
                        for (String value : params.get(key)) {
                            builder.append(key + "=" + UriUtils.encodeQueryParam(value, "UTF-8"));
                        }
                    }
                }
                return new URI(builder.toString());
            } catch (UnsupportedEncodingException ex) {
                // should not happen, UTF-8 is always supported
                throw new IllegalStateException(ex);
            } catch (URISyntaxException ex) {
                throw new IllegalArgumentException("Could not create URI from [" + builder + "]: " + ex, ex);
            }
        }

    }

}