ApiQuerySearchTest.php 3.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133
  1. <?php
  2. /**
  3. * @group medium
  4. * @covers ApiQuerySearch
  5. */
  6. class ApiQuerySearchTest extends ApiTestCase {
  7. public function provideSearchResults() {
  8. return [
  9. 'empty search result' => [ [], [] ],
  10. 'has search results' => [
  11. [ 'Zomg' ],
  12. [ $this->mockResultClosure( 'Zomg' ) ],
  13. ],
  14. 'filters broken search results' => [
  15. [ 'A', 'B' ],
  16. [
  17. $this->mockResultClosure( 'a' ),
  18. $this->mockResultClosure( 'Zomg', [ 'setBrokenTitle' => true ] ),
  19. $this->mockResultClosure( 'b' ),
  20. ],
  21. ],
  22. 'filters results with missing revision' => [
  23. [ 'B', 'A' ],
  24. [
  25. $this->mockResultClosure( 'Zomg', [ 'setMissingRevision' => true ] ),
  26. $this->mockResultClosure( 'b' ),
  27. $this->mockResultClosure( 'a' ),
  28. ],
  29. ],
  30. ];
  31. }
  32. /**
  33. * @dataProvider provideSearchResults
  34. */
  35. public function testSearchResults( $expect, $hits, array $params = [] ) {
  36. MockSearchEngine::addMockResults( 'my query', $hits );
  37. list( $response, $request ) = $this->doApiRequest( $params + [
  38. 'action' => 'query',
  39. 'list' => 'search',
  40. 'srsearch' => 'my query',
  41. ] );
  42. $titles = [];
  43. foreach ( $response['query']['search'] as $result ) {
  44. $titles[] = $result['title'];
  45. }
  46. $this->assertEquals( $expect, $titles );
  47. }
  48. public function provideInterwikiResults() {
  49. return [
  50. 'empty' => [ [], [] ],
  51. 'one wiki response' => [
  52. [ 'utwiki' => [ 'Qwerty' ] ],
  53. [
  54. SearchResultSet::SECONDARY_RESULTS => [
  55. 'utwiki' => new MockSearchResultSet( [
  56. $this->mockResultClosure(
  57. 'Qwerty',
  58. [ 'setInterwikiPrefix' => 'utwiki' ]
  59. ),
  60. ] ),
  61. ],
  62. ]
  63. ],
  64. ];
  65. }
  66. /**
  67. * @dataProvider provideInterwikiResults
  68. */
  69. public function testInterwikiResults( $expect, $hits, array $params = [] ) {
  70. MockSearchEngine::setMockInterwikiResults( $hits );
  71. list( $response, $request ) = $this->doApiRequest( $params + [
  72. 'action' => 'query',
  73. 'list' => 'search',
  74. 'srsearch' => 'my query',
  75. 'srinterwiki' => true,
  76. ] );
  77. if ( !$expect ) {
  78. $this->assertArrayNotHasKey( 'interwikisearch', $response['query'] );
  79. return;
  80. }
  81. $results = [];
  82. $this->assertArrayHasKey( 'interwikisearchinfo', $response['query'] );
  83. foreach ( $response['query']['interwikisearch'] as $wiki => $wikiResults ) {
  84. $results[$wiki] = [];
  85. foreach ( $wikiResults as $wikiResult ) {
  86. $results[$wiki][] = $wikiResult['title'];
  87. }
  88. }
  89. $this->assertEquals( $expect, $results );
  90. }
  91. public function setUp() {
  92. parent::setUp();
  93. MockSearchEngine::clearMockResults();
  94. $this->registerMockSearchEngine();
  95. }
  96. private function registerMockSearchEngine() {
  97. $this->setMwGlobals( [
  98. 'wgSearchType' => MockSearchEngine::class,
  99. ] );
  100. }
  101. /**
  102. * Returns a closure that evaluates to a MockSearchResult, to be resolved by
  103. * MockSearchEngine::addMockResults() or MockresultSet::extractResults().
  104. *
  105. * This is needed because MockSearchResults cannot be instantiated in a data provider,
  106. * since they load revisions. This would hit the "real" database instead of the mock
  107. * database, which in turn may cause cache pollution and other inconsistencies, see T202641.
  108. *
  109. * @param string $title
  110. * @param array $setters
  111. * @return callable function(): MockSearchResult
  112. */
  113. private function mockResultClosure( $title, $setters = [] ) {
  114. return function () use ( $title, $setters ){
  115. $result = MockSearchResult::newFromTitle( Title::newFromText( $title ) );
  116. foreach ( $setters as $method => $param ) {
  117. $result->$method( $param );
  118. }
  119. return $result;
  120. };
  121. }
  122. }