Add a UI element for reviewing older generations of Harbormaster builds
Summary:
See PHI446. Ref T13088. Currently, there's no way to access older generations of a build unless you know the secret `?g=1` URI magic.
When a build has multiple generations, show a history table and let users click to see older run information.
This is currently very basic. It would be nice to show when each generation started, who started/restarted it, and what the build status was at the time the build was restarted. There's currently no convenient source for this information so just add a bare-bones, working version of this for now.
Test Plan:
Viewed pending, single-run and multi-restart builds. Saw table on builds with more than one generation. Clicked table entries to see different build data.
{F5471160}
Subscribers: PHID-OPKG-gm6ozazyms6q6i22gyam
Maniphest Tasks: T13088
Differential Revision: https://secure.phabricator.com/D19217
			
			
This commit is contained in:
		| @@ -71,7 +71,8 @@ final class PhabricatorHarbormasterApplication extends PhabricatorApplication { | |||||||
|         ), |         ), | ||||||
|         'build/' => array( |         'build/' => array( | ||||||
|           $this->getQueryRoutePattern() => 'HarbormasterBuildListController', |           $this->getQueryRoutePattern() => 'HarbormasterBuildListController', | ||||||
|           '(?P<id>\d+)/' => 'HarbormasterBuildViewController', |           '(?P<id>\d+)/(?:(?P<generation>\d+)/)?' | ||||||
|  |             => 'HarbormasterBuildViewController', | ||||||
|           '(?P<action>pause|resume|restart|abort)/'. |           '(?P<action>pause|resume|restart|abort)/'. | ||||||
|             '(?P<id>\d+)/(?:(?P<via>[^/]+)/)?' |             '(?P<id>\d+)/(?:(?P<via>[^/]+)/)?' | ||||||
|             => 'HarbormasterBuildActionController', |             => 'HarbormasterBuildActionController', | ||||||
|   | |||||||
| @@ -8,7 +8,6 @@ final class HarbormasterBuildViewController | |||||||
|     $viewer = $request->getUser(); |     $viewer = $request->getUser(); | ||||||
|  |  | ||||||
|     $id = $request->getURIData('id'); |     $id = $request->getURIData('id'); | ||||||
|     $generation = $request->getInt('g'); |  | ||||||
|  |  | ||||||
|     $build = id(new HarbormasterBuildQuery()) |     $build = id(new HarbormasterBuildQuery()) | ||||||
|       ->setViewer($viewer) |       ->setViewer($viewer) | ||||||
| @@ -21,6 +20,7 @@ final class HarbormasterBuildViewController | |||||||
|     require_celerity_resource('harbormaster-css'); |     require_celerity_resource('harbormaster-css'); | ||||||
|  |  | ||||||
|     $title = pht('Build %d', $id); |     $title = pht('Build %d', $id); | ||||||
|  |     $warnings = array(); | ||||||
|  |  | ||||||
|     $page_header = id(new PHUIHeaderView()) |     $page_header = id(new PHUIHeaderView()) | ||||||
|       ->setHeader($title) |       ->setHeader($title) | ||||||
| @@ -28,7 +28,9 @@ final class HarbormasterBuildViewController | |||||||
|       ->setPolicyObject($build) |       ->setPolicyObject($build) | ||||||
|       ->setHeaderIcon('fa-cubes'); |       ->setHeaderIcon('fa-cubes'); | ||||||
|  |  | ||||||
|     if ($build->isRestarting()) { |     $is_restarting = $build->isRestarting(); | ||||||
|  |  | ||||||
|  |     if ($is_restarting) { | ||||||
|       $page_header->setStatus( |       $page_header->setStatus( | ||||||
|         'fa-exclamation-triangle', 'red', pht('Restarting')); |         'fa-exclamation-triangle', 'red', pht('Restarting')); | ||||||
|     } else if ($build->isPausing()) { |     } else if ($build->isPausing()) { | ||||||
| @@ -42,19 +44,53 @@ final class HarbormasterBuildViewController | |||||||
|         'fa-exclamation-triangle', 'red', pht('Aborting')); |         'fa-exclamation-triangle', 'red', pht('Aborting')); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     $max_generation = (int)$build->getBuildGeneration(); | ||||||
|  |     if ($max_generation === 0) { | ||||||
|  |       $min_generation = 0; | ||||||
|  |     } else { | ||||||
|  |       $min_generation = 1; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     if ($is_restarting) { | ||||||
|  |       $max_generation = $max_generation + 1; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     $generation = $request->getURIData('generation'); | ||||||
|  |     if ($generation === null) { | ||||||
|  |       $generation = $max_generation; | ||||||
|  |     } else { | ||||||
|  |       $generation = (int)$generation; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     if ($generation < $min_generation || $generation > $max_generation) { | ||||||
|  |       return new Aphront404Response(); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     if ($generation < $max_generation) { | ||||||
|  |       $warnings[] = pht( | ||||||
|  |         'You are viewing an older run of this build. %s', | ||||||
|  |         phutil_tag( | ||||||
|  |           'a', | ||||||
|  |           array( | ||||||
|  |             'href' => $build->getURI(), | ||||||
|  |           ), | ||||||
|  |           pht('View Current Build'))); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |  | ||||||
|     $curtain = $this->buildCurtainView($build); |     $curtain = $this->buildCurtainView($build); | ||||||
|     $properties = $this->buildPropertyList($build); |     $properties = $this->buildPropertyList($build); | ||||||
|  |     $history = $this->buildHistoryTable( | ||||||
|  |       $build, | ||||||
|  |       $generation, | ||||||
|  |       $min_generation, | ||||||
|  |       $max_generation); | ||||||
|  |  | ||||||
|     $crumbs = $this->buildApplicationCrumbs(); |     $crumbs = $this->buildApplicationCrumbs(); | ||||||
|     $this->addBuildableCrumb($crumbs, $build->getBuildable()); |     $this->addBuildableCrumb($crumbs, $build->getBuildable()); | ||||||
|     $crumbs->addTextCrumb($title); |     $crumbs->addTextCrumb($title); | ||||||
|     $crumbs->setBorder(true); |     $crumbs->setBorder(true); | ||||||
|  |  | ||||||
|     if ($generation === null || $generation > $build->getBuildGeneration() || |  | ||||||
|       $generation < 0) { |  | ||||||
|       $generation = $build->getBuildGeneration(); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     $build_targets = id(new HarbormasterBuildTargetQuery()) |     $build_targets = id(new HarbormasterBuildTargetQuery()) | ||||||
|       ->setViewer($viewer) |       ->setViewer($viewer) | ||||||
|       ->needBuildSteps(true) |       ->needBuildSteps(true) | ||||||
| @@ -264,14 +300,25 @@ final class HarbormasterBuildViewController | |||||||
|       new HarbormasterBuildTransactionQuery()); |       new HarbormasterBuildTransactionQuery()); | ||||||
|     $timeline->setShouldTerminate(true); |     $timeline->setShouldTerminate(true); | ||||||
|  |  | ||||||
|  |     if ($warnings) { | ||||||
|  |       $warnings = id(new PHUIInfoView()) | ||||||
|  |         ->setErrors($warnings) | ||||||
|  |         ->setSeverity(PHUIInfoView::SEVERITY_WARNING); | ||||||
|  |     } else { | ||||||
|  |       $warnings = null; | ||||||
|  |     } | ||||||
|  |  | ||||||
|     $view = id(new PHUITwoColumnView()) |     $view = id(new PHUITwoColumnView()) | ||||||
|       ->setHeader($page_header) |       ->setHeader($page_header) | ||||||
|       ->setCurtain($curtain) |       ->setCurtain($curtain) | ||||||
|       ->setMainColumn(array( |       ->setMainColumn( | ||||||
|         $properties, |         array( | ||||||
|         $targets, |           $warnings, | ||||||
|         $timeline, |           $properties, | ||||||
|       )); |           $history, | ||||||
|  |           $targets, | ||||||
|  |           $timeline, | ||||||
|  |         )); | ||||||
|  |  | ||||||
|     return $this->newPage() |     return $this->newPage() | ||||||
|       ->setTitle($title) |       ->setTitle($title) | ||||||
| @@ -561,10 +608,6 @@ final class HarbormasterBuildViewController | |||||||
|       pht('Build Plan'), |       pht('Build Plan'), | ||||||
|       $handles[$build->getBuildPlanPHID()]->renderLink()); |       $handles[$build->getBuildPlanPHID()]->renderLink()); | ||||||
|  |  | ||||||
|     $properties->addProperty( |  | ||||||
|       pht('Restarts'), |  | ||||||
|       $build->getBuildGeneration()); |  | ||||||
|  |  | ||||||
|     $properties->addProperty( |     $properties->addProperty( | ||||||
|       pht('Status'), |       pht('Status'), | ||||||
|       $this->getStatus($build)); |       $this->getStatus($build)); | ||||||
| @@ -576,6 +619,53 @@ final class HarbormasterBuildViewController | |||||||
|  |  | ||||||
|   } |   } | ||||||
|  |  | ||||||
|  |   private function buildHistoryTable( | ||||||
|  |     HarbormasterBuild $build, | ||||||
|  |     $generation, | ||||||
|  |     $min_generation, | ||||||
|  |     $max_generation) { | ||||||
|  |  | ||||||
|  |     if ($max_generation === $min_generation) { | ||||||
|  |       return null; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     $viewer = $this->getViewer(); | ||||||
|  |  | ||||||
|  |     $uri = $build->getURI(); | ||||||
|  |  | ||||||
|  |     $rows = array(); | ||||||
|  |     $rowc = array(); | ||||||
|  |     for ($ii = $max_generation; $ii >= $min_generation; $ii--) { | ||||||
|  |       if ($generation == $ii) { | ||||||
|  |         $rowc[] = 'highlighted'; | ||||||
|  |       } else { | ||||||
|  |         $rowc[] = null; | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       $rows[] = array( | ||||||
|  |         phutil_tag( | ||||||
|  |           'a', | ||||||
|  |           array( | ||||||
|  |             'href' => $uri.$ii.'/', | ||||||
|  |           ), | ||||||
|  |           pht('Run %d', $ii)), | ||||||
|  |       ); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     $table = id(new AphrontTableView($rows)) | ||||||
|  |       ->setColumnClasses( | ||||||
|  |         array( | ||||||
|  |           'pri wide', | ||||||
|  |         )) | ||||||
|  |       ->setRowClasses($rowc); | ||||||
|  |  | ||||||
|  |     return id(new PHUIObjectBoxView()) | ||||||
|  |       ->setHeaderText(pht('History')) | ||||||
|  |       ->setBackground(PHUIObjectBoxView::BLUE_PROPERTY) | ||||||
|  |       ->setTable($table); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |  | ||||||
|   private function getStatus(HarbormasterBuild $build) { |   private function getStatus(HarbormasterBuild $build) { | ||||||
|     $status_view = new PHUIStatusListView(); |     $status_view = new PHUIStatusListView(); | ||||||
|  |  | ||||||
|   | |||||||
| @@ -147,7 +147,7 @@ final class HarbormasterBuildEngine extends Phobject { | |||||||
|     // If it is different, they will automatically stop what they're doing |     // If it is different, they will automatically stop what they're doing | ||||||
|     // and abort. |     // and abort. | ||||||
|  |  | ||||||
|     // Previously we used to delete targets, logs and artifacts here.  Instead |     // Previously we used to delete targets, logs and artifacts here. Instead, | ||||||
|     // leave them around so users can view previous generations of this build. |     // leave them around so users can view previous generations of this build. | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user
	 epriestley
					epriestley