1
0
Fork 0
mirror of https://we.phorge.it/source/phorge.git synced 2025-01-10 14:51:06 +01:00

Modernize most Conduit console interfaces

Summary:
Ref T603. Ref T2625.

Long chain of "doing the right thing" here: I want to clean this up, so I can clean up the Conduit logs, so I can add a setup issue for deprecated method calls, so I can remove deprecated methods, so I can get rid of `DifferentialRevisionListData`, so I can make Differntial policy-aware.

Adds modern infrastructure and UI to all of the Conduit interfaces (except only partially for the logs, that will be the next diff).

Test Plan:
{F48201}
{F48202}
{F48203}
{F48204}
{F48206}

This will get further updates in the next diff:

{F48205}

Reviewers: btrahan, chad

Reviewed By: chad

CC: aran

Maniphest Tasks: T603, T2625

Differential Revision: https://secure.phabricator.com/D6331
This commit is contained in:
epriestley 2013-07-01 12:36:34 -07:00
parent e4eeff8140
commit f82e4b0c70
13 changed files with 513 additions and 227 deletions

View file

@ -921,6 +921,8 @@ phutil_register_library_map(array(
'PhabricatorConduitListController' => 'applications/conduit/controller/PhabricatorConduitListController.php', 'PhabricatorConduitListController' => 'applications/conduit/controller/PhabricatorConduitListController.php',
'PhabricatorConduitLogController' => 'applications/conduit/controller/PhabricatorConduitLogController.php', 'PhabricatorConduitLogController' => 'applications/conduit/controller/PhabricatorConduitLogController.php',
'PhabricatorConduitMethodCallLog' => 'applications/conduit/storage/PhabricatorConduitMethodCallLog.php', 'PhabricatorConduitMethodCallLog' => 'applications/conduit/storage/PhabricatorConduitMethodCallLog.php',
'PhabricatorConduitMethodQuery' => 'applications/conduit/query/PhabricatorConduitMethodQuery.php',
'PhabricatorConduitSearchEngine' => 'applications/conduit/query/PhabricatorConduitSearchEngine.php',
'PhabricatorConduitTokenController' => 'applications/conduit/controller/PhabricatorConduitTokenController.php', 'PhabricatorConduitTokenController' => 'applications/conduit/controller/PhabricatorConduitTokenController.php',
'PhabricatorConfigAllController' => 'applications/config/controller/PhabricatorConfigAllController.php', 'PhabricatorConfigAllController' => 'applications/config/controller/PhabricatorConfigAllController.php',
'PhabricatorConfigController' => 'applications/config/controller/PhabricatorConfigController.php', 'PhabricatorConfigController' => 'applications/config/controller/PhabricatorConfigController.php',
@ -2009,6 +2011,11 @@ phutil_register_library_map(array(
'CelerityResourceController' => 'PhabricatorController', 'CelerityResourceController' => 'PhabricatorController',
'CelerityResourceGraph' => 'AbstractDirectedGraph', 'CelerityResourceGraph' => 'AbstractDirectedGraph',
'CelerityResourceTransformerTestCase' => 'PhabricatorTestCase', 'CelerityResourceTransformerTestCase' => 'PhabricatorTestCase',
'ConduitAPIMethod' =>
array(
0 => 'Phobject',
1 => 'PhabricatorPolicyInterface',
),
'ConduitAPI_arcanist_Method' => 'ConduitAPIMethod', 'ConduitAPI_arcanist_Method' => 'ConduitAPIMethod',
'ConduitAPI_arcanist_projectinfo_Method' => 'ConduitAPI_arcanist_Method', 'ConduitAPI_arcanist_projectinfo_Method' => 'ConduitAPI_arcanist_Method',
'ConduitAPI_audit_Method' => 'ConduitAPIMethod', 'ConduitAPI_audit_Method' => 'ConduitAPIMethod',
@ -2807,9 +2814,15 @@ phutil_register_library_map(array(
'PhabricatorConduitConsoleController' => 'PhabricatorConduitController', 'PhabricatorConduitConsoleController' => 'PhabricatorConduitController',
'PhabricatorConduitController' => 'PhabricatorController', 'PhabricatorConduitController' => 'PhabricatorController',
'PhabricatorConduitDAO' => 'PhabricatorLiskDAO', 'PhabricatorConduitDAO' => 'PhabricatorLiskDAO',
'PhabricatorConduitListController' => 'PhabricatorConduitController', 'PhabricatorConduitListController' =>
array(
0 => 'PhabricatorConduitController',
1 => 'PhabricatorApplicationSearchResultsControllerInterface',
),
'PhabricatorConduitLogController' => 'PhabricatorConduitController', 'PhabricatorConduitLogController' => 'PhabricatorConduitController',
'PhabricatorConduitMethodCallLog' => 'PhabricatorConduitDAO', 'PhabricatorConduitMethodCallLog' => 'PhabricatorConduitDAO',
'PhabricatorConduitMethodQuery' => 'PhabricatorCursorPagedPolicyAwareQuery',
'PhabricatorConduitSearchEngine' => 'PhabricatorApplicationSearchEngine',
'PhabricatorConduitTokenController' => 'PhabricatorConduitController', 'PhabricatorConduitTokenController' => 'PhabricatorConduitController',
'PhabricatorConfigAllController' => 'PhabricatorConfigController', 'PhabricatorConfigAllController' => 'PhabricatorConfigController',
'PhabricatorConfigController' => 'PhabricatorController', 'PhabricatorConfigController' => 'PhabricatorController',

View file

@ -38,7 +38,7 @@ final class PhabricatorApplicationConduit extends PhabricatorApplication {
public function getRoutes() { public function getRoutes() {
return array( return array(
'/conduit/' => array( '/conduit/' => array(
'' => 'PhabricatorConduitListController', '(?:query/(?P<queryKey>[^/]+)/)?' => 'PhabricatorConduitListController',
'method/(?P<method>[^/]+)/' => 'PhabricatorConduitConsoleController', 'method/(?P<method>[^/]+)/' => 'PhabricatorConduitConsoleController',
'log/' => 'PhabricatorConduitLogController', 'log/' => 'PhabricatorConduitLogController',
'log/view/(?P<view>[^/]+)/' => 'PhabricatorConduitLogController', 'log/view/(?P<view>[^/]+)/' => 'PhabricatorConduitLogController',

View file

@ -354,6 +354,7 @@ final class PhabricatorConduitAPIController
} }
$param_table = new AphrontTableView($param_rows); $param_table = new AphrontTableView($param_rows);
$param_table->setDeviceReadyTable(true);
$param_table->setColumnClasses( $param_table->setColumnClasses(
array( array(
'header', 'header',
@ -369,6 +370,7 @@ final class PhabricatorConduitAPIController
} }
$result_table = new AphrontTableView($result_rows); $result_table = new AphrontTableView($result_rows);
$result_table->setDeviceReadyTable(true);
$result_table->setColumnClasses( $result_table->setColumnClasses(
array( array(
'header', 'header',
@ -383,13 +385,36 @@ final class PhabricatorConduitAPIController
$result_panel->setHeader('Method Result'); $result_panel->setHeader('Method Result');
$result_panel->appendChild($result_table); $result_panel->appendChild($result_table);
return $this->buildStandardPageResponse( $param_head = id(new PhabricatorHeaderView())
->setHeader(pht('Method Parameters'));
$result_head = id(new PhabricatorHeaderView())
->setHeader(pht('Method Result'));
$method_uri = $this->getApplicationURI('method/'.$method.'/');
$crumbs = $this->buildApplicationCrumbs();
$crumbs
->addCrumb(
id(new PhabricatorCrumbView())
->setName($method)
->setHref($method_uri))
->addCrumb(
id(new PhabricatorCrumbView())
->setName(pht('Call')));
return $this->buildApplicationPage(
array( array(
$param_panel, $crumbs,
$result_panel, $param_head,
$param_table,
$result_head,
$result_table,
), ),
array( array(
'title' => 'Method Call Result', 'title' => 'Method Call Result',
'device' => true,
'dust' => true,
)); ));
} }

View file

@ -15,18 +15,19 @@ final class PhabricatorConduitConsoleController
public function processRequest() { public function processRequest() {
$request = $this->getRequest(); $request = $this->getRequest();
$viewer = $request->getUser();
$methods = $this->getAllMethods(); $method = id(new PhabricatorConduitMethodQuery())
if (empty($methods[$this->method])) { ->setViewer($viewer)
->withMethods(array($this->method))
->executeOne();
if (!$method) {
return new Aphront404Response(); return new Aphront404Response();
} }
$this->setFilter('method/'.$this->method);
$method_class = $methods[$this->method]; $status = $method->getMethodStatus();
$method_object = newv($method_class, array()); $reason = $method->getMethodStatusDescription();
$status = $method_object->getMethodStatus();
$reason = $method_object->getMethodStatusDescription();
$status_view = null; $status_view = null;
if ($status != ConduitAPIMethod::METHOD_STATUS_STABLE) { if ($status != ConduitAPIMethod::METHOD_STATUS_STABLE) {
@ -49,7 +50,7 @@ final class PhabricatorConduitConsoleController
} }
} }
$error_types = $method_object->defineErrorTypes(); $error_types = $method->defineErrorTypes();
if ($error_types) { if ($error_types) {
$error_description = array(); $error_description = array();
foreach ($error_types as $error => $meaning) { foreach ($error_types as $error => $meaning) {
@ -68,14 +69,15 @@ final class PhabricatorConduitConsoleController
->setUser($request->getUser()) ->setUser($request->getUser())
->setAction('/api/'.$this->method) ->setAction('/api/'.$this->method)
->addHiddenInput('allowEmptyParams', 1) ->addHiddenInput('allowEmptyParams', 1)
->setFlexible(true)
->appendChild( ->appendChild(
id(new AphrontFormStaticControl()) id(new AphrontFormStaticControl())
->setLabel('Description') ->setLabel('Description')
->setValue($method_object->getMethodDescription())) ->setValue($method->getMethodDescription()))
->appendChild( ->appendChild(
id(new AphrontFormStaticControl()) id(new AphrontFormStaticControl())
->setLabel('Returns') ->setLabel('Returns')
->setValue($method_object->defineReturnType())) ->setValue($method->defineReturnType()))
->appendChild( ->appendChild(
id(new AphrontFormMarkupControl()) id(new AphrontFormMarkupControl())
->setLabel('Errors') ->setLabel('Errors')
@ -85,7 +87,7 @@ final class PhabricatorConduitConsoleController
'<strong>JSON</strong>. For instance, to enter a list, type: '. '<strong>JSON</strong>. For instance, to enter a list, type: '.
'<tt>["apple", "banana", "cherry"]</tt>')); '<tt>["apple", "banana", "cherry"]</tt>'));
$params = $method_object->defineParamTypes(); $params = $method->defineParamTypes();
foreach ($params as $param => $desc) { foreach ($params as $param => $desc) {
$form->appendChild( $form->appendChild(
id(new AphrontFormTextControl()) id(new AphrontFormTextControl())
@ -106,30 +108,25 @@ final class PhabricatorConduitConsoleController
))) )))
->appendChild( ->appendChild(
id(new AphrontFormSubmitControl()) id(new AphrontFormSubmitControl())
->addCancelButton($this->getApplicationURI())
->setValue('Call Method')); ->setValue('Call Method'));
$panel = new AphrontPanelView(); $crumbs = $this->buildApplicationCrumbs();
$panel->setHeader('Conduit API: '.$this->method); $crumbs->addCrumb(
$panel->appendChild($form); id(new PhabricatorCrumbView())
$panel->setWidth(AphrontPanelView::WIDTH_FULL); ->setName($method->getAPIMethodName()));
return $this->buildStandardPageResponse( return $this->buildApplicationPage(
array( array(
$crumbs,
$status_view, $status_view,
$panel, $form,
), ),
array( array(
'title' => 'Conduit Console - '.$this->method, 'title' => $method->getAPIMethodName(),
'device' => true,
'dust' => true,
)); ));
} }
private function getAllMethods() {
$classes = $this->getAllMethodImplementationClasses();
$methods = array();
foreach ($classes as $class) {
$name = ConduitAPIMethod::getAPIMethodNameFromClassName($class);
$methods[$name] = $class;
}
return $methods;
}
} }

View file

@ -5,122 +5,27 @@
*/ */
abstract class PhabricatorConduitController extends PhabricatorController { abstract class PhabricatorConduitController extends PhabricatorController {
private $filter; protected function buildSideNavView() {
protected $showSideNav; $viewer = $this->getRequest()->getUser();
public function buildStandardPageResponse($view, array $data) { $nav = new AphrontSideNavFilterView();
$page = $this->buildStandardPageView(); $nav->setBaseURI(new PhutilURI($this->getApplicationURI()));
$page->setApplicationName('Conduit'); id(new PhabricatorConduitSearchEngine())
$page->setBaseURI('/conduit/'); ->setViewer($viewer)
$page->setTitle(idx($data, 'title')); ->addNavigationItems($nav->getMenu());
$page->setGlyph("\xE2\x87\xB5");
if ($this->showSideNav()) { $nav->addLabel('Logs');
$nav->addFilter('log', pht('Call Logs'));
$nav = new AphrontSideNavFilterView(); $nav->selectFilter(null);
$nav->setBaseURI(new PhutilURI('/conduit/'));
$method_filters = $this->getMethodFilters();
foreach ($method_filters as $group => $methods) {
$nav->addLabel($group);
foreach ($methods as $method) {
$method_name = $method['full_name'];
$display_name = $method_name; return $nav;
switch ($method['status']) {
case ConduitAPIMethod::METHOD_STATUS_DEPRECATED:
$display_name = '('.$display_name.')';
break;
}
$nav->addFilter('method/'.$method_name,
$display_name);
}
}
$nav->selectFilter($this->getFilter());
$nav->appendChild($view);
$body = $nav;
} else {
$body = $view;
}
$page->appendChild($body);
$response = new AphrontWebpageResponse();
return $response->setContent($page->render());
} }
private function getFilter() { protected function buildApplicationMenu() {
return $this->filter; return $this->buildSideNavView()->getMenu();
} }
protected function setFilter($filter) {
$this->filter = $filter;
return $this;
}
private function showSideNav() {
return $this->showSideNav !== false;
}
protected function setShowSideNav($show_side_nav) {
$this->showSideNav = $show_side_nav;
return $this;
}
protected function getAllMethodImplementationClasses() {
$classes = id(new PhutilSymbolLoader())
->setAncestorClass('ConduitAPIMethod')
->setType('class')
->setConcreteOnly(true)
->selectSymbolsWithoutLoading();
return array_values(ipull($classes, 'name'));
}
protected function getMethodFilters() {
$classes = $this->getAllMethodImplementationClasses();
$method_names = array();
foreach ($classes as $method_class) {
$method_name = ConduitAPIMethod::getAPIMethodNameFromClassName(
$method_class);
$group_name = head(explode('.', $method_name));
$method_object = newv($method_class, array());
$application = $method_object->getApplication();
if ($application && !$application->isInstalled()) {
continue;
}
$status = $method_object->getMethodStatus();
$key = sprintf(
'%02d %s %s',
$this->getOrderForMethodStatus($status),
$group_name,
$method_name);
$method_names[$key] = array(
'full_name' => $method_name,
'group_name' => $group_name,
'status' => $status,
'description' => $method_object->getMethodDescription(),
);
}
ksort($method_names);
$method_names = igroup($method_names, 'group_name');
ksort($method_names);
return $method_names;
}
private function getOrderForMethodStatus($status) {
$map = array(
ConduitAPIMethod::METHOD_STATUS_STABLE => 0,
ConduitAPIMethod::METHOD_STATUS_UNSTABLE => 1,
ConduitAPIMethod::METHOD_STATUS_DEPRECATED => 2,
);
return idx($map, $status, 0);
}
} }

View file

@ -4,78 +4,78 @@
* @group conduit * @group conduit
*/ */
final class PhabricatorConduitListController final class PhabricatorConduitListController
extends PhabricatorConduitController { extends PhabricatorConduitController
implements PhabricatorApplicationSearchResultsControllerInterface {
private $queryKey;
public function willProcessRequest(array $data) {
$this->queryKey = idx($data, 'queryKey');
}
public function processRequest() { public function processRequest() {
$method_groups = $this->getMethodFilters(); $request = $this->getRequest();
$rows = array(); $controller = id(new PhabricatorApplicationSearchController($request))
foreach ($method_groups as $group => $methods) { ->setQueryKey($this->queryKey)
foreach ($methods as $info) { ->setSearchEngine(new PhabricatorConduitSearchEngine())
switch ($info['status']) { ->setNavigation($this->buildSideNavView());
case ConduitAPIMethod::METHOD_STATUS_DEPRECATED: return $this->delegateToController($controller);
$status = 'Deprecated'; }
break;
case ConduitAPIMethod::METHOD_STATUS_UNSTABLE:
$status = 'Unstable';
break;
default:
$status = null;
break;
}
$rows[] = array( public function renderResultsList(array $methods) {
$group, assert_instances_of($methods, 'ConduitAPIMethod');
phutil_tag(
'a', $viewer = $this->getRequest()->getUser();
array(
'href' => '/conduit/method/'.$info['full_name'], $out = array();
),
$info['full_name']), $last = null;
$info['description'], $list = null;
$status, foreach ($methods as $method) {
); $app = $method->getApplicationName();
$group = null; if ($app !== $last) {
$last = $app;
if ($list) {
$out[] = $list;
}
$list = id(new PhabricatorObjectItemListView());
$app_object = $method->getApplication();
if ($app_object) {
$app_name = $app_object->getName();
} else {
$app_name = $app;
}
} }
$method_name = $method->getAPIMethodName();
$item = id(new PhabricatorObjectItemView())
->setHeader($method_name)
->setHref($this->getApplicationURI('method/'.$method_name.'/'))
->addAttribute($method->getMethodDescription());
switch ($method->getMethodStatus()) {
case ConduitAPIMethod::METHOD_STATUS_STABLE:
break;
case ConduitAPIMethod::METHOD_STATUS_UNSTABLE:
$item->addIcon('warning-grey', pht('Unstable'));
$item->setBarColor('yellow');
break;
case ConduitAPIMethod::METHOD_STATUS_DEPRECATED:
$item->addIcon('warning', pht('Deprecated'));
$item->setBarColor('red');
break;
}
$list->addItem($item);
} }
$table = new AphrontTableView($rows); if ($list) {
$table->setHeaders(array( $out[] = $list;
'Group', }
'Name',
'Description',
'Status',
));
$table->setColumnClasses(array(
'pri',
'pri',
'wide',
null,
));
$panel = new AphrontPanelView(); return $out;
$panel->setHeader('Conduit Methods');
$panel->appendChild($table);
$panel->setWidth(AphrontPanelView::WIDTH_FULL);
$utils = new AphrontPanelView();
$utils->setHeader('Utilities');
$utils->appendChild(hsprintf(
'<ul>'.
'<li><a href="/conduit/log/">Log</a> - Conduit Method Calls</li>'.
'<li><a href="/conduit/token/">Token</a> - Certificate Install</li>'.
'</ul>'));
$utils->setWidth(AphrontPanelView::WIDTH_FULL);
$this->setShowSideNav(false);
return $this->buildStandardPageResponse(
array(
$panel,
$utils,
),
array(
'title' => 'Conduit Console',
));
} }
} }

View file

@ -41,10 +41,16 @@ final class PhabricatorConduitLogController
$panel->appendChild($table); $panel->appendChild($table);
$panel->appendChild($pager); $panel->appendChild($pager);
$this->setShowSideNav(false); $crumbs = $this->buildApplicationCrumbs();
$crumbs->addCrumb(
id(new PhabricatorCrumbView())
->setName(pht('Call Logs')));
return $this->buildStandardPageResponse( return $this->buildApplicationPage(
$panel, array(
$crumbs,
$panel,
),
array( array(
'title' => 'Conduit Logs', 'title' => 'Conduit Logs',
)); ));

View file

@ -28,26 +28,40 @@ final class PhabricatorConduitTokenController
->setToken(Filesystem::readRandomCharacters(40)) ->setToken(Filesystem::readRandomCharacters(40))
->save(); ->save();
$panel = new AphrontPanelView(); unset($unguarded);
$panel->setHeader('Certificate Install Token');
$panel->setWidth(AphrontPanelView::WIDTH_FORM);
$panel->appendChild(hsprintf( $pre_instructions = pht(
'<p class="aphront-form-instructions">Copy and paste this token into '. 'Copy and paste this token into the prompt given to you by '.
'the prompt given to you by "arc install-certificate":</p>'. '`arc install-certificate`');
'<p style="padding: 0 0 1em 4em;">'.
'<strong>%s</strong>'.
'</p>'.
'<p class="aphront-form-instructions">arc will then complete the '.
'install process for you.</p>',
$token->getToken()));
$this->setShowSideNav(false); $post_instructions = pht(
'After you copy and paste this token, `arc` will complete '.
'the certificate install process for you.');
return $this->buildStandardPageResponse( $form = id(new AphrontFormView())
$panel, ->setUser($user)
->appendRemarkupInstructions($pre_instructions)
->appendChild(
id(new AphrontFormTextAreaControl())
->setLabel(pht('Token'))
->setHeight(AphrontFormTextAreaControl::HEIGHT_VERY_SHORT)
->setValue($token->getToken()))
->appendRemarkupInstructions($post_instructions);
$crumbs = $this->buildApplicationCrumbs();
$crumbs->addCrumb(
id(new PhabricatorCrumbView())
->setName(pht('Install Certificate')));
return $this->buildApplicationPage(
array( array(
'title' => 'Certificate Install Token', $crumbs,
$form,
),
array(
'title' => pht('Certificate Install Token'),
'device' => true,
'dust' => true,
)); ));
} }
} }

View file

@ -5,7 +5,9 @@
* @task status Method Status * @task status Method Status
* @group conduit * @group conduit
*/ */
abstract class ConduitAPIMethod { abstract class ConduitAPIMethod
extends Phobject
implements PhabricatorPolicyInterface {
const METHOD_STATUS_STABLE = 'stable'; const METHOD_STATUS_STABLE = 'stable';
const METHOD_STATUS_UNSTABLE = 'unstable'; const METHOD_STATUS_UNSTABLE = 'unstable';
@ -21,6 +23,14 @@ abstract class ConduitAPIMethod {
} }
/**
* This is mostly for compatibility with
* @{class:AphrontCursorPagedPolicyAwareQuery}.
*/
public function getID() {
return $this->getAPIMethodName();
}
/** /**
* Get the status for this method (e.g., stable, unstable or deprecated). * Get the status for this method (e.g., stable, unstable or deprecated).
* Should return a METHOD_STATUS_* constant. By default, methods are * Should return a METHOD_STATUS_* constant. By default, methods are
@ -62,6 +72,29 @@ abstract class ConduitAPIMethod {
return self::getAPIMethodNameFromClassName(get_class($this)); return self::getAPIMethodNameFromClassName(get_class($this));
} }
/**
* Return a key which sorts methods by application name, then method status,
* then method name.
*/
public function getSortOrder() {
$name = $this->getAPIMethodName();
$map = array(
ConduitAPIMethod::METHOD_STATUS_STABLE => 0,
ConduitAPIMethod::METHOD_STATUS_UNSTABLE => 1,
ConduitAPIMethod::METHOD_STATUS_DEPRECATED => 2,
);
$ord = idx($map, $this->getMethodStatus(), 0);
list($head, $tail) = explode('.', $name, 2);
return "{$head}.{$ord}.{$tail}";
}
public function getApplicationName() {
return head(explode('.', $this->getAPIMethodName(), 2));
}
public static function getClassNameFromAPIMethodName($method_name) { public static function getClassNameFromAPIMethodName($method_name) {
$method_fragment = str_replace('.', '_', $method_name); $method_fragment = str_replace('.', '_', $method_name);
return 'ConduitAPI_'.$method_fragment.'_Method'; return 'ConduitAPI_'.$method_fragment.'_Method';
@ -129,4 +162,25 @@ abstract class ConduitAPIMethod {
} }
} }
/* -( PhabricatorPolicyInterface )----------------------------------------- */
public function getCapabilities() {
return array(
PhabricatorPolicyCapability::CAN_VIEW,
);
}
public function getPolicy($capability) {
return PhabricatorPolicies::POLICY_USER;
}
public function hasAutomaticCapability($capability, PhabricatorUser $viewer) {
// The policy interface on Conduit calls is currently just to let us hook
// into ApplicationSearch. Calls are always visible (even to logged out
// users).
return true;
}
} }

View file

@ -0,0 +1,124 @@
<?php
final class PhabricatorConduitMethodQuery
extends PhabricatorCursorPagedPolicyAwareQuery {
private $isDeprecated;
private $isStable;
private $isUnstable;
private $applicationNames;
private $nameContains;
private $methods;
public function withMethods(array $methods) {
$this->methods = $methods;
return $this;
}
public function withNameContains($name_contains) {
$this->nameContains = $name_contains;
return $this;
}
public function withApplicationNames(array $application_names) {
$this->applicationNames = $application_names;
return $this;
}
public function withIsStable($is_stable) {
$this->isStable = $is_stable;
return $this;
}
public function withIsUnstable($is_unstable) {
$this->isUnstable = $is_unstable;
return $this;
}
public function withIsDeprecated($is_deprecated) {
$this->isDeprecated = $is_deprecated;
return $this;
}
public function loadPage() {
$methods = $this->getAllMethods();
$methods = $this->filterMethods($methods);
return $methods;
}
private function getAllMethods() {
static $methods;
if ($methods === null) {
$methods = id(new PhutilSymbolLoader())
->setAncestorClass('ConduitAPIMethod')
->loadObjects();
$methods = msort($methods, 'getSortOrder');
}
return $methods;
}
private function filterMethods(array $methods) {
foreach ($methods as $key => $method) {
$application = $method->getApplication();
if (!$application) {
continue;
}
if (!$application->isInstalled()) {
unset($methods[$key]);
}
}
$status = array(
ConduitAPIMethod::METHOD_STATUS_STABLE => $this->isStable,
ConduitAPIMethod::METHOD_STATUS_DEPRECATED => $this->isDeprecated,
ConduitAPIMethod::METHOD_STATUS_UNSTABLE => $this->isUnstable,
);
// Only apply status filters if any of them are set.
if (array_filter($status)) {
foreach ($methods as $key => $method) {
$keep = idx($status, $method->getMethodStatus());
if (!$keep) {
unset($methods[$key]);
}
}
}
if ($this->applicationNames) {
$map = array_fuse($this->applicationNames);
foreach ($methods as $key => $method) {
$needle = $method->getApplicationName();
$needle = phutil_utf8_strtolower($needle);
if (empty($map[$needle])) {
unset($methods[$key]);
}
}
}
if ($this->nameContains) {
$needle = phutil_utf8_strtolower($this->nameContains);
foreach ($methods as $key => $method) {
$haystack = $method->getAPIMethodName();
$haystack = phutil_utf8_strtolower($haystack);
if (strpos($haystack, $needle) === false) {
unset($methods[$key]);
}
}
}
if ($this->methods) {
$map = array_fuse($this->methods);
foreach ($methods as $key => $method) {
$needle = $method->getAPIMethodName();
if (empty($map[$needle])) {
unset($methods[$key]);
}
}
}
return $methods;
}
}

View file

@ -0,0 +1,137 @@
<?php
final class PhabricatorConduitSearchEngine
extends PhabricatorApplicationSearchEngine {
public function getPageSize(PhabricatorSavedQuery $saved) {
return INF;
}
public function buildSavedQueryFromRequest(AphrontRequest $request) {
$saved = new PhabricatorSavedQuery();
$saved->setParameter('isStable', $request->getStr('isStable'));
$saved->setParameter('isUnstable', $request->getStr('isUnstable'));
$saved->setParameter('isDeprecated', $request->getStr('isDeprecated'));
$saved->setParameter(
'applicationNames',
$request->getStrList('applicationNames'));
$saved->setParameter('nameContains', $request->getStr('nameContains'));
return $saved;
}
public function buildQueryFromSavedQuery(PhabricatorSavedQuery $saved) {
$query = id(new PhabricatorConduitMethodQuery());
$query->withIsStable($saved->getParameter('isStable'));
$query->withIsUnstable($saved->getParameter('isUnstable'));
$query->withIsDeprecated($saved->getParameter('isDeprecated'));
$names = $saved->getParameter('applicationNames', array());
if ($names) {
$query->withApplicationNames($names);
}
$contains = $saved->getParameter('nameContains');
if (strlen($contains)) {
$query->withNameContains($contains);
}
return $query;
}
public function buildSearchForm(
AphrontFormView $form,
PhabricatorSavedQuery $saved) {
$form
->appendChild(
id(new AphrontFormTextControl())
->setLabel('Name Contains')
->setName('nameContains')
->setValue($saved->getParameter('nameContains')));
$names = $saved->getParameter('applicationNames', array());
$form
->appendChild(
id(new AphrontFormTextControl())
->setLabel('Applications')
->setName('applicationNames')
->setValue(implode(', ', $names))
->setCaption(
pht('Example: %s', hsprintf('<tt>differential, paste</tt>'))));
$is_stable = $saved->getParameter('isStable');
$is_unstable = $saved->getParameter('isUnstable');
$is_deprecated = $saved->getParameter('isDeprecated');
$form
->appendChild(
id(new AphrontFormCheckboxControl())
->setLabel('Stability')
->addCheckbox(
'isStable',
1,
hsprintf(
'<strong>%s</strong>: %s',
pht('Stable Methods'),
pht('Show established API methods with stable interfaces.')),
$is_stable)
->addCheckbox(
'isUnstable',
1,
hsprintf(
'<strong>%s</strong>: %s',
pht('Unstable Methods'),
pht('Show new methods which are subject to change.')),
$is_unstable)
->addCheckbox(
'isDeprecated',
1,
hsprintf(
'<strong>%s</strong>: %s',
pht('Deprecated Methods'),
pht(
'Show old methods which will be deleted in a future '.
'version of Phabricator.')),
$is_deprecated));
}
protected function getURI($path) {
return '/conduit/'.$path;
}
public function getBuiltinQueryNames() {
$names = array(
'modern' => pht('Modern Methods'),
'all' => pht('All Methods'),
);
return $names;
}
public function buildSavedQueryFromBuiltin($query_key) {
$query = $this->newSavedQuery();
$query->setQueryKey($query_key);
switch ($query_key) {
case 'modern':
return $query
->setParameter('isStable', true)
->setParameter('isUnstable', true);
case 'all':
return $query
->setParameter('isStable', true)
->setParameter('isUnstable', true)
->setParameter('isDeprecated', true);
}
return parent::buildSavedQueryFromBuiltin($query_key);
}
}

View file

@ -173,16 +173,17 @@ final class PhabricatorApplicationSearchController
$pager = new AphrontCursorPagerView(); $pager = new AphrontCursorPagerView();
$pager->readFromRequest($request); $pager->readFromRequest($request);
$pager->setPageSize($engine->getPageSize($saved_query));
$objects = $query->setViewer($request->getUser()) $objects = $query->setViewer($request->getUser())
->executeWithCursorPager($pager); ->executeWithCursorPager($pager);
$list = $parent->renderResultsList($objects); $list = $parent->renderResultsList($objects);
$list->setNoDataString(pht("No results found for this query."));
$nav->appendChild($list); $nav->appendChild($list);
// TODO: This is a bit hacky. // TODO: This is a bit hacky.
if ($list instanceof PhabricatorObjectItemListView) { if ($list instanceof PhabricatorObjectItemListView) {
$list->setNoDataString(pht("No results found for this query."));
$list->setPager($pager); $list->setPager($pager);
} else { } else {
$nav->appendChild($pager); $nav->appendChild($pager);

View file

@ -305,4 +305,14 @@ abstract class PhabricatorApplicationSearchEngine {
->setLabel($end_name) ->setLabel($end_name)
->setValue($end_str)); ->setValue($end_str));
} }
/* -( Pagination )--------------------------------------------------------- */
public function getPageSize(PhabricatorSavedQuery $saved) {
return $saved->getParameter('limit', 100);
}
} }