Page Menu
Home
Phabricator
Search
Configure Global Search
Log In
Files
F15376506
D9807.diff
No One
Temporary
Actions
View File
Edit File
Delete File
View Transforms
Subscribe
Mute Notifications
Award Token
Flag For Later
Size
7 KB
Referenced Files
None
Subscribers
None
D9807.diff
View Options
diff --git a/src/__phutil_library_map__.php b/src/__phutil_library_map__.php
--- a/src/__phutil_library_map__.php
+++ b/src/__phutil_library_map__.php
@@ -1008,6 +1008,7 @@
'HarbormasterQueryBuildablesConduitAPIMethod' => 'applications/harbormaster/conduit/HarbormasterQueryBuildablesConduitAPIMethod.php',
'HarbormasterQueryBuildsConduitAPIMethod' => 'applications/harbormaster/conduit/HarbormasterQueryBuildsConduitAPIMethod.php',
'HarbormasterRemarkupRule' => 'applications/harbormaster/remarkup/HarbormasterRemarkupRule.php',
+ 'HarbormasterRunBuildPlanBuildImplementation' => 'applications/harbormaster/step/HarbormasterRunBuildPlanBuildImplementation.php',
'HarbormasterRunBuildPlansHeraldAction' => 'applications/harbormaster/herald/HarbormasterRunBuildPlansHeraldAction.php',
'HarbormasterSchemaSpec' => 'applications/harbormaster/storage/HarbormasterSchemaSpec.php',
'HarbormasterScratchTable' => 'applications/harbormaster/storage/HarbormasterScratchTable.php',
@@ -4743,6 +4744,7 @@
'HarbormasterQueryBuildablesConduitAPIMethod' => 'HarbormasterConduitAPIMethod',
'HarbormasterQueryBuildsConduitAPIMethod' => 'HarbormasterConduitAPIMethod',
'HarbormasterRemarkupRule' => 'PhabricatorObjectRemarkupRule',
+ 'HarbormasterRunBuildPlanBuildImplementation' => 'HarbormasterBuildStepImplementation',
'HarbormasterRunBuildPlansHeraldAction' => 'HeraldAction',
'HarbormasterSchemaSpec' => 'PhabricatorConfigSchemaSpec',
'HarbormasterScratchTable' => 'HarbormasterDAO',
diff --git a/src/applications/harbormaster/step/HarbormasterRunBuildPlanBuildImplementation.php b/src/applications/harbormaster/step/HarbormasterRunBuildPlanBuildImplementation.php
new file mode 100644
--- /dev/null
+++ b/src/applications/harbormaster/step/HarbormasterRunBuildPlanBuildImplementation.php
@@ -0,0 +1,170 @@
+<?php
+
+final class HarbormasterRunBuildPlanBuildImplementation
+ extends HarbormasterBuildStepImplementation {
+
+ public function getName() {
+ return pht('Start or Wait for Build Plan');
+ }
+
+ public function getGenericDescription() {
+ return pht(
+ 'Start or wait for another build plan to finish on the same buildable.');
+ }
+
+ public function getDescription() {
+ $target_plan_id = $this->getSetting('id');
+
+ $target_plan = id(new HarbormasterBuildPlanQuery())
+ ->setViewer(PhabricatorUser::getOmnipotentUser())
+ ->withIDs(array($target_plan_id))
+ ->executeOne();
+
+ if ($target_plan === null) {
+ return pht(
+ 'Start and / or wait for a non-existent build plan '.
+ 'to finish on this buildable.');
+ }
+
+ $name = $this->formatValueForDescription($target_plan->getName());
+
+ return pht(
+ 'Start and / or wait for build plan %s to finish on this buildable.',
+ $name);
+ }
+
+ public function execute(
+ HarbormasterBuild $build,
+ HarbormasterBuildTarget $build_target) {
+
+ $target_plan_id = $this->getSetting('id');
+
+ // Get the target build plan type.
+ $target_plan = id(new HarbormasterBuildPlanQuery())
+ ->setViewer(PhabricatorUser::getOmnipotentUser())
+ ->withIDs(array($target_plan_id))
+ ->executeOne();
+ if (!$target_plan) {
+ throw new Exception('Unable to find build plan with ID '.$target_plan_id);
+ }
+
+ // If the target plan is the same as the current plan, we
+ // automatically pass.
+ if ($target_plan->getPHID() === $build->getBuildPlanPHID()) {
+ return;
+ }
+
+ // Find all other builds running on this buildable.
+ $buildable = $build->getBuildable();
+ $other_builds = id(new HarbormasterBuildQuery())
+ ->setViewer(PhabricatorUser::getOmnipotentUser())
+ ->withBuildablePHIDs(array($buildable->getPHID()))
+ ->execute();
+
+ // Find the build for that build plan (if it exists).
+ $target_build = null;
+ foreach ($other_builds as $other_build) {
+ if ($other_build->getBuildPlanPHID() === $target_plan->getPHID()) {
+ $target_build = $other_build;
+ break;
+ }
+ }
+
+ if ($target_build === null) {
+ // There is no current build with this build plan on the buildable,
+ // so now we're going to start one.
+ $target_build = $build->getBuildable()->applyPlan($target_plan);
+ } else {
+ // If the build plan has failed on a previous run, restart it.
+ switch ($target_build->getBuildStatus()) {
+ case HarbormasterBuild::STATUS_FAILED:
+ case HarbormasterBuild::STATUS_ERROR: {
+ $harbormaster_phid = id(new PhabricatorHarbormasterApplication())
+ ->getPHID();
+
+ $daemon_source = PhabricatorContentSource::newForSource(
+ PhabricatorContentSource::SOURCE_DAEMON,
+ array());
+
+ $editor = id(new HarbormasterBuildTransactionEditor())
+ ->setActor(PhabricatorUser::getOmnipotentUser())
+ ->setActingAsPHID($harbormaster_phid)
+ ->setContentSource($daemon_source)
+ ->setContinueOnNoEffect(true)
+ ->setContinueOnMissingFields(true);
+
+ $xaction = id(new HarbormasterBuildTransaction())
+ ->setTransactionType(HarbormasterBuildTransaction::TYPE_COMMAND)
+ ->setNewValue(HarbormasterBuildCommand::COMMAND_RESTART);
+
+ $editor->applyTransactions($target_build, array($xaction));
+
+ break;
+ }
+ }
+ }
+
+ $artifact = $build->createArtifact(
+ $build_target,
+ 'build_plan_'.$build_target->getID(),
+ HarbormasterBuildArtifact::TYPE_URI);
+ $artifact->setArtifactData(array(
+ 'name' => 'Target Build Plan',
+ 'uri' => '/harbormaster/plan/'.$target_plan->getID(),
+ ));
+ $artifact->save();
+
+ $artifact = $build->createArtifact(
+ $build_target,
+ 'build_'.$build_target->getID(),
+ HarbormasterBuildArtifact::TYPE_URI);
+ $artifact->setArtifactData(array(
+ 'name' => 'Target Build',
+ 'uri' => '/harbormaster/build/'.$target_build->getID(),
+ ));
+ $artifact->save();
+
+ $target_build_phid = $target_build->getPHID();
+
+ do {
+ $target_build = id(new HarbormasterBuildQuery())
+ ->setViewer(PhabricatorUser::getOmnipotentUser())
+ ->withPHIDs(array($target_build_phid))
+ ->executeOne();
+
+ if (!$target_build) {
+ throw new Exception('Build has disappeared!');
+ }
+
+ // If the build has not yet passed, yield for the moment.
+ if ($target_build->isBuilding() || $target_build->isRestarting() ||
+ $target_build->getBuildStatus() === HarbormasterBuild::STATUS_STOPPED) {
+ sleep(15);
+ continue;
+ }
+
+ // Otherwise check the result and if it's an error or failure result,
+ // then throw an exception.
+ switch ($target_build->getBuildStatus()) {
+ case HarbormasterBuild::STATUS_FAILED:
+ case HarbormasterBuild::STATUS_ERROR:
+ throw new HarbormasterBuildFailureException();
+ }
+
+ // Pass if we get to here.
+ break;
+ } while (true);
+ }
+
+ public function getFieldSpecifications() {
+ return array(
+ 'id' => array(
+ 'name' => pht('Target Build Plan ID'),
+ 'type' => 'text',
+ 'required' => true,
+ ),
+ );
+ }
+
+
+}
File Metadata
Details
Attached
Mime Type
text/plain
Expires
Fri, Mar 14, 4:10 AM (3 w, 19 h ago)
Storage Engine
blob
Storage Format
Encrypted (AES-256-CBC)
Storage Handle
7644967
Default Alt Text
D9807.diff (7 KB)
Attached To
Mode
D9807: [harbormaster/run-build-plan] Implement build step for running another build plan
Attached
Detach File
Event Timeline
Log In to Comment