. * * @file * @ingroup extensions */ use Popups\PopupsContext; use Popups\PopupsHooks; /** * Integration tests for popups hooks * * @group Popups * @coversDefaultClass \Popups\PopupsHooks */ class PopupsHooksTest extends MediaWikiTestCase { protected function tearDown() : void { parent::tearDown(); } /** * @covers ::onGetPreferences */ public function testOnGetPreferencesPreviewsDisabled() { $contextMock = $this->getMockBuilder( PopupsContext::class ) ->disableOriginalConstructor() ->setMethods( [ 'showPreviewsOptInOnPreferencesPage' ] ) ->getMock(); $contextMock->expects( $this->once() ) ->method( 'showPreviewsOptInOnPreferencesPage' ) ->will( $this->returnValue( false ) ); $this->setService( 'Popups.Context', $contextMock ); $prefs = [ 'someNotEmptyValue' => 'notEmpty' ]; PopupsHooks::onGetPreferences( $this->getTestUser()->getUser(), $prefs ); $this->assertCount( 1, $prefs, 'No preferences are retrieved.' ); $this->assertSame( 'notEmpty', $prefs[ 'someNotEmptyValue'], 'No preferences are changed.' ); } /** * @covers ::onGetPreferences */ public function testOnGetPreferencesNavPopupGadgetIsOn() { $userMock = $this->getTestUser()->getUser(); $contextMock = $this->getMockBuilder( PopupsContext::class ) ->disableOriginalConstructor() ->setMethods( [ 'showPreviewsOptInOnPreferencesPage', 'conflictsWithNavPopupsGadget' ] ) ->getMock(); $contextMock->expects( $this->once() ) ->method( 'showPreviewsOptInOnPreferencesPage' ) ->will( $this->returnValue( true ) ); $contextMock->expects( $this->once() ) ->method( 'conflictsWithNavPopupsGadget' ) ->with( $userMock ) ->will( $this->returnValue( true ) ); $this->setService( 'Popups.Context', $contextMock ); $prefs = []; PopupsHooks::onGetPreferences( $this->getTestUser()->getUser(), $prefs ); $this->assertArrayHasKey( PopupsContext::PREVIEWS_OPTIN_PREFERENCE_NAME, $prefs, 'The opt-in preference is retrieved.' ); $this->assertArrayHasKey( 'disabled', $prefs[ PopupsContext::PREVIEWS_OPTIN_PREFERENCE_NAME ], 'The opt-in preference has a status.' ); $this->assertTrue( $prefs[ PopupsContext::PREVIEWS_OPTIN_PREFERENCE_NAME]['disabled'], 'The opt-in preference\'s status is disabled.' ); $this->assertNotEmpty( $prefs[ PopupsContext::PREVIEWS_OPTIN_PREFERENCE_NAME]['help-message'], 'The opt-in preference has a help message.' ); } /** * @covers ::onGetPreferences */ public function testOnGetPreferencesPreviewsEnabled() { $contextMock = $this->getMockBuilder( PopupsContext::class ) ->disableOriginalConstructor() ->setMethods( [ 'showPreviewsOptInOnPreferencesPage', 'conflictsWithNavPopupsGadget' ] ) ->getMock(); $contextMock->expects( $this->once() ) ->method( 'showPreviewsOptInOnPreferencesPage' ) ->will( $this->returnValue( true ) ); $contextMock->expects( $this->once() ) ->method( 'conflictsWithNavPopupsGadget' ) ->will( $this->returnValue( false ) ); $this->setService( 'Popups.Context', $contextMock ); $prefs = [ 'skin' => 'skin stuff', 'someNotEmptyValue' => 'notEmpty', 'other' => 'notEmpty' ]; PopupsHooks::onGetPreferences( $this->getTestUser()->getUser(), $prefs ); $this->assertGreaterThan( 3, count( $prefs ), 'A preference is retrieved.' ); $this->assertSame( 'notEmpty', $prefs[ 'someNotEmptyValue'], 'Unretrieved preferences are unchanged.' ); $this->assertArrayHasKey( PopupsContext::PREVIEWS_OPTIN_PREFERENCE_NAME, $prefs, 'The opt-in preference is retrieved.' ); $this->assertSame( 1, array_search( \Popups\PopupsContext::PREVIEWS_OPTIN_PREFERENCE_NAME, array_keys( $prefs ) ), 'The opt-in preference is injected after Skin select.' ); } /** * @covers ::onGetPreferences */ public function testOnGetPreferencesPreviewsEnabledWhenSkinIsNotAvailable() { $contextMock = $this->getMockBuilder( PopupsContext::class ) ->disableOriginalConstructor() ->setMethods( [ 'showPreviewsOptInOnPreferencesPage', 'conflictsWithNavPopupsGadget' ] ) ->getMock(); $contextMock->expects( $this->once() ) ->method( 'showPreviewsOptInOnPreferencesPage' ) ->will( $this->returnValue( true ) ); $contextMock->expects( $this->once() ) ->method( 'conflictsWithNavPopupsGadget' ) ->will( $this->returnValue( false ) ); $this->setService( 'Popups.Context', $contextMock ); $prefs = [ 'someNotEmptyValue' => 'notEmpty', 'other' => 'notEmpty' ]; PopupsHooks::onGetPreferences( $this->getTestUser()->getUser(), $prefs ); $this->assertGreaterThan( 2, count( $prefs ), 'A preference is retrieved.' ); $this->assertArrayHasKey( PopupsContext::PREVIEWS_OPTIN_PREFERENCE_NAME, $prefs, 'The opt-in preference is retrieved.' ); $this->assertSame( 2, array_search( PopupsContext::PREVIEWS_OPTIN_PREFERENCE_NAME, array_keys( $prefs ) ), 'The opt-in preference is appended.' ); } /** * @covers ::onResourceLoaderGetConfigVars */ public function testOnResourceLoaderGetConfigVars() { $vars = [ 'something' => 'notEmpty' ]; $config = [ 'wgPopupsEventLogging' => false, 'wgPopupsRestGatewayEndpoint' => '/api', 'wgPopupsVirtualPageViews' => true, 'wgPopupsGateway' => 'mwApiPlain', 'wgPopupsStatsvSamplingRate' => 0 ]; $this->setMwGlobals( $config ); PopupsHooks::onResourceLoaderGetConfigVars( $vars, '' ); $this->assertCount( 6, $vars, 'A configuration is retrieved.' ); foreach ( $config as $key => $value ) { $this->assertSame( $value, $vars[ $key ], "It forwards the \"{$key}\" config variable to the client." ); } } /** * @covers ::onBeforePageDisplay */ public function testOnBeforePageDisplayWhenDependenciesAreNotMet() { $skinMock = $this->createMock( Skin::class ); $outPageMock = $this->getMockBuilder( OutputPage::class ) ->setMethods( [ 'addModules' ] ) ->disableOriginalConstructor() ->getMock(); $outPageMock->expects( $this->never() ) ->method( 'addModules' ); $loggerMock = $this->createMock( \Psr\Log\LoggerInterface::class ); $loggerMock->expects( $this->once() ) ->method( 'error' ); $contextMock = $this->getMockBuilder( PopupsContext::class ) ->disableOriginalConstructor() ->setMethods( [ 'areDependenciesMet', 'getLogger', 'isTitleBlacklisted' ] ) ->getMock(); $contextMock->expects( $this->once() ) ->method( 'areDependenciesMet' ) ->will( $this->returnValue( false ) ); $contextMock->expects( $this->once() ) ->method( 'isTitleBlacklisted' ) ->will( $this->returnValue( false ) ); $contextMock->expects( $this->once() ) ->method( 'getLogger' ) ->will( $this->returnValue( $loggerMock ) ); $this->setService( 'Popups.Context', $contextMock ); PopupsHooks::onBeforePageDisplay( $outPageMock, $skinMock ); } public function providerOnBeforePageDisplay() { return [ [ false, false, false ], [ true, true, false ], // Code not sent if title blacklisted [ true, false, true ], // Code not sent if title blacklisted [ false, false, true ] ]; } /** * @covers ::onBeforePageDisplay * @dataProvider providerOnBeforePageDisplay */ public function testOnBeforePageDisplay( $shouldSendModuleToUser, $isCodeLoaded, $isTitleBlacklisted ) { $skinMock = $this->createMock( Skin::class ); $outPageMock = $this->getMockBuilder( OutputPage::class ) ->setMethods( [ 'addModules' ] ) ->disableOriginalConstructor() ->getMock(); $outPageMock->expects( $isCodeLoaded ? $this->once() : $this->never() ) ->method( 'addModules' ) ->with( [ 'ext.popups' ] ); $contextMock = $this->getMockBuilder( PopupsContext::class ) ->setMethods( [ 'areDependenciesMet', 'shouldSendModuleToUser', 'isTitleBlacklisted' ] ) ->disableOriginalConstructor() ->getMock(); if ( !$isTitleBlacklisted ) { $contextMock->expects( $this->once() ) ->method( 'areDependenciesMet' ) ->will( $this->returnValue( true ) ); } $contextMock->expects( $this->any() ) ->method( 'shouldSendModuleToUser' ) ->will( $this->returnValue( $shouldSendModuleToUser ) ); $contextMock->expects( $this->once() ) ->method( 'isTitleBlacklisted' ) ->will( $this->returnValue( $isTitleBlacklisted ) ); $this->setService( 'Popups.Context', $contextMock ); PopupsHooks::onBeforePageDisplay( $outPageMock, $skinMock ); } /** * @covers ::onMakeGlobalVariablesScript */ public function testOnMakeGlobalVariablesScript() { $user = User::newFromId( 0 ); $outputPage = $this->getMockBuilder( OutputPage::class ) ->disableOriginalConstructor() ->getMock(); $outputPage->expects( $this->any() ) ->method( 'getUser' ) ->willReturn( $user ); $contextMock = $this->getMockBuilder( PopupsContext::class ) ->setMethods( [ 'conflictsWithNavPopupsGadget', 'isReferencePreviewsEnabled' ] ) ->disableOriginalConstructor() ->getMock(); $contextMock->expects( $this->any() ) ->method( 'conflictsWithNavPopupsGadget' ) ->with( $user ) ->willReturn( false ); $contextMock->method( 'isReferencePreviewsEnabled' ) ->with( $user ) ->willReturn( true ); $this->setService( 'Popups.Context', $contextMock ); $vars = []; PopupsHooks::onMakeGlobalVariablesScript( $vars, $outputPage ); $this->assertCount( 2, $vars, 'Number of added variables.' ); $this->assertFalse( $vars[ 'wgPopupsConflictsWithNavPopupGadget' ], 'The PopupsConflictsWithNavPopupGadget global is present and false.' ); } /** * @covers ::onUserGetDefaultOptions */ public function testOnUserGetDefaultOptions() { $userOptions = [ 'test' => 'not_empty' ]; $this->setMwGlobals( [ 'wgPopupsOptInDefaultState' => '1' ] ); PopupsHooks::onUserGetDefaultOptions( $userOptions ); $this->assertCount( 3, $userOptions ); $this->assertSame( '1', $userOptions[ \Popups\PopupsContext::PREVIEWS_OPTIN_PREFERENCE_NAME ] ); } /** * @covers ::onUserGetDefaultOptions */ public function testOnLocalUserCreatedForNewlyCreatedUser() { $expectedState = '1'; $userMock = $this->getMockBuilder( User::class ) ->disableOriginalConstructor() ->setMethods( [ 'setOption' ] ) ->getMock(); $userMock->expects( $this->exactly( 2 ) ) ->method( 'setOption' ) ->with( $this->stringStartsWith( 'popups' ), $expectedState ); $this->setMwGlobals( [ 'wgPopupsOptInStateForNewAccounts' => $expectedState ] ); PopupsHooks::onLocalUserCreated( $userMock, false ); } }