noticesWereEnabled = Deprecation::isEnabled(); $this->showSupportedNoticesWasEnabled = Deprecation::getShowNoticesCalledFromSupportedCode(); $this->oldHandler = set_error_handler(function (int $errno, string $errstr, string $errfile, int $errline) { if ($errno === E_USER_DEPRECATED) { if (str_contains($errstr, 'SilverStripe\\Dev\\Tests\\DeprecationTest')) { throw new DeprecationTestException($errstr); } else { // Suppress any E_USER_DEPRECATED unrelated to this unit-test return true; } } if (is_callable($this->oldHandler)) { return call_user_func($this->oldHandler, $errno, $errstr, $errfile, $errline); } // Fallback to default PHP error handler return false; }); // This is required to clear out the notice from instantiating DeprecationTestObject in TableBuilder::buildTables(). Deprecation::outputNotices(); } protected function tearDown(): void { if ($this->noticesWereEnabled) { Deprecation::enable(); } else { Deprecation::disable(); } Deprecation::setShowNoticesCalledFromSupportedCode($this->showSupportedNoticesWasEnabled); restore_error_handler(); $this->oldHandler = null; parent::tearDown(); } private function myDeprecatedMethod(): string { Deprecation::notice('1.2.3', 'My message'); return 'abc'; } private function myDeprecatedMethodNoReplacement(): string { Deprecation::noticeWithNoReplacment('1.2.3'); return 'abc'; } private function enableDeprecationNotices(bool $showNoReplacementNotices = false): void { Deprecation::enable($showNoReplacementNotices); Deprecation::setShowNoticesCalledFromSupportedCode(true); } public function testNotice() { $message = implode(' ', [ 'SilverStripe\Dev\Tests\DeprecationTest->myDeprecatedMethod is deprecated.', 'My message.', 'Called from SilverStripe\Dev\Tests\DeprecationTest->testNotice.' ]); $this->expectException(DeprecationTestException::class); $this->expectExceptionMessage($message); $this->enableDeprecationNotices(); $ret = $this->myDeprecatedMethod(); $this->assertSame('abc', $ret); // call outputNotices() directly because the regular shutdown function that emits // the notices within Deprecation won't be called until after this unit-test has finished Deprecation::outputNotices(); } public function testNoticeNoReplacement() { $message = implode(' ', [ 'SilverStripe\Dev\Tests\DeprecationTest->myDeprecatedMethodNoReplacement is deprecated.', 'Will be removed without equivalent functionality to replace it.', 'Called from SilverStripe\Dev\Tests\DeprecationTest->testNoticeNoReplacement.' ]); $this->expectDeprecation(); $this->expectDeprecationMessage($message); $this->enableDeprecationNotices(true); $ret = $this->myDeprecatedMethodNoReplacement(); $this->assertSame('abc', $ret); Deprecation::outputNotices(); } public function testNoticeNoReplacementNoSupressed() { $this->enableDeprecationNotices(); $ret = $this->myDeprecatedMethodNoReplacement(); $this->assertSame('abc', $ret); Deprecation::outputNotices(); } public function testCallUserFunc() { $message = implode(' ', [ 'SilverStripe\Dev\Tests\DeprecationTest->myDeprecatedMethod is deprecated.', 'My message.', 'Called from SilverStripe\Dev\Tests\DeprecationTest->testCallUserFunc.' ]); $this->expectException(DeprecationTestException::class); $this->expectExceptionMessage($message); $this->enableDeprecationNotices(); $ret = call_user_func([$this, 'myDeprecatedMethod']); $this->assertSame('abc', $ret); Deprecation::outputNotices(); } public function testCallUserFuncArray() { $message = implode(' ', [ 'SilverStripe\Dev\Tests\DeprecationTest->myDeprecatedMethod is deprecated.', 'My message.', 'Called from SilverStripe\Dev\Tests\DeprecationTest->testCallUserFuncArray.' ]); $this->expectException(DeprecationTestException::class); $this->expectExceptionMessage($message); $this->enableDeprecationNotices(); $ret = call_user_func_array([$this, 'myDeprecatedMethod'], []); $this->assertSame('abc', $ret); Deprecation::outputNotices(); } public function testwithSuppressedNoticeDefault() { $this->enableDeprecationNotices(); $ret = Deprecation::withSuppressedNotice(function () { return $this->myDeprecatedMethod(); }); $this->assertSame('abc', $ret); Deprecation::outputNotices(); } public function testwithSuppressedNoticeTrue() { $message = implode(' ', [ 'SilverStripe\Dev\Tests\DeprecationTest->myDeprecatedMethod is deprecated.', 'My message.', 'Called from SilverStripe\Dev\Tests\DeprecationTest->testwithSuppressedNoticeTrue.' ]); $this->expectException(DeprecationTestException::class); $this->expectExceptionMessage($message); $this->enableDeprecationNotices(true); $ret = Deprecation::withSuppressedNotice(function () { return $this->myDeprecatedMethod(); }); $this->assertSame('abc', $ret); Deprecation::outputNotices(); } public function testwithSuppressedNoticeTrueCallUserFunc() { $message = implode(' ', [ 'SilverStripe\Dev\Tests\DeprecationTest->myDeprecatedMethod is deprecated.', 'My message.', 'Called from SilverStripe\Dev\Tests\DeprecationTest->testwithSuppressedNoticeTrueCallUserFunc.' ]); $this->expectException(DeprecationTestException::class); $this->expectExceptionMessage($message); $this->enableDeprecationNotices(true); $ret = Deprecation::withSuppressedNotice(function () { return call_user_func([$this, 'myDeprecatedMethod']); }); $this->assertSame('abc', $ret); Deprecation::outputNotices(); } public function testNoticewithSuppressedNoticeTrue() { $message = implode(' ', [ 'SilverStripe\Dev\Tests\DeprecationTest->testNoticewithSuppressedNoticeTrue is deprecated.', 'My message.', 'Called from PHPUnit\Framework\TestCase->runTest.' ]); $this->expectException(DeprecationTestException::class); $this->expectExceptionMessage($message); $this->enableDeprecationNotices(true); Deprecation::withSuppressedNotice(function () { Deprecation::notice('123', 'My message.'); }); Deprecation::outputNotices(); } public function testClasswithSuppressedNotice() { $message = implode(' ', [ 'SilverStripe\Dev\Tests\DeprecationTest\DeprecationTestObject is deprecated.', 'Some class message.', 'Called from SilverStripe\Dev\Tests\DeprecationTest->testClasswithSuppressedNotice.' ]); $this->expectException(DeprecationTestException::class); $this->expectExceptionMessage($message); $this->enableDeprecationNotices(true); // using this syntax because my IDE was complaining about DeprecationTestObject not existing // when trying to use `new DeprecationTestObject();` $class = DeprecationTestObject::class; new $class(); Deprecation::outputNotices(); } public function testClassWithInjectorwithSuppressedNotice() { $message = implode(' ', [ 'SilverStripe\Dev\Tests\DeprecationTest\DeprecationTestObject is deprecated.', 'Some class message.', 'Called from SilverStripe\Dev\Tests\DeprecationTest->testClassWithInjectorwithSuppressedNotice.' ]); $this->expectException(DeprecationTestException::class); $this->expectExceptionMessage($message); $this->enableDeprecationNotices(true); Injector::inst()->get(DeprecationTestObject::class); Deprecation::outputNotices(); } #[DoesNotPerformAssertions] public function testDisabled() { if ($this->noticesWereEnabled) { $this->markTestSkipped('Notices are enabled for this project outside of this unit test'); } // test that no error error is raised because by default Deprecation is disabled $this->myDeprecatedMethod(); Deprecation::outputNotices(); } public function testshowNoticesCalledFromSupportedCode() { $this->expectNotToPerformAssertions(); $this->enableDeprecationNotices(true); // showNoticesCalledFromSupportedCode is set to true by default for these unit tests // as it is testing code within vendor/silverstripe // This test is to ensure that the method works as expected when we disable this // and we should expect no exceptions to be thrown // // Note specifically NOT testing the following because it's counted as being called // from phpunit itself, which is not considered supported code // Deprecation::withSuppressedNotice(function () { // Deprecation::notice('123', 'My message.'); // }); Deprecation::setShowNoticesCalledFromSupportedCode(false); // notice() $this->myDeprecatedMethod(); // noticeNoReplacement() $this->myDeprecatedMethodNoReplacement(); // callUserFunc() call_user_func([$this, 'myDeprecatedMethod']); // callUserFuncArray() call_user_func_array([$this, 'myDeprecatedMethod'], []); // withSuppressedNotice() Deprecation::withSuppressedNotice( fn() => $this->myDeprecatedMethod() ); // withSuppressedNoticeTrue() Deprecation::withSuppressedNotice(function () { $this->myDeprecatedMethod(); }); // withSuppressedNoticeTrueCallUserFunc() Deprecation::withSuppressedNotice(function () { call_user_func([$this, 'myDeprecatedMethod']); }); // classWithSuppressedNotice() $class = DeprecationTestObject::class; new $class(); // classWithInjectorwithSuppressedNotice() Injector::inst()->get(DeprecationTestObject::class); // Output notices - there should be none Deprecation::outputNotices(); } // The following tests would be better to put in the silverstripe/config module, however this is not // possible to do in a clean way as the config for the DeprecationTestObject will not load if it // is inside the silverstripe/config directory, as there is no _config.php file or _config folder. // Adding a _config.php file will break existing unit-tests within silverstripe/config // It is possible to put DeprecationTestObject in framework and the unit tests in config, however // that's probably messier then just having everything within framework public function testConfigGetFirst() { $message = implode(' ', [ 'Config SilverStripe\Dev\Tests\DeprecationTest\DeprecationTestObject.first_config is deprecated.', 'My first config message.' ]); $this->expectException(DeprecationTestException::class); $this->expectExceptionMessage($message); $this->enableDeprecationNotices(); Config::inst()->get(DeprecationTestObject::class, 'first_config'); Deprecation::outputNotices(); } public function testConfigGetSecond() { $message = implode(' ', [ 'Config SilverStripe\Dev\Tests\DeprecationTest\DeprecationTestObject.second_config is deprecated.', 'My second config message.' ]); $this->expectException(DeprecationTestException::class); $this->expectExceptionMessage($message); $this->enableDeprecationNotices(); Config::inst()->get(DeprecationTestObject::class, 'second_config'); Deprecation::outputNotices(); } public function testConfigGetThird() { $message = 'Config SilverStripe\Dev\Tests\DeprecationTest\DeprecationTestObject.third_config is deprecated.'; $this->expectException(DeprecationTestException::class); $this->expectExceptionMessage($message); $this->enableDeprecationNotices(); Config::inst()->get(DeprecationTestObject::class, 'third_config'); Deprecation::outputNotices(); } public function testConfigSet() { $message = implode(' ', [ 'Config SilverStripe\Dev\Tests\DeprecationTest\DeprecationTestObject.first_config is deprecated.', 'My first config message.' ]); $this->expectException(DeprecationTestException::class); $this->expectExceptionMessage($message); $this->enableDeprecationNotices(); Config::modify()->set(DeprecationTestObject::class, 'first_config', 'abc'); Deprecation::outputNotices(); } public function testConfigMerge() { $message = implode(' ', [ 'Config SilverStripe\Dev\Tests\DeprecationTest\DeprecationTestObject.array_config is deprecated.', 'My array config message.' ]); $this->expectException(DeprecationTestException::class); $this->expectExceptionMessage($message); $this->enableDeprecationNotices(); Config::modify()->merge(DeprecationTestObject::class, 'array_config', ['abc']); Deprecation::outputNotices(); } public static function provideConfigVsEnv() { return [ // env var not set - config setting is respected [ // false is returned when the env isn't set, so this simulates simply not having // set the variable in the first place 'envVal' => 'notset', 'configVal' => false, 'expected' => false, ], [ 'envVal' => 'notset', 'configVal' => true, 'expected' => true, ], // env var is set and truthy, config setting is ignored [ 'envVal' => true, 'configVal' => false, 'expected' => true, ], [ 'envVal' => true, 'configVal' => true, 'expected' => true, ], // env var is set and falsy, config setting is ignored [ 'envVal' => false, 'configVal' => false, 'expected' => false, ], [ 'envVal' => false, 'configVal' => true, 'expected' => false, ], ]; } #[DataProvider('provideConfigVsEnv')] public function testEnabledConfigVsEnv($envVal, bool $configVal, bool $expected) { $this->runConfigVsEnvTest('SS_DEPRECATION_ENABLED', $envVal, $configVal, $expected); } #[DataProvider('provideConfigVsEnv')] public function testShowHttpConfigVsEnv($envVal, bool $configVal, bool $expected) { $this->runConfigVsEnvTest('SS_DEPRECATION_SHOW_HTTP', $envVal, $configVal, $expected); } #[DataProvider('provideConfigVsEnv')] public function testShowCliConfigVsEnv($envVal, bool $configVal, bool $expected) { $this->runConfigVsEnvTest('SS_DEPRECATION_SHOW_CLI', $envVal, $configVal, $expected); } private function runConfigVsEnvTest(string $varName, $envVal, bool $configVal, bool $expected) { $oldVars = Environment::getVariables(); if ($envVal === 'notset') { if (Environment::hasEnv($varName)) { $this->markTestSkipped("$varName is set, so we can't test behaviour when it's not"); return; } } else { Environment::setEnv($varName, $envVal); } switch ($varName) { case 'SS_DEPRECATION_ENABLED': if ($configVal) { $this->enableDeprecationNotices(); } else { Deprecation::disable(); } $result = Deprecation::isEnabled(); break; case 'SS_DEPRECATION_SHOW_HTTP': $oldShouldShow = Deprecation::shouldShowForHttp(); Deprecation::setShouldShowForHttp($configVal); $result = Deprecation::shouldShowForHttp(); Deprecation::setShouldShowForHttp($oldShouldShow); break; case 'SS_DEPRECATION_SHOW_CLI': $oldShouldShow = Deprecation::shouldShowForCli(); Deprecation::setShouldShowForCli($configVal); $result = Deprecation::shouldShowForCli(); Deprecation::setShouldShowForCli($oldShouldShow); break; } Environment::setVariables($oldVars); $this->assertSame($expected, $result); } public static function provideVarAsBoolean() { return [ [ 'rawValue' => true, 'expected' => true, ], [ 'rawValue' => 'true', 'expected' => true, ], [ 'rawValue' => 1, 'expected' => true, ], [ 'rawValue' => '1', 'expected' => true, ], [ 'rawValue' => 'on', 'expected' => true, ], [ 'rawValue' => false, 'expected' => false, ], [ 'rawValue' => 'false', 'expected' => false, ], [ 'rawValue' => 0, 'expected' => false, ], [ 'rawValue' => '0', 'expected' => false, ], [ 'rawValue' => 'off', 'expected' => false, ], ]; } #[DataProvider('provideVarAsBoolean')] public function testVarAsBoolean($rawValue, bool $expected) { $reflectionVarAsBoolean = new ReflectionMethod(Deprecation::class, 'varAsBoolean'); $reflectionVarAsBoolean->setAccessible(true); $this->assertSame($expected, $reflectionVarAsBoolean->invoke(null, $rawValue)); } public static function provideIsEnabled() { return [ 'dev, explicitly enabled' => [ 'envMode' => 'dev', 'envEnabled' => true, 'staticEnabled' => true, 'expected' => true, ], 'dev, explicitly enabled override static' => [ 'envMode' => 'dev', 'envEnabled' => true, 'staticEnabled' => false, 'expected' => true, ], 'dev, explicitly disabled override static' => [ 'envMode' => 'dev', 'envEnabled' => false, 'staticEnabled' => true, 'expected' => false, ], 'dev, explicitly disabled' => [ 'envMode' => 'dev', 'envEnabled' => false, 'staticEnabled' => false, 'expected' => false, ], 'dev, statically disabled' => [ 'envMode' => 'dev', 'envEnabled' => null, 'staticEnabled' => true, 'expected' => true, ], 'dev, statically disabled' => [ 'envMode' => 'dev', 'envEnabled' => null, 'staticEnabled' => false, 'expected' => false, ], 'live, explicitly enabled' => [ 'envMode' => 'live', 'envEnabled' => true, 'staticEnabled' => true, 'expected' => false, ], 'live, explicitly disabled' => [ 'envMode' => 'live', 'envEnabled' => false, 'staticEnabled' => false, 'expected' => false, ], 'live, statically disabled' => [ 'envMode' => 'live', 'envEnabled' => null, 'staticEnabled' => true, 'expected' => false, ], 'live, statically disabled' => [ 'envMode' => 'live', 'envEnabled' => null, 'staticEnabled' => false, 'expected' => false, ], ]; } #[DataProvider('provideIsEnabled')] public function testIsEnabled(string $envMode, ?bool $envEnabled, bool $staticEnabled, bool $expected) { /** @var Kernel $kernel */ $kernel = Injector::inst()->get(Kernel::class); $origMode = $kernel->getEnvironment(); $origEnvEnabled = Environment::getEnv('SS_DEPRECATION_ENABLED'); $reflectionEnabled = new ReflectionProperty(Deprecation::class, 'currentlyEnabled'); $reflectionEnabled->setAccessible(true); $origStaticEnabled = $reflectionEnabled->getValue(); try { $kernel->setEnvironment($envMode); Environment::setEnv('SS_DEPRECATION_ENABLED', $envEnabled); $this->setEnabledViaStatic($staticEnabled); $this->assertSame($expected, Deprecation::isEnabled()); } finally { $kernel->setEnvironment($origMode); Environment::setEnv('SS_DEPRECATION_ENABLED', $origEnvEnabled); $this->setEnabledViaStatic($origStaticEnabled); } } private function setEnabledViaStatic(bool $enabled): void { if ($enabled) { $this->enableDeprecationNotices(); } else { Deprecation::disable(); } } }