Updated to Drupal 8.5. Core Media not yet in use.
[yaffs-website] / vendor / drush / drush / src / Preflight / ArgsPreprocessor.php
diff --git a/vendor/drush/drush/src/Preflight/ArgsPreprocessor.php b/vendor/drush/drush/src/Preflight/ArgsPreprocessor.php
new file mode 100644 (file)
index 0000000..e54bcd4
--- /dev/null
@@ -0,0 +1,191 @@
+<?php
+namespace Drush\Preflight;
+
+use Drush\SiteAlias\SiteAliasName;
+use Drush\SiteAlias\SiteSpecParser;
+
+/**
+ * Preprocess commandline arguments.
+ *
+ * - Record @sitealias, if present
+ * - Record a limited number of global options
+ *
+ * Anything not handled here is processed by Symfony Console.
+ */
+class ArgsPreprocessor
+{
+    /** @var SiteSpecParser */
+    protected $specParser;
+    /** @var ArgsRemapper */
+    protected $remapper;
+
+    /**
+     * ArgsPreprocessor constructor
+     */
+    public function __construct()
+    {
+        $this->specParser = new SiteSpecParser();
+    }
+
+    public function setArgsRemapper(ArgsRemapper $remapper)
+    {
+        $this->remapper = $remapper;
+    }
+
+    /**
+     * Parse the argv array.
+     *
+     * @param string[] $argv
+     *   Commandline arguments. The first element is
+     *   the path to the application, which we will ignore.
+     * @param PreflightArgsInterface $storage
+     *   A storage object to hold the arguments we remove
+     *   from argv, plus the remaining argv arguments.
+     */
+    public function parse($argv, PreflightArgsInterface $storage)
+    {
+        $sawArg = false;
+
+        // Pull off the path to application. Add it to the
+        // 'unprocessed' args list.
+        $appName = array_shift($argv);
+        $storage->addArg($appName);
+
+        if ($this->remapper) {
+            $argv = $this->remapper->remap($argv);
+        }
+
+        $optionsTable = $storage->optionsWithValues();
+        while (!empty($argv)) {
+            $opt = array_shift($argv);
+
+            if ($opt == '--') {
+                $storage->addArg($opt);
+                return $storage->passArgs($argv);
+            }
+
+            if ($this->isAliasOrSiteSpec($opt) && !$storage->hasAlias() && !$sawArg) {
+                $storage->setAlias($opt);
+                continue;
+            }
+
+            if ($opt[0] != '-') {
+                $sawArg = true;
+            }
+
+            list($methodName, $value) = $this->findMethodForOptionWithValues($optionsTable, $opt);
+            if ($methodName) {
+                if (!isset($value)) {
+                    $value = array_shift($argv);
+                }
+                $method = [$storage, $methodName];
+                call_user_func($method, $value);
+            } else {
+                $storage->addArg($opt);
+            }
+        }
+        return $storage;
+    }
+
+    /**
+     * Determine whether the provided argument is an alias or
+     * a site specification.
+     *
+     * @param string $arg
+     *   Argument to test.
+     * @return bool
+     */
+    protected function isAliasOrSiteSpec($arg)
+    {
+        if (SiteAliasName::isAliasName($arg)) {
+            return true;
+        }
+        return $this->specParser->validSiteSpec($arg);
+    }
+
+    /**
+     * Check to see if '$opt' is one of the options that we record
+     * that takes a value.
+     *
+     * @param $optionsTable Table of option names and the name of the
+     *   method that should be called to process that option.
+     * @param $opt The option string to check
+     * @return [$methodName, $optionValue]
+     */
+    protected function findMethodForOptionWithValues($optionsTable, $opt)
+    {
+        // Skip $opt if it is empty, or if it is not an option.
+        if (empty($opt) || ($opt[0] != '-')) {
+            return [false, false];
+        }
+
+        // Check each entry in the option table in turn; return as soon
+        // as there is a match.
+        foreach ($optionsTable as $key => $methodName) {
+            $result = $this->checkMatchingOption($opt, $key, $methodName);
+            if ($result[0]) {
+                return $result;
+            }
+        }
+
+        return [false, false];
+    }
+
+    /**
+     * Check to see if the provided option matches the entry from the
+     * option table.
+     *
+     * @param $opt The option string to check
+     * @param $key The key to test against. Must always start with '-' or
+     *   '--'.  If $key ends with '=', then the option must have a value.
+     *   Otherwise, it cannot be supplied with a value, and always defaults
+     *   to 'true'.
+     * @return [$methodName, $optionValue]
+     */
+    protected function checkMatchingOption($opt, $keyParam, $methodName)
+    {
+        // Test to see if $key ends in '='; remove the character if present.
+        // If the char is removed, it means the option has a value.
+        $key = rtrim($keyParam, '=');
+        $hasValue = $key != $keyParam;
+
+        // If $opt does not begin with $key, then it cannot be a match.
+        if ($key != substr($opt, 0, strlen($key))) {
+            return [false, false];
+        }
+
+        // If $key and $opt are exact matches, then return a positive result.
+        // The returned $optionValue will be 'null' if the option requires
+        // a value; in this case, the value will be provided from the next
+        // argument in the calling function. If this option does not take a
+        // supplied value, then we set its value to 'true'
+        if (strlen($key) == strlen($opt)) {
+            return [$methodName, $hasValue ? null: true];
+        }
+
+        // If the option is not an exact match for the key, then the next
+        // character in the option after the key name must be an '='. Otherwise,
+        // we might confuse `--locale` for `--local`, etc.
+        if ($opt[strlen($key)] != '=') {
+            return [false, false];
+        }
+
+        // If $opt does not take a value, then we will ignore
+        // of the form --opt=value
+        if (!$hasValue) {
+            // TODO: We could fail with "The "--foo" option does not accept a value." here.
+            // It is important that we ignore the value for '--backend', but other options could throw.
+            // For now, we just ignore the value if it is there. This only affects --simulate and --local at the moment.
+            return [$methodName, true];
+        }
+
+        // If $opt is a double-dash option, and it contains an '=', then
+        // the option value is everything after the '='.
+        if ((strlen($key) < strlen($opt)) && ($opt[1] == '-') && ($opt[strlen($key)] == '=')) {
+            $value = substr($opt, strlen($key) + 1);
+            return [$methodName, $value];
+        }
+
+        return [false, false];
+    }
+}