From f5040c6cdedc57bc0e9fde8954b058db9dedcb90 Mon Sep 17 00:00:00 2001
From: Lee Rowlands <lee.rowlands@previousnext.com.au>
Date: Wed, 9 Oct 2019 10:14:28 +1000
Subject: [PATCH] Revert "Issue #3085697 by greg.1.anderson, mbaynton,
 Mixologic, larowlan: Allow scaffold plugin to append to non-scaffolded files"

This reverts commit c5ac07aa8a50e6902c2c516001cb423096c772e1.
---
 .../Scaffold/Operations/AbstractOperation.php |  26 ----
 .../Plugin/Scaffold/Operations/AppendOp.php   | 118 ++----------------
 .../Operations/ConjoinableInterface.php       |  13 ++
 .../Scaffold/Operations/ConjunctionOp.php     |   2 +-
 .../Scaffold/Operations/OperationData.php     |  66 +---------
 .../Scaffold/Operations/OperationFactory.php  |  28 +----
 .../Operations/OperationInterface.php         |  27 ----
 .../Plugin/Scaffold/Operations/ReplaceOp.php  |   2 +-
 .../Operations/ScaffoldFileCollection.php     |  14 +--
 .../Plugin/Scaffold/Operations/SkipOp.php     |   2 +-
 composer/Plugin/Scaffold/README.md            |  24 ----
 .../Plugin/Scaffold/AssertUtilsTrait.php      |  17 ---
 .../Functional/ManageGitIgnoreTest.php        |  21 ----
 .../Scaffold/Functional/ScaffoldTest.php      |  32 +----
 .../assets/append-to-settings.txt             |   1 -
 .../assets/default-settings.txt               |   3 -
 .../composer.json.tmpl                        |  64 ----------
 17 files changed, 40 insertions(+), 420 deletions(-)
 delete mode 100644 composer/Plugin/Scaffold/Operations/AbstractOperation.php
 create mode 100644 composer/Plugin/Scaffold/Operations/ConjoinableInterface.php
 delete mode 100644 core/tests/Drupal/Tests/Composer/Plugin/Scaffold/fixtures/drupal-drupal-append-settings/assets/append-to-settings.txt
 delete mode 100644 core/tests/Drupal/Tests/Composer/Plugin/Scaffold/fixtures/drupal-drupal-append-settings/assets/default-settings.txt
 delete mode 100644 core/tests/Drupal/Tests/Composer/Plugin/Scaffold/fixtures/drupal-drupal-append-settings/composer.json.tmpl

diff --git a/composer/Plugin/Scaffold/Operations/AbstractOperation.php b/composer/Plugin/Scaffold/Operations/AbstractOperation.php
deleted file mode 100644
index 5f7261fd262c..000000000000
--- a/composer/Plugin/Scaffold/Operations/AbstractOperation.php
+++ /dev/null
@@ -1,26 +0,0 @@
-<?php
-
-namespace Drupal\Composer\Plugin\Scaffold\Operations;
-
-use Drupal\Composer\Plugin\Scaffold\ScaffoldFilePath;
-
-/**
- * Provides default behaviors for operations.
- */
-abstract class AbstractOperation implements OperationInterface {
-
-  /**
-   * {@inheritdoc}
-   */
-  public function combineWithConjunctionTarget(OperationInterface $conjunction_target) {
-    return $this;
-  }
-
-  /**
-   * {@inheritdoc}
-   */
-  public function missingConjunctionTarget(ScaffoldFilePath $destination) {
-    return $this;
-  }
-
-}
diff --git a/composer/Plugin/Scaffold/Operations/AppendOp.php b/composer/Plugin/Scaffold/Operations/AppendOp.php
index 7f5c7e8b0dbe..1a3ecd2d86a9 100644
--- a/composer/Plugin/Scaffold/Operations/AppendOp.php
+++ b/composer/Plugin/Scaffold/Operations/AppendOp.php
@@ -9,7 +9,7 @@
 /**
  * Scaffold operation to add to the beginning and/or end of a scaffold file.
  */
-class AppendOp extends AbstractOperation {
+class AppendOp implements OperationInterface, ConjoinableInterface {
 
   /**
    * Identifies Append operations.
@@ -30,23 +30,6 @@ class AppendOp extends AbstractOperation {
    */
   protected $append;
 
-  /**
-   * Path to the default data to use when appending to an empty file.
-   *
-   * @var \Drupal\Composer\Plugin\Scaffold\ScaffoldFilePath
-   */
-  protected $default;
-
-  /**
-   * An indicator of whether the file we are appending to is managed or not.
-   */
-  protected $managed;
-
-  /**
-   * An indicator of whether we are allowed to append to a non-scaffolded file.
-   */
-  protected $forceAppend;
-
   /**
    * Constructs an AppendOp.
    *
@@ -54,17 +37,10 @@ class AppendOp extends AbstractOperation {
    *   The relative path to the prepend file.
    * @param \Drupal\Composer\Plugin\Scaffold\ScaffoldFilePath $append_path
    *   The relative path to the append file.
-   * @param bool $force_append
-   *   TRUE if is okay to append to a file that was not scaffolded.
-   * @param \Drupal\Composer\Plugin\Scaffold\ScaffoldFilePath $default_path
-   *   The relative path to the default data.
    */
-  public function __construct(ScaffoldFilePath $prepend_path = NULL, ScaffoldFilePath $append_path = NULL, $force_append = FALSE, ScaffoldFilePath $default_path = NULL) {
-    $this->forceAppend = $force_append;
+  public function __construct(ScaffoldFilePath $prepend_path = NULL, ScaffoldFilePath $append_path = NULL) {
     $this->prepend = $prepend_path;
     $this->append = $append_path;
-    $this->default = $default_path;
-    $this->managed = TRUE;
   }
 
   /**
@@ -72,25 +48,11 @@ public function __construct(ScaffoldFilePath $prepend_path = NULL, ScaffoldFileP
    */
   public function process(ScaffoldFilePath $destination, IOInterface $io, ScaffoldOptions $options) {
     $destination_path = $destination->fullPath();
-    // This is just a sanity check; the OperationFactory has in theory already
-    // accounted for this, and will return a SkipOp with a warning message.
-    if (!file_exists($destination_path) && empty($this->default)) {
-      throw new \RuntimeException($destination->getInterpolator()->interpolate("Cannot append/prepend because no prior package provided a scaffold file at [dest-rel-path]."));
+    if (!file_exists($destination_path)) {
+      throw new \RuntimeException($destination->getInterpolator()->interpolate("Cannot append/prepend because no prior package provided a scaffold file at that [dest-rel-path]."));
     }
     $interpolator = $destination->getInterpolator();
 
-    // Be extra-noisy of creating a new file or appending to a non-scaffold
-    // file. Note that if the file already has the append contents, then the
-    // OperationFactory will make a SkipOp instead, and we will not get here.
-    if (!$this->managed) {
-      $message = '  - <info>NOTICE</info> Modifying existing file at <info>[dest-rel-path]</info>.';
-      if (!file_exists($destination_path)) {
-        $message = '  - <info>NOTICE</info> Creating a new file at <info>[dest-rel-path]</info>.';
-      }
-      $message .= ' Examine the contents and ensure that it came out correctly.';
-      $io->write($interpolator->interpolate($message));
-    }
-
     // Fetch the prepend contents, if provided.
     $prepend_contents = '';
     if (!empty($this->prepend)) {
@@ -105,82 +67,18 @@ public function process(ScaffoldFilePath $destination, IOInterface $io, Scaffold
       $append_contents = "\n" . file_get_contents($this->append->fullPath());
       $io->write($interpolator->interpolate("  - Append to <info>[dest-rel-path]</info> from <info>[append-rel-path]</info>"));
     }
-    // We typically should always have content if we get here; the
-    // OperationFactory should create a SkipOp instead of an AppendOp if there
-    // is no append / prepend content. The edge case is if there is content
-    // that is all 'trim'ed away. Then we get a message that we are appending,
-    // although nothing will in fact actually happen.
     if (!empty(trim($prepend_contents)) || !empty(trim($append_contents))) {
       // None of our asset files are very large, so we will load each one into
       // memory for processing.
-      $original_contents = file_get_contents(file_exists($destination_path) ? $destination_path : $this->default->fullPath());
+      $original_contents = file_get_contents($destination_path);
       // Write the appended and prepended contents back to the file.
       $altered_contents = $prepend_contents . $original_contents . $append_contents;
       file_put_contents($destination_path, $altered_contents);
     }
-
-    // Return a ScaffoldResult with knowledge of whether this file is managed.
-    return new ScaffoldResult($destination, $this->managed);
-  }
-
-  /**
-   * {@inheritdoc}
-   */
-  public function combineWithConjunctionTarget(OperationInterface $conjunction_target) {
-    return new ConjunctionOp($conjunction_target, $this);
-  }
-
-  /**
-   * {@inheritdoc}
-   */
-  public function missingConjunctionTarget(ScaffoldFilePath $destination) {
-    // If there is no conjunction target (the destination is not scaffolded),
-    // then any append we do will be to an unmanaged file.
-    $this->managed = FALSE;
-
-    // Default: do not allow an append over a file that was not scaffolded.
-    if (!$this->forceAppend) {
-      $message = "  - Skip <info>[dest-rel-path]</info>: cannot append to a path that was not scaffolded unless 'force-append' property is set.";
-      return new SkipOp($message);
-    }
-
-    // If the target file does not exist, then we will allow the append to
-    // happen if we have default data to provide for it.
-    if (!file_exists($destination->fullPath())) {
-      if (!empty($this->default)) {
-        return $this;
-      }
-      $message = "  - Skip <info>[dest-rel-path]</info>: no file exists at the target path, and no default data provided.";
-      return new SkipOp($message);
-    }
-
-    // If the target file DOES exist, and it already contains the append/prepend
-    // data, then we will skip the operation.
-    $existingData = file_get_contents($destination->fullPath());
-    if ($this->existingFileHasData($existingData, $this->append) || $this->existingFileHasData($existingData, $this->prepend)) {
-      $message = "  - Skip <info>[dest-rel-path]</info>: the file already has the append/prepend data.";
-      return new SkipOp($message);
+    else {
+      $io->write($interpolator->interpolate("  - Keep <info>[dest-rel-path]</info> unchanged: no content to prepend / append was provided."));
     }
-
-    return $this;
-  }
-
-  /**
-   * Check to see if the append/prepend data has already been applied.
-   * @param string $contents
-   *   The contents of the target file.
-   * @param \Drupal\Composer\Plugin\Scaffold\ScaffoldFilePath $data_path
-   *   The path to the data to append or prepend
-   * @return bool
-   *   'TRUE' if the append/prepend data already exists in contents.
-   */
-  protected function existingFileHasData($contents, $data_path) {
-    if (empty($data_path)) {
-      return FALSE;
-    }
-    $data = file_get_contents($data_path->fullPath());
-
-    return strpos($contents, $data) !== FALSE;
+    return new ScaffoldResult($destination, TRUE);
   }
 
 }
diff --git a/composer/Plugin/Scaffold/Operations/ConjoinableInterface.php b/composer/Plugin/Scaffold/Operations/ConjoinableInterface.php
new file mode 100644
index 000000000000..ca8d433971ca
--- /dev/null
+++ b/composer/Plugin/Scaffold/Operations/ConjoinableInterface.php
@@ -0,0 +1,13 @@
+<?php
+
+namespace Drupal\Composer\Plugin\Scaffold\Operations;
+
+/**
+ * Marker interface indicating that operation is conjoinable.
+ *
+ * A conjoinable operation is one that runs in addition to any previous
+ * operation defined at the same destination path. Operations that are
+ * not conjoinable simply replace anything at the same destination path.
+ */
+interface ConjoinableInterface {
+}
diff --git a/composer/Plugin/Scaffold/Operations/ConjunctionOp.php b/composer/Plugin/Scaffold/Operations/ConjunctionOp.php
index 83b2441271c7..f855e4e4dee8 100644
--- a/composer/Plugin/Scaffold/Operations/ConjunctionOp.php
+++ b/composer/Plugin/Scaffold/Operations/ConjunctionOp.php
@@ -9,7 +9,7 @@
 /**
  * Joins two operations on the same file into a single operation.
  */
-class ConjunctionOp extends AbstractOperation {
+class ConjunctionOp implements OperationInterface {
 
   /**
    * The first operation.
diff --git a/composer/Plugin/Scaffold/Operations/OperationData.php b/composer/Plugin/Scaffold/Operations/OperationData.php
index 66b85229aed4..c98fe2c2ed84 100644
--- a/composer/Plugin/Scaffold/Operations/OperationData.php
+++ b/composer/Plugin/Scaffold/Operations/OperationData.php
@@ -12,8 +12,6 @@ class OperationData {
   const OVERWRITE = 'overwrite';
   const PREPEND = 'prepend';
   const APPEND = 'append';
-  const DEFAULT = 'default';
-  const FORCE_APPEND = 'force-append';
 
   /**
    * The parameter data.
@@ -87,20 +85,7 @@ public function path() {
    *   Returns true if overwrite mode was selected.
    */
   public function overwrite() {
-    return !empty($this->data[self::OVERWRITE]);
-  }
-
-  /**
-   * Determines whether 'force-append' has been set.
-   *
-   * @return bool
-   *   Returns true if 'force-append' mode was selected.
-   */
-  public function forceAppend() {
-    if ($this->hasDefault()) {
-      return TRUE;
-    }
-    return !empty($this->data[self::FORCE_APPEND]);
+    return isset($this->data[self::OVERWRITE]) ? $this->data[self::OVERWRITE] : TRUE;
   }
 
   /**
@@ -143,26 +128,6 @@ public function append() {
     return $this->data[self::APPEND];
   }
 
-  /**
-   * Checks if default path exists.
-   *
-   * @return bool
-   *   Returns true if there is default data available.
-   */
-  public function hasDefault() {
-    return isset($this->data[self::DEFAULT]);
-  }
-
-  /**
-   * Gets default path.
-   *
-   * @return string
-   *   Path to default data
-   */
-  public function default() {
-    return $this->data[self::DEFAULT];
-  }
-
   /**
    * Normalizes metadata by converting literal values into arrays.
    *
@@ -176,32 +141,9 @@ public function default() {
    *   The metadata for this operation object, which varies by operation type.
    *
    * @return array
-   *   Normalized scaffold metadata with default values.
-   */
-  protected function normalizeScaffoldMetadata($destination, $value) {
-    $defaultScaffoldMetadata = [
-      self::MODE => ReplaceOp::ID,
-      self::PREPEND => NULL,
-      self::APPEND => NULL,
-      self::DEFAULT => NULL,
-      self::OVERWRITE => TRUE,
-    ];
-
-    return $this->convertScaffoldMetadata($destination, $value) + $defaultScaffoldMetadata;
-  }
-
-  /**
-   * Performs the conversion-to-array step in normalizeScaffoldMetadata.
-   *
-   * @param string $destination
-   *   The destination path for the scaffold file.
-   * @param mixed $value
-   *   The metadata for this operation object, which varies by operation type.
-   *
-   * @return array
    *   Normalized scaffold metadata.
    */
-  protected function convertScaffoldMetadata($destination, $value) {
+  protected function normalizeScaffoldMetadata($destination, $value) {
     if (is_bool($value)) {
       if (!$value) {
         return [self::MODE => SkipOp::ID];
@@ -219,6 +161,10 @@ protected function convertScaffoldMetadata($destination, $value) {
     if (!isset($value[self::MODE]) && (isset($value[self::APPEND]) || isset($value[self::PREPEND]))) {
       $value[self::MODE] = AppendOp::ID;
     }
+    // If there is no 'mode', then the default is 'replace'.
+    if (!isset($value[self::MODE])) {
+      $value[self::MODE] = ReplaceOp::ID;
+    }
     return $value;
   }
 
diff --git a/composer/Plugin/Scaffold/Operations/OperationFactory.php b/composer/Plugin/Scaffold/Operations/OperationFactory.php
index 7742e4bb9d7f..4c0efa046291 100644
--- a/composer/Plugin/Scaffold/Operations/OperationFactory.php
+++ b/composer/Plugin/Scaffold/Operations/OperationFactory.php
@@ -99,38 +99,14 @@ protected function createAppendOp(PackageInterface $package, OperationData $oper
     $package_path = $this->getPackagePath($package);
     $prepend_source_file = NULL;
     $append_source_file = NULL;
-    $default_data_file = NULL;
     if ($operation_data->hasPrepend()) {
       $prepend_source_file = ScaffoldFilePath::sourcePath($package_name, $package_path, $operation_data->destination(), $operation_data->prepend());
     }
     if ($operation_data->hasAppend()) {
       $append_source_file = ScaffoldFilePath::sourcePath($package_name, $package_path, $operation_data->destination(), $operation_data->append());
     }
-    if ($operation_data->hasDefault()) {
-      $default_data_file = ScaffoldFilePath::sourcePath($package_name, $package_path, $operation_data->destination(), $operation_data->default());
-    }
-    if (!$this->hasContent($prepend_source_file) && !$this->hasContent($append_source_file)) {
-      $message = '  - Keep <info>[dest-rel-path]</info> unchanged: no content to prepend / append was provided.';
-      return new SkipOp($message);
-    }
-
-    return new AppendOp($prepend_source_file, $append_source_file, $operation_data->forceAppend(), $default_data_file);
-  }
-
-  /**
-   * Checks to see if the specified scaffold file exists and has content.
-   *
-   * @param Drupal\Composer\Plugin\Scaffold\ScaffoldFilePath $file
-   *   Scaffold file to check.
-   * @return bool
-   *   True if the file exists and has content.
-   */
-  protected function hasContent(ScaffoldFilePath $file = NULL) {
-    if (!$file) {
-      return FALSE;
-    }
-    $path = $file->fullPath();
-    return is_file($path) && (filesize($path) > 0);
+    $op = new AppendOp($prepend_source_file, $append_source_file);
+    return $op;
   }
 
   /**
diff --git a/composer/Plugin/Scaffold/Operations/OperationInterface.php b/composer/Plugin/Scaffold/Operations/OperationInterface.php
index 409b4761a2a9..5c1d4506e8de 100644
--- a/composer/Plugin/Scaffold/Operations/OperationInterface.php
+++ b/composer/Plugin/Scaffold/Operations/OperationInterface.php
@@ -26,31 +26,4 @@ interface OperationInterface {
    */
   public function process(ScaffoldFilePath $destination, IOInterface $io, ScaffoldOptions $options);
 
-  /**
-   * Determines what to do if operation is used with a previous operation.
-   *
-   * Default behavior is to scaffold this operation at the specified
-   * destination, ignoring whatever was there before.
-   *
-   * @param OperationInterface $conjunction_target
-   *   Existing file at the destination path that we should combine with.
-   *
-   * @return OperationInterface
-   *   The op to use at this destination.
-   */
-  public function combineWithConjunctionTarget(OperationInterface $conjunction_target);
-
-  /**
-   * Determines what to do if operation is used without a previous operation.
-   *
-   * Default behavior is to scaffold this operation at the specified
-   * destination. Most operations overwrite rather than modify existing files,
-   * and therefore do not need to do anything special when there is no existing
-   * file.
-   *
-   * @return OperationInterface
-   *   The op to use at this destination.
-   */
-  public function missingConjunctionTarget(ScaffoldFilePath $destination);
-
 }
diff --git a/composer/Plugin/Scaffold/Operations/ReplaceOp.php b/composer/Plugin/Scaffold/Operations/ReplaceOp.php
index 5b4c4b9b1d88..c2bf71c296a5 100644
--- a/composer/Plugin/Scaffold/Operations/ReplaceOp.php
+++ b/composer/Plugin/Scaffold/Operations/ReplaceOp.php
@@ -10,7 +10,7 @@
 /**
  * Scaffold operation to copy or symlink from source to destination.
  */
-class ReplaceOp extends AbstractOperation {
+class ReplaceOp implements OperationInterface {
 
   /**
    * Identifies Replace operations.
diff --git a/composer/Plugin/Scaffold/Operations/ScaffoldFileCollection.php b/composer/Plugin/Scaffold/Operations/ScaffoldFileCollection.php
index c000e64a478e..ea97cb873446 100644
--- a/composer/Plugin/Scaffold/Operations/ScaffoldFileCollection.php
+++ b/composer/Plugin/Scaffold/Operations/ScaffoldFileCollection.php
@@ -42,7 +42,6 @@ public function __construct(array $file_mappings, Interpolator $location_replace
     foreach ($file_mappings as $package_name => $package_file_mappings) {
       foreach ($package_file_mappings as $destination_rel_path => $op) {
         $destination = ScaffoldFilePath::destinationPath($package_name, $destination_rel_path, $location_replacements);
-
         // If there was already a scaffolding operation happening at this path,
         // and the new operation is Conjoinable, then use a ConjunctionOp to
         // join together both operations. This will cause both operations to
@@ -51,20 +50,13 @@ public function __construct(array $file_mappings, Interpolator $location_replace
         // path.
         if (isset($scaffoldFiles[$destination_rel_path])) {
           $previous_scaffold_file = $scaffoldFiles[$destination_rel_path];
-          $op = $op->combineWithConjunctionTarget($previous_scaffold_file->op());
-
+          if ($op instanceof ConjoinableInterface) {
+            $op = new ConjunctionOp($previous_scaffold_file->op(), $op);
+          }
           // Remove the previous op so we only touch the destination once.
           $message = "  - Skip <info>[dest-rel-path]</info>: overridden in <comment>{$package_name}</comment>";
           $this->scaffoldFilesByProject[$previous_scaffold_file->packageName()][$destination_rel_path] = new ScaffoldFileInfo($destination, new SkipOp($message));
         }
-        // If there is NOT already a scaffolding operation happening at this
-        // path, but the operation is a ConjunctionOp, then we need to check
-        // to see if there is a strategy for non-conjunction use.
-        else {
-          $op = $op->missingConjunctionTarget($destination);
-        }
-
-        // Combine the scaffold operation with the destination and record it.
         $scaffold_file = new ScaffoldFileInfo($destination, $op);
         $scaffoldFiles[$destination_rel_path] = $scaffold_file;
         $this->scaffoldFilesByProject[$package_name][$destination_rel_path] = $scaffold_file;
diff --git a/composer/Plugin/Scaffold/Operations/SkipOp.php b/composer/Plugin/Scaffold/Operations/SkipOp.php
index 3e6c111bdfcb..10b0ec9bbc35 100644
--- a/composer/Plugin/Scaffold/Operations/SkipOp.php
+++ b/composer/Plugin/Scaffold/Operations/SkipOp.php
@@ -9,7 +9,7 @@
 /**
  * Scaffold operation to skip a scaffold file (do nothing).
  */
-class SkipOp extends AbstractOperation {
+class SkipOp implements OperationInterface {
 
   /**
    * Identifies Skip operations.
diff --git a/composer/Plugin/Scaffold/README.md b/composer/Plugin/Scaffold/README.md
index 659f0ed3f6d2..ae696cea0d67 100644
--- a/composer/Plugin/Scaffold/README.md
+++ b/composer/Plugin/Scaffold/README.md
@@ -302,30 +302,6 @@ path, and it is not possible for multiple entries to have the same key. If
 "prepend" were a separate mode, then it would not be possible to both prepend
 and append to the same file.
 
-By default, append operations may only be applied to files that were scaffolded
-by a previously evaluated project. If the `force-append` attribute is added to
-an `append` operation, though, then the append will be made to non-scaffolded
-files if and only if the append text does not already appear in the file. When
-using this mode, it is also possible to provide default contents to use in the
-event that the destination file is entirely missing.
-
-The example below demonstrates scaffolding a settings-custom.php file, and
-including it from the existing `settings.php` file.
-
-```
-"file-mapping": {
-  "[web-root]/sites/default/settings-custom.php": "assets/settings-custom.php",
-  "[web-root]/sites/default/settings.php": {
-    "append": "assets/include-settings-custom.txt",
-    "force-append": true,
-    "default": "assets/initial-default-settings.txt"
-  }
-}
-```
-
-Note that the example above still works if used with a project that scaffolds
-the settings.php file.
-
 ### gitignore
 
 The `gitignore` configuration setting controls whether or not this plugin will
diff --git a/core/tests/Drupal/Tests/Composer/Plugin/Scaffold/AssertUtilsTrait.php b/core/tests/Drupal/Tests/Composer/Plugin/Scaffold/AssertUtilsTrait.php
index 2f39d832ec80..5634f492dd88 100644
--- a/core/tests/Drupal/Tests/Composer/Plugin/Scaffold/AssertUtilsTrait.php
+++ b/core/tests/Drupal/Tests/Composer/Plugin/Scaffold/AssertUtilsTrait.php
@@ -24,21 +24,4 @@ protected function assertScaffoldedFile($path, $is_link, $contents_contains) {
     $this->assertSame($is_link, is_link($path));
   }
 
-  /**
-   * Asserts that a file does not exist or exists and does not contain a value.
-   *
-   * @param string $path
-   *   The path to check exists.
-   * @param string $contents_not_contains
-   *   A string that is expected should NOT occur in the file contents.
-   */
-  protected function assertScaffoldedFileDoesNotContain($path, $contents_not_contains) {
-    // If the file does not exist at all, we'll count that as a pass.
-    if (!file_exists($path)) {
-      return;
-    }
-    $contents = file_get_contents($path);
-    $this->assertNotContains($contents_not_contains, $contents, basename($path) . ' contains unexpected contents:');
-  }
-
 }
diff --git a/core/tests/Drupal/Tests/Composer/Plugin/Scaffold/Functional/ManageGitIgnoreTest.php b/core/tests/Drupal/Tests/Composer/Plugin/Scaffold/Functional/ManageGitIgnoreTest.php
index c73faf74bdf2..b7ea314c6159 100644
--- a/core/tests/Drupal/Tests/Composer/Plugin/Scaffold/Functional/ManageGitIgnoreTest.php
+++ b/core/tests/Drupal/Tests/Composer/Plugin/Scaffold/Functional/ManageGitIgnoreTest.php
@@ -159,27 +159,6 @@ public function testUnmanagedGitIgnoreWhenDisabled() {
     $this->assertFileNotExists($sut . '/docroot/sites/default/.gitignore');
   }
 
-  /**
-   * Test appending to an unmanaged file, and confirm it is not .gitignored.
-   *
-   * If we append to an unmanaged (not scaffolded) file, and we are managing
-   * .gitignore files, then we expect that the unmanaged file should not be
-   * added to the .gitignore file, because unmanaged files should be committed.
-   */
-  public function testAppendToEmptySettingsIsUnmanaged() {
-    $sut = $this->createSutWithGit('drupal-drupal-append-settings');
-    $this->assertFileNotExists($sut . '/autoload.php');
-    $this->assertFileNotExists($sut . '/index.php');
-    $this->assertFileNotExists($sut . '/sites/.gitignore');
-    // Run the scaffold command.
-    $this->fixtures->runScaffold($sut);
-    $this->assertFileExists($sut . '/autoload.php');
-    $this->assertFileExists($sut . '/index.php');
-
-    $this->assertScaffoldedFile($sut . '/sites/.gitignore', FALSE, 'example.sites.php');
-    $this->assertScaffoldedFileDoesNotContain($sut . '/sites/.gitignore', 'settings.php');
-  }
-
   /**
    * Tests scaffold command disables .gitignore management when git not present.
    *
diff --git a/core/tests/Drupal/Tests/Composer/Plugin/Scaffold/Functional/ScaffoldTest.php b/core/tests/Drupal/Tests/Composer/Plugin/Scaffold/Functional/ScaffoldTest.php
index 948dd8aa12fd..357c6bd1d1ae 100644
--- a/core/tests/Drupal/Tests/Composer/Plugin/Scaffold/Functional/ScaffoldTest.php
+++ b/core/tests/Drupal/Tests/Composer/Plugin/Scaffold/Functional/ScaffoldTest.php
@@ -260,20 +260,7 @@ public function testDrupalDrupalFileWasReplaced() {
   public function scaffoldAppendTestValues() {
     return array_merge(
       $this->scaffoldAppendTestValuesToPermute(FALSE),
-      $this->scaffoldAppendTestValuesToPermute(TRUE),
-      [
-        [
-          'drupal-drupal-append-settings',
-          FALSE,
-          'sites/default/settings.php',
-          '<?php
-
-// Default settings.php contents
-
-include __DIR__ . "/settings-custom-additions.php";',
-          'NOTICE Creating a new file at [web-root]/sites/default/settings.php. Examine the contents and ensure that it came out correctly.',
-        ],
-      ]
+      $this->scaffoldAppendTestValuesToPermute(TRUE)
     );
   }
 
@@ -288,7 +275,6 @@ protected function scaffoldAppendTestValuesToPermute($is_link) {
       [
         'drupal-drupal-test-append',
         $is_link,
-        'robots.txt',
         '# robots.txt fixture scaffolded from "file-mappings" in drupal-drupal-test-append composer.json fixture.
 # This content is prepended to the top of the existing robots.txt fixture.
 # ::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::
@@ -299,13 +285,11 @@ protected function scaffoldAppendTestValuesToPermute($is_link) {
 # This content is appended to the bottom of the existing robots.txt fixture.
 # robots.txt fixture scaffolded from "file-mappings" in drupal-drupal-test-append composer.json fixture.
 ',
-        'Prepend to [web-root]/robots.txt from assets/prepend-to-robots.txt',
       ],
 
       [
         'drupal-drupal-append-to-append',
         $is_link,
-        'robots.txt',
         '# robots.txt fixture scaffolded from "file-mappings" in drupal-drupal-append-to-append composer.json fixture.
 # This content is prepended to the top of the existing robots.txt fixture.
 # ::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::
@@ -319,7 +303,6 @@ protected function scaffoldAppendTestValuesToPermute($is_link) {
 # ::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::
 # This content is appended to the bottom of the existing robots.txt fixture.
 # robots.txt fixture scaffolded from "file-mappings" in drupal-drupal-append-to-append composer.json fixture.',
-        'Append to [web-root]/robots.txt from assets/append-to-robots.txt',
       ],
     ];
   }
@@ -332,20 +315,15 @@ protected function scaffoldAppendTestValuesToPermute($is_link) {
    *   core/tests/Drupal/Tests/Component/Scaffold/fixtures.
    * @param bool $is_link
    *   Whether or not symlinking should be used.
-   * @param string $scaffold_file_path
-   *   Relative path to the scaffold file target we are testing.
-   * @param string $scaffold_file_contents
-   *   A string expected to be contained inside the scaffold file we are testing.
-   * @param string $scaffoldOutputContains
-   *   A string expected to be contained in the scaffold command output.
+   * @param string $robots_txt_contents
+   *   Regular expression matching expectations for robots.txt.
    *
    * @dataProvider scaffoldAppendTestValues
    */
-  public function testDrupalDrupalFileWasAppended($fixture_name, $is_link, $scaffold_file_path, $scaffold_file_contents, $scaffoldOutputContains) {
+  public function testDrupalDrupalFileWasAppended($fixture_name, $is_link, $robots_txt_contents) {
     $result = $this->scaffoldSut($fixture_name, $is_link, FALSE);
-    $this->assertContains($scaffoldOutputContains, $result->scaffoldOutput());
 
-    $this->assertScaffoldedFile($result->docroot() . '/' . $scaffold_file_path, FALSE, $scaffold_file_contents);
+    $this->assertScaffoldedFile($result->docroot() . '/robots.txt', FALSE, $robots_txt_contents);
     $this->assertCommonDrupalAssetsWereScaffolded($result->docroot(), $is_link);
     $this->assertAutoloadFileCorrect($result->docroot());
   }
diff --git a/core/tests/Drupal/Tests/Composer/Plugin/Scaffold/fixtures/drupal-drupal-append-settings/assets/append-to-settings.txt b/core/tests/Drupal/Tests/Composer/Plugin/Scaffold/fixtures/drupal-drupal-append-settings/assets/append-to-settings.txt
deleted file mode 100644
index c6ed5c310161..000000000000
--- a/core/tests/Drupal/Tests/Composer/Plugin/Scaffold/fixtures/drupal-drupal-append-settings/assets/append-to-settings.txt
+++ /dev/null
@@ -1 +0,0 @@
-include __DIR__ . "/settings-custom-additions.php";
diff --git a/core/tests/Drupal/Tests/Composer/Plugin/Scaffold/fixtures/drupal-drupal-append-settings/assets/default-settings.txt b/core/tests/Drupal/Tests/Composer/Plugin/Scaffold/fixtures/drupal-drupal-append-settings/assets/default-settings.txt
deleted file mode 100644
index b3621e07098e..000000000000
--- a/core/tests/Drupal/Tests/Composer/Plugin/Scaffold/fixtures/drupal-drupal-append-settings/assets/default-settings.txt
+++ /dev/null
@@ -1,3 +0,0 @@
-<?php
-
-// Default settings.php contents
diff --git a/core/tests/Drupal/Tests/Composer/Plugin/Scaffold/fixtures/drupal-drupal-append-settings/composer.json.tmpl b/core/tests/Drupal/Tests/Composer/Plugin/Scaffold/fixtures/drupal-drupal-append-settings/composer.json.tmpl
deleted file mode 100644
index 931912467a67..000000000000
--- a/core/tests/Drupal/Tests/Composer/Plugin/Scaffold/fixtures/drupal-drupal-append-settings/composer.json.tmpl
+++ /dev/null
@@ -1,64 +0,0 @@
-{
-  "name": "fixtures/drupal-drupal-append-settings",
-  "type": "project",
-  "minimum-stability": "dev",
-  "prefer-stable": true,
-  "repositories": {
-    "packagist.org": false,
-    "composer-scaffold": {
-      "type": "path",
-      "url": "__PROJECT_ROOT__",
-      "options": {
-        "symlink": true
-      }
-    },
-    "drupal-core-fixture": {
-      "type": "path",
-      "url": "../drupal-core-fixture",
-      "options": {
-        "symlink": true
-      }
-    },
-    "drupal-assets-fixture": {
-      "type": "path",
-      "url": "../drupal-assets-fixture",
-      "options": {
-        "symlink": true
-      }
-    }
-  },
-  "require": {
-    "drupal/core-composer-scaffold": "*",
-    "fixtures/drupal-core-fixture": "*"
-  },
-  "extra": {
-    "composer-scaffold": {
-      "allowed-packages": [
-        "fixtures/drupal-core-fixture"
-      ],
-      "gitignore": true,
-      "locations": {
-        "web-root": "./"
-      },
-      "symlink": __SYMLINK__,
-      "file-mapping": {
-        "[web-root]/.htaccess": false,
-        "[web-root]/sites/default/settings.php": {
-          "default": "assets/default-settings.txt",
-          "append": "assets/append-to-settings.txt"
-        }
-      }
-    },
-    "installer-paths": {
-      "core": ["type:drupal-core"],
-      "modules/contrib/{$name}": ["type:drupal-module"],
-      "modules/custom/{$name}": ["type:drupal-custom-module"],
-      "profiles/contrib/{$name}": ["type:drupal-profile"],
-      "profiles/custom/{$name}": ["type:drupal-custom-profile"],
-      "themes/contrib/{$name}": ["type:drupal-theme"],
-      "themes/custom/{$name}": ["type:drupal-custom-theme"],
-      "libraries/{$name}": ["type:drupal-library"],
-      "drush/Commands/contrib/{$name}": ["type:drupal-drush"]
-    }
-  }
-}
-- 
GitLab