blob: b6189db57ad838b6b86132eed13ed76c5219b7cb [file] [log] [blame]
/*
* Copyright (c) 2020 Oracle and/or its affiliates. All rights reserved.
*
* This program and the accompanying materials are made available under the
* terms of the Eclipse Public License v. 2.0, which is available at
* http://www.eclipse.org/legal/epl-2.0.
*
* This Source Code may also be made available under the following Secondary
* Licenses when the conditions for such availability set forth in the
* Eclipse Public License v. 2.0 are satisfied: GNU General Public License,
* version 2 with the GNU Classpath Exception, which is available at
* https://www.gnu.org/software/classpath/license.html.
*
* SPDX-License-Identifier: EPL-2.0 OR GPL-2.0 WITH Classpath-exception-2.0
*/
package org.glassfish.jersey.helidon.connector;
import org.glassfish.jersey.client.ClientConfig;
import org.glassfish.jersey.logging.LoggingFeature;
import org.glassfish.jersey.server.ResourceConfig;
import org.glassfish.jersey.test.JerseyTest;
import org.hamcrest.Matchers;
import org.junit.Test;
import javax.ws.rs.GET;
import javax.ws.rs.POST;
import javax.ws.rs.Path;
import javax.ws.rs.client.Entity;
import javax.ws.rs.container.AsyncResponse;
import javax.ws.rs.container.Suspended;
import javax.ws.rs.container.TimeoutHandler;
import javax.ws.rs.core.Application;
import javax.ws.rs.core.Response;
import java.util.concurrent.Future;
import java.util.concurrent.TimeUnit;
import java.util.logging.Logger;
import static org.junit.Assert.assertEquals;
import static org.junit.Assert.assertThat;
/**
* Asynchronous connector test.
*
* @author Arul Dhesiaseelan (aruld at acm.org)
* @author Marek Potociar
*/
public class AsyncTest extends JerseyTest {
private static final Logger LOGGER = Logger.getLogger(AsyncTest.class.getName());
private static final String PATH = "async";
/**
* Asynchronous test resource.
*/
@Path(PATH)
public static class AsyncResource {
/**
* Typical long-running operation duration.
*/
public static final long OPERATION_DURATION = 1000;
/**
* Long-running asynchronous post.
*
* @param asyncResponse async response.
* @param id post request id (received as request payload).
*/
@POST
public void asyncPost(@Suspended final AsyncResponse asyncResponse, final String id) {
LOGGER.info("Long running post operation called with id " + id + " on thread " + Thread.currentThread().getName());
new Thread(new Runnable() {
@Override
public void run() {
final String result = veryExpensiveOperation();
asyncResponse.resume(result);
}
private String veryExpensiveOperation() {
// ... very expensive operation that typically finishes within 1 seconds, simulated using sleep()
try {
Thread.sleep(OPERATION_DURATION);
return "DONE-" + id;
} catch (final InterruptedException e) {
Thread.currentThread().interrupt();
return "INTERRUPTED-" + id;
} finally {
LOGGER.info("Long running post operation finished on thread " + Thread.currentThread().getName());
}
}
}, "async-post-runner-" + id).start();
}
/**
* Long-running async get request that times out.
*
* @param asyncResponse async response.
*/
@GET
@Path("timeout")
public void asyncGetWithTimeout(@Suspended final AsyncResponse asyncResponse) {
LOGGER.info("Async long-running get with timeout called on thread " + Thread.currentThread().getName());
asyncResponse.setTimeoutHandler(new TimeoutHandler() {
@Override
public void handleTimeout(final AsyncResponse asyncResponse) {
asyncResponse.resume(Response.status(Response.Status.SERVICE_UNAVAILABLE)
.entity("Operation time out.").build());
}
});
asyncResponse.setTimeout(1, TimeUnit.SECONDS);
new Thread(new Runnable() {
@Override
public void run() {
final String result = veryExpensiveOperation();
asyncResponse.resume(result);
}
private String veryExpensiveOperation() {
// very expensive operation that typically finishes within 1 second but can take up to 5 seconds,
// simulated using sleep()
try {
Thread.sleep(5 * OPERATION_DURATION);
return "DONE";
} catch (final InterruptedException e) {
Thread.currentThread().interrupt();
return "INTERRUPTED";
} finally {
LOGGER.info("Async long-running get with timeout finished on thread " + Thread.currentThread().getName());
}
}
}).start();
}
}
@Override
protected Application configure() {
return new ResourceConfig(AsyncResource.class)
.register(new LoggingFeature(LOGGER, LoggingFeature.Verbosity.PAYLOAD_ANY));
}
@Override
protected void configureClient(final ClientConfig config) {
config.register(new LoggingFeature(LOGGER, LoggingFeature.Verbosity.PAYLOAD_ANY));
config.connectorProvider(new HelidonConnectorProvider());
}
/**
* Test asynchronous POST.
* <p/>
* Send 3 async POST requests and wait to receive the responses. Check the response content and
* assert that the operation did not take more than twice as long as a single long operation duration
* (this ensures async request execution).
*
* @throws Exception in case of a test error.
*/
@Test
public void testAsyncPost() throws Exception {
final Future<Response> warmUp1 = target(PATH).request().async().post(Entity.text("100"));
final Future<Response> warmUp2 = target(PATH).request().async().post(Entity.text("200"));
final Future<Response> warmUp3 = target(PATH).request().async().post(Entity.text("300"));
assertEquals("DONE-100", warmUp1.get().readEntity(String.class));
assertEquals("DONE-200", warmUp2.get().readEntity(String.class));
assertEquals("DONE-300", warmUp3.get().readEntity(String.class));
final long tic = System.currentTimeMillis();
// Submit requests asynchronously.
final Future<Response> rf1 = target(PATH).request().async().post(Entity.text("1"));
final Future<Response> rf2 = target(PATH).request().async().post(Entity.text("2"));
final Future<Response> rf3 = target(PATH).request().async().post(Entity.text("3"));
// get() waits for the response
final String r1 = rf1.get().readEntity(String.class);
final String r2 = rf2.get().readEntity(String.class);
final String r3 = rf3.get().readEntity(String.class);
final long toc = System.currentTimeMillis();
assertEquals("DONE-1", r1);
assertEquals("DONE-2", r2);
assertEquals("DONE-3", r3);
assertThat("Async processing took too long.", toc - tic, Matchers.lessThan(3 * AsyncResource.OPERATION_DURATION));
}
/**
* Test accessing an operation that times out on the server.
*
* @throws Exception in case of a test error.
*/
@Test
public void testAsyncGetWithTimeout() throws Exception {
final Future<Response> responseFuture = target(PATH).path("timeout").request().async().get();
// Request is being processed asynchronously.
final Response response = responseFuture.get();
// get() waits for the response
assertEquals(503, response.getStatus());
assertEquals("Operation time out.", response.readEntity(String.class));
}
}