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=