diff --git a/endpoints/api/search.py b/endpoints/api/search.py index 2d96b0384..7cb1a1fda 100644 --- a/endpoints/api/search.py +++ b/endpoints/api/search.py @@ -2,9 +2,11 @@ from endpoints.api import (ApiResource, parse_args, query_param, truthy_bool, ni require_scope) from data import model from auth.permissions import (OrganizationMemberPermission, ViewTeamPermission, - ReadRepositoryPermission, UserAdminPermission) + ReadRepositoryPermission, UserAdminPermission, + AdministerOrganizationPermission) from auth.auth_context import get_authenticated_user from auth import scopes +from util.gravatar import compute_hash @resource('/v1/entities/') @@ -14,10 +16,12 @@ class EntitySearch(ApiResource): @query_param('namespace', 'Namespace to use when querying for org entities.', type=str, default='') @query_param('includeTeams', 'Whether to include team names.', type=truthy_bool, default=False) + @query_param('includeOrgs', 'Whether to include orgs names.', type=truthy_bool, default=False) @nickname('getMatchingEntities') def get(self, args, prefix): """ Get a list of entities that match the specified prefix. """ teams = [] + org_data = [] namespace_name = args['namespace'] robot_namespace = None @@ -34,6 +38,15 @@ class EntitySearch(ApiResource): if args['includeTeams']: teams = model.get_matching_teams(prefix, organization) + if args['includeOrgs'] and AdministerOrganizationPermission(namespace_name) \ + and namespace_name.startswith(prefix): + org_data = [{ + 'name': namespace_name, + 'kind': 'org', + 'is_org_member': True, + 'gravatar': compute_hash(organization.email), + }] + except model.InvalidOrganizationException: # namespace name was a user user = get_authenticated_user() @@ -69,7 +82,7 @@ class EntitySearch(ApiResource): user_data = [user_view(user) for user in users] return { - 'results': team_data + user_data + 'results': team_data + user_data + org_data } @@ -113,4 +126,4 @@ class FindRepositories(ApiResource): 'repositories': [repo_view(repo) for repo in matching if (repo.visibility.name == 'public' or ReadRepositoryPermission(repo.namespace, repo.name).can())] - } \ No newline at end of file + } diff --git a/endpoints/notificationmethod.py b/endpoints/notificationmethod.py index 5f0f8247f..9f57fc495 100644 --- a/endpoints/notificationmethod.py +++ b/endpoints/notificationmethod.py @@ -67,6 +67,17 @@ class QuayNotificationMethod(NotificationMethod): # Just to be safe. return True + target_users.append(target) + elif target_info['kind'] == 'org': + target = model.get_organization(target_info['name']) + if not target: + # Just to be safe. + return True + + # Only repositories under the organization can cause notifications to that org. + if target_info['name'] != repository.namespace: + return False + target_users.append(target) elif target_info['kind'] == 'team': # Lookup the team. diff --git a/static/css/quay.css b/static/css/quay.css index 0ffc62b68..3e9c4a746 100644 --- a/static/css/quay.css +++ b/static/css/quay.css @@ -4469,11 +4469,11 @@ i.quay-icon { width: 50px; } -.external-notification-view-element .side-controls { - opacity: 0; - transition: opacity 300ms ease-in-out; +.external-notification-view-element .side-controls button { + border: 1px solid transparent; + transition: all 300ms ease-in-out; } -.external-notification-view-element:hover .side-controls { - opacity: 1; +.external-notification-view-element:hover .side-controls button { + border: 1px solid #eee; } \ No newline at end of file diff --git a/static/directives/create-external-notification-dialog.html b/static/directives/create-external-notification-dialog.html index dc5548b85..712a203da 100644 --- a/static/directives/create-external-notification-dialog.html +++ b/static/directives/create-external-notification-dialog.html @@ -66,7 +66,7 @@ placeholder="''" current-entity="currentConfig[field.name]" ng-model="currentConfig[field.name]" - allowed-entities="['user', 'team']" + allowed-entities="['user', 'team', 'org']" ng-switch-when="entity"> @@ -89,7 +89,7 @@