mirror of
https://we.phorge.it/source/phorge.git
synced 2024-11-18 12:52:42 +01:00
Convert OAuthServer to Transactions + EditEngine
Summary: Ref T7303. This application is currently stone-age tech (no transactions, hard "delete" action). Bring it up to modern specs. Test Plan: - Created and edited an OAuth application. - Viewed transaction record. - Tried to create something with no name, invalid redirect URI, etc. Was gently rebuffed with detailed explanatory errors. Reviewers: chad Reviewed By: chad Maniphest Tasks: T7303 Differential Revision: https://secure.phabricator.com/D15609
This commit is contained in:
parent
e2685a248b
commit
57f016b166
13 changed files with 416 additions and 166 deletions
19
resources/sql/autopatches/20160404.oauth.1.xaction.sql
Normal file
19
resources/sql/autopatches/20160404.oauth.1.xaction.sql
Normal file
|
@ -0,0 +1,19 @@
|
||||||
|
CREATE TABLE {$NAMESPACE}_oauth_server.oauth_server_transaction (
|
||||||
|
id INT UNSIGNED NOT NULL AUTO_INCREMENT PRIMARY KEY,
|
||||||
|
phid VARBINARY(64) NOT NULL,
|
||||||
|
authorPHID VARBINARY(64) NOT NULL,
|
||||||
|
objectPHID VARBINARY(64) NOT NULL,
|
||||||
|
viewPolicy VARBINARY(64) NOT NULL,
|
||||||
|
editPolicy VARBINARY(64) NOT NULL,
|
||||||
|
commentPHID VARBINARY(64) DEFAULT NULL,
|
||||||
|
commentVersion INT UNSIGNED NOT NULL,
|
||||||
|
transactionType VARCHAR(32) COLLATE {$COLLATE_TEXT} NOT NULL,
|
||||||
|
oldValue LONGTEXT COLLATE {$COLLATE_TEXT} NOT NULL,
|
||||||
|
newValue LONGTEXT COLLATE {$COLLATE_TEXT} NOT NULL,
|
||||||
|
contentSource LONGTEXT COLLATE {$COLLATE_TEXT} NOT NULL,
|
||||||
|
metadata LONGTEXT COLLATE {$COLLATE_TEXT} NOT NULL,
|
||||||
|
dateCreated INT UNSIGNED NOT NULL,
|
||||||
|
dateModified INT UNSIGNED NOT NULL,
|
||||||
|
UNIQUE KEY `key_phid` (`phid`),
|
||||||
|
KEY `key_object` (`objectPHID`)
|
||||||
|
) ENGINE=InnoDB, COLLATE {$COLLATE_TEXT};
|
|
@ -2723,9 +2723,13 @@ phutil_register_library_map(array(
|
||||||
'PhabricatorOAuthServerController' => 'applications/oauthserver/controller/PhabricatorOAuthServerController.php',
|
'PhabricatorOAuthServerController' => 'applications/oauthserver/controller/PhabricatorOAuthServerController.php',
|
||||||
'PhabricatorOAuthServerCreateClientsCapability' => 'applications/oauthserver/capability/PhabricatorOAuthServerCreateClientsCapability.php',
|
'PhabricatorOAuthServerCreateClientsCapability' => 'applications/oauthserver/capability/PhabricatorOAuthServerCreateClientsCapability.php',
|
||||||
'PhabricatorOAuthServerDAO' => 'applications/oauthserver/storage/PhabricatorOAuthServerDAO.php',
|
'PhabricatorOAuthServerDAO' => 'applications/oauthserver/storage/PhabricatorOAuthServerDAO.php',
|
||||||
|
'PhabricatorOAuthServerEditEngine' => 'applications/oauthserver/editor/PhabricatorOAuthServerEditEngine.php',
|
||||||
|
'PhabricatorOAuthServerEditor' => 'applications/oauthserver/editor/PhabricatorOAuthServerEditor.php',
|
||||||
'PhabricatorOAuthServerScope' => 'applications/oauthserver/PhabricatorOAuthServerScope.php',
|
'PhabricatorOAuthServerScope' => 'applications/oauthserver/PhabricatorOAuthServerScope.php',
|
||||||
'PhabricatorOAuthServerTestCase' => 'applications/oauthserver/__tests__/PhabricatorOAuthServerTestCase.php',
|
'PhabricatorOAuthServerTestCase' => 'applications/oauthserver/__tests__/PhabricatorOAuthServerTestCase.php',
|
||||||
'PhabricatorOAuthServerTokenController' => 'applications/oauthserver/controller/PhabricatorOAuthServerTokenController.php',
|
'PhabricatorOAuthServerTokenController' => 'applications/oauthserver/controller/PhabricatorOAuthServerTokenController.php',
|
||||||
|
'PhabricatorOAuthServerTransaction' => 'applications/oauthserver/storage/PhabricatorOAuthServerTransaction.php',
|
||||||
|
'PhabricatorOAuthServerTransactionQuery' => 'applications/oauthserver/query/PhabricatorOAuthServerTransactionQuery.php',
|
||||||
'PhabricatorObjectHandle' => 'applications/phid/PhabricatorObjectHandle.php',
|
'PhabricatorObjectHandle' => 'applications/phid/PhabricatorObjectHandle.php',
|
||||||
'PhabricatorObjectHasAsanaSubtaskEdgeType' => 'applications/doorkeeper/edge/PhabricatorObjectHasAsanaSubtaskEdgeType.php',
|
'PhabricatorObjectHasAsanaSubtaskEdgeType' => 'applications/doorkeeper/edge/PhabricatorObjectHasAsanaSubtaskEdgeType.php',
|
||||||
'PhabricatorObjectHasAsanaTaskEdgeType' => 'applications/doorkeeper/edge/PhabricatorObjectHasAsanaTaskEdgeType.php',
|
'PhabricatorObjectHasAsanaTaskEdgeType' => 'applications/doorkeeper/edge/PhabricatorObjectHasAsanaTaskEdgeType.php',
|
||||||
|
@ -7206,6 +7210,7 @@ phutil_register_library_map(array(
|
||||||
'PhabricatorOAuthServerClient' => array(
|
'PhabricatorOAuthServerClient' => array(
|
||||||
'PhabricatorOAuthServerDAO',
|
'PhabricatorOAuthServerDAO',
|
||||||
'PhabricatorPolicyInterface',
|
'PhabricatorPolicyInterface',
|
||||||
|
'PhabricatorApplicationTransactionInterface',
|
||||||
'PhabricatorDestructibleInterface',
|
'PhabricatorDestructibleInterface',
|
||||||
),
|
),
|
||||||
'PhabricatorOAuthServerClientAuthorizationPHIDType' => 'PhabricatorPHIDType',
|
'PhabricatorOAuthServerClientAuthorizationPHIDType' => 'PhabricatorPHIDType',
|
||||||
|
@ -7215,9 +7220,13 @@ phutil_register_library_map(array(
|
||||||
'PhabricatorOAuthServerController' => 'PhabricatorController',
|
'PhabricatorOAuthServerController' => 'PhabricatorController',
|
||||||
'PhabricatorOAuthServerCreateClientsCapability' => 'PhabricatorPolicyCapability',
|
'PhabricatorOAuthServerCreateClientsCapability' => 'PhabricatorPolicyCapability',
|
||||||
'PhabricatorOAuthServerDAO' => 'PhabricatorLiskDAO',
|
'PhabricatorOAuthServerDAO' => 'PhabricatorLiskDAO',
|
||||||
|
'PhabricatorOAuthServerEditEngine' => 'PhabricatorEditEngine',
|
||||||
|
'PhabricatorOAuthServerEditor' => 'PhabricatorApplicationTransactionEditor',
|
||||||
'PhabricatorOAuthServerScope' => 'Phobject',
|
'PhabricatorOAuthServerScope' => 'Phobject',
|
||||||
'PhabricatorOAuthServerTestCase' => 'PhabricatorTestCase',
|
'PhabricatorOAuthServerTestCase' => 'PhabricatorTestCase',
|
||||||
'PhabricatorOAuthServerTokenController' => 'PhabricatorOAuthServerController',
|
'PhabricatorOAuthServerTokenController' => 'PhabricatorOAuthServerController',
|
||||||
|
'PhabricatorOAuthServerTransaction' => 'PhabricatorApplicationTransaction',
|
||||||
|
'PhabricatorOAuthServerTransactionQuery' => 'PhabricatorApplicationTransactionQuery',
|
||||||
'PhabricatorObjectHandle' => array(
|
'PhabricatorObjectHandle' => array(
|
||||||
'Phobject',
|
'Phobject',
|
||||||
'PhabricatorPolicyInterface',
|
'PhabricatorPolicyInterface',
|
||||||
|
|
|
@ -188,24 +188,49 @@ final class PhabricatorOAuthServer extends Phobject {
|
||||||
return $authorization;
|
return $authorization;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public function validateRedirectURI($uri) {
|
||||||
|
try {
|
||||||
|
$this->assertValidRedirectURI($uri);
|
||||||
|
return true;
|
||||||
|
} catch (Exception $ex) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* See http://tools.ietf.org/html/draft-ietf-oauth-v2-23#section-3.1.2
|
* See http://tools.ietf.org/html/draft-ietf-oauth-v2-23#section-3.1.2
|
||||||
* for details on what makes a given redirect URI "valid".
|
* for details on what makes a given redirect URI "valid".
|
||||||
*/
|
*/
|
||||||
public function validateRedirectURI(PhutilURI $uri) {
|
public function assertValidRedirectURI($raw_uri) {
|
||||||
if (!PhabricatorEnv::isValidRemoteURIForLink($uri)) {
|
// This covers basics like reasonable formatting and the existence of a
|
||||||
return false;
|
// protocol.
|
||||||
|
PhabricatorEnv::requireValidRemoteURIForLink($raw_uri);
|
||||||
|
|
||||||
|
$uri = new PhutilURI($raw_uri);
|
||||||
|
|
||||||
|
$fragment = $uri->getFragment();
|
||||||
|
if (strlen($fragment)) {
|
||||||
|
throw new Exception(
|
||||||
|
pht(
|
||||||
|
'OAuth application redirect URIs must not contain URI '.
|
||||||
|
'fragments, but the URI "%s" has a fragment ("%s").',
|
||||||
|
$raw_uri,
|
||||||
|
$fragment));
|
||||||
}
|
}
|
||||||
|
|
||||||
if ($uri->getFragment()) {
|
$protocol = $uri->getProtocol();
|
||||||
return false;
|
switch ($protocol) {
|
||||||
|
case 'http':
|
||||||
|
case 'https':
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
throw new Exception(
|
||||||
|
pht(
|
||||||
|
'OAuth application redirect URIs must only use the "http" or '.
|
||||||
|
'"https" protocols, but the URI "%s" uses the "%s" protocol.',
|
||||||
|
$raw_uri,
|
||||||
|
$protocol));
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!$uri->getDomain()) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
return true;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|
|
@ -51,10 +51,10 @@ final class PhabricatorOAuthServerApplication extends PhabricatorApplication {
|
||||||
=> 'PhabricatorOAuthClientListController',
|
=> 'PhabricatorOAuthClientListController',
|
||||||
'auth/' => 'PhabricatorOAuthServerAuthController',
|
'auth/' => 'PhabricatorOAuthServerAuthController',
|
||||||
'token/' => 'PhabricatorOAuthServerTokenController',
|
'token/' => 'PhabricatorOAuthServerTokenController',
|
||||||
|
$this->getEditRoutePattern('edit/') =>
|
||||||
|
'PhabricatorOAuthClientEditController',
|
||||||
'client/' => array(
|
'client/' => array(
|
||||||
'create/' => 'PhabricatorOAuthClientEditController',
|
|
||||||
'delete/(?P<id>\d+)/' => 'PhabricatorOAuthClientDeleteController',
|
'delete/(?P<id>\d+)/' => 'PhabricatorOAuthClientDeleteController',
|
||||||
'edit/(?P<id>\d+)/' => 'PhabricatorOAuthClientEditController',
|
|
||||||
'view/(?P<id>\d+)/' => 'PhabricatorOAuthClientViewController',
|
'view/(?P<id>\d+)/' => 'PhabricatorOAuthClientViewController',
|
||||||
'secret/(?P<id>\d+)/' => 'PhabricatorOAuthClientSecretController',
|
'secret/(?P<id>\d+)/' => 'PhabricatorOAuthClientSecretController',
|
||||||
'test/(?P<id>\d+)/' => 'PhabricatorOAuthClientTestController',
|
'test/(?P<id>\d+)/' => 'PhabricatorOAuthClientTestController',
|
||||||
|
|
|
@ -4,129 +4,9 @@ final class PhabricatorOAuthClientEditController
|
||||||
extends PhabricatorOAuthClientController {
|
extends PhabricatorOAuthClientController {
|
||||||
|
|
||||||
public function handleRequest(AphrontRequest $request) {
|
public function handleRequest(AphrontRequest $request) {
|
||||||
$viewer = $this->getViewer();
|
return id(new PhabricatorOAuthServerEditEngine())
|
||||||
$id = $request->getURIData('id');
|
->setController($this)
|
||||||
|
->buildResponse();
|
||||||
if ($id) {
|
|
||||||
$client = id(new PhabricatorOAuthServerClientQuery())
|
|
||||||
->setViewer($viewer)
|
|
||||||
->withIDs(array($id))
|
|
||||||
->requireCapabilities(
|
|
||||||
array(
|
|
||||||
PhabricatorPolicyCapability::CAN_VIEW,
|
|
||||||
PhabricatorPolicyCapability::CAN_EDIT,
|
|
||||||
))
|
|
||||||
->executeOne();
|
|
||||||
if (!$client) {
|
|
||||||
return new Aphront404Response();
|
|
||||||
}
|
|
||||||
|
|
||||||
$title = pht('Edit OAuth Application: %s', $client->getName());
|
|
||||||
$submit_button = pht('Save Application');
|
|
||||||
$crumb_text = pht('Edit');
|
|
||||||
$cancel_uri = $client->getViewURI();
|
|
||||||
$is_new = false;
|
|
||||||
} else {
|
|
||||||
$this->requireApplicationCapability(
|
|
||||||
PhabricatorOAuthServerCreateClientsCapability::CAPABILITY);
|
|
||||||
|
|
||||||
$client = PhabricatorOAuthServerClient::initializeNewClient($viewer);
|
|
||||||
|
|
||||||
$title = pht('Create OAuth Application');
|
|
||||||
$submit_button = pht('Create Application');
|
|
||||||
$crumb_text = pht('Create Application');
|
|
||||||
$cancel_uri = $this->getApplicationURI();
|
|
||||||
$is_new = true;
|
|
||||||
}
|
|
||||||
|
|
||||||
$errors = array();
|
|
||||||
$e_redirect = true;
|
|
||||||
$e_name = true;
|
|
||||||
if ($request->isFormPost()) {
|
|
||||||
$redirect_uri = $request->getStr('redirect_uri');
|
|
||||||
$client->setName($request->getStr('name'));
|
|
||||||
$client->setRedirectURI($redirect_uri);
|
|
||||||
|
|
||||||
if (!strlen($client->getName())) {
|
|
||||||
$errors[] = pht('You must choose a name for this OAuth application.');
|
|
||||||
$e_name = pht('Required');
|
|
||||||
}
|
|
||||||
|
|
||||||
$server = new PhabricatorOAuthServer();
|
|
||||||
$uri = new PhutilURI($redirect_uri);
|
|
||||||
if (!$server->validateRedirectURI($uri)) {
|
|
||||||
$errors[] = pht(
|
|
||||||
'Redirect URI must be a fully qualified domain name '.
|
|
||||||
'with no fragments. See %s for more information on the correct '.
|
|
||||||
'format.',
|
|
||||||
'http://tools.ietf.org/html/draft-ietf-oauth-v2-23#section-3.1.2');
|
|
||||||
$e_redirect = pht('Invalid');
|
|
||||||
}
|
|
||||||
|
|
||||||
$client->setViewPolicy($request->getStr('viewPolicy'));
|
|
||||||
$client->setEditPolicy($request->getStr('editPolicy'));
|
|
||||||
if (!$errors) {
|
|
||||||
$client->save();
|
|
||||||
$view_uri = $client->getViewURI();
|
|
||||||
return id(new AphrontRedirectResponse())->setURI($view_uri);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
$policies = id(new PhabricatorPolicyQuery())
|
|
||||||
->setViewer($viewer)
|
|
||||||
->setObject($client)
|
|
||||||
->execute();
|
|
||||||
|
|
||||||
$form = id(new AphrontFormView())
|
|
||||||
->setUser($viewer)
|
|
||||||
->appendChild(
|
|
||||||
id(new AphrontFormTextControl())
|
|
||||||
->setLabel(pht('Name'))
|
|
||||||
->setName('name')
|
|
||||||
->setValue($client->getName())
|
|
||||||
->setError($e_name))
|
|
||||||
->appendChild(
|
|
||||||
id(new AphrontFormTextControl())
|
|
||||||
->setLabel(pht('Redirect URI'))
|
|
||||||
->setName('redirect_uri')
|
|
||||||
->setValue($client->getRedirectURI())
|
|
||||||
->setError($e_redirect))
|
|
||||||
->appendChild(
|
|
||||||
id(new AphrontFormPolicyControl())
|
|
||||||
->setUser($viewer)
|
|
||||||
->setCapability(PhabricatorPolicyCapability::CAN_VIEW)
|
|
||||||
->setPolicyObject($client)
|
|
||||||
->setPolicies($policies)
|
|
||||||
->setName('viewPolicy'))
|
|
||||||
->appendChild(
|
|
||||||
id(new AphrontFormPolicyControl())
|
|
||||||
->setUser($viewer)
|
|
||||||
->setCapability(PhabricatorPolicyCapability::CAN_EDIT)
|
|
||||||
->setPolicyObject($client)
|
|
||||||
->setPolicies($policies)
|
|
||||||
->setName('editPolicy'))
|
|
||||||
->appendChild(
|
|
||||||
id(new AphrontFormSubmitControl())
|
|
||||||
->addCancelButton($cancel_uri)
|
|
||||||
->setValue($submit_button));
|
|
||||||
|
|
||||||
$crumbs = $this->buildApplicationCrumbs();
|
|
||||||
if (!$is_new) {
|
|
||||||
$crumbs->addTextCrumb(
|
|
||||||
$client->getName(),
|
|
||||||
$client->getViewURI());
|
|
||||||
}
|
|
||||||
$crumbs->addTextCrumb($crumb_text);
|
|
||||||
|
|
||||||
$box = id(new PHUIObjectBoxView())
|
|
||||||
->setHeaderText($title)
|
|
||||||
->setFormErrors($errors)
|
|
||||||
->setForm($form);
|
|
||||||
|
|
||||||
return $this->newPage()
|
|
||||||
->setCrumbs($crumbs)
|
|
||||||
->setTitle($title)
|
|
||||||
->appendChild($box);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -3,38 +3,22 @@
|
||||||
final class PhabricatorOAuthClientListController
|
final class PhabricatorOAuthClientListController
|
||||||
extends PhabricatorOAuthClientController {
|
extends PhabricatorOAuthClientController {
|
||||||
|
|
||||||
private $queryKey;
|
|
||||||
|
|
||||||
public function shouldAllowPublic() {
|
public function shouldAllowPublic() {
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
public function willProcessRequest(array $data) {
|
public function handleRequest(AphrontRequest $request) {
|
||||||
$this->queryKey = idx($data, 'queryKey');
|
return id(new PhabricatorOAuthServerClientSearchEngine())
|
||||||
}
|
->setController($this)
|
||||||
|
->buildResponse();
|
||||||
public function processRequest() {
|
|
||||||
$controller = id(new PhabricatorApplicationSearchController())
|
|
||||||
->setQueryKey($this->queryKey)
|
|
||||||
->setSearchEngine(new PhabricatorOAuthServerClientSearchEngine())
|
|
||||||
->setNavigation($this->buildSideNavView());
|
|
||||||
|
|
||||||
return $this->delegateToController($controller);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
protected function buildApplicationCrumbs() {
|
protected function buildApplicationCrumbs() {
|
||||||
$crumbs = parent::buildApplicationCrumbs();
|
$crumbs = parent::buildApplicationCrumbs();
|
||||||
|
|
||||||
$can_create = $this->hasApplicationCapability(
|
id(new PhabricatorOAuthServerEditEngine())
|
||||||
PhabricatorOAuthServerCreateClientsCapability::CAPABILITY);
|
->setViewer($this->getViewer())
|
||||||
|
->addActionToCrumbs($crumbs);
|
||||||
$crumbs->addAction(
|
|
||||||
id(new PHUIListItemView())
|
|
||||||
->setHref($this->getApplicationURI('client/create/'))
|
|
||||||
->setName(pht('Create Application'))
|
|
||||||
->setDisabled(!$can_create)
|
|
||||||
->setWorkflow(!$can_create)
|
|
||||||
->setIcon('fa-plus-square'));
|
|
||||||
|
|
||||||
return $crumbs;
|
return $crumbs;
|
||||||
}
|
}
|
||||||
|
|
|
@ -22,6 +22,11 @@ final class PhabricatorOAuthClientViewController
|
||||||
$crumbs = $this->buildApplicationCrumbs();
|
$crumbs = $this->buildApplicationCrumbs();
|
||||||
$crumbs->addTextCrumb($client->getName());
|
$crumbs->addTextCrumb($client->getName());
|
||||||
|
|
||||||
|
$timeline = $this->buildTransactionTimeline(
|
||||||
|
$client,
|
||||||
|
new PhabricatorOAuthServerTransactionQuery());
|
||||||
|
$timeline->setShouldTerminate(true);
|
||||||
|
|
||||||
$box = id(new PHUIObjectBoxView())
|
$box = id(new PHUIObjectBoxView())
|
||||||
->setHeader($header)
|
->setHeader($header)
|
||||||
->addPropertyList($properties);
|
->addPropertyList($properties);
|
||||||
|
@ -31,7 +36,11 @@ final class PhabricatorOAuthClientViewController
|
||||||
return $this->newPage()
|
return $this->newPage()
|
||||||
->setCrumbs($crumbs)
|
->setCrumbs($crumbs)
|
||||||
->setTitle($title)
|
->setTitle($title)
|
||||||
->appendChild($box);
|
->appendChild(
|
||||||
|
array(
|
||||||
|
$box,
|
||||||
|
$timeline,
|
||||||
|
));
|
||||||
}
|
}
|
||||||
|
|
||||||
private function buildHeaderView(PhabricatorOAuthServerClient $client) {
|
private function buildHeaderView(PhabricatorOAuthServerClient $client) {
|
||||||
|
|
|
@ -0,0 +1,100 @@
|
||||||
|
<?php
|
||||||
|
|
||||||
|
final class PhabricatorOAuthServerEditEngine
|
||||||
|
extends PhabricatorEditEngine {
|
||||||
|
|
||||||
|
const ENGINECONST = 'oauthserver.application';
|
||||||
|
|
||||||
|
public function isEngineConfigurable() {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getEngineName() {
|
||||||
|
return pht('OAuth Applications');
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getSummaryHeader() {
|
||||||
|
return pht('Edit OAuth Applications');
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getSummaryText() {
|
||||||
|
return pht('This engine manages OAuth client applications.');
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getEngineApplicationClass() {
|
||||||
|
return 'PhabricatorOAuthServerApplication';
|
||||||
|
}
|
||||||
|
|
||||||
|
protected function newEditableObject() {
|
||||||
|
return PhabricatorOAuthServerClient::initializeNewClient(
|
||||||
|
$this->getViewer());
|
||||||
|
}
|
||||||
|
|
||||||
|
protected function newObjectQuery() {
|
||||||
|
return id(new PhabricatorOAuthServerClientQuery());
|
||||||
|
}
|
||||||
|
|
||||||
|
protected function getObjectCreateTitleText($object) {
|
||||||
|
return pht('Create OAuth Server');
|
||||||
|
}
|
||||||
|
|
||||||
|
protected function getObjectCreateButtonText($object) {
|
||||||
|
return pht('Create OAuth Server');
|
||||||
|
}
|
||||||
|
|
||||||
|
protected function getObjectEditTitleText($object) {
|
||||||
|
return pht('Edit OAuth Server: %s', $object->getName());
|
||||||
|
}
|
||||||
|
|
||||||
|
protected function getObjectEditShortText($object) {
|
||||||
|
return pht('Edit OAuth Server');
|
||||||
|
}
|
||||||
|
|
||||||
|
protected function getObjectCreateShortText() {
|
||||||
|
return pht('Create OAuth Server');
|
||||||
|
}
|
||||||
|
|
||||||
|
protected function getObjectName() {
|
||||||
|
return pht('OAuth Server');
|
||||||
|
}
|
||||||
|
|
||||||
|
protected function getObjectViewURI($object) {
|
||||||
|
return $object->getViewURI();
|
||||||
|
}
|
||||||
|
|
||||||
|
protected function getCreateNewObjectPolicy() {
|
||||||
|
return $this->getApplication()->getPolicy(
|
||||||
|
PhabricatorOAuthServerCreateClientsCapability::CAPABILITY);
|
||||||
|
}
|
||||||
|
|
||||||
|
protected function buildCustomEditFields($object) {
|
||||||
|
return array(
|
||||||
|
id(new PhabricatorTextEditField())
|
||||||
|
->setKey('name')
|
||||||
|
->setLabel(pht('Name'))
|
||||||
|
->setIsRequired(true)
|
||||||
|
->setTransactionType(PhabricatorOAuthServerTransaction::TYPE_NAME)
|
||||||
|
->setDescription(pht('The name of the OAuth application.'))
|
||||||
|
->setConduitDescription(pht('Rename the application.'))
|
||||||
|
->setConduitTypeDescription(pht('New application name.'))
|
||||||
|
->setValue($object->getName()),
|
||||||
|
id(new PhabricatorTextEditField())
|
||||||
|
->setKey('redirectURI')
|
||||||
|
->setLabel(pht('Redirect URI'))
|
||||||
|
->setIsRequired(true)
|
||||||
|
->setTransactionType(
|
||||||
|
PhabricatorOAuthServerTransaction::TYPE_REDIRECT_URI)
|
||||||
|
->setDescription(
|
||||||
|
pht('The redirect URI for OAuth handshakes.'))
|
||||||
|
->setConduitDescription(
|
||||||
|
pht(
|
||||||
|
'Change where this application redirects users to during OAuth '.
|
||||||
|
'handshakes.'))
|
||||||
|
->setConduitTypeDescription(
|
||||||
|
pht(
|
||||||
|
'New OAuth application redirect URI.'))
|
||||||
|
->setValue($object->getRedirectURI()),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -0,0 +1,137 @@
|
||||||
|
<?php
|
||||||
|
|
||||||
|
final class PhabricatorOAuthServerEditor
|
||||||
|
extends PhabricatorApplicationTransactionEditor {
|
||||||
|
|
||||||
|
public function getEditorApplicationClass() {
|
||||||
|
return 'PhabricatorOAuthServerApplication';
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getEditorObjectsDescription() {
|
||||||
|
return pht('OAuth Applications');
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getTransactionTypes() {
|
||||||
|
$types = parent::getTransactionTypes();
|
||||||
|
|
||||||
|
$types[] = PhabricatorOAuthServerTransaction::TYPE_NAME;
|
||||||
|
$types[] = PhabricatorOAuthServerTransaction::TYPE_REDIRECT_URI;
|
||||||
|
|
||||||
|
$types[] = PhabricatorTransactions::TYPE_VIEW_POLICY;
|
||||||
|
$types[] = PhabricatorTransactions::TYPE_EDIT_POLICY;
|
||||||
|
|
||||||
|
return $types;
|
||||||
|
}
|
||||||
|
|
||||||
|
protected function getCustomTransactionOldValue(
|
||||||
|
PhabricatorLiskDAO $object,
|
||||||
|
PhabricatorApplicationTransaction $xaction) {
|
||||||
|
|
||||||
|
switch ($xaction->getTransactionType()) {
|
||||||
|
case PhabricatorOAuthServerTransaction::TYPE_NAME:
|
||||||
|
return $object->getName();
|
||||||
|
case PhabricatorOAuthServerTransaction::TYPE_REDIRECT_URI:
|
||||||
|
return $object->getRedirectURI();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
protected function getCustomTransactionNewValue(
|
||||||
|
PhabricatorLiskDAO $object,
|
||||||
|
PhabricatorApplicationTransaction $xaction) {
|
||||||
|
|
||||||
|
switch ($xaction->getTransactionType()) {
|
||||||
|
case PhabricatorOAuthServerTransaction::TYPE_NAME:
|
||||||
|
case PhabricatorOAuthServerTransaction::TYPE_REDIRECT_URI:
|
||||||
|
return $xaction->getNewValue();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
protected function applyCustomInternalTransaction(
|
||||||
|
PhabricatorLiskDAO $object,
|
||||||
|
PhabricatorApplicationTransaction $xaction) {
|
||||||
|
|
||||||
|
switch ($xaction->getTransactionType()) {
|
||||||
|
case PhabricatorOAuthServerTransaction::TYPE_NAME:
|
||||||
|
$object->setName($xaction->getNewValue());
|
||||||
|
return;
|
||||||
|
case PhabricatorOAuthServerTransaction::TYPE_REDIRECT_URI:
|
||||||
|
$object->setRedirectURI($xaction->getNewValue());
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
return parent::applyCustomInternalTransaction($object, $xaction);
|
||||||
|
}
|
||||||
|
|
||||||
|
protected function applyCustomExternalTransaction(
|
||||||
|
PhabricatorLiskDAO $object,
|
||||||
|
PhabricatorApplicationTransaction $xaction) {
|
||||||
|
|
||||||
|
switch ($xaction->getTransactionType()) {
|
||||||
|
case PhabricatorOAuthServerTransaction::TYPE_NAME:
|
||||||
|
case PhabricatorOAuthServerTransaction::TYPE_REDIRECT_URI:
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
return parent::applyCustomExternalTransaction($object, $xaction);
|
||||||
|
}
|
||||||
|
|
||||||
|
protected function validateTransaction(
|
||||||
|
PhabricatorLiskDAO $object,
|
||||||
|
$type,
|
||||||
|
array $xactions) {
|
||||||
|
|
||||||
|
$errors = parent::validateTransaction($object, $type, $xactions);
|
||||||
|
|
||||||
|
switch ($type) {
|
||||||
|
case PhabricatorOAuthServerTransaction::TYPE_NAME:
|
||||||
|
$missing = $this->validateIsEmptyTextField(
|
||||||
|
$object->getName(),
|
||||||
|
$xactions);
|
||||||
|
|
||||||
|
if ($missing) {
|
||||||
|
$error = new PhabricatorApplicationTransactionValidationError(
|
||||||
|
$type,
|
||||||
|
pht('Required'),
|
||||||
|
pht('OAuth applications must have a name.'),
|
||||||
|
nonempty(last($xactions), null));
|
||||||
|
|
||||||
|
$error->setIsMissingFieldError(true);
|
||||||
|
$errors[] = $error;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case PhabricatorOAuthServerTransaction::TYPE_REDIRECT_URI:
|
||||||
|
$missing = $this->validateIsEmptyTextField(
|
||||||
|
$object->getRedirectURI(),
|
||||||
|
$xactions);
|
||||||
|
if ($missing) {
|
||||||
|
$error = new PhabricatorApplicationTransactionValidationError(
|
||||||
|
$type,
|
||||||
|
pht('Required'),
|
||||||
|
pht('OAuth applications must have a valid redirect URI.'),
|
||||||
|
nonempty(last($xactions), null));
|
||||||
|
|
||||||
|
$error->setIsMissingFieldError(true);
|
||||||
|
$errors[] = $error;
|
||||||
|
} else {
|
||||||
|
foreach ($xactions as $xaction) {
|
||||||
|
$redirect_uri = $xaction->getNewValue();
|
||||||
|
|
||||||
|
try {
|
||||||
|
$server = new PhabricatorOAuthServer();
|
||||||
|
$server->assertValidRedirectURI($redirect_uri);
|
||||||
|
} catch (Exception $ex) {
|
||||||
|
$errors[] = new PhabricatorApplicationTransactionValidationError(
|
||||||
|
$type,
|
||||||
|
pht('Invalid'),
|
||||||
|
$ex->getMessage(),
|
||||||
|
$xaction);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
return $errors;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -0,0 +1,10 @@
|
||||||
|
<?php
|
||||||
|
|
||||||
|
final class PhabricatorOAuthServerTransactionQuery
|
||||||
|
extends PhabricatorApplicationTransactionQuery {
|
||||||
|
|
||||||
|
public function getTemplateApplicationTransaction() {
|
||||||
|
return new PhabricatorOAuthServerTransaction();
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -4,6 +4,7 @@ final class PhabricatorOAuthServerClient
|
||||||
extends PhabricatorOAuthServerDAO
|
extends PhabricatorOAuthServerDAO
|
||||||
implements
|
implements
|
||||||
PhabricatorPolicyInterface,
|
PhabricatorPolicyInterface,
|
||||||
|
PhabricatorApplicationTransactionInterface,
|
||||||
PhabricatorDestructibleInterface {
|
PhabricatorDestructibleInterface {
|
||||||
|
|
||||||
protected $secret;
|
protected $secret;
|
||||||
|
@ -16,7 +17,7 @@ final class PhabricatorOAuthServerClient
|
||||||
|
|
||||||
public function getEditURI() {
|
public function getEditURI() {
|
||||||
$id = $this->getID();
|
$id = $this->getID();
|
||||||
return "/oauthserver/client/edit/{$id}/";
|
return "/oauthserver/edit/{$id}/";
|
||||||
}
|
}
|
||||||
|
|
||||||
public function getViewURI() {
|
public function getViewURI() {
|
||||||
|
@ -92,8 +93,32 @@ final class PhabricatorOAuthServerClient
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/* -( PhabricatorApplicationTransactionInterface )------------------------- */
|
||||||
|
|
||||||
|
|
||||||
|
public function getApplicationTransactionEditor() {
|
||||||
|
return new PhabricatorOAuthServerEditor();
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getApplicationTransactionObject() {
|
||||||
|
return $this;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getApplicationTransactionTemplate() {
|
||||||
|
return new PhabricatorOAuthServerTransaction();
|
||||||
|
}
|
||||||
|
|
||||||
|
public function willRenderTimeline(
|
||||||
|
PhabricatorApplicationTransactionView $timeline,
|
||||||
|
AphrontRequest $request) {
|
||||||
|
return $timeline;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
/* -( PhabricatorDestructibleInterface )----------------------------------- */
|
/* -( PhabricatorDestructibleInterface )----------------------------------- */
|
||||||
|
|
||||||
|
|
||||||
public function destroyObjectPermanently(
|
public function destroyObjectPermanently(
|
||||||
PhabricatorDestructionEngine $engine) {
|
PhabricatorDestructionEngine $engine) {
|
||||||
|
|
||||||
|
|
|
@ -0,0 +1,52 @@
|
||||||
|
<?php
|
||||||
|
|
||||||
|
final class PhabricatorOAuthServerTransaction
|
||||||
|
extends PhabricatorApplicationTransaction {
|
||||||
|
|
||||||
|
const TYPE_NAME = 'oauthserver.name';
|
||||||
|
const TYPE_REDIRECT_URI = 'oauthserver.redirect-uri';
|
||||||
|
|
||||||
|
public function getApplicationName() {
|
||||||
|
return 'oauth_server';
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getTableName() {
|
||||||
|
return 'oauth_server_transaction';
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getApplicationTransactionType() {
|
||||||
|
return PhabricatorOAuthServerClientPHIDType::TYPECONST;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getApplicationTransactionCommentObject() {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getTitle() {
|
||||||
|
$author_phid = $this->getAuthorPHID();
|
||||||
|
$old = $this->getOldValue();
|
||||||
|
$new = $this->getNewValue();
|
||||||
|
|
||||||
|
switch ($this->getTransactionType()) {
|
||||||
|
case PhabricatorTransactions::TYPE_CREATE:
|
||||||
|
return pht(
|
||||||
|
'%s created this OAuth application.',
|
||||||
|
$this->renderHandleLink($author_phid));
|
||||||
|
case self::TYPE_NAME:
|
||||||
|
return pht(
|
||||||
|
'%s renamed this application from "%s" to "%s".',
|
||||||
|
$this->renderHandleLink($author_phid),
|
||||||
|
$old,
|
||||||
|
$new);
|
||||||
|
case self::TYPE_REDIRECT_URI:
|
||||||
|
return pht(
|
||||||
|
'%s changed the application redirect URI from "%s" to "%s".',
|
||||||
|
$this->renderHandleLink($author_phid),
|
||||||
|
$old,
|
||||||
|
$new);
|
||||||
|
}
|
||||||
|
|
||||||
|
return parent::getTitle();
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
10
src/infrastructure/env/PhabricatorEnv.php
vendored
10
src/infrastructure/env/PhabricatorEnv.php
vendored
|
@ -575,8 +575,8 @@ final class PhabricatorEnv extends Phobject {
|
||||||
* @return void
|
* @return void
|
||||||
* @task uri
|
* @task uri
|
||||||
*/
|
*/
|
||||||
public static function requireValidRemoteURIForLink($uri) {
|
public static function requireValidRemoteURIForLink($raw_uri) {
|
||||||
$uri = new PhutilURI($uri);
|
$uri = new PhutilURI($raw_uri);
|
||||||
|
|
||||||
$proto = $uri->getProtocol();
|
$proto = $uri->getProtocol();
|
||||||
if (!strlen($proto)) {
|
if (!strlen($proto)) {
|
||||||
|
@ -584,7 +584,7 @@ final class PhabricatorEnv extends Phobject {
|
||||||
pht(
|
pht(
|
||||||
'URI "%s" is not a valid linkable resource. A valid linkable '.
|
'URI "%s" is not a valid linkable resource. A valid linkable '.
|
||||||
'resource URI must specify a protocol.',
|
'resource URI must specify a protocol.',
|
||||||
$uri));
|
$raw_uri));
|
||||||
}
|
}
|
||||||
|
|
||||||
$protocols = self::getEnvConfig('uri.allowed-protocols');
|
$protocols = self::getEnvConfig('uri.allowed-protocols');
|
||||||
|
@ -593,7 +593,7 @@ final class PhabricatorEnv extends Phobject {
|
||||||
pht(
|
pht(
|
||||||
'URI "%s" is not a valid linkable resource. A valid linkable '.
|
'URI "%s" is not a valid linkable resource. A valid linkable '.
|
||||||
'resource URI must use one of these protocols: %s.',
|
'resource URI must use one of these protocols: %s.',
|
||||||
$uri,
|
$raw_uri,
|
||||||
implode(', ', array_keys($protocols))));
|
implode(', ', array_keys($protocols))));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -603,7 +603,7 @@ final class PhabricatorEnv extends Phobject {
|
||||||
pht(
|
pht(
|
||||||
'URI "%s" is not a valid linkable resource. A valid linkable '.
|
'URI "%s" is not a valid linkable resource. A valid linkable '.
|
||||||
'resource URI must specify a domain.',
|
'resource URI must specify a domain.',
|
||||||
$uri));
|
$raw_uri));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue