Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

[#1849] Exceptions from controllers are not detected in FunctionalTests #937

Open
wants to merge 5 commits into
base: 1.4.x
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
39 changes: 38 additions & 1 deletion framework/src/play/test/FunctionalTest.java
Original file line number Diff line number Diff line change
Expand Up @@ -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;

Expand Down Expand Up @@ -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 {
Expand Down Expand Up @@ -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<String, Http.Cookie>();
}
Expand All @@ -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);
Expand Down
36 changes: 36 additions & 0 deletions samples-and-tests/just-test-cases/app/controllers/StatusCodes.java
Original file line number Diff line number Diff line change
@@ -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<String>() {
@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");
}

}
Original file line number Diff line number Diff line change
Expand Up @@ -163,14 +163,14 @@ public static void rollbackWithContinuationsThatWorks() {

public static void streamedResult() {
response.contentType = "text/html";
response.writeChunk("<h1>This page should load progressively in about 3 second</h1>");
response.writeChunk("<h1>This page should load progressively in about a few seconds</h1>");
long s = System.currentTimeMillis();
for(int i=0; i<100; i++) {
await(10);
response.writeChunk("<h2>Hello " + i + "</h2>");
}
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() {
Expand Down Expand Up @@ -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));
}
}
};
Expand Down
6 changes: 6 additions & 0 deletions samples-and-tests/just-test-cases/conf/routes
Original file line number Diff line number Diff line change
Expand Up @@ -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
14 changes: 8 additions & 6 deletions samples-and-tests/just-test-cases/test/BinaryTest.java
Original file line number Diff line number Diff line change
Expand Up @@ -28,7 +28,7 @@ public void setUp() {
Response deletedResponse = GET(deleteURL);
assertStatus(200, deletedResponse);
}

@Test
public void testUploadSomething() {
URL imageURL = reverse(); {
Expand Down Expand Up @@ -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);
}
}
}
70 changes: 69 additions & 1 deletion samples-and-tests/just-test-cases/test/FunctionalTestTest.java
Original file line number Diff line number Diff line change
Expand Up @@ -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;
Expand Down Expand Up @@ -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);
}

}