Merge pull request #2864 from coreos-inc/partial-autocomplete
Partial autocomplete
This commit is contained in:
commit
c44cc072fa
6 changed files with 41 additions and 2 deletions
|
@ -273,6 +273,9 @@ class DefaultConfig(ImmutableConfig):
|
||||||
# rather than only write access or admin access.
|
# rather than only write access or admin access.
|
||||||
FEATURE_READER_BUILD_LOGS = False
|
FEATURE_READER_BUILD_LOGS = False
|
||||||
|
|
||||||
|
# Feature Flag: If set to true, autocompletion will apply to partial usernames.
|
||||||
|
FEATURE_PARTIAL_USER_AUTOCOMPLETE = True
|
||||||
|
|
||||||
# If a namespace is defined in the public namespace list, then it will appear on *all*
|
# If a namespace is defined in the public namespace list, then it will appear on *all*
|
||||||
# user's repository list pages, regardless of whether that user is a member of the namespace.
|
# user's repository list pages, regardless of whether that user is a member of the namespace.
|
||||||
# Typically, this is used by an enterprise customer in configuring a set of "well-known"
|
# Typically, this is used by an enterprise customer in configuring a set of "well-known"
|
||||||
|
|
|
@ -605,8 +605,12 @@ def get_matching_user_namespaces(namespace_prefix, username, limit=10):
|
||||||
|
|
||||||
return _basequery.filter_to_repos_for_user(base_query, username).limit(limit)
|
return _basequery.filter_to_repos_for_user(base_query, username).limit(limit)
|
||||||
|
|
||||||
def get_matching_users(username_prefix, robot_namespace=None, organization=None, limit=20):
|
def get_matching_users(username_prefix, robot_namespace=None, organization=None, limit=20,
|
||||||
|
exact_matches_only=False):
|
||||||
user_search = prefix_search(User.username, username_prefix)
|
user_search = prefix_search(User.username, username_prefix)
|
||||||
|
if exact_matches_only:
|
||||||
|
user_search = (User.username == username_prefix)
|
||||||
|
|
||||||
direct_user_query = (user_search & (User.organization == False) & (User.robot == False))
|
direct_user_query = (user_search & (User.organization == False) & (User.robot == False))
|
||||||
|
|
||||||
if robot_namespace:
|
if robot_namespace:
|
||||||
|
|
|
@ -1,5 +1,7 @@
|
||||||
""" Conduct searches against all registry context. """
|
""" Conduct searches against all registry context. """
|
||||||
|
|
||||||
|
import features
|
||||||
|
|
||||||
from endpoints.api import (ApiResource, parse_args, query_param, truthy_bool, nickname, resource,
|
from endpoints.api import (ApiResource, parse_args, query_param, truthy_bool, nickname, resource,
|
||||||
require_scope, path_param, internal_only, Unauthorized, InvalidRequest,
|
require_scope, path_param, internal_only, Unauthorized, InvalidRequest,
|
||||||
show_if)
|
show_if)
|
||||||
|
@ -107,7 +109,8 @@ class EntitySearch(ApiResource):
|
||||||
robot_namespace = namespace_name
|
robot_namespace = namespace_name
|
||||||
|
|
||||||
# Lookup users in the database for the prefix query.
|
# Lookup users in the database for the prefix query.
|
||||||
users = model.user.get_matching_users(prefix, robot_namespace, organization, limit=10)
|
users = model.user.get_matching_users(prefix, robot_namespace, organization, limit=10,
|
||||||
|
exact_matches_only=not features.PARTIAL_USER_AUTOCOMPLETE)
|
||||||
|
|
||||||
# Lookup users via the user system for the prefix query. We'll filter out any users that
|
# Lookup users via the user system for the prefix query. We'll filter out any users that
|
||||||
# already exist in the database.
|
# already exist in the database.
|
||||||
|
|
|
@ -1239,6 +1239,17 @@
|
||||||
</div>
|
</div>
|
||||||
</td>
|
</td>
|
||||||
</tr>
|
</tr>
|
||||||
|
<tr>
|
||||||
|
<td class="non-input">Prefix username autocompletion:</td>
|
||||||
|
<td colspan="2">
|
||||||
|
<div class="config-bool-field" binding="config.FEATURE_PARTIAL_USER_AUTOCOMPLETE">
|
||||||
|
Allow prefix username autocompletion
|
||||||
|
</div>
|
||||||
|
<div class="help-text">
|
||||||
|
If disabled, autocompletion for users will only match on exact usernames.
|
||||||
|
</div>
|
||||||
|
</td>
|
||||||
|
</tr>
|
||||||
<tr ng-show="config.FEATURE_MAILING">
|
<tr ng-show="config.FEATURE_MAILING">
|
||||||
<td class="non-input">Team Invitations:</td>
|
<td class="non-input">Team Invitations:</td>
|
||||||
<td colspan="2">
|
<td colspan="2">
|
||||||
|
|
|
@ -1017,6 +1017,23 @@ class TestGetMatchingEntities(ApiTestCase):
|
||||||
assert 'outsideorg' in names
|
assert 'outsideorg' in names
|
||||||
assert not 'owners' in names
|
assert not 'owners' in names
|
||||||
|
|
||||||
|
def test_prefix_disabled(self):
|
||||||
|
with patch('features.PARTIAL_USER_AUTOCOMPLETE', False):
|
||||||
|
self.login(NO_ACCESS_USER)
|
||||||
|
|
||||||
|
json = self.getJsonResponse(EntitySearch, params=dict(prefix='o', namespace=ORGANIZATION,
|
||||||
|
includeTeams='true'))
|
||||||
|
|
||||||
|
names = set([r['name'] for r in json['results']])
|
||||||
|
assert not 'outsideorg' in names
|
||||||
|
assert not 'owners' in names
|
||||||
|
|
||||||
|
json = self.getJsonResponse(EntitySearch, params=dict(prefix='outsideorg', namespace=ORGANIZATION,
|
||||||
|
includeTeams='true'))
|
||||||
|
names = set([r['name'] for r in json['results']])
|
||||||
|
assert 'outsideorg' in names
|
||||||
|
assert not 'owners' in names
|
||||||
|
|
||||||
def test_inorg(self):
|
def test_inorg(self):
|
||||||
self.login(ADMIN_ACCESS_USER)
|
self.login(ADMIN_ACCESS_USER)
|
||||||
|
|
||||||
|
|
|
@ -20,6 +20,7 @@ def add_enterprise_config_defaults(config_obj, current_secret_key, hostname):
|
||||||
config_obj['FEATURE_CHANGE_TAG_EXPIRATION'] = config_obj.get('FEATURE_CHANGE_TAG_EXPIRATION',
|
config_obj['FEATURE_CHANGE_TAG_EXPIRATION'] = config_obj.get('FEATURE_CHANGE_TAG_EXPIRATION',
|
||||||
True)
|
True)
|
||||||
config_obj['FEATURE_DIRECT_LOGIN'] = config_obj.get('FEATURE_DIRECT_LOGIN', True)
|
config_obj['FEATURE_DIRECT_LOGIN'] = config_obj.get('FEATURE_DIRECT_LOGIN', True)
|
||||||
|
config_obj['FEATURE_PARTIAL_USER_AUTOCOMPLETE'] = config_obj.get('FEATURE_PARTIAL_USER_AUTOCOMPLETE', True)
|
||||||
|
|
||||||
# Default features that are off.
|
# Default features that are off.
|
||||||
config_obj['FEATURE_MAILING'] = config_obj.get('FEATURE_MAILING', False)
|
config_obj['FEATURE_MAILING'] = config_obj.get('FEATURE_MAILING', False)
|
||||||
|
|
Reference in a new issue