2008-08-09 03:19:54 +00:00
|
|
|
<?php
|
|
|
|
|
|
|
|
/**
|
2008-10-30 22:03:21 +00:00
|
|
|
* Tests for RequestHandler and HTTPRequest.
|
2008-08-09 03:19:54 +00:00
|
|
|
* We've set up a simple URL handling model based on
|
|
|
|
*/
|
|
|
|
class RequestHandlingTest extends SapphireTest {
|
|
|
|
static $fixture_file = null;
|
|
|
|
|
|
|
|
function testMethodCallingOnController() {
|
|
|
|
/* Calling a controller works just like it always has */
|
|
|
|
$response = Director::test("testGoodBase1");
|
|
|
|
$this->assertEquals("This is the controller", $response->getBody());
|
|
|
|
|
|
|
|
/* ID and OtherID are extracted from the URL and passed in $request->params. */
|
|
|
|
$response = Director::test("testGoodBase1/method/1/2");
|
|
|
|
$this->assertEquals("This is a method on the controller: 1, 2", $response->getBody());
|
|
|
|
|
|
|
|
/* In addition, these values are availalbe in $controller->urlParams. This is mainly for backward compatability. */
|
|
|
|
$response = Director::test("testGoodBase1/legacymethod/3/4");
|
|
|
|
$this->assertEquals("\$this->urlParams can be used, for backward compatibility: 3, 4", $response->getBody());
|
|
|
|
}
|
|
|
|
|
|
|
|
function testPostRequests() {
|
|
|
|
/* The HTTP Request handler can trigger special behaviour for GET and POST. */
|
|
|
|
$response = Director::test("testGoodBase1/TestForm", array("MyField" => 3), null, "POST");
|
|
|
|
$this->assertEquals("Form posted", $response->getBody());
|
|
|
|
|
|
|
|
$response = Director::test("testGoodBase1/TestForm");
|
|
|
|
$this->assertEquals("Get request on form", $response->getBody());
|
|
|
|
}
|
|
|
|
|
|
|
|
function testRequestHandlerChaining() {
|
|
|
|
/* Request handlers can be chained, from Director to Controller to Form to FormField. Here, we can make a get
|
|
|
|
request on a FormField. */
|
|
|
|
$response = Director::test("testGoodBase1/TestForm/fields/MyField");
|
|
|
|
$this->assertEquals("MyField requested", $response->getBody());
|
|
|
|
|
|
|
|
/* We can also make a POST request on a form field, which could be used for in-place editing, for example. */
|
|
|
|
$response = Director::test("testGoodBase1/TestForm/fields/MyField" ,array("MyField" => 5));
|
|
|
|
$this->assertEquals("MyField posted, update to 5", $response->getBody());
|
|
|
|
}
|
|
|
|
|
|
|
|
function testBadBase() {
|
|
|
|
/* Without a double-slash indicator in the URL, the entire URL is popped off the stack. The controller's default
|
|
|
|
action handlers have been designed for this to an extend: simple actions can still be called. This is the set-up
|
|
|
|
of URL rules written before this new request handler. */
|
|
|
|
$response = Director::test("testBadBase/method/1/2");
|
|
|
|
$this->assertEquals("This is a method on the controller: 1, 2", $response->getBody());
|
|
|
|
|
|
|
|
$response = Director::test("testBadBase/TestForm", array("MyField" => 3), null, "POST");
|
|
|
|
$this->assertEquals("Form posted", $response->getBody());
|
|
|
|
|
|
|
|
/* It won't, however, let you chain requests to access methods on forms, or form fields. In order to do that,
|
|
|
|
you need to have a // marker in your URL parsing rule */
|
|
|
|
$response = Director::test("testBadBase/TestForm/fields/MyField");
|
|
|
|
$this->assertNotEquals("MyField requested", $response->getBody());
|
|
|
|
}
|
2008-09-16 20:37:46 +00:00
|
|
|
|
|
|
|
function testBaseWithExtension() {
|
|
|
|
/* Rules with an extension always default to the index() action */
|
|
|
|
$response = Director::test("testBaseWithExtension/virtualfile.xml");
|
|
|
|
$this->assertEquals("This is the controller", $response->getBody());
|
|
|
|
|
|
|
|
/* Without the extension, the methodname should be matched */
|
|
|
|
$response = Director::test("testBaseWithExtension/virtualfile");
|
|
|
|
$this->assertEquals("This is the virtualfile method", $response->getBody());
|
|
|
|
}
|
2008-10-05 19:21:35 +00:00
|
|
|
|
|
|
|
function testNestedBase() {
|
|
|
|
/* Nested base should leave out the two parts and correctly map arguments */
|
|
|
|
$response = Director::test("testParentBase/testChildBase/method/1/2");
|
|
|
|
$this->assertEquals("This is a method on the controller: 1, 2", $response->getBody());
|
|
|
|
}
|
2008-10-30 22:28:01 +00:00
|
|
|
|
|
|
|
function testInheritedUrlHandlers() {
|
|
|
|
/* $url_handlers can be defined on any class, and */
|
|
|
|
$response = Director::test("testGoodBase1/TestForm/fields/SubclassedField/something");
|
|
|
|
$this->assertEquals("customSomething", $response->getBody());
|
|
|
|
|
|
|
|
/* However, if the subclass' url_handlers don't match, then the parent class' url_handlers will be used */
|
|
|
|
$response = Director::test("testGoodBase1/TestForm/fields/SubclassedField");
|
|
|
|
$this->assertEquals("SubclassedField requested", $response->getBody());
|
|
|
|
}
|
2008-10-31 02:16:51 +00:00
|
|
|
|
|
|
|
function testDisallowedExtendedActions() {
|
|
|
|
/* Actions on magic methods are only accessible if explicitly allowed on the controller. */
|
|
|
|
$response = Director::test("testGoodBase1/extendedMethod");
|
|
|
|
$this->assertEquals(403, $response->getStatusCode());
|
|
|
|
|
|
|
|
/* Actions on an extension are allowed because they specifically provided appropriate allowed_actions items */
|
|
|
|
$response = Director::test("testGoodBase1/otherExtendedMethod");
|
|
|
|
$this->assertEquals("otherExtendedMethod", $response->getBody());
|
|
|
|
|
|
|
|
/* The failoverMethod action wasn't explicitly listed and so isnt' allowed */
|
|
|
|
$response = Director::test("testGoodBase1/failoverMethod");
|
|
|
|
$this->assertEquals(403, $response->getStatusCode());
|
|
|
|
|
|
|
|
/* However, on RequestHandlingTest_AllowedController it has been explicitly allowed */
|
|
|
|
$response = Director::test("RequestHandlingTest_AllowedController/failoverMethod");
|
|
|
|
$this->assertEquals("failoverMethod", $response->getBody());
|
|
|
|
|
|
|
|
/* The action on the extension has also been explicitly allowed even though it wasn't on the extension */
|
|
|
|
$response = Director::test("RequestHandlingTest_AllowedController/extendedMethod");
|
|
|
|
$this->assertEquals("extendedMethod", $response->getBody());
|
|
|
|
|
|
|
|
}
|
2009-06-27 08:48:44 +00:00
|
|
|
|
|
|
|
public function testHTTPException() {
|
|
|
|
$exception = Director::test('RequestHandlingTest_Controller/throwexception');
|
|
|
|
$this->assertEquals(400, $exception->getStatusCode());
|
|
|
|
$this->assertEquals('This request was invalid.', $exception->getBody());
|
|
|
|
|
|
|
|
$responseException = (Director::test('RequestHandlingTest_Controller/throwresponseexception'));
|
|
|
|
$this->assertEquals(500, $responseException->getStatusCode());
|
|
|
|
$this->assertEquals('There was an internal server error.', $responseException->getBody());
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testHTTPError() {
|
|
|
|
$response = Director::test('RequestHandlingTest_Controller/throwhttperror');
|
|
|
|
$this->assertEquals(404, $response->getStatusCode());
|
|
|
|
$this->assertEquals('This page does not exist.', $response->getBody());
|
|
|
|
}
|
|
|
|
|
2008-08-09 03:19:54 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Director rules for the test
|
|
|
|
*/
|
|
|
|
Director::addRules(50, array(
|
|
|
|
// If we don't request any variables, then the whole URL will get shifted off. This is fine, but it means that the
|
|
|
|
// controller will have to parse the Action from the URL itself.
|
|
|
|
'testGoodBase1' => "RequestHandlingTest_Controller",
|
|
|
|
|
|
|
|
// The double-slash indicates how much of the URL should be shifted off the stack. This is important for dealing
|
|
|
|
// with nested request handlers appropriately.
|
|
|
|
'testGoodBase2//$Action/$ID/$OtherID' => "RequestHandlingTest_Controller",
|
|
|
|
|
|
|
|
// By default, the entire URL will be shifted off. This creates a bit of backward-incompatability, but makes the
|
|
|
|
// URL rules much more explicit.
|
|
|
|
'testBadBase/$Action/$ID/$OtherID' => "RequestHandlingTest_Controller",
|
2008-09-16 20:37:46 +00:00
|
|
|
|
|
|
|
// Rules with an extension always default to the index() action
|
|
|
|
'testBaseWithExtension/virtualfile.xml' => "RequestHandlingTest_Controller",
|
|
|
|
|
|
|
|
// Without the extension, the methodname should be matched
|
|
|
|
'testBaseWithExtension//$Action/$ID/$OtherID' => "RequestHandlingTest_Controller",
|
2008-10-05 19:21:35 +00:00
|
|
|
|
|
|
|
// Test nested base
|
|
|
|
'testParentBase/testChildBase//$Action/$ID/$OtherID' => "RequestHandlingTest_Controller",
|
2008-08-09 03:19:54 +00:00
|
|
|
));
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Controller for the test
|
|
|
|
*/
|
|
|
|
class RequestHandlingTest_Controller extends Controller {
|
|
|
|
static $url_handlers = array(
|
|
|
|
// The double-slash is need here to ensure that
|
|
|
|
'$Action//$ID/$OtherID' => "handleAction",
|
|
|
|
);
|
2008-10-31 02:16:51 +00:00
|
|
|
|
|
|
|
static $extensions = array(
|
|
|
|
'RequestHandlingTest_ControllerExtension',
|
|
|
|
'RequestHandlingTest_AllowedControllerExtension',
|
|
|
|
);
|
|
|
|
|
|
|
|
function __construct() {
|
|
|
|
$this->failover = new RequestHandlingTest_ControllerFailover();
|
|
|
|
parent::__construct();
|
|
|
|
}
|
2008-08-09 03:19:54 +00:00
|
|
|
|
|
|
|
function index($request) {
|
|
|
|
return "This is the controller";
|
|
|
|
}
|
|
|
|
|
|
|
|
function method($request) {
|
|
|
|
return "This is a method on the controller: " . $request->param('ID') . ', ' . $request->param('OtherID');
|
|
|
|
}
|
|
|
|
|
|
|
|
function legacymethod($request) {
|
|
|
|
return "\$this->urlParams can be used, for backward compatibility: " . $this->urlParams['ID'] . ', ' . $this->urlParams['OtherID'];
|
|
|
|
}
|
|
|
|
|
2008-09-16 20:37:46 +00:00
|
|
|
function virtualfile($request) {
|
|
|
|
return "This is the virtualfile method";
|
|
|
|
}
|
|
|
|
|
2008-08-09 03:19:54 +00:00
|
|
|
function TestForm() {
|
|
|
|
return new RequestHandlingTest_Form($this, "TestForm", new FieldSet(
|
2008-10-30 22:28:01 +00:00
|
|
|
new RequestHandlingTest_FormField("MyField"),
|
|
|
|
new RequestHandlingTest_SubclassedFormField("SubclassedField")
|
2008-08-09 03:19:54 +00:00
|
|
|
), new FieldSet(
|
|
|
|
new FormAction("myAction")
|
|
|
|
));
|
|
|
|
}
|
2009-06-27 08:48:44 +00:00
|
|
|
|
|
|
|
public function throwexception() {
|
|
|
|
throw new HTTPResponse_Exception('This request was invalid.', 400);
|
|
|
|
}
|
|
|
|
|
|
|
|
public function throwresponseexception() {
|
|
|
|
throw new HTTPResponse_Exception(new HTTPResponse('There was an internal server error.', 500));
|
|
|
|
}
|
|
|
|
|
|
|
|
public function throwhttperror() {
|
|
|
|
$this->httpError(404, 'This page does not exist.');
|
|
|
|
}
|
|
|
|
|
2008-08-09 03:19:54 +00:00
|
|
|
}
|
|
|
|
|
2008-10-31 02:16:51 +00:00
|
|
|
/**
|
|
|
|
* Simple extension for the test controller
|
|
|
|
*/
|
|
|
|
class RequestHandlingTest_ControllerExtension extends Extension {
|
|
|
|
function extendedMethod() {
|
|
|
|
return "extendedMethod";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Controller for the test
|
|
|
|
*/
|
|
|
|
class RequestHandlingTest_AllowedController extends Controller {
|
|
|
|
static $url_handlers = array(
|
|
|
|
// The double-slash is need here to ensure that
|
|
|
|
'$Action//$ID/$OtherID' => "handleAction",
|
|
|
|
);
|
|
|
|
|
|
|
|
static $allowed_actions = array(
|
|
|
|
'failoverMethod', // part of the failover object
|
|
|
|
'extendedMethod', // part of the RequestHandlingTest_ControllerExtension object
|
|
|
|
);
|
|
|
|
|
|
|
|
static $extensions = array(
|
|
|
|
'RequestHandlingTest_ControllerExtension',
|
|
|
|
'RequestHandlingTest_AllowedControllerExtension',
|
|
|
|
);
|
|
|
|
|
|
|
|
function __construct() {
|
|
|
|
$this->failover = new RequestHandlingTest_ControllerFailover();
|
|
|
|
parent::__construct();
|
|
|
|
}
|
|
|
|
|
|
|
|
function index($request) {
|
|
|
|
return "This is the controller";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Simple extension for the test controller - with allowed_actions define
|
|
|
|
*/
|
|
|
|
class RequestHandlingTest_AllowedControllerExtension extends Extension {
|
|
|
|
static $allowed_actions = array(
|
|
|
|
'otherExtendedMethod'
|
|
|
|
);
|
|
|
|
|
|
|
|
function otherExtendedMethod() {
|
|
|
|
return "otherExtendedMethod";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
class RequestHandlingTest_ControllerFailover extends ViewableData {
|
|
|
|
function failoverMethod() {
|
|
|
|
return "failoverMethod";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2008-08-09 03:19:54 +00:00
|
|
|
/**
|
|
|
|
* Form for the test
|
|
|
|
*/
|
|
|
|
class RequestHandlingTest_Form extends Form {
|
|
|
|
static $url_handlers = array(
|
|
|
|
'fields/$FieldName' => 'handleField',
|
|
|
|
"POST " => "handleSubmission",
|
|
|
|
"GET " => "handleGet",
|
|
|
|
);
|
|
|
|
|
2008-09-11 00:15:31 +00:00
|
|
|
// These are a different case from those in url_handlers to confirm that it's all case-insensitive
|
|
|
|
static $allowed_actions = array(
|
|
|
|
'handlesubmission',
|
|
|
|
'handlefield',
|
|
|
|
'handleget',
|
|
|
|
);
|
|
|
|
|
2008-08-09 03:19:54 +00:00
|
|
|
function handleField($request) {
|
|
|
|
return $this->dataFieldByName($request->param('FieldName'));
|
|
|
|
}
|
|
|
|
|
|
|
|
function handleSubmission($request) {
|
|
|
|
return "Form posted";
|
|
|
|
}
|
|
|
|
|
|
|
|
function handleGet($request) {
|
|
|
|
return "Get request on form";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Form field for the test
|
|
|
|
*/
|
|
|
|
class RequestHandlingTest_FormField extends FormField {
|
|
|
|
static $url_handlers = array(
|
|
|
|
"POST " => "handleInPlaceEdit",
|
|
|
|
'' => 'handleField',
|
|
|
|
'$Action' => '$Action',
|
|
|
|
);
|
2008-09-11 00:15:31 +00:00
|
|
|
|
|
|
|
// These contain uppercase letters to test that allowed_actions doesn't need to be all lowercase
|
|
|
|
static $allowed_actions = array(
|
|
|
|
'TEST',
|
|
|
|
'handleField',
|
|
|
|
'handleInPLACEEDIT',
|
|
|
|
);
|
2008-08-09 03:19:54 +00:00
|
|
|
|
|
|
|
function test() {
|
|
|
|
return "Test method on $this->name";
|
|
|
|
}
|
|
|
|
|
|
|
|
function handleField() {
|
|
|
|
return "$this->name requested";
|
|
|
|
}
|
|
|
|
|
|
|
|
function handleInPlaceEdit($request) {
|
|
|
|
return "$this->name posted, update to " . $request->postVar($this->name);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2008-10-30 22:28:01 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Form field for the test
|
|
|
|
*/
|
|
|
|
class RequestHandlingTest_SubclassedFormField extends RequestHandlingTest_FormField {
|
|
|
|
// We have some url_handlers defined that override RequestHandlingTest_FormField handlers.
|
|
|
|
// We will confirm that the url_handlers inherit.
|
|
|
|
static $url_handlers = array(
|
|
|
|
'something' => 'customSomething',
|
|
|
|
);
|
2008-10-31 02:16:51 +00:00
|
|
|
|
2008-10-30 22:28:01 +00:00
|
|
|
|
|
|
|
function customSomething() {
|
|
|
|
return "customSomething";
|
|
|
|
}
|
2009-06-27 08:48:44 +00:00
|
|
|
}
|