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

Use herald to trigger builds of revisions and commits.

Summary:
Depends on D7500.

This seemed like a pretty good idea once I thought of it.  Instead of having some custom triggering logic instead Harbormaster, I figured it best to leverage all of Herald's power so that users can create rules to apply builds to commits and differential revisions.  This gives the added advantage that they can trigger off builds for particular types of revisions and commits, which seems like it could be really useful (e.g. run extra tests against revisions that touch sensitive areas of the code).

Test Plan: Ran the usual daemons + the Harbormaster daemon.  Pushed a commit to the repository and saw both the buildable and build get created when the commit worked picked it up.  Submitted a diff and saw both the buildable and build get created when the Herald rules were evaluated for the diff.

Reviewers: epriestley, #blessed_reviewers

Reviewed By: epriestley

CC: Korvin, epriestley, aran, hwinkel

Maniphest Tasks: T1049

Differential Revision: https://secure.phabricator.com/D7501
This commit is contained in:
James Rhodes 2013-11-08 16:48:17 -08:00 committed by epriestley
parent 3de4e91dac
commit 7ffea0463e
13 changed files with 194 additions and 26 deletions

View file

@ -273,6 +273,11 @@ final class DifferentialRevisionEditor extends PhabricatorEditor {
$rem_ccs = $adapter->getCCsRemovedByHerald(); $rem_ccs = $adapter->getCCsRemovedByHerald();
$blocking_reviewers = array_keys( $blocking_reviewers = array_keys(
$adapter->getBlockingReviewersAddedByHerald()); $adapter->getBlockingReviewersAddedByHerald());
HarbormasterBuildable::applyBuildPlans(
$diff->getPHID(),
$revision->getPHID(),
$adapter->getBuildPlans());
} else { } else {
$sub = array( $sub = array(
'rev' => array(), 'rev' => array(),

View file

@ -31,17 +31,10 @@ final class HarbormasterBuildableApplyController
->withIDs(array($request->getInt('build-plan'))) ->withIDs(array($request->getInt('build-plan')))
->executeOne(); ->executeOne();
$build = HarbormasterBuild::initializeNewBuild($viewer); HarbormasterBuildable::applyBuildPlans(
$build->setBuildablePHID($buildable->getPHID()); $buildable->getBuildablePHID(),
$build->setBuildPlanPHID($plan->getPHID()); $buildable->getContainerPHID(),
$build->setBuildStatus(HarbormasterBuild::STATUS_PENDING); array($plan->getPHID()));
$build->save();
PhabricatorWorker::scheduleTask(
'HarbormasterBuildWorker',
array(
'buildID' => $build->getID()
));
return id(new AphrontRedirectResponse())->setURI($buildable_uri); return id(new AphrontRedirectResponse())->setURI($buildable_uri);
} }

View file

@ -19,7 +19,7 @@ final class HarbormasterBuildableViewController
->setViewer($viewer) ->setViewer($viewer)
->withIDs(array($id)) ->withIDs(array($id))
->needBuildableHandles(true) ->needBuildableHandles(true)
->needContainerObjects(true) ->needContainerHandles(true)
->executeOne(); ->executeOne();
if (!$buildable) { if (!$buildable) {
return new Aphront404Response(); return new Aphront404Response();
@ -139,6 +139,12 @@ final class HarbormasterBuildableViewController
pht('Buildable'), pht('Buildable'),
$buildable->getBuildableHandle()->renderLink()); $buildable->getBuildableHandle()->renderLink());
if ($buildable->getContainerHandle() !== null) {
$properties->addProperty(
pht('Container'),
$buildable->getContainerHandle()->renderLink());
}
} }
} }

View file

@ -31,6 +31,8 @@ final class HarbormasterPHIDTypeBuildPlan extends PhabricatorPHIDType {
foreach ($handles as $phid => $handle) { foreach ($handles as $phid => $handle) {
$build_plan = $objects[$phid]; $build_plan = $objects[$phid];
$handles[$phid]->setName($build_plan->getName());
$handles[$phid]->setURI('/harbormaster/plan/'.$build_plan->getID());
} }
} }

View file

@ -9,6 +9,7 @@ final class HarbormasterBuildableQuery
private $containerPHIDs; private $containerPHIDs;
private $needContainerObjects; private $needContainerObjects;
private $needContainerHandles;
private $needBuildableHandles; private $needBuildableHandles;
private $needBuilds; private $needBuilds;
@ -37,6 +38,11 @@ final class HarbormasterBuildableQuery
return $this; return $this;
} }
public function needContainerHandles($need) {
$this->needContainerHandles = $need;
return $this;
}
public function needBuildableHandles($need) { public function needBuildableHandles($need) {
$this->needBuildableHandles = $need; $this->needBuildableHandles = $need;
return $this; return $this;
@ -88,10 +94,12 @@ final class HarbormasterBuildableQuery
} }
protected function didFilterPage(array $page) { protected function didFilterPage(array $page) {
if ($this->needContainerObjects || $this->needContainerHandles) {
$container_phids = array_filter(mpull($page, 'getContainerPHID'));
if ($this->needContainerObjects) { if ($this->needContainerObjects) {
$containers = array(); $containers = array();
$container_phids = array_filter(mpull($page, 'getContainerPHID'));
if ($container_phids) { if ($container_phids) {
$containers = id(new PhabricatorObjectQuery()) $containers = id(new PhabricatorObjectQuery())
->setViewer($this->getViewer()) ->setViewer($this->getViewer())
@ -107,6 +115,24 @@ final class HarbormasterBuildableQuery
} }
} }
if ($this->needContainerHandles) {
$handles = array();
if ($container_phids) {
$handles = id(new PhabricatorHandleQuery())
->setViewer($this->getViewer())
->withPHIDs($container_phids)
->setParentQuery($this)
->execute();
}
foreach ($page as $key => $buildable) {
$container_phid = $buildable->getContainerPHID();
$buildable->attachContainerHandle(idx($handles, $container_phid));
}
}
}
if ($this->needBuildableHandles) { if ($this->needBuildableHandles) {
$handles = array(); $handles = array();

View file

@ -11,6 +11,7 @@ final class HarbormasterBuildable extends HarbormasterDAO
private $buildableObject = self::ATTACHABLE; private $buildableObject = self::ATTACHABLE;
private $containerObject = self::ATTACHABLE; private $containerObject = self::ATTACHABLE;
private $buildableHandle = self::ATTACHABLE; private $buildableHandle = self::ATTACHABLE;
private $containerHandle = self::ATTACHABLE;
private $builds = self::ATTACHABLE; private $builds = self::ATTACHABLE;
const STATUS_WHATEVER = 'whatever'; const STATUS_WHATEVER = 'whatever';
@ -21,6 +22,75 @@ final class HarbormasterBuildable extends HarbormasterDAO
->setBuildableStatus(self::STATUS_WHATEVER); ->setBuildableStatus(self::STATUS_WHATEVER);
} }
/**
* Returns an existing buildable for the object's PHID or creates a
* new buildable implicitly if needed.
*/
public static function createOrLoadExisting(
PhabricatorUser $actor,
$buildable_object_phid,
$container_object_phid) {
$buildable = id(new HarbormasterBuildableQuery())
->setViewer($actor)
->withBuildablePHIDs(array($buildable_object_phid))
->executeOne();
if ($buildable) {
return $buildable;
}
$buildable = HarbormasterBuildable::initializeNewBuildable($actor)
->setBuildablePHID($buildable_object_phid)
->setContainerPHID($container_object_phid);
$buildable->save();
return $buildable;
}
/**
* Looks up the plan PHIDs and applies the plans to the specified
* object identified by it's PHID.
*/
public static function applyBuildPlans(
$phid,
$container_phid,
array $plan_phids) {
if (count($plan_phids) === 0) {
return;
}
// Skip all of this logic if the Harbormaster application
// isn't currently installed.
$harbormaster_app = 'PhabricatorApplicationHarbormaster';
if (!PhabricatorApplication::isClassInstalled($harbormaster_app)) {
return;
}
$buildable = HarbormasterBuildable::createOrLoadExisting(
PhabricatorUser::getOmnipotentUser(),
$phid,
$container_phid);
$plans = id(new HarbormasterBuildPlanQuery())
->setViewer(PhabricatorUser::getOmnipotentUser())
->withPHIDs($plan_phids)
->execute();
foreach ($plans as $plan) {
$build = HarbormasterBuild::initializeNewBuild(
PhabricatorUser::getOmnipotentUser());
$build->setBuildablePHID($buildable->getPHID());
$build->setBuildPlanPHID($plan->getPHID());
$build->setBuildStatus(HarbormasterBuild::STATUS_PENDING);
$build->save();
PhabricatorWorker::scheduleTask(
'HarbormasterBuildWorker',
array(
'buildID' => $build->getID()
));
}
}
public function getConfiguration() { public function getConfiguration() {
return array( return array(
self::CONFIG_AUX_PHID => true, self::CONFIG_AUX_PHID => true,
@ -50,6 +120,15 @@ final class HarbormasterBuildable extends HarbormasterDAO
return $this->assertAttached($this->containerObject); return $this->assertAttached($this->containerObject);
} }
public function attachContainerHandle($container_handle) {
$this->containerHandle = $container_handle;
return $this;
}
public function getContainerHandle() {
return $this->assertAttached($this->containerHandle);
}
public function attachBuildableHandle($buildable_handle) { public function attachBuildableHandle($buildable_handle) {
$this->buildableHandle = $buildable_handle; $this->buildableHandle = $buildable_handle;
return $this; return $this;

View file

@ -54,6 +54,7 @@ abstract class HeraldAdapter {
const ACTION_ADD_PROJECTS = 'addprojects'; const ACTION_ADD_PROJECTS = 'addprojects';
const ACTION_ADD_REVIEWERS = 'addreviewers'; const ACTION_ADD_REVIEWERS = 'addreviewers';
const ACTION_ADD_BLOCKING_REVIEWERS = 'addblockingreviewers'; const ACTION_ADD_BLOCKING_REVIEWERS = 'addblockingreviewers';
const ACTION_APPLY_BUILD_PLANS = 'applybuildplans';
const VALUE_TEXT = 'text'; const VALUE_TEXT = 'text';
const VALUE_NONE = 'none'; const VALUE_NONE = 'none';
@ -67,6 +68,7 @@ abstract class HeraldAdapter {
const VALUE_FLAG_COLOR = 'flagcolor'; const VALUE_FLAG_COLOR = 'flagcolor';
const VALUE_CONTENT_SOURCE = 'contentsource'; const VALUE_CONTENT_SOURCE = 'contentsource';
const VALUE_USER_OR_PROJECT = 'userorproject'; const VALUE_USER_OR_PROJECT = 'userorproject';
const VALUE_BUILD_PLAN = 'buildplan';
private $contentSource; private $contentSource;
@ -490,6 +492,7 @@ abstract class HeraldAdapter {
self::ACTION_ADD_PROJECTS => pht('Add projects'), self::ACTION_ADD_PROJECTS => pht('Add projects'),
self::ACTION_ADD_REVIEWERS => pht('Add reviewers'), self::ACTION_ADD_REVIEWERS => pht('Add reviewers'),
self::ACTION_ADD_BLOCKING_REVIEWERS => pht('Add blocking reviewers'), self::ACTION_ADD_BLOCKING_REVIEWERS => pht('Add blocking reviewers'),
self::ACTION_APPLY_BUILD_PLANS => pht('Apply build plans'),
); );
case HeraldRuleTypeConfig::RULE_TYPE_PERSONAL: case HeraldRuleTypeConfig::RULE_TYPE_PERSONAL:
return array( return array(
@ -657,6 +660,8 @@ abstract class HeraldAdapter {
case self::ACTION_ADD_REVIEWERS: case self::ACTION_ADD_REVIEWERS:
case self::ACTION_ADD_BLOCKING_REVIEWERS: case self::ACTION_ADD_BLOCKING_REVIEWERS:
return self::VALUE_USER_OR_PROJECT; return self::VALUE_USER_OR_PROJECT;
case self::ACTION_APPLY_BUILD_PLANS:
return self::VALUE_BUILD_PLAN;
default: default:
throw new Exception("Unknown or invalid action '{$action}'."); throw new Exception("Unknown or invalid action '{$action}'.");
} }

View file

@ -22,6 +22,7 @@ final class HeraldCommitAdapter extends HeraldAdapter {
protected $emailPHIDs = array(); protected $emailPHIDs = array();
protected $addCCPHIDs = array(); protected $addCCPHIDs = array();
protected $auditMap = array(); protected $auditMap = array();
protected $buildPlans = array();
protected $affectedPaths; protected $affectedPaths;
protected $affectedRevision; protected $affectedRevision;
@ -120,7 +121,8 @@ final class HeraldCommitAdapter extends HeraldAdapter {
self::ACTION_ADD_CC, self::ACTION_ADD_CC,
self::ACTION_EMAIL, self::ACTION_EMAIL,
self::ACTION_AUDIT, self::ACTION_AUDIT,
self::ACTION_NOTHING, self::ACTION_APPLY_BUILD_PLANS,
self::ACTION_NOTHING
); );
case HeraldRuleTypeConfig::RULE_TYPE_PERSONAL: case HeraldRuleTypeConfig::RULE_TYPE_PERSONAL:
return array( return array(
@ -176,6 +178,10 @@ final class HeraldCommitAdapter extends HeraldAdapter {
return $this->auditMap; return $this->auditMap;
} }
public function getBuildPlans() {
return $this->buildPlans;
}
public function getHeraldName() { public function getHeraldName() {
return return
'r'. 'r'.
@ -422,6 +428,15 @@ final class HeraldCommitAdapter extends HeraldAdapter {
true, true,
pht('Triggered an audit.')); pht('Triggered an audit.'));
break; break;
case self::ACTION_APPLY_BUILD_PLANS:
foreach ($effect->getTarget() as $phid) {
$this->buildPlans[] = $phid;
}
$result[] = new HeraldApplyTranscript(
$effect,
true,
pht('Applied build plans.'));
break;
case self::ACTION_FLAG: case self::ACTION_FLAG:
$result[] = parent::applyFlagEffect( $result[] = parent::applyFlagEffect(
$effect, $effect,

View file

@ -14,6 +14,7 @@ final class HeraldDifferentialRevisionAdapter extends HeraldAdapter {
protected $emailPHIDs = array(); protected $emailPHIDs = array();
protected $addReviewerPHIDs = array(); protected $addReviewerPHIDs = array();
protected $blockingReviewerPHIDs = array(); protected $blockingReviewerPHIDs = array();
protected $buildPlans = array();
protected $repository; protected $repository;
protected $affectedPackages; protected $affectedPackages;
@ -117,6 +118,10 @@ final class HeraldDifferentialRevisionAdapter extends HeraldAdapter {
return $this->blockingReviewerPHIDs; return $this->blockingReviewerPHIDs;
} }
public function getBuildPlans() {
return $this->buildPlans;
}
public function getPHID() { public function getPHID() {
return $this->revision->getPHID(); return $this->revision->getPHID();
} }
@ -349,6 +354,7 @@ final class HeraldDifferentialRevisionAdapter extends HeraldAdapter {
self::ACTION_EMAIL, self::ACTION_EMAIL,
self::ACTION_ADD_REVIEWERS, self::ACTION_ADD_REVIEWERS,
self::ACTION_ADD_BLOCKING_REVIEWERS, self::ACTION_ADD_BLOCKING_REVIEWERS,
self::ACTION_APPLY_BUILD_PLANS,
self::ACTION_NOTHING, self::ACTION_NOTHING,
); );
case HeraldRuleTypeConfig::RULE_TYPE_PERSONAL: case HeraldRuleTypeConfig::RULE_TYPE_PERSONAL:
@ -468,6 +474,15 @@ final class HeraldDifferentialRevisionAdapter extends HeraldAdapter {
true, true,
pht('Added blocking reviewers.')); pht('Added blocking reviewers.'));
break; break;
case self::ACTION_APPLY_BUILD_PLANS:
foreach ($effect->getTarget() as $phid) {
$this->buildPlans[] = $phid;
}
$result[] = new HeraldApplyTranscript(
$effect,
true,
pht('Applied build plans.'));
break;
default: default:
throw new Exception("No rules to handle action '{$action}'."); throw new Exception("No rules to handle action '{$action}'.");
} }

View file

@ -511,6 +511,7 @@ final class HeraldRuleController extends HeraldController {
'package' => '/typeahead/common/packages/', 'package' => '/typeahead/common/packages/',
'project' => '/typeahead/common/projects/', 'project' => '/typeahead/common/projects/',
'userorproject' => '/typeahead/common/accountsorprojects/', 'userorproject' => '/typeahead/common/accountsorprojects/',
'buildplan' => '/typeahead/common/buildplans/',
), ),
'markup' => $template, 'markup' => $template,
); );

View file

@ -66,6 +66,11 @@ final class PhabricatorRepositoryCommitHeraldWorker
$this->createAudits($commit, $audit_phids, $cc_phids, $rules); $this->createAudits($commit, $audit_phids, $cc_phids, $rules);
} }
HarbormasterBuildable::applyBuildPlans(
$commit->getPHID(),
$repository->getPHID(),
$adapter->getBuildPlans());
$explicit_auditors = $this->createAuditsFromCommitMessage($commit, $data); $explicit_auditors = $this->createAuditsFromCommitMessage($commit, $data);
if ($repository->getDetail('herald-disabled')) { if ($repository->getDetail('herald-disabled')) {

View file

@ -35,6 +35,7 @@ final class PhabricatorTypeaheadCommonDatasourceController
$need_noproject = false; $need_noproject = false;
$need_symbols = false; $need_symbols = false;
$need_jump_objects = false; $need_jump_objects = false;
$need_build_plans = false;
switch ($this->type) { switch ($this->type) {
case 'mainsearch': case 'mainsearch':
$need_users = true; $need_users = true;
@ -93,6 +94,9 @@ final class PhabricatorTypeaheadCommonDatasourceController
case 'arcanistprojects': case 'arcanistprojects':
$need_arcanist_projects = true; $need_arcanist_projects = true;
break; break;
case 'buildplans':
$need_build_plans = true;
break;
} }
$results = array(); $results = array();
@ -219,6 +223,17 @@ final class PhabricatorTypeaheadCommonDatasourceController
} }
} }
if ($need_build_plans) {
$plans = id(new HarbormasterBuildPlanQuery())
->setViewer($viewer)
->execute();
foreach ($plans as $plan) {
$results[] = id(new PhabricatorTypeaheadResult())
->setName($plan->getName())
->setPHID($plan->getPHID());
}
}
if ($need_projs) { if ($need_projs) {
$projs = id(new PhabricatorProjectQuery()) $projs = id(new PhabricatorProjectQuery())
->setViewer($viewer) ->setViewer($viewer)

View file

@ -221,6 +221,7 @@ JX.install('HeraldRuleEditor', {
case 'package': case 'package':
case 'project': case 'project':
case 'userorproject': case 'userorproject':
case 'buildplan':
var tokenizer = this._newTokenizer(type); var tokenizer = this._newTokenizer(type);
input = tokenizer[0]; input = tokenizer[0];
get_fn = tokenizer[1]; get_fn = tokenizer[1];