| margaretha | f093afb | 2017-11-12 21:38:23 +0100 | [diff] [blame] | 1 | package de.ids_mannheim.korap.service; |
| 2 | |
| margaretha | f7abb36 | 2018-09-18 20:09:37 +0200 | [diff] [blame] | 3 | import java.sql.SQLException; |
| margaretha | 351777b | 2017-12-13 19:55:04 +0100 | [diff] [blame] | 4 | import java.util.ArrayList; |
| margaretha | 4457383 | 2018-03-21 16:59:59 +0100 | [diff] [blame] | 5 | import java.util.Collections; |
| margaretha | 541b8cc | 2018-01-10 13:02:46 +0100 | [diff] [blame] | 6 | import java.util.Iterator; |
| margaretha | 351777b | 2017-12-13 19:55:04 +0100 | [diff] [blame] | 7 | import java.util.List; |
| margaretha | 2e1781f | 2018-08-21 11:45:26 +0200 | [diff] [blame] | 8 | import java.util.regex.Pattern; |
| margaretha | 351777b | 2017-12-13 19:55:04 +0100 | [diff] [blame] | 9 | |
| margaretha | 49cb688 | 2018-07-04 04:19:54 +0200 | [diff] [blame] | 10 | import org.apache.logging.log4j.LogManager; |
| 11 | import org.apache.logging.log4j.Logger; |
| margaretha | f093afb | 2017-11-12 21:38:23 +0100 | [diff] [blame] | 12 | import org.springframework.beans.factory.annotation.Autowired; |
| 13 | import org.springframework.stereotype.Service; |
| 14 | |
| 15 | import com.fasterxml.jackson.core.JsonProcessingException; |
| 16 | import com.fasterxml.jackson.databind.JsonNode; |
| 17 | |
| margaretha | 7c1f428 | 2021-11-29 17:27:53 +0100 | [diff] [blame] | 18 | import de.ids_mannheim.korap.cache.VirtualCorpusCache; |
| margaretha | 56e8e55 | 2017-12-05 16:31:21 +0100 | [diff] [blame] | 19 | import de.ids_mannheim.korap.config.FullConfiguration; |
| margaretha | b874ef5 | 2018-01-23 20:26:31 +0100 | [diff] [blame] | 20 | import de.ids_mannheim.korap.constant.GroupMemberStatus; |
| margaretha | b097fb0 | 2021-02-22 19:28:33 +0100 | [diff] [blame] | 21 | import de.ids_mannheim.korap.constant.QueryAccessStatus; |
| margaretha | 7c1f428 | 2021-11-29 17:27:53 +0100 | [diff] [blame] | 22 | import de.ids_mannheim.korap.constant.QueryType; |
| Akron | da08015 | 2020-12-03 13:53:29 +0100 | [diff] [blame] | 23 | import de.ids_mannheim.korap.constant.ResourceType; |
| margaretha | 4edc70e | 2018-03-14 22:34:29 +0100 | [diff] [blame] | 24 | import de.ids_mannheim.korap.dao.AdminDao; |
| margaretha | b097fb0 | 2021-02-22 19:28:33 +0100 | [diff] [blame] | 25 | import de.ids_mannheim.korap.dao.QueryAccessDao; |
| 26 | import de.ids_mannheim.korap.dao.QueryDao; |
| 27 | import de.ids_mannheim.korap.dto.QueryAccessDto; |
| 28 | import de.ids_mannheim.korap.dto.QueryDto; |
| 29 | import de.ids_mannheim.korap.dto.converter.QueryAccessConverter; |
| 30 | import de.ids_mannheim.korap.dto.converter.QueryConverter; |
| margaretha | 7c1f428 | 2021-11-29 17:27:53 +0100 | [diff] [blame] | 31 | import de.ids_mannheim.korap.entity.QueryAccess; |
| 32 | import de.ids_mannheim.korap.entity.QueryDO; |
| margaretha | 45dde68 | 2018-01-04 21:33:46 +0100 | [diff] [blame] | 33 | import de.ids_mannheim.korap.entity.UserGroup; |
| margaretha | cfea1ae | 2018-01-15 20:27:26 +0100 | [diff] [blame] | 34 | import de.ids_mannheim.korap.entity.UserGroupMember; |
| margaretha | f093afb | 2017-11-12 21:38:23 +0100 | [diff] [blame] | 35 | import de.ids_mannheim.korap.exceptions.KustvaktException; |
| 36 | import de.ids_mannheim.korap.exceptions.StatusCodes; |
| 37 | import de.ids_mannheim.korap.query.serialize.QuerySerializer; |
| margaretha | f093afb | 2017-11-12 21:38:23 +0100 | [diff] [blame] | 38 | import de.ids_mannheim.korap.user.User.CorpusAccess; |
| 39 | import de.ids_mannheim.korap.utils.JsonUtils; |
| 40 | import de.ids_mannheim.korap.utils.KoralCollectionQueryBuilder; |
| margaretha | d3bc71f | 2018-01-03 20:35:06 +0100 | [diff] [blame] | 41 | import de.ids_mannheim.korap.utils.ParameterChecker; |
| margaretha | f093afb | 2017-11-12 21:38:23 +0100 | [diff] [blame] | 42 | import de.ids_mannheim.korap.web.SearchKrill; |
| margaretha | 89bd8f5 | 2021-02-26 17:08:01 +0100 | [diff] [blame] | 43 | import de.ids_mannheim.korap.web.controller.QueryReferenceController; |
| margaretha | 0b63de4 | 2017-12-20 18:48:09 +0100 | [diff] [blame] | 44 | import de.ids_mannheim.korap.web.controller.VirtualCorpusController; |
| Akron | da08015 | 2020-12-03 13:53:29 +0100 | [diff] [blame] | 45 | import de.ids_mannheim.korap.web.input.QueryJson; |
| margaretha | 96c309d | 2023-08-16 12:24:12 +0200 | [diff] [blame] | 46 | import jakarta.ws.rs.core.Response.Status; |
| margaretha | f093afb | 2017-11-12 21:38:23 +0100 | [diff] [blame] | 47 | |
| margaretha | 2e1781f | 2018-08-21 11:45:26 +0200 | [diff] [blame] | 48 | /** |
| margaretha | 89bd8f5 | 2021-02-26 17:08:01 +0100 | [diff] [blame] | 49 | * QueryService handles the logic behind |
| 50 | * {@link VirtualCorpusController} and |
| 51 | * {@link QueryReferenceController}. Virtual corpora and |
| 52 | * stored-queries are both treated as queries of different types. |
| 53 | * Thus, they are handled logically similarly. |
| 54 | * |
| 55 | * QueryService communicates with {@link QueryDao}, handles |
| 56 | * {@link QueryDO} and |
| 57 | * returns |
| 58 | * {@link QueryDto} to {@link VirtualCorpusController} and |
| 59 | * {@link QueryReferenceController}. |
| margaretha | 0b63de4 | 2017-12-20 18:48:09 +0100 | [diff] [blame] | 60 | * |
| 61 | * @author margaretha |
| 62 | * |
| 63 | */ |
| margaretha | f093afb | 2017-11-12 21:38:23 +0100 | [diff] [blame] | 64 | @Service |
| margaretha | b097fb0 | 2021-02-22 19:28:33 +0100 | [diff] [blame] | 65 | public class QueryService { |
| margaretha | f093afb | 2017-11-12 21:38:23 +0100 | [diff] [blame] | 66 | |
| margaretha | 923c207 | 2022-01-31 14:45:47 +0100 | [diff] [blame] | 67 | public static Logger jlog = LogManager.getLogger(QueryService.class); |
| margaretha | f093afb | 2017-11-12 21:38:23 +0100 | [diff] [blame] | 68 | |
| margaretha | dda4ef7 | 2018-12-06 14:20:51 +0100 | [diff] [blame] | 69 | public static boolean DEBUG = false; |
| margaretha | 8c20396 | 2019-01-14 17:01:33 +0100 | [diff] [blame] | 70 | |
| margaretha | 9645ab0 | 2022-03-31 11:33:03 +0200 | [diff] [blame] | 71 | public static Pattern queryNamePattern = Pattern |
| 72 | .compile("[a-zA-Z0-9]+[a-zA-Z_0-9-.]+"); |
| margaretha | 2e1781f | 2018-08-21 11:45:26 +0200 | [diff] [blame] | 73 | |
| margaretha | f093afb | 2017-11-12 21:38:23 +0100 | [diff] [blame] | 74 | @Autowired |
| margaretha | b097fb0 | 2021-02-22 19:28:33 +0100 | [diff] [blame] | 75 | private QueryDao queryDao; |
| margaretha | 541b8cc | 2018-01-10 13:02:46 +0100 | [diff] [blame] | 76 | @Autowired |
| margaretha | b097fb0 | 2021-02-22 19:28:33 +0100 | [diff] [blame] | 77 | private QueryAccessDao accessDao; |
| margaretha | f093afb | 2017-11-12 21:38:23 +0100 | [diff] [blame] | 78 | @Autowired |
| margaretha | 4edc70e | 2018-03-14 22:34:29 +0100 | [diff] [blame] | 79 | private AdminDao adminDao; |
| 80 | @Autowired |
| margaretha | 45dde68 | 2018-01-04 21:33:46 +0100 | [diff] [blame] | 81 | private UserGroupService userGroupService; |
| 82 | @Autowired |
| margaretha | f093afb | 2017-11-12 21:38:23 +0100 | [diff] [blame] | 83 | private SearchKrill krill; |
| margaretha | f093afb | 2017-11-12 21:38:23 +0100 | [diff] [blame] | 84 | @Autowired |
| margaretha | 56e8e55 | 2017-12-05 16:31:21 +0100 | [diff] [blame] | 85 | private FullConfiguration config; |
| margaretha | 351777b | 2017-12-13 19:55:04 +0100 | [diff] [blame] | 86 | @Autowired |
| margaretha | b097fb0 | 2021-02-22 19:28:33 +0100 | [diff] [blame] | 87 | private QueryConverter converter; |
| margaretha | fc7d777 | 2018-01-16 17:48:17 +0100 | [diff] [blame] | 88 | @Autowired |
| margaretha | b097fb0 | 2021-02-22 19:28:33 +0100 | [diff] [blame] | 89 | private QueryAccessConverter accessConverter; |
| margaretha | f093afb | 2017-11-12 21:38:23 +0100 | [diff] [blame] | 90 | |
| margaretha | 3ccaeb7 | 2019-02-28 18:40:22 +0100 | [diff] [blame] | 91 | private void verifyUsername (String contextUsername, String pathUsername) |
| margaretha | cfea1ae | 2018-01-15 20:27:26 +0100 | [diff] [blame] | 92 | throws KustvaktException { |
| margaretha | 3ccaeb7 | 2019-02-28 18:40:22 +0100 | [diff] [blame] | 93 | if (!contextUsername.equals(pathUsername) |
| 94 | && !adminDao.isAdmin(contextUsername)) { |
| 95 | throw new KustvaktException(StatusCodes.AUTHORIZATION_FAILED, |
| 96 | "Unauthorized operation for user: " + contextUsername, |
| 97 | contextUsername); |
| 98 | } |
| 99 | } |
| 100 | |
| margaretha | 35e1ca2 | 2023-11-16 22:00:01 +0100 | [diff] [blame^] | 101 | public List<QueryDto> listOwnerQuery (String username, QueryType queryType) |
| 102 | throws KustvaktException { |
| margaretha | b097fb0 | 2021-02-22 19:28:33 +0100 | [diff] [blame] | 103 | List<QueryDO> list = queryDao.retrieveOwnerQuery(username, queryType); |
| 104 | return createQueryDtos(list, queryType); |
| margaretha | cfea1ae | 2018-01-15 20:27:26 +0100 | [diff] [blame] | 105 | } |
| margaretha | 923c207 | 2022-01-31 14:45:47 +0100 | [diff] [blame] | 106 | |
| margaretha | aec6b41 | 2021-12-03 15:42:33 +0100 | [diff] [blame] | 107 | public List<QueryDto> listSystemQuery (QueryType queryType) |
| 108 | throws KustvaktException { |
| 109 | List<QueryDO> list = queryDao.retrieveQueryByType(ResourceType.SYSTEM, |
| 110 | null, queryType); |
| 111 | return createQueryDtos(list, queryType); |
| 112 | } |
| margaretha | cfea1ae | 2018-01-15 20:27:26 +0100 | [diff] [blame] | 113 | |
| margaretha | cf869e1 | 2023-04-03 15:47:45 +0200 | [diff] [blame] | 114 | public List<QueryDto> listAvailableQueryForUser (String username, |
| 115 | QueryType queryType) throws KustvaktException { |
| margaretha | 923c207 | 2022-01-31 14:45:47 +0100 | [diff] [blame] | 116 | List<QueryDO> list = queryDao.retrieveQueryByUser(username, queryType); |
| margaretha | b097fb0 | 2021-02-22 19:28:33 +0100 | [diff] [blame] | 117 | return createQueryDtos(list, queryType); |
| margaretha | cfea1ae | 2018-01-15 20:27:26 +0100 | [diff] [blame] | 118 | } |
| margaretha | b874ef5 | 2018-01-23 20:26:31 +0100 | [diff] [blame] | 119 | |
| margaretha | bdde7f4 | 2023-02-10 08:24:03 +0100 | [diff] [blame] | 120 | public List<QueryDto> listQueryByType (String createdBy, ResourceType type, |
| 121 | QueryType queryType) throws KustvaktException { |
| margaretha | 4457383 | 2018-03-21 16:59:59 +0100 | [diff] [blame] | 122 | |
| margaretha | 35e1ca2 | 2023-11-16 22:00:01 +0100 | [diff] [blame^] | 123 | List<QueryDO> virtualCorpora = queryDao.retrieveQueryByType(type, |
| 124 | createdBy, queryType); |
| margaretha | bdde7f4 | 2023-02-10 08:24:03 +0100 | [diff] [blame] | 125 | Collections.sort(virtualCorpora); |
| 126 | return createQueryDtos(virtualCorpora, queryType); |
| margaretha | 4457383 | 2018-03-21 16:59:59 +0100 | [diff] [blame] | 127 | |
| margaretha | 4457383 | 2018-03-21 16:59:59 +0100 | [diff] [blame] | 128 | } |
| 129 | |
| margaretha | 923c207 | 2022-01-31 14:45:47 +0100 | [diff] [blame] | 130 | private ArrayList<QueryDto> createQueryDtos (List<QueryDO> queryList, |
| 131 | QueryType queryType) throws KustvaktException { |
| margaretha | b097fb0 | 2021-02-22 19:28:33 +0100 | [diff] [blame] | 132 | ArrayList<QueryDto> dtos = new ArrayList<>(queryList.size()); |
| 133 | QueryDO query; |
| 134 | Iterator<QueryDO> i = queryList.iterator(); |
| margaretha | cfea1ae | 2018-01-15 20:27:26 +0100 | [diff] [blame] | 135 | while (i.hasNext()) { |
| margaretha | b097fb0 | 2021-02-22 19:28:33 +0100 | [diff] [blame] | 136 | query = i.next(); |
| margaretha | 923c207 | 2022-01-31 14:45:47 +0100 | [diff] [blame] | 137 | // String json = query.getKoralQuery(); |
| margaretha | 652c4dc | 2021-02-12 17:07:44 +0100 | [diff] [blame] | 138 | String statistics = null; |
| margaretha | 923c207 | 2022-01-31 14:45:47 +0100 | [diff] [blame] | 139 | // if (queryType.equals(QueryType.VIRTUAL_CORPUS)) { |
| 140 | // statistics = krill.getStatistics(json); |
| 141 | // } |
| 142 | QueryDto dto = converter.createQueryDto(query, statistics); |
| margaretha | b097fb0 | 2021-02-22 19:28:33 +0100 | [diff] [blame] | 143 | dtos.add(dto); |
| margaretha | cfea1ae | 2018-01-15 20:27:26 +0100 | [diff] [blame] | 144 | } |
| 145 | return dtos; |
| 146 | } |
| 147 | |
| margaretha | b097fb0 | 2021-02-22 19:28:33 +0100 | [diff] [blame] | 148 | public void deleteQueryByName (String username, String queryName, |
| margaretha | 89bd8f5 | 2021-02-26 17:08:01 +0100 | [diff] [blame] | 149 | String createdBy, QueryType type) throws KustvaktException { |
| margaretha | 8c20396 | 2019-01-14 17:01:33 +0100 | [diff] [blame] | 150 | |
| margaretha | b097fb0 | 2021-02-22 19:28:33 +0100 | [diff] [blame] | 151 | QueryDO query = queryDao.retrieveQueryByName(queryName, createdBy); |
| margaretha | 8c20396 | 2019-01-14 17:01:33 +0100 | [diff] [blame] | 152 | |
| margaretha | b097fb0 | 2021-02-22 19:28:33 +0100 | [diff] [blame] | 153 | if (query == null) { |
| 154 | String code = createdBy + "/" + queryName; |
| margaretha | b5e1e0a | 2019-01-29 22:11:57 +0100 | [diff] [blame] | 155 | throw new KustvaktException(StatusCodes.NO_RESOURCE_FOUND, |
| margaretha | 923c207 | 2022-01-31 14:45:47 +0100 | [diff] [blame] | 156 | "Query " + code + " is not found.", String.valueOf(code)); |
| margaretha | 4af3f1e | 2019-01-16 17:53:26 +0100 | [diff] [blame] | 157 | } |
| margaretha | b097fb0 | 2021-02-22 19:28:33 +0100 | [diff] [blame] | 158 | else if (query.getCreatedBy().equals(username) |
| margaretha | 4af3f1e | 2019-01-16 17:53:26 +0100 | [diff] [blame] | 159 | || adminDao.isAdmin(username)) { |
| margaretha | 8c20396 | 2019-01-14 17:01:33 +0100 | [diff] [blame] | 160 | |
| margaretha | b097fb0 | 2021-02-22 19:28:33 +0100 | [diff] [blame] | 161 | if (query.getType().equals(ResourceType.PUBLISHED)) { |
| margaretha | 35e1ca2 | 2023-11-16 22:00:01 +0100 | [diff] [blame^] | 162 | QueryAccess access = accessDao |
| 163 | .retrieveHiddenAccess(query.getId()); |
| margaretha | 8c20396 | 2019-01-14 17:01:33 +0100 | [diff] [blame] | 164 | accessDao.deleteAccess(access, "system"); |
| 165 | userGroupService.deleteAutoHiddenGroup( |
| 166 | access.getUserGroup().getId(), "system"); |
| 167 | } |
| margaretha | 89bd8f5 | 2021-02-26 17:08:01 +0100 | [diff] [blame] | 168 | if (type.equals(QueryType.VIRTUAL_CORPUS) |
| margaretha | 7c1f428 | 2021-11-29 17:27:53 +0100 | [diff] [blame] | 169 | && VirtualCorpusCache.contains(queryName)) { |
| margaretha | 923c207 | 2022-01-31 14:45:47 +0100 | [diff] [blame] | 170 | VirtualCorpusCache.delete(queryName); |
| margaretha | 47a72a8 | 2019-07-03 16:00:54 +0200 | [diff] [blame] | 171 | } |
| margaretha | b097fb0 | 2021-02-22 19:28:33 +0100 | [diff] [blame] | 172 | queryDao.deleteQuery(query); |
| margaretha | 8c20396 | 2019-01-14 17:01:33 +0100 | [diff] [blame] | 173 | } |
| 174 | else { |
| 175 | throw new KustvaktException(StatusCodes.AUTHORIZATION_FAILED, |
| 176 | "Unauthorized operation for user: " + username, username); |
| 177 | } |
| 178 | } |
| 179 | |
| margaretha | b097fb0 | 2021-02-22 19:28:33 +0100 | [diff] [blame] | 180 | public Status handlePutRequest (String username, String queryCreator, |
| 181 | String queryName, QueryJson queryJson) throws KustvaktException { |
| margaretha | cfea1ae | 2018-01-15 20:27:26 +0100 | [diff] [blame] | 182 | |
| margaretha | b097fb0 | 2021-02-22 19:28:33 +0100 | [diff] [blame] | 183 | verifyUsername(username, queryCreator); |
| 184 | QueryDO query = queryDao.retrieveQueryByName(queryName, queryCreator); |
| margaretha | 923c207 | 2022-01-31 14:45:47 +0100 | [diff] [blame] | 185 | |
| margaretha | b097fb0 | 2021-02-22 19:28:33 +0100 | [diff] [blame] | 186 | if (query == null) { |
| margaretha | b3ecbe3 | 2021-08-16 12:55:54 +0200 | [diff] [blame] | 187 | storeQuery(queryJson, queryName, queryCreator, username); |
| margaretha | bdd47ac | 2019-08-15 14:22:38 +0200 | [diff] [blame] | 188 | return Status.CREATED; |
| margaretha | 4af3f1e | 2019-01-16 17:53:26 +0100 | [diff] [blame] | 189 | } |
| 190 | else { |
| margaretha | b097fb0 | 2021-02-22 19:28:33 +0100 | [diff] [blame] | 191 | editQuery(query, queryJson, queryName, username); |
| margaretha | bdd47ac | 2019-08-15 14:22:38 +0200 | [diff] [blame] | 192 | return Status.NO_CONTENT; |
| margaretha | 4af3f1e | 2019-01-16 17:53:26 +0100 | [diff] [blame] | 193 | } |
| 194 | } |
| 195 | |
| margaretha | b097fb0 | 2021-02-22 19:28:33 +0100 | [diff] [blame] | 196 | public void editQuery (QueryDO existingQuery, QueryJson newQuery, |
| 197 | String queryName, String username) throws KustvaktException { |
| margaretha | 4af3f1e | 2019-01-16 17:53:26 +0100 | [diff] [blame] | 198 | |
| margaretha | b097fb0 | 2021-02-22 19:28:33 +0100 | [diff] [blame] | 199 | if (!username.equals(existingQuery.getCreatedBy()) |
| margaretha | 4edc70e | 2018-03-14 22:34:29 +0100 | [diff] [blame] | 200 | && !adminDao.isAdmin(username)) { |
| margaretha | cfea1ae | 2018-01-15 20:27:26 +0100 | [diff] [blame] | 201 | throw new KustvaktException(StatusCodes.AUTHORIZATION_FAILED, |
| 202 | "Unauthorized operation for user: " + username, username); |
| 203 | } |
| 204 | |
| 205 | String koralQuery = null; |
| 206 | CorpusAccess requiredAccess = null; |
| margaretha | b097fb0 | 2021-02-22 19:28:33 +0100 | [diff] [blame] | 207 | String corpusQuery = newQuery.getCorpusQuery(); |
| margaretha | 1703c16 | 2021-08-13 17:13:42 +0200 | [diff] [blame] | 208 | String query = newQuery.getQuery(); |
| 209 | String queryLanguage = newQuery.getQueryLanguage(); |
| margaretha | ed053fb | 2019-04-11 15:15:13 +0200 | [diff] [blame] | 210 | if (corpusQuery != null && !corpusQuery.isEmpty()) { |
| margaretha | 4af3f1e | 2019-01-16 17:53:26 +0100 | [diff] [blame] | 211 | koralQuery = serializeCorpusQuery(corpusQuery); |
| margaretha | 923c207 | 2022-01-31 14:45:47 +0100 | [diff] [blame] | 212 | requiredAccess = determineRequiredAccess(newQuery.isCached(), |
| 213 | queryName, koralQuery); |
| margaretha | cfea1ae | 2018-01-15 20:27:26 +0100 | [diff] [blame] | 214 | } |
| margaretha | 1703c16 | 2021-08-13 17:13:42 +0200 | [diff] [blame] | 215 | else if (query != null && !query.isEmpty() && queryLanguage != null |
| 216 | && !queryLanguage.isEmpty()) { |
| 217 | koralQuery = serializeQuery(query, queryLanguage); |
| 218 | } |
| margaretha | cfea1ae | 2018-01-15 20:27:26 +0100 | [diff] [blame] | 219 | |
| margaretha | b097fb0 | 2021-02-22 19:28:33 +0100 | [diff] [blame] | 220 | ResourceType type = newQuery.getType(); |
| margaretha | b874ef5 | 2018-01-23 20:26:31 +0100 | [diff] [blame] | 221 | if (type != null) { |
| margaretha | b097fb0 | 2021-02-22 19:28:33 +0100 | [diff] [blame] | 222 | if (existingQuery.getType().equals(ResourceType.PUBLISHED)) { |
| margaretha | b874ef5 | 2018-01-23 20:26:31 +0100 | [diff] [blame] | 223 | // withdraw from publication |
| Akron | da08015 | 2020-12-03 13:53:29 +0100 | [diff] [blame] | 224 | if (!type.equals(ResourceType.PUBLISHED)) { |
| margaretha | 923c207 | 2022-01-31 14:45:47 +0100 | [diff] [blame] | 225 | QueryAccess hiddenAccess = accessDao |
| 226 | .retrieveHiddenAccess(existingQuery.getId()); |
| margaretha | b097fb0 | 2021-02-22 19:28:33 +0100 | [diff] [blame] | 227 | deleteQueryAccess(hiddenAccess.getId(), "system"); |
| margaretha | b874ef5 | 2018-01-23 20:26:31 +0100 | [diff] [blame] | 228 | int groupId = hiddenAccess.getUserGroup().getId(); |
| 229 | userGroupService.deleteAutoHiddenGroup(groupId, "system"); |
| margaretha | 652c4dc | 2021-02-12 17:07:44 +0100 | [diff] [blame] | 230 | // EM: should the users within the hidden group |
| 231 | // receive |
| 232 | // notifications? |
| margaretha | b874ef5 | 2018-01-23 20:26:31 +0100 | [diff] [blame] | 233 | } |
| 234 | // else remains the same |
| 235 | } |
| Akron | da08015 | 2020-12-03 13:53:29 +0100 | [diff] [blame] | 236 | else if (type.equals(ResourceType.PUBLISHED)) { |
| margaretha | b097fb0 | 2021-02-22 19:28:33 +0100 | [diff] [blame] | 237 | publishQuery(existingQuery.getId()); |
| margaretha | b874ef5 | 2018-01-23 20:26:31 +0100 | [diff] [blame] | 238 | } |
| 239 | } |
| 240 | |
| margaretha | b097fb0 | 2021-02-22 19:28:33 +0100 | [diff] [blame] | 241 | queryDao.editQuery(existingQuery, queryName, type, requiredAccess, |
| 242 | koralQuery, newQuery.getDefinition(), newQuery.getDescription(), |
| margaretha | 923c207 | 2022-01-31 14:45:47 +0100 | [diff] [blame] | 243 | newQuery.getStatus(), newQuery.isCached(), query, |
| 244 | queryLanguage); |
| margaretha | b874ef5 | 2018-01-23 20:26:31 +0100 | [diff] [blame] | 245 | } |
| margaretha | cfea1ae | 2018-01-15 20:27:26 +0100 | [diff] [blame] | 246 | |
| margaretha | b097fb0 | 2021-02-22 19:28:33 +0100 | [diff] [blame] | 247 | private void publishQuery (int queryId) throws KustvaktException { |
| margaretha | b874ef5 | 2018-01-23 20:26:31 +0100 | [diff] [blame] | 248 | |
| margaretha | b097fb0 | 2021-02-22 19:28:33 +0100 | [diff] [blame] | 249 | QueryAccess access = accessDao.retrieveHiddenAccess(queryId); |
| margaretha | b874ef5 | 2018-01-23 20:26:31 +0100 | [diff] [blame] | 250 | // check if hidden access exists |
| 251 | if (access == null) { |
| margaretha | b097fb0 | 2021-02-22 19:28:33 +0100 | [diff] [blame] | 252 | QueryDO query = queryDao.retrieveQueryById(queryId); |
| margaretha | f7abb36 | 2018-09-18 20:09:37 +0200 | [diff] [blame] | 253 | // create and assign a new hidden group |
| margaretha | a18ab2b | 2019-11-11 12:55:26 +0100 | [diff] [blame] | 254 | int groupId = userGroupService.createAutoHiddenGroup(); |
| margaretha | 35e1ca2 | 2023-11-16 22:00:01 +0100 | [diff] [blame^] | 255 | UserGroup autoHidden = userGroupService |
| 256 | .retrieveUserGroupById(groupId); |
| margaretha | b097fb0 | 2021-02-22 19:28:33 +0100 | [diff] [blame] | 257 | accessDao.createAccessToQuery(query, autoHidden, "system", |
| 258 | QueryAccessStatus.HIDDEN); |
| margaretha | b874ef5 | 2018-01-23 20:26:31 +0100 | [diff] [blame] | 259 | } |
| 260 | else { |
| margaretha | f7abb36 | 2018-09-18 20:09:37 +0200 | [diff] [blame] | 261 | // should not happened |
| margaretha | b097fb0 | 2021-02-22 19:28:33 +0100 | [diff] [blame] | 262 | jlog.error("Cannot publish query with id: " + queryId |
| margaretha | f7abb36 | 2018-09-18 20:09:37 +0200 | [diff] [blame] | 263 | + ". Hidden access exists! Access id: " + access.getId()); |
| margaretha | cfea1ae | 2018-01-15 20:27:26 +0100 | [diff] [blame] | 264 | } |
| 265 | } |
| margaretha | 923c207 | 2022-01-31 14:45:47 +0100 | [diff] [blame] | 266 | |
| margaretha | b3ecbe3 | 2021-08-16 12:55:54 +0200 | [diff] [blame] | 267 | public void storeQuery (QueryJson query, String queryName, |
| 268 | String queryCreator, String username) throws KustvaktException { |
| margaretha | 9e73c0e | 2023-05-05 16:51:49 +0200 | [diff] [blame] | 269 | QueryType queryType = query.getQueryType(); |
| 270 | if (!checkNumberOfQueryLimit(username, queryType)) { |
| 271 | String type = queryType.displayName().toLowerCase(); |
| margaretha | 35e1ca2 | 2023-11-16 22:00:01 +0100 | [diff] [blame^] | 272 | throw new KustvaktException(StatusCodes.NOT_ALLOWED, |
| 273 | "Cannot create " + type + ". The maximum number " + "of " |
| 274 | + type + " has been reached."); |
| Akron | da08015 | 2020-12-03 13:53:29 +0100 | [diff] [blame] | 275 | } |
| margaretha | 35e1ca2 | 2023-11-16 22:00:01 +0100 | [diff] [blame^] | 276 | |
| margaretha | 9e73c0e | 2023-05-05 16:51:49 +0200 | [diff] [blame] | 277 | String koralQuery = computeKoralQuery(query); |
| margaretha | b3ecbe3 | 2021-08-16 12:55:54 +0200 | [diff] [blame] | 278 | storeQuery(username, queryName, query.getType(), query.getQueryType(), |
| 279 | koralQuery, query.getDefinition(), query.getDescription(), |
| 280 | query.getStatus(), query.isCached(), queryCreator, |
| margaretha | 479a447 | 2021-02-18 12:00:55 +0100 | [diff] [blame] | 281 | query.getQuery(), query.getQueryLanguage()); |
| margaretha | f7abb36 | 2018-09-18 20:09:37 +0200 | [diff] [blame] | 282 | } |
| margaretha | 35e1ca2 | 2023-11-16 22:00:01 +0100 | [diff] [blame^] | 283 | |
| margaretha | 9e73c0e | 2023-05-05 16:51:49 +0200 | [diff] [blame] | 284 | private boolean checkNumberOfQueryLimit (String username, |
| 285 | QueryType queryType) throws KustvaktException { |
| 286 | Long num = queryDao.countNumberOfQuery(username, queryType); |
| margaretha | 35e1ca2 | 2023-11-16 22:00:01 +0100 | [diff] [blame^] | 287 | if (num < config.getMaxNumberOfUserQueries()) |
| 288 | return true; |
| 289 | else |
| 290 | return false; |
| margaretha | 9e73c0e | 2023-05-05 16:51:49 +0200 | [diff] [blame] | 291 | } |
| margaretha | 35e1ca2 | 2023-11-16 22:00:01 +0100 | [diff] [blame^] | 292 | |
| 293 | private String computeKoralQuery (QueryJson query) |
| 294 | throws KustvaktException { |
| margaretha | 9e73c0e | 2023-05-05 16:51:49 +0200 | [diff] [blame] | 295 | if (query.getQueryType().equals(QueryType.VIRTUAL_CORPUS)) { |
| 296 | ParameterChecker.checkStringValue(query.getCorpusQuery(), |
| 297 | "corpusQuery"); |
| 298 | return serializeCorpusQuery(query.getCorpusQuery()); |
| 299 | } |
| margaretha | 35e1ca2 | 2023-11-16 22:00:01 +0100 | [diff] [blame^] | 300 | |
| margaretha | 9e73c0e | 2023-05-05 16:51:49 +0200 | [diff] [blame] | 301 | if (query.getQueryType().equals(QueryType.QUERY)) { |
| 302 | ParameterChecker.checkStringValue(query.getQuery(), "query"); |
| 303 | ParameterChecker.checkStringValue(query.getQueryLanguage(), |
| 304 | "queryLanguage"); |
| margaretha | 35e1ca2 | 2023-11-16 22:00:01 +0100 | [diff] [blame^] | 305 | return serializeQuery(query.getQuery(), query.getQueryLanguage()); |
| margaretha | 9e73c0e | 2023-05-05 16:51:49 +0200 | [diff] [blame] | 306 | } |
| margaretha | 35e1ca2 | 2023-11-16 22:00:01 +0100 | [diff] [blame^] | 307 | |
| margaretha | 9e73c0e | 2023-05-05 16:51:49 +0200 | [diff] [blame] | 308 | return null; |
| 309 | } |
| margaretha | f7abb36 | 2018-09-18 20:09:37 +0200 | [diff] [blame] | 310 | |
| margaretha | b3ecbe3 | 2021-08-16 12:55:54 +0200 | [diff] [blame] | 311 | public void storeQuery (String username, String queryName, |
| 312 | ResourceType type, QueryType queryType, String koralQuery, |
| 313 | String definition, String description, String status, |
| 314 | boolean isCached, String queryCreator, String query, |
| margaretha | 5213ced | 2021-02-17 12:27:59 +0100 | [diff] [blame] | 315 | String queryLanguage) throws KustvaktException { |
| margaretha | a8c364b | 2021-02-19 13:00:31 +0100 | [diff] [blame] | 316 | ParameterChecker.checkNameValue(queryName, "queryName"); |
| margaretha | f7abb36 | 2018-09-18 20:09:37 +0200 | [diff] [blame] | 317 | ParameterChecker.checkObjectValue(type, "type"); |
| 318 | |
| margaretha | 479a447 | 2021-02-18 12:00:55 +0100 | [diff] [blame] | 319 | if (!queryNamePattern.matcher(queryName).matches()) { |
| margaretha | 923c207 | 2022-01-31 14:45:47 +0100 | [diff] [blame] | 320 | throw new KustvaktException(StatusCodes.INVALID_ARGUMENT, queryType |
| margaretha | 35e1ca2 | 2023-11-16 22:00:01 +0100 | [diff] [blame^] | 321 | .displayName() |
| 322 | + " must consists of alphanumerical characters " |
| 323 | + "(limited to ASCII), underscores, dashes and periods. " |
| 324 | + "The name has to start with an alphanumerical character.", |
| margaretha | 479a447 | 2021-02-18 12:00:55 +0100 | [diff] [blame] | 325 | queryName); |
| margaretha | 2e1781f | 2018-08-21 11:45:26 +0200 | [diff] [blame] | 326 | } |
| margaretha | 45dde68 | 2018-01-04 21:33:46 +0100 | [diff] [blame] | 327 | |
| margaretha | 923c207 | 2022-01-31 14:45:47 +0100 | [diff] [blame] | 328 | if (type.equals(ResourceType.SYSTEM)) { |
| margaretha | b3ecbe3 | 2021-08-16 12:55:54 +0200 | [diff] [blame] | 329 | if (adminDao.isAdmin(username)) { |
| margaretha | 923c207 | 2022-01-31 14:45:47 +0100 | [diff] [blame] | 330 | queryCreator = "system"; |
| margaretha | b3ecbe3 | 2021-08-16 12:55:54 +0200 | [diff] [blame] | 331 | } |
| 332 | else if (!username.equals("system")) { |
| 333 | throw new KustvaktException(StatusCodes.AUTHORIZATION_FAILED, |
| margaretha | 923c207 | 2022-01-31 14:45:47 +0100 | [diff] [blame] | 334 | "Unauthorized operation for user: " + username, |
| 335 | username); |
| margaretha | b3ecbe3 | 2021-08-16 12:55:54 +0200 | [diff] [blame] | 336 | } |
| margaretha | f093afb | 2017-11-12 21:38:23 +0100 | [diff] [blame] | 337 | } |
| 338 | |
| margaretha | 479a447 | 2021-02-18 12:00:55 +0100 | [diff] [blame] | 339 | CorpusAccess requiredAccess = CorpusAccess.PUB; |
| 340 | if (queryType.equals(QueryType.VIRTUAL_CORPUS)) { |
| margaretha | 35e1ca2 | 2023-11-16 22:00:01 +0100 | [diff] [blame^] | 341 | requiredAccess = determineRequiredAccess(isCached, queryName, |
| 342 | koralQuery); |
| margaretha | 479a447 | 2021-02-18 12:00:55 +0100 | [diff] [blame] | 343 | } |
| margaretha | 8c20396 | 2019-01-14 17:01:33 +0100 | [diff] [blame] | 344 | |
| margaretha | 923c207 | 2022-01-31 14:45:47 +0100 | [diff] [blame] | 345 | if (DEBUG) { |
| margaretha | b097fb0 | 2021-02-22 19:28:33 +0100 | [diff] [blame] | 346 | jlog.debug("Storing query: " + queryName + "in the database "); |
| 347 | } |
| margaretha | 923c207 | 2022-01-31 14:45:47 +0100 | [diff] [blame] | 348 | |
| margaretha | 479a447 | 2021-02-18 12:00:55 +0100 | [diff] [blame] | 349 | int queryId = 0; |
| margaretha | f7abb36 | 2018-09-18 20:09:37 +0200 | [diff] [blame] | 350 | try { |
| margaretha | b097fb0 | 2021-02-22 19:28:33 +0100 | [diff] [blame] | 351 | queryId = queryDao.createQuery(queryName, type, queryType, |
| Akron | da08015 | 2020-12-03 13:53:29 +0100 | [diff] [blame] | 352 | requiredAccess, koralQuery, definition, description, status, |
| margaretha | b3ecbe3 | 2021-08-16 12:55:54 +0200 | [diff] [blame] | 353 | isCached, queryCreator, query, queryLanguage); |
| margaretha | 45dde68 | 2018-01-04 21:33:46 +0100 | [diff] [blame] | 354 | |
| margaretha | f7abb36 | 2018-09-18 20:09:37 +0200 | [diff] [blame] | 355 | } |
| 356 | catch (Exception e) { |
| 357 | Throwable cause = e; |
| 358 | Throwable lastCause = null; |
| 359 | while ((cause = cause.getCause()) != null |
| 360 | && !cause.equals(lastCause)) { |
| 361 | if (cause instanceof SQLException) { |
| 362 | break; |
| 363 | } |
| 364 | lastCause = cause; |
| 365 | } |
| 366 | throw new KustvaktException(StatusCodes.DB_INSERT_FAILED, |
| 367 | cause.getMessage()); |
| 368 | } |
| Akron | da08015 | 2020-12-03 13:53:29 +0100 | [diff] [blame] | 369 | if (type.equals(ResourceType.PUBLISHED)) { |
| margaretha | b097fb0 | 2021-02-22 19:28:33 +0100 | [diff] [blame] | 370 | publishQuery(queryId); |
| margaretha | 541b8cc | 2018-01-10 13:02:46 +0100 | [diff] [blame] | 371 | } |
| margaretha | f093afb | 2017-11-12 21:38:23 +0100 | [diff] [blame] | 372 | } |
| 373 | |
| margaretha | 541b8cc | 2018-01-10 13:02:46 +0100 | [diff] [blame] | 374 | private String serializeCorpusQuery (String corpusQuery) |
| margaretha | f093afb | 2017-11-12 21:38:23 +0100 | [diff] [blame] | 375 | throws KustvaktException { |
| 376 | QuerySerializer serializer = new QuerySerializer(); |
| margaretha | 541b8cc | 2018-01-10 13:02:46 +0100 | [diff] [blame] | 377 | serializer.setCollection(corpusQuery); |
| margaretha | f093afb | 2017-11-12 21:38:23 +0100 | [diff] [blame] | 378 | String koralQuery; |
| 379 | try { |
| 380 | koralQuery = serializer.convertCollectionToJson(); |
| 381 | } |
| 382 | catch (JsonProcessingException e) { |
| 383 | throw new KustvaktException(StatusCodes.INVALID_ARGUMENT, |
| margaretha | 541b8cc | 2018-01-10 13:02:46 +0100 | [diff] [blame] | 384 | "Invalid argument: " + corpusQuery, corpusQuery); |
| margaretha | f093afb | 2017-11-12 21:38:23 +0100 | [diff] [blame] | 385 | } |
| margaretha | dda4ef7 | 2018-12-06 14:20:51 +0100 | [diff] [blame] | 386 | if (DEBUG) { |
| 387 | jlog.debug(koralQuery); |
| 388 | } |
| margaretha | f093afb | 2017-11-12 21:38:23 +0100 | [diff] [blame] | 389 | return koralQuery; |
| 390 | } |
| margaretha | 652c4dc | 2021-02-12 17:07:44 +0100 | [diff] [blame] | 391 | |
| Akron | da08015 | 2020-12-03 13:53:29 +0100 | [diff] [blame] | 392 | private String serializeQuery (String query, String queryLanguage) |
| 393 | throws KustvaktException { |
| 394 | QuerySerializer serializer = new QuerySerializer(); |
| 395 | String koralQuery; |
| margaretha | 652c4dc | 2021-02-12 17:07:44 +0100 | [diff] [blame] | 396 | koralQuery = serializer.setQuery(query, queryLanguage).toJSON(); |
| Akron | da08015 | 2020-12-03 13:53:29 +0100 | [diff] [blame] | 397 | if (DEBUG) { |
| 398 | jlog.debug(koralQuery); |
| 399 | } |
| 400 | return koralQuery; |
| 401 | } |
| margaretha | f093afb | 2017-11-12 21:38:23 +0100 | [diff] [blame] | 402 | |
| margaretha | 4af3f1e | 2019-01-16 17:53:26 +0100 | [diff] [blame] | 403 | public CorpusAccess determineRequiredAccess (boolean isCached, String name, |
| 404 | String koralQuery) throws KustvaktException { |
| 405 | |
| 406 | if (isCached) { |
| margaretha | 35e1ca2 | 2023-11-16 22:00:01 +0100 | [diff] [blame^] | 407 | KoralCollectionQueryBuilder koral = new KoralCollectionQueryBuilder(); |
| margaretha | 4af3f1e | 2019-01-16 17:53:26 +0100 | [diff] [blame] | 408 | koral.with("referTo " + name); |
| 409 | koralQuery = koral.toJSON(); |
| 410 | if (DEBUG) { |
| 411 | jlog.debug("Determine vc access with vc ref: " + koralQuery); |
| 412 | } |
| 413 | |
| 414 | } |
| margaretha | f093afb | 2017-11-12 21:38:23 +0100 | [diff] [blame] | 415 | |
| 416 | if (findDocWithLicense(koralQuery, config.getAllOnlyRegex())) { |
| 417 | return CorpusAccess.ALL; |
| 418 | } |
| 419 | else if (findDocWithLicense(koralQuery, config.getPublicOnlyRegex())) { |
| 420 | return CorpusAccess.PUB; |
| 421 | } |
| 422 | else { |
| 423 | return CorpusAccess.FREE; |
| 424 | } |
| 425 | } |
| 426 | |
| 427 | private boolean findDocWithLicense (String koralQuery, String license) |
| 428 | throws KustvaktException { |
| 429 | KoralCollectionQueryBuilder koral = new KoralCollectionQueryBuilder(); |
| 430 | koral.setBaseQuery(koralQuery); |
| 431 | koral.with("availability=/" + license + "/"); |
| 432 | String json = koral.toJSON(); |
| 433 | |
| 434 | String statistics = krill.getStatistics(json); |
| 435 | JsonNode node = JsonUtils.readTree(statistics); |
| 436 | int numberOfDoc = node.at("/documents").asInt(); |
| margaretha | dda4ef7 | 2018-12-06 14:20:51 +0100 | [diff] [blame] | 437 | if (DEBUG) { |
| 438 | jlog.debug( |
| 439 | "License: " + license + ", number of docs: " + numberOfDoc); |
| 440 | } |
| margaretha | f093afb | 2017-11-12 21:38:23 +0100 | [diff] [blame] | 441 | return (numberOfDoc > 0) ? true : false; |
| 442 | } |
| margaretha | 351777b | 2017-12-13 19:55:04 +0100 | [diff] [blame] | 443 | |
| margaretha | b097fb0 | 2021-02-22 19:28:33 +0100 | [diff] [blame] | 444 | public void shareQuery (String username, String createdBy, String queryName, |
| margaretha | 3ccaeb7 | 2019-02-28 18:40:22 +0100 | [diff] [blame] | 445 | String groupName) throws KustvaktException { |
| margaretha | cfea1ae | 2018-01-15 20:27:26 +0100 | [diff] [blame] | 446 | |
| margaretha | b097fb0 | 2021-02-22 19:28:33 +0100 | [diff] [blame] | 447 | QueryDO query = queryDao.retrieveQueryByName(queryName, createdBy); |
| 448 | if (query == null) { |
| 449 | String code = createdBy + "/" + queryName; |
| margaretha | ed053fb | 2019-04-11 15:15:13 +0200 | [diff] [blame] | 450 | throw new KustvaktException(StatusCodes.NO_RESOURCE_FOUND, |
| margaretha | 923c207 | 2022-01-31 14:45:47 +0100 | [diff] [blame] | 451 | "Query " + code + " is not found.", String.valueOf(code)); |
| margaretha | ed053fb | 2019-04-11 15:15:13 +0200 | [diff] [blame] | 452 | } |
| margaretha | b097fb0 | 2021-02-22 19:28:33 +0100 | [diff] [blame] | 453 | if (!username.equals(query.getCreatedBy()) |
| margaretha | 4edc70e | 2018-03-14 22:34:29 +0100 | [diff] [blame] | 454 | && !adminDao.isAdmin(username)) { |
| margaretha | cfea1ae | 2018-01-15 20:27:26 +0100 | [diff] [blame] | 455 | throw new KustvaktException(StatusCodes.AUTHORIZATION_FAILED, |
| 456 | "Unauthorized operation for user: " + username, username); |
| 457 | } |
| 458 | |
| margaretha | 35e1ca2 | 2023-11-16 22:00:01 +0100 | [diff] [blame^] | 459 | UserGroup userGroup = userGroupService |
| 460 | .retrieveUserGroupByName(groupName); |
| margaretha | cfea1ae | 2018-01-15 20:27:26 +0100 | [diff] [blame] | 461 | |
| margaretha | b097fb0 | 2021-02-22 19:28:33 +0100 | [diff] [blame] | 462 | if (!isQueryAccessAdmin(userGroup, username) |
| margaretha | 4edc70e | 2018-03-14 22:34:29 +0100 | [diff] [blame] | 463 | && !adminDao.isAdmin(username)) { |
| margaretha | cfea1ae | 2018-01-15 20:27:26 +0100 | [diff] [blame] | 464 | throw new KustvaktException(StatusCodes.AUTHORIZATION_FAILED, |
| 465 | "Unauthorized operation for user: " + username, username); |
| 466 | } |
| 467 | else { |
| margaretha | f7abb36 | 2018-09-18 20:09:37 +0200 | [diff] [blame] | 468 | try { |
| margaretha | b097fb0 | 2021-02-22 19:28:33 +0100 | [diff] [blame] | 469 | accessDao.createAccessToQuery(query, userGroup, username, |
| 470 | QueryAccessStatus.ACTIVE); |
| margaretha | f7abb36 | 2018-09-18 20:09:37 +0200 | [diff] [blame] | 471 | } |
| 472 | catch (Exception e) { |
| 473 | Throwable cause = e; |
| 474 | Throwable lastCause = null; |
| 475 | while ((cause = cause.getCause()) != null |
| 476 | && !cause.equals(lastCause)) { |
| 477 | if (cause instanceof SQLException) { |
| 478 | break; |
| 479 | } |
| 480 | lastCause = cause; |
| 481 | } |
| 482 | throw new KustvaktException(StatusCodes.DB_INSERT_FAILED, |
| 483 | cause.getMessage()); |
| 484 | } |
| margaretha | 652c4dc | 2021-02-12 17:07:44 +0100 | [diff] [blame] | 485 | |
| margaretha | b097fb0 | 2021-02-22 19:28:33 +0100 | [diff] [blame] | 486 | queryDao.editQuery(query, null, ResourceType.PROJECT, null, null, |
| margaretha | 1703c16 | 2021-08-13 17:13:42 +0200 | [diff] [blame] | 487 | null, null, null, query.isCached(), null, null); |
| margaretha | cfea1ae | 2018-01-15 20:27:26 +0100 | [diff] [blame] | 488 | } |
| 489 | } |
| 490 | |
| margaretha | b097fb0 | 2021-02-22 19:28:33 +0100 | [diff] [blame] | 491 | private boolean isQueryAccessAdmin (UserGroup userGroup, String username) |
| margaretha | cfea1ae | 2018-01-15 20:27:26 +0100 | [diff] [blame] | 492 | throws KustvaktException { |
| margaretha | 35e1ca2 | 2023-11-16 22:00:01 +0100 | [diff] [blame^] | 493 | List<UserGroupMember> accessAdmins = userGroupService |
| 494 | .retrieveQueryAccessAdmins(userGroup); |
| margaretha | cfea1ae | 2018-01-15 20:27:26 +0100 | [diff] [blame] | 495 | for (UserGroupMember m : accessAdmins) { |
| 496 | if (username.equals(m.getUserId())) { |
| 497 | return true; |
| 498 | } |
| 499 | } |
| 500 | return false; |
| 501 | } |
| 502 | |
| margaretha | 2e1781f | 2018-08-21 11:45:26 +0200 | [diff] [blame] | 503 | // public void editVCAccess (VirtualCorpusAccess access, String |
| 504 | // username) |
| 505 | // throws KustvaktException { |
| margaretha | 4edc70e | 2018-03-14 22:34:29 +0100 | [diff] [blame] | 506 | // |
| margaretha | 2e1781f | 2018-08-21 11:45:26 +0200 | [diff] [blame] | 507 | // // get all the VCA admins |
| 508 | // UserGroup userGroup = access.getUserGroup(); |
| 509 | // List<UserGroupMember> accessAdmins = |
| 510 | // userGroupService.retrieveVCAccessAdmins(userGroup); |
| margaretha | 4edc70e | 2018-03-14 22:34:29 +0100 | [diff] [blame] | 511 | // |
| margaretha | 2e1781f | 2018-08-21 11:45:26 +0200 | [diff] [blame] | 512 | // User user = authManager.getUser(username); |
| 513 | // if (!user.isSystemAdmin()) { |
| 514 | // throw new KustvaktException(StatusCodes.AUTHORIZATION_FAILED, |
| 515 | // "Unauthorized operation for user: " + username, username); |
| 516 | // } |
| 517 | // } |
| margaretha | 541b8cc | 2018-01-10 13:02:46 +0100 | [diff] [blame] | 518 | |
| margaretha | b097fb0 | 2021-02-22 19:28:33 +0100 | [diff] [blame] | 519 | public List<QueryAccessDto> listQueryAccessByUsername (String username) |
| margaretha | 3ccaeb7 | 2019-02-28 18:40:22 +0100 | [diff] [blame] | 520 | throws KustvaktException { |
| margaretha | b097fb0 | 2021-02-22 19:28:33 +0100 | [diff] [blame] | 521 | List<QueryAccess> accessList = new ArrayList<>(); |
| margaretha | 4edc70e | 2018-03-14 22:34:29 +0100 | [diff] [blame] | 522 | if (adminDao.isAdmin(username)) { |
| margaretha | 3ccaeb7 | 2019-02-28 18:40:22 +0100 | [diff] [blame] | 523 | accessList = accessDao.retrieveAllAccess(); |
| margaretha | cfea1ae | 2018-01-15 20:27:26 +0100 | [diff] [blame] | 524 | } |
| margaretha | fc7d777 | 2018-01-16 17:48:17 +0100 | [diff] [blame] | 525 | else { |
| margaretha | 35e1ca2 | 2023-11-16 22:00:01 +0100 | [diff] [blame^] | 526 | List<UserGroup> groups = userGroupService |
| 527 | .retrieveUserGroup(username); |
| margaretha | 652c4dc | 2021-02-12 17:07:44 +0100 | [diff] [blame] | 528 | for (UserGroup g : groups) { |
| margaretha | b097fb0 | 2021-02-22 19:28:33 +0100 | [diff] [blame] | 529 | if (isQueryAccessAdmin(g, username)) { |
| margaretha | 652c4dc | 2021-02-12 17:07:44 +0100 | [diff] [blame] | 530 | accessList.addAll( |
| 531 | accessDao.retrieveActiveAccessByGroup(g.getId())); |
| 532 | } |
| margaretha | cfea1ae | 2018-01-15 20:27:26 +0100 | [diff] [blame] | 533 | } |
| 534 | } |
| margaretha | b097fb0 | 2021-02-22 19:28:33 +0100 | [diff] [blame] | 535 | return accessConverter.createQueryAccessDto(accessList); |
| margaretha | cfea1ae | 2018-01-15 20:27:26 +0100 | [diff] [blame] | 536 | } |
| margaretha | 652c4dc | 2021-02-12 17:07:44 +0100 | [diff] [blame] | 537 | |
| margaretha | b097fb0 | 2021-02-22 19:28:33 +0100 | [diff] [blame] | 538 | public List<QueryAccessDto> listQueryAccessByQuery (String username, |
| 539 | String queryCreator, String queryName) throws KustvaktException { |
| margaretha | 4af3f1e | 2019-01-16 17:53:26 +0100 | [diff] [blame] | 540 | |
| margaretha | b097fb0 | 2021-02-22 19:28:33 +0100 | [diff] [blame] | 541 | List<QueryAccess> accessList; |
| margaretha | 4af3f1e | 2019-01-16 17:53:26 +0100 | [diff] [blame] | 542 | if (adminDao.isAdmin(username)) { |
| margaretha | 35e1ca2 | 2023-11-16 22:00:01 +0100 | [diff] [blame^] | 543 | accessList = accessDao.retrieveAllAccessByQuery(queryCreator, |
| 544 | queryName); |
| margaretha | 4af3f1e | 2019-01-16 17:53:26 +0100 | [diff] [blame] | 545 | } |
| 546 | else { |
| margaretha | 923c207 | 2022-01-31 14:45:47 +0100 | [diff] [blame] | 547 | accessList = accessDao.retrieveActiveAccessByQuery(queryCreator, |
| 548 | queryName); |
| margaretha | b097fb0 | 2021-02-22 19:28:33 +0100 | [diff] [blame] | 549 | List<QueryAccess> filteredAccessList = new ArrayList<>(); |
| 550 | for (QueryAccess access : accessList) { |
| margaretha | 4af3f1e | 2019-01-16 17:53:26 +0100 | [diff] [blame] | 551 | UserGroup userGroup = access.getUserGroup(); |
| margaretha | b097fb0 | 2021-02-22 19:28:33 +0100 | [diff] [blame] | 552 | if (isQueryAccessAdmin(userGroup, username)) { |
| margaretha | 4af3f1e | 2019-01-16 17:53:26 +0100 | [diff] [blame] | 553 | filteredAccessList.add(access); |
| 554 | } |
| 555 | } |
| 556 | accessList = filteredAccessList; |
| 557 | } |
| margaretha | b097fb0 | 2021-02-22 19:28:33 +0100 | [diff] [blame] | 558 | return accessConverter.createQueryAccessDto(accessList); |
| margaretha | 4af3f1e | 2019-01-16 17:53:26 +0100 | [diff] [blame] | 559 | } |
| 560 | |
| margaretha | 3ccaeb7 | 2019-02-28 18:40:22 +0100 | [diff] [blame] | 561 | @Deprecated |
| margaretha | b097fb0 | 2021-02-22 19:28:33 +0100 | [diff] [blame] | 562 | public List<QueryAccessDto> listVCAccessByGroup (String username, |
| margaretha | cfea1ae | 2018-01-15 20:27:26 +0100 | [diff] [blame] | 563 | int groupId) throws KustvaktException { |
| margaretha | cfea1ae | 2018-01-15 20:27:26 +0100 | [diff] [blame] | 564 | UserGroup userGroup = userGroupService.retrieveUserGroupById(groupId); |
| margaretha | fc7d777 | 2018-01-16 17:48:17 +0100 | [diff] [blame] | 565 | |
| margaretha | b097fb0 | 2021-02-22 19:28:33 +0100 | [diff] [blame] | 566 | List<QueryAccess> accessList; |
| margaretha | 4edc70e | 2018-03-14 22:34:29 +0100 | [diff] [blame] | 567 | if (adminDao.isAdmin(username)) { |
| margaretha | fc7d777 | 2018-01-16 17:48:17 +0100 | [diff] [blame] | 568 | accessList = accessDao.retrieveAllAccessByGroup(groupId); |
| 569 | } |
| margaretha | b097fb0 | 2021-02-22 19:28:33 +0100 | [diff] [blame] | 570 | else if (isQueryAccessAdmin(userGroup, username)) { |
| margaretha | fc7d777 | 2018-01-16 17:48:17 +0100 | [diff] [blame] | 571 | accessList = accessDao.retrieveActiveAccessByGroup(groupId); |
| 572 | } |
| 573 | else { |
| margaretha | cfea1ae | 2018-01-15 20:27:26 +0100 | [diff] [blame] | 574 | throw new KustvaktException(StatusCodes.AUTHORIZATION_FAILED, |
| 575 | "Unauthorized operation for user: " + username, username); |
| 576 | } |
| margaretha | fc7d777 | 2018-01-16 17:48:17 +0100 | [diff] [blame] | 577 | |
| margaretha | b097fb0 | 2021-02-22 19:28:33 +0100 | [diff] [blame] | 578 | return accessConverter.createQueryAccessDto(accessList); |
| margaretha | 541b8cc | 2018-01-10 13:02:46 +0100 | [diff] [blame] | 579 | } |
| margaretha | 652c4dc | 2021-02-12 17:07:44 +0100 | [diff] [blame] | 580 | |
| margaretha | b097fb0 | 2021-02-22 19:28:33 +0100 | [diff] [blame] | 581 | public List<QueryAccessDto> listQueryAccessByGroup (String username, |
| margaretha | 3ccaeb7 | 2019-02-28 18:40:22 +0100 | [diff] [blame] | 582 | String groupName) throws KustvaktException { |
| margaretha | 35e1ca2 | 2023-11-16 22:00:01 +0100 | [diff] [blame^] | 583 | UserGroup userGroup = userGroupService |
| 584 | .retrieveUserGroupByName(groupName); |
| margaretha | 3ccaeb7 | 2019-02-28 18:40:22 +0100 | [diff] [blame] | 585 | |
| margaretha | b097fb0 | 2021-02-22 19:28:33 +0100 | [diff] [blame] | 586 | List<QueryAccess> accessList; |
| margaretha | 3ccaeb7 | 2019-02-28 18:40:22 +0100 | [diff] [blame] | 587 | if (adminDao.isAdmin(username)) { |
| 588 | accessList = accessDao.retrieveAllAccessByGroup(userGroup.getId()); |
| 589 | } |
| margaretha | b097fb0 | 2021-02-22 19:28:33 +0100 | [diff] [blame] | 590 | else if (isQueryAccessAdmin(userGroup, username)) { |
| margaretha | 35e1ca2 | 2023-11-16 22:00:01 +0100 | [diff] [blame^] | 591 | accessList = accessDao |
| 592 | .retrieveActiveAccessByGroup(userGroup.getId()); |
| margaretha | 3ccaeb7 | 2019-02-28 18:40:22 +0100 | [diff] [blame] | 593 | } |
| 594 | else { |
| 595 | throw new KustvaktException(StatusCodes.AUTHORIZATION_FAILED, |
| 596 | "Unauthorized operation for user: " + username, username); |
| 597 | } |
| margaretha | b097fb0 | 2021-02-22 19:28:33 +0100 | [diff] [blame] | 598 | return accessConverter.createQueryAccessDto(accessList); |
| margaretha | 3ccaeb7 | 2019-02-28 18:40:22 +0100 | [diff] [blame] | 599 | } |
| margaretha | e8ab51d | 2018-01-16 19:27:40 +0100 | [diff] [blame] | 600 | |
| margaretha | b097fb0 | 2021-02-22 19:28:33 +0100 | [diff] [blame] | 601 | public void deleteQueryAccess (int accessId, String username) |
| margaretha | e8ab51d | 2018-01-16 19:27:40 +0100 | [diff] [blame] | 602 | throws KustvaktException { |
| 603 | |
| margaretha | b097fb0 | 2021-02-22 19:28:33 +0100 | [diff] [blame] | 604 | QueryAccess access = accessDao.retrieveAccessById(accessId); |
| margaretha | e8ab51d | 2018-01-16 19:27:40 +0100 | [diff] [blame] | 605 | UserGroup userGroup = access.getUserGroup(); |
| margaretha | b097fb0 | 2021-02-22 19:28:33 +0100 | [diff] [blame] | 606 | if (isQueryAccessAdmin(userGroup, username) |
| margaretha | 4edc70e | 2018-03-14 22:34:29 +0100 | [diff] [blame] | 607 | || adminDao.isAdmin(username)) { |
| margaretha | b874ef5 | 2018-01-23 20:26:31 +0100 | [diff] [blame] | 608 | accessDao.deleteAccess(access, username); |
| margaretha | e8ab51d | 2018-01-16 19:27:40 +0100 | [diff] [blame] | 609 | } |
| 610 | else { |
| 611 | throw new KustvaktException(StatusCodes.AUTHORIZATION_FAILED, |
| 612 | "Unauthorized operation for user: " + username, username); |
| 613 | } |
| 614 | |
| 615 | } |
| margaretha | 923c207 | 2022-01-31 14:45:47 +0100 | [diff] [blame] | 616 | |
| margaretha | 86c9be3 | 2021-12-08 19:09:18 +0100 | [diff] [blame] | 617 | public JsonNode retrieveKoralQuery (String username, String queryName, |
| 618 | String createdBy, QueryType queryType) throws KustvaktException { |
| margaretha | 35e1ca2 | 2023-11-16 22:00:01 +0100 | [diff] [blame^] | 619 | QueryDO query = searchQueryByName(username, queryName, createdBy, |
| 620 | queryType); |
| margaretha | 86c9be3 | 2021-12-08 19:09:18 +0100 | [diff] [blame] | 621 | String koralQuery = query.getKoralQuery(); |
| margaretha | 923c207 | 2022-01-31 14:45:47 +0100 | [diff] [blame] | 622 | JsonNode kq = JsonUtils.readTree(koralQuery); |
| margaretha | 86c9be3 | 2021-12-08 19:09:18 +0100 | [diff] [blame] | 623 | return kq; |
| 624 | } |
| margaretha | e8ab51d | 2018-01-16 19:27:40 +0100 | [diff] [blame] | 625 | |
| margaretha | 923c207 | 2022-01-31 14:45:47 +0100 | [diff] [blame] | 626 | public JsonNode retrieveFieldValues (String username, String queryName, |
| 627 | String createdBy, QueryType queryType, String fieldName) |
| 628 | throws KustvaktException { |
| margaretha | 35e1ca2 | 2023-11-16 22:00:01 +0100 | [diff] [blame^] | 629 | |
| margaretha | 21a8386 | 2022-03-07 16:56:08 +0100 | [diff] [blame] | 630 | ParameterChecker.checkStringValue(fieldName, "fieldName"); |
| margaretha | 35e1ca2 | 2023-11-16 22:00:01 +0100 | [diff] [blame^] | 631 | |
| 632 | // if (!adminDao.isAdmin(username)) { |
| 633 | // throw new KustvaktException(StatusCodes.AUTHORIZATION_FAILED, |
| 634 | // "Unauthorized operation for user: " + username, username); |
| 635 | // } |
| 636 | |
| margaretha | 923c207 | 2022-01-31 14:45:47 +0100 | [diff] [blame] | 637 | if (fieldName.equals("tokens") || fieldName.equals("base")) { |
| 638 | throw new KustvaktException(StatusCodes.NOT_ALLOWED, |
| margaretha | 35e1ca2 | 2023-11-16 22:00:01 +0100 | [diff] [blame^] | 639 | "Retrieving values of field " + fieldName |
| 640 | + " is not allowed."); |
| margaretha | 923c207 | 2022-01-31 14:45:47 +0100 | [diff] [blame] | 641 | } |
| 642 | else { |
| 643 | QueryDO query = searchQueryByName(username, queryName, createdBy, |
| 644 | queryType); |
| 645 | String koralQuery = query.getKoralQuery(); |
| 646 | return krill.getFieldValuesForVC(koralQuery, fieldName); |
| 647 | } |
| 648 | } |
| 649 | |
| margaretha | b097fb0 | 2021-02-22 19:28:33 +0100 | [diff] [blame] | 650 | public QueryDO searchQueryByName (String username, String queryName, |
| margaretha | 652c4dc | 2021-02-12 17:07:44 +0100 | [diff] [blame] | 651 | String createdBy, QueryType queryType) throws KustvaktException { |
| margaretha | b097fb0 | 2021-02-22 19:28:33 +0100 | [diff] [blame] | 652 | QueryDO query = queryDao.retrieveQueryByName(queryName, createdBy); |
| 653 | if (query == null) { |
| 654 | String code = createdBy + "/" + queryName; |
| margaretha | b5e1e0a | 2019-01-29 22:11:57 +0100 | [diff] [blame] | 655 | throw new KustvaktException(StatusCodes.NO_RESOURCE_FOUND, |
| margaretha | 923c207 | 2022-01-31 14:45:47 +0100 | [diff] [blame] | 656 | queryType.displayName() + " " + code + " is not found.", |
| margaretha | b097fb0 | 2021-02-22 19:28:33 +0100 | [diff] [blame] | 657 | String.valueOf(code)); |
| margaretha | 4af3f1e | 2019-01-16 17:53:26 +0100 | [diff] [blame] | 658 | } |
| margaretha | b097fb0 | 2021-02-22 19:28:33 +0100 | [diff] [blame] | 659 | checkQueryAccess(query, username); |
| 660 | return query; |
| margaretha | 563aabe | 2018-09-13 20:39:45 +0200 | [diff] [blame] | 661 | } |
| 662 | |
| margaretha | b097fb0 | 2021-02-22 19:28:33 +0100 | [diff] [blame] | 663 | public QueryDto retrieveQueryByName (String username, String queryName, |
| margaretha | 652c4dc | 2021-02-12 17:07:44 +0100 | [diff] [blame] | 664 | String createdBy, QueryType queryType) throws KustvaktException { |
| margaretha | 35e1ca2 | 2023-11-16 22:00:01 +0100 | [diff] [blame^] | 665 | QueryDO query = searchQueryByName(username, queryName, createdBy, |
| 666 | queryType); |
| margaretha | 923c207 | 2022-01-31 14:45:47 +0100 | [diff] [blame] | 667 | // String json = query.getKoralQuery(); |
| Akron | da08015 | 2020-12-03 13:53:29 +0100 | [diff] [blame] | 668 | String statistics = null; |
| margaretha | 923c207 | 2022-01-31 14:45:47 +0100 | [diff] [blame] | 669 | // long start,end; |
| 670 | // start = System.currentTimeMillis(); |
| 671 | // if (query.getQueryType().equals(QueryType.VIRTUAL_CORPUS)) |
| 672 | // { |
| 673 | // statistics = krill.getStatistics(json); |
| 674 | // } |
| 675 | // end = System.currentTimeMillis(); |
| 676 | // jlog.debug("{} statistics duration: {}", queryName, (end - |
| 677 | // start)); |
| margaretha | b097fb0 | 2021-02-22 19:28:33 +0100 | [diff] [blame] | 678 | return converter.createQueryDto(query, statistics); |
| margaretha | 8c20396 | 2019-01-14 17:01:33 +0100 | [diff] [blame] | 679 | } |
| 680 | |
| margaretha | b097fb0 | 2021-02-22 19:28:33 +0100 | [diff] [blame] | 681 | public QueryDto searchQueryById (String username, int queryId) |
| margaretha | e8ab51d | 2018-01-16 19:27:40 +0100 | [diff] [blame] | 682 | throws KustvaktException { |
| margaretha | e8ab51d | 2018-01-16 19:27:40 +0100 | [diff] [blame] | 683 | |
| margaretha | b097fb0 | 2021-02-22 19:28:33 +0100 | [diff] [blame] | 684 | QueryDO query = queryDao.retrieveQueryById(queryId); |
| 685 | checkQueryAccess(query, username); |
| margaretha | 923c207 | 2022-01-31 14:45:47 +0100 | [diff] [blame] | 686 | // String json = query.getKoralQuery(); |
| 687 | // String statistics = krill.getStatistics(json); |
| margaretha | 326520b | 2021-12-08 17:58:09 +0100 | [diff] [blame] | 688 | return converter.createQueryDto(query, null); |
| margaretha | 563aabe | 2018-09-13 20:39:45 +0200 | [diff] [blame] | 689 | } |
| 690 | |
| margaretha | b097fb0 | 2021-02-22 19:28:33 +0100 | [diff] [blame] | 691 | private void checkQueryAccess (QueryDO query, String username) |
| margaretha | 563aabe | 2018-09-13 20:39:45 +0200 | [diff] [blame] | 692 | throws KustvaktException { |
| margaretha | b097fb0 | 2021-02-22 19:28:33 +0100 | [diff] [blame] | 693 | ResourceType type = query.getType(); |
| margaretha | e8ab51d | 2018-01-16 19:27:40 +0100 | [diff] [blame] | 694 | |
| margaretha | 4edc70e | 2018-03-14 22:34:29 +0100 | [diff] [blame] | 695 | if (!adminDao.isAdmin(username) |
| margaretha | b097fb0 | 2021-02-22 19:28:33 +0100 | [diff] [blame] | 696 | && !username.equals(query.getCreatedBy())) { |
| Akron | da08015 | 2020-12-03 13:53:29 +0100 | [diff] [blame] | 697 | if (type.equals(ResourceType.PRIVATE) |
| 698 | || (type.equals(ResourceType.PROJECT) |
| margaretha | b097fb0 | 2021-02-22 19:28:33 +0100 | [diff] [blame] | 699 | && !hasAccess(username, query.getId()))) { |
| margaretha | e8ab51d | 2018-01-16 19:27:40 +0100 | [diff] [blame] | 700 | throw new KustvaktException(StatusCodes.AUTHORIZATION_FAILED, |
| 701 | "Unauthorized operation for user: " + username, |
| 702 | username); |
| 703 | } |
| margaretha | 2c24991 | 2018-01-17 20:07:20 +0100 | [diff] [blame] | 704 | |
| Akron | da08015 | 2020-12-03 13:53:29 +0100 | [diff] [blame] | 705 | else if (ResourceType.PUBLISHED.equals(type) |
| margaretha | 0e1fc55 | 2019-08-08 15:31:01 +0200 | [diff] [blame] | 706 | && !username.equals("guest")) { |
| margaretha | b097fb0 | 2021-02-22 19:28:33 +0100 | [diff] [blame] | 707 | // add user in the query's auto group |
| margaretha | 563aabe | 2018-09-13 20:39:45 +0200 | [diff] [blame] | 708 | UserGroup userGroup = userGroupService |
| margaretha | b097fb0 | 2021-02-22 19:28:33 +0100 | [diff] [blame] | 709 | .retrieveHiddenUserGroupByQuery(query.getId()); |
| margaretha | 4566792 | 2018-01-25 21:23:03 +0100 | [diff] [blame] | 710 | try { |
| margaretha | 18533fd | 2018-03-28 16:01:06 +0200 | [diff] [blame] | 711 | userGroupService.addGroupMember(username, userGroup, |
| margaretha | b874ef5 | 2018-01-23 20:26:31 +0100 | [diff] [blame] | 712 | "system", GroupMemberStatus.ACTIVE); |
| margaretha | 0e1fc55 | 2019-08-08 15:31:01 +0200 | [diff] [blame] | 713 | // member roles are not set (not necessary) |
| margaretha | b874ef5 | 2018-01-23 20:26:31 +0100 | [diff] [blame] | 714 | } |
| 715 | catch (KustvaktException e) { |
| 716 | // member exists |
| 717 | // skip adding user to hidden group |
| margaretha | e8ab51d | 2018-01-16 19:27:40 +0100 | [diff] [blame] | 718 | } |
| 719 | } |
| margaretha | 4457383 | 2018-03-21 16:59:59 +0100 | [diff] [blame] | 720 | // else VirtualCorpusType.SYSTEM |
| margaretha | e8ab51d | 2018-01-16 19:27:40 +0100 | [diff] [blame] | 721 | } |
| margaretha | e8ab51d | 2018-01-16 19:27:40 +0100 | [diff] [blame] | 722 | } |
| margaretha | 2c24991 | 2018-01-17 20:07:20 +0100 | [diff] [blame] | 723 | |
| margaretha | b097fb0 | 2021-02-22 19:28:33 +0100 | [diff] [blame] | 724 | private boolean hasAccess (String username, int queryId) |
| margaretha | 2c24991 | 2018-01-17 20:07:20 +0100 | [diff] [blame] | 725 | throws KustvaktException { |
| 726 | UserGroup userGroup; |
| margaretha | 35e1ca2 | 2023-11-16 22:00:01 +0100 | [diff] [blame^] | 727 | List<QueryAccess> accessList = accessDao |
| 728 | .retrieveActiveAccessByQuery(queryId); |
| margaretha | b097fb0 | 2021-02-22 19:28:33 +0100 | [diff] [blame] | 729 | for (QueryAccess access : accessList) { |
| margaretha | 2c24991 | 2018-01-17 20:07:20 +0100 | [diff] [blame] | 730 | userGroup = access.getUserGroup(); |
| 731 | if (userGroupService.isMember(username, userGroup)) { |
| 732 | return true; |
| 733 | } |
| 734 | } |
| 735 | return false; |
| 736 | } |
| margaretha | f093afb | 2017-11-12 21:38:23 +0100 | [diff] [blame] | 737 | } |