mirror of
https://we.phorge.it/source/phorge.git
synced 2024-11-24 15:52:41 +01:00
04b9f94602
Summary: Ref T4065. Give administrators an "Edit Settings" link from profiles, which allows selective edit of settings panels. Enable Conduit, SSH Keys, and VCS Password. Test Plan: - Used these panels for a bot. - Used these panels on my own account. - Tried to use these panels for a non-bot account, was denied. Reviewers: btrahan Reviewed By: btrahan Subscribers: epriestley Maniphest Tasks: T4065 Differential Revision: https://secure.phabricator.com/D8668
161 lines
3.7 KiB
PHP
161 lines
3.7 KiB
PHP
<?php
|
|
|
|
final class PhabricatorSettingsMainController
|
|
extends PhabricatorController {
|
|
|
|
private $id;
|
|
private $key;
|
|
private $user;
|
|
|
|
private function getUser() {
|
|
return $this->user;
|
|
}
|
|
|
|
private function isSelf() {
|
|
$viewer_phid = $this->getRequest()->getUser()->getPHID();
|
|
$user_phid = $this->getUser()->getPHID();
|
|
return ($viewer_phid == $user_phid);
|
|
}
|
|
|
|
public function willProcessRequest(array $data) {
|
|
$this->id = idx($data, 'id');
|
|
$this->key = idx($data, 'key');
|
|
}
|
|
|
|
public function processRequest() {
|
|
$request = $this->getRequest();
|
|
$viewer = $request->getUser();
|
|
|
|
if ($this->id) {
|
|
$user = id(new PhabricatorPeopleQuery())
|
|
->setViewer($viewer)
|
|
->withIDs(array($this->id))
|
|
->requireCapabilities(
|
|
array(
|
|
PhabricatorPolicyCapability::CAN_VIEW,
|
|
PhabricatorPolicyCapability::CAN_EDIT,
|
|
))
|
|
->executeOne();
|
|
|
|
if (!$user) {
|
|
return new Aphront404Response();
|
|
}
|
|
|
|
$this->user = $user;
|
|
} else {
|
|
$this->user = $viewer;
|
|
}
|
|
|
|
$panels = $this->buildPanels();
|
|
$nav = $this->renderSideNav($panels);
|
|
|
|
$key = $nav->selectFilter($this->key, head($panels)->getPanelKey());
|
|
|
|
|
|
$panel = $panels[$key];
|
|
$panel->setUser($this->getUser());
|
|
$panel->setViewer($viewer);
|
|
|
|
$response = $panel->processRequest($request);
|
|
if ($response instanceof AphrontResponse) {
|
|
return $response;
|
|
}
|
|
|
|
$crumbs = $this->buildApplicationCrumbs();
|
|
if (!$this->isSelf()) {
|
|
$crumbs->addTextCrumb(
|
|
$this->getUser()->getUsername(),
|
|
'/p/'.$this->getUser()->getUsername().'/');
|
|
}
|
|
$crumbs->addTextCrumb($panel->getPanelName());
|
|
$nav->appendChild(
|
|
array(
|
|
$crumbs,
|
|
$response,
|
|
));
|
|
|
|
return $this->buildApplicationPage(
|
|
$nav,
|
|
array(
|
|
'title' => $panel->getPanelName(),
|
|
'device' => true,
|
|
));
|
|
}
|
|
|
|
private function buildPanels() {
|
|
$panel_specs = id(new PhutilSymbolLoader())
|
|
->setAncestorClass('PhabricatorSettingsPanel')
|
|
->setConcreteOnly(true)
|
|
->selectAndLoadSymbols();
|
|
|
|
$panels = array();
|
|
foreach ($panel_specs as $spec) {
|
|
$class = newv($spec['name'], array());
|
|
$panels[] = $class->buildPanels();
|
|
}
|
|
|
|
$panels = array_mergev($panels);
|
|
$panels = mpull($panels, null, 'getPanelKey');
|
|
|
|
$result = array();
|
|
foreach ($panels as $key => $panel) {
|
|
if (!$panel->isEnabled()) {
|
|
continue;
|
|
}
|
|
|
|
if (!$this->isSelf()) {
|
|
if (!$panel->isEditableByAdministrators()) {
|
|
continue;
|
|
}
|
|
}
|
|
|
|
if (!empty($result[$key])) {
|
|
throw new Exception(pht(
|
|
"Two settings panels share the same panel key ('%s'): %s, %s.",
|
|
$key,
|
|
get_class($panel),
|
|
get_class($result[$key])));
|
|
}
|
|
|
|
$result[$key] = $panel;
|
|
}
|
|
|
|
$result = msort($result, 'getPanelSortKey');
|
|
|
|
if (!$result) {
|
|
throw new Exception(pht('No settings panels are available.'));
|
|
}
|
|
|
|
return $result;
|
|
}
|
|
|
|
private function renderSideNav(array $panels) {
|
|
$nav = new AphrontSideNavFilterView();
|
|
|
|
if ($this->isSelf()) {
|
|
$base_uri = 'panel/';
|
|
} else {
|
|
$base_uri = $this->getUser()->getID().'/panel/';
|
|
}
|
|
|
|
$nav->setBaseURI(new PhutilURI($this->getApplicationURI($base_uri)));
|
|
|
|
$group = null;
|
|
foreach ($panels as $panel) {
|
|
if ($panel->getPanelGroup() != $group) {
|
|
$group = $panel->getPanelGroup();
|
|
$nav->addLabel($group);
|
|
}
|
|
|
|
$nav->addFilter($panel->getPanelKey(), $panel->getPanelName());
|
|
}
|
|
|
|
return $nav;
|
|
}
|
|
|
|
public function buildApplicationMenu() {
|
|
$panels = $this->buildPanels();
|
|
return $this->renderSideNav($panels)->getMenu();
|
|
}
|
|
|
|
}
|