1
0
Fork 0
mirror of https://we.phorge.it/source/phorge.git synced 2025-01-11 07:11:04 +01:00

Synchronize (hosted, clustered, Git) repositories over Conduit + HTTP

Summary:
Ref T4292. We currently synchronize hosted, clustered, Git repositories when we receive an SSH pull or push.

Additionally:

  - Synchronize before HTTP reads and writes.
  - Synchronize reads before Conduit requests.

We could relax Conduit eventually and allow Diffusion to say "it's OK to give me stale data".

We could also redirect some set of these actions to just go to the up-to-date host instead of connecting to a random host and synchronizing it. However, this potentially won't work as well at scale: if you have a larger number of servers, it sends all of the traffic to the leader immediately following a write. That can cause "thundering herd" issues, and isn't efficient if replicas are in different geographical regions and the write just went to the east coast but most clients are on the west coast. In large-scale cases, it's better to go to the local replica, wait for an update, then serve traffic from it -- particularly given that writes are relatively rare. But we can finesse this later once things are solid.

Test Plan:
  - Pushed and pulled a Git repository over HTTP.
  - Browsed a Git repository from the web UI.

Reviewers: chad

Reviewed By: chad

Maniphest Tasks: T4292

Differential Revision: https://secure.phabricator.com/D15758
This commit is contained in:
epriestley 2016-04-19 08:03:12 -07:00
parent 31bc023eff
commit d87c500002
4 changed files with 68 additions and 9 deletions

View file

@ -145,6 +145,12 @@ abstract class DiffusionQueryConduitAPIMethod
$this->setDiffusionRequest($drequest); $this->setDiffusionRequest($drequest);
// TODO: Allow web UI queries opt out of this if they don't care about
// fetching the most up-to-date data? Synchronization can be slow, and a
// lot of web reads are probably fine if they're a few seconds out of
// date.
$repository->synchronizeWorkingCopyBeforeRead();
return $this->getResult($request); return $this->getResult($request);
} }
} }

View file

@ -538,10 +538,35 @@ final class DiffusionServeController extends DiffusionController {
$command = csprintf('%s', $bin); $command = csprintf('%s', $bin);
$command = PhabricatorDaemon::sudoCommandAsDaemonUser($command); $command = PhabricatorDaemon::sudoCommandAsDaemonUser($command);
list($err, $stdout, $stderr) = id(new ExecFuture('%C', $command)) $unguarded = AphrontWriteGuard::beginScopedUnguardedWrites();
->setEnv($env, true)
->write($input) $did_write_lock = false;
->resolve(); if ($this->isReadOnlyRequest($repository)) {
$repository->synchronizeWorkingCopyBeforeRead();
} else {
$did_write_lock = true;
$repository->synchronizeWorkingCopyBeforeWrite($viewer);
}
$caught = null;
try {
list($err, $stdout, $stderr) = id(new ExecFuture('%C', $command))
->setEnv($env, true)
->write($input)
->resolve();
} catch (Exception $ex) {
$caught = $ex;
}
if ($did_write_lock) {
$repository->synchronizeWorkingCopyAfterWrite();
}
unset($unguarded);
if ($caught) {
throw $caught;
}
if ($err) { if ($err) {
if ($this->isValidGitShallowCloneResponse($stdout, $stderr)) { if ($this->isValidGitShallowCloneResponse($stdout, $stderr)) {

View file

@ -8,6 +8,7 @@ abstract class DiffusionCommandEngine extends Phobject {
private $argv; private $argv;
private $passthru; private $passthru;
private $connectAsDevice; private $connectAsDevice;
private $sudoAsDaemon;
public static function newCommandEngine(PhabricatorRepository $repository) { public static function newCommandEngine(PhabricatorRepository $repository) {
$engines = self::newCommandEngines(); $engines = self::newCommandEngines();
@ -92,10 +93,25 @@ abstract class DiffusionCommandEngine extends Phobject {
return $this->connectAsDevice; return $this->connectAsDevice;
} }
public function setSudoAsDaemon($sudo_as_daemon) {
$this->sudoAsDaemon = $sudo_as_daemon;
return $this;
}
public function getSudoAsDaemon() {
return $this->sudoAsDaemon;
}
public function newFuture() { public function newFuture() {
$argv = $this->newCommandArgv(); $argv = $this->newCommandArgv();
$env = $this->newCommandEnvironment(); $env = $this->newCommandEnvironment();
if ($this->getSudoAsDaemon()) {
$command = call_user_func_array('csprintf', $argv);
$command = PhabricatorDaemon::sudoCommandAsDaemonUser($command);
$argv = array('%C', $command);
}
if ($this->getPassthru()) { if ($this->getPassthru()) {
$future = newv('PhutilExecPassthru', $argv); $future = newv('PhutilExecPassthru', $argv);
} else { } else {

View file

@ -1954,6 +1954,10 @@ final class PhabricatorRepository extends PhabricatorRepositoryDAO
$uris = array(); $uris = array();
foreach ($bindings as $binding) { foreach ($bindings as $binding) {
if ($binding->getIsDisabled()) {
continue;
}
$iface = $binding->getInterface(); $iface = $binding->getInterface();
// If we're never proxying this and it's locally satisfiable, return // If we're never proxying this and it's locally satisfiable, return
@ -2197,11 +2201,6 @@ final class PhabricatorRepository extends PhabricatorRepositoryDAO
private function shouldEnableSynchronization() { private function shouldEnableSynchronization() {
$device = AlmanacKeys::getLiveDevice();
if (!$device) {
return false;
}
$service_phid = $this->getAlmanacServicePHID(); $service_phid = $this->getAlmanacServicePHID();
if (!$service_phid) { if (!$service_phid) {
return false; return false;
@ -2212,6 +2211,18 @@ final class PhabricatorRepository extends PhabricatorRepositoryDAO
return false; return false;
} }
// TODO: It may eventually make sense to try to version and synchronize
// observed repositories (so that daemons don't do reads against out-of
// date hosts), but don't bother for now.
if (!$this->isHosted()) {
return false;
}
$device = AlmanacKeys::getLiveDevice();
if (!$device) {
return false;
}
return true; return true;
} }
@ -2451,6 +2462,7 @@ final class PhabricatorRepository extends PhabricatorRepositoryDAO
$future = DiffusionCommandEngine::newCommandEngine($this) $future = DiffusionCommandEngine::newCommandEngine($this)
->setArgv($argv) ->setArgv($argv)
->setConnectAsDevice(true) ->setConnectAsDevice(true)
->setSudoAsDaemon(true)
->setProtocol($fetch_uri->getProtocol()) ->setProtocol($fetch_uri->getProtocol())
->newFuture(); ->newFuture();