198 lines
4.9 KiB
PHP
198 lines
4.9 KiB
PHP
<?php
|
|
|
|
namespace MediaWiki\Rest;
|
|
|
|
use ExtensionRegistry;
|
|
use IContextSource;
|
|
use MediaWiki;
|
|
use MediaWiki\Config\ServiceOptions;
|
|
use MediaWiki\MediaWikiServices;
|
|
use MediaWiki\Rest\BasicAccess\CompoundAuthorizer;
|
|
use MediaWiki\Rest\BasicAccess\MWBasicAuthorizer;
|
|
use MediaWiki\Rest\Validator\Validator;
|
|
use RequestContext;
|
|
use Title;
|
|
use WebResponse;
|
|
use Wikimedia\Message\ITextFormatter;
|
|
|
|
class EntryPoint {
|
|
/** @var RequestInterface */
|
|
private $request;
|
|
/** @var WebResponse */
|
|
private $webResponse;
|
|
/** @var Router */
|
|
private $router;
|
|
/** @var RequestContext */
|
|
private $context;
|
|
/** @var CorsUtils */
|
|
private $cors;
|
|
|
|
/**
|
|
* @param IContextSource $context
|
|
* @param RequestInterface $request
|
|
* @param ResponseFactory $responseFactory
|
|
* @param CorsUtils $cors
|
|
* @return Router
|
|
*/
|
|
private static function createRouter(
|
|
IContextSource $context, RequestInterface $request, ResponseFactory $responseFactory, CorsUtils $cors
|
|
): Router {
|
|
global $IP;
|
|
|
|
$services = MediaWikiServices::getInstance();
|
|
$conf = $services->getMainConfig();
|
|
|
|
$authorizer = new CompoundAuthorizer();
|
|
$authorizer
|
|
->addAuthorizer( new MWBasicAuthorizer( $context->getUser(),
|
|
$services->getPermissionManager() ) )
|
|
->addAuthorizer( $cors );
|
|
|
|
$objectFactory = $services->getObjectFactory();
|
|
$restValidator = new Validator( $objectFactory,
|
|
$services->getPermissionManager(),
|
|
$request,
|
|
RequestContext::getMain()->getUser()
|
|
);
|
|
|
|
// Always include the "official" routes. Include additional routes if specified.
|
|
$routeFiles = array_merge(
|
|
[ 'includes/Rest/coreRoutes.json' ],
|
|
$conf->get( 'RestAPIAdditionalRouteFiles' )
|
|
);
|
|
array_walk( $routeFiles, function ( &$val, $key ) {
|
|
global $IP;
|
|
$val = "$IP/$val";
|
|
} );
|
|
|
|
return ( new Router(
|
|
$routeFiles,
|
|
ExtensionRegistry::getInstance()->getAttribute( 'RestRoutes' ),
|
|
$conf->get( 'CanonicalServer' ),
|
|
$conf->get( 'RestPath' ),
|
|
$services->getLocalServerObjectCache(),
|
|
$responseFactory,
|
|
$authorizer,
|
|
$objectFactory,
|
|
$restValidator,
|
|
$services->getHookContainer()
|
|
) )->setCors( $cors );
|
|
}
|
|
|
|
public static function main() {
|
|
// URL safety checks
|
|
global $wgRequest;
|
|
|
|
$context = RequestContext::getMain();
|
|
|
|
// Set $wgTitle and the title in RequestContext, as in api.php
|
|
global $wgTitle;
|
|
$wgTitle = Title::makeTitle( NS_SPECIAL, 'Badtitle/rest.php' );
|
|
$context->setTitle( $wgTitle );
|
|
|
|
$services = MediaWikiServices::getInstance();
|
|
$conf = $services->getMainConfig();
|
|
|
|
$request = new RequestFromGlobals( [
|
|
'cookiePrefix' => $conf->get( 'CookiePrefix' )
|
|
] );
|
|
|
|
$responseFactory = new ResponseFactory( self::getTextFormatters( $services ) );
|
|
|
|
$cors = new CorsUtils(
|
|
new ServiceOptions(
|
|
CorsUtils::CONSTRUCTOR_OPTIONS, $services->getMainConfig()
|
|
),
|
|
$responseFactory,
|
|
$context->getUser()
|
|
);
|
|
|
|
$router = self::createRouter( $context, $request, $responseFactory, $cors );
|
|
|
|
$entryPoint = new self(
|
|
$context,
|
|
$request,
|
|
$wgRequest->response(),
|
|
$router,
|
|
$cors
|
|
);
|
|
$entryPoint->execute();
|
|
}
|
|
|
|
/**
|
|
* Get a TextFormatter array from MediaWikiServices
|
|
*
|
|
* @param MediaWikiServices $services
|
|
* @return ITextFormatter[]
|
|
*/
|
|
public static function getTextFormatters( MediaWikiServices $services ) {
|
|
$langs = array_unique( [
|
|
$services->getMainConfig()->get( 'ContLang' )->getCode(),
|
|
'en'
|
|
] );
|
|
$textFormatters = [];
|
|
$factory = $services->getMessageFormatterFactory();
|
|
foreach ( $langs as $lang ) {
|
|
$textFormatters[] = $factory->getTextFormatter( $lang );
|
|
}
|
|
return $textFormatters;
|
|
}
|
|
|
|
public function __construct( RequestContext $context, RequestInterface $request,
|
|
WebResponse $webResponse, Router $router, CorsUtils $cors
|
|
) {
|
|
$this->context = $context;
|
|
$this->request = $request;
|
|
$this->webResponse = $webResponse;
|
|
$this->router = $router;
|
|
$this->cors = $cors;
|
|
}
|
|
|
|
public function execute() {
|
|
ob_start();
|
|
$response = $this->cors->modifyResponse(
|
|
$this->request,
|
|
$this->router->execute( $this->request )
|
|
);
|
|
|
|
$this->webResponse->header(
|
|
'HTTP/' . $response->getProtocolVersion() . ' ' .
|
|
$response->getStatusCode() . ' ' .
|
|
$response->getReasonPhrase() );
|
|
|
|
foreach ( $response->getRawHeaderLines() as $line ) {
|
|
$this->webResponse->header( $line );
|
|
}
|
|
|
|
foreach ( $response->getCookies() as $cookie ) {
|
|
$this->webResponse->setCookie(
|
|
$cookie['name'],
|
|
$cookie['value'],
|
|
$cookie['expiry'],
|
|
$cookie['options'] );
|
|
}
|
|
|
|
// Clear all errors that might have been displayed if display_errors=On
|
|
ob_end_clean();
|
|
|
|
$stream = $response->getBody();
|
|
$stream->rewind();
|
|
|
|
MediaWiki::preOutputCommit( $this->context );
|
|
|
|
if ( $stream instanceof CopyableStreamInterface ) {
|
|
$stream->copyToStream( fopen( 'php://output', 'w' ) );
|
|
} else {
|
|
while ( true ) {
|
|
$buffer = $stream->read( 65536 );
|
|
if ( $buffer === '' ) {
|
|
break;
|
|
}
|
|
echo $buffer;
|
|
}
|
|
}
|
|
|
|
$mw = new MediaWiki;
|
|
$mw->doPostOutputShutdown();
|
|
}
|
|
}
|