org.apache.brooklyn.util.HttpAssertsTest.java Source code

Java tutorial

Introduction

Here is the source code for org.apache.brooklyn.util.HttpAssertsTest.java

Source

/*
 * Licensed to the Apache Software Foundation (ASF) under one
 * or more contributor license agreements.  See the NOTICE file
 * distributed with this work for additional information
 * regarding copyright ownership.  The ASF licenses this file
 * to you under the Apache License, Version 2.0 (the
 * "License"); you may not use this file except in compliance
 * with the License.  You may obtain a copy of the License at
 *
 *     http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing,
 * software distributed under the License is distributed on an
 * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
 * KIND, either express or implied.  See the License for the
 * specific language governing permissions and limitations
 * under the License.
 */
package org.apache.brooklyn.util;

import java.net.URI;
import java.util.concurrent.ExecutionException;
import java.util.concurrent.Executors;
import java.util.concurrent.ScheduledExecutorService;
import java.util.concurrent.TimeUnit;

import org.apache.brooklyn.test.http.TestHttpRequestHandler;
import org.apache.brooklyn.test.http.TestHttpServer;
import org.apache.brooklyn.util.exceptions.Exceptions;
import org.apache.brooklyn.util.http.HttpAsserts;
import org.apache.brooklyn.util.http.HttpTool;
import org.apache.brooklyn.util.http.HttpToolResponse;
import org.apache.brooklyn.util.javalang.JavaClassNames;
import org.apache.brooklyn.util.net.Networking;
import org.apache.brooklyn.util.time.Duration;
import org.apache.brooklyn.util.time.Time;
import org.apache.http.HttpStatus;
import org.apache.http.client.HttpClient;
import org.apache.http.localserver.RequestBasicAuth;
import org.apache.http.localserver.ResponseBasicUnauthorized;
import org.apache.http.protocol.ResponseServer;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.testng.annotations.AfterMethod;
import org.testng.annotations.BeforeMethod;
import org.testng.annotations.Test;

import com.google.common.collect.ImmutableMap;
import com.google.common.util.concurrent.ListenableFuture;
import com.google.common.util.concurrent.ListeningExecutorService;
import com.google.common.util.concurrent.MoreExecutors;

/**
 * Tests on {@link HttpAsserts}.
 *
 * @todo Restructure this to avoid sleeps, according to conversation at
 * <a href="https://github.com/apache/incubator-brooklyn/pull/994#issuecomment-154074295">#994</a> on github.
 */
@Test(groups = "Integration")
public class HttpAssertsTest {

    private static final Logger LOG = LoggerFactory.getLogger(HttpAssertsTest.class);
    private static Duration DELAY_FOR_SERVER_TO_SETTLE = Duration.seconds(2);

    HttpClient httpClient;
    URI baseUri;
    private TestHttpServer server;
    private String baseUrl;
    private String simpleEndpoint;
    private ScheduledExecutorService executor;

    @BeforeMethod(alwaysRun = true)
    public void setUp() throws Exception {
        server = initializeServer();
        initVars();
        Time.sleep(DELAY_FOR_SERVER_TO_SETTLE);
    }

    private void initVars() {
        baseUrl = server.getUrl();
        httpClient = HttpTool.httpClientBuilder().uri(baseUrl).build();
        baseUri = URI.create(baseUrl);
        simpleEndpoint = testUri("/simple");
        executor = Executors.newScheduledThreadPool(3);
    }

    private TestHttpServer initializeServerUnstarted() {
        return new TestHttpServer().interceptor(new ResponseServer()).interceptor(new ResponseBasicUnauthorized())
                .interceptor(new RequestBasicAuth()).handler("/simple", new TestHttpRequestHandler().response("OK"))
                .handler("/empty", new TestHttpRequestHandler().code(HttpStatus.SC_NO_CONTENT))
                .handler("/missing", new TestHttpRequestHandler().code(HttpStatus.SC_NOT_FOUND).response("Missing"))
                .handler("/redirect",
                        new TestHttpRequestHandler().code(HttpStatus.SC_MOVED_TEMPORARILY).response("Redirect")
                                .header("Location", "/simple"))
                .handler("/cycle",
                        new TestHttpRequestHandler().code(HttpStatus.SC_MOVED_TEMPORARILY).response("Redirect")
                                .header("Location", "/cycle"))
                .handler("/secure", new TestHttpRequestHandler().code(HttpStatus.SC_MOVED_TEMPORARILY)
                        .response("Redirect").header("Location", "https://0.0.0.0/"));
    }

    private TestHttpServer initializeServer() {
        return initializeServerUnstarted().start();
    }

    @AfterMethod(alwaysRun = true)
    public void tearDown() throws Exception {
        if (executor != null)
            executor.shutdownNow();
        server.stop();
        Time.sleep(DELAY_FOR_SERVER_TO_SETTLE);
    }

    // schedule a stop of server after n seconds
    private void stopAfter(final Duration time) {
        final TestHttpServer serverCached = server;
        executor.schedule(new Runnable() {
            @Override
            public void run() {
                LOG.info("stopping server (" + time + " elapsed)");
                serverCached.stop();
            }
        }, time.toMilliseconds(), TimeUnit.MILLISECONDS);
    }

    // stop server and pause to wait for it to finish
    private void stopServer() {
        server.stop();
        Time.sleep(DELAY_FOR_SERVER_TO_SETTLE);
    }

    // schedule a start of server after n seconds
    private void startAfter(final Duration time) {
        // find the port before delay so that callers can get the right url
        // (sometimes if stopped and started it can't bind to the same port;
        // at least that is one suspicion for failures on hosted jenkins)
        server = initializeServerUnstarted();
        server.basePort(Networking.nextAvailablePort(50606));
        initVars();
        executor.schedule(new Runnable() {
            @Override
            public void run() {
                LOG.info("starting server (" + time + " elapsed)");
                server.start();
            }
        }, time.toMilliseconds(), TimeUnit.MILLISECONDS);
    }

    private HttpToolResponse doGet(String str) {
        final URI address = baseUri.resolve(str);
        return HttpTool.httpGet(httpClient, address, ImmutableMap.<String, String>of());
    }

    @Test
    public void shouldAssertHealthyStatusCode() {
        final HttpToolResponse response = doGet("/simple");
        HttpAsserts.assertHealthyStatusCode(response.getResponseCode());
    }

    @Test
    public void shouldAssertUrlReachable() {
        HttpAsserts.assertUrlReachable(baseUrl);
    }

    @Test
    public void shouldAssertUrlUnreachable() {
        stopServer();
        HttpAsserts.assertUrlUnreachable(simpleEndpoint);
    }

    @Test
    public void shouldAssertUrlUnreachableEventually() {
        HttpAsserts.assertUrlReachable(baseUrl);
        stopAfter(Duration.seconds(1));
        HttpAsserts.assertUrlUnreachableEventually(baseUrl);
    }

    @Test
    public void shouldAssertUrlUnreachableEventuallyWithFlags() throws Exception {
        String baseUrlOrig = baseUrl;
        LOG.info("testing " + JavaClassNames.niceClassAndMethod() + ", server " + server.getUrl());
        stopAfter(DELAY_FOR_SERVER_TO_SETTLE);
        startAfter(DELAY_FOR_SERVER_TO_SETTLE.add(DELAY_FOR_SERVER_TO_SETTLE).add(DELAY_FOR_SERVER_TO_SETTLE));
        LOG.info(JavaClassNames.niceClassAndMethod() + " queued server changes");
        HttpAsserts.assertUrlReachable(baseUrlOrig);
        HttpAsserts.assertUrlUnreachableEventually(ImmutableMap.of("timeout", "10s"), baseUrlOrig);
    }

    @Test(expectedExceptions = AssertionError.class)
    public void shouldFailAssertUrlUnreachableEventuallyWithTimeout() throws Exception {
        HttpAsserts.assertUrlReachable(baseUrl);
        HttpAsserts.assertUrlUnreachableEventually(ImmutableMap.of("timeout", "3s"), baseUrl);
    }

    @Test
    public void shouldAssertHttpStatusCodeEquals() {
        HttpAsserts.assertHttpStatusCodeEquals(baseUrl, 500, 501);
        HttpAsserts.assertHttpStatusCodeEquals(simpleEndpoint, 201, 200);
        HttpAsserts.assertHttpStatusCodeEquals(testUri("/missing"), 400, 404);
    }

    @Test
    public void shouldAssertHttpStatusCodeEventuallyEquals() throws Exception {
        stopServer();
        HttpAsserts.assertUrlUnreachable(simpleEndpoint);
        startAfter(DELAY_FOR_SERVER_TO_SETTLE);
        try {
            HttpAsserts.assertHttpStatusCodeEventuallyEquals(simpleEndpoint, 200);
        } catch (Throwable t) {
            LOG.warn("Failed waiting for simple with start after: " + t, t);
            LOG.warn("Detail: server at " + server.getUrl() + " (" + server + "), looking at " + simpleEndpoint);
            throw Exceptions.propagate(t);
        }
    }

    private String testUri(String str) {
        return baseUri.resolve(str).toString();
    }

    @Test
    public void shouldAssertContentContainsText() {
        HttpAsserts.assertContentContainsText(simpleEndpoint, "OK");
    }

    @Test
    public void shouldAssertContentNotContainsText() {
        HttpAsserts.assertContentNotContainsText(simpleEndpoint, "Bad");
    }

    @Test
    public void shouldAssertErrorContentsContainsText() {
        HttpAsserts.assertErrorContentContainsText(testUri("/missing"), "Missing");
    }

    @Test
    public void shouldAssertErrorContentNotContainsText() {
        HttpAsserts.assertErrorContentNotContainsText(testUri("/missing"), "Bogus");
    }

    @Test
    public void shouldAssertContentEventuallyContainsText() {
        stopServer();
        HttpAsserts.assertUrlUnreachable(simpleEndpoint);
        startAfter(DELAY_FOR_SERVER_TO_SETTLE);
        HttpAsserts.assertContentEventuallyContainsText(simpleEndpoint, "OK");
    }

    @Test
    public void shouldAssertContentEventuallyContainsTextWithFlags() {
        stopServer();
        HttpAsserts.assertUrlUnreachable(simpleEndpoint);
        startAfter(DELAY_FOR_SERVER_TO_SETTLE);
        HttpAsserts.assertContentEventuallyContainsText(
                ImmutableMap.of("timeout", DELAY_FOR_SERVER_TO_SETTLE.add(Duration.ONE_SECOND).toStringRounded()),
                simpleEndpoint, "OK");
    }

    @Test(expectedExceptions = AssertionError.class)
    public void shouldAssertContentEventuallyContainsTextWithTimeout() {
        stopServer();
        HttpAsserts.assertUrlUnreachable(simpleEndpoint);
        startAfter(DELAY_FOR_SERVER_TO_SETTLE.add(Duration.seconds(2)));
        HttpAsserts.assertContentEventuallyContainsText(
                ImmutableMap.of("timeout", DELAY_FOR_SERVER_TO_SETTLE.add(Duration.ONE_SECOND).toStringRounded()),
                simpleEndpoint, "OK");
    }

    @Test
    public void shouldAssertContentMatches() {
        HttpAsserts.assertContentMatches(simpleEndpoint, "[Oo][Kk]");
    }

    @Test
    public void shouldAssertContentEventuallyMatches() throws Exception {
        stopServer();
        Time.sleep(DELAY_FOR_SERVER_TO_SETTLE);
        HttpAsserts.assertUrlUnreachable(simpleEndpoint);
        Time.sleep(DELAY_FOR_SERVER_TO_SETTLE);
        startAfter(DELAY_FOR_SERVER_TO_SETTLE);
        HttpAsserts.assertContentEventuallyMatches(simpleEndpoint, "[Oo][Kk]");
    }

    @Test
    public void shouldAssertContentEventuallyMatchesWithFlags() {
        stopServer();
        HttpAsserts.assertUrlUnreachable(simpleEndpoint);
        startAfter(DELAY_FOR_SERVER_TO_SETTLE);
        HttpAsserts.assertContentEventuallyMatches(ImmutableMap.of("timeout", "3s"), simpleEndpoint, "[Oo][Kk]");
    }

    @Test(expectedExceptions = AssertionError.class)
    public void shouldAssertContentEventuallyMatchesWithTimeout() {
        stopServer();
        HttpAsserts.assertUrlUnreachable(simpleEndpoint);
        startAfter(DELAY_FOR_SERVER_TO_SETTLE.add(Duration.seconds(2)));
        HttpAsserts.assertContentEventuallyMatches(
                ImmutableMap.of("timeout", DELAY_FOR_SERVER_TO_SETTLE.add(Duration.ONE_SECOND).toStringRounded()),
                simpleEndpoint, "[Oo][Kk]");
    }

    @Test
    public void shouldAssertAsyncHttpStatusCodeContinuallyEquals() throws Exception {
        stopServer();
        ListeningExecutorService listeningExecutor = MoreExecutors.listeningDecorator(executor);
        final ListenableFuture<?> future = HttpAsserts.assertAsyncHttpStatusCodeContinuallyEquals(listeningExecutor,
                simpleEndpoint, 200);
        startAfter(DELAY_FOR_SERVER_TO_SETTLE.add(Duration.seconds(1)));
        if (future.isDone()) {
            future.get(); // should not throw exception
        }
        future.cancel(true);
    }

    @Test(expectedExceptions = ExecutionException.class)
    public void shouldAssertAsyncHttpStatusCodeContinuallyEqualsFails() throws Exception {
        stopServer();
        ListeningExecutorService listeningExecutor = MoreExecutors.listeningDecorator(executor);
        final ListenableFuture<?> future = HttpAsserts.assertAsyncHttpStatusCodeContinuallyEquals(listeningExecutor,
                testUri("/missing"), 200);
        startAfter(DELAY_FOR_SERVER_TO_SETTLE.add(Duration.seconds(1)));
        Time.sleep(DELAY_FOR_SERVER_TO_SETTLE);
        if (future.isDone()) {
            Object result = future.get(); // should throw exception
            LOG.warn("Should have failed, instead gave " + result + " (accessing " + server + ")");
        } else {
            LOG.warn("Future should have been done");
        }
        future.cancel(true);
    }
}