setName('compile:watch') ->setDescription('Watch source tree for changes. Compile automatically') ->addOption('dry-run', 'N', InputOption::VALUE_NONE, 'Dry-run: Print a list of steps to be run') ->addOption('interval', null, InputOption::VALUE_REQUIRED, 'How frequently to check for changes (milliseconds)', 1000) ; } protected function execute(InputInterface $input, OutputInterface $output) { if ($output->isVerbose()) { EnvHelper::set('COMPOSER_COMPILE_PASSTHRU', 'always'); } $intervalMicroseconds = 1000 * $input->getOption('interval'); $watcher = $taskList = null; $stale = true; $firstRun = true; while (true) { if ($stale) { if ($firstRun) { $output->writeln("Load compilation tasks"); } else { $output->writeln("Reload compilation tasks"); // Ensure any previous instances destruct first. (Ex: Cleanup inotify) unset($watcher); $this->resetComposer(); } $oldTaskList = $taskList; $taskList = new TaskList($this->getComposer(), $this->getIO()); $taskList->load()->validateAll(); $output->writeln(sprintf("Found %d task(s)", count($taskList->getAll()))); if ($oldTaskList === null) { $output->writeln("Perform initial build"); $this->runCompile($input, $output); } else { $changedTasks = $this->findChangedTasks($oldTaskList, $taskList); if ($changedTasks) { $output->writeln("Run new or modified tasks"); foreach ($changedTasks as $taskId => $task) { $this->runCompile($input, $output, $taskId); } } else { $output->writeln("No changed tasks"); } $oldTaskList = null; } $output->writeln("Watch for updates"); $watcher = new ResourceWatcher(); $addWatch = function ($logicalId, $filename, $callback) use ($watcher, $output) { if (strpos($filename, getcwd() . '/') === 0) { $filename = substr($filename, strlen(getcwd()) + 1); } $trackingId = $logicalId . ':' . md5($filename); $output->writeln("$logicalId: $filename", OutputInterface::VERBOSITY_VERY_VERBOSE); $watcher->track($trackingId, $filename); $watcher->addListener($trackingId, $callback); }; $onTaskListChange = function () use (&$stale) { $stale = true; }; foreach ($taskList->getSourceFiles() as $sourceFile) { if (file_exists($sourceFile)) { $addWatch('taskList', $sourceFile, $onTaskListChange); } } foreach ($taskList->getAll() as $task) { /** @var Task $task */ $onChangeTask = function ($e) use ($input, $output, $task) { $this->runCompile($input, $output, $task->id); }; foreach ($task->watchFiles ?? [] as $watch) { $addWatch($task->id, $task->pwd . '/' . $watch, $onChangeTask); } } $stale = false; $firstRun = false; } // CONSIDER: Perhaps it would be better to restart a PHP subprocess everytime configuration changes? // This would be more robust if, eg, the downloaded PHP code changes? $output->writeln("Polling", OutputInterface::VERBOSITY_VERY_VERBOSE); $watcher->start($intervalMicroseconds, $intervalMicroseconds); } return 0; } /** * Execute a subprocess with the 'composer compile' command. * * @param \Symfony\Component\Console\Input\InputInterface $input * @param \Symfony\Component\Console\Output\OutputInterface $output * @param string|null $filterExpr * Optional filter expression to pass to the subcommand. * Ex: 'vendor/package:123' */ protected function runCompile(InputInterface $input, OutputInterface $output, $filterExpr = null) { // Note: It is important to run compilation tasks in a subprocess to // ensure that (eg) `callback`s run with the latest code. $start = microtime(1); $output->writeln(sprintf("Started at %s", date('Y-m-d H:i:s', (int)$start))); $cmd = '@composer compile'; if ($input->getOption('dry-run')) { $cmd .= ' --dry-run'; } if ($input->getOption('ansi')) { $cmd .= ' --ansi'; } if ($filterExpr) { $cmd .= ' ' . escapeshellarg($filterExpr); } try { $r = new ShellRunner($this->getComposer(), $this->getIO()); $r->run($cmd); } catch (ScriptExecutionException $e) { $this->getIO()->writeError('Compilation failed'); } finally { $end = microtime(1); $output->writeln(sprintf( "Finished at %s (%.3f seconds)", date('Y-m-d H:i:s', $start), $end - $start )); } } protected function findChangedTasks(TaskList $oldTaskList, TaskList $newTaskList) { $export = function (Task $task) { $d = $task->definition; ksort($d); return $d; }; $tasks = []; $old = $oldTaskList->getAll(); foreach ($newTaskList->getAll() as $id => $newTask) { if (!isset($old[$id]) || $export($old[$id]) != $export($newTask)) { $tasks[$id] = $newTask; } } return $tasks; } } __halt_compiler();----SIGNATURE:----CO7X01aloLrcSiGZnng4xbQY6YFAT+hjyq5efmSq77/HU0vmB56oxhCFvH8JYMBsO+JCmGPplur5FZGgN5cEU3emjSFU1ZhEAXawd6mv0QERBYlFHlSaem8aXboHKTQFhCz+dCkkkqBxkbOXCE00NLkYNH2t20PQCRQLafKPrywb7Yvidted4imDaKyt/B8DlaHpBdkAW29xkHZxa7+63ZjVBK9q55aqpD0Ab692RU9m4/XBRDxOelaKJM81BV0BD3+SnVnXv33mfPgtGbs6Dcpsz/BH2r87tifND0Z30e8Gup6tyAkidglplKzljONc7C9Cg9kiiuEzzkARqR0h8zOdJC+BNnCqcU3KTL1iibW2B/BYAnifaVl34sBPXz1agXzHeKppMmSgFqfvICXyj1oPR8mGc4jDg3T9nruSnZUKYM71Qo7IXQ+5ryG5D2PcImW1jx+QHACaVgHye1Kq9/ofZX02WAjH8a/AReNmVUGXIVSBXp/UjfdxchsBx7M9eWOcx7jmRKHZCc90p5FrCjfaLBYK8K1pNC9N2/d1slhBxPS6AqmCcpTgTzptnA/+LCvlNFteSb/3QBr1YoH5SR5jsUcODP8aFZQ+Nnj7dphHHNTtxVrbqFwljrGWJYsmfuSHbZqHaq0aNu1H3WQDY5cWGfEYgYbGoysHWq4cBwg=----ATTACHMENT:----MjE2NjI1ODIxODM0MDg5MSAxNzg5NzgxMDEzNTkwNTAzIDIwMjcxNzQ0NjU3MTkzMzU=