2020-03-06 15:53:01 +00:00
|
|
|
<?php
|
|
|
|
|
|
|
|
|
|
namespace MediaWiki\Tests\Rest\Handler;
|
|
|
|
|
|
2024-09-25 16:17:29 +00:00
|
|
|
use MediaWiki\Api\ApiUsageException;
|
2023-09-20 07:54:42 +00:00
|
|
|
use MediaWiki\Config\HashConfig;
|
2024-08-06 13:40:20 +00:00
|
|
|
use MediaWiki\Content\WikitextContent;
|
2023-06-19 19:21:47 +00:00
|
|
|
use MediaWiki\MainConfigNames;
|
2024-04-02 21:15:08 +00:00
|
|
|
use MediaWiki\Message\Message;
|
2020-03-06 15:53:01 +00:00
|
|
|
use MediaWiki\Rest\Handler\CreationHandler;
|
|
|
|
|
use MediaWiki\Rest\LocalizedHttpException;
|
|
|
|
|
use MediaWiki\Rest\RequestData;
|
2022-04-14 20:54:04 +00:00
|
|
|
use MediaWiki\Revision\MutableRevisionRecord;
|
2020-03-06 15:53:01 +00:00
|
|
|
use MediaWiki\Revision\RevisionLookup;
|
2022-04-09 18:09:35 +00:00
|
|
|
use MediaWiki\Revision\SlotRecord;
|
2023-08-25 12:29:41 +00:00
|
|
|
use MediaWiki\Status\Status;
|
2021-05-03 20:38:48 +00:00
|
|
|
use MediaWiki\Tests\Unit\DummyServicesTrait;
|
2020-03-06 15:53:01 +00:00
|
|
|
use MediaWikiIntegrationTestCase;
|
2021-01-13 17:34:01 +00:00
|
|
|
use MockTitleTrait;
|
2020-03-06 15:53:01 +00:00
|
|
|
use PHPUnit\Framework\MockObject\MockObject;
|
2024-03-12 10:52:13 +00:00
|
|
|
use Wikimedia\Message\DataMessageValue;
|
2020-03-06 15:53:01 +00:00
|
|
|
use Wikimedia\Message\MessageValue;
|
|
|
|
|
use Wikimedia\Message\ParamType;
|
|
|
|
|
use Wikimedia\Message\ScalarParam;
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* @covers \MediaWiki\Rest\Handler\CreationHandler
|
|
|
|
|
*/
|
|
|
|
|
class CreationHandlerTest extends MediaWikiIntegrationTestCase {
|
|
|
|
|
use ActionModuleBasedHandlerTestTrait;
|
2021-05-03 20:38:48 +00:00
|
|
|
use DummyServicesTrait;
|
2021-01-13 17:34:01 +00:00
|
|
|
use MockTitleTrait;
|
2020-03-06 15:53:01 +00:00
|
|
|
|
2020-03-12 12:54:51 +00:00
|
|
|
private function newHandler( $resultData, $throwException = null, $csrfSafe = false ) {
|
2020-03-06 15:53:01 +00:00
|
|
|
$config = new HashConfig( [
|
2023-06-19 19:21:47 +00:00
|
|
|
MainConfigNames::RightsUrl => 'https://creativecommons.org/licenses/by-sa/4.0/',
|
|
|
|
|
MainConfigNames::RightsText => 'CC-BY-SA 4.0'
|
2020-03-06 15:53:01 +00:00
|
|
|
] );
|
|
|
|
|
|
2022-12-14 23:40:12 +00:00
|
|
|
// Claims that wikitext and plaintext are defined, but trying to get the actual
|
|
|
|
|
// content handlers would break
|
|
|
|
|
$contentHandlerFactory = $this->getDummyContentHandlerFactory( [
|
|
|
|
|
CONTENT_MODEL_WIKITEXT => true,
|
|
|
|
|
CONTENT_MODEL_TEXT => true,
|
|
|
|
|
] );
|
2020-03-06 15:53:01 +00:00
|
|
|
|
2021-05-03 20:38:48 +00:00
|
|
|
$titleCodec = $this->getDummyMediaWikiTitleCodec();
|
2020-03-06 15:53:01 +00:00
|
|
|
|
|
|
|
|
/** @var RevisionLookup|MockObject $revisionLookup */
|
|
|
|
|
$revisionLookup = $this->createNoOpMock( RevisionLookup::class, [ 'getRevisionById' ] );
|
|
|
|
|
$revisionLookup->method( 'getRevisionById' )
|
|
|
|
|
->willReturnCallback( function ( $id ) {
|
|
|
|
|
$title = $this->makeMockTitle( __CLASS__ );
|
|
|
|
|
$rev = new MutableRevisionRecord( $title );
|
|
|
|
|
$rev->setId( $id );
|
|
|
|
|
$rev->setContent( SlotRecord::MAIN, new WikitextContent( "Content of revision $id" ) );
|
|
|
|
|
return $rev;
|
|
|
|
|
} );
|
|
|
|
|
|
|
|
|
|
$handler = new CreationHandler(
|
|
|
|
|
$config,
|
|
|
|
|
$contentHandlerFactory,
|
|
|
|
|
$titleCodec,
|
|
|
|
|
$titleCodec,
|
|
|
|
|
$revisionLookup
|
|
|
|
|
);
|
2020-03-12 12:54:51 +00:00
|
|
|
|
|
|
|
|
$apiMain = $this->getApiMain( $csrfSafe );
|
|
|
|
|
$dummyModule = $this->getDummyApiModule( $apiMain, 'edit', $resultData, $throwException );
|
|
|
|
|
|
|
|
|
|
$handler->setApiMain( $apiMain );
|
2020-03-06 15:53:01 +00:00
|
|
|
$handler->overrideActionModule(
|
|
|
|
|
'edit',
|
|
|
|
|
'action',
|
2020-03-12 12:54:51 +00:00
|
|
|
$dummyModule
|
2020-03-06 15:53:01 +00:00
|
|
|
);
|
|
|
|
|
|
|
|
|
|
return $handler;
|
|
|
|
|
}
|
|
|
|
|
|
2023-03-23 11:36:19 +00:00
|
|
|
public static function provideExecute() {
|
2020-07-22 19:12:00 +00:00
|
|
|
// NOTE: Prefix hard coded in a fake for Router::getRouteUrl() in HandlerTestTrait
|
|
|
|
|
$baseUrl = 'https://wiki.example.com/rest/v1/page/';
|
|
|
|
|
|
2020-03-12 12:54:51 +00:00
|
|
|
yield "create with token" => [
|
2020-03-06 15:53:01 +00:00
|
|
|
[ // Request data received by CreationHandler
|
|
|
|
|
'method' => 'POST',
|
|
|
|
|
'headers' => [
|
|
|
|
|
'Content-Type' => 'application/json',
|
|
|
|
|
],
|
|
|
|
|
'bodyContents' => json_encode( [
|
|
|
|
|
'token' => 'TOKEN',
|
|
|
|
|
'title' => 'Foo',
|
|
|
|
|
'source' => 'Lorem Ipsum',
|
|
|
|
|
'comment' => 'Testing'
|
|
|
|
|
] ),
|
|
|
|
|
],
|
|
|
|
|
[ // Fake request expected to be passed into ApiEditPage
|
|
|
|
|
'title' => 'Foo',
|
|
|
|
|
'text' => 'Lorem Ipsum',
|
|
|
|
|
'summary' => 'Testing',
|
|
|
|
|
'createonly' => '1',
|
|
|
|
|
],
|
|
|
|
|
[ // Mock response returned by ApiEditPage
|
|
|
|
|
"edit" => [
|
|
|
|
|
"new" => true,
|
|
|
|
|
"result" => "Success",
|
|
|
|
|
"pageid" => 94542,
|
2020-07-22 19:12:00 +00:00
|
|
|
"title" => "Foo",
|
2020-03-06 15:53:01 +00:00
|
|
|
"contentmodel" => "wikitext",
|
|
|
|
|
"oldrevid" => 0,
|
|
|
|
|
"newrevid" => 371707,
|
|
|
|
|
"newtimestamp" => "2018-12-18T16:59:42Z",
|
|
|
|
|
]
|
|
|
|
|
],
|
|
|
|
|
[ // Response expected to be generated by CreationHandler
|
|
|
|
|
'id' => 94542,
|
2021-05-03 20:38:48 +00:00
|
|
|
'title' => 'Foo',
|
|
|
|
|
'key' => 'Foo',
|
2020-03-06 15:53:01 +00:00
|
|
|
'content_model' => 'wikitext',
|
|
|
|
|
'latest' => [
|
|
|
|
|
'id' => 371707,
|
|
|
|
|
'timestamp' => "2018-12-18T16:59:42Z"
|
|
|
|
|
],
|
|
|
|
|
'license' => [
|
|
|
|
|
'url' => 'https://creativecommons.org/licenses/by-sa/4.0/',
|
|
|
|
|
'title' => 'CC-BY-SA 4.0'
|
|
|
|
|
],
|
|
|
|
|
'source' => 'Content of revision 371707'
|
2020-03-12 12:54:51 +00:00
|
|
|
],
|
2020-07-22 19:12:00 +00:00
|
|
|
$baseUrl . 'Foo',
|
2020-03-12 12:54:51 +00:00
|
|
|
false
|
2020-03-06 15:53:01 +00:00
|
|
|
];
|
|
|
|
|
|
|
|
|
|
yield "create with model" => [
|
|
|
|
|
[ // Request data received by CreationHandler
|
|
|
|
|
'method' => 'POST',
|
|
|
|
|
'headers' => [
|
|
|
|
|
'Content-Type' => 'application/json',
|
|
|
|
|
],
|
|
|
|
|
'bodyContents' => json_encode( [
|
2020-07-22 19:12:00 +00:00
|
|
|
'title' => 'Talk:Foo',
|
2020-03-06 15:53:01 +00:00
|
|
|
'source' => 'Lorem Ipsum',
|
|
|
|
|
'comment' => 'Testing',
|
2020-07-22 19:12:00 +00:00
|
|
|
'content_model' => CONTENT_MODEL_TEXT,
|
2020-03-06 15:53:01 +00:00
|
|
|
] ),
|
|
|
|
|
],
|
|
|
|
|
[ // Fake request expected to be passed into ApiEditPage
|
2020-07-22 19:12:00 +00:00
|
|
|
'title' => 'Talk:Foo',
|
2020-03-06 15:53:01 +00:00
|
|
|
'text' => 'Lorem Ipsum',
|
|
|
|
|
'summary' => 'Testing',
|
2020-07-22 19:12:00 +00:00
|
|
|
'contentmodel' => CONTENT_MODEL_TEXT,
|
2020-03-06 15:53:01 +00:00
|
|
|
'createonly' => '1',
|
2020-03-12 12:54:51 +00:00
|
|
|
'token' => '+\\',
|
|
|
|
|
],
|
|
|
|
|
[ // Mock response returned by ApiEditPage
|
|
|
|
|
"edit" => [
|
|
|
|
|
"new" => true,
|
|
|
|
|
"result" => "Success",
|
|
|
|
|
"pageid" => 94542,
|
2020-07-22 19:12:00 +00:00
|
|
|
"title" => "Talk:Foo",
|
|
|
|
|
"contentmodel" => CONTENT_MODEL_TEXT,
|
2020-03-12 12:54:51 +00:00
|
|
|
"oldrevid" => 0,
|
|
|
|
|
"newrevid" => 371707,
|
|
|
|
|
"newtimestamp" => "2018-12-18T16:59:42Z",
|
|
|
|
|
]
|
|
|
|
|
],
|
|
|
|
|
[ // Response expected to be generated by CreationHandler
|
|
|
|
|
'id' => 94542,
|
2021-05-03 20:38:48 +00:00
|
|
|
'title' => 'Talk:Foo',
|
|
|
|
|
'key' => 'Talk:Foo',
|
2020-07-22 19:12:00 +00:00
|
|
|
'content_model' => CONTENT_MODEL_TEXT,
|
2020-03-12 12:54:51 +00:00
|
|
|
'latest' => [
|
|
|
|
|
'id' => 371707,
|
|
|
|
|
'timestamp' => "2018-12-18T16:59:42Z"
|
|
|
|
|
],
|
|
|
|
|
'license' => [
|
|
|
|
|
'url' => 'https://creativecommons.org/licenses/by-sa/4.0/',
|
|
|
|
|
'title' => 'CC-BY-SA 4.0'
|
|
|
|
|
],
|
|
|
|
|
'source' => 'Content of revision 371707'
|
|
|
|
|
],
|
2020-07-22 19:12:00 +00:00
|
|
|
$baseUrl . 'Talk:Foo',
|
2020-03-12 12:54:51 +00:00
|
|
|
true
|
|
|
|
|
];
|
|
|
|
|
|
|
|
|
|
yield "create without token" => [
|
|
|
|
|
[ // Request data received by CreationHandler
|
|
|
|
|
'method' => 'POST',
|
|
|
|
|
'headers' => [
|
|
|
|
|
'Content-Type' => 'application/json',
|
|
|
|
|
],
|
|
|
|
|
'bodyContents' => json_encode( [
|
2020-07-22 19:12:00 +00:00
|
|
|
'title' => 'foo/bar',
|
2020-03-12 12:54:51 +00:00
|
|
|
'source' => 'Lorem Ipsum',
|
|
|
|
|
'comment' => 'Testing',
|
|
|
|
|
'content_model' => CONTENT_MODEL_WIKITEXT,
|
|
|
|
|
] ),
|
|
|
|
|
],
|
|
|
|
|
[ // Fake request expected to be passed into ApiEditPage
|
2020-07-22 19:12:00 +00:00
|
|
|
'title' => 'foo/bar',
|
2020-03-12 12:54:51 +00:00
|
|
|
'text' => 'Lorem Ipsum',
|
|
|
|
|
'summary' => 'Testing',
|
|
|
|
|
'contentmodel' => 'wikitext',
|
|
|
|
|
'createonly' => '1',
|
|
|
|
|
'token' => '+\\', // use known-good token for current user (anon)
|
2020-03-06 15:53:01 +00:00
|
|
|
],
|
|
|
|
|
[ // Mock response returned by ApiEditPage
|
|
|
|
|
"edit" => [
|
|
|
|
|
"new" => true,
|
|
|
|
|
"result" => "Success",
|
|
|
|
|
"pageid" => 94542,
|
2020-07-22 19:12:00 +00:00
|
|
|
"title" => "Foo/bar",
|
2020-03-06 15:53:01 +00:00
|
|
|
"contentmodel" => "wikitext",
|
|
|
|
|
"oldrevid" => 0,
|
|
|
|
|
"newrevid" => 371707,
|
|
|
|
|
"newtimestamp" => "2018-12-18T16:59:42Z",
|
|
|
|
|
]
|
|
|
|
|
],
|
|
|
|
|
[ // Response expected to be generated by CreationHandler
|
|
|
|
|
'id' => 94542,
|
2021-05-03 20:38:48 +00:00
|
|
|
'title' => 'Foo/bar',
|
|
|
|
|
'key' => 'Foo/bar',
|
2020-07-22 19:12:00 +00:00
|
|
|
'content_model' => 'wikitext',
|
|
|
|
|
'latest' => [
|
|
|
|
|
'id' => 371707,
|
|
|
|
|
'timestamp' => "2018-12-18T16:59:42Z"
|
|
|
|
|
],
|
|
|
|
|
'license' => [
|
|
|
|
|
'url' => 'https://creativecommons.org/licenses/by-sa/4.0/',
|
|
|
|
|
'title' => 'CC-BY-SA 4.0'
|
|
|
|
|
],
|
|
|
|
|
'source' => 'Content of revision 371707'
|
|
|
|
|
],
|
|
|
|
|
$baseUrl . 'Foo%2Fbar',
|
|
|
|
|
true
|
|
|
|
|
];
|
|
|
|
|
|
|
|
|
|
yield "create with space" => [
|
|
|
|
|
[ // Request data received by CreationHandler
|
|
|
|
|
'method' => 'POST',
|
|
|
|
|
'headers' => [
|
|
|
|
|
'Content-Type' => 'application/json',
|
|
|
|
|
],
|
|
|
|
|
'bodyContents' => json_encode( [
|
|
|
|
|
'title' => 'foo (ba+r)',
|
|
|
|
|
'source' => 'Lorem Ipsum',
|
|
|
|
|
'comment' => 'Testing'
|
|
|
|
|
] ),
|
|
|
|
|
],
|
|
|
|
|
[ // Fake request expected to be passed into ApiEditPage
|
|
|
|
|
'title' => 'foo (ba+r)',
|
|
|
|
|
'text' => 'Lorem Ipsum',
|
|
|
|
|
'summary' => 'Testing',
|
|
|
|
|
'createonly' => '1',
|
|
|
|
|
'token' => '+\\', // use known-good token for current user (anon)
|
|
|
|
|
],
|
|
|
|
|
[ // Mock response returned by ApiEditPage
|
|
|
|
|
"edit" => [
|
|
|
|
|
"new" => true,
|
|
|
|
|
"result" => "Success",
|
|
|
|
|
"pageid" => 94542,
|
|
|
|
|
"title" => "Foo (ba+r)",
|
|
|
|
|
"contentmodel" => "wikitext",
|
|
|
|
|
"oldrevid" => 0,
|
|
|
|
|
"newrevid" => 371707,
|
|
|
|
|
"newtimestamp" => "2018-12-18T16:59:42Z",
|
|
|
|
|
]
|
|
|
|
|
],
|
|
|
|
|
[ // Response expected to be generated by CreationHandler
|
|
|
|
|
'id' => 94542,
|
2021-05-03 20:38:48 +00:00
|
|
|
'title' => 'Foo (ba+r)',
|
|
|
|
|
'key' => 'Foo_(ba+r)',
|
2020-03-06 15:53:01 +00:00
|
|
|
'content_model' => 'wikitext',
|
|
|
|
|
'latest' => [
|
|
|
|
|
'id' => 371707,
|
|
|
|
|
'timestamp' => "2018-12-18T16:59:42Z"
|
|
|
|
|
],
|
|
|
|
|
'license' => [
|
|
|
|
|
'url' => 'https://creativecommons.org/licenses/by-sa/4.0/',
|
|
|
|
|
'title' => 'CC-BY-SA 4.0'
|
|
|
|
|
],
|
|
|
|
|
'source' => 'Content of revision 371707'
|
2020-03-12 12:54:51 +00:00
|
|
|
],
|
2020-07-22 19:12:00 +00:00
|
|
|
$baseUrl . 'Foo_(ba%2Br)',
|
2020-03-12 12:54:51 +00:00
|
|
|
true
|
2020-03-06 15:53:01 +00:00
|
|
|
];
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* @dataProvider provideExecute
|
|
|
|
|
*/
|
|
|
|
|
public function testExecute(
|
|
|
|
|
$requestData,
|
|
|
|
|
$expectedActionParams,
|
|
|
|
|
$actionResult,
|
2020-03-12 12:54:51 +00:00
|
|
|
$expectedResponse,
|
2020-07-22 19:12:00 +00:00
|
|
|
$expectedRedirect,
|
2020-03-12 12:54:51 +00:00
|
|
|
$csrfSafe
|
2020-03-06 15:53:01 +00:00
|
|
|
) {
|
|
|
|
|
$request = new RequestData( $requestData );
|
|
|
|
|
|
2020-03-12 12:54:51 +00:00
|
|
|
$handler = $this->newHandler( $actionResult, null, $csrfSafe );
|
2020-03-06 15:53:01 +00:00
|
|
|
|
Make it possible to override the session in REST API tests
The current signature of the various execute methods only takes a
boolean parameter to determine if the session should be safe against
CSRF, but that does not give callers fine-grained control over the
Session object, including setting a specific token.
Also, do not use createNoOpMock in getSession(), since it implies
strong assertions on what methods are called. This way, getSession
can also be used to get a simple mock session that tests may further
manipulate.
Make $csrfSafe parameter of SessionHelperTestTrait::getSession
mandatory. This way, callers are forced to think what makes sense in
each use case. The various methods in HandlerTestTrait now default to
a session that is safe against CSRF. This assumes that most REST
handlers don't care about the session, and that any handler that does
care about the session and where someone needs to test the behaviour
in case of bad/missing token will explicitly provide a Session that
is NOT safe against CSRF.
Typehint the return value of Session(Backend)::getUser so that PHPUnit
will automatically make it return a mock User object even if the method
is not explicitly mocked. Remove a useless PHPUnit assertion -- setting
the return value to be X and then veryfing that is equal to X is a
tautology, and can only fail if the test itself is flawed (as was the
case, since it was using stdClass as the return type for all
methods). Remove the getUser test case altogether, there's no way to
make it work given the DummySessionBackend, and the test isn't that
helpful anyway. More and more methods will have the same issue as soon
as their return value is typehinted.
Follow-up: I2a9215bf909b83564247ded95ecdb4ead0615150
Change-Id: Ic51dc3e7bf47c81f2ac4705308bb9ecd8275bbaf
2023-02-06 15:18:00 +00:00
|
|
|
$response = $this->executeHandler( $handler, $request, [], [], [], [], null, $this->getSession( $csrfSafe ) );
|
2020-03-06 15:53:01 +00:00
|
|
|
|
|
|
|
|
$this->assertSame( 201, $response->getStatusCode() );
|
|
|
|
|
$this->assertSame(
|
2020-07-22 19:12:00 +00:00
|
|
|
$expectedRedirect,
|
2020-03-06 15:53:01 +00:00
|
|
|
$response->getHeaderLine( 'Location' )
|
|
|
|
|
);
|
|
|
|
|
$this->assertSame( 'application/json', $response->getHeaderLine( 'Content-Type' ) );
|
|
|
|
|
|
|
|
|
|
$responseData = json_decode( $response->getBody(), true );
|
|
|
|
|
$this->assertIsArray( $responseData, 'Body must be a JSON array' );
|
|
|
|
|
|
|
|
|
|
// Check parameters passed to ApiEditPage by CreationHandler based on $requestData
|
|
|
|
|
foreach ( $expectedActionParams as $key => $value ) {
|
|
|
|
|
$this->assertSame(
|
|
|
|
|
$value,
|
2020-07-22 19:12:00 +00:00
|
|
|
$handler->getApiMain()->getVal( $key ),
|
2020-03-06 15:53:01 +00:00
|
|
|
"ApiEditPage param: $key"
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Check response that CreationHandler created after receiving $actionResult from ApiEditPage
|
|
|
|
|
foreach ( $expectedResponse as $key => $value ) {
|
|
|
|
|
$this->assertArrayHasKey( $key, $responseData );
|
|
|
|
|
$this->assertSame(
|
|
|
|
|
$value,
|
|
|
|
|
$responseData[ $key ],
|
|
|
|
|
"CreationHandler response field: $key"
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2023-03-23 11:36:19 +00:00
|
|
|
public static function provideBodyValidation() {
|
2020-03-06 15:53:01 +00:00
|
|
|
yield "missing source field" => [
|
|
|
|
|
[ // Request data received by CreationHandler
|
|
|
|
|
'method' => 'POST',
|
|
|
|
|
'headers' => [
|
|
|
|
|
'Content-Type' => 'application/json',
|
|
|
|
|
],
|
|
|
|
|
'bodyContents' => json_encode( [
|
|
|
|
|
'token' => 'TOKEN',
|
|
|
|
|
'title' => 'Foo',
|
|
|
|
|
'comment' => 'Testing',
|
|
|
|
|
'content_model' => CONTENT_MODEL_WIKITEXT,
|
|
|
|
|
] ),
|
|
|
|
|
],
|
2024-04-29 07:00:12 +00:00
|
|
|
MessageValue::new( 'rest-body-validation-error', [
|
2024-03-12 10:52:13 +00:00
|
|
|
DataMessageValue::new( 'paramvalidator-missingparam', [], 'missingparam' )
|
|
|
|
|
->plaintextParams( 'source' )
|
2024-04-29 07:00:12 +00:00
|
|
|
] ),
|
2020-03-06 15:53:01 +00:00
|
|
|
];
|
|
|
|
|
yield "missing comment field" => [
|
|
|
|
|
[ // Request data received by CreationHandler
|
|
|
|
|
'method' => 'POST',
|
|
|
|
|
'headers' => [
|
|
|
|
|
'Content-Type' => 'application/json',
|
|
|
|
|
],
|
|
|
|
|
'bodyContents' => json_encode( [
|
|
|
|
|
'token' => 'TOKEN',
|
|
|
|
|
'title' => 'Foo',
|
|
|
|
|
'source' => 'Lorem Ipsum',
|
|
|
|
|
'content_model' => CONTENT_MODEL_WIKITEXT,
|
|
|
|
|
] ),
|
|
|
|
|
],
|
2024-04-29 07:00:12 +00:00
|
|
|
MessageValue::new( 'rest-body-validation-error', [
|
2024-03-12 10:52:13 +00:00
|
|
|
DataMessageValue::new( 'paramvalidator-missingparam', [], 'missingparam' )
|
|
|
|
|
->plaintextParams( 'comment' )
|
2024-04-29 07:00:12 +00:00
|
|
|
] ),
|
2020-03-06 15:53:01 +00:00
|
|
|
];
|
|
|
|
|
yield "missing title field" => [
|
|
|
|
|
[ // Request data received by CreationHandler
|
|
|
|
|
'method' => 'POST',
|
|
|
|
|
'headers' => [
|
|
|
|
|
'Content-Type' => 'application/json',
|
|
|
|
|
],
|
|
|
|
|
'bodyContents' => json_encode( [
|
|
|
|
|
'token' => 'TOKEN',
|
|
|
|
|
'comment' => 'Testing',
|
|
|
|
|
'source' => 'Lorem Ipsum',
|
|
|
|
|
'content_model' => CONTENT_MODEL_WIKITEXT,
|
|
|
|
|
] ),
|
|
|
|
|
],
|
2024-04-29 07:00:12 +00:00
|
|
|
MessageValue::new( 'rest-body-validation-error', [
|
2024-03-12 10:52:13 +00:00
|
|
|
DataMessageValue::new( 'paramvalidator-missingparam', [], 'missingparam' )
|
|
|
|
|
->plaintextParams( 'title' )
|
2024-04-29 07:00:12 +00:00
|
|
|
] ),
|
2020-03-06 15:53:01 +00:00
|
|
|
];
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* @dataProvider provideBodyValidation
|
|
|
|
|
*/
|
|
|
|
|
public function testBodyValidation( array $requestData, MessageValue $expectedMessage ) {
|
|
|
|
|
$request = new RequestData( $requestData );
|
|
|
|
|
|
|
|
|
|
$handler = $this->newHandler( [] );
|
|
|
|
|
|
|
|
|
|
$exception = $this->executeHandlerAndGetHttpException( $handler, $request );
|
|
|
|
|
|
|
|
|
|
$this->assertSame( 400, $exception->getCode(), 'HTTP status' );
|
|
|
|
|
$this->assertInstanceOf( LocalizedHttpException::class, $exception );
|
|
|
|
|
|
|
|
|
|
/** @var LocalizedHttpException $exception */
|
|
|
|
|
$this->assertEquals( $expectedMessage, $exception->getMessageValue() );
|
|
|
|
|
}
|
|
|
|
|
|
2023-03-23 11:36:19 +00:00
|
|
|
public static function provideHeaderValidation() {
|
2020-03-06 15:53:01 +00:00
|
|
|
yield "bad content type" => [
|
|
|
|
|
[ // Request data received by CreationHandler
|
|
|
|
|
'method' => 'POST',
|
|
|
|
|
'headers' => [
|
|
|
|
|
'Content-Type' => 'text/plain',
|
|
|
|
|
],
|
|
|
|
|
'bodyContents' => json_encode( [
|
|
|
|
|
'title' => 'Foo',
|
|
|
|
|
'source' => 'Lorem Ipsum',
|
|
|
|
|
'comment' => 'Testing',
|
|
|
|
|
'content_model' => CONTENT_MODEL_WIKITEXT,
|
|
|
|
|
] ),
|
|
|
|
|
],
|
|
|
|
|
415
|
|
|
|
|
];
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* @dataProvider provideHeaderValidation
|
|
|
|
|
*/
|
|
|
|
|
public function testHeaderValidation( array $requestData, $expectedStatus ) {
|
|
|
|
|
$request = new RequestData( $requestData );
|
|
|
|
|
|
|
|
|
|
$handler = $this->newHandler( [] );
|
|
|
|
|
|
|
|
|
|
$exception = $this->executeHandlerAndGetHttpException( $handler, $request );
|
|
|
|
|
|
|
|
|
|
$this->assertSame( $expectedStatus, $exception->getCode(), 'HTTP status' );
|
|
|
|
|
}
|
|
|
|
|
|
2021-12-08 14:25:20 +00:00
|
|
|
/*
|
|
|
|
|
* FIXME: Status::newFatal invokes MediaWikiServices, which is not allowed in a dataProvider.
|
|
|
|
|
*/
|
2023-03-23 11:36:19 +00:00
|
|
|
public static function provideErrorMapping() {
|
2020-03-06 15:53:01 +00:00
|
|
|
yield "missingtitle" => [
|
|
|
|
|
new ApiUsageException( null, Status::newFatal( 'apierror-missingtitle' ) ),
|
|
|
|
|
new LocalizedHttpException( new MessageValue( 'apierror-missingtitle' ), 404 ),
|
|
|
|
|
];
|
|
|
|
|
yield "protectedpage" => [
|
|
|
|
|
new ApiUsageException( null, Status::newFatal( 'apierror-protectedpage' ) ),
|
|
|
|
|
new LocalizedHttpException( new MessageValue( 'apierror-protectedpage' ), 403 ),
|
|
|
|
|
];
|
|
|
|
|
yield "articleexists" => [
|
|
|
|
|
new ApiUsageException( null, Status::newFatal( 'apierror-articleexists' ) ),
|
|
|
|
|
new LocalizedHttpException( new MessageValue( 'apierror-articleexists' ), 409 ),
|
|
|
|
|
];
|
|
|
|
|
yield "editconflict" => [
|
|
|
|
|
new ApiUsageException( null, Status::newFatal( 'apierror-editconflict' ) ),
|
|
|
|
|
new LocalizedHttpException( new MessageValue( 'apierror-editconflict' ), 409 ),
|
|
|
|
|
];
|
|
|
|
|
yield "ratelimited" => [
|
|
|
|
|
new ApiUsageException( null, Status::newFatal( 'apierror-ratelimited' ) ),
|
|
|
|
|
new LocalizedHttpException( new MessageValue( 'apierror-ratelimited' ), 429 ),
|
|
|
|
|
];
|
|
|
|
|
yield "badtoken" => [
|
|
|
|
|
new ApiUsageException(
|
|
|
|
|
null,
|
2024-04-02 21:15:08 +00:00
|
|
|
Status::newFatal( 'apierror-badtoken', Message::plaintextParam( 'BAD' ) )
|
2020-03-06 15:53:01 +00:00
|
|
|
),
|
|
|
|
|
new LocalizedHttpException(
|
|
|
|
|
new MessageValue(
|
|
|
|
|
'apierror-badtoken',
|
|
|
|
|
[ new ScalarParam( ParamType::PLAINTEXT, 'BAD' ) ]
|
|
|
|
|
), 403
|
|
|
|
|
),
|
|
|
|
|
];
|
|
|
|
|
|
|
|
|
|
// Unmapped errors should be passed through with a status 400.
|
|
|
|
|
yield "no-direct-editing" => [
|
|
|
|
|
new ApiUsageException( null, Status::newFatal( 'apierror-no-direct-editing' ) ),
|
|
|
|
|
new LocalizedHttpException( new MessageValue( 'apierror-no-direct-editing' ), 400 ),
|
|
|
|
|
];
|
|
|
|
|
yield "badformat" => [
|
|
|
|
|
new ApiUsageException( null, Status::newFatal( 'apierror-badformat' ) ),
|
|
|
|
|
new LocalizedHttpException( new MessageValue( 'apierror-badformat' ), 400 ),
|
|
|
|
|
];
|
|
|
|
|
yield "emptypage" => [
|
|
|
|
|
new ApiUsageException( null, Status::newFatal( 'apierror-emptypage' ) ),
|
|
|
|
|
new LocalizedHttpException( new MessageValue( 'apierror-emptypage' ), 400 ),
|
|
|
|
|
];
|
|
|
|
|
}
|
|
|
|
|
|
2021-12-08 14:25:20 +00:00
|
|
|
public function testErrorMapping() {
|
|
|
|
|
$provideErrorMapping = $this->provideErrorMapping();
|
|
|
|
|
foreach ( $provideErrorMapping as $expected ) {
|
|
|
|
|
$apiUsageException = $expected[0];
|
|
|
|
|
$expectedHttpException = $expected[1];
|
|
|
|
|
$requestData = [ // Request data received by CreationHandler
|
|
|
|
|
'method' => 'POST',
|
|
|
|
|
'headers' => [
|
|
|
|
|
'Content-Type' => 'application/json',
|
|
|
|
|
],
|
|
|
|
|
'bodyContents' => json_encode( [
|
|
|
|
|
'title' => 'Foo',
|
|
|
|
|
'source' => 'Lorem Ipsum',
|
|
|
|
|
'comment' => 'Testing',
|
|
|
|
|
'content_model' => CONTENT_MODEL_WIKITEXT,
|
|
|
|
|
] ),
|
|
|
|
|
];
|
|
|
|
|
$request = new RequestData( $requestData );
|
2020-03-06 15:53:01 +00:00
|
|
|
|
2021-12-08 14:25:20 +00:00
|
|
|
$handler = $this->newHandler( [], $apiUsageException );
|
2020-03-06 15:53:01 +00:00
|
|
|
|
2021-12-08 14:25:20 +00:00
|
|
|
$exception = $this->executeHandlerAndGetHttpException( $handler, $request );
|
2020-03-06 15:53:01 +00:00
|
|
|
|
2021-12-08 14:25:20 +00:00
|
|
|
$this->assertSame( $expectedHttpException->getMessage(), $exception->getMessage() );
|
|
|
|
|
$this->assertSame( $expectedHttpException->getCode(), $exception->getCode(), 'HTTP status' );
|
2020-03-06 15:53:01 +00:00
|
|
|
|
2021-12-08 14:25:20 +00:00
|
|
|
$errorData = $exception->getErrorData();
|
|
|
|
|
if ( $expectedHttpException->getErrorData() ) {
|
|
|
|
|
foreach ( $expectedHttpException->getErrorData() as $key => $value ) {
|
|
|
|
|
$this->assertSame( $value, $errorData[$key], 'Error data key $key' );
|
|
|
|
|
}
|
2020-03-06 15:53:01 +00:00
|
|
|
}
|
|
|
|
|
|
2021-12-08 14:25:20 +00:00
|
|
|
if ( $expectedHttpException instanceof LocalizedHttpException ) {
|
|
|
|
|
/** @var LocalizedHttpException $exception */
|
|
|
|
|
$this->assertEquals(
|
|
|
|
|
$expectedHttpException->getMessageValue(),
|
|
|
|
|
$exception->getMessageValue()
|
|
|
|
|
);
|
|
|
|
|
}
|
2020-03-06 15:53:01 +00:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
}
|