mirror of
https://we.phorge.it/source/phorge.git
synced 2025-01-13 16:21:07 +01:00
Make repository daemons periodically check for out-of-sync repositories
Summary: See PHI1015. If you add new repository nodes to a cluster, we may not actually sync some repositories for up to 6 hours (if they've had no commits for 30 days). Add an explicit check for out-of-sync repositories to trigger background sync. Test Plan: - Ran `bin/phd debug pullocal`. - Fiddled with the `repository_workingcopy` version table to put the local node in and out of sync with the cluster. - Saw appropriate responses in the daemon (sync; wait if the last sync trigger was too recent). Reviewers: amckinley Reviewed By: amckinley Differential Revision: https://secure.phabricator.com/D20078
This commit is contained in:
parent
f3e154eb02
commit
7dd55a728f
2 changed files with 72 additions and 1 deletions
|
@ -73,6 +73,9 @@ final class PhabricatorRepositoryPullLocalDaemon
|
|||
$futures = array();
|
||||
$queue = array();
|
||||
|
||||
$sync_wait = phutil_units('2 minutes in seconds');
|
||||
$last_sync = array();
|
||||
|
||||
while (!$this->shouldExit()) {
|
||||
PhabricatorCaches::destroyRequestCache();
|
||||
$device = AlmanacKeys::getLiveDevice();
|
||||
|
@ -96,6 +99,37 @@ final class PhabricatorRepositoryPullLocalDaemon
|
|||
$retry_after[$message->getRepositoryID()] = time();
|
||||
}
|
||||
|
||||
if ($device) {
|
||||
$unsynchronized = $this->loadUnsynchronizedRepositories($device);
|
||||
$now = PhabricatorTime::getNow();
|
||||
foreach ($unsynchronized as $repository) {
|
||||
$id = $repository->getID();
|
||||
|
||||
$this->log(
|
||||
pht(
|
||||
'Cluster repository ("%s") is out of sync on this node ("%s").',
|
||||
$repository->getDisplayName(),
|
||||
$device->getName()));
|
||||
|
||||
// Don't let out-of-sync conditions trigger updates too frequently,
|
||||
// since we don't want to get trapped in a death spiral if sync is
|
||||
// failing.
|
||||
$sync_at = idx($last_sync, $id, 0);
|
||||
$wait_duration = ($now - $sync_at);
|
||||
if ($wait_duration < $sync_wait) {
|
||||
$this->log(
|
||||
pht(
|
||||
'Skipping forced out-of-sync update because the last update '.
|
||||
'was too recent (%s seconds ago).',
|
||||
$wait_duration));
|
||||
continue;
|
||||
}
|
||||
|
||||
$last_sync[$id] = $now;
|
||||
$retry_after[$id] = $now;
|
||||
}
|
||||
}
|
||||
|
||||
// If any repositories were deleted, remove them from the retry timer map
|
||||
// so we don't end up with a retry timer that never gets updated and
|
||||
// causes us to sleep for the minimum amount of time.
|
||||
|
@ -521,4 +555,41 @@ final class PhabricatorRepositoryPullLocalDaemon
|
|||
return false;
|
||||
}
|
||||
|
||||
private function loadUnsynchronizedRepositories(AlmanacDevice $device) {
|
||||
$viewer = $this->getViewer();
|
||||
$table = new PhabricatorRepositoryWorkingCopyVersion();
|
||||
$conn = $table->establishConnection('r');
|
||||
|
||||
$our_versions = queryfx_all(
|
||||
$conn,
|
||||
'SELECT repositoryPHID, repositoryVersion FROM %R WHERE devicePHID = %s',
|
||||
$table,
|
||||
$device->getPHID());
|
||||
$our_versions = ipull($our_versions, 'repositoryVersion', 'repositoryPHID');
|
||||
|
||||
$max_versions = queryfx_all(
|
||||
$conn,
|
||||
'SELECT repositoryPHID, MAX(repositoryVersion) maxVersion FROM %R
|
||||
GROUP BY repositoryPHID',
|
||||
$table);
|
||||
$max_versions = ipull($max_versions, 'maxVersion', 'repositoryPHID');
|
||||
|
||||
$unsynchronized_phids = array();
|
||||
foreach ($max_versions as $repository_phid => $max_version) {
|
||||
$our_version = idx($our_versions, $repository_phid);
|
||||
if (($our_version === null) || ($our_version < $max_version)) {
|
||||
$unsynchronized_phids[] = $repository_phid;
|
||||
}
|
||||
}
|
||||
|
||||
if (!$unsynchronized_phids) {
|
||||
return array();
|
||||
}
|
||||
|
||||
return id(new PhabricatorRepositoryQuery())
|
||||
->setViewer($viewer)
|
||||
->withPHIDs($unsynchronized_phids)
|
||||
->execute();
|
||||
}
|
||||
|
||||
}
|
||||
|
|
|
@ -7,7 +7,7 @@ abstract class PhabricatorRepositoryManagementWorkflow
|
|||
$identifiers = $args->getArg($param);
|
||||
|
||||
if (!$identifiers) {
|
||||
return null;
|
||||
return array();
|
||||
}
|
||||
|
||||
$query = id(new PhabricatorRepositoryQuery())
|
||||
|
|
Loading…
Reference in a new issue