1
0
Fork 0
mirror of https://we.phorge.it/source/phorge.git synced 2025-01-24 05:28:18 +01:00

Legalpad - add a view signatures page

Summary: ...needs to add a LegalpadDocumentSignatureQuery class to get this done, which is also re-deployed everywhere we were issuing raw queries. Ref T3116.

Test Plan: viewed some signatures. Verified color and footer icons showed up how I wanted them to.

Reviewers: epriestley

Reviewed By: epriestley

CC: Korvin, epriestley, aran

Maniphest Tasks: T3116

Differential Revision: https://secure.phabricator.com/D7986
This commit is contained in:
Bob Trahan 2014-01-17 11:40:26 -08:00
parent 117519f396
commit 4e6390774f
9 changed files with 295 additions and 30 deletions

View file

@ -842,6 +842,8 @@ phutil_register_library_map(array(
'LegalpadDocumentSearchEngine' => 'applications/legalpad/query/LegalpadDocumentSearchEngine.php', 'LegalpadDocumentSearchEngine' => 'applications/legalpad/query/LegalpadDocumentSearchEngine.php',
'LegalpadDocumentSignController' => 'applications/legalpad/controller/LegalpadDocumentSignController.php', 'LegalpadDocumentSignController' => 'applications/legalpad/controller/LegalpadDocumentSignController.php',
'LegalpadDocumentSignature' => 'applications/legalpad/storage/LegalpadDocumentSignature.php', 'LegalpadDocumentSignature' => 'applications/legalpad/storage/LegalpadDocumentSignature.php',
'LegalpadDocumentSignatureListController' => 'applications/legalpad/controller/LegalpadDocumentSignatureListController.php',
'LegalpadDocumentSignatureQuery' => 'applications/legalpad/query/LegalpadDocumentSignatureQuery.php',
'LegalpadDocumentSignatureVerificationController' => 'applications/legalpad/controller/LegalpadDocumentSignatureVerificationController.php', 'LegalpadDocumentSignatureVerificationController' => 'applications/legalpad/controller/LegalpadDocumentSignatureVerificationController.php',
'LegalpadDocumentViewController' => 'applications/legalpad/controller/LegalpadDocumentViewController.php', 'LegalpadDocumentViewController' => 'applications/legalpad/controller/LegalpadDocumentViewController.php',
'LegalpadMockMailReceiver' => 'applications/legalpad/mail/LegalpadMockMailReceiver.php', 'LegalpadMockMailReceiver' => 'applications/legalpad/mail/LegalpadMockMailReceiver.php',
@ -3374,7 +3376,13 @@ phutil_register_library_map(array(
'LegalpadDocumentQuery' => 'PhabricatorCursorPagedPolicyAwareQuery', 'LegalpadDocumentQuery' => 'PhabricatorCursorPagedPolicyAwareQuery',
'LegalpadDocumentSearchEngine' => 'PhabricatorApplicationSearchEngine', 'LegalpadDocumentSearchEngine' => 'PhabricatorApplicationSearchEngine',
'LegalpadDocumentSignController' => 'LegalpadController', 'LegalpadDocumentSignController' => 'LegalpadController',
'LegalpadDocumentSignature' => 'LegalpadDAO', 'LegalpadDocumentSignature' =>
array(
0 => 'LegalpadDAO',
1 => 'PhabricatorPolicyInterface',
),
'LegalpadDocumentSignatureListController' => 'LegalpadController',
'LegalpadDocumentSignatureQuery' => 'PhabricatorCursorPagedPolicyAwareQuery',
'LegalpadDocumentSignatureVerificationController' => 'LegalpadController', 'LegalpadDocumentSignatureVerificationController' => 'LegalpadController',
'LegalpadDocumentViewController' => 'LegalpadController', 'LegalpadDocumentViewController' => 'LegalpadController',
'LegalpadMockMailReceiver' => 'PhabricatorObjectMailReceiver', 'LegalpadMockMailReceiver' => 'PhabricatorObjectMailReceiver',

View file

@ -49,7 +49,8 @@ final class PhabricatorApplicationLegalpad extends PhabricatorApplication {
'comment/(?P<id>\d+)/' => 'LegalpadDocumentCommentController', 'comment/(?P<id>\d+)/' => 'LegalpadDocumentCommentController',
'view/(?P<id>\d+)/' => 'LegalpadDocumentViewController', 'view/(?P<id>\d+)/' => 'LegalpadDocumentViewController',
'verify/(?P<code>[^/]+)/' => 'verify/(?P<code>[^/]+)/' =>
'LegalpadDocumentSignatureVerificationController', 'LegalpadDocumentSignatureVerificationController',
'signatures/(?P<id>\d+)/' => 'LegalpadDocumentSignatureListController',
'document/' => array( 'document/' => array(
'preview/' => 'PhabricatorMarkupPreviewController'), 'preview/' => 'PhabricatorMarkupPreviewController'),
)); ));

View file

@ -1,8 +1,5 @@
<?php <?php
/**
* @group legalpad
*/
final class LegalpadDocumentSignController extends LegalpadController { final class LegalpadDocumentSignController extends LegalpadController {
private $id; private $id;
@ -56,12 +53,12 @@ final class LegalpadDocumentSignController extends LegalpadController {
} }
if ($signer_phid) { if ($signer_phid) {
$signature = id(new LegalpadDocumentSignature()) $signature = id(new LegalpadDocumentSignatureQuery())
->loadOneWhere( ->setViewer($user)
'documentPHID = %s AND signerPHID = %s AND documentVersion = %d', ->withDocumentPHIDs(array($document->getPHID()))
$document->getPHID(), ->withSignerPHIDs(array($signer_phid))
$signer_phid, ->withDocumentVersions(array($document->getVersions()))
$document->getVersions()); ->executeOne();
} }
if (!$signature) { if (!$signature) {

View file

@ -0,0 +1,122 @@
<?php
final class LegalpadDocumentSignatureListController extends LegalpadController {
private $documentId;
public function willProcessRequest(array $data) {
$this->documentId = $data['id'];
}
public function processRequest() {
$request = $this->getRequest();
$user = $request->getUser();
$document = id(new LegalpadDocumentQuery())
->setViewer($user)
->withIDs(array($this->documentId))
->executeOne();
if (!$document) {
return new Aphront404Response();
}
$title = pht('Signatures for %s', $document->getMonogram());
$pager = id(new AphrontCursorPagerView())
->readFromRequest($request);
$signatures = id(new LegalpadDocumentSignatureQuery())
->setViewer($user)
->withDocumentPHIDs(array($document->getPHID()))
->executeWithCursorPager($pager);
$crumbs = $this->buildApplicationCrumbs($this->buildSideNav());
$crumbs->addTextCrumb(
$document->getMonogram(),
$this->getApplicationURI('view/'.$document->getID()));
$crumbs->addTextCrumb(
pht('Signatures'));
$list = $this->renderResultsList($document, $signatures);
$list->setPager($pager);
return $this->buildApplicationPage(
array(
$crumbs,
$list,
),
array(
'title' => $title,
'device' => true,
));
}
private function renderResultsList(
LegalpadDocument $document,
array $signatures) {
assert_instances_of($signatures, 'LegalpadDocumentSignature');
$user = $this->getRequest()->getUser();
$list = new PHUIObjectItemListView();
$list->setUser($user);
foreach ($signatures as $signature) {
$created = phabricator_date($signature->getDateCreated(), $user);
$data = $signature->getSignatureData();
$sig_data = phutil_tag(
'div',
array(),
array(
phutil_tag(
'div',
array(),
phutil_tag(
'a',
array(
'href' => 'mailto:'.$data['email'],
),
$data['email'])),
phutil_tag(
'div',
array(),
$data['address_1']),
phutil_tag(
'div',
array(),
$data['address_2']),
phutil_tag(
'div',
array(),
$data['phone'])
));
$item = id(new PHUIObjectItemView())
->setObject($signature)
->setHeader($data['name'])
->setSubhead($sig_data)
->addIcon('none', pht('Signed %s', $created));
$good_sig = true;
if (!$signature->isVerified()) {
$item->addFootIcon('disable', 'Unverified Email');
$good_sig = false;
}
if ($signature->getDocumentVersion() != $document->getVersions()) {
$item->addFootIcon('delete', 'Stale Signature');
$good_sig = false;
}
if ($good_sig) {
$item->setBarColor('green');
}
$list->addItem($item);
}
return $list;
}
}

View file

@ -21,8 +21,18 @@ extends LegalpadController {
$request = $this->getRequest(); $request = $this->getRequest();
$user = $request->getUser(); $user = $request->getUser();
$signature = id(new LegalpadDocumentSignature()) // this page can be accessed by not logged in users to valid their
->loadOneWhere('secretKey = %s', $this->code); // signatures. use the omnipotent user for these cases.
if (!$user->isLoggedIn()) {
$viewer = PhabricatorUser::getOmnipotentUser();
} else {
$viewer = $user;
}
$signature = id(new LegalpadDocumentSignatureQuery())
->setViewer($viewer)
->withSecretKeys(array($this->code))
->executeOne();
if (!$signature) { if (!$signature) {
$title = pht('Unable to Verify Signature'); $title = pht('Unable to Verify Signature');

View file

@ -132,11 +132,13 @@ final class LegalpadDocumentViewController extends LegalpadController {
$document, $document,
PhabricatorPolicyCapability::CAN_EDIT); PhabricatorPolicyCapability::CAN_EDIT);
$doc_id = $document->getID();
$actions->addAction( $actions->addAction(
id(new PhabricatorActionView()) id(new PhabricatorActionView())
->setIcon('edit') ->setIcon('edit')
->setName(pht('Edit Document')) ->setName(pht('Edit Document'))
->setHref($this->getApplicationURI('/edit/'.$document->getID().'/')) ->setHref($this->getApplicationURI('/edit/'.$doc_id.'/'))
->setDisabled(!$can_edit) ->setDisabled(!$can_edit)
->setWorkflow(!$can_edit)); ->setWorkflow(!$can_edit));
@ -146,6 +148,12 @@ final class LegalpadDocumentViewController extends LegalpadController {
->setName(pht('Sign Document')) ->setName(pht('Sign Document'))
->setHref('/'.$document->getMonogram())); ->setHref('/'.$document->getMonogram()));
$actions->addAction(
id(new PhabricatorActionView())
->setIcon('transcript')
->setName(pht('View Signatures'))
->setHref($this->getApplicationURI('/signatures/'.$doc_id.'/')));
return $actions; return $actions;
} }

View file

@ -1,8 +1,5 @@
<?php <?php
/**
* @group legalpad
*/
final class LegalpadDocumentQuery final class LegalpadDocumentQuery
extends PhabricatorCursorPagedPolicyAwareQuery { extends PhabricatorCursorPagedPolicyAwareQuery {
@ -89,11 +86,11 @@ final class LegalpadDocumentQuery
protected function willFilterPage(array $documents) { protected function willFilterPage(array $documents) {
if ($this->signerPHIDs) { if ($this->signerPHIDs) {
$document_map = mpull($documents, null, 'getPHID'); $document_map = mpull($documents, null, 'getPHID');
$signatures = id(new LegalpadDocumentSignature()) $signatures = id(new LegalpadDocumentSignatureQuery())
->loadAllWhere( ->setViewer($this->getViewer())
'documentPHID IN (%Ls) AND signerPHID IN (%Ls)', ->withDocumentPHIDs(array_keys($document_map))
array_keys($document_map), ->wtihSignerPHIDs($this->signerPHIDs)
$this->signerPHIDs); ->execute();
$signatures = mgroup($signatures, 'getDocumentPHID'); $signatures = mgroup($signatures, 'getDocumentPHID');
foreach ($document_map as $document_phid => $document) { foreach ($document_map as $document_phid => $document) {
$sigs = idx($signatures, $document_phid, array()); $sigs = idx($signatures, $document_phid, array());
@ -222,10 +219,10 @@ final class LegalpadDocumentQuery
private function loadSignatures(array $documents) { private function loadSignatures(array $documents) {
$document_map = mpull($documents, null, 'getPHID'); $document_map = mpull($documents, null, 'getPHID');
$signatures = id(new LegalpadDocumentSignature()) $signatures = id(new LegalpadDocumentSignatureQuery())
->loadAllWhere( ->setViewer($this->getViewer())
'documentPHID IN (%Ls)', ->withDocumentPHIDs(array_keys($document_map))
array_keys($document_map)); ->execute();
$signatures = mgroup($signatures, 'getDocumentPHID'); $signatures = mgroup($signatures, 'getDocumentPHID');
foreach ($documents as $document) { foreach ($documents as $document) {

View file

@ -0,0 +1,101 @@
<?php
final class LegalpadDocumentSignatureQuery
extends PhabricatorCursorPagedPolicyAwareQuery {
private $ids;
private $documentPHIDs;
private $signerPHIDs;
private $documentVersions;
private $secretKeys;
public function withIDs(array $ids) {
$this->ids = $ids;
return $this;
}
public function withDocumentPHIDs(array $phids) {
$this->documentPHIDs = $phids;
return $this;
}
public function withSignerPHIDs(array $phids) {
$this->signerPHIDs = $phids;
return $this;
}
public function withDocumentVersions(array $versions) {
$this->documentVersions = $versions;
return $this;
}
public function withSecretKeys(array $keys) {
$this->secretKeys = $keys;
return $this;
}
protected function loadPage() {
$table = new LegalpadDocumentSignature();
$conn_r = $table->establishConnection('r');
$data = queryfx_all(
$conn_r,
'SELECT * FROM %T %Q %Q %Q',
$table->getTableName(),
$this->buildWhereClause($conn_r),
$this->buildOrderClause($conn_r),
$this->buildLimitClause($conn_r));
$documents = $table->loadAllFromArray($data);
return $documents;
}
protected function buildWhereClause($conn_r) {
$where = array();
$where[] = $this->buildPagingClause($conn_r);
if ($this->ids) {
$where[] = qsprintf(
$conn_r,
'id IN (%Ld)',
$this->ids);
}
if ($this->documentPHIDs) {
$where[] = qsprintf(
$conn_r,
'documentPHID IN (%Ls)',
$this->documentPHIDs);
}
if ($this->signerPHIDs) {
$where[] = qsprintf(
$conn_r,
'signerPHID IN (%Ls)',
$this->signerPHIDs);
}
if ($this->documentVersions) {
$where[] = qsprintf(
$conn_r,
'documentVersion IN (%Ld)',
$this->documentVersions);
}
if ($this->secretKeys) {
$where[] = qsprintf(
$conn_r,
'secretKey IN (%Ls)',
$this->secretKeys);
}
return $this->formatWhereClause($where);
}
public function getQueryApplicationClass() {
return 'PhabricatorApplicationLegalpad';
}
}

View file

@ -1,9 +1,8 @@
<?php <?php
/** final class LegalpadDocumentSignature
* @group legalpad extends LegalpadDAO
*/ implements PhabricatorPolicyInterface {
final class LegalpadDocumentSignature extends LegalpadDAO {
const VERIFIED = 0; const VERIFIED = 0;
const UNVERIFIED = 1; const UNVERIFIED = 1;
@ -33,5 +32,27 @@ final class LegalpadDocumentSignature extends LegalpadDAO {
public function isVerified() { public function isVerified() {
return $this->getVerified() != self::UNVERIFIED; return $this->getVerified() != self::UNVERIFIED;
} }
/* -( PhabricatorPolicyInterface )----------------------------------------- */
public function getCapabilities() {
return array(
PhabricatorPolicyCapability::CAN_VIEW,
);
}
public function getPolicy($capability) {
switch ($capability) {
case PhabricatorPolicyCapability::CAN_VIEW:
return PhabricatorPolicies::POLICY_USER;
}
}
public function hasAutomaticCapability($capability, PhabricatorUser $viewer) {
return false;
}
public function describeAutomaticCapability($capability) {
return null;
}
} }