diff --git a/framework/src/play/test/FunctionalTest.java b/framework/src/play/test/FunctionalTest.java index 76ad2884b3..db2b196214 100644 --- a/framework/src/play/test/FunctionalTest.java +++ b/framework/src/play/test/FunctionalTest.java @@ -31,11 +31,15 @@ import play.Invoker; import play.Invoker.InvocationContext; import play.classloading.enhancers.ControllersEnhancer.ControllerInstrumentation; +import play.exceptions.JavaExecutionException; +import play.exceptions.UnexpectedException; import play.mvc.ActionInvoker; import play.mvc.Controller; import play.mvc.Http; import play.mvc.Http.Request; import play.mvc.Http.Response; +import java.util.concurrent.ExecutionException; +import java.util.concurrent.Future; import play.mvc.Router.ActionDefinition; import play.mvc.Scope.RenderArgs; @@ -307,7 +311,7 @@ public static Response DELETE(Request request, Object url) { public static void makeRequest(final Request request, final Response response) { final CountDownLatch actionCompleted = new CountDownLatch(1); - TestEngine.functionalTestsExecutor.submit(new Invoker.Invocation() { + final Future invocationResult = TestEngine.functionalTestsExecutor.submit(new Invoker.Invocation() { @Override public void execute() throws Exception { @@ -350,9 +354,27 @@ public InvocationContext getInvocationContext() { }); try { + // We can not simply wait on the future result because of how continuations + // are implemented. Only when the latch is counted down the action is really + // completed. Therefore, wait on the latch. if (!actionCompleted.await(30, TimeUnit.SECONDS)) { throw new TimeoutException("Request did not complete in time"); } + } catch (Exception ex) { + throw new RuntimeException(ex); + } + try { + // We still call this to raise any exception that might have + // occurred during execution of the invocation. + invocationResult.get(); + } + catch (ExecutionException e) { + throw unwrapOriginalException(e); + } + catch (Exception e) { + throw new RuntimeException(e); + } + try { if (savedCookies == null) { savedCookies = new HashMap(); } @@ -374,6 +396,21 @@ public InvocationContext getInvocationContext() { } } + private static RuntimeException unwrapOriginalException(final ExecutionException e) { + // Check if the original exceptions fits the usual patterns. If yes, throw the very + // original runtime exception + final Throwable executionCause = e.getCause(); + if (executionCause != null + && (executionCause instanceof JavaExecutionException || executionCause instanceof UnexpectedException)) { + final Throwable originalCause = executionCause.getCause(); + if (originalCause != null && originalCause instanceof RuntimeException) { + throw (RuntimeException) originalCause; + } + } + // As a last fallback, just wrap everything up + return new RuntimeException(e); + } + public static Response makeRequest(final Request request) { Response response = newResponse(); makeRequest(request, response); diff --git a/samples-and-tests/just-test-cases/app/controllers/StatusCodes.java b/samples-and-tests/just-test-cases/app/controllers/StatusCodes.java new file mode 100644 index 0000000000..2be98462a2 --- /dev/null +++ b/samples-and-tests/just-test-cases/app/controllers/StatusCodes.java @@ -0,0 +1,36 @@ +package controllers; + +import play.jobs.Job; +import play.mvc.Controller; +import play.mvc.Http.Response; + +public class StatusCodes extends Controller { + + public static void justOkay() { + renderText("Okay"); + } + + public static void rendersNotFound() { + notFound(); + } + + public static void rendersUnauthorized() { + unauthorized(); + } + + public static void usesContinuation() { + final String text = await(new Job() { + @Override + public String doJobWithResult() throws Exception { + return "Job completed successfully"; + } + }.now()); + Response.current().status = Integer.valueOf(201); + renderText(text); + } + + public static void throwsException() throws Exception { + throw new UnsupportedOperationException("Whoops"); + } + +} diff --git a/samples-and-tests/just-test-cases/app/controllers/WithContinuations.java b/samples-and-tests/just-test-cases/app/controllers/WithContinuations.java index 643fc5caf5..0e1f3eaaed 100644 --- a/samples-and-tests/just-test-cases/app/controllers/WithContinuations.java +++ b/samples-and-tests/just-test-cases/app/controllers/WithContinuations.java @@ -163,14 +163,14 @@ public static void rollbackWithContinuationsThatWorks() { public static void streamedResult() { response.contentType = "text/html"; - response.writeChunk("

This page should load progressively in about 3 second

"); + response.writeChunk("

This page should load progressively in about a few seconds

"); long s = System.currentTimeMillis(); for(int i=0; i<100; i++) { await(10); response.writeChunk("

Hello " + i + "

"); } long t = System.currentTimeMillis() - s; - response.writeChunk("Time: " + t + ", isOk->" + (t > 1000 && t < 10000)); + response.writeChunk("Time: " + t + ", isOk->" + (t > 1000 && t < 25000)); } public static void loopWithCallback() { @@ -206,7 +206,7 @@ public void invoke() { await(10, this); } else { long t = System.currentTimeMillis() - s.get(); - response.writeChunk("Time: " + t + ", isOk->" + (t > 1000 && t < 10000)); + response.writeChunk("Time: " + t + ", isOk->" + (t > 1000 && t < 25000)); } } }; diff --git a/samples-and-tests/just-test-cases/conf/routes b/samples-and-tests/just-test-cases/conf/routes index 1cd69ef827..9a2550af40 100644 --- a/samples-and-tests/just-test-cases/conf/routes +++ b/samples-and-tests/just-test-cases/conf/routes @@ -80,3 +80,9 @@ GET /databinding/changeLanguage/{lang}/? DataBinding.changeLa GET /useAwaitViaOtherClass WithContinuations.ControllerWithoutContinuations.useAwaitViaOtherClass + +GET /status/ok/ StatusCodes.justOkay +GET /status/not-found/ StatusCodes.rendersNotFound +GET /status/unauthorized/ StatusCodes.rendersUnauthorized +POST /status/job/ StatusCodes.usesContinuation +GET /status/failure/ StatusCodes.throwsException diff --git a/samples-and-tests/just-test-cases/test/BinaryTest.java b/samples-and-tests/just-test-cases/test/BinaryTest.java index beaf21c626..d4ec910663 100755 --- a/samples-and-tests/just-test-cases/test/BinaryTest.java +++ b/samples-and-tests/just-test-cases/test/BinaryTest.java @@ -28,7 +28,7 @@ public void setUp() { Response deletedResponse = GET(deleteURL); assertStatus(200, deletedResponse); } - + @Test public void testUploadSomething() { URL imageURL = reverse(); { @@ -235,11 +235,13 @@ public void testGetEmptyBinary() { assertTrue(Binary.emptyInputStreamClosed); } - @Test + @Test(expected = Exception.class) public void testGetErrorBinary() { - Response response = GET("/binary/getErrorBinary"); - // This does not work. See Lighthouse ticket #1637. - // assertStatus(500, response); - assertTrue(Binary.errorInputStreamClosed); + try { + GET("/binary/getErrorBinary"); + } + finally { + assertTrue(Binary.errorInputStreamClosed); + } } } diff --git a/samples-and-tests/just-test-cases/test/FunctionalTestTest.java b/samples-and-tests/just-test-cases/test/FunctionalTestTest.java index 3916be0379..28c53d07dc 100644 --- a/samples-and-tests/just-test-cases/test/FunctionalTestTest.java +++ b/samples-and-tests/just-test-cases/test/FunctionalTestTest.java @@ -3,6 +3,7 @@ import play.test.*; import play.libs.WS; import play.mvc.Http.*; +import play.mvc.results.*; import models.*; import java.util.HashMap; @@ -126,5 +127,72 @@ public void testGettingStaticFile() { Response response = GET("http://localhost:9003/public/session.test?req=1"); assertIsOk(response); } -} + /** + * A simple call that should always work. + */ + @Test + public void testOk() { + final Response response = GET("/status/ok/"); + assertStatus(200, response); + assertContentEquals("Okay", response); + } + + /** + * When a route is called that is not even defined, an exception is expected. + */ + @Test(expected = NotFound.class) + public void testNoRoute() { + GET("/status/route-not-defined/"); + } + + /** + * When a defined route is called but the controller decides to render a 404, + * the test code is expected to pass and we can assert on the status. + */ + @Test + public void testNotFound() { + final Response response = GET("/status/not-found/"); + assertStatus(404, response); + } + + /** + * When a controller throws a normal exception, an exception is expected in + * the test method as well. + */ + @Test(expected = UnsupportedOperationException.class) + public void testFailure() { + GET("/status/failure/"); + } + + /** + * When a controller renders a non-standard result code (which is, actually, implemented + * through exception), the call is expected to pass and we can assert on the status. + */ + @Test + public void testUnauthorized() { + final Response response = GET("/status/unauthorized/"); + assertStatus(401, response); + } + + /** + * Even when a controller makes use of continuations, e.g. by calling and waiting for a + * job, it is expected that we can assert on the status code. + */ + @Test + public void testContinuationCustomStatus() { + final Response response = POST("/status/job/"); + assertStatus(201, response); + } + + /** + * Even when a controller makes use of continuations, e.g. by calling and waiting for a + * job, it is expected that we can assert on the content. + */ + @Test + public void testContinuationContent() { + final Response response = POST("/status/job/"); + assertContentEquals("Job completed successfully", response); + } + +}