mirror of
https://we.phorge.it/source/phorge.git
synced 2025-01-19 19:21:10 +01:00
Restructure HookEngine to use PushLog records for all operations
Summary: Ref T4195. This pulls the central logic of HookEngine up one level and makes all the git stuff genrate PushLogs. In future diffs, everything will generate PushLogs and we can hand those off to Herald. Test Plan: Pushed a pile of valid/invalid stuff: {F89256} Reviewers: btrahan Reviewed By: btrahan CC: aran Maniphest Tasks: T4195 Differential Revision: https://secure.phabricator.com/D7761
This commit is contained in:
parent
488d4c509d
commit
2725586baf
3 changed files with 313 additions and 193 deletions
|
@ -83,6 +83,10 @@ final class DiffusionPushLogListController extends DiffusionController
|
||||||
'href' => '/r'.$callsign.$log->getRefNew(),
|
'href' => '/r'.$callsign.$log->getRefNew(),
|
||||||
),
|
),
|
||||||
$log->getRefNewShort()),
|
$log->getRefNewShort()),
|
||||||
|
|
||||||
|
// TODO: Make these human-readable.
|
||||||
|
$log->getChangeFlags(),
|
||||||
|
$log->getRejectCode(),
|
||||||
phabricator_datetime($log->getEpoch(), $viewer),
|
phabricator_datetime($log->getEpoch(), $viewer),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
@ -98,6 +102,8 @@ final class DiffusionPushLogListController extends DiffusionController
|
||||||
pht('Name'),
|
pht('Name'),
|
||||||
pht('Old'),
|
pht('Old'),
|
||||||
pht('New'),
|
pht('New'),
|
||||||
|
pht('Flags'),
|
||||||
|
pht('Code'),
|
||||||
pht('Date'),
|
pht('Date'),
|
||||||
))
|
))
|
||||||
->setColumnClasses(
|
->setColumnClasses(
|
||||||
|
|
|
@ -1,9 +1,12 @@
|
||||||
<?php
|
<?php
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @task git Git Hooks
|
* @task config Configuring the Hook Engine
|
||||||
* @task hg Mercurial Hooks
|
* @task hook Hook Execution
|
||||||
* @task svn Subversion Hooks
|
* @task git Git Hooks
|
||||||
|
* @task hg Mercurial Hooks
|
||||||
|
* @task svn Subversion Hooks
|
||||||
|
* @task internal Internals
|
||||||
*/
|
*/
|
||||||
final class DiffusionCommitHookEngine extends Phobject {
|
final class DiffusionCommitHookEngine extends Phobject {
|
||||||
|
|
||||||
|
@ -11,6 +14,8 @@ final class DiffusionCommitHookEngine extends Phobject {
|
||||||
const ENV_REMOTE_ADDRESS = 'PHABRICATOR_REMOTE_ADDRESS';
|
const ENV_REMOTE_ADDRESS = 'PHABRICATOR_REMOTE_ADDRESS';
|
||||||
const ENV_REMOTE_PROTOCOL = 'PHABRICATOR_REMOTE_PROTOCOL';
|
const ENV_REMOTE_PROTOCOL = 'PHABRICATOR_REMOTE_PROTOCOL';
|
||||||
|
|
||||||
|
const EMPTY_HASH = '0000000000000000000000000000000000000000';
|
||||||
|
|
||||||
private $viewer;
|
private $viewer;
|
||||||
private $repository;
|
private $repository;
|
||||||
private $stdin;
|
private $stdin;
|
||||||
|
@ -20,6 +25,10 @@ final class DiffusionCommitHookEngine extends Phobject {
|
||||||
private $remoteProtocol;
|
private $remoteProtocol;
|
||||||
private $transactionKey;
|
private $transactionKey;
|
||||||
|
|
||||||
|
|
||||||
|
/* -( Config )------------------------------------------------------------- */
|
||||||
|
|
||||||
|
|
||||||
public function setRemoteProtocol($remote_protocol) {
|
public function setRemoteProtocol($remote_protocol) {
|
||||||
$this->remoteProtocol = $remote_protocol;
|
$this->remoteProtocol = $remote_protocol;
|
||||||
return $this;
|
return $this;
|
||||||
|
@ -88,171 +97,190 @@ final class DiffusionCommitHookEngine extends Phobject {
|
||||||
return $this->viewer;
|
return $this->viewer;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/* -( Hook Execution )----------------------------------------------------- */
|
||||||
|
|
||||||
|
|
||||||
public function execute() {
|
public function execute() {
|
||||||
$type = $this->getRepository()->getVersionControlSystem();
|
$ref_updates = $this->findRefUpdates();
|
||||||
switch ($type) {
|
$all_updates = $ref_updates;
|
||||||
case PhabricatorRepositoryType::REPOSITORY_TYPE_GIT:
|
|
||||||
$err = $this->executeGitHook();
|
|
||||||
break;
|
|
||||||
case PhabricatorRepositoryType::REPOSITORY_TYPE_SVN:
|
|
||||||
$err = $this->executeSubversionHook();
|
|
||||||
break;
|
|
||||||
case PhabricatorRepositoryType::REPOSITORY_TYPE_MERCURIAL:
|
|
||||||
$err = $this->executeMercurialHook();
|
|
||||||
break;
|
|
||||||
default:
|
|
||||||
throw new Exception(pht('Unsupported repository type "%s"!', $type));
|
|
||||||
}
|
|
||||||
|
|
||||||
return $err;
|
$caught = null;
|
||||||
}
|
try {
|
||||||
|
|
||||||
private function newPushLog() {
|
try {
|
||||||
return PhabricatorRepositoryPushLog::initializeNewLog($this->getViewer())
|
$this->rejectDangerousChanges($ref_updates);
|
||||||
->setRepositoryPHID($this->getRepository()->getPHID())
|
} catch (DiffusionCommitHookRejectException $ex) {
|
||||||
->setEpoch(time())
|
// If we're rejecting dangerous changes, flag everything that we've
|
||||||
->setRemoteAddress($this->getRemoteAddressForLog())
|
// seen as rejected so it's clear that none of it was accepted.
|
||||||
->setRemoteProtocol($this->getRemoteProtocol())
|
foreach ($all_updates as $update) {
|
||||||
->setTransactionKey($this->getTransactionKey())
|
$update->setRejectCode(
|
||||||
->setRejectCode(PhabricatorRepositoryPushLog::REJECT_ACCEPT)
|
PhabricatorRepositoryPushLog::REJECT_DANGEROUS);
|
||||||
->setRejectDetails(null);
|
}
|
||||||
}
|
throw $ex;
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @task git
|
|
||||||
*/
|
|
||||||
private function executeGitHook() {
|
|
||||||
$updates = $this->parseGitUpdates($this->getStdin());
|
|
||||||
|
|
||||||
$this->rejectGitDangerousChanges($updates);
|
|
||||||
|
|
||||||
// TODO: Do cheap checks: non-ff commits, mutating refs without access,
|
|
||||||
// creating or deleting things you can't touch. We can do all non-content
|
|
||||||
// checks here.
|
|
||||||
|
|
||||||
$updates = $this->findGitNewCommits($updates);
|
|
||||||
|
|
||||||
// TODO: Now, do content checks.
|
|
||||||
|
|
||||||
// TODO: Generalize this; just getting some data in the database for now.
|
|
||||||
|
|
||||||
$logs = array();
|
|
||||||
foreach ($updates as $update) {
|
|
||||||
$log = $this->newPushLog()
|
|
||||||
->setRefType($update['type'])
|
|
||||||
->setRefNameHash(PhabricatorHash::digestForIndex($update['ref']))
|
|
||||||
->setRefNameRaw($update['ref'])
|
|
||||||
->setRefNameEncoding(phutil_is_utf8($update['ref']) ? 'utf8' : null)
|
|
||||||
->setRefOld($update['old'])
|
|
||||||
->setRefNew($update['new'])
|
|
||||||
->setMergeBase(idx($update, 'merge-base'));
|
|
||||||
|
|
||||||
$flags = 0;
|
|
||||||
if ($update['operation'] == 'create') {
|
|
||||||
$flags = $flags | PhabricatorRepositoryPushLog::CHANGEFLAG_ADD;
|
|
||||||
} else if ($update['operation'] == 'delete') {
|
|
||||||
$flags = $flags | PhabricatorRepositoryPushLog::CHANGEFLAG_DELETE;
|
|
||||||
} else {
|
|
||||||
// TODO: This isn't correct; these might be APPEND or REWRITE, and
|
|
||||||
// if they're REWRITE they might be DANGEROUS. Fix this when this
|
|
||||||
// gets generalized.
|
|
||||||
$flags = $flags | PhabricatorRepositoryPushLog::CHANGEFLAG_APPEND;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
$log->setChangeFlags($flags);
|
// TODO: Fire ref herald rules.
|
||||||
$logs[] = $log;
|
|
||||||
|
$content_updates = $this->findContentUpdates($ref_updates);
|
||||||
|
$all_updates = array_merge($all_updates, $content_updates);
|
||||||
|
|
||||||
|
// TODO: Fire content Herald rules.
|
||||||
|
// TODO: Fire external hooks.
|
||||||
|
|
||||||
|
// If we make it this far, we're accepting these changes. Mark all the
|
||||||
|
// logs as accepted.
|
||||||
|
foreach ($all_updates as $update) {
|
||||||
|
$update->setRejectCode(PhabricatorRepositoryPushLog::REJECT_ACCEPT);
|
||||||
|
}
|
||||||
|
} catch (Exception $ex) {
|
||||||
|
// We'll throw this again in a minute, but we want to save all the logs
|
||||||
|
// first.
|
||||||
|
$caught = $ex;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Now, build logs for all the commits.
|
// Save all the logs no matter what the outcome was.
|
||||||
// TODO: Generalize this, too.
|
foreach ($all_updates as $update) {
|
||||||
$commits = array_mergev(ipull($updates, 'commits'));
|
$update->save();
|
||||||
$commits = array_unique($commits);
|
|
||||||
foreach ($commits as $commit) {
|
|
||||||
$log = $this->newPushLog()
|
|
||||||
->setRefType(PhabricatorRepositoryPushLog::REFTYPE_COMMIT)
|
|
||||||
->setRefNew($commit)
|
|
||||||
->setChangeFlags(PhabricatorRepositoryPushLog::CHANGEFLAG_ADD);
|
|
||||||
$logs[] = $log;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
foreach ($logs as $log) {
|
if ($caught) {
|
||||||
$log->save();
|
throw $caught;
|
||||||
}
|
}
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private function findRefUpdates() {
|
||||||
|
$type = $this->getRepository()->getVersionControlSystem();
|
||||||
|
switch ($type) {
|
||||||
|
case PhabricatorRepositoryType::REPOSITORY_TYPE_GIT:
|
||||||
|
return $this->findGitRefUpdates();
|
||||||
|
case PhabricatorRepositoryType::REPOSITORY_TYPE_MERCURIAL:
|
||||||
|
return $this->findMercurialRefUpdates();
|
||||||
|
case PhabricatorRepositoryType::REPOSITORY_TYPE_SVN:
|
||||||
|
return $this->findSubversionRefUpdates();
|
||||||
|
default:
|
||||||
|
throw new Exception(pht('Unsupported repository type "%s"!', $type));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
private function rejectDangerousChanges(array $ref_updates) {
|
||||||
* @task git
|
assert_instances_of($ref_updates, 'PhabricatorRepositoryPushLog');
|
||||||
*/
|
|
||||||
private function parseGitUpdates($stdin) {
|
|
||||||
$updates = array();
|
|
||||||
|
|
||||||
|
$repository = $this->getRepository();
|
||||||
|
if ($repository->shouldAllowDangerousChanges()) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
$flag_dangerous = PhabricatorRepositoryPushLog::CHANGEFLAG_DANGEROUS;
|
||||||
|
|
||||||
|
foreach ($ref_updates as $ref_update) {
|
||||||
|
if (!$ref_update->hasChangeFlags($flag_dangerous)) {
|
||||||
|
// This is not a dangerous change.
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
// We either have a branch deletion or a non fast-forward branch update.
|
||||||
|
// Format a message and reject the push.
|
||||||
|
|
||||||
|
$message = pht(
|
||||||
|
"DANGEROUS CHANGE: %s\n".
|
||||||
|
"Dangerous change protection is enabled for this repository.\n".
|
||||||
|
"Edit the repository configuration before making dangerous changes.",
|
||||||
|
$ref_update->getDangerousChangeDescription());
|
||||||
|
|
||||||
|
throw new DiffusionCommitHookRejectException($message);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private function findContentUpdates(array $ref_updates) {
|
||||||
|
assert_instances_of($ref_updates, 'PhabricatorRepositoryPushLog');
|
||||||
|
|
||||||
|
$type = $this->getRepository()->getVersionControlSystem();
|
||||||
|
switch ($type) {
|
||||||
|
case PhabricatorRepositoryType::REPOSITORY_TYPE_GIT:
|
||||||
|
return $this->findGitContentUpdates($ref_updates);
|
||||||
|
case PhabricatorRepositoryType::REPOSITORY_TYPE_MERCURIAL:
|
||||||
|
return $this->findMercurialContentUpdates($ref_updates);
|
||||||
|
case PhabricatorRepositoryType::REPOSITORY_TYPE_SVN:
|
||||||
|
return $this->findSubversionContentUpdates($ref_updates);
|
||||||
|
default:
|
||||||
|
throw new Exception(pht('Unsupported repository type "%s"!', $type));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/* -( Git )---------------------------------------------------------------- */
|
||||||
|
|
||||||
|
|
||||||
|
private function findGitRefUpdates() {
|
||||||
|
$ref_updates = array();
|
||||||
|
|
||||||
|
// First, parse stdin, which lists all the ref changes. The input looks
|
||||||
|
// like this:
|
||||||
|
//
|
||||||
|
// <old hash> <new hash> <ref>
|
||||||
|
|
||||||
|
$stdin = $this->getStdin();
|
||||||
$lines = phutil_split_lines($stdin, $retain_endings = false);
|
$lines = phutil_split_lines($stdin, $retain_endings = false);
|
||||||
foreach ($lines as $line) {
|
foreach ($lines as $line) {
|
||||||
$parts = explode(' ', $line, 3);
|
$parts = explode(' ', $line, 3);
|
||||||
if (count($parts) != 3) {
|
if (count($parts) != 3) {
|
||||||
throw new Exception(pht('Expected "old new ref", got "%s".', $line));
|
throw new Exception(pht('Expected "old new ref", got "%s".', $line));
|
||||||
}
|
}
|
||||||
$update = array(
|
|
||||||
'old' => $parts[0],
|
|
||||||
'old.short' => substr($parts[0], 0, 8),
|
|
||||||
'new' => $parts[1],
|
|
||||||
'new.short' => substr($parts[1], 0, 8),
|
|
||||||
'ref' => $parts[2],
|
|
||||||
);
|
|
||||||
|
|
||||||
if (preg_match('(^refs/heads/)', $update['ref'])) {
|
$ref_old = $parts[0];
|
||||||
$update['type'] = 'branch';
|
$ref_new = $parts[1];
|
||||||
$update['ref.short'] = substr($update['ref'], strlen('refs/heads/'));
|
$ref_raw = $parts[2];
|
||||||
} else if (preg_match('(^refs/tags/)', $update['ref'])) {
|
|
||||||
$update['type'] = 'tag';
|
if (preg_match('(^refs/heads/)', $ref_raw)) {
|
||||||
$update['ref.short'] = substr($update['ref'], strlen('refs/tags/'));
|
$ref_type = PhabricatorRepositoryPushLog::REFTYPE_BRANCH;
|
||||||
|
} else if (preg_match('(^refs/tags/)', $ref_raw)) {
|
||||||
|
$ref_type = PhabricatorRepositoryPushLog::REFTYPE_TAG;
|
||||||
} else {
|
} else {
|
||||||
$update['type'] = 'unknown';
|
$ref_type = PhabricatorRepositoryPushLog::REFTYPE_UNKNOWN;
|
||||||
}
|
}
|
||||||
|
|
||||||
$updates[] = $update;
|
$ref_update = $this->newPushLog()
|
||||||
|
->setRefType($ref_type)
|
||||||
|
->setRefName($ref_raw)
|
||||||
|
->setRefOld($ref_old)
|
||||||
|
->setRefNew($ref_new);
|
||||||
|
|
||||||
|
$ref_updates[] = $ref_update;
|
||||||
}
|
}
|
||||||
|
|
||||||
$updates = $this->findGitMergeBases($updates);
|
$this->findGitMergeBases($ref_updates);
|
||||||
|
$this->findGitChangeFlags($ref_updates);
|
||||||
|
|
||||||
return $updates;
|
return $ref_updates;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/**
|
private function findGitMergeBases(array $ref_updates) {
|
||||||
* @task git
|
assert_instances_of($ref_updates, 'PhabricatorRepositoryPushLog');
|
||||||
*/
|
|
||||||
private function findGitMergeBases(array $updates) {
|
|
||||||
$empty = str_repeat('0', 40);
|
|
||||||
|
|
||||||
$futures = array();
|
$futures = array();
|
||||||
foreach ($updates as $key => $update) {
|
foreach ($ref_updates as $key => $ref_update) {
|
||||||
// Updates are in the form:
|
|
||||||
//
|
|
||||||
// <old hash> <new hash> <ref>
|
|
||||||
//
|
|
||||||
// If the old hash is "00000...", the ref is being created (either a new
|
// If the old hash is "00000...", the ref is being created (either a new
|
||||||
// branch, or a new tag). If the new hash is "00000...", the ref is being
|
// branch, or a new tag). If the new hash is "00000...", the ref is being
|
||||||
// deleted. If both are nonempty, the ref is being updated. For updates,
|
// deleted. If both are nonempty, the ref is being updated. For updates,
|
||||||
// we'll figure out the `merge-base` of the old and new objects here. This
|
// we'll figure out the `merge-base` of the old and new objects here. This
|
||||||
// lets us reject non-FF changes cheaply; later, we'll figure out exactly
|
// lets us reject non-FF changes cheaply; later, we'll figure out exactly
|
||||||
// which commits are new.
|
// which commits are new.
|
||||||
|
$ref_old = $ref_update->getRefOld();
|
||||||
|
$ref_new = $ref_update->getRefNew();
|
||||||
|
|
||||||
if ($update['old'] == $empty) {
|
if (($ref_old === self::EMPTY_HASH) ||
|
||||||
$updates[$key]['operation'] = 'create';
|
($ref_new === self::EMPTY_HASH)) {
|
||||||
} else if ($update['new'] == $empty) {
|
continue;
|
||||||
$updates[$key]['operation'] = 'delete';
|
|
||||||
} else {
|
|
||||||
$updates[$key]['operation'] = 'update';
|
|
||||||
$futures[$key] = $this->getRepository()->getLocalCommandFuture(
|
|
||||||
'merge-base %s %s',
|
|
||||||
$update['old'],
|
|
||||||
$update['new']);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
$futures[$key] = $this->getRepository()->getLocalCommandFuture(
|
||||||
|
'merge-base %s %s',
|
||||||
|
$ref_old,
|
||||||
|
$ref_new);
|
||||||
}
|
}
|
||||||
|
|
||||||
foreach (Futures($futures)->limit(8) as $key => $future) {
|
foreach (Futures($futures)->limit(8) as $key => $future) {
|
||||||
|
@ -264,20 +292,86 @@ final class DiffusionCommitHookEngine extends Phobject {
|
||||||
// T4224. Assume this means there are no ancestors.
|
// T4224. Assume this means there are no ancestors.
|
||||||
|
|
||||||
list($err, $stdout) = $future->resolve();
|
list($err, $stdout) = $future->resolve();
|
||||||
|
|
||||||
if ($err) {
|
if ($err) {
|
||||||
$updates[$key]['merge-base'] = null;
|
$merge_base = null;
|
||||||
} else {
|
} else {
|
||||||
$updates[$key]['merge-base'] = rtrim($stdout, "\n");
|
$merge_base = rtrim($stdout, "\n");
|
||||||
|
}
|
||||||
|
|
||||||
|
$ref_update->setMergeBase($merge_base);
|
||||||
|
}
|
||||||
|
|
||||||
|
return $ref_updates;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
private function findGitChangeFlags(array $ref_updates) {
|
||||||
|
assert_instances_of($ref_updates, 'PhabricatorRepositoryPushLog');
|
||||||
|
|
||||||
|
foreach ($ref_updates as $key => $ref_update) {
|
||||||
|
$ref_old = $ref_update->getRefOld();
|
||||||
|
$ref_new = $ref_update->getRefNew();
|
||||||
|
$ref_type = $ref_update->getRefType();
|
||||||
|
|
||||||
|
$ref_flags = 0;
|
||||||
|
$dangerous = null;
|
||||||
|
|
||||||
|
if ($ref_old === self::EMPTY_HASH) {
|
||||||
|
$ref_flags |= PhabricatorRepositoryPushLog::CHANGEFLAG_ADD;
|
||||||
|
} else if ($ref_new === self::EMPTY_HASH) {
|
||||||
|
$ref_flags |= PhabricatorRepositoryPushLog::CHANGEFLAG_DELETE;
|
||||||
|
if ($ref_type == PhabricatorRepositoryPushLog::REFTYPE_BRANCH) {
|
||||||
|
$ref_flags |= PhabricatorRepositoryPushLog::CHANGEFLAG_DANGEROUS;
|
||||||
|
$dangerous = pht(
|
||||||
|
"The change you're attempting to push deletes the branch '%s'.",
|
||||||
|
$ref_update->getRefName());
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
$merge_base = $ref_update->getMergeBase();
|
||||||
|
if ($merge_base == $ref_old) {
|
||||||
|
// This is a fast-forward update to an existing branch.
|
||||||
|
// These are safe.
|
||||||
|
$ref_flags |= PhabricatorRepositoryPushLog::CHANGEFLAG_APPEND;
|
||||||
|
} else {
|
||||||
|
$ref_flags |= PhabricatorRepositoryPushLog::CHANGEFLAG_REWRITE;
|
||||||
|
|
||||||
|
// For now, we don't consider deleting or moving tags to be a
|
||||||
|
// "dangerous" update. It's way harder to get wrong and should be easy
|
||||||
|
// to recover from once we have better logging. Only add the dangerous
|
||||||
|
// flag if this ref is a branch.
|
||||||
|
|
||||||
|
if ($ref_type == PhabricatorRepositoryPushLog::REFTYPE_BRANCH) {
|
||||||
|
$ref_flags |= PhabricatorRepositoryPushLog::CHANGEFLAG_DANGEROUS;
|
||||||
|
|
||||||
|
$dangerous = pht(
|
||||||
|
"DANGEROUS CHANGE: The change you're attempting to push updates ".
|
||||||
|
"the branch '%s' from '%s' to '%s', but this is not a ".
|
||||||
|
"fast-forward. Pushes which rewrite published branch history ".
|
||||||
|
"are dangerous.",
|
||||||
|
$ref_update->getRefName(),
|
||||||
|
$ref_update->getRefOldShort(),
|
||||||
|
$ref_update->getRefNewShort());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
$ref_update->setChangeFlags($ref_flags);
|
||||||
|
if ($dangerous !== null) {
|
||||||
|
$ref_update->attachDangerousChangeDescription($dangerous);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return $updates;
|
return $ref_updates;
|
||||||
}
|
}
|
||||||
|
|
||||||
private function findGitNewCommits(array $updates) {
|
|
||||||
|
private function findGitContentUpdates(array $ref_updates) {
|
||||||
|
$flag_delete = PhabricatorRepositoryPushLog::CHANGEFLAG_DELETE;
|
||||||
|
|
||||||
$futures = array();
|
$futures = array();
|
||||||
foreach ($updates as $key => $update) {
|
foreach ($ref_updates as $key => $ref_update) {
|
||||||
if ($update['operation'] == 'delete') {
|
if ($ref_update->hasChangeFlags($flag_delete)) {
|
||||||
// Deleting a branch or tag can never create any new commits.
|
// Deleting a branch or tag can never create any new commits.
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
@ -289,84 +383,78 @@ final class DiffusionCommitHookEngine extends Phobject {
|
||||||
$futures[$key] = $this->getRepository()->getLocalCommandFuture(
|
$futures[$key] = $this->getRepository()->getLocalCommandFuture(
|
||||||
'log --format=%s %s --not --all',
|
'log --format=%s %s --not --all',
|
||||||
'%H',
|
'%H',
|
||||||
$update['new']);
|
$ref_update->getRefNew());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
$content_updates = array();
|
||||||
foreach (Futures($futures)->limit(8) as $key => $future) {
|
foreach (Futures($futures)->limit(8) as $key => $future) {
|
||||||
list($stdout) = $future->resolvex();
|
list($stdout) = $future->resolvex();
|
||||||
|
|
||||||
|
if (!strlen(trim($stdout))) {
|
||||||
|
// This change doesn't have any new commits. One common case of this
|
||||||
|
// is creating a new tag which points at an existing commit.
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
$commits = phutil_split_lines($stdout, $retain_newlines = false);
|
$commits = phutil_split_lines($stdout, $retain_newlines = false);
|
||||||
$updates[$key]['commits'] = $commits;
|
|
||||||
|
foreach ($commits as $commit) {
|
||||||
|
$content_updates[$commit] = $this->newPushLog()
|
||||||
|
->setRefType(PhabricatorRepositoryPushLog::REFTYPE_COMMIT)
|
||||||
|
->setRefNew($commit)
|
||||||
|
->setChangeFlags(PhabricatorRepositoryPushLog::CHANGEFLAG_ADD);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return $updates;
|
return $content_updates;
|
||||||
}
|
}
|
||||||
|
|
||||||
private function rejectGitDangerousChanges(array $updates) {
|
|
||||||
$repository = $this->getRepository();
|
|
||||||
if ($repository->shouldAllowDangerousChanges()) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
foreach ($updates as $update) {
|
/* -( Mercurial )---------------------------------------------------------- */
|
||||||
if ($update['type'] != 'branch') {
|
|
||||||
// For now, we don't consider deleting or moving tags to be a
|
|
||||||
// "dangerous" update. It's way harder to get wrong and should be easy
|
|
||||||
// to recover from once we have better logging.
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
if ($update['operation'] == 'create') {
|
|
||||||
// Creating a branch is never dangerous.
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
if ($update['operation'] == 'update') {
|
private function findMercurialRefUpdates() {
|
||||||
if ($update['old'] == $update['merge-base']) {
|
// TODO: Implement.
|
||||||
// This is a fast-forward update to an existing branch.
|
return array();
|
||||||
// These are safe.
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// We either have a branch deletion or a non fast-forward branch update.
|
|
||||||
// Format a message and reject the push.
|
|
||||||
|
|
||||||
if ($update['operation'] == 'delete') {
|
|
||||||
$message = pht(
|
|
||||||
"DANGEROUS CHANGE: The change you're attempting to push deletes ".
|
|
||||||
"the branch '%s'.",
|
|
||||||
$update['ref.short']);
|
|
||||||
} else {
|
|
||||||
$message = pht(
|
|
||||||
"DANGEROUS CHANGE: The change you're attempting to push updates ".
|
|
||||||
"the branch '%s' from '%s' to '%s', but this is not a fast-forward. ".
|
|
||||||
"Pushes which rewrite published branch history are dangerous.",
|
|
||||||
$update['ref.short'],
|
|
||||||
$update['old.short'],
|
|
||||||
$update['new.short']);
|
|
||||||
}
|
|
||||||
|
|
||||||
$boilerplate = pht(
|
|
||||||
"Dangerous change protection is enabled for this repository.\n".
|
|
||||||
"Edit the repository configuration before making dangerous changes.");
|
|
||||||
|
|
||||||
$message = $message."\n".$boilerplate;
|
|
||||||
|
|
||||||
throw new DiffusionCommitHookRejectException($message);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private function executeSubversionHook() {
|
private function findMercurialContentUpdates(array $ref_updates) {
|
||||||
|
// TODO: Implement.
|
||||||
// TODO: Do useful things here, too.
|
return array();
|
||||||
|
|
||||||
return 0;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private function executeMercurialHook() {
|
|
||||||
|
|
||||||
// TODO: Here, too, useful things should be done.
|
/* -( Subversion )--------------------------------------------------------- */
|
||||||
|
|
||||||
return 0;
|
|
||||||
|
private function findSubversionRefUpdates() {
|
||||||
|
// TODO: Implement.
|
||||||
|
return array();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private function findSubversionContentUpdates(array $ref_updates) {
|
||||||
|
// TODO: Implement.
|
||||||
|
return array();
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/* -( Internals )---------------------------------------------------------- */
|
||||||
|
|
||||||
|
|
||||||
|
private function newPushLog() {
|
||||||
|
// NOTE: By default, we create these with REJECT_BROKEN as the reject
|
||||||
|
// code. This indicates a broken hook, and covers the case where we
|
||||||
|
// encounter some unexpected exception and consequently reject the changes.
|
||||||
|
|
||||||
|
return PhabricatorRepositoryPushLog::initializeNewLog($this->getViewer())
|
||||||
|
->attachRepository($this->getRepository())
|
||||||
|
->setRepositoryPHID($this->getRepository()->getPHID())
|
||||||
|
->setEpoch(time())
|
||||||
|
->setRemoteAddress($this->getRemoteAddressForLog())
|
||||||
|
->setRemoteProtocol($this->getRemoteProtocol())
|
||||||
|
->setTransactionKey($this->getTransactionKey())
|
||||||
|
->setRejectCode(PhabricatorRepositoryPushLog::REJECT_BROKEN)
|
||||||
|
->setRejectDetails(null);
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -18,6 +18,7 @@ final class PhabricatorRepositoryPushLog
|
||||||
const REFTYPE_BOOKMARK = 'bookmark';
|
const REFTYPE_BOOKMARK = 'bookmark';
|
||||||
const REFTYPE_SVN = 'svn';
|
const REFTYPE_SVN = 'svn';
|
||||||
const REFTYPE_COMMIT = 'commit';
|
const REFTYPE_COMMIT = 'commit';
|
||||||
|
const REFTYPE_UNKNOWN = 'unknown';
|
||||||
|
|
||||||
const CHANGEFLAG_ADD = 1;
|
const CHANGEFLAG_ADD = 1;
|
||||||
const CHANGEFLAG_DELETE = 2;
|
const CHANGEFLAG_DELETE = 2;
|
||||||
|
@ -29,6 +30,7 @@ final class PhabricatorRepositoryPushLog
|
||||||
const REJECT_DANGEROUS = 1;
|
const REJECT_DANGEROUS = 1;
|
||||||
const REJECT_HERALD = 2;
|
const REJECT_HERALD = 2;
|
||||||
const REJECT_EXTERNAL = 3;
|
const REJECT_EXTERNAL = 3;
|
||||||
|
const REJECT_BROKEN = 4;
|
||||||
|
|
||||||
protected $repositoryPHID;
|
protected $repositoryPHID;
|
||||||
protected $epoch;
|
protected $epoch;
|
||||||
|
@ -47,6 +49,7 @@ final class PhabricatorRepositoryPushLog
|
||||||
protected $rejectCode;
|
protected $rejectCode;
|
||||||
protected $rejectDetails;
|
protected $rejectDetails;
|
||||||
|
|
||||||
|
private $dangerousChangeDescription = self::ATTACHABLE;
|
||||||
private $repository = self::ATTACHABLE;
|
private $repository = self::ATTACHABLE;
|
||||||
|
|
||||||
public static function initializeNewLog(PhabricatorUser $viewer) {
|
public static function initializeNewLog(PhabricatorUser $viewer) {
|
||||||
|
@ -76,6 +79,16 @@ final class PhabricatorRepositoryPushLog
|
||||||
return phutil_utf8ize($this->getRefNameRaw());
|
return phutil_utf8ize($this->getRefNameRaw());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public function setRefName($ref_raw) {
|
||||||
|
$encoding = phutil_is_utf8($ref_raw) ? 'utf8' : null;
|
||||||
|
|
||||||
|
$this->setRefNameRaw($ref_raw);
|
||||||
|
$this->setRefNameHash(PhabricatorHash::digestForIndex($ref_raw));
|
||||||
|
$this->setRefNameEncoding($encoding);
|
||||||
|
|
||||||
|
return $this;
|
||||||
|
}
|
||||||
|
|
||||||
public function getRefOldShort() {
|
public function getRefOldShort() {
|
||||||
if ($this->getRepository()->isSVN()) {
|
if ($this->getRepository()->isSVN()) {
|
||||||
return $this->getRefOld();
|
return $this->getRefOld();
|
||||||
|
@ -90,6 +103,19 @@ final class PhabricatorRepositoryPushLog
|
||||||
return substr($this->getRefNew(), 0, 12);
|
return substr($this->getRefNew(), 0, 12);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public function hasChangeFlags($mask) {
|
||||||
|
return ($this->changeFlags & $mask);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function attachDangerousChangeDescription($description) {
|
||||||
|
$this->dangerousChangeDescription = $description;
|
||||||
|
return $this;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getDangerousChangeDescription() {
|
||||||
|
return $this->assertAttached($this->dangerousChangeDescription);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
/* -( PhabricatorPolicyInterface )----------------------------------------- */
|
/* -( PhabricatorPolicyInterface )----------------------------------------- */
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue