mirror of
https://we.phorge.it/source/phorge.git
synced 2025-01-24 21:48:21 +01:00
Make repository pulls testable
Summary: Ref T2784. This moves us toward being able to test the background and Conduit pipelines for repositories. In particular: - Separate the logic for pulling repositories (`git pull`, `hg pull`) out of `PhabricatorRepositoryPullLocalDaemon` and put it in `PhabricatorRepositoryPullEngine`. This allows repositories to be pulled directly without invoking the daemons. - Add tests for the engine, including a future-looking base test case. - Add basic `PhutilDirectoryFixture`-based repositories. Next steps: # Do the same for repo discovery. # Then we can start writing tests against specific Conduit methods. Test Plan: Ran unit tests. Ran `bin/repository pull` on SVN, Hg and Git repositories. Ran `bin/phd debug pulllocal`. Reviewers: btrahan Reviewed By: btrahan CC: aran, nh Maniphest Tasks: T2784 Differential Revision: https://secure.phabricator.com/D5904
This commit is contained in:
parent
98e2ad4ebc
commit
105dd1899c
9 changed files with 405 additions and 225 deletions
|
@ -1300,6 +1300,7 @@ phutil_register_library_map(array(
|
||||||
'PhabricatorRepositoryManagementWorkflow' => 'applications/repository/management/PhabricatorRepositoryManagementWorkflow.php',
|
'PhabricatorRepositoryManagementWorkflow' => 'applications/repository/management/PhabricatorRepositoryManagementWorkflow.php',
|
||||||
'PhabricatorRepositoryMercurialCommitChangeParserWorker' => 'applications/repository/worker/commitchangeparser/PhabricatorRepositoryMercurialCommitChangeParserWorker.php',
|
'PhabricatorRepositoryMercurialCommitChangeParserWorker' => 'applications/repository/worker/commitchangeparser/PhabricatorRepositoryMercurialCommitChangeParserWorker.php',
|
||||||
'PhabricatorRepositoryMercurialCommitMessageParserWorker' => 'applications/repository/worker/commitmessageparser/PhabricatorRepositoryMercurialCommitMessageParserWorker.php',
|
'PhabricatorRepositoryMercurialCommitMessageParserWorker' => 'applications/repository/worker/commitmessageparser/PhabricatorRepositoryMercurialCommitMessageParserWorker.php',
|
||||||
|
'PhabricatorRepositoryPullEngine' => 'applications/repository/engine/PhabricatorRepositoryPullEngine.php',
|
||||||
'PhabricatorRepositoryPullLocalDaemon' => 'applications/repository/daemon/PhabricatorRepositoryPullLocalDaemon.php',
|
'PhabricatorRepositoryPullLocalDaemon' => 'applications/repository/daemon/PhabricatorRepositoryPullLocalDaemon.php',
|
||||||
'PhabricatorRepositoryPullLocalDaemonTestCase' => 'applications/repository/daemon/__tests__/PhabricatorRepositoryPullLocalDaemonTestCase.php',
|
'PhabricatorRepositoryPullLocalDaemonTestCase' => 'applications/repository/daemon/__tests__/PhabricatorRepositoryPullLocalDaemonTestCase.php',
|
||||||
'PhabricatorRepositoryQuery' => 'applications/repository/query/PhabricatorRepositoryQuery.php',
|
'PhabricatorRepositoryQuery' => 'applications/repository/query/PhabricatorRepositoryQuery.php',
|
||||||
|
@ -1487,6 +1488,8 @@ phutil_register_library_map(array(
|
||||||
'PhabricatorWorkerTaskDetailController' => 'applications/daemon/controller/PhabricatorWorkerTaskDetailController.php',
|
'PhabricatorWorkerTaskDetailController' => 'applications/daemon/controller/PhabricatorWorkerTaskDetailController.php',
|
||||||
'PhabricatorWorkerTaskUpdateController' => 'applications/daemon/controller/PhabricatorWorkerTaskUpdateController.php',
|
'PhabricatorWorkerTaskUpdateController' => 'applications/daemon/controller/PhabricatorWorkerTaskUpdateController.php',
|
||||||
'PhabricatorWorkerTestCase' => 'infrastructure/daemon/workers/__tests__/PhabricatorWorkerTestCase.php',
|
'PhabricatorWorkerTestCase' => 'infrastructure/daemon/workers/__tests__/PhabricatorWorkerTestCase.php',
|
||||||
|
'PhabricatorWorkingCopyPullTestCase' => 'applications/repository/engine/__tests__/PhabricatorWorkingCopyPullTestCase.php',
|
||||||
|
'PhabricatorWorkingCopyTestCase' => 'applications/repository/engine/__tests__/PhabricatorWorkingCopyTestCase.php',
|
||||||
'PhabricatorWorkpanelView' => 'view/layout/PhabricatorWorkpanelView.php',
|
'PhabricatorWorkpanelView' => 'view/layout/PhabricatorWorkpanelView.php',
|
||||||
'PhabricatorXHPASTViewController' => 'applications/phpast/controller/PhabricatorXHPASTViewController.php',
|
'PhabricatorXHPASTViewController' => 'applications/phpast/controller/PhabricatorXHPASTViewController.php',
|
||||||
'PhabricatorXHPASTViewDAO' => 'applications/phpast/storage/PhabricatorXHPASTViewDAO.php',
|
'PhabricatorXHPASTViewDAO' => 'applications/phpast/storage/PhabricatorXHPASTViewDAO.php',
|
||||||
|
@ -3194,6 +3197,8 @@ phutil_register_library_map(array(
|
||||||
'PhabricatorWorkerTaskDetailController' => 'PhabricatorDaemonController',
|
'PhabricatorWorkerTaskDetailController' => 'PhabricatorDaemonController',
|
||||||
'PhabricatorWorkerTaskUpdateController' => 'PhabricatorDaemonController',
|
'PhabricatorWorkerTaskUpdateController' => 'PhabricatorDaemonController',
|
||||||
'PhabricatorWorkerTestCase' => 'PhabricatorTestCase',
|
'PhabricatorWorkerTestCase' => 'PhabricatorTestCase',
|
||||||
|
'PhabricatorWorkingCopyPullTestCase' => 'PhabricatorWorkingCopyTestCase',
|
||||||
|
'PhabricatorWorkingCopyTestCase' => 'PhabricatorTestCase',
|
||||||
'PhabricatorWorkpanelView' => 'AphrontView',
|
'PhabricatorWorkpanelView' => 'AphrontView',
|
||||||
'PhabricatorXHPASTViewController' => 'PhabricatorController',
|
'PhabricatorXHPASTViewController' => 'PhabricatorController',
|
||||||
'PhabricatorXHPASTViewDAO' => 'PhabricatorLiskDAO',
|
'PhabricatorXHPASTViewDAO' => 'PhabricatorLiskDAO',
|
||||||
|
|
|
@ -120,7 +120,9 @@ final class PhabricatorRepositoryPullLocalDaemon
|
||||||
$callsign = $repository->getCallsign();
|
$callsign = $repository->getCallsign();
|
||||||
$this->log("Updating repository '{$callsign}'.");
|
$this->log("Updating repository '{$callsign}'.");
|
||||||
|
|
||||||
$this->pullRepository($repository);
|
id(new PhabricatorRepositoryPullEngine())
|
||||||
|
->setRepository($repository)
|
||||||
|
->pullRepository();
|
||||||
|
|
||||||
if (!$no_discovery) {
|
if (!$no_discovery) {
|
||||||
// TODO: It would be nice to discover only if we pulled something,
|
// TODO: It would be nice to discover only if we pulled something,
|
||||||
|
@ -175,54 +177,6 @@ final class PhabricatorRepositoryPullLocalDaemon
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @task pull
|
|
||||||
*/
|
|
||||||
public function pullRepository(PhabricatorRepository $repository) {
|
|
||||||
$vcs = $repository->getVersionControlSystem();
|
|
||||||
|
|
||||||
$is_svn = ($vcs == PhabricatorRepositoryType::REPOSITORY_TYPE_SVN);
|
|
||||||
$is_git = ($vcs == PhabricatorRepositoryType::REPOSITORY_TYPE_GIT);
|
|
||||||
$is_hg = ($vcs == PhabricatorRepositoryType::REPOSITORY_TYPE_MERCURIAL);
|
|
||||||
|
|
||||||
if ($is_svn) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
$callsign = $repository->getCallsign();
|
|
||||||
|
|
||||||
if (!$is_git && !$is_hg) {
|
|
||||||
throw new Exception(
|
|
||||||
"Unknown VCS '{$vcs}' for repository '{$callsign}'!");
|
|
||||||
}
|
|
||||||
|
|
||||||
$local_path = $repository->getDetail('local-path');
|
|
||||||
if (!$local_path) {
|
|
||||||
throw new Exception(
|
|
||||||
"No local path is available for repository '{$callsign}'.");
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!Filesystem::pathExists($local_path)) {
|
|
||||||
$dirname = dirname($local_path);
|
|
||||||
if (!Filesystem::pathExists($dirname)) {
|
|
||||||
Filesystem::createDirectory($dirname, 0755, $recursive = true);
|
|
||||||
}
|
|
||||||
|
|
||||||
if ($is_git) {
|
|
||||||
return $this->executeGitCreate($repository, $local_path);
|
|
||||||
} else if ($is_hg) {
|
|
||||||
return $this->executeHgCreate($repository, $local_path);
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
if ($is_git) {
|
|
||||||
return $this->executeGitUpdate($repository, $local_path);
|
|
||||||
} else if ($is_hg) {
|
|
||||||
return $this->executeHgUpdate($repository, $local_path);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public function discoverRepository(PhabricatorRepository $repository) {
|
public function discoverRepository(PhabricatorRepository $repository) {
|
||||||
$vcs = $repository->getVersionControlSystem();
|
$vcs = $repository->getVersionControlSystem();
|
||||||
switch ($vcs) {
|
switch ($vcs) {
|
||||||
|
@ -468,121 +422,6 @@ final class PhabricatorRepositoryPullLocalDaemon
|
||||||
/* -( Git Implementation )------------------------------------------------- */
|
/* -( Git Implementation )------------------------------------------------- */
|
||||||
|
|
||||||
|
|
||||||
private function canWrite($path) {
|
|
||||||
$default_path =
|
|
||||||
PhabricatorEnv::getEnvConfig('repository.default-local-path');
|
|
||||||
return Filesystem::isDescendant($path, $default_path);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @task git
|
|
||||||
*/
|
|
||||||
private function executeGitCreate(
|
|
||||||
PhabricatorRepository $repository,
|
|
||||||
$path) {
|
|
||||||
|
|
||||||
$repository->execxRemoteCommand(
|
|
||||||
'clone --origin origin %s %s',
|
|
||||||
$repository->getRemoteURI(),
|
|
||||||
rtrim($path, '/'));
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @task git
|
|
||||||
*/
|
|
||||||
private function executeGitUpdate(
|
|
||||||
PhabricatorRepository $repository,
|
|
||||||
$path) {
|
|
||||||
|
|
||||||
// Run a bunch of sanity checks to detect people checking out repositories
|
|
||||||
// inside other repositories, making empty directories, pointing the local
|
|
||||||
// path at some random file or path, etc.
|
|
||||||
|
|
||||||
list($err, $stdout) = $repository->execLocalCommand(
|
|
||||||
'rev-parse --show-toplevel');
|
|
||||||
$msg = '';
|
|
||||||
|
|
||||||
if ($err) {
|
|
||||||
|
|
||||||
// Try to raise a more tailored error message in the more common case
|
|
||||||
// of the user creating an empty directory. (We could try to remove it,
|
|
||||||
// but might not be able to, and it's much simpler to raise a good
|
|
||||||
// message than try to navigate those waters.)
|
|
||||||
if (is_dir($path)) {
|
|
||||||
$files = Filesystem::listDirectory($path, $include_hidden = true);
|
|
||||||
if (!$files) {
|
|
||||||
$msg =
|
|
||||||
"Expected to find a git repository at '{$path}', but there ".
|
|
||||||
"is an empty directory there. Remove the directory: the daemon ".
|
|
||||||
"will run 'git clone' for you.";
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
$msg =
|
|
||||||
"Expected to find a git repository at '{$path}', but there is ".
|
|
||||||
"a non-repository directory (with other stuff in it) there. Move or ".
|
|
||||||
"remove this directory (or reconfigure the repository to use a ".
|
|
||||||
"different directory), and then either clone a repository yourself ".
|
|
||||||
"or let the daemon do it.";
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
$repo_path = rtrim($stdout, "\n");
|
|
||||||
|
|
||||||
if (empty($repo_path)) {
|
|
||||||
$err = true;
|
|
||||||
$msg =
|
|
||||||
"Expected to find a git repository at '{$path}', but ".
|
|
||||||
"there was no result from `git rev-parse --show-toplevel`. ".
|
|
||||||
"Something is misconfigured or broken. The git repository ".
|
|
||||||
"may be inside a '.git/' directory.";
|
|
||||||
} else if (!Filesystem::pathsAreEquivalent($repo_path, $path)) {
|
|
||||||
$err = true;
|
|
||||||
$msg =
|
|
||||||
"Expected to find repo at '{$path}', but the actual ".
|
|
||||||
"git repository root for this directory is '{$repo_path}'. ".
|
|
||||||
"Something is misconfigured. The repository's 'Local Path' should ".
|
|
||||||
"be set to some place where the daemon can check out a working ".
|
|
||||||
"copy, and should not be inside another git repository.";
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if ($err && $this->canWrite($path)) {
|
|
||||||
phlog("{$path} failed sanity check; recloning. ({$msg})");
|
|
||||||
Filesystem::remove($path);
|
|
||||||
$this->executeGitCreate($repository, $path);
|
|
||||||
} else if ($err) {
|
|
||||||
throw new Exception($msg);
|
|
||||||
}
|
|
||||||
|
|
||||||
$retry = false;
|
|
||||||
do {
|
|
||||||
// This is a local command, but needs credentials.
|
|
||||||
$future = $repository->getRemoteCommandFuture('fetch --all --prune');
|
|
||||||
$future->setCWD($path);
|
|
||||||
list($err, $stdout, $stderr) = $future->resolve();
|
|
||||||
|
|
||||||
if ($err && !$retry && $this->canWrite($path)) {
|
|
||||||
$retry = true;
|
|
||||||
// Fix remote origin url if it doesn't match our configuration
|
|
||||||
$origin_url =
|
|
||||||
$repository->execLocalCommand('config --get remote.origin.url');
|
|
||||||
$remote_uri = $repository->getDetail('remote-uri');
|
|
||||||
if ($origin_url != $remote_uri) {
|
|
||||||
$repository->execLocalCommand('remote set-url origin %s',
|
|
||||||
$remote_uri);
|
|
||||||
}
|
|
||||||
} else if ($err) {
|
|
||||||
throw new Exception(
|
|
||||||
"git fetch failed with error #{$err}:\n".
|
|
||||||
"stdout:{$stdout}\n\n".
|
|
||||||
"stderr:{$stderr}\n");
|
|
||||||
} else {
|
|
||||||
$retry = false;
|
|
||||||
}
|
|
||||||
} while ($retry);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @task git
|
* @task git
|
||||||
*/
|
*/
|
||||||
|
@ -787,60 +626,6 @@ final class PhabricatorRepositoryPullLocalDaemon
|
||||||
/* -( Mercurial Implementation )------------------------------------------- */
|
/* -( Mercurial Implementation )------------------------------------------- */
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @task hg
|
|
||||||
*/
|
|
||||||
private function executeHgCreate(
|
|
||||||
PhabricatorRepository $repository,
|
|
||||||
$path) {
|
|
||||||
|
|
||||||
$repository->execxRemoteCommand(
|
|
||||||
'clone %s %s',
|
|
||||||
$repository->getRemoteURI(),
|
|
||||||
rtrim($path, '/'));
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @task hg
|
|
||||||
*/
|
|
||||||
private function executeHgUpdate(
|
|
||||||
PhabricatorRepository $repository,
|
|
||||||
$path) {
|
|
||||||
|
|
||||||
// This is a local command, but needs credentials.
|
|
||||||
$future = $repository->getRemoteCommandFuture('pull -u');
|
|
||||||
$future->setCWD($path);
|
|
||||||
|
|
||||||
try {
|
|
||||||
$future->resolvex();
|
|
||||||
} catch (CommandException $ex) {
|
|
||||||
$err = $ex->getError();
|
|
||||||
$stdout = $ex->getStdOut();
|
|
||||||
|
|
||||||
// NOTE: Between versions 2.1 and 2.1.1, Mercurial changed the behavior
|
|
||||||
// of "hg pull" to return 1 in case of a successful pull with no changes.
|
|
||||||
// This behavior has been reverted, but users who updated between Feb 1,
|
|
||||||
// 2012 and Mar 1, 2012 will have the erroring version. Do a dumb test
|
|
||||||
// against stdout to check for this possibility.
|
|
||||||
// See: https://github.com/facebook/phabricator/issues/101/
|
|
||||||
|
|
||||||
// NOTE: Mercurial has translated versions, which translate this error
|
|
||||||
// string. In a translated version, the string will be something else,
|
|
||||||
// like "aucun changement trouve". There didn't seem to be an easy way
|
|
||||||
// to handle this (there are hard ways but this is not a common problem
|
|
||||||
// and only creates log spam, not application failures). Assume English.
|
|
||||||
|
|
||||||
// TODO: Remove this once we're far enough in the future that deployment
|
|
||||||
// of 2.1 is exceedingly rare?
|
|
||||||
if ($err == 1 && preg_match('/no changes found/', $stdout)) {
|
|
||||||
return;
|
|
||||||
} else {
|
|
||||||
throw $ex;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private function executeHgDiscover(PhabricatorRepository $repository) {
|
private function executeHgDiscover(PhabricatorRepository $repository) {
|
||||||
// NOTE: "--debug" gives us 40-character hashes.
|
// NOTE: "--debug" gives us 40-character hashes.
|
||||||
list($stdout) = $repository->execxLocalCommand('--debug branches');
|
list($stdout) = $repository->execxLocalCommand('--debug branches');
|
||||||
|
|
|
@ -0,0 +1,267 @@
|
||||||
|
<?php
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Manages execution of `git pull` and `hg pull` commands for
|
||||||
|
* @{class:PhabricatorRepository} objects. Used by
|
||||||
|
* @{class:PhabricatorRepositoryPullLocalDaemon}.
|
||||||
|
*
|
||||||
|
* @task pull Pulling Working Copies
|
||||||
|
* @task git Pulling Git Working Copies
|
||||||
|
* @task hg Pulling Mercurial Working Copies
|
||||||
|
* @task internal Internals
|
||||||
|
*/
|
||||||
|
final class PhabricatorRepositoryPullEngine {
|
||||||
|
|
||||||
|
private $repository;
|
||||||
|
|
||||||
|
|
||||||
|
/* -( Pulling Working Copies )--------------------------------------------- */
|
||||||
|
|
||||||
|
|
||||||
|
public function setRepository(PhabricatorRepository $repository) {
|
||||||
|
$this->repository = $repository;
|
||||||
|
return $this;
|
||||||
|
}
|
||||||
|
|
||||||
|
private function getRepository() {
|
||||||
|
return $this->repository;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function pullRepository() {
|
||||||
|
$repository = $this->getRepository();
|
||||||
|
|
||||||
|
if (!$repository) {
|
||||||
|
throw new Exception("Call setRepository() before pullRepository()!");
|
||||||
|
}
|
||||||
|
|
||||||
|
$is_hg = false;
|
||||||
|
$is_git = false;
|
||||||
|
|
||||||
|
$vcs = $repository->getVersionControlSystem();
|
||||||
|
switch ($vcs) {
|
||||||
|
case PhabricatorRepositoryType::REPOSITORY_TYPE_SVN:
|
||||||
|
// We never pull a local copy of Subversion repositories.
|
||||||
|
return;
|
||||||
|
case PhabricatorRepositoryType::REPOSITORY_TYPE_GIT:
|
||||||
|
$is_git = true;
|
||||||
|
break;
|
||||||
|
case PhabricatorRepositoryType::REPOSITORY_TYPE_MERCURIAL:
|
||||||
|
$is_hg = true;
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
throw new Exception("Unsupported VCS '{$vcs}'!");
|
||||||
|
}
|
||||||
|
|
||||||
|
$callsign = $repository->getCallsign();
|
||||||
|
$local_path = $repository->getLocalPath();
|
||||||
|
if ($local_path === null) {
|
||||||
|
throw new Exception(
|
||||||
|
"No local path is configured for repository '{$callsign}'.");
|
||||||
|
}
|
||||||
|
|
||||||
|
$dirname = dirname($local_path);
|
||||||
|
if (!Filesystem::pathExists($dirname)) {
|
||||||
|
Filesystem::createDirectory($dirname, 0755, $recursive = true);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!Filesystem::pathExists($local_path)) {
|
||||||
|
if ($is_git) {
|
||||||
|
$this->executeGitCreate();
|
||||||
|
} else {
|
||||||
|
$this->executeMercurialCreate();
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if ($is_git) {
|
||||||
|
$this->executeGitUpdate();
|
||||||
|
} else {
|
||||||
|
$this->executeMercurialUpdate();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return $this;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/* -( Pulling Git Working Copies )----------------------------------------- */
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @task git
|
||||||
|
*/
|
||||||
|
private function executeGitCreate() {
|
||||||
|
$repository = $this->getRepository();
|
||||||
|
|
||||||
|
$repository->execxRemoteCommand(
|
||||||
|
'clone --origin origin %s %s',
|
||||||
|
$repository->getRemoteURI(),
|
||||||
|
rtrim($repository->getLocalPath(), '/'));
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @task git
|
||||||
|
*/
|
||||||
|
private function executeGitUpdate() {
|
||||||
|
$repository = $this->getRepository();
|
||||||
|
|
||||||
|
list($err, $stdout) = $repository->execLocalCommand(
|
||||||
|
'rev-parse --show-toplevel');
|
||||||
|
|
||||||
|
$message = null;
|
||||||
|
$path = $repository->getLocalPath();
|
||||||
|
if ($err) {
|
||||||
|
// Try to raise a more tailored error message in the more common case
|
||||||
|
// of the user creating an empty directory. (We could try to remove it,
|
||||||
|
// but might not be able to, and it's much simpler to raise a good
|
||||||
|
// message than try to navigate those waters.)
|
||||||
|
if (is_dir($path)) {
|
||||||
|
$files = Filesystem::listDirectory($path, $include_hidden = true);
|
||||||
|
if (!$files) {
|
||||||
|
$message =
|
||||||
|
"Expected to find a git repository at '{$path}', but there ".
|
||||||
|
"is an empty directory there. Remove the directory: the daemon ".
|
||||||
|
"will run 'git clone' for you.";
|
||||||
|
} else {
|
||||||
|
$message =
|
||||||
|
"Expected to find a git repository at '{$path}', but there is ".
|
||||||
|
"a non-repository directory (with other stuff in it) there. Move ".
|
||||||
|
"or remove this directory (or reconfigure the repository to use a ".
|
||||||
|
"different directory), and then either clone a repository ".
|
||||||
|
"yourself or let the daemon do it.";
|
||||||
|
}
|
||||||
|
} else if (is_file($path)) {
|
||||||
|
$message =
|
||||||
|
"Expected to find a git repository at '{$path}', but there is a ".
|
||||||
|
"file there instead. Remove it and let the daemon clone a ".
|
||||||
|
"repository for you.";
|
||||||
|
} else {
|
||||||
|
$message =
|
||||||
|
"Expected to find a git repository at '{$path}', but did not.";
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
$repo_path = rtrim($stdout, "\n");
|
||||||
|
|
||||||
|
if (empty($repo_path)) {
|
||||||
|
$err = true;
|
||||||
|
$msg =
|
||||||
|
"Expected to find a git repository at '{$path}', but ".
|
||||||
|
"there was no result from `git rev-parse --show-toplevel`. ".
|
||||||
|
"Something is misconfigured or broken. The git repository ".
|
||||||
|
"may be inside a '.git/' directory.";
|
||||||
|
} else if (!Filesystem::pathsAreEquivalent($repo_path, $path)) {
|
||||||
|
$err = true;
|
||||||
|
$msg =
|
||||||
|
"Expected to find repo at '{$path}', but the actual ".
|
||||||
|
"git repository root for this directory is '{$repo_path}'. ".
|
||||||
|
"Something is misconfigured. The repository's 'Local Path' should ".
|
||||||
|
"be set to some place where the daemon can check out a working ".
|
||||||
|
"copy, and should not be inside another git repository.";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($err && $this->canDestroyWorkingCopy($path)) {
|
||||||
|
phlog("Repository working copy at '{$path}' failed sanity check; ".
|
||||||
|
"destroying and re-cloning. {$message}");
|
||||||
|
Filesystem::remove($path);
|
||||||
|
$this->executeGitCreate();
|
||||||
|
} else if ($err) {
|
||||||
|
throw new Exception($message);
|
||||||
|
}
|
||||||
|
|
||||||
|
$retry = false;
|
||||||
|
do {
|
||||||
|
// This is a local command, but needs credentials.
|
||||||
|
$future = $repository->getRemoteCommandFuture('fetch --all --prune');
|
||||||
|
$future->setCWD($path);
|
||||||
|
list($err, $stdout, $stderr) = $future->resolve();
|
||||||
|
|
||||||
|
if ($err && !$retry && $this->canDestroyWorkingCopy($path)) {
|
||||||
|
$retry = true;
|
||||||
|
// Fix remote origin url if it doesn't match our configuration
|
||||||
|
$origin_url = $repository->execLocalCommand(
|
||||||
|
'config --get remote.origin.url');
|
||||||
|
$remote_uri = $repository->getDetail('remote-uri');
|
||||||
|
if ($origin_url != $remote_uri) {
|
||||||
|
$repository->execLocalCommand(
|
||||||
|
'remote set-url origin %s',
|
||||||
|
$remote_uri);
|
||||||
|
}
|
||||||
|
} else if ($err) {
|
||||||
|
throw new Exception(
|
||||||
|
"git fetch failed with error #{$err}:\n".
|
||||||
|
"stdout:{$stdout}\n\n".
|
||||||
|
"stderr:{$stderr}\n");
|
||||||
|
} else {
|
||||||
|
$retry = false;
|
||||||
|
}
|
||||||
|
} while ($retry);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/* -( Pulling Mercurial Working Copies )----------------------------------- */
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @task hg
|
||||||
|
*/
|
||||||
|
private function executeMercurialCreate() {
|
||||||
|
$repository = $this->getRepository();
|
||||||
|
|
||||||
|
$repository->execxRemoteCommand(
|
||||||
|
'clone %s %s',
|
||||||
|
$repository->getRemoteURI(),
|
||||||
|
rtrim($repository->getLocalPath(), '/'));
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @task hg
|
||||||
|
*/
|
||||||
|
private function executeMercurialUpdate() {
|
||||||
|
$repository = $this->getRepository();
|
||||||
|
$path = $repository->getLocalPath();
|
||||||
|
|
||||||
|
// This is a local command, but needs credentials.
|
||||||
|
$future = $repository->getRemoteCommandFuture('pull -u');
|
||||||
|
$future->setCWD($path);
|
||||||
|
|
||||||
|
try {
|
||||||
|
$future->resolvex();
|
||||||
|
} catch (CommandException $ex) {
|
||||||
|
$err = $ex->getError();
|
||||||
|
$stdout = $ex->getStdOut();
|
||||||
|
|
||||||
|
// NOTE: Between versions 2.1 and 2.1.1, Mercurial changed the behavior
|
||||||
|
// of "hg pull" to return 1 in case of a successful pull with no changes.
|
||||||
|
// This behavior has been reverted, but users who updated between Feb 1,
|
||||||
|
// 2012 and Mar 1, 2012 will have the erroring version. Do a dumb test
|
||||||
|
// against stdout to check for this possibility.
|
||||||
|
// See: https://github.com/facebook/phabricator/issues/101/
|
||||||
|
|
||||||
|
// NOTE: Mercurial has translated versions, which translate this error
|
||||||
|
// string. In a translated version, the string will be something else,
|
||||||
|
// like "aucun changement trouve". There didn't seem to be an easy way
|
||||||
|
// to handle this (there are hard ways but this is not a common problem
|
||||||
|
// and only creates log spam, not application failures). Assume English.
|
||||||
|
|
||||||
|
// TODO: Remove this once we're far enough in the future that deployment
|
||||||
|
// of 2.1 is exceedingly rare?
|
||||||
|
if ($err == 1 && preg_match('/no changes found/', $stdout)) {
|
||||||
|
return;
|
||||||
|
} else {
|
||||||
|
throw $ex;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/* -( Internals )---------------------------------------------------------- */
|
||||||
|
|
||||||
|
|
||||||
|
private function canDestroyWorkingCopy($path) {
|
||||||
|
$default_path = PhabricatorEnv::getEnvConfig(
|
||||||
|
'repository.default-local-path');
|
||||||
|
return Filesystem::isDescendant($path, $default_path);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -0,0 +1,32 @@
|
||||||
|
<?php
|
||||||
|
|
||||||
|
final class PhabricatorWorkingCopyPullTestCase
|
||||||
|
extends PhabricatorWorkingCopyTestCase {
|
||||||
|
|
||||||
|
public function testGitPullBasic() {
|
||||||
|
$repo = $this->buildPulledRepository('GT');
|
||||||
|
|
||||||
|
$this->assertEqual(
|
||||||
|
true,
|
||||||
|
Filesystem::pathExists($repo->getLocalPath().'/.git'));
|
||||||
|
}
|
||||||
|
|
||||||
|
public function testHgPullBasic() {
|
||||||
|
$repo = $this->buildPulledRepository('HT');
|
||||||
|
|
||||||
|
$this->assertEqual(
|
||||||
|
true,
|
||||||
|
Filesystem::pathExists($repo->getLocalPath().'/.hg'));
|
||||||
|
}
|
||||||
|
|
||||||
|
public function testSVNPullBasic() {
|
||||||
|
$repo = $this->buildPulledRepository('ST');
|
||||||
|
|
||||||
|
// We don't pull local clones for SVN, so we don't expect there to be
|
||||||
|
// a working copy.
|
||||||
|
$this->assertEqual(
|
||||||
|
false,
|
||||||
|
Filesystem::pathExists($repo->getLocalPath()));
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -0,0 +1,95 @@
|
||||||
|
<?php
|
||||||
|
|
||||||
|
abstract class PhabricatorWorkingCopyTestCase extends PhabricatorTestCase {
|
||||||
|
|
||||||
|
private $dirs = array();
|
||||||
|
private $repos = array();
|
||||||
|
private $pulled = array();
|
||||||
|
|
||||||
|
protected function getPhabricatorTestCaseConfiguration() {
|
||||||
|
return array(
|
||||||
|
self::PHABRICATOR_TESTCONFIG_BUILD_STORAGE_FIXTURES => true,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
protected function buildBareRepository($callsign) {
|
||||||
|
if (isset($this->repos[$callsign])) {
|
||||||
|
return $this->repos[$callsign];
|
||||||
|
}
|
||||||
|
|
||||||
|
$data_dir = dirname(__FILE__).'/data/';
|
||||||
|
|
||||||
|
$types = array(
|
||||||
|
'svn' => PhabricatorRepositoryType::REPOSITORY_TYPE_SVN,
|
||||||
|
'hg' => PhabricatorRepositoryType::REPOSITORY_TYPE_MERCURIAL,
|
||||||
|
'git' => PhabricatorRepositoryType::REPOSITORY_TYPE_GIT,
|
||||||
|
);
|
||||||
|
|
||||||
|
$hits = array();
|
||||||
|
foreach ($types as $type => $const) {
|
||||||
|
$path = $data_dir.$callsign.'.'.$type.'.tgz';
|
||||||
|
if (Filesystem::pathExists($path)) {
|
||||||
|
$hits[$const] = $path;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!$hits) {
|
||||||
|
throw new Exception(
|
||||||
|
"No test data for callsign '{$callsign}'. Expected an archive ".
|
||||||
|
"like '{$callsign}.git.tgz' in '{$data_dir}'.");
|
||||||
|
}
|
||||||
|
|
||||||
|
if (count($hits) > 1) {
|
||||||
|
throw new Exception(
|
||||||
|
"Expected exactly one archive matching callsign '{$callsign}', ".
|
||||||
|
"found too many: ".implode(', ', $hits));
|
||||||
|
}
|
||||||
|
|
||||||
|
$path = head($hits);
|
||||||
|
$vcs_type = head_key($hits);
|
||||||
|
|
||||||
|
$dir = PhutilDirectoryFixture::newFromArchive($path);
|
||||||
|
$local = new TempFile('.ignore');
|
||||||
|
|
||||||
|
$repo = id(new PhabricatorRepository())
|
||||||
|
->setCallsign($callsign)
|
||||||
|
->setName(pht('Test Repo "%s"', $callsign))
|
||||||
|
->setVersionControlSystem($vcs_type)
|
||||||
|
->setDetail('local-path', dirname($local).'/'.$callsign)
|
||||||
|
->setDetail('remote-uri', 'file://'.$dir->getPath().'/');
|
||||||
|
|
||||||
|
$this->didConstructRepository($repo);
|
||||||
|
|
||||||
|
$repo->save();
|
||||||
|
$repo->makeEphemeral();
|
||||||
|
|
||||||
|
// Keep the disk resources around until we exit.
|
||||||
|
$this->dirs[] = $dir;
|
||||||
|
$this->dirs[] = $local;
|
||||||
|
|
||||||
|
$this->repos[$callsign] = $repo;
|
||||||
|
|
||||||
|
return $repo;
|
||||||
|
}
|
||||||
|
|
||||||
|
protected function didConstructRepository(PhabricatorRepository $repository) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
protected function buildPulledRepository($callsign) {
|
||||||
|
$repository = $this->buildBareRepository($callsign);
|
||||||
|
|
||||||
|
if (isset($this->pulled[$callsign])) {
|
||||||
|
return $repository;
|
||||||
|
}
|
||||||
|
|
||||||
|
id(new PhabricatorRepositoryPullEngine())
|
||||||
|
->setRepository($repository)
|
||||||
|
->pullRepository();
|
||||||
|
|
||||||
|
$this->pulled[$callsign] = true;
|
||||||
|
|
||||||
|
return $repository;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
BIN
src/applications/repository/engine/__tests__/data/GT.git.tgz
Normal file
BIN
src/applications/repository/engine/__tests__/data/GT.git.tgz
Normal file
Binary file not shown.
BIN
src/applications/repository/engine/__tests__/data/HT.hg.tgz
Normal file
BIN
src/applications/repository/engine/__tests__/data/HT.hg.tgz
Normal file
Binary file not shown.
BIN
src/applications/repository/engine/__tests__/data/ST.svn.tgz
Normal file
BIN
src/applications/repository/engine/__tests__/data/ST.svn.tgz
Normal file
Binary file not shown.
|
@ -10,10 +10,6 @@ final class PhabricatorRepositoryManagementPullWorkflow
|
||||||
->setSynopsis('Pull __repository__, named by callsign or PHID.')
|
->setSynopsis('Pull __repository__, named by callsign or PHID.')
|
||||||
->setArguments(
|
->setArguments(
|
||||||
array(
|
array(
|
||||||
array(
|
|
||||||
'name' => 'verbose',
|
|
||||||
'help' => 'Show additional debugging information.',
|
|
||||||
),
|
|
||||||
array(
|
array(
|
||||||
'name' => 'repos',
|
'name' => 'repos',
|
||||||
'wildcard' => true,
|
'wildcard' => true,
|
||||||
|
@ -34,9 +30,9 @@ final class PhabricatorRepositoryManagementPullWorkflow
|
||||||
foreach ($repos as $repo) {
|
foreach ($repos as $repo) {
|
||||||
$console->writeOut("Pulling '%s'...\n", $repo->getCallsign());
|
$console->writeOut("Pulling '%s'...\n", $repo->getCallsign());
|
||||||
|
|
||||||
$daemon = new PhabricatorRepositoryPullLocalDaemon(array());
|
id(new PhabricatorRepositoryPullEngine())
|
||||||
$daemon->setVerbose($args->getArg('verbose'));
|
->setRepository($repo)
|
||||||
$daemon->pullRepository($repo);
|
->pullRepository();
|
||||||
}
|
}
|
||||||
|
|
||||||
$console->writeOut("Done.\n");
|
$console->writeOut("Done.\n");
|
||||||
|
|
Loading…
Add table
Reference in a new issue