2013-05-31 10:51:20 -07:00
|
|
|
<?php
|
|
|
|
|
|
|
|
final class PhabricatorPeopleSearchEngine
|
|
|
|
extends PhabricatorApplicationSearchEngine {
|
|
|
|
|
2014-06-12 13:22:20 -07:00
|
|
|
public function getResultTypeDescription() {
|
|
|
|
return pht('Users');
|
|
|
|
}
|
|
|
|
|
2015-02-04 15:47:48 -08:00
|
|
|
public function getApplicationClassName() {
|
2014-07-23 10:03:09 +10:00
|
|
|
return 'PhabricatorPeopleApplication';
|
2014-05-15 19:17:09 -07:00
|
|
|
}
|
|
|
|
|
2015-06-08 12:21:48 -07:00
|
|
|
public function newQuery() {
|
|
|
|
return id(new PhabricatorPeopleQuery())
|
|
|
|
->needPrimaryEmail(true)
|
|
|
|
->needProfileImage(true);
|
Integrate ApplicationSearch with CustomField
Summary:
Ref T2625. Ref T3794. Ref T418. Ref T1703.
This is a more general version of D5278. It expands CustomField support to include real integration with ApplicationSearch.
Broadly, custom fields may elect to:
- build indicies when objects are updated;
- populate ApplicationSearch forms with new controls;
- read inputs entered into those controls out of the request; and
- apply constraints to search queries.
Some utility/helper stuff is provided to make this easier. This part could be cleaner, but seems reasonable for a first cut. In particular, the Query and SearchEngine must manually call all the hooks right now instead of everything happening magically. I think that's fine for the moment; they're pretty easy to get right.
Test Plan:
I added a new searchable "Company" field to People:
{F58229}
This also cleaned up the disable/reorder view a little bit:
{F58230}
As it did before, this field appears on the edit screen:
{F58231}
However, because it has `search`, it also appears on the search screen:
{F58232}
When queried, it returns the expected results:
{F58233}
And the actually good bit of all this is that the query can take advantage of indexes:
mysql> explain SELECT * FROM `user` user JOIN `user_customfieldstringindex` `appsearch_0` ON `appsearch_0`.objectPHID = user.phid AND `appsearch_0`.indexKey = 'mk3Ndy476ge6' AND `appsearch_0`.indexValue IN ('phacility') ORDER BY user.id DESC LIMIT 101;
+----+-------------+-------------+--------+-------------------+----------+---------+------------------------------------------+------+----------------------------------------------+
| id | select_type | table | type | possible_keys | key | key_len | ref | rows | Extra |
+----+-------------+-------------+--------+-------------------+----------+---------+------------------------------------------+------+----------------------------------------------+
| 1 | SIMPLE | appsearch_0 | ref | key_join,key_find | key_find | 232 | const,const | 1 | Using where; Using temporary; Using filesort |
| 1 | SIMPLE | user | eq_ref | phid | phid | 194 | phabricator2_user.appsearch_0.objectPHID | 1 | |
+----+-------------+-------------+--------+-------------------+----------+---------+------------------------------------------+------+----------------------------------------------+
2 rows in set (0.00 sec)
Reviewers: btrahan
Reviewed By: btrahan
CC: aran
Maniphest Tasks: T418, T1703, T2625, T3794
Differential Revision: https://secure.phabricator.com/D6992
2013-09-16 13:44:34 -07:00
|
|
|
}
|
|
|
|
|
2015-06-08 12:20:35 -07:00
|
|
|
protected function buildCustomSearchFields() {
|
|
|
|
return array(
|
|
|
|
id(new PhabricatorSearchStringListField())
|
|
|
|
->setLabel(pht('Usernames'))
|
|
|
|
->setKey('usernames')
|
|
|
|
->setAliases(array('username')),
|
|
|
|
id(new PhabricatorSearchTextField())
|
|
|
|
->setLabel(pht('Name Contains'))
|
|
|
|
->setKey('nameLike'),
|
|
|
|
id(new PhabricatorSearchThreeStateField())
|
|
|
|
->setLabel(pht('Administrators'))
|
|
|
|
->setKey('isAdmin')
|
|
|
|
->setOptions(
|
|
|
|
pht('(Show All)'),
|
|
|
|
pht('Show Only Administrators'),
|
|
|
|
pht('Hide Administrators')),
|
|
|
|
id(new PhabricatorSearchThreeStateField())
|
|
|
|
->setLabel(pht('Disabled'))
|
|
|
|
->setKey('isDisabled')
|
|
|
|
->setOptions(
|
|
|
|
pht('(Show All)'),
|
|
|
|
pht('Show Only Disabled Users'),
|
|
|
|
pht('Hide Disabled Users')),
|
|
|
|
id(new PhabricatorSearchThreeStateField())
|
|
|
|
->setLabel(pht('Bots'))
|
|
|
|
->setKey('isSystemAgent')
|
|
|
|
->setOptions(
|
|
|
|
pht('(Show All)'),
|
|
|
|
pht('Show Only Bots'),
|
|
|
|
pht('Hide Bots')),
|
|
|
|
id(new PhabricatorSearchThreeStateField())
|
|
|
|
->setLabel(pht('Mailing Lists'))
|
|
|
|
->setKey('isMailingList')
|
|
|
|
->setOptions(
|
|
|
|
pht('(Show All)'),
|
|
|
|
pht('Show Only Mailing Lists'),
|
|
|
|
pht('Hide Mailing Lists')),
|
|
|
|
id(new PhabricatorSearchThreeStateField())
|
|
|
|
->setLabel(pht('Needs Approval'))
|
|
|
|
->setKey('needsApproval')
|
|
|
|
->setOptions(
|
|
|
|
pht('(Show All)'),
|
|
|
|
pht('Show Only Unapproved Users'),
|
|
|
|
pht('Hide Unappproved Users')),
|
|
|
|
id(new PhabricatorSearchDateField())
|
|
|
|
->setKey('createdStart')
|
|
|
|
->setLabel(pht('Joined After')),
|
|
|
|
id(new PhabricatorSearchDateField())
|
|
|
|
->setKey('createdEnd')
|
|
|
|
->setLabel(pht('Joined Before')),
|
|
|
|
);
|
|
|
|
}
|
Integrate ApplicationSearch with CustomField
Summary:
Ref T2625. Ref T3794. Ref T418. Ref T1703.
This is a more general version of D5278. It expands CustomField support to include real integration with ApplicationSearch.
Broadly, custom fields may elect to:
- build indicies when objects are updated;
- populate ApplicationSearch forms with new controls;
- read inputs entered into those controls out of the request; and
- apply constraints to search queries.
Some utility/helper stuff is provided to make this easier. This part could be cleaner, but seems reasonable for a first cut. In particular, the Query and SearchEngine must manually call all the hooks right now instead of everything happening magically. I think that's fine for the moment; they're pretty easy to get right.
Test Plan:
I added a new searchable "Company" field to People:
{F58229}
This also cleaned up the disable/reorder view a little bit:
{F58230}
As it did before, this field appears on the edit screen:
{F58231}
However, because it has `search`, it also appears on the search screen:
{F58232}
When queried, it returns the expected results:
{F58233}
And the actually good bit of all this is that the query can take advantage of indexes:
mysql> explain SELECT * FROM `user` user JOIN `user_customfieldstringindex` `appsearch_0` ON `appsearch_0`.objectPHID = user.phid AND `appsearch_0`.indexKey = 'mk3Ndy476ge6' AND `appsearch_0`.indexValue IN ('phacility') ORDER BY user.id DESC LIMIT 101;
+----+-------------+-------------+--------+-------------------+----------+---------+------------------------------------------+------+----------------------------------------------+
| id | select_type | table | type | possible_keys | key | key_len | ref | rows | Extra |
+----+-------------+-------------+--------+-------------------+----------+---------+------------------------------------------+------+----------------------------------------------+
| 1 | SIMPLE | appsearch_0 | ref | key_join,key_find | key_find | 232 | const,const | 1 | Using where; Using temporary; Using filesort |
| 1 | SIMPLE | user | eq_ref | phid | phid | 194 | phabricator2_user.appsearch_0.objectPHID | 1 | |
+----+-------------+-------------+--------+-------------------+----------+---------+------------------------------------------+------+----------------------------------------------+
2 rows in set (0.00 sec)
Reviewers: btrahan
Reviewed By: btrahan
CC: aran
Maniphest Tasks: T418, T1703, T2625, T3794
Differential Revision: https://secure.phabricator.com/D6992
2013-09-16 13:44:34 -07:00
|
|
|
|
2015-06-08 12:20:35 -07:00
|
|
|
protected function getDefaultFieldOrder() {
|
|
|
|
return array(
|
|
|
|
'...',
|
|
|
|
'createdStart',
|
|
|
|
'createdEnd',
|
|
|
|
);
|
2013-05-31 10:51:20 -07:00
|
|
|
}
|
|
|
|
|
2015-06-09 07:09:58 +10:00
|
|
|
protected function buildQueryFromParameters(array $map) {
|
2015-06-08 12:21:48 -07:00
|
|
|
$query = $this->newQuery();
|
2013-05-31 10:51:20 -07:00
|
|
|
|
2014-05-15 19:17:09 -07:00
|
|
|
$viewer = $this->requireViewer();
|
|
|
|
|
|
|
|
// If the viewer can't browse the user directory, restrict the query to
|
|
|
|
// just the user's own profile. This is a little bit silly, but serves to
|
|
|
|
// restrict users from creating a dashboard panel which essentially just
|
|
|
|
// contains a user directory anyway.
|
|
|
|
$can_browse = PhabricatorPolicyFilter::hasCapability(
|
|
|
|
$viewer,
|
|
|
|
$this->getApplication(),
|
2014-07-25 08:20:39 +10:00
|
|
|
PeopleBrowseUserDirectoryCapability::CAPABILITY);
|
2014-05-15 19:17:09 -07:00
|
|
|
if (!$can_browse) {
|
|
|
|
$query->withPHIDs(array($viewer->getPHID()));
|
|
|
|
}
|
|
|
|
|
2015-06-08 12:20:35 -07:00
|
|
|
if ($map['usernames']) {
|
|
|
|
$query->withUsernames($map['usernames']);
|
2013-05-31 10:51:20 -07:00
|
|
|
}
|
|
|
|
|
2015-06-08 12:20:35 -07:00
|
|
|
if ($map['nameLike']) {
|
|
|
|
$query->withNameLike($map['nameLike']);
|
2013-05-31 10:51:20 -07:00
|
|
|
}
|
|
|
|
|
2015-06-08 12:20:35 -07:00
|
|
|
if ($map['isAdmin'] !== null) {
|
|
|
|
$query->withIsAdmin($map['isAdmin']);
|
2013-05-31 10:51:20 -07:00
|
|
|
}
|
|
|
|
|
2015-06-08 12:20:35 -07:00
|
|
|
if ($map['isDisabled'] !== null) {
|
|
|
|
$query->withIsDisabled($map['isDisabled']);
|
2013-05-31 10:51:20 -07:00
|
|
|
}
|
|
|
|
|
2015-06-08 12:20:35 -07:00
|
|
|
if ($map['isMailingList'] !== null) {
|
|
|
|
$query->withIsMailingList($map['isMailingList']);
|
2013-05-31 10:51:20 -07:00
|
|
|
}
|
|
|
|
|
2015-06-08 12:20:35 -07:00
|
|
|
if ($map['isSystemAgent'] !== null) {
|
|
|
|
$query->withIsSystemAgent($map['isSystemAgent']);
|
2015-06-02 08:52:00 -07:00
|
|
|
}
|
|
|
|
|
2015-06-08 12:20:35 -07:00
|
|
|
if ($map['needsApproval'] !== null) {
|
|
|
|
$query->withIsApproved(!$map['needsApproval']);
|
2013-11-13 11:24:18 -08:00
|
|
|
}
|
|
|
|
|
2015-06-08 12:20:35 -07:00
|
|
|
if ($map['createdStart']) {
|
|
|
|
$query->withDateCreatedAfter($map['createdStart']);
|
2013-05-31 10:51:20 -07:00
|
|
|
}
|
|
|
|
|
2015-06-08 12:20:35 -07:00
|
|
|
if ($map['createdEnd']) {
|
|
|
|
$query->withDateCreatedBefore($map['createdEnd']);
|
2013-05-31 10:51:20 -07:00
|
|
|
}
|
Integrate ApplicationSearch with CustomField
Summary:
Ref T2625. Ref T3794. Ref T418. Ref T1703.
This is a more general version of D5278. It expands CustomField support to include real integration with ApplicationSearch.
Broadly, custom fields may elect to:
- build indicies when objects are updated;
- populate ApplicationSearch forms with new controls;
- read inputs entered into those controls out of the request; and
- apply constraints to search queries.
Some utility/helper stuff is provided to make this easier. This part could be cleaner, but seems reasonable for a first cut. In particular, the Query and SearchEngine must manually call all the hooks right now instead of everything happening magically. I think that's fine for the moment; they're pretty easy to get right.
Test Plan:
I added a new searchable "Company" field to People:
{F58229}
This also cleaned up the disable/reorder view a little bit:
{F58230}
As it did before, this field appears on the edit screen:
{F58231}
However, because it has `search`, it also appears on the search screen:
{F58232}
When queried, it returns the expected results:
{F58233}
And the actually good bit of all this is that the query can take advantage of indexes:
mysql> explain SELECT * FROM `user` user JOIN `user_customfieldstringindex` `appsearch_0` ON `appsearch_0`.objectPHID = user.phid AND `appsearch_0`.indexKey = 'mk3Ndy476ge6' AND `appsearch_0`.indexValue IN ('phacility') ORDER BY user.id DESC LIMIT 101;
+----+-------------+-------------+--------+-------------------+----------+---------+------------------------------------------+------+----------------------------------------------+
| id | select_type | table | type | possible_keys | key | key_len | ref | rows | Extra |
+----+-------------+-------------+--------+-------------------+----------+---------+------------------------------------------+------+----------------------------------------------+
| 1 | SIMPLE | appsearch_0 | ref | key_join,key_find | key_find | 232 | const,const | 1 | Using where; Using temporary; Using filesort |
| 1 | SIMPLE | user | eq_ref | phid | phid | 194 | phabricator2_user.appsearch_0.objectPHID | 1 | |
+----+-------------+-------------+--------+-------------------+----------+---------+------------------------------------------+------+----------------------------------------------+
2 rows in set (0.00 sec)
Reviewers: btrahan
Reviewed By: btrahan
CC: aran
Maniphest Tasks: T418, T1703, T2625, T3794
Differential Revision: https://secure.phabricator.com/D6992
2013-09-16 13:44:34 -07:00
|
|
|
|
2013-05-31 10:51:20 -07:00
|
|
|
return $query;
|
|
|
|
}
|
|
|
|
|
|
|
|
protected function getURI($path) {
|
|
|
|
return '/people/'.$path;
|
|
|
|
}
|
|
|
|
|
2015-01-07 07:34:51 +11:00
|
|
|
protected function getBuiltinQueryNames() {
|
2013-05-31 10:51:20 -07:00
|
|
|
$names = array(
|
2015-06-02 10:22:26 -07:00
|
|
|
'active' => pht('Active'),
|
2013-05-31 10:51:20 -07:00
|
|
|
'all' => pht('All'),
|
|
|
|
);
|
|
|
|
|
2013-11-13 11:24:18 -08:00
|
|
|
$viewer = $this->requireViewer();
|
|
|
|
if ($viewer->getIsAdmin()) {
|
|
|
|
$names['approval'] = pht('Approval Queue');
|
|
|
|
}
|
|
|
|
|
2013-05-31 10:51:20 -07:00
|
|
|
return $names;
|
|
|
|
}
|
|
|
|
|
|
|
|
public function buildSavedQueryFromBuiltin($query_key) {
|
|
|
|
$query = $this->newSavedQuery();
|
|
|
|
$query->setQueryKey($query_key);
|
|
|
|
|
|
|
|
switch ($query_key) {
|
|
|
|
case 'all':
|
|
|
|
return $query;
|
2015-06-02 10:22:26 -07:00
|
|
|
case 'active':
|
|
|
|
return $query
|
|
|
|
->setParameter('isDisabled', false);
|
2013-11-13 11:24:18 -08:00
|
|
|
case 'approval':
|
|
|
|
return $query
|
|
|
|
->setParameter('needsApproval', true)
|
2015-06-06 06:37:56 -07:00
|
|
|
->setParameter('isDisabled', false);
|
2013-05-31 10:51:20 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
return parent::buildSavedQueryFromBuiltin($query_key);
|
|
|
|
}
|
|
|
|
|
2014-05-15 19:17:09 -07:00
|
|
|
protected function renderResultList(
|
|
|
|
array $users,
|
|
|
|
PhabricatorSavedQuery $query,
|
|
|
|
array $handles) {
|
|
|
|
|
|
|
|
assert_instances_of($users, 'PhabricatorUser');
|
|
|
|
|
|
|
|
$request = $this->getRequest();
|
|
|
|
$viewer = $this->requireViewer();
|
|
|
|
|
|
|
|
$list = new PHUIObjectItemListView();
|
|
|
|
|
|
|
|
$is_approval = ($query->getQueryKey() == 'approval');
|
|
|
|
|
|
|
|
foreach ($users as $user) {
|
|
|
|
$primary_email = $user->loadPrimaryEmail();
|
|
|
|
if ($primary_email && $primary_email->getIsVerified()) {
|
|
|
|
$email = pht('Verified');
|
|
|
|
} else {
|
|
|
|
$email = pht('Unverified');
|
|
|
|
}
|
|
|
|
|
|
|
|
$item = new PHUIObjectItemView();
|
|
|
|
$item->setHeader($user->getFullName())
|
|
|
|
->setHref('/p/'.$user->getUsername().'/')
|
|
|
|
->addAttribute(phabricator_datetime($user->getDateCreated(), $viewer))
|
|
|
|
->addAttribute($email)
|
|
|
|
->setImageURI($user->getProfileImageURI());
|
|
|
|
|
|
|
|
if ($is_approval && $primary_email) {
|
|
|
|
$item->addAttribute($primary_email->getAddress());
|
|
|
|
}
|
|
|
|
|
|
|
|
if ($user->getIsDisabled()) {
|
|
|
|
$item->addIcon('fa-ban', pht('Disabled'));
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!$is_approval) {
|
|
|
|
if (!$user->getIsApproved()) {
|
|
|
|
$item->addIcon('fa-clock-o', pht('Needs Approval'));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if ($user->getIsAdmin()) {
|
|
|
|
$item->addIcon('fa-star', pht('Admin'));
|
|
|
|
}
|
|
|
|
|
|
|
|
if ($user->getIsSystemAgent()) {
|
2015-06-02 10:22:26 -07:00
|
|
|
$item->addIcon('fa-desktop', pht('Bot'));
|
2014-05-15 19:17:09 -07:00
|
|
|
}
|
|
|
|
|
2015-06-02 08:52:00 -07:00
|
|
|
if ($user->getIsMailingList()) {
|
|
|
|
$item->addIcon('fa-envelope-o', pht('Mailing List'));
|
|
|
|
}
|
|
|
|
|
2014-05-15 19:17:09 -07:00
|
|
|
if ($viewer->getIsAdmin()) {
|
|
|
|
$user_id = $user->getID();
|
|
|
|
if ($is_approval) {
|
|
|
|
$item->addAction(
|
|
|
|
id(new PHUIListItemView())
|
|
|
|
->setIcon('fa-ban')
|
|
|
|
->setName(pht('Disable'))
|
|
|
|
->setWorkflow(true)
|
|
|
|
->setHref($this->getApplicationURI('disapprove/'.$user_id.'/')));
|
|
|
|
$item->addAction(
|
|
|
|
id(new PHUIListItemView())
|
|
|
|
->setIcon('fa-thumbs-o-up')
|
|
|
|
->setName(pht('Approve'))
|
|
|
|
->setWorkflow(true)
|
|
|
|
->setHref($this->getApplicationURI('approve/'.$user_id.'/')));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
$list->addItem($item);
|
|
|
|
}
|
|
|
|
|
2015-06-19 11:46:20 +01:00
|
|
|
$result = new PhabricatorApplicationSearchResultView();
|
|
|
|
$result->setObjectList($list);
|
|
|
|
$result->setNoDataString(pht('No accounts found.'));
|
|
|
|
|
|
|
|
return $result;
|
2014-05-15 19:17:09 -07:00
|
|
|
}
|
|
|
|
|
2013-05-31 10:51:20 -07:00
|
|
|
}
|