| margaretha | 0b63de4 | 2017-12-20 18:48:09 +0100 | [diff] [blame] | 1 | package de.ids_mannheim.korap.web.controller; |
| 2 | |
| 3 | import java.util.List; |
| 4 | |
| margaretha | 9d3eb04 | 2017-12-22 11:02:30 +0100 | [diff] [blame] | 5 | import javax.ws.rs.Consumes; |
| margaretha | 4566792 | 2018-01-25 21:23:03 +0100 | [diff] [blame] | 6 | import javax.ws.rs.DELETE; |
| margaretha | 9d3eb04 | 2017-12-22 11:02:30 +0100 | [diff] [blame] | 7 | import javax.ws.rs.FormParam; |
| margaretha | 0b63de4 | 2017-12-20 18:48:09 +0100 | [diff] [blame] | 8 | import javax.ws.rs.GET; |
| margaretha | 9d3eb04 | 2017-12-22 11:02:30 +0100 | [diff] [blame] | 9 | import javax.ws.rs.POST; |
| margaretha | ca7cff8 | 2019-11-12 12:06:37 +0100 | [diff] [blame] | 10 | import javax.ws.rs.PUT; |
| margaretha | 0b63de4 | 2017-12-20 18:48:09 +0100 | [diff] [blame] | 11 | import javax.ws.rs.Path; |
| margaretha | 4457383 | 2018-03-21 16:59:59 +0100 | [diff] [blame] | 12 | import javax.ws.rs.PathParam; |
| margaretha | 0b63de4 | 2017-12-20 18:48:09 +0100 | [diff] [blame] | 13 | import javax.ws.rs.Produces; |
| margaretha | 4566792 | 2018-01-25 21:23:03 +0100 | [diff] [blame] | 14 | import javax.ws.rs.QueryParam; |
| margaretha | 0b63de4 | 2017-12-20 18:48:09 +0100 | [diff] [blame] | 15 | import javax.ws.rs.core.Context; |
| 16 | import javax.ws.rs.core.MediaType; |
| 17 | import javax.ws.rs.core.Response; |
| 18 | import javax.ws.rs.core.SecurityContext; |
| 19 | |
| margaretha | ca7cff8 | 2019-11-12 12:06:37 +0100 | [diff] [blame] | 20 | import org.apache.http.HttpStatus; |
| margaretha | 0b63de4 | 2017-12-20 18:48:09 +0100 | [diff] [blame] | 21 | import org.springframework.beans.factory.annotation.Autowired; |
| 22 | import org.springframework.stereotype.Controller; |
| 23 | |
| 24 | import com.sun.jersey.spi.container.ResourceFilters; |
| 25 | |
| margaretha | 2df0660 | 2018-11-14 19:10:30 +0100 | [diff] [blame] | 26 | import de.ids_mannheim.korap.constant.OAuth2Scope; |
| margaretha | 293ee03 | 2018-03-20 20:11:52 +0100 | [diff] [blame] | 27 | import de.ids_mannheim.korap.constant.UserGroupStatus; |
| margaretha | 0b63de4 | 2017-12-20 18:48:09 +0100 | [diff] [blame] | 28 | import de.ids_mannheim.korap.dto.UserGroupDto; |
| 29 | import de.ids_mannheim.korap.exceptions.KustvaktException; |
| margaretha | 835178d | 2018-08-15 19:04:03 +0200 | [diff] [blame] | 30 | import de.ids_mannheim.korap.oauth2.service.OAuth2ScopeService; |
| margaretha | 0e8f4e7 | 2018-04-05 14:11:52 +0200 | [diff] [blame] | 31 | import de.ids_mannheim.korap.security.context.TokenContext; |
| margaretha | 0b63de4 | 2017-12-20 18:48:09 +0100 | [diff] [blame] | 32 | import de.ids_mannheim.korap.service.UserGroupService; |
| margaretha | f7abb36 | 2018-09-18 20:09:37 +0200 | [diff] [blame] | 33 | import de.ids_mannheim.korap.web.KustvaktResponseHandler; |
| margaretha | 398f472 | 2019-01-09 19:07:20 +0100 | [diff] [blame] | 34 | import de.ids_mannheim.korap.web.filter.APIVersionFilter; |
| margaretha | 0b63de4 | 2017-12-20 18:48:09 +0100 | [diff] [blame] | 35 | import de.ids_mannheim.korap.web.filter.AuthenticationFilter; |
| margaretha | 23aae22 | 2017-12-22 15:08:23 +0100 | [diff] [blame] | 36 | import de.ids_mannheim.korap.web.filter.BlockingFilter; |
| margaretha | 0b63de4 | 2017-12-20 18:48:09 +0100 | [diff] [blame] | 37 | import de.ids_mannheim.korap.web.filter.PiwikFilter; |
| 38 | |
| margaretha | 4566792 | 2018-01-25 21:23:03 +0100 | [diff] [blame] | 39 | /** |
| 40 | * UserGroupController defines web APIs related to user groups, |
| 41 | * such as creating a user group, listing groups of a user, |
| 42 | * adding members to a group and subscribing (confirming an |
| 43 | * invitation) to a group. |
| 44 | * |
| margaretha | 20f3123 | 2018-07-09 17:49:39 +0200 | [diff] [blame] | 45 | * These APIs are only available to logged-in users and not available |
| 46 | * via third-party apps. |
| margaretha | 4566792 | 2018-01-25 21:23:03 +0100 | [diff] [blame] | 47 | * |
| margaretha | a0d4d3c | 2018-01-02 12:06:11 +0100 | [diff] [blame] | 48 | * @author margaretha |
| 49 | * |
| 50 | */ |
| margaretha | 0b63de4 | 2017-12-20 18:48:09 +0100 | [diff] [blame] | 51 | @Controller |
| margaretha | ee0cbfe | 2018-08-28 17:47:14 +0200 | [diff] [blame] | 52 | @Path("{version}/group") |
| 53 | @ResourceFilters({ APIVersionFilter.class, AuthenticationFilter.class, |
| 54 | BlockingFilter.class, PiwikFilter.class }) |
| margaretha | 0b63de4 | 2017-12-20 18:48:09 +0100 | [diff] [blame] | 55 | public class UserGroupController { |
| 56 | |
| margaretha | 0b63de4 | 2017-12-20 18:48:09 +0100 | [diff] [blame] | 57 | @Autowired |
| margaretha | da3c785 | 2018-06-14 20:35:11 +0200 | [diff] [blame] | 58 | private KustvaktResponseHandler kustvaktResponseHandler; |
| margaretha | 0b63de4 | 2017-12-20 18:48:09 +0100 | [diff] [blame] | 59 | @Autowired |
| 60 | private UserGroupService service; |
| margaretha | 835178d | 2018-08-15 19:04:03 +0200 | [diff] [blame] | 61 | @Autowired |
| 62 | private OAuth2ScopeService scopeService; |
| margaretha | 9d3eb04 | 2017-12-22 11:02:30 +0100 | [diff] [blame] | 63 | |
| margaretha | 20f3123 | 2018-07-09 17:49:39 +0200 | [diff] [blame] | 64 | /** |
| 65 | * Returns all user-groups in which a user is an active or a |
| 66 | * pending member. |
| margaretha | a18ab2b | 2019-11-11 12:55:26 +0100 | [diff] [blame] | 67 | * |
| margaretha | ca7cff8 | 2019-11-12 12:06:37 +0100 | [diff] [blame] | 68 | * Not suitable for system-admin, instead use |
| 69 | * {@link UserGroupController# |
| margaretha | 20f3123 | 2018-07-09 17:49:39 +0200 | [diff] [blame] | 70 | * getUserGroupBySystemAdmin(SecurityContext, String, UserGroupStatus)} |
| margaretha | e6c711b | 2018-02-06 21:55:04 +0100 | [diff] [blame] | 71 | * |
| 72 | * @param securityContext |
| 73 | * @return a list of user-groups |
| margaretha | 293ee03 | 2018-03-20 20:11:52 +0100 | [diff] [blame] | 74 | * |
| margaretha | e6c711b | 2018-02-06 21:55:04 +0100 | [diff] [blame] | 75 | */ |
| margaretha | 0b63de4 | 2017-12-20 18:48:09 +0100 | [diff] [blame] | 76 | @GET |
| margaretha | 0e8f4e7 | 2018-04-05 14:11:52 +0200 | [diff] [blame] | 77 | @Produces(MediaType.APPLICATION_JSON + ";charset=utf-8") |
| margaretha | ca7cff8 | 2019-11-12 12:06:37 +0100 | [diff] [blame] | 78 | public List<UserGroupDto> listUserGroups ( |
| margaretha | 0e8f4e7 | 2018-04-05 14:11:52 +0200 | [diff] [blame] | 79 | @Context SecurityContext securityContext) { |
| margaretha | 0b63de4 | 2017-12-20 18:48:09 +0100 | [diff] [blame] | 80 | TokenContext context = |
| 81 | (TokenContext) securityContext.getUserPrincipal(); |
| 82 | try { |
| margaretha | 835178d | 2018-08-15 19:04:03 +0200 | [diff] [blame] | 83 | scopeService.verifyScope(context, OAuth2Scope.USER_GROUP_INFO); |
| margaretha | 3ccaeb7 | 2019-02-28 18:40:22 +0100 | [diff] [blame] | 84 | return service.retrieveUserGroupDto(context.getUsername()); |
| margaretha | 0b63de4 | 2017-12-20 18:48:09 +0100 | [diff] [blame] | 85 | } |
| 86 | catch (KustvaktException e) { |
| margaretha | da3c785 | 2018-06-14 20:35:11 +0200 | [diff] [blame] | 87 | throw kustvaktResponseHandler.throwit(e); |
| margaretha | 0b63de4 | 2017-12-20 18:48:09 +0100 | [diff] [blame] | 88 | } |
| margaretha | 0b63de4 | 2017-12-20 18:48:09 +0100 | [diff] [blame] | 89 | } |
| margaretha | 9d3eb04 | 2017-12-22 11:02:30 +0100 | [diff] [blame] | 90 | |
| margaretha | 20f3123 | 2018-07-09 17:49:39 +0200 | [diff] [blame] | 91 | /** |
| margaretha | ca7cff8 | 2019-11-12 12:06:37 +0100 | [diff] [blame] | 92 | * Lists user-groups for system-admin purposes. If username is |
| 93 | * specified, lists user-groups of the given user, otherwise list |
| 94 | * user-groups of all users. If status specified, list only |
| 95 | * user-groups with the given status, otherwise list user-groups |
| 96 | * regardless of their status. |
| margaretha | 293ee03 | 2018-03-20 20:11:52 +0100 | [diff] [blame] | 97 | * |
| 98 | * @param securityContext |
| margaretha | 20f3123 | 2018-07-09 17:49:39 +0200 | [diff] [blame] | 99 | * @param username |
| margaretha | ca7cff8 | 2019-11-12 12:06:37 +0100 | [diff] [blame] | 100 | * a username |
| margaretha | 20f3123 | 2018-07-09 17:49:39 +0200 | [diff] [blame] | 101 | * @param status |
| 102 | * {@link UserGroupStatus} |
| margaretha | 293ee03 | 2018-03-20 20:11:52 +0100 | [diff] [blame] | 103 | * @return a list of user-groups |
| 104 | */ |
| 105 | @GET |
| 106 | @Path("list/system-admin") |
| margaretha | 0e8f4e7 | 2018-04-05 14:11:52 +0200 | [diff] [blame] | 107 | @Produces(MediaType.APPLICATION_JSON + ";charset=utf-8") |
| 108 | public List<UserGroupDto> getUserGroupBySystemAdmin ( |
| margaretha | 293ee03 | 2018-03-20 20:11:52 +0100 | [diff] [blame] | 109 | @Context SecurityContext securityContext, |
| 110 | @QueryParam("username") String username, |
| 111 | @QueryParam("status") UserGroupStatus status) { |
| 112 | TokenContext context = |
| 113 | (TokenContext) securityContext.getUserPrincipal(); |
| 114 | try { |
| margaretha | 835178d | 2018-08-15 19:04:03 +0200 | [diff] [blame] | 115 | scopeService.verifyScope(context, OAuth2Scope.ADMIN); |
| margaretha | 0e8f4e7 | 2018-04-05 14:11:52 +0200 | [diff] [blame] | 116 | return service.retrieveUserGroupByStatus(username, |
| 117 | context.getUsername(), status); |
| margaretha | 293ee03 | 2018-03-20 20:11:52 +0100 | [diff] [blame] | 118 | } |
| 119 | catch (KustvaktException e) { |
| margaretha | da3c785 | 2018-06-14 20:35:11 +0200 | [diff] [blame] | 120 | throw kustvaktResponseHandler.throwit(e); |
| margaretha | 293ee03 | 2018-03-20 20:11:52 +0100 | [diff] [blame] | 121 | } |
| 122 | } |
| margaretha | 9d3eb04 | 2017-12-22 11:02:30 +0100 | [diff] [blame] | 123 | |
| margaretha | 20f3123 | 2018-07-09 17:49:39 +0200 | [diff] [blame] | 124 | /** |
| margaretha | a18ab2b | 2019-11-11 12:55:26 +0100 | [diff] [blame] | 125 | * Retrieves a specific user-group. Only system admins are |
| 126 | * allowed. |
| margaretha | 4457383 | 2018-03-21 16:59:59 +0100 | [diff] [blame] | 127 | * |
| 128 | * @param securityContext |
| margaretha | a18ab2b | 2019-11-11 12:55:26 +0100 | [diff] [blame] | 129 | * @param groupName |
| 130 | * group name |
| margaretha | 4457383 | 2018-03-21 16:59:59 +0100 | [diff] [blame] | 131 | * @return a user-group |
| 132 | */ |
| 133 | @GET |
| margaretha | 03b195a | 2019-11-12 14:57:15 +0100 | [diff] [blame] | 134 | @Path("@{groupName}") |
| margaretha | 0e8f4e7 | 2018-04-05 14:11:52 +0200 | [diff] [blame] | 135 | @Produces(MediaType.APPLICATION_JSON + ";charset=utf-8") |
| margaretha | a18ab2b | 2019-11-11 12:55:26 +0100 | [diff] [blame] | 136 | public UserGroupDto retrieveUserGroup ( |
| margaretha | 0e8f4e7 | 2018-04-05 14:11:52 +0200 | [diff] [blame] | 137 | @Context SecurityContext securityContext, |
| margaretha | a18ab2b | 2019-11-11 12:55:26 +0100 | [diff] [blame] | 138 | @PathParam("groupName") String groupName) { |
| margaretha | 4457383 | 2018-03-21 16:59:59 +0100 | [diff] [blame] | 139 | TokenContext context = |
| 140 | (TokenContext) securityContext.getUserPrincipal(); |
| 141 | try { |
| margaretha | 835178d | 2018-08-15 19:04:03 +0200 | [diff] [blame] | 142 | scopeService.verifyScope(context, OAuth2Scope.ADMIN); |
| margaretha | a18ab2b | 2019-11-11 12:55:26 +0100 | [diff] [blame] | 143 | return service.searchByName(context.getUsername(), groupName); |
| margaretha | 4457383 | 2018-03-21 16:59:59 +0100 | [diff] [blame] | 144 | } |
| 145 | catch (KustvaktException e) { |
| margaretha | da3c785 | 2018-06-14 20:35:11 +0200 | [diff] [blame] | 146 | throw kustvaktResponseHandler.throwit(e); |
| margaretha | 4457383 | 2018-03-21 16:59:59 +0100 | [diff] [blame] | 147 | } |
| margaretha | 0e8f4e7 | 2018-04-05 14:11:52 +0200 | [diff] [blame] | 148 | |
| margaretha | 4457383 | 2018-03-21 16:59:59 +0100 | [diff] [blame] | 149 | } |
| margaretha | 18533fd | 2018-03-28 16:01:06 +0200 | [diff] [blame] | 150 | |
| margaretha | 20f3123 | 2018-07-09 17:49:39 +0200 | [diff] [blame] | 151 | /** |
| margaretha | d8aa135 | 2019-12-19 11:04:41 +0100 | [diff] [blame] | 152 | * Creates a user group with the group owner as the only group |
| 153 | * member. The group owner is the authenticated user in the token |
| 154 | * context. |
| margaretha | 9d3eb04 | 2017-12-22 11:02:30 +0100 | [diff] [blame] | 155 | * |
| 156 | * @param securityContext |
| margaretha | d8aa135 | 2019-12-19 11:04:41 +0100 | [diff] [blame] | 157 | * @param groupName the name of the group |
| margaretha | ca7cff8 | 2019-11-12 12:06:37 +0100 | [diff] [blame] | 158 | * @return if a new group created, HTTP response status 201 |
| 159 | * Created, otherwise 204 No Content. |
| margaretha | 9d3eb04 | 2017-12-22 11:02:30 +0100 | [diff] [blame] | 160 | */ |
| margaretha | ca7cff8 | 2019-11-12 12:06:37 +0100 | [diff] [blame] | 161 | @PUT |
| margaretha | 03b195a | 2019-11-12 14:57:15 +0100 | [diff] [blame] | 162 | @Path("@{groupName}") |
| margaretha | ca7cff8 | 2019-11-12 12:06:37 +0100 | [diff] [blame] | 163 | @Consumes(MediaType.APPLICATION_FORM_URLENCODED) |
| margaretha | d8aa135 | 2019-12-19 11:04:41 +0100 | [diff] [blame] | 164 | public Response createUpdateUserGroup (@Context SecurityContext securityContext, |
| margaretha | ca7cff8 | 2019-11-12 12:06:37 +0100 | [diff] [blame] | 165 | @PathParam("groupName") String groupName, |
| margaretha | d8aa135 | 2019-12-19 11:04:41 +0100 | [diff] [blame] | 166 | @FormParam("description") String description) { |
| margaretha | 9d3eb04 | 2017-12-22 11:02:30 +0100 | [diff] [blame] | 167 | TokenContext context = |
| 168 | (TokenContext) securityContext.getUserPrincipal(); |
| 169 | try { |
| margaretha | 835178d | 2018-08-15 19:04:03 +0200 | [diff] [blame] | 170 | scopeService.verifyScope(context, OAuth2Scope.CREATE_USER_GROUP); |
| margaretha | d8aa135 | 2019-12-19 11:04:41 +0100 | [diff] [blame] | 171 | boolean groupExists = service.createUpdateUserGroup(groupName, |
| 172 | description, context.getUsername()); |
| margaretha | ca7cff8 | 2019-11-12 12:06:37 +0100 | [diff] [blame] | 173 | if (groupExists) { |
| 174 | return Response.noContent().build(); |
| 175 | } |
| 176 | else { |
| 177 | return Response.status(HttpStatus.SC_CREATED).build(); |
| 178 | } |
| margaretha | 9d3eb04 | 2017-12-22 11:02:30 +0100 | [diff] [blame] | 179 | } |
| 180 | catch (KustvaktException e) { |
| margaretha | da3c785 | 2018-06-14 20:35:11 +0200 | [diff] [blame] | 181 | throw kustvaktResponseHandler.throwit(e); |
| margaretha | 9d3eb04 | 2017-12-22 11:02:30 +0100 | [diff] [blame] | 182 | } |
| margaretha | 9d3eb04 | 2017-12-22 11:02:30 +0100 | [diff] [blame] | 183 | } |
| 184 | |
| margaretha | 20f3123 | 2018-07-09 17:49:39 +0200 | [diff] [blame] | 185 | /** |
| margaretha | ca7cff8 | 2019-11-12 12:06:37 +0100 | [diff] [blame] | 186 | * Deletes a user-group specified by the group name. Only group |
| margaretha | 835178d | 2018-08-15 19:04:03 +0200 | [diff] [blame] | 187 | * owner and system admins can delete groups. |
| margaretha | 2c019fa | 2018-02-01 19:50:51 +0100 | [diff] [blame] | 188 | * |
| 189 | * @param securityContext |
| margaretha | ca7cff8 | 2019-11-12 12:06:37 +0100 | [diff] [blame] | 190 | * @param groupName |
| 191 | * the name of the group to delete |
| margaretha | 2c019fa | 2018-02-01 19:50:51 +0100 | [diff] [blame] | 192 | * @return HTTP 200, if successful. |
| 193 | */ |
| margaretha | 4566792 | 2018-01-25 21:23:03 +0100 | [diff] [blame] | 194 | @DELETE |
| margaretha | 03b195a | 2019-11-12 14:57:15 +0100 | [diff] [blame] | 195 | @Path("@{groupName}") |
| margaretha | a18ab2b | 2019-11-11 12:55:26 +0100 | [diff] [blame] | 196 | public Response deleteUserGroup (@Context SecurityContext securityContext, |
| margaretha | 39cec60 | 2019-02-05 19:48:49 +0100 | [diff] [blame] | 197 | @PathParam("groupName") String groupName) { |
| 198 | TokenContext context = |
| 199 | (TokenContext) securityContext.getUserPrincipal(); |
| 200 | try { |
| 201 | scopeService.verifyScope(context, OAuth2Scope.DELETE_USER_GROUP); |
| 202 | service.deleteGroup(groupName, context.getUsername()); |
| 203 | return Response.ok().build(); |
| 204 | } |
| 205 | catch (KustvaktException e) { |
| 206 | throw kustvaktResponseHandler.throwit(e); |
| 207 | } |
| 208 | } |
| margaretha | e6c711b | 2018-02-06 21:55:04 +0100 | [diff] [blame] | 209 | |
| margaretha | 20f3123 | 2018-07-09 17:49:39 +0200 | [diff] [blame] | 210 | /** |
| margaretha | ca7cff8 | 2019-11-12 12:06:37 +0100 | [diff] [blame] | 211 | * Removes a user-group member. Group owner cannot be removed. |
| margaretha | a18ab2b | 2019-11-11 12:55:26 +0100 | [diff] [blame] | 212 | * Only group admins, system admins and the member himself can |
| margaretha | ca7cff8 | 2019-11-12 12:06:37 +0100 | [diff] [blame] | 213 | * remove a member. |
| margaretha | 4566792 | 2018-01-25 21:23:03 +0100 | [diff] [blame] | 214 | * |
| 215 | * @param securityContext |
| margaretha | a18ab2b | 2019-11-11 12:55:26 +0100 | [diff] [blame] | 216 | * @param memberUsername |
| margaretha | 20f3123 | 2018-07-09 17:49:39 +0200 | [diff] [blame] | 217 | * a username of a group member |
| margaretha | a18ab2b | 2019-11-11 12:55:26 +0100 | [diff] [blame] | 218 | * @param groupName |
| 219 | * a group name |
| margaretha | 4566792 | 2018-01-25 21:23:03 +0100 | [diff] [blame] | 220 | * @return if successful, HTTP response status OK |
| 221 | */ |
| 222 | @DELETE |
| margaretha | 03b195a | 2019-11-12 14:57:15 +0100 | [diff] [blame] | 223 | @Path("@{groupName}/~{memberUsername}") |
| margaretha | a18ab2b | 2019-11-11 12:55:26 +0100 | [diff] [blame] | 224 | public Response removeUserFromGroup ( |
| margaretha | 4566792 | 2018-01-25 21:23:03 +0100 | [diff] [blame] | 225 | @Context SecurityContext securityContext, |
| margaretha | a18ab2b | 2019-11-11 12:55:26 +0100 | [diff] [blame] | 226 | @PathParam("memberUsername") String memberUsername, |
| 227 | @PathParam("groupName") String groupName) { |
| margaretha | 4566792 | 2018-01-25 21:23:03 +0100 | [diff] [blame] | 228 | TokenContext context = |
| 229 | (TokenContext) securityContext.getUserPrincipal(); |
| 230 | try { |
| margaretha | ee0cbfe | 2018-08-28 17:47:14 +0200 | [diff] [blame] | 231 | scopeService.verifyScope(context, |
| 232 | OAuth2Scope.DELETE_USER_GROUP_MEMBER); |
| margaretha | a18ab2b | 2019-11-11 12:55:26 +0100 | [diff] [blame] | 233 | service.deleteGroupMember(memberUsername, groupName, |
| 234 | context.getUsername()); |
| margaretha | 4566792 | 2018-01-25 21:23:03 +0100 | [diff] [blame] | 235 | return Response.ok().build(); |
| 236 | } |
| 237 | catch (KustvaktException e) { |
| margaretha | da3c785 | 2018-06-14 20:35:11 +0200 | [diff] [blame] | 238 | throw kustvaktResponseHandler.throwit(e); |
| margaretha | 4566792 | 2018-01-25 21:23:03 +0100 | [diff] [blame] | 239 | } |
| 240 | } |
| 241 | |
| margaretha | 20f3123 | 2018-07-09 17:49:39 +0200 | [diff] [blame] | 242 | /** |
| margaretha | ca7cff8 | 2019-11-12 12:06:37 +0100 | [diff] [blame] | 243 | * Invites users to join a user-group specified by the |
| margaretha | d8aa135 | 2019-12-19 11:04:41 +0100 | [diff] [blame] | 244 | * groupName. Only user-group admins and system admins are |
| 245 | * allowed to use this service. |
| 246 | * |
| 247 | * The invited users are added as group members with status |
| 248 | * GroupMemberStatus.PENDING. |
| 249 | * |
| 250 | * If a user accepts the invitation by using the service: |
| 251 | * {@link UserGroupController#subscribeToGroup(SecurityContext, String)}, |
| 252 | * his GroupMemberStatus will be updated to |
| 253 | * GroupMemberStatus.ACTIVE. |
| 254 | * |
| 255 | * If a user rejects the invitation by using the service: |
| 256 | * {@link UserGroupController#unsubscribeFromGroup(SecurityContext, String)}, |
| 257 | * his GroupMemberStatus will be updated to |
| 258 | * GroupMemberStatus.DELETED. |
| margaretha | 31a9f52 | 2018-04-03 20:40:45 +0200 | [diff] [blame] | 259 | * |
| margaretha | 18533fd | 2018-03-28 16:01:06 +0200 | [diff] [blame] | 260 | * @param securityContext |
| margaretha | ca7cff8 | 2019-11-12 12:06:37 +0100 | [diff] [blame] | 261 | * @param members |
| 262 | * usernames separated by comma |
| margaretha | 18533fd | 2018-03-28 16:01:06 +0200 | [diff] [blame] | 263 | * @return if successful, HTTP response status OK |
| 264 | */ |
| margaretha | b874ef5 | 2018-01-23 20:26:31 +0100 | [diff] [blame] | 265 | @POST |
| margaretha | 03b195a | 2019-11-12 14:57:15 +0100 | [diff] [blame] | 266 | @Path("@{groupName}/invite") |
| margaretha | ca7cff8 | 2019-11-12 12:06:37 +0100 | [diff] [blame] | 267 | @Consumes(MediaType.APPLICATION_FORM_URLENCODED) |
| margaretha | 4edc70e | 2018-03-14 22:34:29 +0100 | [diff] [blame] | 268 | public Response inviteGroupMembers ( |
| margaretha | a18ab2b | 2019-11-11 12:55:26 +0100 | [diff] [blame] | 269 | @Context SecurityContext securityContext, |
| 270 | @PathParam("groupName") String groupName, |
| margaretha | ca7cff8 | 2019-11-12 12:06:37 +0100 | [diff] [blame] | 271 | @FormParam("members") String members) { |
| margaretha | b874ef5 | 2018-01-23 20:26:31 +0100 | [diff] [blame] | 272 | TokenContext context = |
| 273 | (TokenContext) securityContext.getUserPrincipal(); |
| 274 | try { |
| margaretha | ee0cbfe | 2018-08-28 17:47:14 +0200 | [diff] [blame] | 275 | scopeService.verifyScope(context, |
| 276 | OAuth2Scope.ADD_USER_GROUP_MEMBER); |
| margaretha | ca7cff8 | 2019-11-12 12:06:37 +0100 | [diff] [blame] | 277 | service.inviteGroupMembers(groupName, members, |
| 278 | context.getUsername()); |
| margaretha | c9f40e2 | 2019-08-07 17:32:19 +0200 | [diff] [blame] | 279 | return Response.ok("SUCCESS").build(); |
| margaretha | b874ef5 | 2018-01-23 20:26:31 +0100 | [diff] [blame] | 280 | } |
| 281 | catch (KustvaktException e) { |
| margaretha | da3c785 | 2018-06-14 20:35:11 +0200 | [diff] [blame] | 282 | throw kustvaktResponseHandler.throwit(e); |
| margaretha | b874ef5 | 2018-01-23 20:26:31 +0100 | [diff] [blame] | 283 | } |
| 284 | } |
| margaretha | 9d3eb04 | 2017-12-22 11:02:30 +0100 | [diff] [blame] | 285 | |
| margaretha | da5a6ab | 2019-11-08 10:06:05 +0100 | [diff] [blame] | 286 | /** |
| 287 | * Very similar to addMemberRoles web-service, but allows deletion |
| margaretha | 0e1fc55 | 2019-08-08 15:31:01 +0200 | [diff] [blame] | 288 | * as well. |
| 289 | * |
| 290 | * @param securityContext |
| margaretha | d8aa135 | 2019-12-19 11:04:41 +0100 | [diff] [blame] | 291 | * @param groupName |
| 292 | * the group name |
| 293 | * @param memberUsername |
| 294 | * the username of a group-member |
| margaretha | d575bd9 | 2021-06-14 09:42:18 +0200 | [diff] [blame] | 295 | * @param roleId |
| 296 | * a role id or multiple role ids |
| margaretha | 0e1fc55 | 2019-08-08 15:31:01 +0200 | [diff] [blame] | 297 | * @return |
| 298 | */ |
| margaretha | f7abb36 | 2018-09-18 20:09:37 +0200 | [diff] [blame] | 299 | @POST |
| margaretha | 03b195a | 2019-11-12 14:57:15 +0100 | [diff] [blame] | 300 | @Path("@{groupName}/role/edit") |
| margaretha | f7abb36 | 2018-09-18 20:09:37 +0200 | [diff] [blame] | 301 | @Consumes(MediaType.APPLICATION_FORM_URLENCODED) |
| 302 | public Response editMemberRoles (@Context SecurityContext securityContext, |
| margaretha | a18ab2b | 2019-11-11 12:55:26 +0100 | [diff] [blame] | 303 | @PathParam("groupName") String groupName, |
| margaretha | f7abb36 | 2018-09-18 20:09:37 +0200 | [diff] [blame] | 304 | @FormParam("memberUsername") String memberUsername, |
| margaretha | d575bd9 | 2021-06-14 09:42:18 +0200 | [diff] [blame] | 305 | @FormParam("roleId") List<Integer> roleIds) { |
| margaretha | f7abb36 | 2018-09-18 20:09:37 +0200 | [diff] [blame] | 306 | TokenContext context = |
| 307 | (TokenContext) securityContext.getUserPrincipal(); |
| 308 | try { |
| 309 | scopeService.verifyScope(context, |
| 310 | OAuth2Scope.EDIT_USER_GROUP_MEMBER_ROLE); |
| margaretha | a18ab2b | 2019-11-11 12:55:26 +0100 | [diff] [blame] | 311 | service.editMemberRoles(context.getUsername(), groupName, |
| margaretha | f7abb36 | 2018-09-18 20:09:37 +0200 | [diff] [blame] | 312 | memberUsername, roleIds); |
| margaretha | c9f40e2 | 2019-08-07 17:32:19 +0200 | [diff] [blame] | 313 | return Response.ok("SUCCESS").build(); |
| margaretha | f7abb36 | 2018-09-18 20:09:37 +0200 | [diff] [blame] | 314 | } |
| 315 | catch (KustvaktException e) { |
| 316 | throw kustvaktResponseHandler.throwit(e); |
| 317 | } |
| 318 | } |
| 319 | |
| margaretha | 20f3123 | 2018-07-09 17:49:39 +0200 | [diff] [blame] | 320 | /** |
| 321 | * Adds roles of an active member of a user-group. Only user-group |
| margaretha | 0e1fc55 | 2019-08-08 15:31:01 +0200 | [diff] [blame] | 322 | * admins and system admins are allowed. |
| margaretha | 18533fd | 2018-03-28 16:01:06 +0200 | [diff] [blame] | 323 | * |
| 324 | * @param securityContext |
| margaretha | a18ab2b | 2019-11-11 12:55:26 +0100 | [diff] [blame] | 325 | * @param groupName |
| 326 | * a group name |
| margaretha | 20f3123 | 2018-07-09 17:49:39 +0200 | [diff] [blame] | 327 | * @param memberUsername |
| margaretha | a18ab2b | 2019-11-11 12:55:26 +0100 | [diff] [blame] | 328 | * a username of a group member |
| margaretha | d575bd9 | 2021-06-14 09:42:18 +0200 | [diff] [blame] | 329 | * @param roleId |
| 330 | * a role id or multiple role ids |
| margaretha | 18533fd | 2018-03-28 16:01:06 +0200 | [diff] [blame] | 331 | * @return if successful, HTTP response status OK |
| 332 | */ |
| 333 | @POST |
| margaretha | 03b195a | 2019-11-12 14:57:15 +0100 | [diff] [blame] | 334 | @Path("@{groupName}/role/add") |
| margaretha | 18533fd | 2018-03-28 16:01:06 +0200 | [diff] [blame] | 335 | @Consumes(MediaType.APPLICATION_FORM_URLENCODED) |
| 336 | public Response addMemberRoles (@Context SecurityContext securityContext, |
| margaretha | a18ab2b | 2019-11-11 12:55:26 +0100 | [diff] [blame] | 337 | @PathParam("groupName") String groupName, |
| margaretha | 18533fd | 2018-03-28 16:01:06 +0200 | [diff] [blame] | 338 | @FormParam("memberUsername") String memberUsername, |
| margaretha | d575bd9 | 2021-06-14 09:42:18 +0200 | [diff] [blame] | 339 | @FormParam("roleId") List<Integer> roleIds) { |
| margaretha | 18533fd | 2018-03-28 16:01:06 +0200 | [diff] [blame] | 340 | TokenContext context = |
| 341 | (TokenContext) securityContext.getUserPrincipal(); |
| 342 | try { |
| margaretha | ee0cbfe | 2018-08-28 17:47:14 +0200 | [diff] [blame] | 343 | scopeService.verifyScope(context, |
| 344 | OAuth2Scope.ADD_USER_GROUP_MEMBER_ROLE); |
| margaretha | a18ab2b | 2019-11-11 12:55:26 +0100 | [diff] [blame] | 345 | service.addMemberRoles(context.getUsername(), groupName, |
| margaretha | 18533fd | 2018-03-28 16:01:06 +0200 | [diff] [blame] | 346 | memberUsername, roleIds); |
| margaretha | c9f40e2 | 2019-08-07 17:32:19 +0200 | [diff] [blame] | 347 | return Response.ok("SUCCESS").build(); |
| margaretha | 18533fd | 2018-03-28 16:01:06 +0200 | [diff] [blame] | 348 | } |
| 349 | catch (KustvaktException e) { |
| margaretha | da3c785 | 2018-06-14 20:35:11 +0200 | [diff] [blame] | 350 | throw kustvaktResponseHandler.throwit(e); |
| margaretha | 18533fd | 2018-03-28 16:01:06 +0200 | [diff] [blame] | 351 | } |
| 352 | } |
| 353 | |
| margaretha | 20f3123 | 2018-07-09 17:49:39 +0200 | [diff] [blame] | 354 | /** |
| margaretha | a18ab2b | 2019-11-11 12:55:26 +0100 | [diff] [blame] | 355 | * Updates the roles of a member of a user-group by removing the |
| margaretha | ca7cff8 | 2019-11-12 12:06:37 +0100 | [diff] [blame] | 356 | * given roles. Only user-group admins and system admins are |
| 357 | * allowed. |
| margaretha | 18533fd | 2018-03-28 16:01:06 +0200 | [diff] [blame] | 358 | * |
| 359 | * @param securityContext |
| margaretha | a18ab2b | 2019-11-11 12:55:26 +0100 | [diff] [blame] | 360 | * @param groupName |
| 361 | * a group name |
| margaretha | 20f3123 | 2018-07-09 17:49:39 +0200 | [diff] [blame] | 362 | * @param memberUsername |
| margaretha | a18ab2b | 2019-11-11 12:55:26 +0100 | [diff] [blame] | 363 | * a username of a group member |
| margaretha | d575bd9 | 2021-06-14 09:42:18 +0200 | [diff] [blame] | 364 | * @param roleId |
| 365 | * a role id or multiple role ids |
| margaretha | 18533fd | 2018-03-28 16:01:06 +0200 | [diff] [blame] | 366 | * @return if successful, HTTP response status OK |
| 367 | */ |
| 368 | @POST |
| margaretha | 03b195a | 2019-11-12 14:57:15 +0100 | [diff] [blame] | 369 | @Path("@{groupName}/role/delete") |
| margaretha | 18533fd | 2018-03-28 16:01:06 +0200 | [diff] [blame] | 370 | @Consumes(MediaType.APPLICATION_FORM_URLENCODED) |
| 371 | public Response deleteMemberRoles (@Context SecurityContext securityContext, |
| margaretha | a18ab2b | 2019-11-11 12:55:26 +0100 | [diff] [blame] | 372 | @PathParam("groupName") String groupName, |
| margaretha | 18533fd | 2018-03-28 16:01:06 +0200 | [diff] [blame] | 373 | @FormParam("memberUsername") String memberUsername, |
| margaretha | d575bd9 | 2021-06-14 09:42:18 +0200 | [diff] [blame] | 374 | @FormParam("roleId") List<Integer> roleIds) { |
| margaretha | 18533fd | 2018-03-28 16:01:06 +0200 | [diff] [blame] | 375 | TokenContext context = |
| 376 | (TokenContext) securityContext.getUserPrincipal(); |
| 377 | try { |
| margaretha | ee0cbfe | 2018-08-28 17:47:14 +0200 | [diff] [blame] | 378 | scopeService.verifyScope(context, |
| 379 | OAuth2Scope.DELETE_USER_GROUP_MEMBER_ROLE); |
| margaretha | a18ab2b | 2019-11-11 12:55:26 +0100 | [diff] [blame] | 380 | service.deleteMemberRoles(context.getUsername(), groupName, |
| margaretha | 18533fd | 2018-03-28 16:01:06 +0200 | [diff] [blame] | 381 | memberUsername, roleIds); |
| margaretha | c9f40e2 | 2019-08-07 17:32:19 +0200 | [diff] [blame] | 382 | return Response.ok("SUCCESS").build(); |
| margaretha | 18533fd | 2018-03-28 16:01:06 +0200 | [diff] [blame] | 383 | } |
| 384 | catch (KustvaktException e) { |
| margaretha | da3c785 | 2018-06-14 20:35:11 +0200 | [diff] [blame] | 385 | throw kustvaktResponseHandler.throwit(e); |
| margaretha | 18533fd | 2018-03-28 16:01:06 +0200 | [diff] [blame] | 386 | } |
| 387 | } |
| 388 | |
| margaretha | 20f3123 | 2018-07-09 17:49:39 +0200 | [diff] [blame] | 389 | /** |
| 390 | * Handles requests to accept membership invitation. Only invited |
| margaretha | da5a6ab | 2019-11-08 10:06:05 +0100 | [diff] [blame] | 391 | * users can subscribe to the corresponding user-group. |
| margaretha | 18533fd | 2018-03-28 16:01:06 +0200 | [diff] [blame] | 392 | * |
| 393 | * @param securityContext |
| margaretha | a18ab2b | 2019-11-11 12:55:26 +0100 | [diff] [blame] | 394 | * @param groupName |
| 395 | * a group name |
| margaretha | 18533fd | 2018-03-28 16:01:06 +0200 | [diff] [blame] | 396 | * @return if successful, HTTP response status OK |
| 397 | */ |
| margaretha | 9d3eb04 | 2017-12-22 11:02:30 +0100 | [diff] [blame] | 398 | @POST |
| margaretha | 03b195a | 2019-11-12 14:57:15 +0100 | [diff] [blame] | 399 | @Path("@{groupName}/subscribe") |
| margaretha | 9d3eb04 | 2017-12-22 11:02:30 +0100 | [diff] [blame] | 400 | public Response subscribeToGroup (@Context SecurityContext securityContext, |
| margaretha | a18ab2b | 2019-11-11 12:55:26 +0100 | [diff] [blame] | 401 | @PathParam("groupName") String groupName) { |
| margaretha | 9d3eb04 | 2017-12-22 11:02:30 +0100 | [diff] [blame] | 402 | TokenContext context = |
| 403 | (TokenContext) securityContext.getUserPrincipal(); |
| 404 | try { |
| margaretha | ee0cbfe | 2018-08-28 17:47:14 +0200 | [diff] [blame] | 405 | scopeService.verifyScope(context, |
| 406 | OAuth2Scope.ADD_USER_GROUP_MEMBER); |
| margaretha | a18ab2b | 2019-11-11 12:55:26 +0100 | [diff] [blame] | 407 | service.acceptInvitation(groupName, context.getUsername()); |
| margaretha | c9f40e2 | 2019-08-07 17:32:19 +0200 | [diff] [blame] | 408 | return Response.ok("SUCCESS").build(); |
| margaretha | 9d3eb04 | 2017-12-22 11:02:30 +0100 | [diff] [blame] | 409 | } |
| 410 | catch (KustvaktException e) { |
| margaretha | da3c785 | 2018-06-14 20:35:11 +0200 | [diff] [blame] | 411 | throw kustvaktResponseHandler.throwit(e); |
| margaretha | 9d3eb04 | 2017-12-22 11:02:30 +0100 | [diff] [blame] | 412 | } |
| margaretha | 9d3eb04 | 2017-12-22 11:02:30 +0100 | [diff] [blame] | 413 | } |
| 414 | |
| margaretha | 20f3123 | 2018-07-09 17:49:39 +0200 | [diff] [blame] | 415 | /** |
| 416 | * Handles requests to reject membership invitation. A member can |
| margaretha | da5a6ab | 2019-11-08 10:06:05 +0100 | [diff] [blame] | 417 | * only unsubscribe him/herself from a group. |
| margaretha | 18533fd | 2018-03-28 16:01:06 +0200 | [diff] [blame] | 418 | * |
| margaretha | ca7cff8 | 2019-11-12 12:06:37 +0100 | [diff] [blame] | 419 | * Implemented identical to |
| 420 | * {@link #removeUserFromGroup(SecurityContext, String, String)}. |
| margaretha | 18533fd | 2018-03-28 16:01:06 +0200 | [diff] [blame] | 421 | * |
| 422 | * @param securityContext |
| margaretha | a18ab2b | 2019-11-11 12:55:26 +0100 | [diff] [blame] | 423 | * @param groupName |
| margaretha | 18533fd | 2018-03-28 16:01:06 +0200 | [diff] [blame] | 424 | * @return if successful, HTTP response status OK |
| 425 | */ |
| margaretha | a18ab2b | 2019-11-11 12:55:26 +0100 | [diff] [blame] | 426 | @DELETE |
| margaretha | 03b195a | 2019-11-12 14:57:15 +0100 | [diff] [blame] | 427 | @Path("@{groupName}/unsubscribe") |
| margaretha | 9d3eb04 | 2017-12-22 11:02:30 +0100 | [diff] [blame] | 428 | public Response unsubscribeFromGroup ( |
| 429 | @Context SecurityContext securityContext, |
| margaretha | a18ab2b | 2019-11-11 12:55:26 +0100 | [diff] [blame] | 430 | @PathParam("groupName") String groupName) { |
| margaretha | 9d3eb04 | 2017-12-22 11:02:30 +0100 | [diff] [blame] | 431 | TokenContext context = |
| 432 | (TokenContext) securityContext.getUserPrincipal(); |
| 433 | try { |
| margaretha | ee0cbfe | 2018-08-28 17:47:14 +0200 | [diff] [blame] | 434 | scopeService.verifyScope(context, |
| 435 | OAuth2Scope.DELETE_USER_GROUP_MEMBER); |
| margaretha | a18ab2b | 2019-11-11 12:55:26 +0100 | [diff] [blame] | 436 | service.deleteGroupMember(context.getUsername(), groupName, |
| margaretha | e6c711b | 2018-02-06 21:55:04 +0100 | [diff] [blame] | 437 | context.getUsername()); |
| margaretha | c9f40e2 | 2019-08-07 17:32:19 +0200 | [diff] [blame] | 438 | return Response.ok("SUCCESS").build(); |
| margaretha | 9d3eb04 | 2017-12-22 11:02:30 +0100 | [diff] [blame] | 439 | } |
| 440 | catch (KustvaktException e) { |
| margaretha | da3c785 | 2018-06-14 20:35:11 +0200 | [diff] [blame] | 441 | throw kustvaktResponseHandler.throwit(e); |
| margaretha | 9d3eb04 | 2017-12-22 11:02:30 +0100 | [diff] [blame] | 442 | } |
| margaretha | 9d3eb04 | 2017-12-22 11:02:30 +0100 | [diff] [blame] | 443 | } |
| margaretha | 0b63de4 | 2017-12-20 18:48:09 +0100 | [diff] [blame] | 444 | } |