getMathoidChecker( $input ); $realKey = $checker->getCacheKey(); $this->assertStringEndsWith( $expected, $realKey ); } /** * @covers \MediaWiki\Extension\Math\InputCheck\MathoidChecker::getCheckResponse */ public function testResponseFromCache() { $fakeWAN = new WANObjectCache( [ 'cache' => new HashBagOStuff() ] ); $fakeWAN->set( self::SAMPLE_KEY, [ 999, 'expected' ], WANObjectCache::TTL_INDEFINITE, [ 'version' => MathoidChecker::VERSION ] ); // double check that the fake works $this->assertSame( [ 999, 'expected' ], $fakeWAN->get( self::SAMPLE_KEY ) ); $this->setService( 'MainWANObjectCache', $fakeWAN ); $checker = $this->getMathoidChecker(); $this->assertSame( [ 999, 'expected' ], $checker->getCheckResponse() ); } /** * @covers \MediaWiki\Extension\Math\InputCheck\MathoidChecker::getCheckResponse */ public function testResponseFromResponse() { $fakeWAN = WANObjectCache::newEmpty(); $fakeWAN->set( self::SAMPLE_KEY, 'expected' ); // double check that the fake does not works $this->assertSame( false, $fakeWAN->get( self::SAMPLE_KEY ) ); $this->setService( 'MainWANObjectCache', $fakeWAN ); $this->setFakeRequest( 200, 'expected' ); $checker = $this->getMathoidChecker(); $this->assertSame( [ 200, 'expected' ], $checker->getCheckResponse() ); } /** * @covers \MediaWiki\Extension\Math\InputCheck\MathoidChecker::getCheckResponse */ public function testFailedResponse() { $fakeWAN = WANObjectCache::newEmpty(); $fakeWAN->set( self::SAMPLE_KEY, 'expected' ); // double check that the fake does not works $this->assertSame( false, $fakeWAN->get( self::SAMPLE_KEY ) ); $this->setService( 'MainWANObjectCache', $fakeWAN ); $this->setFakeRequest( 401, false ); $checker = $this->getMathoidChecker(); $this->expectException( 'MWException' ); $checker->getCheckResponse(); } /** * @covers \MediaWiki\Extension\Math\InputCheck\MathoidChecker::isValid * @dataProvider provideMathoidSamples * @param string $input LaTeX input to check * @param string $mockRequestBody * @param int $mockResponseStatus * @param array $expeted */ public function testIsValid( $input, $mockRequestBody, $mockResponseStatus, $expeted ) { $request = $this->makeFakeHttpRequest( $mockRequestBody, $mockResponseStatus ); $this->installMockHttp( $request ); $checker = $this->getMathoidChecker( $input ); $this->assertSame( $expeted['valid'], $checker->isValid() ); } /** * @covers \MediaWiki\Extension\Math\InputCheck\MathoidChecker::getValidTex * @dataProvider provideMathoidSamples * @param string $input LaTeX input to check * @param string $mockRequestBody * @param int $mockResponseStatus * @param array $expeted */ public function testGetChecked( $input, $mockRequestBody, $mockResponseStatus, $expeted ) { $request = $this->makeFakeHttpRequest( $mockRequestBody, $mockResponseStatus ); $this->installMockHttp( $request ); $checker = $this->getMathoidChecker( $input ); $this->assertSame( $expeted['checked'], $checker->getValidTex() ); } /** * @covers \MediaWiki\Extension\Math\InputCheck\MathoidChecker::getError * @dataProvider provideMathoidSamples * @param string $input LaTeX input to check * @param string $mockRequestBody * @param int $mockResponseStatus * @param array $expeted */ public function testGetError( $input, $mockRequestBody, $mockResponseStatus, $expeted ) { $request = $this->makeFakeHttpRequest( $mockRequestBody, $mockResponseStatus ); $this->installMockHttp( $request ); $checker = $this->getMathoidChecker( $input ); if ( array_key_exists( 'error', $expeted ) ) { $checkerError = $checker->getError(); $this->assertNotNull( $checkerError ); $renderedCheckerError = ( new MathMathML( 'a' ) ) ->getError( $checkerError->getKey(), ...$checkerError->getParams() ); $this->assertStringContainsString( $expeted['error'], $renderedCheckerError ); } else { $this->assertNull( $checker->getError() ); } } /** * @param string $tex * @return MathoidChecker */ private function getMathoidChecker( $tex = '\sin x' ): MathoidChecker { return Math::getCheckerFactory() ->newMathoidChecker( $tex, 'tex' ); } private function setFakeRequest( $returnStatus, $content ): void { $fakeHTTP = $this->createMock( HttpRequestFactory::class ); $fakeRequest = $this->createMock( \MWHttpRequest::class ); $fakeRequest->expects( $this->once() )->method( 'execute' )->willReturn( true ); $fakeRequest->expects( $this->once() )->method( 'getStatus' )->willReturn( $returnStatus ); if ( $content ) { $fakeRequest->expects( $this->once() )->method( 'getContent' )->willReturn( $content ); } else { $fakeRequest->expects( $this->never() )->method( 'getContent' ); } $fakeHTTP->expects( $this->once() )->method( 'create' )->willReturn( $fakeRequest ); $this->setService( 'HttpRequestFactory', $fakeHTTP ); } public static function provideMathoidSamples() { yield '\ sin x' => [ '\sin x', file_get_contents( __DIR__ . '/data/mathoid/sinx.json' ), 200, [ 'valid' => true, 'checked' => '\sin x' ], ]; yield 'invalid F' => [ '1+\invalid', file_get_contents( __DIR__ . '/data/mathoid/invalidF.json' ), 400, [ 'valid' => false, 'checked' => null, 'error' => 'unknown function' ], ]; yield 'unescaped' => [ '1.5%', file_get_contents( __DIR__ . '/data/mathoid/deprecated.json' ), 200, [ 'valid' => true, 'checked' => '1.5\%' ], ]; yield 'syntax error' => [ '\left( x', file_get_contents( __DIR__ . '/data/mathoid/syntaxE.json' ), 400, [ 'valid' => false, 'checked' => null, 'error' => 'Failed to parse' ], ]; } }