diff --git a/README.md b/README.md index 1e6fd4d..b98bbbd 100644 --- a/README.md +++ b/README.md @@ -168,48 +168,13 @@ information: - `{{ logs }}` will inject the commit logs with before/after delimiters, so they can be updated later without destroying any other changes to the contents. -## Synchronising data to supported modules +## GitHub API ratelimit -Cow includes commands to help synchronise standardised data to all -[commercially supported modules](https://www.silverstripe.org/software/addons/silverstripe-commercially-supported-module-list/): - -* `cow github:synclabels` Pushes a centralised list of labels to all supported module GitHub repositories -* `cow github:ratelimit` Check your current GitHub API rate limiting status (sync commands can use this up quickly) +Run `cow github:ratelimit` to check your current GitHub API rate limiting status **Note:** All GitHub API commands require a `GITHUB_ACCESS_TOKEN` environment variable to be set before they can be used. It can be in the .env file (see [dev mode](#dev_mode)). -### Labels - -[Centralised label configuration](https://github.com/silverstripe/supported-modules/blob/gh-pages/labels.json) can be -pushed out to all [supported modules](https://github.com/silverstripe/supported-modules/blob/gh-pages/modules.json) -using the `cow github:synclabels` command. - -This command takes an optional argument to specify which module(s) to update: - -* `modules` Optionally sync to specific modules (comma delimited) - -If the `modules` argument is not provided, the list of supported modules will be loaded from the `supported-modules` -GitHub repository. You can then confirm the list, before you will be shown a list of the labels that will be sync'd -and finally all labels will either be created or updated on the target repositories. - -This command can max out your GitHub API rate limiting credits, so use it sparingly. If you exceed the limit you may -need to go and make a coffee and come back in an hour (check current rate limits with `cow github:ratelimit`). - -### Metadata files - -[File templates](https://github.com/silverstripe/supported-modules/tree/gh-pages/templates) for supported modules can -be synchronised out to all supported modules using the `module:sync:metadata` command. - -This command will pull the latest version from the supported-modules repository, write the contents to each repository -then stage, commit and push directly to the default branch. - -This command takes an optional argument to skip the clone/pull of each repository beforehand: - -* `--skip-update` Optionally skip the clone/fetch/pull for each repository before running the sync - -You will need `git` available in your system path, as well as write permission to push to each repository. - ## Schema The [cow schema file](cow.schema.json) is in the root of this project. diff --git a/src/Application.php b/src/Application.php index 2acc25b..8c58e4c 100644 --- a/src/Application.php +++ b/src/Application.php @@ -3,7 +3,6 @@ namespace SilverStripe\Cow; use SilverStripe\Cow\Commands; -use SilverStripe\Cow\Utility\SupportedModuleLoader; use SilverStripe\Cow\Utility\Config; use SilverStripe\Cow\Utility\GitHubApi; use SilverStripe\Cow\Utility\Twig; @@ -92,7 +91,6 @@ protected function getDefaultCommands(): array // Create dependencies $githubApi = new GitHubApi(); - $supportedModuleLoader = new SupportedModuleLoader(); // What is this cow doing in here, stop it, get out $commands[] = new Commands\MooCommand(); @@ -112,15 +110,11 @@ protected function getDefaultCommands(): array $commands[] = new Commands\Release\Release(); $commands[] = new Commands\Release\Publish(); - // Module commands - $commands[] = new Commands\Module\Sync\Metadata($supportedModuleLoader); - // Schema commands $commands[] = new Commands\Schema\Validate(); // GitHub commands $commands[] = new Commands\GitHub\RateLimit($githubApi); - $commands[] = new Commands\GitHub\SyncLabels($supportedModuleLoader, $githubApi); return $commands; } diff --git a/src/Commands/GitHub/SyncLabels.php b/src/Commands/GitHub/SyncLabels.php deleted file mode 100644 index fad833b..0000000 --- a/src/Commands/GitHub/SyncLabels.php +++ /dev/null @@ -1,226 +0,0 @@ -supportedModuleLoader = $supportedModuleLoader; - $this->github = $github; - } - - protected function configureOptions() - { - $this->addArgument( - 'modules', - InputArgument::IS_ARRAY, - 'Optionally sync to specific modules (space delimited)' - ); - } - - protected function fire() - { - $io = new SymfonyStyle($this->input, $this->output); - - $modules = $this->input->getArgument('modules'); - if (empty($modules)) { - // Loading data and confirming steps with user - $io->section('Loading supported modules'); - $modules = $this->supportedModuleLoader->getModules(); - } - - if (!$this->confirmModules($io, $modules)) { - return; - } - - $labelConfig = $this->loadLabelConfig(); - if (!$this->confirmLabels($io, $labelConfig)) { - return; - } - - // Proceed with synchronisation - $result = ['success' => 0, 'error' => 0]; - $current = 0; - $total = count($modules); - foreach ($modules as $githubSlug) { - $io->text(++$current . '/' . $total . ': Processing ' . $githubSlug . '...'); - - // Set the progress bar: total is the number of label operations to make - $io->progressStart(array_sum(array_map('count', $labelConfig))); - - try { - $this->syncLabelsToModule($githubSlug, $labelConfig, $io); - $result['success']++; - } catch (Exception $ex) { - $io->error('Error updating ' . $githubSlug . ': ' . $ex->getMessage()); - $result['error']++; - } - - $io->progressFinish(); - } - $result['total'] = array_sum($result); - - $io->success('Finished! ' . $result['success'] . '/' . $result['total'] . ' succeeded.'); - } - - /** - * Show a summary of the modules that are going to be processed and ask the user for confirmation before - * proceeding. - * - * @param SymfonyStyle $io - * @param array $modules - */ - protected function confirmModules(SymfonyStyle $io, array $modules) - { - // Show summary table - $rows = []; - foreach ($modules as $github) { - $rows[] = [$github]; - } - $io->table(['GitHub repo'], $rows); - - return $io->confirm('Continue with sync?', true); - } - - /** - * Show a summary of the labels that are going to be synchronised and ask for confirmation before proceeding - * - * @param SymfonyStyle $io - * @param array $labelData - */ - protected function confirmLabels(SymfonyStyle $io, array $labels) - { - $io->text('The following labels will be pushed to each repository:'); - // Default labels - $rows = []; - foreach ($labels['default_labels'] as $label => $hexCode) { - $rows[] = [$label, $hexCode]; - } - $io->table(['Label', 'Hex code'], $rows); - - // Renaming - $io->text('The following labels will be renamed:'); - foreach ($labels['rename_labels'] as $old => $new) { - $io->writeln(' * ' . $old . ' => ' . $new . ''); - } - $io->newLine(); - - // Removing - $io->text('The following labels will be deleted:'); - foreach ($labels['remove_labels'] as $label) { - $io->writeln(' * ' . $label . ''); - } - $io->newLine(); - - return $io->confirm('Continue with sync?', true); - } - - /** - * @return array - * @throws UnexpectedValueException - */ - protected function loadLabelConfig() - { - $labels = $this->supportedModuleLoader->getLabels(); - if (empty($labels)) { - throw new UnexpectedValueException('labels.json data could not be loaded!'); - } - return $labels; - } - - /** - * Given a GitHub repository slug and an array of labels, synchronise them to GitHub - * - * @param string $githubSlug - * @param array $labelConfig - * @param SymfonyStyle $io - */ - protected function syncLabelsToModule($githubSlug, array $labelConfig, SymfonyStyle $io) - { - /** @var LabelsApi $labelsApi */ - $labelsApi = $this->github->getClient()->api('issue')->labels(); - - list ($organisation, $repository) = explode('/', $githubSlug); - - // Rename labels - foreach ($labelConfig['rename_labels'] as $oldLabel => $newLabel) { - $io->progressAdvance(); - - // Assign white as a placeholder, it'll be updated further down - try { - $labelsApi->update($organisation, $repository, $oldLabel, $newLabel, 'FFFFFF'); - } catch (Exception $ex) { - if (strpos($ex->getMessage(), 'Not Found') === false) { - // Only log messages that aren't "Not Found", which come when the labels don't exist - $io->error($ex->getMessage()); - } - } - } - - // (Create and) update colours on labels - foreach ($labelConfig['default_labels'] as $label => $hexCode) { - $io->progressAdvance(); - - try { - $exists = $labelsApi->show($organisation, $repository, $label); - } catch (Exception $ex) { - $exists = false; - } - - if ($exists) { - // Existing, update - $labelsApi->update($organisation, $repository, $label, $label, $hexCode); - continue; - } - // Create new label - $labelsApi->create($organisation, $repository, [ - 'name' => $label, - 'color' => $hexCode, - ]); - } - - // Delete labels - foreach ($labelConfig['remove_labels'] as $label) { - $io->progressAdvance(); - - try { - $labelsApi->deleteLabel($organisation, $repository, $label); - } catch (Exception $ex) { - if (strpos($ex->getMessage(), 'Not Found') === false) { - // Only log messages that aren't "Not Found", which come when the labels don't exist - $io->error($ex->getMessage()); - } - } - } - } -} diff --git a/src/Commands/Module/AbstractSyncCommand.php b/src/Commands/Module/AbstractSyncCommand.php deleted file mode 100644 index 1f06ac8..0000000 --- a/src/Commands/Module/AbstractSyncCommand.php +++ /dev/null @@ -1,163 +0,0 @@ -setSupportedModuleLoader($supportedModuleLoader); - } - - /** - * Either clones or pulls a shallow cloned copy of each of the supported modules - */ - protected function syncRepositories() - { - $repositories = $this->getSupportedModuleLoader()->getModules(); - $baseDir = $this->getBaseDir(); - $this->output->writeln('Temporary directory: ' . $baseDir . ''); - - foreach ($repositories as $repository) { - $repositoryHash = sha1($repository); - - // Temporary: only process SilverStripe repos - // @todo should this be part of the module configuration? e.g. "sync_files": true - if (substr($repository, 0, 12) !== 'silverstripe') { - $this->output->writeln('Skipping ' . $repository . ''); - continue; - } - - if ($this->tempFolderExists($baseDir . $repositoryHash)) { - $this->output->writeln('Updating ' . $repository . '...'); - $this->updateRepository($repository); - } else { - $this->output->writeln('Cloning ' . $repository . '...'); - $this->cloneRepository($repository); - } - } - - $this->output->writeln('All repositories updated.'); - } - - /** - * Returns the base directory for storing Git repositories. Will create it if it doesn't exist yet. - * - * @return string - */ - protected function getBaseDir() - { - $baseDir = __DIR__ . '/../../../temp/'; - if (!is_dir($baseDir)) { - mkdir($baseDir); - } - return realpath($baseDir) . '/'; - } - - /** - * @param string $folder - * @return bool - */ - protected function tempFolderExists($folder) - { - return is_dir($folder); - } - - /** - * Updates an existing Git repository - * - * @param string $repository - */ - protected function updateRepository($repository) - { - $process = new Process([ - '/usr/bin/env', - 'git', - 'pull', - ]); - $process->setWorkingDirectory($this->getRepositoryPath($repository)); - - $this->getHelper('process')->run($this->output, $process); - } - - /** - * Clones a Git repository - * - * @param string $repository - */ - protected function cloneRepository($repository) - { - $process = new Process([ - '/usr/bin/env', - 'git', - 'clone', - '--depth', - '1', - $this->getRepositoryUrl($repository), - $this->getRepositoryPath($repository), - ]); - - $this->getHelper('process')->run($this->output, $process); - } - - /** - * @param string $repository - * @return string - */ - protected function getRepositoryPath($repository) - { - return $this->getBaseDir() . $this->getRepositoryHash($repository); - } - - /** - * Returns the GitHub repository URL. Uses SSH protocol. - * - * @param string $repository - * @return string - */ - protected function getRepositoryUrl($repository) - { - return 'git@github.com:' . $repository; - } - - /** - * @param string $repository - * @return string - */ - protected function getRepositoryHash($repository) - { - return sha1($repository); - } - - /** - * @param SupportedModuleLoader $supportedModuleLoader - * @return $this - */ - public function setSupportedModuleLoader(SupportedModuleLoader $supportedModuleLoader) - { - $this->supportedModuleLoader = $supportedModuleLoader; - return $this; - } - - /** - * @return SupportedModuleLoader - */ - public function getSupportedModuleLoader() - { - return $this->supportedModuleLoader; - } -} diff --git a/src/Commands/Module/Sync/Metadata.php b/src/Commands/Module/Sync/Metadata.php deleted file mode 100644 index 1cc3d81..0000000 --- a/src/Commands/Module/Sync/Metadata.php +++ /dev/null @@ -1,157 +0,0 @@ -addOption( - 'skip-update', - null, - InputOption::VALUE_NONE, - 'Skip fetching latest repository changes' - ); - } - - protected function fire() - { - // Ensure that all repositories are available and up to date - if (!$this->input->getOption('skip-update')) { - $this->syncRepositories(); - } - $repositories = $this->getSupportedModuleLoader()->getModules(); - - /** @var QuestionHelper $questionHelper */ - $questionHelper = $this->getHelper('question'); - - foreach (self::SYNC_FILES as $filename) { - $data = $this->getSupportedModuleLoader()->getRemoteData($filename); - $baseFilename = basename($filename); - - // Confirm diff before proceeding - $this->output->writeln($data); - $question = new ConfirmationQuestion( - 'Confirm file contents for ' . $baseFilename . '? ' - ); - $confirm = $questionHelper->ask($this->input, $this->output, $question); - - if (!$confirm) { - $this->output->writeln('Skipping ' . $baseFilename . ''); - continue; - } - - foreach ($repositories as $repository) { - // @todo implement check e.g. "sync_files": true - if (substr($repository, 0, 12) !== 'silverstripe') { - continue; - } - $basePath = $this->getRepositoryPath($repository); - - $this->writeDataToFile($basePath . '/' . $baseFilename, $data); - $this->stageFile($basePath, $baseFilename); - if (!$this->hasChanges($basePath)) { - continue; - } - $this->commitChanges($basePath, $baseFilename); - $this->pushChanges($basePath); - } - } - - $this->output->writeln('Done'); - } - - /** - * Writes the given data to the given filename - * - * @param string $filename - * @param string $data - */ - protected function writeDataToFile($filename, $data) - { - file_put_contents($filename, $data); - } - - /** - * Adds the given filename to stage - * - * @param string $basePath - * @param string $filename - */ - protected function stageFile($basePath, $filename) - { - $process = new Process(['/usr/bin/env', 'git', 'add', $filename, strtolower($filename)]); - $process->setWorkingDirectory($basePath); - // We don't need to know if one of the two filenames didn't exist - $process->disableOutput(); - $this->getHelper('process')->run($this->output, $process); - } - - /** - * Returns whether the given path has any staged changes in it - * - * @param string $basePath - * @return bool - */ - protected function hasChanges($basePath) - { - $process = new Process(['/usr/bin/env', 'git', 'diff', '--staged']); - $process->setWorkingDirectory($basePath); - $process->run(); - $result = $process->getOutput(); - return trim($result) !== ''; - } - - /** - * Commit stages changes - * - * @param string $basePath - * @param string $filename - */ - protected function commitChanges($basePath, $filename) - { - $process = new Process(['/usr/bin/env', 'git', 'commit', '-m', 'Update ' . $filename]); - $process->setWorkingDirectory($basePath); - $this->getHelper('process')->run($this->output, $process); - } - - /** - * Pushes any new commits to origin - * - * @param string $basePath - */ - protected function pushChanges($basePath) - { - if (Application::isDevMode()) { - echo "Not pushing changes because DEV_MODE is enabled\n"; - return; - } - $process = new Process(['/usr/bin/env', 'git', 'push']); - $process->setWorkingDirectory($basePath); - $this->getHelper('process')->run($this->output, $process); - } -} diff --git a/src/Utility/SupportedModuleLoader.php b/src/Utility/SupportedModuleLoader.php deleted file mode 100644 index 4f12847..0000000 --- a/src/Utility/SupportedModuleLoader.php +++ /dev/null @@ -1,106 +0,0 @@ -getRemoteData('modules.json'); - $modules = json_decode($data, true) ?: []; - - if ($this->getFilter()) { - $modules = $this->getFilter()->filter($modules); - } - - return array_column($modules, 'github'); - } - - /** - * Get the supported module labels configuration data - * - * @return array - */ - public function getLabels() - { - $data = $this->getRemoteData('labels.json'); - return json_decode($data, true) ?: []; - } - - /** - * Returns the full URL to a filename on the supported modules repository - * - * @param string $filename - * @return string - */ - public function getFilePath($filename) - { - return $this->getBaseUrl() . ltrim($filename, '/'); - } - - /** - * Gets data from the remote supported-modules repository by filename - * - * @param string $filename - * @return string - */ - public function getRemoteData($filename) - { - $data = file_get_contents($this->getFilePath($filename)); - return $data ?: ''; - } - - /** - * @param string $baseUrl - * @return $this - */ - public function setBaseUrl($baseUrl) - { - $this->baseUrl = $baseUrl; - return $this; - } - - /** - * @return string - */ - public function getBaseUrl() - { - return $this->baseUrl; - } - - /** - * @param FilterInterface $filter - * @return $this - */ - public function setFilter(FilterInterface $filter) - { - $this->filter = $filter; - return $this; - } - - /** - * @return FilterInterface - */ - public function getFilter() - { - return $this->filter; - } -} diff --git a/tests/Commands/Module/Sync/MetadataTest.php b/tests/Commands/Module/Sync/MetadataTest.php deleted file mode 100644 index d3fe75f..0000000 --- a/tests/Commands/Module/Sync/MetadataTest.php +++ /dev/null @@ -1,131 +0,0 @@ -moduleLoader = $this->getMockBuilder(SupportedModuleLoader::class) - ->setMethods(['getModules', 'getRemoteData']) - ->getMock(); - - $this->moduleLoader->expects($this->once())->method('getModules')->willReturn([ - 'silverstripe/foo', - 'someoneelse/foo', - ]); - - $this->moduleLoader->expects($this->once()) - ->method('getRemoteData') - ->with('templates/LICENSE.md') - ->willReturn('foo'); - - $this->metadata = $this->getMockBuilder(Metadata::class) - ->setMethods([ - 'syncRepositories', - 'writeDataToFile', - 'stageFile', - 'hasChanges', - 'commitChanges', - 'pushChanges', - ]) - ->setConstructorArgs([$this->moduleLoader]) - ->getMock(); - } - - public function testSyncRepositoriesByDefault() - { - $this->metadata->expects($this->once())->method('syncRepositories'); - - $output = $this->executeCommand([], ['yes']); - $this->assertStringContainsString('Done', $output); - } - - public function testDoesNotSyncRepositoriesWithSkipUpdateOption() - { - $this->metadata->expects($this->never())->method('syncRepositories'); - - $output = $this->executeCommand(['--skip-update' => true], ['yes']); - $this->assertStringContainsString('Done', $output); - } - - public function testSkippingFiles() - { - $output = $this->executeCommand([], ['no']); - $this->assertStringContainsString('Skipping LICENSE.md', $output); - $this->assertStringContainsString('Done', $output); - } - - public function testSkipThirdPartyRepositories() - { - $this->metadata->expects($this->once())->method('writeDataToFile'); - - $output = $this->executeCommand([], ['yes']); - $this->assertStringContainsString('Done', $output); - } - - public function testApplyChanges() - { - $this->metadata->expects($this->once())->method('writeDataToFile')->with($this->anything(), 'foo'); - $this->metadata->expects($this->once())->method('stageFile'); - $this->metadata->expects($this->once())->method('hasChanges')->willReturn(true); - $this->metadata->expects($this->once())->method('commitChanges'); - $this->metadata->expects($this->once())->method('pushChanges'); - - $output = $this->executeCommand([], ['yes']); - $this->assertStringContainsString('Done', $output); - } - - public function testCommitAndPushIsSkippedWithoutChanges() - { - $this->metadata->expects($this->once())->method('writeDataToFile')->with($this->anything(), 'foo'); - $this->metadata->expects($this->once())->method('stageFile'); - $this->metadata->expects($this->once())->method('hasChanges')->willReturn(false); - $this->metadata->expects($this->never())->method('commitChanges'); - $this->metadata->expects($this->never())->method('pushChanges'); - - $output = $this->executeCommand([], ['yes']); - $this->assertStringContainsString('Done', $output); - } - - /** - * Wrapper for executing a command and returning its output - * - * @param array $extraArgs - * @param array $inputs - * @return string - */ - protected function executeCommand(array $extraArgs = [], array $inputs = []) - { - $application = new Application(); - $application->add($this->metadata); - - $command = $application->find('module:sync:metadata'); - $commandTester = new CommandTester($command); - - if (!empty($inputs)) { - $commandTester->setInputs($inputs); - } - - $commandTester->execute(array_merge(['command' => $command->getName()], $extraArgs)); - return $commandTester->getDisplay(); - } -} diff --git a/tests/Utility/SupportedModuleLoaderTest.php b/tests/Utility/SupportedModuleLoaderTest.php deleted file mode 100644 index e899097..0000000 --- a/tests/Utility/SupportedModuleLoaderTest.php +++ /dev/null @@ -1,101 +0,0 @@ -getMockBuilder(SupportedModuleLoader::class) - ->setMethods(['getRemoteData']) - ->getMock(); - - $loader->expects($this->once())->method('getRemoteData')->willReturn(<<getModules(); - $this->assertContains('silverstripe/silverstripe-framework', $result, 'Supported modules are returned'); - $this->assertContains('silverstripe/silverstripe-fulltextsearch', $result, 'Supported modules are returned'); - } - - public function testGetLabels() - { - /** @var SupportedModuleLoader|PHPUnit_Framework_MockObject_MockObject $loader */ - $loader = $this->getMockBuilder(SupportedModuleLoader::class) - ->setMethods(['getRemoteData']) - ->getMock(); - - $loader->expects($this->once())->method('getRemoteData')->willReturn(<<getLabels(); - $this->assertNotEmpty($result['default_labels']); - $this->assertSame('ff0000', $result['default_labels']['affects/v3']); - - $this->assertNotEmpty($result['rename_labels']); - $this->assertSame('type/bug', $result['rename_labels']['bug']); - - $this->assertNotEmpty($result['remove_labels']); - $this->assertSame('good first issue', reset($result['remove_labels'])); - } - - public function testBrokenApiResponses() - { - /** @var SupportedModuleLoader|PHPUnit_Framework_MockObject_MockObject $loader */ - $loader = $this->getMockBuilder(SupportedModuleLoader::class) - ->setMethods(['getRemoteData']) - ->getMock(); - - $loader->expects($this->exactly(2))->method('getRemoteData')->willReturn(false); - - $this->assertSame([], $loader->getModules(), 'Broken HTTP response still returns an empty array'); - $this->assertSame([], $loader->getLabels(), 'Broken HTTP response still returns an empty array'); - } - - public function testGetFilePath() - { - $loader = new SupportedModuleLoader(); - - $this->assertSame( - 'https://raw.githubusercontent.com/silverstripe/supported-modules/gh-pages/modules.json', - $loader->getFilePath('modules.json') - ); - $this->assertSame( - 'https://raw.githubusercontent.com/silverstripe/supported-modules/gh-pages/labels.json', - $loader->getFilePath('/labels.json') - ); - } -}