mirror of
https://we.phorge.it/source/phorge.git
synced 2025-01-19 19:21:10 +01:00
Make Drydock blueprint create workflow somewhat more standard
Summary: Ref T2015. This workflow is a little weird (runs in a dialog, no edit-before-create step, lots of internal classnames). Make it a little more standard. Test Plan: See screenshots. Reviewers: btrahan Reviewed By: btrahan CC: aran Maniphest Tasks: T2015 Differential Revision: https://secure.phabricator.com/D7908
This commit is contained in:
parent
324ee4e4d5
commit
5502fca5f4
9 changed files with 126 additions and 79 deletions
|
@ -38,7 +38,7 @@ final class PhabricatorApplicationDrydock extends PhabricatorApplication {
|
||||||
'(?:query/(?P<queryKey>[^/]+)/)?' => 'DrydockBlueprintListController',
|
'(?:query/(?P<queryKey>[^/]+)/)?' => 'DrydockBlueprintListController',
|
||||||
'(?P<id>[1-9]\d*)/' => 'DrydockBlueprintViewController',
|
'(?P<id>[1-9]\d*)/' => 'DrydockBlueprintViewController',
|
||||||
'create/' => 'DrydockBlueprintCreateController',
|
'create/' => 'DrydockBlueprintCreateController',
|
||||||
'edit/(?P<id>[1-9]\d*)/' => 'DrydockBlueprintEditController',
|
'edit/(?:(?P<id>[1-9]\d*)/)?' => 'DrydockBlueprintEditController',
|
||||||
),
|
),
|
||||||
'resource/' => array(
|
'resource/' => array(
|
||||||
'(?:query/(?P<queryKey>[^/]+)/)?' => 'DrydockResourceListController',
|
'(?:query/(?P<queryKey>[^/]+)/)?' => 'DrydockResourceListController',
|
||||||
|
|
|
@ -19,6 +19,7 @@ abstract class DrydockBlueprintImplementation {
|
||||||
|
|
||||||
abstract public function isEnabled();
|
abstract public function isEnabled();
|
||||||
|
|
||||||
|
abstract public function getBlueprintName();
|
||||||
abstract public function getDescription();
|
abstract public function getDescription();
|
||||||
|
|
||||||
public function getBlueprintClass() {
|
public function getBlueprintClass() {
|
||||||
|
@ -388,6 +389,10 @@ abstract class DrydockBlueprintImplementation {
|
||||||
return idx($groups, $type, array());
|
return idx($groups, $type, array());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public static function getNamedImplementation($class) {
|
||||||
|
return idx(self::getAllBlueprintImplementations(), $class);
|
||||||
|
}
|
||||||
|
|
||||||
protected function newResourceTemplate($name) {
|
protected function newResourceTemplate($name) {
|
||||||
$resource = id(new DrydockResource())
|
$resource = id(new DrydockResource())
|
||||||
->setBlueprintPHID($this->getInstance()->getPHID())
|
->setBlueprintPHID($this->getInstance()->getPHID())
|
||||||
|
|
|
@ -7,8 +7,13 @@ final class DrydockLocalHostBlueprintImplementation
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public function getBlueprintName() {
|
||||||
|
return pht('Local Host');
|
||||||
|
}
|
||||||
|
|
||||||
public function getDescription() {
|
public function getDescription() {
|
||||||
return pht('Allocates storage on the local host.');
|
return pht(
|
||||||
|
'Allows Drydock to run on the local host.');
|
||||||
}
|
}
|
||||||
|
|
||||||
public function canAllocateMoreResources(array $pool) {
|
public function canAllocateMoreResources(array $pool) {
|
||||||
|
|
|
@ -7,8 +7,12 @@ final class DrydockPreallocatedHostBlueprintImplementation
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public function getBlueprintName() {
|
||||||
|
return pht('Remote Host (Preallocated)');
|
||||||
|
}
|
||||||
|
|
||||||
public function getDescription() {
|
public function getDescription() {
|
||||||
return pht('Leases out preallocated, remote hosts.');
|
return pht('Allows Drydock to run on specific remote hosts you configure.');
|
||||||
}
|
}
|
||||||
|
|
||||||
public function canAllocateMoreResources(array $pool) {
|
public function canAllocateMoreResources(array $pool) {
|
||||||
|
|
|
@ -7,8 +7,12 @@ final class DrydockWorkingCopyBlueprintImplementation
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public function getBlueprintName() {
|
||||||
|
return pht('Working Copy');
|
||||||
|
}
|
||||||
|
|
||||||
public function getDescription() {
|
public function getDescription() {
|
||||||
return pht('Allocates out working copies of repositories.');
|
return pht('Allows Drydock to check out working copies of repositories.');
|
||||||
}
|
}
|
||||||
|
|
||||||
protected function canAllocateLease(
|
protected function canAllocateLease(
|
||||||
|
|
|
@ -10,56 +10,75 @@ final class DrydockBlueprintCreateController
|
||||||
$implementations =
|
$implementations =
|
||||||
DrydockBlueprintImplementation::getAllBlueprintImplementations();
|
DrydockBlueprintImplementation::getAllBlueprintImplementations();
|
||||||
|
|
||||||
|
$errors = array();
|
||||||
|
$e_blueprint = null;
|
||||||
|
|
||||||
if ($request->isFormPost()) {
|
if ($request->isFormPost()) {
|
||||||
$class = $request->getStr('blueprint-type');
|
$class = $request->getStr('blueprint-type');
|
||||||
if (!isset($implementations[$class])) {
|
if (!isset($implementations[$class])) {
|
||||||
return $this->createDialog($implementations);
|
$e_blueprint = pht('Required');
|
||||||
|
$errors[] = pht('You must choose a blueprint type.');
|
||||||
}
|
}
|
||||||
|
|
||||||
$blueprint = id(new DrydockBlueprint())
|
if (!$errors) {
|
||||||
->setClassName($class)
|
$edit_uri = $this->getApplicationURI('blueprint/edit/?class='.$class);
|
||||||
->setDetails(array())
|
|
||||||
->setViewPolicy(PhabricatorPolicies::POLICY_ADMIN)
|
|
||||||
->setEditPolicy(PhabricatorPolicies::POLICY_ADMIN)
|
|
||||||
->save();
|
|
||||||
|
|
||||||
$edit_uri = $this->getApplicationURI(
|
|
||||||
"blueprint/edit/".$blueprint->getID()."/");
|
|
||||||
|
|
||||||
return id(new AphrontRedirectResponse())->setURI($edit_uri);
|
return id(new AphrontRedirectResponse())->setURI($edit_uri);
|
||||||
}
|
}
|
||||||
|
|
||||||
return $this->createDialog($implementations);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
function createDialog(array $implementations) {
|
$error_view = null;
|
||||||
$request = $this->getRequest();
|
if ($errors) {
|
||||||
$viewer = $request->getUser();
|
$error_view = id(new AphrontErrorView())
|
||||||
|
->setErrors($errors);
|
||||||
|
}
|
||||||
|
|
||||||
$control = id(new AphrontFormRadioButtonControl())
|
$control = id(new AphrontFormRadioButtonControl())
|
||||||
->setName('blueprint-type');
|
->setName('blueprint-type')
|
||||||
|
->setLabel(pht('Blueprint Type'))
|
||||||
|
->setError($e_blueprint);
|
||||||
|
|
||||||
foreach ($implementations as $implementation_name => $implementation) {
|
foreach ($implementations as $implementation_name => $implementation) {
|
||||||
$control
|
$disabled = !$implementation->isEnabled();
|
||||||
->addButton(
|
|
||||||
|
$control->addButton(
|
||||||
$implementation_name,
|
$implementation_name,
|
||||||
$implementation->getBlueprintClass(),
|
$implementation->getBlueprintName(),
|
||||||
$implementation->getDescription());
|
array(
|
||||||
|
pht('Provides: %s', $implementation->getType()),
|
||||||
|
phutil_tag('br'),
|
||||||
|
phutil_tag('br'),
|
||||||
|
$implementation->getDescription(),
|
||||||
|
),
|
||||||
|
$disabled ? 'disabled' : null,
|
||||||
|
$disabled);
|
||||||
}
|
}
|
||||||
|
|
||||||
$dialog = new AphrontDialogView();
|
$title = pht('Create New Blueprint');
|
||||||
$dialog->setTitle(pht('Create New Blueprint'))
|
$crumbs = $this->buildApplicationCrumbs();
|
||||||
|
$crumbs->addTextCrumb(pht('New Blueprint'));
|
||||||
|
|
||||||
|
$form = id(new AphrontFormView())
|
||||||
->setUser($viewer)
|
->setUser($viewer)
|
||||||
->addSubmitButton(pht('Create Blueprint'))
|
->appendChild($control)
|
||||||
->addCancelButton($this->getApplicationURI('blueprint/'));
|
->appendChild(
|
||||||
$dialog->appendChild(
|
id(new AphrontFormSubmitControl())
|
||||||
phutil_tag(
|
->addCancelButton($this->getApplicationURI('blueprint/'))
|
||||||
'p',
|
->setValue(pht('Continue')));
|
||||||
array(),
|
|
||||||
pht(
|
$box = id(new PHUIObjectBoxView())
|
||||||
'Select what type of blueprint you want to create: ')));
|
->setFormError($error_view)
|
||||||
$dialog->appendChild($control);
|
->setHeaderText($title)
|
||||||
return id(new AphrontDialogResponse())->setDialog($dialog);
|
->setForm($form);
|
||||||
|
|
||||||
|
return $this->buildApplicationPage(
|
||||||
|
array(
|
||||||
|
$crumbs,
|
||||||
|
$box,
|
||||||
|
),
|
||||||
|
array(
|
||||||
|
'title' => $title,
|
||||||
|
'device' => true,
|
||||||
|
));
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -25,10 +25,23 @@ final class DrydockBlueprintEditController extends DrydockBlueprintController {
|
||||||
if (!$blueprint) {
|
if (!$blueprint) {
|
||||||
return new Aphront404Response();
|
return new Aphront404Response();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
$impl = $blueprint->getImplementation();
|
||||||
|
$cancel_uri = $this->getApplicationURI('blueprint/'.$this->id.'/');
|
||||||
} else {
|
} else {
|
||||||
$blueprint = new DrydockBlueprint();
|
$class = $request->getStr('class');
|
||||||
|
|
||||||
|
$impl = DrydockBlueprintImplementation::getNamedImplementation($class);
|
||||||
|
if (!$impl || !$impl->isEnabled()) {
|
||||||
|
return new Aphront400Response();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
$blueprint = new DrydockBlueprint();
|
||||||
|
$blueprint->setClassName($class);
|
||||||
|
$cancel_uri = $this->getApplicationURI('blueprint/');
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
if ($request->isFormPost()) {
|
if ($request->isFormPost()) {
|
||||||
$v_view_policy = $request->getStr('viewPolicy');
|
$v_view_policy = $request->getStr('viewPolicy');
|
||||||
$v_edit_policy = $request->getStr('editPolicy');
|
$v_edit_policy = $request->getStr('editPolicy');
|
||||||
|
@ -39,8 +52,10 @@ final class DrydockBlueprintEditController extends DrydockBlueprintController {
|
||||||
|
|
||||||
$blueprint->save();
|
$blueprint->save();
|
||||||
|
|
||||||
return id(new AphrontRedirectResponse())
|
$id = $blueprint->getID();
|
||||||
->setURI('/drydock/blueprint/');
|
$save_uri = $this->getApplicationURI("blueprint/{$id}/");
|
||||||
|
|
||||||
|
return id(new AphrontRedirectResponse())->setURI($save_uri);
|
||||||
}
|
}
|
||||||
|
|
||||||
$policies = id(new PhabricatorPolicyQuery())
|
$policies = id(new PhabricatorPolicyQuery())
|
||||||
|
@ -48,21 +63,13 @@ final class DrydockBlueprintEditController extends DrydockBlueprintController {
|
||||||
->setObject($blueprint)
|
->setObject($blueprint)
|
||||||
->execute();
|
->execute();
|
||||||
|
|
||||||
if ($request->isAjax()) {
|
|
||||||
$form = id(new PHUIFormLayoutView())
|
|
||||||
->setUser($viewer);
|
|
||||||
} else {
|
|
||||||
$form = id(new AphrontFormView())
|
$form = id(new AphrontFormView())
|
||||||
->setUser($viewer);
|
->setUser($viewer)
|
||||||
}
|
->addHiddenInput('class', $request->getStr('class'))
|
||||||
|
|
||||||
$form
|
|
||||||
->appendChild(
|
->appendChild(
|
||||||
id(new AphrontFormTextControl())
|
id(new AphrontFormStaticControl())
|
||||||
->setName('className')
|
->setLabel(pht('Blueprint Type'))
|
||||||
->setLabel(pht('Implementation'))
|
->setValue($impl->getBlueprintName()))
|
||||||
->setValue($blueprint->getClassName())
|
|
||||||
->setDisabled(true))
|
|
||||||
->appendChild(
|
->appendChild(
|
||||||
id(new AphrontFormPolicyControl())
|
id(new AphrontFormPolicyControl())
|
||||||
->setName('viewPolicy')
|
->setName('viewPolicy')
|
||||||
|
@ -78,27 +85,23 @@ final class DrydockBlueprintEditController extends DrydockBlueprintController {
|
||||||
|
|
||||||
$crumbs = $this->buildApplicationCrumbs();
|
$crumbs = $this->buildApplicationCrumbs();
|
||||||
|
|
||||||
|
if ($blueprint->getID()) {
|
||||||
$title = pht('Edit Blueprint');
|
$title = pht('Edit Blueprint');
|
||||||
$header = pht('Edit Blueprint %d', $blueprint->getID());
|
$header = pht('Edit Blueprint %d', $blueprint->getID());
|
||||||
$crumbs->addTextCrumb(pht('Blueprint %d', $blueprint->getID()));
|
$crumbs->addTextCrumb(pht('Blueprint %d', $blueprint->getID()));
|
||||||
$crumbs->addTextCrumb(pht('Edit'));
|
$crumbs->addTextCrumb(pht('Edit'));
|
||||||
|
$submit = pht('Save Blueprint');
|
||||||
if ($request->isAjax()) {
|
} else {
|
||||||
$dialog = id(new AphrontDialogView())
|
$title = pht('New Blueprint');
|
||||||
->setUser($viewer)
|
$header = pht('New Blueprint');
|
||||||
->setWidth(AphrontDialogView::WIDTH_FORM)
|
$crumbs->addTextCrumb(pht('New Blueprint'));
|
||||||
->setTitle($title)
|
$submit = pht('Create Blueprint');
|
||||||
->appendChild($form)
|
|
||||||
->addSubmitButton(pht('Edit Blueprint'))
|
|
||||||
->addCancelButton($this->getApplicationURI());
|
|
||||||
|
|
||||||
return id(new AphrontDialogResponse())->setDialog($dialog);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
$form->appendChild(
|
$form->appendChild(
|
||||||
id(new AphrontFormSubmitControl())
|
id(new AphrontFormSubmitControl())
|
||||||
->setValue(pht('Save'))
|
->setValue($submit)
|
||||||
->addCancelButton($this->getApplicationURI()));
|
->addCancelButton($cancel_uri));
|
||||||
|
|
||||||
$box = id(new PHUIObjectBoxView())
|
$box = id(new PHUIObjectBoxView())
|
||||||
->setHeaderText($header)
|
->setHeaderText($header)
|
||||||
|
|
|
@ -65,21 +65,28 @@ final class DrydockBlueprintViewController extends DrydockBlueprintController {
|
||||||
}
|
}
|
||||||
|
|
||||||
private function buildActionListView(DrydockBlueprint $blueprint) {
|
private function buildActionListView(DrydockBlueprint $blueprint) {
|
||||||
|
$viewer = $this->getRequest()->getUser();
|
||||||
|
|
||||||
$view = id(new PhabricatorActionListView())
|
$view = id(new PhabricatorActionListView())
|
||||||
->setUser($this->getRequest()->getUser())
|
->setUser($viewer)
|
||||||
->setObjectURI($this->getRequest()->getRequestURI())
|
->setObjectURI($this->getRequest()->getRequestURI())
|
||||||
->setObject($blueprint);
|
->setObject($blueprint);
|
||||||
|
|
||||||
$uri = '/blueprint/edit/'.$blueprint->getID().'/';
|
$uri = '/blueprint/edit/'.$blueprint->getID().'/';
|
||||||
$uri = $this->getApplicationURI($uri);
|
$uri = $this->getApplicationURI($uri);
|
||||||
|
|
||||||
|
$can_edit = PhabricatorPolicyFilter::hasCapability(
|
||||||
|
$viewer,
|
||||||
|
$blueprint,
|
||||||
|
PhabricatorPolicyCapability::CAN_EDIT);
|
||||||
|
|
||||||
$view->addAction(
|
$view->addAction(
|
||||||
id(new PhabricatorActionView())
|
id(new PhabricatorActionView())
|
||||||
->setHref($uri)
|
->setHref($uri)
|
||||||
->setName(pht('Edit Blueprint Policies'))
|
->setName(pht('Edit Blueprint'))
|
||||||
->setIcon('edit')
|
->setIcon('edit')
|
||||||
->setWorkflow(true)
|
->setWorkflow(!$can_edit)
|
||||||
->setDisabled(false));
|
->setDisabled(!$can_edit));
|
||||||
|
|
||||||
return $view;
|
return $view;
|
||||||
}
|
}
|
||||||
|
|
|
@ -33,7 +33,7 @@ final class HarbormasterStepAddController
|
||||||
if ($request->isDialogFormPost()) {
|
if ($request->isDialogFormPost()) {
|
||||||
$class = $request->getStr('step-type');
|
$class = $request->getStr('step-type');
|
||||||
if (!in_array($class, $implementations)) {
|
if (!in_array($class, $implementations)) {
|
||||||
return $this->createDialog($implementations);
|
return $this->createDialog($implementations, $cancel_uri);
|
||||||
}
|
}
|
||||||
|
|
||||||
$steps = $plan->loadOrderedBuildSteps();
|
$steps = $plan->loadOrderedBuildSteps();
|
||||||
|
|
Loading…
Reference in a new issue