From 6a3ba025c415c84c2abd42230ff7ae94e594bf8d Mon Sep 17 00:00:00 2001
From: Lee Rowlands <lee.rowlands@previousnext.com.au>
Date: Thu, 18 Jan 2018 07:34:26 +1000
Subject: [PATCH] Issue #2936802 by alexpott, Wim Leers: expectDeprecation()
 doesn't work in isolated tests

---
 .../Drupal/Tests/ExpectDeprecationTest.php    | 12 +++
 .../Listeners/DeprecationListenerTrait.php    | 39 ++++++++++
 .../Drupal/Tests/Listeners/DrupalListener.php |  7 ++
 .../Tests/Listeners/Legacy/DrupalListener.php |  7 ++
 .../Tests/Traits/ExpectDeprecationTrait.php   | 78 +++++++++++++------
 5 files changed, 118 insertions(+), 25 deletions(-)

diff --git a/core/tests/Drupal/Tests/ExpectDeprecationTest.php b/core/tests/Drupal/Tests/ExpectDeprecationTest.php
index 8ebf3b5a210d..aefb6a0ad927 100644
--- a/core/tests/Drupal/Tests/ExpectDeprecationTest.php
+++ b/core/tests/Drupal/Tests/ExpectDeprecationTest.php
@@ -21,4 +21,16 @@ public function testExpectDeprecation() {
     @trigger_error('Test deprecation', E_USER_DEPRECATED);
   }
 
+  /**
+   * @covers ::expectDeprecation
+   * @runInSeparateProcess
+   * @preserveGlobalState disabled
+   */
+  public function testExpectDeprecationInIsolation() {
+    $this->expectDeprecation('Test isolated deprecation');
+    $this->expectDeprecation('Test isolated deprecation2');
+    @trigger_error('Test isolated deprecation', E_USER_DEPRECATED);
+    @trigger_error('Test isolated deprecation2', E_USER_DEPRECATED);
+  }
+
 }
diff --git a/core/tests/Drupal/Tests/Listeners/DeprecationListenerTrait.php b/core/tests/Drupal/Tests/Listeners/DeprecationListenerTrait.php
index 027ead6c24f8..26967fae9b68 100644
--- a/core/tests/Drupal/Tests/Listeners/DeprecationListenerTrait.php
+++ b/core/tests/Drupal/Tests/Listeners/DeprecationListenerTrait.php
@@ -2,6 +2,9 @@
 
 namespace Drupal\Tests\Listeners;
 
+use Drupal\Tests\Traits\ExpectDeprecationTrait;
+use PHPUnit\Framework\TestCase;
+
 /**
  * Removes deprecations that we are yet to fix.
  *
@@ -10,6 +13,15 @@
  *   fixed.
  */
 trait DeprecationListenerTrait {
+  use ExpectDeprecationTrait;
+
+  protected function deprecationStartTest($test) {
+    if ($test instanceof \PHPUnit_Framework_TestCase || $test instanceof TestCase) {
+      if ($this->willBeIsolated($test)) {
+        putenv('DRUPAL_EXPECTED_DEPRECATIONS_SERIALIZE=' . tempnam(sys_get_temp_dir(), 'exdep'));
+      }
+    }
+  }
 
   /**
    * Reacts to the end of a test.
@@ -21,6 +33,13 @@ trait DeprecationListenerTrait {
    */
   protected function deprecationEndTest($test, $time) {
     /** @var \PHPUnit\Framework\Test $test */
+    if ($file = getenv('DRUPAL_EXPECTED_DEPRECATIONS_SERIALIZE')) {
+      putenv('DRUPAL_EXPECTED_DEPRECATIONS_SERIALIZE');
+      $expected_deprecations = file_get_contents($file);
+      if ($expected_deprecations) {
+        $test->expectedDeprecations(unserialize($expected_deprecations));
+      }
+    }
     if ($file = getenv('SYMFONY_DEPRECATIONS_SERIALIZE')) {
       $util_test_class = class_exists('PHPUnit_Util_Test') ? 'PHPUnit_Util_Test' : 'PHPUnit\Util\Test';
       $method = $test->getName(FALSE);
@@ -50,6 +69,26 @@ protected function deprecationEndTest($test, $time) {
     }
   }
 
+  /**
+   * Determines if a test is isolated.
+   *
+   * @param \PHPUnit_Framework_TestCase|\PHPUnit\Framework\TestCase $test
+   *   The test to check.
+   *
+   * @return bool
+   *   TRUE if the isolated, FALSE if not.
+   */
+  private function willBeIsolated($test) {
+    if ($test->isInIsolation()) {
+      return FALSE;
+    }
+
+    $r = new \ReflectionProperty($test, 'runTestInSeparateProcess');
+    $r->setAccessible(TRUE);
+
+    return $r->getValue($test);
+  }
+
   /**
    * A list of deprecations to ignore whilst fixes are put in place.
    *
diff --git a/core/tests/Drupal/Tests/Listeners/DrupalListener.php b/core/tests/Drupal/Tests/Listeners/DrupalListener.php
index 9ed976f76c44..cafaa2292b37 100644
--- a/core/tests/Drupal/Tests/Listeners/DrupalListener.php
+++ b/core/tests/Drupal/Tests/Listeners/DrupalListener.php
@@ -23,6 +23,13 @@ class DrupalListener extends BaseTestListener {
     use DrupalComponentTestListenerTrait;
     use DrupalStandardsListenerTrait;
 
+    /**
+     * {@inheritdoc}
+     */
+    public function startTest(Test $test) {
+      $this->deprecationStartTest($test);
+    }
+
     /**
      * {@inheritdoc}
      */
diff --git a/core/tests/Drupal/Tests/Listeners/Legacy/DrupalListener.php b/core/tests/Drupal/Tests/Listeners/Legacy/DrupalListener.php
index f7c2c76668cb..1fc603a6f4f1 100644
--- a/core/tests/Drupal/Tests/Listeners/Legacy/DrupalListener.php
+++ b/core/tests/Drupal/Tests/Listeners/Legacy/DrupalListener.php
@@ -17,6 +17,13 @@ class DrupalListener extends \PHPUnit_Framework_BaseTestListener {
   use DrupalComponentTestListenerTrait;
   use DrupalStandardsListenerTrait;
 
+  /**
+   * {@inheritdoc}
+   */
+  public function startTest(\PHPUnit_Framework_Test $test) {
+    $this->deprecationStartTest($test);
+  }
+
   /**
    * {@inheritdoc}
    */
diff --git a/core/tests/Drupal/Tests/Traits/ExpectDeprecationTrait.php b/core/tests/Drupal/Tests/Traits/ExpectDeprecationTrait.php
index 279612e94369..4e47f77fe598 100644
--- a/core/tests/Drupal/Tests/Traits/ExpectDeprecationTrait.php
+++ b/core/tests/Drupal/Tests/Traits/ExpectDeprecationTrait.php
@@ -4,6 +4,7 @@
 
 use Symfony\Bridge\PhpUnit\Legacy\SymfonyTestsListener as LegacySymfonyTestsListener;
 use Symfony\Bridge\PhpUnit\SymfonyTestsListener;
+use PHPUnit\Framework\TestCase;
 
 /**
  * Adds the ability to dynamically set expected deprecation messages in tests.
@@ -19,11 +20,20 @@ trait ExpectDeprecationTrait {
   /**
    * Sets an expected deprecation message.
    *
-   * @param string $msg
+   * @param string $message
    *   The expected deprecation message.
    */
-  protected function expectDeprecation($msg) {
-    // Ensure the class or method is in the legacy group.
+  protected function expectDeprecation($message) {
+    $this->expectedDeprecations([$message]);
+  }
+
+  /**
+   * Sets expected deprecation messages.
+   *
+   * @param string[] $messages
+   *   The expected deprecation messages.
+   */
+  public function expectedDeprecations(array $messages) {
     if (class_exists('PHPUnit_Util_Test', FALSE)) {
       $test_util = 'PHPUnit_Util_Test';
       $assertion_failed_error = 'PHPUnit_Framework_AssertionFailedError';
@@ -32,36 +42,54 @@ protected function expectDeprecation($msg) {
       $test_util = 'PHPUnit\Util\Test';
       $assertion_failed_error = 'PHPUnit\Framework\AssertionFailedError';
     }
-    $groups = $test_util::getGroups(get_class($this), $this->getName(FALSE));
-    if (!in_array('legacy', $groups, TRUE)) {
-      throw new $assertion_failed_error('Only tests with the `@group legacy` annotation can call `setExpectedDeprecation()`.');
-    }
+    if ($this instanceof \PHPUnit_Framework_TestCase || $this instanceof TestCase) {
+      // Ensure the class or method is in the legacy group.
+      $groups = $test_util::getGroups(get_class($this), $this->getName(FALSE));
+      if (!in_array('legacy', $groups, TRUE)) {
+        throw new $assertion_failed_error('Only tests with the `@group legacy` annotation can call `setExpectedDeprecation()`.');
+      }
 
-    if ($trait = $this->getSymfonyTestListenerTrait()) {
       // If setting an expected deprecation there is no need to be strict about
       // testing nothing as this is an assertion.
-      $this->getTestResultObject()->beStrictAboutTestsThatDoNotTestAnything(FALSE);
+      $this->getTestResultObject()
+        ->beStrictAboutTestsThatDoNotTestAnything(FALSE);
 
-      // Add the expected deprecation message to the class property.
-      $reflection_class = new \ReflectionClass($trait);
-      $expected_deprecations_property = $reflection_class->getProperty('expectedDeprecations');
-      $expected_deprecations_property->setAccessible(TRUE);
-      $expected_deprecations = $expected_deprecations_property->getValue($trait);
-      $expected_deprecations[] = $msg;
-      $expected_deprecations_property->setValue($trait, $expected_deprecations);
+      if ($trait = $this->getSymfonyTestListenerTrait()) {
+        // Add the expected deprecation message to the class property.
+        $reflection_class = new \ReflectionClass($trait);
+        $expected_deprecations_property = $reflection_class->getProperty('expectedDeprecations');
+        $expected_deprecations_property->setAccessible(TRUE);
+        $expected_deprecations = $expected_deprecations_property->getValue($trait);
+        $expected_deprecations = array_merge($expected_deprecations, $messages);
+        $expected_deprecations_property->setValue($trait, $expected_deprecations);
 
-      // Register the error handler if necessary.
-      $previous_error_handler_property = $reflection_class->getProperty('previousErrorHandler');
-      $previous_error_handler_property->setAccessible(TRUE);
-      $previous_error_handler = $previous_error_handler_property->getValue($trait);
-      if (!$previous_error_handler) {
-        $previous_error_handler = set_error_handler([$trait, 'handleError']);
-        $previous_error_handler_property->setValue($trait, $previous_error_handler);
+        // Register the error handler if necessary.
+        $previous_error_handler_property = $reflection_class->getProperty('previousErrorHandler');
+        $previous_error_handler_property->setAccessible(TRUE);
+        $previous_error_handler = $previous_error_handler_property->getValue($trait);
+        if (!$previous_error_handler) {
+          $previous_error_handler = set_error_handler([$trait, 'handleError']);
+          $previous_error_handler_property->setValue($trait, $previous_error_handler);
+        }
+        return;
       }
     }
-    else {
-      throw new $assertion_failed_error('Can not set an expected deprecation message because the Symfony\Bridge\PhpUnit\SymfonyTestsListener is not registered as a PHPUnit test listener.');
+
+    // Expected deprecations set by isolated tests need to be written to a file
+    // so that the test running process can take account of them.
+    if ($file = getenv('DRUPAL_EXPECTED_DEPRECATIONS_SERIALIZE')) {
+      $expected_deprecations = file_get_contents($file);
+      if ($expected_deprecations) {
+        $expected_deprecations = array_merge(unserialize($expected_deprecations), $messages);
+      }
+      else {
+        $expected_deprecations = $messages;
+      }
+      file_put_contents($file, serialize($expected_deprecations));
+      return;
     }
+
+    throw new $assertion_failed_error('Can not set an expected deprecation message because the Symfony\Bridge\PhpUnit\SymfonyTestsListener is not registered as a PHPUnit test listener.');
   }
 
   /**
-- 
GitLab