4 * Base class for testing special pages.
9 * @author Jeroen De Dauw < jeroendedauw@gmail.com >
10 * @author Daniel Kinzler
11 * @author Adam Shorland
12 * @author Thiemo Mättig
14 abstract class SpecialPageTestBase
extends MediaWikiTestCase
{
18 protected function setUp() {
21 $this->obLevel
= ob_get_level();
24 protected function tearDown() {
25 $obLevel = ob_get_level();
27 while ( ob_get_level() > $this->obLevel
) {
31 if ( $obLevel !== $this->obLevel
) {
33 "Test changed output buffer level: was {$this->obLevel} before test, but $obLevel after test."
41 * Returns a new instance of the special page under test.
45 abstract protected function newSpecialPage();
48 * @param string $subPage The subpage parameter to call the page with
49 * @param WebRequest|null $request Web request that may contain URL parameters, etc
50 * @param Language|string|null $language The language which should be used in the context
51 * @param User|null $user The user which should be used in the context of this special page
54 * @return array( string, WebResponse ) A two-elements array containing the HTML output
55 * generated by the special page as well as the response object.
57 protected function executeSpecialPage(
59 WebRequest
$request = null,
63 $context = $this->newContext( $request, $language, $user );
65 $output = new OutputPage( $context );
66 $context->setOutput( $output );
68 $page = $this->newSpecialPage();
69 $page->setContext( $context );
70 $output->setTitle( $page->getPageTitle() );
72 $html = $this->getHTMLFromSpecialPage( $page, $subPage );
73 $response = $context->getRequest()->response();
75 if ( $response instanceof FauxResponse
) {
76 $code = $response->getStatusCode();
79 $response->header( 'Status: ' . $code . ' ' . HttpStatus
::getMessage( $code ) );
83 return array( $html, $response );
87 * @param WebRequest|null $request
88 * @param Language|string|null $language
89 * @param User|null $user
91 * @return DerivativeContext
93 private function newContext(
94 WebRequest
$request = null,
98 $context = new DerivativeContext( RequestContext
::getMain() );
100 $context->setRequest( $request ?
: new FauxRequest() );
102 if ( $language !== null ) {
103 $context->setLanguage( $language );
106 if ( $user !== null ) {
107 $context->setUser( $user );
110 $this->setEditTokenFromUser( $context );
116 * If we are trying to edit and no token is set, supply one.
118 * @param DerivativeContext $context
120 private function setEditTokenFromUser( DerivativeContext
$context ) {
121 $request = $context->getRequest();
123 // Edits via GET are a security issue and should not succeed. On the other hand, not all
124 // POST requests are edits, but should ignore unused parameters.
125 if ( !$request->getCheck( 'wpEditToken' ) && $request->wasPosted() ) {
126 $request->setVal( 'wpEditToken', $context->getUser()->getEditToken() );
131 * @param SpecialPage $page
132 * @param string $subPage
135 * @return string HTML
137 private function getHTMLFromSpecialPage( SpecialPage
$page, $subPage ) {
141 $page->execute( $subPage );
143 $output = $page->getOutput();
145 if ( $output->getRedirect() !== '' ) {
147 $html = ob_get_contents();
148 } elseif ( $output->isDisabled() ) {
149 $html = ob_get_contents();
151 $html = $output->getHTML();
153 } catch ( Exception
$ex ) {
156 // Re-throw exception after "finally" handling because PHP 5.3 doesn't have "finally".