Only return the team and repo permissions when listing robots when we absolutely need them.
This commit is contained in:
parent
561f2c7db0
commit
f858caf6cd
4 changed files with 81 additions and 34 deletions
|
@ -336,16 +336,21 @@ class TupleSelector(object):
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
def list_entity_robot_permission_teams(entity_name):
|
def list_entity_robot_permission_teams(entity_name, include_permissions=False):
|
||||||
query = (_list_entity_robots(entity_name)
|
query = (_list_entity_robots(entity_name))
|
||||||
.join(RepositoryPermission, JOIN_LEFT_OUTER,
|
|
||||||
on=(RepositoryPermission.user == FederatedLogin.user))
|
fields = [User.username, FederatedLogin.service_ident]
|
||||||
.join(Repository, JOIN_LEFT_OUTER)
|
if include_permissions:
|
||||||
.switch(User)
|
query = (query.join(RepositoryPermission, JOIN_LEFT_OUTER,
|
||||||
.join(TeamMember, JOIN_LEFT_OUTER)
|
on=(RepositoryPermission.user == FederatedLogin.user))
|
||||||
.join(Team, JOIN_LEFT_OUTER))
|
.join(Repository, JOIN_LEFT_OUTER)
|
||||||
|
.switch(User)
|
||||||
|
.join(TeamMember, JOIN_LEFT_OUTER)
|
||||||
|
.join(Team, JOIN_LEFT_OUTER))
|
||||||
|
|
||||||
|
fields.append(Repository.name)
|
||||||
|
fields.append(Team.name)
|
||||||
|
|
||||||
fields = [User.username, FederatedLogin.service_ident, Repository.name, Team.name]
|
|
||||||
return TupleSelector(query, fields)
|
return TupleSelector(query, fields)
|
||||||
|
|
||||||
|
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
from endpoints.api import (resource, nickname, ApiResource, log_action, related_user_resource,
|
from endpoints.api import (resource, nickname, ApiResource, log_action, related_user_resource,
|
||||||
Unauthorized, require_user_admin, internal_only, require_scope,
|
Unauthorized, require_user_admin, internal_only, require_scope,
|
||||||
path_param)
|
path_param, parse_args, truthy_bool, query_param)
|
||||||
from auth.permissions import AdministerOrganizationPermission, OrganizationMemberPermission
|
from auth.permissions import AdministerOrganizationPermission, OrganizationMemberPermission
|
||||||
from auth.auth_context import get_authenticated_user
|
from auth.auth_context import get_authenticated_user
|
||||||
from auth import scopes
|
from auth import scopes
|
||||||
|
@ -27,8 +27,8 @@ def permission_view(permission):
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
def robots_list(prefix):
|
def robots_list(prefix, include_permissions=False):
|
||||||
tuples = model.list_entity_robot_permission_teams(prefix)
|
tuples = model.list_entity_robot_permission_teams(prefix, include_permissions=include_permissions)
|
||||||
|
|
||||||
robots = {}
|
robots = {}
|
||||||
robot_teams = set()
|
robot_teams = set()
|
||||||
|
@ -38,27 +38,32 @@ def robots_list(prefix):
|
||||||
if not robot_name in robots:
|
if not robot_name in robots:
|
||||||
robots[robot_name] = {
|
robots[robot_name] = {
|
||||||
'name': robot_name,
|
'name': robot_name,
|
||||||
'token': robot_tuple.get(FederatedLogin.service_ident),
|
'token': robot_tuple.get(FederatedLogin.service_ident)
|
||||||
'teams': [],
|
|
||||||
'repositories': []
|
|
||||||
}
|
}
|
||||||
|
|
||||||
team_name = robot_tuple.get(Team.name)
|
if include_permissions:
|
||||||
repository_name = robot_tuple.get(Repository.name)
|
robots[robot_name].update({
|
||||||
|
'teams': [],
|
||||||
if team_name is not None:
|
'repositories': []
|
||||||
check_key = robot_name + ':' + team_name
|
|
||||||
if not check_key in robot_teams:
|
|
||||||
robot_teams.add(check_key)
|
|
||||||
|
|
||||||
robots[robot_name]['teams'].append({
|
|
||||||
'name': team_name,
|
|
||||||
'avatar': avatar.get_data(team_name, team_name, 'team')
|
|
||||||
})
|
})
|
||||||
|
|
||||||
if repository_name is not None:
|
if include_permissions:
|
||||||
if not repository_name in robots[robot_name]['repositories']:
|
team_name = robot_tuple.get(Team.name)
|
||||||
robots[robot_name]['repositories'].append(repository_name)
|
repository_name = robot_tuple.get(Repository.name)
|
||||||
|
|
||||||
|
if team_name is not None:
|
||||||
|
check_key = robot_name + ':' + team_name
|
||||||
|
if not check_key in robot_teams:
|
||||||
|
robot_teams.add(check_key)
|
||||||
|
|
||||||
|
robots[robot_name]['teams'].append({
|
||||||
|
'name': team_name,
|
||||||
|
'avatar': avatar.get_data(team_name, team_name, 'team')
|
||||||
|
})
|
||||||
|
|
||||||
|
if repository_name is not None:
|
||||||
|
if not repository_name in robots[robot_name]['repositories']:
|
||||||
|
robots[robot_name]['repositories'].append(repository_name)
|
||||||
|
|
||||||
return {'robots': robots.values()}
|
return {'robots': robots.values()}
|
||||||
|
|
||||||
|
@ -68,10 +73,14 @@ class UserRobotList(ApiResource):
|
||||||
""" Resource for listing user robots. """
|
""" Resource for listing user robots. """
|
||||||
@require_user_admin
|
@require_user_admin
|
||||||
@nickname('getUserRobots')
|
@nickname('getUserRobots')
|
||||||
def get(self):
|
@parse_args
|
||||||
|
@query_param('permissions',
|
||||||
|
'Whether to include repostories and teams in which the robots have permission.',
|
||||||
|
type=truthy_bool, default=False)
|
||||||
|
def get(self, args):
|
||||||
""" List the available robots for the user. """
|
""" List the available robots for the user. """
|
||||||
user = get_authenticated_user()
|
user = get_authenticated_user()
|
||||||
return robots_list(user.username)
|
return robots_list(user.username, include_permissions=args.get('permissions', False))
|
||||||
|
|
||||||
|
|
||||||
@resource('/v1/user/robots/<robot_shortname>')
|
@resource('/v1/user/robots/<robot_shortname>')
|
||||||
|
@ -113,11 +122,15 @@ class OrgRobotList(ApiResource):
|
||||||
""" Resource for listing an organization's robots. """
|
""" Resource for listing an organization's robots. """
|
||||||
@require_scope(scopes.ORG_ADMIN)
|
@require_scope(scopes.ORG_ADMIN)
|
||||||
@nickname('getOrgRobots')
|
@nickname('getOrgRobots')
|
||||||
def get(self, orgname):
|
@parse_args
|
||||||
|
@query_param('permissions',
|
||||||
|
'Whether to include repostories and teams in which the robots have permission.',
|
||||||
|
type=truthy_bool, default=False)
|
||||||
|
def get(self, args, orgname):
|
||||||
""" List the organization's robots. """
|
""" List the organization's robots. """
|
||||||
permission = OrganizationMemberPermission(orgname)
|
permission = OrganizationMemberPermission(orgname)
|
||||||
if permission.can():
|
if permission.can():
|
||||||
return robots_list(orgname)
|
return robots_list(orgname, include_permissions=args.get('permissions', False))
|
||||||
|
|
||||||
raise Unauthorized()
|
raise Unauthorized()
|
||||||
|
|
||||||
|
|
|
@ -143,8 +143,12 @@ angular.module('quay').directive('robotsManager', function () {
|
||||||
if (!$scope.user && !$scope.organization) { return; }
|
if (!$scope.user && !$scope.organization) { return; }
|
||||||
if ($scope.loading || !$scope.isEnabled) { return; }
|
if ($scope.loading || !$scope.isEnabled) { return; }
|
||||||
|
|
||||||
|
var params = {
|
||||||
|
'permissions': true
|
||||||
|
};
|
||||||
|
|
||||||
$scope.loading = true;
|
$scope.loading = true;
|
||||||
ApiService.getRobots($scope.organization).then(function(resp) {
|
ApiService.getRobots($scope.organization, null, params).then(function(resp) {
|
||||||
$scope.robots = resp.robots;
|
$scope.robots = resp.robots;
|
||||||
$scope.loading = false;
|
$scope.loading = false;
|
||||||
|
|
||||||
|
|
|
@ -2173,6 +2173,31 @@ class TestUserRobots(ApiTestCase):
|
||||||
def getRobotNames(self):
|
def getRobotNames(self):
|
||||||
return [r['name'] for r in self.getJsonResponse(UserRobotList)['robots']]
|
return [r['name'] for r in self.getJsonResponse(UserRobotList)['robots']]
|
||||||
|
|
||||||
|
def test_robot_list(self):
|
||||||
|
self.login(NO_ACCESS_USER)
|
||||||
|
|
||||||
|
# Create some robots.
|
||||||
|
self.putJsonResponse(UserRobot,
|
||||||
|
params=dict(robot_shortname='bender'),
|
||||||
|
expected_code=201)
|
||||||
|
|
||||||
|
self.putJsonResponse(UserRobot,
|
||||||
|
params=dict(robot_shortname='goldy'),
|
||||||
|
expected_code=201)
|
||||||
|
|
||||||
|
self.putJsonResponse(UserRobot,
|
||||||
|
params=dict(robot_shortname='coolbot'),
|
||||||
|
expected_code=201)
|
||||||
|
|
||||||
|
# Queries: Base + the list query
|
||||||
|
with assert_query_count(BASE_ACCESS_QUERY_COUNT + 1):
|
||||||
|
self.getJsonResponse(UserRobotList)
|
||||||
|
|
||||||
|
# Queries: Base + the list query
|
||||||
|
with assert_query_count(BASE_ACCESS_QUERY_COUNT + 1):
|
||||||
|
self.getJsonResponse(UserRobotList, params=dict(permissions=True))
|
||||||
|
|
||||||
|
|
||||||
def test_robots(self):
|
def test_robots(self):
|
||||||
self.login(NO_ACCESS_USER)
|
self.login(NO_ACCESS_USER)
|
||||||
|
|
||||||
|
|
Reference in a new issue