mirror of
https://we.phorge.it/source/phorge.git
synced 2025-01-07 13:21:02 +01:00
81fa847bc5
Summary: Currently, users get an error when making any changes to this field if they don't have a linked JIRA account. Instead: - We should only raise an error if they're trying to //add// issues, and only on the new issues. It's always fine to remove issues, and existing issues the author can't see are also fine. - When we can't add things because there's no account (vs because there's a permissions error or they don't exist), raise a more tailored exception. Test Plan: - As JIRA and non-JIRA users, made various edits to this field. - Got appropriate exceptions, including better tailoring. Reviewers: btrahan Reviewed By: btrahan Subscribers: mbishopim3, epriestley Differential Revision: https://secure.phabricator.com/D8676
310 lines
7.9 KiB
PHP
310 lines
7.9 KiB
PHP
<?php
|
|
|
|
final class DifferentialJIRAIssuesField
|
|
extends DifferentialStoredCustomField {
|
|
|
|
private $error;
|
|
|
|
public function getFieldKey() {
|
|
return 'phabricator:jira-issues';
|
|
}
|
|
|
|
public function getFieldKeyForConduit() {
|
|
return 'jira.issues';
|
|
}
|
|
|
|
public function isFieldEnabled() {
|
|
return (bool)PhabricatorAuthProviderOAuth1JIRA::getJIRAProvider();
|
|
}
|
|
|
|
public function canDisableField() {
|
|
return false;
|
|
}
|
|
|
|
public function getValueForStorage() {
|
|
return json_encode($this->getValue());
|
|
}
|
|
|
|
public function setValueFromStorage($value) {
|
|
$this->setValue(phutil_json_decode($value));
|
|
return $this;
|
|
}
|
|
|
|
public function getFieldName() {
|
|
return pht('JIRA Issues');
|
|
}
|
|
|
|
public function getFieldDescription() {
|
|
return pht('Lists associated JIRA issues.');
|
|
}
|
|
|
|
public function shouldAppearInPropertyView() {
|
|
return true;
|
|
}
|
|
|
|
public function renderPropertyViewLabel() {
|
|
return $this->getFieldName();
|
|
}
|
|
|
|
public function renderPropertyViewValue(array $handles) {
|
|
$xobjs = $this->loadDoorkeeperExternalObjects($this->getValue());
|
|
if (!$xobjs) {
|
|
return null;
|
|
}
|
|
|
|
$links = array();
|
|
foreach ($xobjs as $xobj) {
|
|
$links[] = id(new DoorkeeperTagView())
|
|
->setExternalObject($xobj);
|
|
}
|
|
|
|
return phutil_implode_html(phutil_tag('br'), $links);
|
|
}
|
|
|
|
private function buildDoorkeeperRefs($value) {
|
|
$provider = PhabricatorAuthProviderOAuth1JIRA::getJIRAProvider();
|
|
|
|
$refs = array();
|
|
if ($value) {
|
|
foreach ($value as $jira_key) {
|
|
$refs[] = id(new DoorkeeperObjectRef())
|
|
->setApplicationType(DoorkeeperBridgeJIRA::APPTYPE_JIRA)
|
|
->setApplicationDomain($provider->getProviderDomain())
|
|
->setObjectType(DoorkeeperBridgeJIRA::OBJTYPE_ISSUE)
|
|
->setObjectID($jira_key);
|
|
}
|
|
}
|
|
|
|
return $refs;
|
|
}
|
|
|
|
private function loadDoorkeeperExternalObjects($value) {
|
|
$refs = $this->buildDoorkeeperRefs($value);
|
|
if (!$refs) {
|
|
return array();
|
|
}
|
|
|
|
$xobjs = id(new DoorkeeperExternalObjectQuery())
|
|
->setViewer($this->getViewer())
|
|
->withObjectKeys(mpull($refs, 'getObjectKey'))
|
|
->execute();
|
|
|
|
return $xobjs;
|
|
}
|
|
|
|
public function shouldAppearInEditView() {
|
|
return PhabricatorAuthProviderOAuth1JIRA::getJIRAProvider();
|
|
}
|
|
|
|
public function shouldAppearInApplicationTransactions() {
|
|
return PhabricatorAuthProviderOAuth1JIRA::getJIRAProvider();
|
|
}
|
|
|
|
public function readValueFromRequest(AphrontRequest $request) {
|
|
$this->setValue($request->getStrList($this->getFieldKey()));
|
|
return $this;
|
|
}
|
|
|
|
public function renderEditControl(array $handles) {
|
|
return id(new AphrontFormTextControl())
|
|
->setLabel(pht('JIRA Issues'))
|
|
->setCaption(
|
|
pht('Example: %s', phutil_tag('tt', array(), 'JIS-3, JIS-9')))
|
|
->setName($this->getFieldKey())
|
|
->setValue(implode(', ', nonempty($this->getValue(), array())))
|
|
->setError($this->error);
|
|
}
|
|
|
|
public function getOldValueForApplicationTransactions() {
|
|
return array_unique(nonempty($this->getValue(), array()));
|
|
}
|
|
|
|
public function getNewValueForApplicationTransactions() {
|
|
return array_unique(nonempty($this->getValue(), array()));
|
|
}
|
|
|
|
public function validateApplicationTransactions(
|
|
PhabricatorApplicationTransactionEditor $editor,
|
|
$type,
|
|
array $xactions) {
|
|
|
|
$this->error = null;
|
|
|
|
$errors = parent::validateApplicationTransactions(
|
|
$editor,
|
|
$type,
|
|
$xactions);
|
|
|
|
$transaction = null;
|
|
foreach ($xactions as $xaction) {
|
|
$old = $xaction->getOldValue();
|
|
$new = $xaction->getNewValue();
|
|
|
|
$add = array_diff($new, $old);
|
|
if (!$add) {
|
|
continue;
|
|
}
|
|
|
|
// Only check that the actor can see newly added JIRA refs. You're
|
|
// allowed to remove refs or make no-op changes even if you aren't
|
|
// linked to JIRA.
|
|
|
|
try {
|
|
$refs = id(new DoorkeeperImportEngine())
|
|
->setViewer($this->getViewer())
|
|
->setRefs($this->buildDoorkeeperRefs($add))
|
|
->setThrowOnMissingLink(true)
|
|
->execute();
|
|
} catch (DoorkeeperMissingLinkException $ex) {
|
|
$this->error = pht('Not Linked');
|
|
$errors[] = new PhabricatorApplicationTransactionValidationError(
|
|
$type,
|
|
pht('Not Linked'),
|
|
pht(
|
|
'You can not add JIRA issues (%s) to this revision because your '.
|
|
'Phabricator account is not linked to a JIRA account.',
|
|
implode(', ', $add)),
|
|
$xaction);
|
|
continue;
|
|
}
|
|
|
|
$bad = array();
|
|
foreach ($refs as $ref) {
|
|
if (!$ref->getIsVisible()) {
|
|
$bad[] = $ref->getObjectID();
|
|
}
|
|
}
|
|
|
|
if ($bad) {
|
|
$bad = implode(', ', $bad);
|
|
$this->error = pht('Invalid');
|
|
|
|
$errors[] = new PhabricatorApplicationTransactionValidationError(
|
|
$type,
|
|
pht('Invalid'),
|
|
pht(
|
|
"Some JIRA issues could not be loaded. They may not exist, or ".
|
|
"you may not have permission to view them: %s",
|
|
$bad),
|
|
$xaction);
|
|
}
|
|
}
|
|
|
|
return $errors;
|
|
}
|
|
|
|
public function getApplicationTransactionTitle(
|
|
PhabricatorApplicationTransaction $xaction) {
|
|
|
|
$old = $xaction->getOldValue();
|
|
if (!is_array($old)) {
|
|
$old = array();
|
|
}
|
|
|
|
$new = $xaction->getNewValue();
|
|
if (!is_array($new)) {
|
|
$new = array();
|
|
}
|
|
|
|
$add = array_diff($new, $old);
|
|
$rem = array_diff($old, $new);
|
|
|
|
$author_phid = $xaction->getAuthorPHID();
|
|
if ($add && $rem) {
|
|
return pht(
|
|
'%s updated JIRA issue(s): added %d %s; removed %d %s.',
|
|
$xaction->renderHandleLink($author_phid),
|
|
new PhutilNumber(count($add)),
|
|
implode(', ', $add),
|
|
new PhutilNumber(count($rem)),
|
|
implode(', ', $rem));
|
|
} else if ($add) {
|
|
return pht(
|
|
'%s added %d JIRA issue(s): %s.',
|
|
$xaction->renderHandleLink($author_phid),
|
|
new PhutilNumber(count($add)),
|
|
implode(', ', $add));
|
|
} else if ($rem) {
|
|
return pht(
|
|
'%s removed %d JIRA issue(s): %s.',
|
|
$xaction->renderHandleLink($author_phid),
|
|
new PhutilNumber(count($rem)),
|
|
implode(', ', $rem));
|
|
}
|
|
|
|
return parent::getApplicationTransactionTitle($xaction);
|
|
}
|
|
|
|
public function applyApplicationTransactionExternalEffects(
|
|
PhabricatorApplicationTransaction $xaction) {
|
|
|
|
// Update the CustomField storage.
|
|
parent::applyApplicationTransactionExternalEffects($xaction);
|
|
|
|
// Now, synchronize the Doorkeeper edges.
|
|
$revision = $this->getObject();
|
|
$revision_phid = $revision->getPHID();
|
|
|
|
$edge_type = PhabricatorEdgeConfig::TYPE_PHOB_HAS_JIRAISSUE;
|
|
$xobjs = $this->loadDoorkeeperExternalObjects($xaction->getNewValue());
|
|
$edge_dsts = mpull($xobjs, 'getPHID');
|
|
|
|
$edges = PhabricatorEdgeQuery::loadDestinationPHIDs(
|
|
$revision_phid,
|
|
$edge_type);
|
|
|
|
$editor = id(new PhabricatorEdgeEditor())
|
|
->setActor($this->getViewer());
|
|
|
|
foreach (array_diff($edges, $edge_dsts) as $rem_edge) {
|
|
$editor->removeEdge($revision_phid, $edge_type, $rem_edge);
|
|
}
|
|
|
|
foreach (array_diff($edge_dsts, $edges) as $add_edge) {
|
|
$editor->addEdge($revision_phid, $edge_type, $add_edge);
|
|
}
|
|
|
|
$editor->save();
|
|
}
|
|
|
|
public function shouldAppearInCommitMessage() {
|
|
return true;
|
|
}
|
|
|
|
public function shouldAppearInCommitMessageTemplate() {
|
|
return true;
|
|
}
|
|
|
|
public function getCommitMessageLabels() {
|
|
return array(
|
|
'JIRA',
|
|
'JIRA Issues',
|
|
'JIRA Issue',
|
|
);
|
|
}
|
|
|
|
public function parseValueFromCommitMessage($value) {
|
|
return preg_split('/[\s,]+/', $value, $limit = -1, PREG_SPLIT_NO_EMPTY);
|
|
}
|
|
|
|
public function readValueFromCommitMessage($value) {
|
|
$this->setValue($value);
|
|
return $this;
|
|
}
|
|
|
|
|
|
|
|
public function renderCommitMessageValue(array $handles) {
|
|
$value = $this->getValue();
|
|
if (!$value) {
|
|
return null;
|
|
}
|
|
return implode(', ', $value);
|
|
}
|
|
|
|
public function shouldAppearInConduitDictionary() {
|
|
return true;
|
|
}
|
|
|
|
|
|
}
|