Make "bin/files" parsing of working set arguments more consistent
Summary: Fixes T13326. In D20571, I slightly generalized construction of an iterator over a set of files, but missed some code in other "bin/files ..." commands which was also affected. Today, basically all of these workflows define their own "--all" and "names" flags. Pull these definitions up and implement them more consistently. Test Plan: Ran multiple different `bin/files` commands with different combinations of arguments, saw consistent handling of iterator construction. Reviewers: amckinley Reviewed By: amckinley Maniphest Tasks: T13326 Differential Revision: https://secure.phabricator.com/D20614
This commit is contained in:
		| @@ -4,41 +4,25 @@ final class PhabricatorFilesManagementCompactWorkflow | |||||||
|   extends PhabricatorFilesManagementWorkflow { |   extends PhabricatorFilesManagementWorkflow { | ||||||
|  |  | ||||||
|   protected function didConstruct() { |   protected function didConstruct() { | ||||||
|  |     $arguments = $this->newIteratorArguments(); | ||||||
|  |     $arguments[] = array( | ||||||
|  |       'name' => 'dry-run', | ||||||
|  |       'help' => pht('Show what would be compacted.'), | ||||||
|  |     ); | ||||||
|  |  | ||||||
|     $this |     $this | ||||||
|       ->setName('compact') |       ->setName('compact') | ||||||
|       ->setSynopsis( |       ->setSynopsis( | ||||||
|         pht( |         pht( | ||||||
|           'Merge identical files to share the same storage. In some cases, '. |           'Merge identical files to share the same storage. In some cases, '. | ||||||
|           'this can repair files with missing data.')) |           'this can repair files with missing data.')) | ||||||
|       ->setArguments( |       ->setArguments($arguments); | ||||||
|         array( |  | ||||||
|           array( |  | ||||||
|             'name'      => 'dry-run', |  | ||||||
|             'help'      => pht('Show what would be compacted.'), |  | ||||||
|           ), |  | ||||||
|           array( |  | ||||||
|             'name'      => 'all', |  | ||||||
|             'help'      => pht('Compact all files.'), |  | ||||||
|           ), |  | ||||||
|           array( |  | ||||||
|             'name'      => 'names', |  | ||||||
|             'wildcard'  => true, |  | ||||||
|           ), |  | ||||||
|         )); |  | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   public function execute(PhutilArgumentParser $args) { |   public function execute(PhutilArgumentParser $args) { | ||||||
|     $console = PhutilConsole::getConsole(); |     $console = PhutilConsole::getConsole(); | ||||||
|  |  | ||||||
|     $iterator = $this->buildIterator($args); |     $iterator = $this->buildIterator($args); | ||||||
|     if (!$iterator) { |  | ||||||
|       throw new PhutilArgumentUsageException( |  | ||||||
|         pht( |  | ||||||
|           'Either specify a list of files to compact, or use `%s` '. |  | ||||||
|           'to compact all files.', |  | ||||||
|           '--all')); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     $is_dry_run = $args->getArg('dry-run'); |     $is_dry_run = $args->getArg('dry-run'); | ||||||
|  |  | ||||||
|     foreach ($iterator as $file) { |     foreach ($iterator as $file) { | ||||||
|   | |||||||
| @@ -4,36 +4,22 @@ final class PhabricatorFilesManagementCycleWorkflow | |||||||
|   extends PhabricatorFilesManagementWorkflow { |   extends PhabricatorFilesManagementWorkflow { | ||||||
|  |  | ||||||
|   protected function didConstruct() { |   protected function didConstruct() { | ||||||
|  |     $arguments = $this->newIteratorArguments(); | ||||||
|  |     $arguments[] = array( | ||||||
|  |       'name' => 'key', | ||||||
|  |       'param' => 'keyname', | ||||||
|  |       'help' => pht('Select a specific storage key to cycle to.'), | ||||||
|  |     ); | ||||||
|  |  | ||||||
|     $this |     $this | ||||||
|       ->setName('cycle') |       ->setName('cycle') | ||||||
|       ->setSynopsis( |       ->setSynopsis( | ||||||
|         pht('Cycle master key for encrypted files.')) |         pht('Cycle master key for encrypted files.')) | ||||||
|       ->setArguments( |       ->setArguments($arguments); | ||||||
|         array( |  | ||||||
|           array( |  | ||||||
|             'name' => 'key', |  | ||||||
|             'param' => 'keyname', |  | ||||||
|             'help' => pht('Select a specific storage key to cycle to.'), |  | ||||||
|           ), |  | ||||||
|           array( |  | ||||||
|             'name' => 'all', |  | ||||||
|             'help' => pht('Change encoding for all files.'), |  | ||||||
|           ), |  | ||||||
|           array( |  | ||||||
|             'name' => 'names', |  | ||||||
|             'wildcard' => true, |  | ||||||
|           ), |  | ||||||
|         )); |  | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   public function execute(PhutilArgumentParser $args) { |   public function execute(PhutilArgumentParser $args) { | ||||||
|     $iterator = $this->buildIterator($args); |     $iterator = $this->buildIterator($args); | ||||||
|     if (!$iterator) { |  | ||||||
|       throw new PhutilArgumentUsageException( |  | ||||||
|         pht( |  | ||||||
|           'Either specify a list of files to cycle, or use --all to cycle '. |  | ||||||
|           'all files.')); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     $format_map = PhabricatorFileStorageFormat::getAllFormats(); |     $format_map = PhabricatorFileStorageFormat::getAllFormats(); | ||||||
|     $engines = PhabricatorFileStorageEngine::loadAllEngines(); |     $engines = PhabricatorFileStorageEngine::loadAllEngines(); | ||||||
|   | |||||||
| @@ -4,47 +4,36 @@ final class PhabricatorFilesManagementEncodeWorkflow | |||||||
|   extends PhabricatorFilesManagementWorkflow { |   extends PhabricatorFilesManagementWorkflow { | ||||||
|  |  | ||||||
|   protected function didConstruct() { |   protected function didConstruct() { | ||||||
|     $this |     $arguments = $this->newIteratorArguments(); | ||||||
|       ->setName('encode') |  | ||||||
|       ->setSynopsis( |     $arguments[] = array( | ||||||
|         pht('Change the storage encoding of files.')) |  | ||||||
|       ->setArguments( |  | ||||||
|         array( |  | ||||||
|           array( |  | ||||||
|       'name' => 'as', |       'name' => 'as', | ||||||
|       'param' => 'format', |       'param' => 'format', | ||||||
|       'help' => pht('Select the storage format to use.'), |       'help' => pht('Select the storage format to use.'), | ||||||
|           ), |     ); | ||||||
|           array( |  | ||||||
|  |     $arguments[] = array( | ||||||
|       'name' => 'key', |       'name' => 'key', | ||||||
|       'param' => 'keyname', |       'param' => 'keyname', | ||||||
|       'help' => pht('Select a specific storage key.'), |       'help' => pht('Select a specific storage key.'), | ||||||
|           ), |     ); | ||||||
|           array( |  | ||||||
|             'name' => 'all', |     $arguments[] = array( | ||||||
|             'help' => pht('Change encoding for all files.'), |  | ||||||
|           ), |  | ||||||
|           array( |  | ||||||
|       'name' => 'force', |       'name' => 'force', | ||||||
|       'help' => pht( |       'help' => pht( | ||||||
|         'Re-encode files which are already stored in the target '. |         'Re-encode files which are already stored in the target '. | ||||||
|         'encoding.'), |         'encoding.'), | ||||||
|           ), |     ); | ||||||
|           array( |  | ||||||
|             'name' => 'names', |     $this | ||||||
|             'wildcard' => true, |       ->setName('encode') | ||||||
|           ), |       ->setSynopsis( | ||||||
|         )); |         pht('Change the storage encoding of files.')) | ||||||
|  |       ->setArguments($arguments); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   public function execute(PhutilArgumentParser $args) { |   public function execute(PhutilArgumentParser $args) { | ||||||
|     $iterator = $this->buildIterator($args); |     $iterator = $this->buildIterator($args); | ||||||
|     if (!$iterator) { |  | ||||||
|       throw new PhutilArgumentUsageException( |  | ||||||
|         pht( |  | ||||||
|           'Either specify a list of files to encode, or use --all to '. |  | ||||||
|           'encode all files.')); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     $force = (bool)$args->getArg('force'); |     $force = (bool)$args->getArg('force'); | ||||||
|  |  | ||||||
|   | |||||||
| @@ -4,52 +4,50 @@ final class PhabricatorFilesManagementIntegrityWorkflow | |||||||
|   extends PhabricatorFilesManagementWorkflow { |   extends PhabricatorFilesManagementWorkflow { | ||||||
|  |  | ||||||
|   protected function didConstruct() { |   protected function didConstruct() { | ||||||
|     $this |     $arguments = $this->newIteratorArguments(); | ||||||
|       ->setName('integrity') |  | ||||||
|       ->setSynopsis(pht('Verify or recalculate file integrity hashes.')) |     $arguments[] = array( | ||||||
|       ->setArguments( |  | ||||||
|         array( |  | ||||||
|           array( |  | ||||||
|             'name'      => 'all', |  | ||||||
|             'help'      => pht('Affect all files.'), |  | ||||||
|           ), |  | ||||||
|           array( |  | ||||||
|       'name' => 'strip', |       'name' => 'strip', | ||||||
|       'help' => pht( |       'help' => pht( | ||||||
|         'DANGEROUS. Strip integrity hashes from files. This makes '. |         'DANGEROUS. Strip integrity hashes from files. This makes '. | ||||||
|         'files vulnerable to corruption or tampering.'), |         'files vulnerable to corruption or tampering.'), | ||||||
|           ), |     ); | ||||||
|           array( |  | ||||||
|  |     $arguments[] = array( | ||||||
|       'name' => 'corrupt', |       'name' => 'corrupt', | ||||||
|       'help' => pht( |       'help' => pht( | ||||||
|         'Corrupt integrity hashes for given files. This is intended '. |         'Corrupt integrity hashes for given files. This is intended '. | ||||||
|         'for debugging.'), |         'for debugging.'), | ||||||
|           ), |     ); | ||||||
|           array( |  | ||||||
|  |     $arguments[] = array( | ||||||
|       'name' => 'compute', |       'name' => 'compute', | ||||||
|       'help' => pht( |       'help' => pht( | ||||||
|         'Compute and update integrity hashes for files which do not '. |         'Compute and update integrity hashes for files which do not '. | ||||||
|         'yet have them.'), |         'yet have them.'), | ||||||
|           ), |     ); | ||||||
|           array( |  | ||||||
|  |     $arguments[] = array( | ||||||
|       'name' => 'overwrite', |       'name' => 'overwrite', | ||||||
|       'help' => pht( |       'help' => pht( | ||||||
|         'DANGEROUS. Recompute and update integrity hashes, overwriting '. |         'DANGEROUS. Recompute and update integrity hashes, overwriting '. | ||||||
|         'invalid hashes. This may mark corrupt or dangerous files as '. |         'invalid hashes. This may mark corrupt or dangerous files as '. | ||||||
|         'valid.'), |         'valid.'), | ||||||
|           ), |     ); | ||||||
|           array( |  | ||||||
|  |     $arguments[] = array( | ||||||
|       'name' => 'force', |       'name' => 'force', | ||||||
|       'short' => 'f', |       'short' => 'f', | ||||||
|       'help' => pht( |       'help' => pht( | ||||||
|         'Execute dangerous operations without prompting for '. |         'Execute dangerous operations without prompting for '. | ||||||
|         'confirmation.'), |         'confirmation.'), | ||||||
|           ), |     ); | ||||||
|           array( |  | ||||||
|             'name'      => 'names', |  | ||||||
|             'wildcard'  => true, |     $this | ||||||
|           ), |       ->setName('integrity') | ||||||
|         )); |       ->setSynopsis(pht('Verify or recalculate file integrity hashes.')) | ||||||
|  |       ->setArguments($arguments); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   public function execute(PhutilArgumentParser $args) { |   public function execute(PhutilArgumentParser $args) { | ||||||
| @@ -120,12 +118,6 @@ final class PhabricatorFilesManagementIntegrityWorkflow | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     $iterator = $this->buildIterator($args); |     $iterator = $this->buildIterator($args); | ||||||
|     if (!$iterator) { |  | ||||||
|       throw new PhutilArgumentUsageException( |  | ||||||
|         pht( |  | ||||||
|           'Either specify a list of files to affect, or use "--all" to '. |  | ||||||
|           'affect all files.')); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     $failure_count = 0; |     $failure_count = 0; | ||||||
|     $total_count = 0; |     $total_count = 0; | ||||||
|   | |||||||
| @@ -4,61 +4,54 @@ final class PhabricatorFilesManagementMigrateWorkflow | |||||||
|   extends PhabricatorFilesManagementWorkflow { |   extends PhabricatorFilesManagementWorkflow { | ||||||
|  |  | ||||||
|   protected function didConstruct() { |   protected function didConstruct() { | ||||||
|     $this |     $arguments = $this->newIteratorArguments(); | ||||||
|       ->setName('migrate') |  | ||||||
|       ->setSynopsis(pht('Migrate files between storage engines.')) |     $arguments[] = array( | ||||||
|       ->setArguments( |  | ||||||
|         array( |  | ||||||
|           array( |  | ||||||
|       'name' => 'engine', |       'name' => 'engine', | ||||||
|             'param'     => 'storage_engine', |       'param' => 'storage-engine', | ||||||
|       'help' => pht('Migrate to the named storage engine.'), |       'help' => pht('Migrate to the named storage engine.'), | ||||||
|           ), |     ); | ||||||
|           array( |  | ||||||
|  |     $arguments[] = array( | ||||||
|       'name' => 'dry-run', |       'name' => 'dry-run', | ||||||
|       'help' => pht('Show what would be migrated.'), |       'help' => pht('Show what would be migrated.'), | ||||||
|           ), |     ); | ||||||
|           array( |  | ||||||
|  |     $arguments[] = array( | ||||||
|       'name' => 'min-size', |       'name' => 'min-size', | ||||||
|       'param' => 'bytes', |       'param' => 'bytes', | ||||||
|       'help' => pht( |       'help' => pht( | ||||||
|         'Do not migrate data for files which are smaller than a given '. |         'Do not migrate data for files which are smaller than a given '. | ||||||
|         'filesize.'), |         'filesize.'), | ||||||
|           ), |     ); | ||||||
|           array( |  | ||||||
|  |     $arguments[] = array( | ||||||
|       'name' => 'max-size', |       'name' => 'max-size', | ||||||
|       'param' => 'bytes', |       'param' => 'bytes', | ||||||
|       'help' => pht( |       'help' => pht( | ||||||
|         'Do not migrate data for files which are larger than a given '. |         'Do not migrate data for files which are larger than a given '. | ||||||
|         'filesize.'), |         'filesize.'), | ||||||
|           ), |     ); | ||||||
|           array( |  | ||||||
|             'name'      => 'all', |     $arguments[] = array( | ||||||
|             'help'      => pht('Migrate all files.'), |  | ||||||
|           ), |  | ||||||
|           array( |  | ||||||
|       'name' => 'copy', |       'name' => 'copy', | ||||||
|       'help' => pht( |       'help' => pht( | ||||||
|         'Copy file data instead of moving it: after migrating, do not '. |         'Copy file data instead of moving it: after migrating, do not '. | ||||||
|         'remove the old data even if it is no longer referenced.'), |         'remove the old data even if it is no longer referenced.'), | ||||||
|           ), |     ); | ||||||
|           array( |  | ||||||
|             'name'      => 'names', |     $arguments[] = array( | ||||||
|             'wildcard'  => true, |  | ||||||
|           ), |  | ||||||
|           array( |  | ||||||
|             'name' => 'from-engine', |  | ||||||
|             'param' => 'engine', |  | ||||||
|             'help' => pht('Migrate files from the named storage engine.'), |  | ||||||
|           ), |  | ||||||
|           array( |  | ||||||
|       'name' => 'local-disk-source', |       'name' => 'local-disk-source', | ||||||
|       'param' => 'path', |       'param' => 'path', | ||||||
|       'help' => pht( |       'help' => pht( | ||||||
|         'When migrating from a local disk source, use the specified '. |         'When migrating from a local disk source, use the specified '. | ||||||
|         'path as the root directory.'), |         'path as the root directory.'), | ||||||
|           ), |     ); | ||||||
|         )); |  | ||||||
|  |     $this | ||||||
|  |       ->setName('migrate') | ||||||
|  |       ->setSynopsis(pht('Migrate files between storage engines.')) | ||||||
|  |       ->setArguments($arguments); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   public function execute(PhutilArgumentParser $args) { |   public function execute(PhutilArgumentParser $args) { | ||||||
| @@ -97,14 +90,6 @@ final class PhabricatorFilesManagementMigrateWorkflow | |||||||
|     $target_engine = PhabricatorFile::buildEngine($target_key); |     $target_engine = PhabricatorFile::buildEngine($target_key); | ||||||
|  |  | ||||||
|     $iterator = $this->buildIterator($args); |     $iterator = $this->buildIterator($args); | ||||||
|     if (!$iterator) { |  | ||||||
|       throw new PhutilArgumentUsageException( |  | ||||||
|         pht( |  | ||||||
|           'Either specify a list of files to migrate, or use `%s` '. |  | ||||||
|           'to migrate all files.', |  | ||||||
|           '--all')); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     $is_dry_run = $args->getArg('dry-run'); |     $is_dry_run = $args->getArg('dry-run'); | ||||||
|  |  | ||||||
|     $min_size = (int)$args->getArg('min-size'); |     $min_size = (int)$args->getArg('min-size'); | ||||||
|   | |||||||
| @@ -4,45 +4,33 @@ final class PhabricatorFilesManagementRebuildWorkflow | |||||||
|   extends PhabricatorFilesManagementWorkflow { |   extends PhabricatorFilesManagementWorkflow { | ||||||
|  |  | ||||||
|   protected function didConstruct() { |   protected function didConstruct() { | ||||||
|  |     $arguments = $this->newIteratorArguments(); | ||||||
|  |  | ||||||
|  |     $arguments[] = array( | ||||||
|  |       'name' => 'dry-run', | ||||||
|  |       'help' => pht('Show what would be updated.'), | ||||||
|  |     ); | ||||||
|  |  | ||||||
|  |     $arguments[] = array( | ||||||
|  |       'name' => 'rebuild-mime', | ||||||
|  |       'help' => pht('Rebuild MIME information.'), | ||||||
|  |     ); | ||||||
|  |  | ||||||
|  |     $arguments[] = array( | ||||||
|  |       'name' => 'rebuild-dimensions', | ||||||
|  |       'help' => pht('Rebuild image dimension information.'), | ||||||
|  |     ); | ||||||
|  |  | ||||||
|     $this |     $this | ||||||
|       ->setName('rebuild') |       ->setName('rebuild') | ||||||
|       ->setSynopsis(pht('Rebuild metadata of old files.')) |       ->setSynopsis(pht('Rebuild metadata of old files.')) | ||||||
|       ->setArguments( |       ->setArguments($arguments); | ||||||
|         array( |  | ||||||
|           array( |  | ||||||
|             'name'      => 'all', |  | ||||||
|             'help'      => pht('Update all files.'), |  | ||||||
|           ), |  | ||||||
|           array( |  | ||||||
|             'name'      => 'dry-run', |  | ||||||
|             'help'      => pht('Show what would be updated.'), |  | ||||||
|           ), |  | ||||||
|           array( |  | ||||||
|             'name'      => 'rebuild-mime', |  | ||||||
|             'help'      => pht('Rebuild MIME information.'), |  | ||||||
|           ), |  | ||||||
|           array( |  | ||||||
|             'name'      => 'rebuild-dimensions', |  | ||||||
|             'help'      => pht('Rebuild image dimension information.'), |  | ||||||
|           ), |  | ||||||
|           array( |  | ||||||
|             'name'      => 'names', |  | ||||||
|             'wildcard'  => true, |  | ||||||
|           ), |  | ||||||
|         )); |  | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   public function execute(PhutilArgumentParser $args) { |   public function execute(PhutilArgumentParser $args) { | ||||||
|     $console = PhutilConsole::getConsole(); |     $console = PhutilConsole::getConsole(); | ||||||
|  |  | ||||||
|     $iterator = $this->buildIterator($args); |     $iterator = $this->buildIterator($args); | ||||||
|     if (!$iterator) { |  | ||||||
|       throw new PhutilArgumentUsageException( |  | ||||||
|         pht( |  | ||||||
|           'Either specify a list of files to update, or use `%s` '. |  | ||||||
|           'to update all files.', |  | ||||||
|           '--all')); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     $update = array( |     $update = array( | ||||||
|       'mime'          => $args->getArg('rebuild-mime'), |       'mime'          => $args->getArg('rebuild-mime'), | ||||||
|   | |||||||
| @@ -3,11 +3,30 @@ | |||||||
| abstract class PhabricatorFilesManagementWorkflow | abstract class PhabricatorFilesManagementWorkflow | ||||||
|   extends PhabricatorManagementWorkflow { |   extends PhabricatorManagementWorkflow { | ||||||
|  |  | ||||||
|  |   protected function newIteratorArguments() { | ||||||
|  |     return array( | ||||||
|  |       array( | ||||||
|  |         'name' => 'all', | ||||||
|  |         'help' => pht('Operate on all files.'), | ||||||
|  |       ), | ||||||
|  |       array( | ||||||
|  |         'name' => 'names', | ||||||
|  |         'wildcard' => true, | ||||||
|  |       ), | ||||||
|  |       array( | ||||||
|  |         'name' => 'from-engine', | ||||||
|  |         'param' => 'storage-engine', | ||||||
|  |         'help' => pht('Operate on files stored in a specified engine.'), | ||||||
|  |       ), | ||||||
|  |     ); | ||||||
|  |   } | ||||||
|  |  | ||||||
|   protected function buildIterator(PhutilArgumentParser $args) { |   protected function buildIterator(PhutilArgumentParser $args) { | ||||||
|     $viewer = $this->getViewer(); |     $viewer = $this->getViewer(); | ||||||
|     $names = $args->getArg('names'); |  | ||||||
|  |  | ||||||
|     $is_all = $args->getArg('all'); |     $is_all = $args->getArg('all'); | ||||||
|  |  | ||||||
|  |     $names = $args->getArg('names'); | ||||||
|     $from_engine = $args->getArg('from-engine'); |     $from_engine = $args->getArg('from-engine'); | ||||||
|  |  | ||||||
|     $any_constraint = ($from_engine || $names); |     $any_constraint = ($from_engine || $names); | ||||||
| @@ -15,15 +34,16 @@ abstract class PhabricatorFilesManagementWorkflow | |||||||
|     if (!$is_all && !$any_constraint) { |     if (!$is_all && !$any_constraint) { | ||||||
|       throw new PhutilArgumentUsageException( |       throw new PhutilArgumentUsageException( | ||||||
|         pht( |         pht( | ||||||
|           'Use "--all" to migrate all files, or choose files to migrate '. |           'Specify which files to operate on, or use "--all" to operate on '. | ||||||
|           'with "--names" or "--from-engine".')); |           'all files.')); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     if ($is_all && $any_constraint) { |     if ($is_all && $any_constraint) { | ||||||
|       throw new PhutilArgumentUsageException( |       throw new PhutilArgumentUsageException( | ||||||
|         pht( |         pht( | ||||||
|           'You can not migrate all files with "--all" and also migrate only '. |           'You can not operate on all files with "--all" and also operate '. | ||||||
|           'a subset of files with "--from-engine" or "--names".')); |           'on a subset of files by naming them explicitly or using '. | ||||||
|  |           'constraint flags like "--from-engine".')); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     // If we're migrating specific named files, convert the names into IDs |     // If we're migrating specific named files, convert the names into IDs | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user
	 epriestley
					epriestley