2011-01-24 09:00:29 -08:00
|
|
|
<?php
|
|
|
|
|
2012-03-09 15:46:25 -08:00
|
|
|
final class PhabricatorConduitConsoleController
|
2011-01-24 09:00:29 -08:00
|
|
|
extends PhabricatorConduitController {
|
|
|
|
|
2014-04-21 15:32:48 -07:00
|
|
|
public function shouldAllowPublic() {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2015-05-08 12:19:52 -07:00
|
|
|
public function handleRequest(AphrontRequest $request) {
|
|
|
|
$viewer = $this->getViewer();
|
|
|
|
$method_name = $request->getURIData('method');
|
2011-01-30 18:52:29 -08:00
|
|
|
|
2013-07-01 12:36:34 -07:00
|
|
|
$method = id(new PhabricatorConduitMethodQuery())
|
|
|
|
->setViewer($viewer)
|
2015-05-08 12:19:52 -07:00
|
|
|
->withMethods(array($method_name))
|
2013-07-01 12:36:34 -07:00
|
|
|
->executeOne();
|
|
|
|
if (!$method) {
|
2012-04-26 22:25:05 -07:00
|
|
|
return new Aphront404Response();
|
2011-01-24 09:00:29 -08:00
|
|
|
}
|
|
|
|
|
2015-05-08 12:19:52 -07:00
|
|
|
$call_uri = '/api/'.$method->getAPIMethodName();
|
2014-04-21 15:32:48 -07:00
|
|
|
|
2013-07-01 12:36:34 -07:00
|
|
|
$status = $method->getMethodStatus();
|
|
|
|
$reason = $method->getMethodStatusDescription();
|
2014-04-21 15:32:48 -07:00
|
|
|
$errors = array();
|
|
|
|
|
|
|
|
switch ($status) {
|
|
|
|
case ConduitAPIMethod::METHOD_STATUS_DEPRECATED:
|
|
|
|
$reason = nonempty($reason, pht('This method is deprecated.'));
|
|
|
|
$errors[] = pht('Deprecated Method: %s', $reason);
|
|
|
|
break;
|
|
|
|
case ConduitAPIMethod::METHOD_STATUS_UNSTABLE:
|
|
|
|
$reason = nonempty(
|
|
|
|
$reason,
|
|
|
|
pht(
|
|
|
|
'This method is new and unstable. Its interface is subject '.
|
|
|
|
'to change.'));
|
|
|
|
$errors[] = pht('Unstable Method: %s', $reason);
|
|
|
|
break;
|
2012-04-18 14:25:27 -07:00
|
|
|
}
|
2011-01-24 09:00:29 -08:00
|
|
|
|
2015-05-08 12:19:52 -07:00
|
|
|
$form = id(new AphrontFormView())
|
|
|
|
->setAction($call_uri)
|
2011-01-30 18:52:29 -08:00
|
|
|
->setUser($request->getUser())
|
2015-05-08 12:19:52 -07:00
|
|
|
->appendRemarkupInstructions(
|
|
|
|
pht(
|
|
|
|
'Enter parameters using **JSON**. For instance, to enter a '.
|
2015-05-22 17:27:56 +10:00
|
|
|
'list, type: `%s`',
|
|
|
|
'["apple", "banana", "cherry"]'));
|
2011-01-24 09:00:29 -08:00
|
|
|
|
2015-04-12 15:59:07 -07:00
|
|
|
$params = $method->getParamTypes();
|
2011-01-24 09:00:29 -08:00
|
|
|
foreach ($params as $param => $desc) {
|
|
|
|
$form->appendChild(
|
|
|
|
id(new AphrontFormTextControl())
|
|
|
|
->setLabel($param)
|
|
|
|
->setName("params[{$param}]")
|
2013-02-05 13:23:05 -08:00
|
|
|
->setCaption($desc));
|
2011-01-24 09:00:29 -08:00
|
|
|
}
|
|
|
|
|
2014-04-21 15:32:48 -07:00
|
|
|
$must_login = !$viewer->isLoggedIn() &&
|
|
|
|
$method->shouldRequireAuthentication();
|
|
|
|
if ($must_login) {
|
|
|
|
$errors[] = pht(
|
|
|
|
'Login Required: This method requires authentication. You must '.
|
|
|
|
'log in before you can make calls to it.');
|
|
|
|
} else {
|
|
|
|
$form
|
|
|
|
->appendChild(
|
|
|
|
id(new AphrontFormSelectControl())
|
2015-05-22 17:27:56 +10:00
|
|
|
->setLabel(pht('Output Format'))
|
2014-04-21 15:32:48 -07:00
|
|
|
->setName('output')
|
|
|
|
->setOptions(
|
|
|
|
array(
|
2015-05-22 17:27:56 +10:00
|
|
|
'human' => pht('Human Readable'),
|
|
|
|
'json' => pht('JSON'),
|
2014-04-21 15:32:48 -07:00
|
|
|
)))
|
|
|
|
->appendChild(
|
|
|
|
id(new AphrontFormSubmitControl())
|
|
|
|
->addCancelButton($this->getApplicationURI())
|
|
|
|
->setValue(pht('Call Method')));
|
|
|
|
}
|
|
|
|
|
|
|
|
$header = id(new PHUIHeaderView())
|
|
|
|
->setUser($viewer)
|
|
|
|
->setHeader($method->getAPIMethodName());
|
2011-01-24 09:00:29 -08:00
|
|
|
|
2013-09-25 11:23:29 -07:00
|
|
|
$form_box = id(new PHUIObjectBoxView())
|
2015-05-08 12:19:52 -07:00
|
|
|
->setHeaderText(pht('Call Method'))
|
[Redesign] Add Table, Collapse support to ObjectBox
Summary: Converts most all tables to be directly set via `setTable` to an ObjectBox. I think this path is more flexible design wise, as we can change the box based on children, and not just CSS. We also already do this with PropertyList, Forms, ObjectList, and Header. `setCollapsed` is added to ObjectBox to all children objects to bleed to the edges (like diffs).
Test Plan: I did a grep of `appendChild($table)` as well as searches for `PHUIObjectBoxView`, also with manual opening of hundreds of files. I'm sure I missed 5-8 places. If you just appendChild($table) nothing breaks, it just looks a little funny.
Reviewers: epriestley, btrahan
Subscribers: Korvin, epriestley
Differential Revision: https://secure.phabricator.com/D12955
2015-05-20 12:43:34 -07:00
|
|
|
->setForm($form);
|
2015-04-12 16:26:57 -07:00
|
|
|
|
|
|
|
$content = array();
|
|
|
|
|
2015-05-08 12:19:52 -07:00
|
|
|
$properties = $this->buildMethodProperties($method);
|
|
|
|
|
|
|
|
$info_box = id(new PHUIObjectBoxView())
|
|
|
|
->setHeaderText(pht('API Method: %s', $method->getAPIMethodName()))
|
|
|
|
->setFormErrors($errors)
|
|
|
|
->appendChild($properties);
|
|
|
|
|
|
|
|
$content[] = $info_box;
|
|
|
|
$content[] = $form_box;
|
|
|
|
$content[] = $this->renderExampleBox($method, null);
|
|
|
|
|
2015-04-12 16:26:57 -07:00
|
|
|
$query = $method->newQueryObject();
|
|
|
|
if ($query) {
|
|
|
|
$orders = $query->getBuiltinOrders();
|
|
|
|
|
|
|
|
$rows = array();
|
|
|
|
foreach ($orders as $key => $order) {
|
|
|
|
$rows[] = array(
|
|
|
|
$key,
|
|
|
|
$order['name'],
|
|
|
|
implode(', ', $order['vector']),
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
$table = id(new AphrontTableView($rows))
|
|
|
|
->setHeaders(
|
|
|
|
array(
|
|
|
|
pht('Key'),
|
|
|
|
pht('Description'),
|
|
|
|
pht('Columns'),
|
|
|
|
))
|
|
|
|
->setColumnClasses(
|
|
|
|
array(
|
|
|
|
'pri',
|
|
|
|
'',
|
|
|
|
'wide',
|
|
|
|
));
|
|
|
|
$content[] = id(new PHUIObjectBoxView())
|
|
|
|
->setHeaderText(pht('Builtin Orders'))
|
[Redesign] Add Table, Collapse support to ObjectBox
Summary: Converts most all tables to be directly set via `setTable` to an ObjectBox. I think this path is more flexible design wise, as we can change the box based on children, and not just CSS. We also already do this with PropertyList, Forms, ObjectList, and Header. `setCollapsed` is added to ObjectBox to all children objects to bleed to the edges (like diffs).
Test Plan: I did a grep of `appendChild($table)` as well as searches for `PHUIObjectBoxView`, also with manual opening of hundreds of files. I'm sure I missed 5-8 places. If you just appendChild($table) nothing breaks, it just looks a little funny.
Reviewers: epriestley, btrahan
Subscribers: Korvin, epriestley
Differential Revision: https://secure.phabricator.com/D12955
2015-05-20 12:43:34 -07:00
|
|
|
->setTable($table);
|
2015-04-12 16:26:57 -07:00
|
|
|
|
|
|
|
$columns = $query->getOrderableColumns();
|
|
|
|
|
|
|
|
$rows = array();
|
|
|
|
foreach ($columns as $key => $column) {
|
|
|
|
$rows[] = array(
|
|
|
|
$key,
|
|
|
|
idx($column, 'unique') ? pht('Yes') : pht('No'),
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
$table = id(new AphrontTableView($rows))
|
|
|
|
->setHeaders(
|
|
|
|
array(
|
|
|
|
pht('Key'),
|
|
|
|
pht('Unique'),
|
|
|
|
))
|
|
|
|
->setColumnClasses(
|
|
|
|
array(
|
|
|
|
'pri',
|
|
|
|
'wide',
|
|
|
|
));
|
|
|
|
$content[] = id(new PHUIObjectBoxView())
|
|
|
|
->setHeaderText(pht('Column Orders'))
|
[Redesign] Add Table, Collapse support to ObjectBox
Summary: Converts most all tables to be directly set via `setTable` to an ObjectBox. I think this path is more flexible design wise, as we can change the box based on children, and not just CSS. We also already do this with PropertyList, Forms, ObjectList, and Header. `setCollapsed` is added to ObjectBox to all children objects to bleed to the edges (like diffs).
Test Plan: I did a grep of `appendChild($table)` as well as searches for `PHUIObjectBoxView`, also with manual opening of hundreds of files. I'm sure I missed 5-8 places. If you just appendChild($table) nothing breaks, it just looks a little funny.
Reviewers: epriestley, btrahan
Subscribers: Korvin, epriestley
Differential Revision: https://secure.phabricator.com/D12955
2015-05-20 12:43:34 -07:00
|
|
|
->setTable($table);
|
2015-04-12 16:26:57 -07:00
|
|
|
}
|
2013-08-26 15:45:58 -07:00
|
|
|
|
2013-07-01 12:36:34 -07:00
|
|
|
$crumbs = $this->buildApplicationCrumbs();
|
2013-12-18 17:47:34 -08:00
|
|
|
$crumbs->addTextCrumb($method->getAPIMethodName());
|
2011-01-24 09:00:29 -08:00
|
|
|
|
2013-07-01 12:36:34 -07:00
|
|
|
return $this->buildApplicationPage(
|
2012-04-18 14:25:27 -07:00
|
|
|
array(
|
2013-07-01 12:36:34 -07:00
|
|
|
$crumbs,
|
2015-04-12 16:26:57 -07:00
|
|
|
$content,
|
2012-04-18 14:25:27 -07:00
|
|
|
),
|
2011-01-24 09:00:29 -08:00
|
|
|
array(
|
2013-07-01 12:36:34 -07:00
|
|
|
'title' => $method->getAPIMethodName(),
|
2011-01-24 09:00:29 -08:00
|
|
|
));
|
|
|
|
}
|
|
|
|
|
2015-05-08 12:19:52 -07:00
|
|
|
private function buildMethodProperties(ConduitAPIMethod $method) {
|
|
|
|
$viewer = $this->getViewer();
|
|
|
|
|
|
|
|
$view = id(new PHUIPropertyListView());
|
|
|
|
|
|
|
|
$view->addProperty(
|
|
|
|
pht('Returns'),
|
|
|
|
$method->getReturnType());
|
|
|
|
|
|
|
|
$error_types = $method->getErrorTypes();
|
|
|
|
$error_types['ERR-CONDUIT-CORE'] = pht('See error message for details.');
|
|
|
|
$error_description = array();
|
|
|
|
foreach ($error_types as $error => $meaning) {
|
|
|
|
$error_description[] = hsprintf(
|
|
|
|
'<li><strong>%s:</strong> %s</li>',
|
|
|
|
$error,
|
|
|
|
$meaning);
|
|
|
|
}
|
|
|
|
$error_description = phutil_tag('ul', array(), $error_description);
|
|
|
|
|
|
|
|
$view->addProperty(
|
|
|
|
pht('Errors'),
|
|
|
|
$error_description);
|
|
|
|
|
|
|
|
|
|
|
|
$description = $method->getMethodDescription();
|
|
|
|
$description = PhabricatorMarkupEngine::renderOneObject(
|
|
|
|
id(new PhabricatorMarkupOneOff())->setContent($description),
|
|
|
|
'default',
|
|
|
|
$viewer);
|
2015-09-19 11:29:01 -07:00
|
|
|
$view->addSectionHeader(
|
|
|
|
pht('Description'), PHUIPropertyListView::ICON_SUMMARY);
|
2015-05-08 12:19:52 -07:00
|
|
|
$view->addTextContent($description);
|
|
|
|
|
|
|
|
return $view;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-01-24 09:00:29 -08:00
|
|
|
}
|