wiki.techinc.nl/includes/api/ApiChangeContentModel.php
daniel a8ee61d9d6 Implement rate limiting in Authority.
Rate limits will automatically be checked by definitelyCan(),
authorizeRead() and authorizeWrite(). The authorize methods also
increment the counter.

UserAuthority tracks which limits have already been incremented during
the current request, to avoid duplicate increments caused by code that
still calls pingLimiter directly.

DEPLOY: Risky! We SHOULD not hit rate limits more often, but we might.
Rate limit metrics should be monitored closely, see
<https://grafana.wikimedia.org/d/8oA6CWr4z/mw-rate-limiting-overview>.

Bug: T310476
Depends-On: Iebd62b0487af9172edaeae41c9b31aaf2f20fd06
Change-Id: Ic349f03b7040343815b60b0a2c84a5780326c797
2023-07-23 17:16:56 +00:00

157 lines
3.9 KiB
PHP

<?php
use MediaWiki\Content\IContentHandlerFactory;
use MediaWiki\Page\ContentModelChangeFactory;
use MediaWiki\Title\Title;
use Wikimedia\ParamValidator\ParamValidator;
/**
* Api module to change the content model of existing pages
*
* For new pages, use the edit api and specify the desired content model and format.
*
* @since 1.35
* @ingroup API
* @author DannyS712
*/
class ApiChangeContentModel extends ApiBase {
/** @var IContentHandlerFactory */
private $contentHandlerFactory;
/** @var ContentModelChangeFactory */
private $contentModelChangeFactory;
/**
* @param ApiMain $main
* @param string $action
* @param IContentHandlerFactory $contentHandlerFactory
* @param ContentModelChangeFactory $contentModelChangeFactory
*/
public function __construct(
ApiMain $main,
$action,
IContentHandlerFactory $contentHandlerFactory,
ContentModelChangeFactory $contentModelChangeFactory
) {
parent::__construct( $main, $action );
$this->contentHandlerFactory = $contentHandlerFactory;
$this->contentModelChangeFactory = $contentModelChangeFactory;
}
/**
* A lot of this code is based on SpecialChangeContentModel
*/
public function execute() {
$params = $this->extractRequestParams();
$wikiPage = $this->getTitleOrPageId( $params );
$title = $wikiPage->getTitle();
$this->getErrorFormatter()->setContextTitle( $title );
if ( !$title->exists() ) {
$this->dieWithError( 'apierror-changecontentmodel-missingtitle' );
}
$newModel = $params['model'];
$this->checkUserRightsAny( 'editcontentmodel' );
$changer = $this->contentModelChangeFactory->newContentModelChange(
$this->getAuthority(),
$wikiPage,
$newModel
);
// Status messages should be apierror-*
$changer->setMessagePrefix( 'apierror-' );
$permissionStatus = $changer->authorizeChange();
if ( !$permissionStatus->isGood() ) {
$this->dieStatus( $permissionStatus );
}
if ( $params['tags'] ) {
$tagStatus = $changer->setTags( $params['tags'] );
if ( !$tagStatus->isGood() ) {
$this->dieStatus( $tagStatus );
}
}
// Everything passed, make the conversion
$status = $changer->doContentModelChange(
$this->getContext(),
$params['summary'],
$params['bot']
);
if ( !$status->isGood() ) {
// Failed
$this->dieStatus( $status );
}
$logid = $status->getValue()['logid'];
$result = [
'result' => 'Success',
'title' => $title->getPrefixedText(),
'pageid' => $title->getArticleID(),
'contentmodel' => $title->getContentModel( Title::READ_LATEST ),
'logid' => $logid,
'revid' => $title->getLatestRevID( Title::READ_LATEST ),
];
$this->getResult()->addValue( null, $this->getModuleName(), $result );
}
public function getAllowedParams() {
$models = $this->contentHandlerFactory->getContentModels();
$modelOptions = [];
foreach ( $models as $model ) {
$handler = $this->contentHandlerFactory->getContentHandler( $model );
if ( !$handler->supportsDirectEditing() ) {
continue;
}
$modelOptions[] = $model;
}
return [
'title' => [
ParamValidator::PARAM_TYPE => 'string',
],
'pageid' => [
ParamValidator::PARAM_TYPE => 'integer',
],
'summary' => [
ParamValidator::PARAM_TYPE => 'string',
],
'tags' => [
ParamValidator::PARAM_TYPE => 'tags',
ParamValidator::PARAM_ISMULTI => true,
],
'model' => [
ParamValidator::PARAM_TYPE => $modelOptions,
ParamValidator::PARAM_REQUIRED => true,
],
'bot' => false,
];
}
public function mustBePosted() {
return true;
}
public function isWriteMode() {
return true;
}
public function needsToken() {
return 'csrf';
}
public function getHelpUrls() {
return 'https://www.mediawiki.org/wiki/Special:MyLanguage/Help:ChangeContentModel';
}
protected function getExamplesMessages() {
return [
'action=changecontentmodel&title=Main Page&model=text&token=123ABC'
=> 'apihelp-changecontentmodel-example'
];
}
}