2017-08-16 05:50:44 +00:00
|
|
|
<?php
|
|
|
|
|
2021-04-07 22:22:05 +00:00
|
|
|
namespace MediaWiki\Extension\Math;
|
|
|
|
|
|
|
|
use Exception;
|
2017-08-16 05:50:44 +00:00
|
|
|
use MediaWiki\Logger\LoggerFactory;
|
2019-10-11 17:45:11 +00:00
|
|
|
use MediaWiki\MediaWikiServices;
|
2021-04-07 22:22:05 +00:00
|
|
|
use MWException;
|
|
|
|
use stdClass;
|
2017-08-16 05:50:44 +00:00
|
|
|
|
|
|
|
/**
|
2020-03-17 07:14:13 +00:00
|
|
|
* @author Moritz Schubotz
|
2017-08-16 05:50:44 +00:00
|
|
|
*/
|
|
|
|
class MathMathMLCli extends MathMathML {
|
|
|
|
|
|
|
|
/**
|
2021-08-04 19:49:10 +00:00
|
|
|
* @param MathRenderer[] $renderers
|
2017-08-16 05:50:44 +00:00
|
|
|
* @return bool
|
|
|
|
* @throws MWException
|
|
|
|
*/
|
2021-08-04 19:49:10 +00:00
|
|
|
public static function batchEvaluate( array $renderers ) {
|
2017-08-16 05:50:44 +00:00
|
|
|
$req = [];
|
2021-08-04 19:49:10 +00:00
|
|
|
foreach ( $renderers as $key => $renderer ) {
|
|
|
|
'@phan-var MathMathMLCli $renderer';
|
2017-08-16 05:50:44 +00:00
|
|
|
// checking if the rendering is in the database is no security issue since only the md5
|
|
|
|
// hash of the user input string will be sent to the database
|
|
|
|
if ( !$renderer->isInDatabase() ) {
|
|
|
|
$req[] = $renderer->getMathoidCliQuery();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if ( count( $req ) === 0 ) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
$exitCode = 1;
|
|
|
|
$res = self::evaluateWithCli( $req, $exitCode );
|
2021-08-04 19:49:10 +00:00
|
|
|
foreach ( $renderers as $key => $renderer ) {
|
|
|
|
'@phan-var MathMathMLCli $renderer';
|
2017-08-16 05:50:44 +00:00
|
|
|
if ( !$renderer->isInDatabase() ) {
|
|
|
|
$renderer->initializeFromCliResponse( $res );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2020-12-17 20:53:28 +00:00
|
|
|
* @param stdClass $res
|
2017-08-16 05:50:44 +00:00
|
|
|
* @return bool
|
|
|
|
*/
|
|
|
|
private function initializeFromCliResponse( $res ) {
|
|
|
|
global $wgMathoidCli;
|
|
|
|
if ( !property_exists( $res, $this->getMd5() ) ) {
|
|
|
|
$this->lastError =
|
|
|
|
$this->getError( 'math_mathoid_error', 'cli',
|
2019-05-31 18:31:51 +00:00
|
|
|
var_export( get_object_vars( $res ), true ) );
|
2017-08-16 05:50:44 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if ( $this->isEmpty() ) {
|
2021-08-05 01:08:52 +00:00
|
|
|
$this->lastError = $this->getError( 'math_empty_tex' );
|
2017-08-16 05:50:44 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
$response = $res->{$this->getMd5()};
|
|
|
|
if ( !$response->success ) {
|
|
|
|
$this->lastError = $this->renderError( $response );
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
$this->texSecure = true;
|
|
|
|
$this->tex = $response->sanetex;
|
|
|
|
// The host name is only relevant for the debugging. So using file:// to indicate that the
|
|
|
|
// cli interface seems to be OK.
|
|
|
|
$this->processJsonResult( $response, 'file://' . $wgMathoidCli[0] );
|
|
|
|
$this->mathStyle = $response->mathoidStyle;
|
2022-11-08 13:58:00 +00:00
|
|
|
if ( property_exists( $response, 'png' ) ) {
|
2020-04-05 13:18:28 +00:00
|
|
|
$this->png = implode( array_map( "chr", $response->png->data ) );
|
|
|
|
} else {
|
|
|
|
LoggerFactory::getInstance( 'Math' )->error( 'Mathoid did not return a PNG image.' .
|
|
|
|
' Check your librsvg installation https://github.com/wikimedia/mathoid/.' );
|
|
|
|
}
|
2017-08-16 05:50:44 +00:00
|
|
|
$this->changed = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
public function renderError( $response ) {
|
|
|
|
$msg = $response->error;
|
|
|
|
try {
|
|
|
|
switch ( $response->detail->status ) {
|
|
|
|
case "F":
|
|
|
|
$msg .= "\n Found {$response->detail->details}" .
|
|
|
|
$this->appendLocationInfo( $response );
|
|
|
|
break;
|
|
|
|
case 'S':
|
|
|
|
case "C":
|
|
|
|
$msg .= $this->appendLocationInfo( $response );
|
|
|
|
break;
|
|
|
|
case '-':
|
|
|
|
// we do not know any cases that triggers this error
|
|
|
|
}
|
2022-06-23 00:22:59 +00:00
|
|
|
} catch ( Exception $e ) {
|
2017-08-16 05:50:44 +00:00
|
|
|
// use default error message
|
|
|
|
}
|
|
|
|
|
|
|
|
return $this->getError( 'math_mathoid_error', 'cli', $msg );
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return array
|
|
|
|
*/
|
|
|
|
public function getMathoidCliQuery() {
|
|
|
|
return [
|
|
|
|
'query' => [
|
|
|
|
'q' => $this->getTex(),
|
|
|
|
'type' => $this->getInputType(),
|
|
|
|
'hash' => $this->getMd5(),
|
|
|
|
],
|
|
|
|
];
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2018-04-07 19:03:56 +00:00
|
|
|
* @param mixed $req request
|
2018-05-26 04:26:21 +00:00
|
|
|
* @param int|null &$exitCode exit code
|
2017-08-16 05:50:44 +00:00
|
|
|
* @return mixed
|
|
|
|
* @throws MWException
|
|
|
|
*/
|
2022-06-21 09:23:05 +00:00
|
|
|
private static function evaluateWithCli( $req, &$exitCode = null ) {
|
2017-08-16 05:50:44 +00:00
|
|
|
global $wgMathoidCli;
|
|
|
|
$json_req = json_encode( $req );
|
|
|
|
$cmd = MediaWikiServices::getInstance()->getShellCommandFactory()->create();
|
|
|
|
$cmd->params( $wgMathoidCli );
|
|
|
|
$cmd->input( $json_req );
|
|
|
|
$result = $cmd->execute();
|
|
|
|
if ( $result->getExitCode() != 0 ) {
|
2018-11-20 10:34:41 +00:00
|
|
|
$errorMsg = $result->getStderr();
|
2017-08-16 05:50:44 +00:00
|
|
|
LoggerFactory::getInstance( 'Math' )->error( 'Can not process {req} with config
|
|
|
|
{conf} returns {res}', [
|
|
|
|
'req' => $req,
|
|
|
|
'conf' => var_export( $wgMathoidCli, true ),
|
|
|
|
'res' => var_export( $result, true ),
|
|
|
|
] );
|
2018-11-20 10:34:41 +00:00
|
|
|
throw new MWException( "Failed to execute Mathoid cli '$wgMathoidCli[0]', reason: $errorMsg" );
|
2017-08-16 05:50:44 +00:00
|
|
|
}
|
|
|
|
$res = json_decode( $result->getStdout() );
|
|
|
|
if ( !$res ) {
|
|
|
|
throw new MWException( "Mathoid cli response '$res' is no valid JSON file." );
|
|
|
|
}
|
|
|
|
|
|
|
|
return $res;
|
|
|
|
}
|
|
|
|
|
2022-06-21 09:23:05 +00:00
|
|
|
public function render() {
|
2017-08-16 05:50:44 +00:00
|
|
|
if ( $this->getLastError() ) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
protected function doCheck() {
|
|
|
|
// avoid that restbase is called if check is set to always
|
|
|
|
return $this->texSecure;
|
|
|
|
}
|
|
|
|
|
|
|
|
protected function initializeFromDatabaseRow( $rpage ) {
|
|
|
|
if ( !empty( $rpage->math_svg ) ) {
|
|
|
|
$this->png = $rpage->math_png;
|
|
|
|
}
|
|
|
|
parent::initializeFromDatabaseRow( $rpage ); // TODO: Change the autogenerated stub
|
|
|
|
}
|
|
|
|
|
|
|
|
protected function dbOutArray() {
|
|
|
|
$out = parent::dbOutArray();
|
|
|
|
$out['math_png'] = $this->png;
|
|
|
|
|
|
|
|
return $out;
|
|
|
|
}
|
|
|
|
|
|
|
|
protected function dbInArray() {
|
|
|
|
$out = parent::dbInArray();
|
|
|
|
$out[] = 'math_png';
|
|
|
|
|
|
|
|
return $out;
|
|
|
|
}
|
|
|
|
|
|
|
|
public function getPng() {
|
|
|
|
if ( !$this->png ) {
|
|
|
|
$this->initializeFromCliResponse( self::evaluateWithCli( [
|
|
|
|
$this->getMathoidCliQuery(),
|
|
|
|
] ) );
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
return parent::getPng();
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2020-01-14 07:43:50 +00:00
|
|
|
* @param stdClass $response object from cli
|
2017-08-16 05:50:44 +00:00
|
|
|
* @return string containing the location information
|
|
|
|
*/
|
|
|
|
private function appendLocationInfo( $response ) {
|
|
|
|
return "in {$response->detail->line}:{$response->detail->column}";
|
|
|
|
}
|
|
|
|
}
|