diff --git a/src/main/java/org/highmed/numportal/domain/dto/QueryDto.java b/src/main/java/org/highmed/numportal/domain/dto/QueryDto.java new file mode 100644 index 00000000..362a94e2 --- /dev/null +++ b/src/main/java/org/highmed/numportal/domain/dto/QueryDto.java @@ -0,0 +1,13 @@ +package org.highmed.numportal.domain.dto; + +import io.swagger.v3.oas.annotations.media.Schema; +import jakarta.validation.constraints.NotNull; +import lombok.Data; + +@Data +@Schema +public class QueryDto { + + @NotNull private String aql; + +} \ No newline at end of file diff --git a/src/main/java/org/highmed/numportal/properties/FeatureProperties.java b/src/main/java/org/highmed/numportal/properties/FeatureProperties.java index 1cb9144b..a431579e 100644 --- a/src/main/java/org/highmed/numportal/properties/FeatureProperties.java +++ b/src/main/java/org/highmed/numportal/properties/FeatureProperties.java @@ -3,7 +3,11 @@ import lombok.Data; import org.springframework.boot.context.properties.ConfigurationProperties; + @Data @ConfigurationProperties(prefix = "feature") public class FeatureProperties { + + private boolean searchWithAql = false; + } diff --git a/src/main/java/org/highmed/numportal/service/ProjectService.java b/src/main/java/org/highmed/numportal/service/ProjectService.java index 7fb87368..98f6866f 100644 --- a/src/main/java/org/highmed/numportal/service/ProjectService.java +++ b/src/main/java/org/highmed/numportal/service/ProjectService.java @@ -2,7 +2,6 @@ import org.highmed.numportal.attachment.domain.dto.LightAttachmentDto; import org.highmed.numportal.attachment.service.AttachmentService; -import org.highmed.numportal.domain.dto.CohortDto; import org.highmed.numportal.domain.dto.Language; import org.highmed.numportal.domain.dto.ProjectDto; import org.highmed.numportal.domain.dto.ProjectInfoDto; @@ -10,7 +9,6 @@ import org.highmed.numportal.domain.dto.TemplateInfoDto; import org.highmed.numportal.domain.dto.UserDetailsDto; import org.highmed.numportal.domain.dto.ZarsInfoDto; -import org.highmed.numportal.domain.model.Cohort; import org.highmed.numportal.domain.model.ExportType; import org.highmed.numportal.domain.model.Organization; import org.highmed.numportal.domain.model.Project; @@ -23,15 +21,12 @@ import org.highmed.numportal.domain.repository.ProjectTransitionRepository; import org.highmed.numportal.domain.specification.ProjectSpecification; import org.highmed.numportal.mapper.ProjectMapper; -import org.highmed.numportal.properties.ConsentProperties; -import org.highmed.numportal.properties.PrivacyProperties; import org.highmed.numportal.service.atna.AtnaService; import org.highmed.numportal.service.ehrbase.EhrBaseService; import org.highmed.numportal.service.ehrbase.ResponseFilter; import org.highmed.numportal.service.email.ZarsService; import org.highmed.numportal.service.exception.BadRequestException; import org.highmed.numportal.service.exception.ForbiddenException; -import org.highmed.numportal.service.exception.PrivacyException; import org.highmed.numportal.service.exception.ResourceNotFound; import org.highmed.numportal.service.exception.SystemException; import org.highmed.numportal.service.executors.CohortQueryLister; @@ -43,11 +38,9 @@ import org.highmed.numportal.service.notification.dto.ProjectStartNotification; import org.highmed.numportal.service.notification.dto.ProjectStatusChangeNotification; import org.highmed.numportal.service.notification.dto.ProjectStatusChangeRequestNotification; -import org.highmed.numportal.service.policy.EhrPolicy; -import org.highmed.numportal.service.policy.EuropeanConsentPolicy; import org.highmed.numportal.service.policy.Policy; import org.highmed.numportal.service.policy.ProjectPolicyService; -import org.highmed.numportal.service.policy.TemplatesPolicy; +import org.highmed.numportal.service.util.ExportUtil; import com.fasterxml.jackson.core.JsonProcessingException; import com.fasterxml.jackson.databind.ObjectMapper; @@ -55,8 +48,6 @@ import lombok.AllArgsConstructor; import lombok.extern.slf4j.Slf4j; import org.apache.commons.collections.CollectionUtils; -import org.apache.commons.csv.CSVFormat; -import org.apache.commons.csv.CSVPrinter; import org.apache.commons.lang3.BooleanUtils; import org.apache.commons.lang3.StringUtils; import org.ehrbase.openehr.sdk.aql.dto.AqlQuery; @@ -68,25 +59,15 @@ import org.springframework.data.domain.PageRequest; import org.springframework.data.domain.Pageable; import org.springframework.data.domain.Sort; -import org.springframework.http.HttpHeaders; -import org.springframework.http.MediaType; import org.springframework.lang.Nullable; import org.springframework.stereotype.Service; import org.springframework.transaction.annotation.Transactional; -import org.springframework.util.LinkedMultiValueMap; -import org.springframework.util.MultiValueMap; import org.springframework.web.multipart.MultipartFile; import org.springframework.web.servlet.mvc.method.annotation.StreamingResponseBody; import java.io.IOException; -import java.io.OutputStream; -import java.io.OutputStreamWriter; -import java.nio.charset.StandardCharsets; -import java.time.LocalDate; -import java.time.LocalDateTime; import java.time.OffsetDateTime; import java.time.format.DateTimeFormatter; -import java.time.temporal.ChronoUnit; import java.util.ArrayList; import java.util.Arrays; import java.util.Collections; @@ -100,8 +81,6 @@ import java.util.Optional; import java.util.Set; import java.util.stream.Collectors; -import java.util.zip.ZipEntry; -import java.util.zip.ZipOutputStream; import static java.util.Objects.nonNull; import static org.highmed.numportal.domain.templates.ExceptionsTemplate.AN_ISSUE_HAS_OCCURRED_CANNOT_EXECUTE_AQL; @@ -113,10 +92,7 @@ import static org.highmed.numportal.domain.templates.ExceptionsTemplate.CANNOT_DELETE_PROJECT_INVALID_STATUS; import static org.highmed.numportal.domain.templates.ExceptionsTemplate.CANNOT_UPDATE_PROJECT_INVALID_PROJECT_STATUS; import static org.highmed.numportal.domain.templates.ExceptionsTemplate.DATA_EXPLORER_AVAILABLE_FOR_PUBLISHED_PROJECTS_ONLY; -import static org.highmed.numportal.domain.templates.ExceptionsTemplate.ERROR_CREATING_A_ZIP_FILE_FOR_DATA_EXPORT; import static org.highmed.numportal.domain.templates.ExceptionsTemplate.ERROR_CREATING_THE_PROJECT_PDF; -import static org.highmed.numportal.domain.templates.ExceptionsTemplate.ERROR_WHILE_CREATING_THE_CSV_FILE; -import static org.highmed.numportal.domain.templates.ExceptionsTemplate.ERROR_WHILE_RETRIEVING_DATA; import static org.highmed.numportal.domain.templates.ExceptionsTemplate.INVALID_PROJECT_STATUS; import static org.highmed.numportal.domain.templates.ExceptionsTemplate.INVALID_PROJECT_STATUS_PARAM; import static org.highmed.numportal.domain.templates.ExceptionsTemplate.MORE_THAN_ONE_TRANSITION_FROM_PUBLISHED_TO_CLOSED_FOR_PROJECT; @@ -128,17 +104,12 @@ import static org.highmed.numportal.domain.templates.ExceptionsTemplate.PROJECT_TEMPLATES_CANNOT_BE_NULL; import static org.highmed.numportal.domain.templates.ExceptionsTemplate.RESEARCHER_NOT_APPROVED; import static org.highmed.numportal.domain.templates.ExceptionsTemplate.RESEARCHER_NOT_FOUND; -import static org.highmed.numportal.domain.templates.ExceptionsTemplate.RESULTS_WITHHELD_FOR_PRIVACY_REASONS; @Service @Slf4j @AllArgsConstructor public class ProjectService { - private static final String ZIP_FILE_ENDING = ".zip"; - private static final String JSON_FILE_ENDING = ".json"; - private static final String ZIP_MEDIA_TYPE = "application/zip"; - private static final String CSV_FILE_PATTERN = "%s_%s.csv"; private static final String AUTHOR_NAME = "author"; private static final String ORGANIZATION_NAME = "organization"; private static final String PROJECT_NAME = "name"; @@ -172,14 +143,10 @@ public class ProjectService { private final CohortService cohortService; - private final ConsentProperties consentProperties; + private final ExportUtil exportUtil; private final ResponseFilter responseFilter; - private final PrivacyProperties privacyProperties; - - private final TemplateService templateService; - private final ProjectDocCreator projectDocCreator; private final ProjectMapper projectMapper; @@ -271,7 +238,7 @@ public String retrieveData(String query, Long projectId, String userId, Boolean List responseData; if (BooleanUtils.isTrue(defaultConfiguration)) { responseData = - executeDefaultConfiguration(projectId, project.getCohort(), project.getTemplates()); + exportUtil.executeDefaultConfiguration(projectId, project.getCohort(), project.getTemplates()); } else { responseData = executeCustomConfiguration(query, projectId, userId); } @@ -283,55 +250,12 @@ public String retrieveData(String query, Long projectId, String userId, Boolean } } - private List executeDefaultConfiguration(Long projectId, Cohort cohort, Map templates) { - - if (templates == null || templates.isEmpty()) { - return List.of(); - } - - Set ehrIds = cohortService.executeCohort(cohort, false); - - if (ehrIds.size() < privacyProperties.getMinHits()) { - log.warn(RESULTS_WITHHELD_FOR_PRIVACY_REASONS); - throw new PrivacyException(ProjectService.class, RESULTS_WITHHELD_FOR_PRIVACY_REASONS); - } - - List response = new LinkedList<>(); - - templates.forEach( - (templateId, v) -> - response.addAll(retrieveTemplateData(ehrIds, templateId, projectId, false))); - return responseFilter.filterResponse(response); - } private List executeCustomConfiguration(String query, Long projectId, String userId) { List response = executeAql(query, projectId, userId); return responseFilter.filterResponse(response); } - private List retrieveTemplateData( - Set ehrIds, String templateId, Long projectId, Boolean usedOutsideEu) { - try { - AqlQuery aql = templateService.createSelectCompositionQuery(templateId); - - List policies = - collectProjectPolicies(ehrIds, Map.of(templateId, templateId), usedOutsideEu); - projectPolicyService.apply(aql, policies); - - List response = ehrBaseService.executeRawQuery(aql, projectId); - response.forEach(data -> data.setName(templateId)); - return response; - - } catch (ResourceNotFound e) { - log.error("Could not retrieve data for template {} and project {}. Failed with message {} ", templateId, projectId, e.getMessage(), e); - log.error(e.getMessage(), e); - } catch (Exception e) { - log.error(e.getMessage(), e); - } - QueryResponseData response = new QueryResponseData(); - response.setName(templateId); - return List.of(response); - } public List executeAql(String query, Long projectId, String userId) { List queryResponseData; @@ -347,7 +271,7 @@ public List executeAql(String query, Long projectId, String u AqlQuery aql = AqlQueryParser.parse(query); List policies = - collectProjectPolicies(ehrIds, project.getTemplates(), project.isUsedOutsideEu()); + exportUtil.collectProjectPolicies(ehrIds, project.getTemplates(), project.isUsedOutsideEu()); projectPolicyService.apply(aql, policies); queryResponseData = ehrBaseService.executeRawQuery(aql, projectId); @@ -360,80 +284,6 @@ public List executeAql(String query, Long projectId, String u return queryResponseData; } - public String executeManagerProject(CohortDto cohortDto, List templates, String userId) { - var queryResponse = StringUtils.EMPTY; - var project = createManagerProject(); - try { - userDetailsService.checkIsUserApproved(userId); - var templateMap = CollectionUtils.isNotEmpty(templates) ? templates.stream().collect(Collectors.toMap(k -> k, v -> v)) : Collections.emptyMap(); - - List responseData = - executeDefaultConfiguration( - project.getId(), cohortService.toCohort(cohortDto), (Map) templateMap); - - queryResponse = mapper.writeValueAsString(responseData); - - } catch (Exception e) { - atnaService.logDataExport(userId, project.getId(), project, false); - throw new SystemException(ProjectService.class, ERROR_WHILE_RETRIEVING_DATA, - String.format(ERROR_WHILE_RETRIEVING_DATA, e.getLocalizedMessage())); - } - atnaService.logDataExport(userId, project.getId(), project, true); - return queryResponse; - } - - public boolean streamResponseAsZip( - List queryResponseDataList, - String filenameStart, - OutputStream outputStream) { - - try (var zipOutputStream = new ZipOutputStream(outputStream, StandardCharsets.UTF_8)) { - - var index = 0; - for (QueryResponseData queryResponseData : queryResponseDataList) { - - String responseName = queryResponseData.getName(); - if (StringUtils.isEmpty(responseName)) { - responseName = String.valueOf(index); - } - zipOutputStream.putNextEntry( - new ZipEntry(String.format(CSV_FILE_PATTERN, filenameStart, responseName))); - addResponseAsCsv(zipOutputStream, queryResponseData); - zipOutputStream.closeEntry(); - index++; - } - } catch (IOException e) { - log.error("Error creating a zip file for data export.", e); - throw new SystemException(ProjectService.class, ERROR_CREATING_A_ZIP_FILE_FOR_DATA_EXPORT, - String.format(ERROR_CREATING_A_ZIP_FILE_FOR_DATA_EXPORT, e.getLocalizedMessage())); - } - return true; - } - - private void addResponseAsCsv(ZipOutputStream zipOutputStream, QueryResponseData queryResponseData) { - List paths = new ArrayList<>(); - - for (Map column : queryResponseData.getColumns()) { - paths.add(column.get("path")); - } - CSVPrinter printer; - try { - printer = - CSVFormat.EXCEL.builder() - .setHeader(paths.toArray(new String[]{})) - .build() - .print(new OutputStreamWriter(zipOutputStream, StandardCharsets.UTF_8)); - - for (List row : queryResponseData.getRows()) { - printer.printRecord(row); - } - printer.flush(); - } catch (IOException e) { - throw new SystemException(ProjectService.class, ERROR_WHILE_CREATING_THE_CSV_FILE, - String.format(ERROR_WHILE_CREATING_THE_CSV_FILE, e.getMessage())); - } - } - public StreamingResponseBody getExportResponseBody( String query, Long projectId, @@ -447,70 +297,18 @@ public StreamingResponseBody getExportResponseBody( if (BooleanUtils.isTrue(defaultConfiguration)) { response = - executeDefaultConfiguration(projectId, project.getCohort(), project.getTemplates()); + exportUtil.executeDefaultConfiguration(projectId, project.getCohort(), project.getTemplates()); } else { response = executeCustomConfiguration(query, projectId, userId); } if (format == ExportType.json) { - return exportJson(response); - } else { - return exportCsv(response, projectId); - } - } - - public StreamingResponseBody getManagerExportResponseBody(CohortDto cohortDto, List templates, String userId, ExportType format) { - userDetailsService.checkIsUserApproved(userId); - var project = createManagerProject(); - - var templateMap = templates.stream().collect(Collectors.toMap(k -> k, v -> v)); - - List response = - executeDefaultConfiguration( - project.getId(), cohortService.toCohort(cohortDto), templateMap); - - if (format == ExportType.json) { - return exportJson(response); + return exportUtil.exportJson(response); } else { - return exportCsv(response, project.getId()); + return exportUtil.exportCsv(response, projectId); } } - private StreamingResponseBody exportCsv(List response, Long projectId) { - return outputStream -> - streamResponseAsZip(response, getExportFilenameBody(projectId), outputStream); - } - - private StreamingResponseBody exportJson(List response) { - String json; - try { - json = mapper.writeValueAsString(response); - } catch (JsonProcessingException e) { - throw new SystemException(ProjectService.class, AN_ISSUE_HAS_OCCURRED_CANNOT_EXECUTE_AQL); - } - return outputStream -> { - outputStream.write(json.getBytes()); - outputStream.flush(); - outputStream.close(); - }; - } - - public MultiValueMap getExportHeaders(ExportType format, Long projectId) { - MultiValueMap headers = new LinkedMultiValueMap<>(); - String fileEnding; - if (format == ExportType.json) { - fileEnding = JSON_FILE_ENDING; - headers.add(HttpHeaders.CONTENT_TYPE, MediaType.APPLICATION_JSON_VALUE); - } else { - fileEnding = ZIP_FILE_ENDING; - headers.add(HttpHeaders.CONTENT_TYPE, ZIP_MEDIA_TYPE); - } - headers.add( - HttpHeaders.CONTENT_DISPOSITION, - "attachment; filename=" + getExportFilenameBody(projectId) + fileEnding); - return headers; - } - public Optional getProjectById(String loggedInUserId, Long projectId) { userDetailsService.checkIsUserApproved(loggedInUserId); return projectRepository.findById(projectId); @@ -771,22 +569,6 @@ private void deleteAttachments(Set attachmentsToBeDeleted, List ro } } - private List collectProjectPolicies( - Set ehrIds, Map templates, boolean usedOutsideEu) { - List policies = new LinkedList<>(); - policies.add(EhrPolicy.builder().cohortEhrIds(ehrIds).build()); - policies.add(TemplatesPolicy.builder().templatesMap(templates).build()); - - if (usedOutsideEu) { - policies.add( - EuropeanConsentPolicy.builder() - .oid(consentProperties.getAllowUsageOutsideEuOid()) - .build()); - } - - return policies; - } - private List collectNotifications(Project project, ProjectStatus newStatus, ProjectStatus oldStatus, String coordinatorUserId, @@ -1054,16 +836,6 @@ private void sortProjects(List projects, Sort sortBy) { } } - public String getExportFilenameBody(Long projectId) { - return String.format( - "Project_%d_%s", - projectId, - LocalDateTime.now() - .truncatedTo(ChronoUnit.MINUTES) - .format(DateTimeFormatter.ISO_LOCAL_DATE)) - .replace('-', '_'); - } - private void setTemplates(Project project, ProjectDto projectDto) { if (projectDto.getTemplates() != null) { Map map = @@ -1296,24 +1068,6 @@ private Project validateAndRetrieveProject(Long projectId, String userId) { return project; } - private Project createManagerProject() { - var undef = "undef"; - return Project.builder() - .id(0L) - .name("Manager data retrieval project") - .createDate(OffsetDateTime.now()) - .startDate(LocalDate.now()) - .description("Adhoc temp project for manager data retrieval") - .goal(undef) - .usedOutsideEu(false) - .firstHypotheses(undef) - .secondHypotheses(undef) - .description("Temporary project for manager data retrieval") - .coordinator(UserDetails.builder().userId(undef).organization(Organization.builder().id(0L).build()).build()) - .status(ProjectStatus.DENIED) - .build(); - } - public byte[] getInfoDocBytes(Long id, String userId, Locale locale) { userDetailsService.checkIsUserApproved(userId); Project project = diff --git a/src/main/java/org/highmed/numportal/service/util/ExportUtil.java b/src/main/java/org/highmed/numportal/service/util/ExportUtil.java new file mode 100644 index 00000000..bdfbba14 --- /dev/null +++ b/src/main/java/org/highmed/numportal/service/util/ExportUtil.java @@ -0,0 +1,238 @@ +package org.highmed.numportal.service.util; + +import org.highmed.numportal.domain.model.Cohort; +import org.highmed.numportal.domain.model.ExportType; +import org.highmed.numportal.properties.ConsentProperties; +import org.highmed.numportal.properties.PrivacyProperties; +import org.highmed.numportal.service.CohortService; +import org.highmed.numportal.service.ProjectService; +import org.highmed.numportal.service.TemplateService; +import org.highmed.numportal.service.ehrbase.EhrBaseService; +import org.highmed.numportal.service.ehrbase.ResponseFilter; +import org.highmed.numportal.service.exception.PrivacyException; +import org.highmed.numportal.service.exception.ResourceNotFound; +import org.highmed.numportal.service.exception.SystemException; +import org.highmed.numportal.service.policy.EhrPolicy; +import org.highmed.numportal.service.policy.EuropeanConsentPolicy; +import org.highmed.numportal.service.policy.Policy; +import org.highmed.numportal.service.policy.ProjectPolicyService; +import org.highmed.numportal.service.policy.TemplatesPolicy; + +import com.fasterxml.jackson.core.JsonProcessingException; +import com.fasterxml.jackson.databind.ObjectMapper; +import lombok.AllArgsConstructor; +import lombok.extern.slf4j.Slf4j; +import org.apache.commons.csv.CSVFormat; +import org.apache.commons.csv.CSVPrinter; +import org.apache.commons.lang3.StringUtils; +import org.ehrbase.openehr.sdk.aql.dto.AqlQuery; +import org.ehrbase.openehr.sdk.response.dto.QueryResponseData; +import org.springframework.http.HttpHeaders; +import org.springframework.http.MediaType; +import org.springframework.util.LinkedMultiValueMap; +import org.springframework.util.MultiValueMap; +import org.springframework.web.servlet.mvc.method.annotation.StreamingResponseBody; + +import java.io.IOException; +import java.io.OutputStream; +import java.io.OutputStreamWriter; +import java.nio.charset.StandardCharsets; +import java.time.LocalDateTime; +import java.time.format.DateTimeFormatter; +import java.time.temporal.ChronoUnit; +import java.util.ArrayList; +import java.util.LinkedList; +import java.util.List; +import java.util.Map; +import java.util.Set; +import java.util.zip.ZipEntry; +import java.util.zip.ZipOutputStream; + +import static org.highmed.numportal.domain.templates.ExceptionsTemplate.AN_ISSUE_HAS_OCCURRED_CANNOT_EXECUTE_AQL; +import static org.highmed.numportal.domain.templates.ExceptionsTemplate.ERROR_CREATING_A_ZIP_FILE_FOR_DATA_EXPORT; +import static org.highmed.numportal.domain.templates.ExceptionsTemplate.ERROR_WHILE_CREATING_THE_CSV_FILE; +import static org.highmed.numportal.domain.templates.ExceptionsTemplate.RESULTS_WITHHELD_FOR_PRIVACY_REASONS; + +@Slf4j +@AllArgsConstructor +public class ExportUtil { + + private static final String CSV_FILE_PATTERN = "%s_%s.csv"; + private static final String ZIP_FILE_ENDING = ".zip"; + private static final String JSON_FILE_ENDING = ".json"; + private static final String ZIP_MEDIA_TYPE = "application/zip"; + + private final CohortService cohortService; + + private final TemplateService templateService; + + private final EhrBaseService ehrBaseService; + + private final ResponseFilter responseFilter; + + private final PrivacyProperties privacyProperties; + + private final ConsentProperties consentProperties; + + private final ProjectPolicyService projectPolicyService; + + private final ObjectMapper mapper; + + public MultiValueMap getExportHeaders(ExportType format, Long projectId) { + MultiValueMap headers = new LinkedMultiValueMap<>(); + String fileEnding; + if (format == ExportType.json) { + fileEnding = JSON_FILE_ENDING; + headers.add(HttpHeaders.CONTENT_TYPE, MediaType.APPLICATION_JSON_VALUE); + } else { + fileEnding = ZIP_FILE_ENDING; + headers.add(HttpHeaders.CONTENT_TYPE, ZIP_MEDIA_TYPE); + } + headers.add( + HttpHeaders.CONTENT_DISPOSITION, + "attachment; filename=" + getExportFilenameBody(projectId) + fileEnding); + return headers; + } + + public String getExportFilenameBody(Long projectId) { + return String.format( + "Project_%d_%s", + projectId, + LocalDateTime.now() + .truncatedTo(ChronoUnit.MINUTES) + .format(DateTimeFormatter.ISO_LOCAL_DATE)) + .replace('-', '_'); + } + + public List executeDefaultConfiguration(Long projectId, Cohort cohort, Map templates) { + + if (templates == null || templates.isEmpty()) { + return List.of(); + } + + Set ehrIds = cohortService.executeCohort(cohort, false); + + if (ehrIds.size() < privacyProperties.getMinHits()) { + log.warn(RESULTS_WITHHELD_FOR_PRIVACY_REASONS); + throw new PrivacyException(ProjectService.class, RESULTS_WITHHELD_FOR_PRIVACY_REASONS); + } + + List response = new LinkedList<>(); + + templates.forEach( + (templateId, v) -> + response.addAll(retrieveTemplateData(ehrIds, templateId, projectId, false))); + return responseFilter.filterResponse(response); + } + + private List retrieveTemplateData( + Set ehrIds, String templateId, Long projectId, Boolean usedOutsideEu) { + try { + AqlQuery aql = templateService.createSelectCompositionQuery(templateId); + + List policies = + collectProjectPolicies(ehrIds, Map.of(templateId, templateId), usedOutsideEu); + projectPolicyService.apply(aql, policies); + + List response = ehrBaseService.executeRawQuery(aql, projectId); + response.forEach(data -> data.setName(templateId)); + return response; + + } catch (ResourceNotFound e) { + log.error("Could not retrieve data for template {} and project {}. Failed with message {} ", templateId, projectId, e.getMessage(), e); + log.error(e.getMessage(), e); + } catch (Exception e) { + log.error(e.getMessage(), e); + } + QueryResponseData response = new QueryResponseData(); + response.setName(templateId); + return List.of(response); + } + + public List collectProjectPolicies( + Set ehrIds, Map templates, boolean usedOutsideEu) { + List policies = new LinkedList<>(); + policies.add(EhrPolicy.builder().cohortEhrIds(ehrIds).build()); + policies.add(TemplatesPolicy.builder().templatesMap(templates).build()); + + if (usedOutsideEu) { + policies.add( + EuropeanConsentPolicy.builder() + .oid(consentProperties.getAllowUsageOutsideEuOid()) + .build()); + } + + return policies; + } + + public StreamingResponseBody exportJson(List response) { + String json; + try { + json = mapper.writeValueAsString(response); + } catch (JsonProcessingException e) { + throw new SystemException(ProjectService.class, AN_ISSUE_HAS_OCCURRED_CANNOT_EXECUTE_AQL); + } + return outputStream -> { + outputStream.write(json.getBytes()); + outputStream.flush(); + outputStream.close(); + }; + } + + public StreamingResponseBody exportCsv(List response, Long projectId) { + return outputStream -> + streamResponseAsZip(response, getExportFilenameBody(projectId), outputStream); + } + + public void streamResponseAsZip( + List queryResponseDataList, + String filenameStart, + OutputStream outputStream) { + + try (var zipOutputStream = new ZipOutputStream(outputStream, StandardCharsets.UTF_8)) { + + var index = 0; + for (QueryResponseData queryResponseData : queryResponseDataList) { + + String responseName = queryResponseData.getName(); + if (StringUtils.isEmpty(responseName)) { + responseName = String.valueOf(index); + } + zipOutputStream.putNextEntry( + new ZipEntry(String.format(CSV_FILE_PATTERN, filenameStart, responseName))); + addResponseAsCsv(zipOutputStream, queryResponseData); + zipOutputStream.closeEntry(); + index++; + } + } catch (IOException e) { + log.error("Error creating a zip file for data export.", e); + throw new SystemException(ProjectService.class, ERROR_CREATING_A_ZIP_FILE_FOR_DATA_EXPORT, + String.format(ERROR_CREATING_A_ZIP_FILE_FOR_DATA_EXPORT, e.getLocalizedMessage())); + } + } + + private void addResponseAsCsv(ZipOutputStream zipOutputStream, QueryResponseData queryResponseData) { + List paths = new ArrayList<>(); + + for (Map column : queryResponseData.getColumns()) { + paths.add(column.get("path")); + } + CSVPrinter printer; + try { + printer = + CSVFormat.EXCEL.builder() + .setHeader(paths.toArray(new String[]{})) + .build() + .print(new OutputStreamWriter(zipOutputStream, StandardCharsets.UTF_8)); + + for (List row : queryResponseData.getRows()) { + printer.printRecord(row); + } + printer.flush(); + } catch (IOException e) { + throw new SystemException(ProjectService.class, ERROR_WHILE_CREATING_THE_CSV_FILE, + String.format(ERROR_WHILE_CREATING_THE_CSV_FILE, e.getMessage())); + } + } + +} diff --git a/src/main/java/org/highmed/numportal/web/controller/ManagerController.java b/src/main/java/org/highmed/numportal/web/controller/ManagerController.java new file mode 100644 index 00000000..f224fe56 --- /dev/null +++ b/src/main/java/org/highmed/numportal/web/controller/ManagerController.java @@ -0,0 +1,86 @@ +package org.highmed.numportal.web.controller; + +import org.highmed.numportal.domain.dto.ManagerProjectDto; +import org.highmed.numportal.domain.dto.QueryDto; +import org.highmed.numportal.domain.model.ExportType; +import org.highmed.numportal.service.ManagerService; +import org.highmed.numportal.service.ehrbase.EhrBaseService; +import org.highmed.numportal.service.logger.ContextLog; +import org.highmed.numportal.service.util.ExportUtil; +import org.highmed.numportal.web.config.Role; + +import io.swagger.v3.oas.annotations.Operation; +import io.swagger.v3.oas.annotations.Parameter; +import io.swagger.v3.oas.annotations.security.SecurityRequirement; +import jakarta.validation.Valid; +import jakarta.validation.constraints.NotNull; +import lombok.AllArgsConstructor; +import org.ehrbase.openehr.sdk.response.dto.QueryResponseData; +import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty; +import org.springframework.http.HttpStatus; +import org.springframework.http.ResponseEntity; +import org.springframework.security.access.prepost.PreAuthorize; +import org.springframework.security.core.annotation.AuthenticationPrincipal; +import org.springframework.security.oauth2.jwt.Jwt; +import org.springframework.util.MultiValueMap; +import org.springframework.web.bind.annotation.PostMapping; +import org.springframework.web.bind.annotation.RequestBody; +import org.springframework.web.bind.annotation.RequestMapping; +import org.springframework.web.bind.annotation.RequestParam; +import org.springframework.web.bind.annotation.RestController; +import org.springframework.web.servlet.mvc.method.annotation.StreamingResponseBody; + +@RestController +@AllArgsConstructor +@RequestMapping(value = "/query", produces = "application/json") +@SecurityRequirement(name = "security_auth") +@ConditionalOnProperty(value = "feature.search-with-aql", havingValue = "true") +public class ManagerController { + + private final EhrBaseService ehrBaseService; + private final ManagerService managerService; + private final ExportUtil exportUtil; + + @ContextLog(type = "Manager", description = "Execute AQL queries") + @PostMapping("execute") + @Operation(description = "Executes an AQL query") + @PreAuthorize(Role.MANAGER) + public ResponseEntity executeManagerQuery( + @RequestBody @Valid QueryDto queryDto) { + return ResponseEntity.ok( + ehrBaseService.executePlainQuery(queryDto.getAql()) + ); + } + + @PostMapping("/manager/execute") + @Operation( + description = "Executes the manager project aql in the cohort returning medical data matching the templates") + @PreAuthorize(Role.MANAGER) + public ResponseEntity executeManagerProject( + @AuthenticationPrincipal @NotNull Jwt principal, + @RequestBody @Valid ManagerProjectDto managerProjectDto) { + return ResponseEntity.ok( + managerService.executeManagerProject( + managerProjectDto.getCohort(), + managerProjectDto.getTemplates(), + principal.getSubject())); + } + + @PostMapping(value = "/manager/export") + @Operation(description = "Executes the cohort default configuration returns the result as a csv file attachment") + @PreAuthorize(Role.MANAGER) + public ResponseEntity exportManagerResults( + @AuthenticationPrincipal @NotNull Jwt principal, + @RequestBody @Valid ManagerProjectDto managerProjectDto, + @RequestParam(required = false) + @Parameter(description = "A string defining the output format. Valid values are 'csv' and 'json'. Default is csv.") + ExportType format) { + StreamingResponseBody streamingResponseBody = + managerService.getManagerExportResponseBody( + managerProjectDto.getCohort(), managerProjectDto.getTemplates(), principal.getSubject(), + format); + MultiValueMap headers = exportUtil.getExportHeaders(format, 0L); + + return new ResponseEntity<>(streamingResponseBody, headers, HttpStatus.OK); + } +} diff --git a/src/main/java/org/highmed/numportal/web/controller/ProjectController.java b/src/main/java/org/highmed/numportal/web/controller/ProjectController.java index 6b9afeef..87216cd9 100644 --- a/src/main/java/org/highmed/numportal/web/controller/ProjectController.java +++ b/src/main/java/org/highmed/numportal/web/controller/ProjectController.java @@ -1,7 +1,6 @@ package org.highmed.numportal.web.controller; import org.highmed.numportal.domain.dto.CommentDto; -import org.highmed.numportal.domain.dto.ManagerProjectDto; import org.highmed.numportal.domain.dto.ProjectDto; import org.highmed.numportal.domain.dto.ProjectViewDto; import org.highmed.numportal.domain.dto.RawQueryDto; @@ -18,6 +17,7 @@ import org.highmed.numportal.service.exception.CustomizedExceptionHandler; import org.highmed.numportal.service.exception.ResourceNotFound; import org.highmed.numportal.service.logger.ContextLog; +import org.highmed.numportal.service.util.ExportUtil; import org.highmed.numportal.web.config.Role; import io.swagger.v3.oas.annotations.Operation; @@ -69,6 +69,7 @@ public class ProjectController extends CustomizedExceptionHandler { private final CommentService commentService; private final ProjectMapper projectMapper; private final CommentMapper commentMapper; + private final ExportUtil exportUtil; private final ProjectViewMapper projectViewMapper; @@ -180,19 +181,6 @@ public ResponseEntity executeAql( query.getQuery(), projectId, principal.getSubject(), defaultConfiguration)); } - @PostMapping("/manager/execute") - @Operation( - description = "Executes the manager project aql in the cohort returning medical data matching the templates") - @PreAuthorize(Role.MANAGER) - public ResponseEntity executeManagerProject( - @AuthenticationPrincipal @NotNull Jwt principal, - @RequestBody @Valid ManagerProjectDto managerProjectDto) { - return ResponseEntity.ok( - projectService.executeManagerProject( - managerProjectDto.getCohort(), - managerProjectDto.getTemplates(), - principal.getSubject())); - } @PostMapping(value = "/{projectId}/export") @Operation(description = "Executes the aql and returns the result as a csv file attachment") @@ -208,25 +196,7 @@ public ResponseEntity exportResults( StreamingResponseBody streamingResponseBody = projectService.getExportResponseBody( query.getQuery(), projectId, principal.getSubject(), format, defaultConfiguration); - MultiValueMap headers = projectService.getExportHeaders(format, projectId); - - return new ResponseEntity<>(streamingResponseBody, headers, HttpStatus.OK); - } - - @PostMapping(value = "/manager/export") - @Operation(description = "Executes the cohort default configuration returns the result as a csv file attachment") - @PreAuthorize(Role.MANAGER) - public ResponseEntity exportManagerResults( - @AuthenticationPrincipal @NotNull Jwt principal, - @RequestBody @Valid ManagerProjectDto managerProjectDto, - @RequestParam(required = false) - @Parameter(description = "A string defining the output format. Valid values are 'csv' and 'json'. Default is csv.") - ExportType format) { - StreamingResponseBody streamingResponseBody = - projectService.getManagerExportResponseBody( - managerProjectDto.getCohort(), managerProjectDto.getTemplates(), principal.getSubject(), - format); - MultiValueMap headers = projectService.getExportHeaders(format, 0L); + MultiValueMap headers = exportUtil.getExportHeaders(format, projectId); return new ResponseEntity<>(streamingResponseBody, headers, HttpStatus.OK); } diff --git a/src/main/resources/application-local.yml b/src/main/resources/application-local.yml index a3075bdf..eba0048b 100644 --- a/src/main/resources/application-local.yml +++ b/src/main/resources/application-local.yml @@ -31,7 +31,7 @@ spring: userStoreClient: authorization-grant-type: client_credentials client-id: num-portal - client-secret: ZO29JPc5OLIZ29Rv83tEHOdxiAt2P9eC + client-secret: Tp3MiedEuSaLEIxaOhYHsA0bOHQZ7G2C provider: userStoreClient: token-uri: ${keycloak.url}/realms/crr/protocol/openid-connect/token @@ -107,3 +107,4 @@ user-service: delete-users-cron: 0 0 5 * * * feature: + search-with-aql: false \ No newline at end of file diff --git a/src/main/resources/application.yml b/src/main/resources/application.yml index 5e1c11e1..01017a0e 100644 --- a/src/main/resources/application.yml +++ b/src/main/resources/application.yml @@ -205,4 +205,7 @@ pseudonymity: fake3rdPartyPseudonymEnabled: false user-service: - delete-users-cron: 0 0 5 * * * \ No newline at end of file + delete-users-cron: 0 0 5 * * * + +feature: + search-with-aql: false \ No newline at end of file diff --git a/src/test/java/org/highmed/numportal/service/ManagerServiceTest.java b/src/test/java/org/highmed/numportal/service/ManagerServiceTest.java new file mode 100644 index 00000000..d8377fc3 --- /dev/null +++ b/src/test/java/org/highmed/numportal/service/ManagerServiceTest.java @@ -0,0 +1,119 @@ +package org.highmed.numportal.service; + +import org.highmed.numportal.domain.dto.CohortDto; +import org.highmed.numportal.domain.model.ExportType; +import org.highmed.numportal.domain.model.admin.UserDetails; +import org.highmed.numportal.service.exception.SystemException; +import org.highmed.numportal.service.util.ExportUtil; + +import com.fasterxml.jackson.core.JsonProcessingException; +import com.fasterxml.jackson.databind.ObjectMapper; +import org.ehrbase.openehr.sdk.response.dto.QueryResponseData; +import org.junit.Assert; +import org.junit.Test; +import org.junit.runner.RunWith; +import org.mockito.InjectMocks; +import org.mockito.Mock; +import org.mockito.Mockito; +import org.mockito.Spy; +import org.mockito.junit.MockitoJUnitRunner; + +import java.io.ByteArrayInputStream; +import java.io.ByteArrayOutputStream; +import java.io.IOException; +import java.util.ArrayList; +import java.util.Arrays; +import java.util.Collections; +import java.util.List; +import java.util.Map; +import java.util.zip.ZipEntry; +import java.util.zip.ZipInputStream; + +import static org.hamcrest.MatcherAssert.assertThat; +import static org.hamcrest.core.Is.is; +import static org.mockito.ArgumentMatchers.any; +import static org.mockito.Mockito.when; + +@RunWith(MockitoJUnitRunner.class) +public class ManagerServiceTest { + + private static final String CORONA_TEMPLATE = "Corona_Anamnese"; + + @Mock + private CohortService cohortService; + + @InjectMocks + private ExportUtil exportUtil; + + @InjectMocks + private ManagerService managerService; + + @Spy + private ObjectMapper mapper; + + @Test(expected = SystemException.class) + public void executeManagerProjectSystemException() throws JsonProcessingException { + CohortDto cohortDto = CohortDto.builder().name("Cohort name").id(2L).build(); + when(mapper.writeValueAsString(any(Object.class))).thenThrow(new JsonProcessingException("Error"){}); + managerService.executeManagerProject(cohortDto, Arrays.asList("1", "2"), "ownerCoordinatorId"); + } + + @Test + public void shouldSuccessfullyExecuteManagerProject() { + CohortDto cohortDto = CohortDto.builder().name("Cohort name").id(2L).build(); + + UserDetails userDetails = + UserDetails.builder().userId("approvedCoordinatorId").approved(true).build(); + + String result = + managerService.executeManagerProject( + cohortDto, List.of(CORONA_TEMPLATE), userDetails.getUserId()); + + assertThat(result, is("[{\"name\":\"Corona_Anamnese\",\"columns\":null,\"rows\":null}]")); + } + + @Test + public void shouldHandleExecuteManagerProjectWithEmptyTemplates() { + executeManagerProjectWithoutTemplates(Collections.EMPTY_LIST); + } + + @Test + public void shouldHandleExecuteManagerProjectWithNullTemplates() { + executeManagerProjectWithoutTemplates(null); + } + + private void executeManagerProjectWithoutTemplates(List templates) { + CohortDto cohortDto = CohortDto.builder().name("Cohort name").id(2L).build(); + UserDetails userDetails = + UserDetails.builder().userId("approvedCoordinatorId").approved(true).build(); + String result = + managerService.executeManagerProject( + cohortDto, templates, userDetails.getUserId()); + assertThat(result, is("[]")); + } + + @Test + public void streamResponseBody() throws IOException { + QueryResponseData response = new QueryResponseData(); + response.setName("response-one"); + response.setColumns(new ArrayList<>(List.of(Map.of("path", "/ehr_id/value"), Map.of("uuid", "c/uuid")))); + response.setRows( List.of( + new ArrayList<>(List.of("ehr-id-1", Map.of("_type", "OBSERVATION", "uuid", "12345"))), + new ArrayList<>(List.of("ehr-id-2", Map.of("_type", "SECTION", "uuid", "bla"))))); + ByteArrayOutputStream out = new ByteArrayOutputStream(); + exportUtil.streamResponseAsZip(List.of(response), "testFile", out); + + ZipInputStream zipInputStream = new ZipInputStream(new ByteArrayInputStream(out.toByteArray())); + ZipEntry expectedFile = zipInputStream.getNextEntry(); + Assert.assertEquals("testFile_response-one.csv", expectedFile.getName()); + } + + @Test + public void getManagerExportResponseBodyTest() { + CohortDto cohortDto = CohortDto.builder() + .name("alter cohort") + .projectId(2L).build(); + managerService.getManagerExportResponseBody(cohortDto, List.of("Alter"), "approvedCoordinatorId", ExportType.json); + Mockito.verify(cohortService, Mockito.times(1)).toCohort(Mockito.any(CohortDto.class)); + } +} diff --git a/src/test/java/org/highmed/numportal/service/ProjectServiceTest.java b/src/test/java/org/highmed/numportal/service/ProjectServiceTest.java index b16af735..db7aa89a 100644 --- a/src/test/java/org/highmed/numportal/service/ProjectServiceTest.java +++ b/src/test/java/org/highmed/numportal/service/ProjectServiceTest.java @@ -32,6 +32,8 @@ import org.highmed.numportal.service.notification.dto.ProjectStartNotification; import org.highmed.numportal.service.notification.dto.ProjectStatusChangeRequestNotification; import org.highmed.numportal.service.policy.ProjectPolicyService; +import org.highmed.numportal.service.util.ExportUtil; + import org.junit.Assert; import org.junit.Before; import org.junit.Ignore; @@ -123,11 +125,9 @@ public class ProjectServiceTest { @Mock private PrivacyProperties privacyProperties; - @Mock - private ProjectsMetrics projectsMetrics; + @Mock private ProjectsMetrics projectsMetrics; - @Mock - private ConsentProperties consentProperties; + @Mock private ConsentProperties consentProperties; @Mock private UserService userService; @@ -151,6 +151,8 @@ public class ProjectServiceTest { @InjectMocks private ProjectService projectService; + @InjectMocks private ExportUtil exportUtil; + @Spy private AqlEditorAqlService aqlEditorAqlService; @Captor ArgumentCaptor> notificationCaptor; @@ -266,12 +268,6 @@ public void retrieveDataPrivacyExceptionMinHits() { projectService.retrieveData("query", 8L, "researcher2", Boolean.TRUE); } - @Test(expected = SystemException.class) - public void executeManagerProjectSystemException() throws JsonProcessingException { - CohortDto cohortDto = CohortDto.builder().name("Cohort name").id(2L).build(); - when(mapper.writeValueAsString(any(Object.class))).thenThrow(new JsonProcessingException("Error"){}); - projectService.executeManagerProject(cohortDto, Arrays.asList("1", "2"), "ownerCoordinatorId"); - } @Test(expected = BadRequestException.class) public void getResearchersBadRequestException() { @@ -1282,40 +1278,6 @@ public void shouldRejectEditingClosedStudies() { verify(projectRepository, times(1)).save(any()); } - @Test - public void shouldSuccessfullyExecuteManagerProject() { - CohortDto cohortDto = CohortDto.builder().name("Cohort name").id(2L).build(); - - UserDetails userDetails = - UserDetails.builder().userId("approvedCoordinatorId").approved(true).build(); - - String result = - projectService.executeManagerProject( - cohortDto, List.of(CORONA_TEMPLATE), userDetails.getUserId()); - - assertThat(result, is("[{\"name\":\"Corona_Anamnese\",\"columns\":null,\"rows\":null}]")); - } - - @Test - public void shouldHandleExecuteManagerProjectWithEmptyTemplates() { - executeManagerProjectWithoutTemplates(Collections.EMPTY_LIST); - } - - @Test - public void shouldHandleExecuteManagerProjectWithNullTemplates() { - executeManagerProjectWithoutTemplates(null); - } - - private void executeManagerProjectWithoutTemplates(List templates) { - CohortDto cohortDto = CohortDto.builder().name("Cohort name").id(2L).build(); - UserDetails userDetails = - UserDetails.builder().userId("approvedCoordinatorId").approved(true).build(); - String result = - projectService.executeManagerProject( - cohortDto, templates, userDetails.getUserId()); - assertThat(result, is("[]")); - } - @Test public void shouldSendNotificationWhenProjectStarts() { Project projectToEdit = @@ -1585,7 +1547,7 @@ public void retrieveDataForProjectWithoutTemplatesTest() { public void getExportFilenameBodyTest() { String currentDate = LocalDateTime.now().truncatedTo(ChronoUnit.MINUTES).format(DateTimeFormatter.ISO_LOCAL_DATE); String expected = "Project_3_" + currentDate.replace("-","_"); - String projectFilename = projectService.getExportFilenameBody(3L); + String projectFilename = exportUtil.getExportFilenameBody(3L); Assert.assertEquals(expected, projectFilename); } @@ -1598,13 +1560,13 @@ public void getInfoDocBytesTest() throws IOException { @Test public void getExportHeadersAsJsonTest() { - MultiValueMap headers = projectService.getExportHeaders(ExportType.json, 3L); + MultiValueMap headers = exportUtil.getExportHeaders(ExportType.json, 3L); Assert.assertEquals(MediaType.APPLICATION_JSON_VALUE, headers.getFirst(HttpHeaders.CONTENT_TYPE)); } @Test public void getExportHeadersAsCSVTest() { - MultiValueMap headers = projectService.getExportHeaders(ExportType.csv, 3L); + MultiValueMap headers = exportUtil.getExportHeaders(ExportType.csv, 3L); Assert.assertEquals("application/zip", headers.getFirst(HttpHeaders.CONTENT_TYPE)); } @@ -1622,30 +1584,7 @@ public void getExportResponseBodyAsCSVTest() { Mockito.verify(cohortService, times(1)).executeCohort(Mockito.eq(2L), Mockito.eq(false)); } - @Test - public void streamResponseBody() throws IOException { - QueryResponseData response = new QueryResponseData(); - response.setName("response-one"); - response.setColumns(new ArrayList<>(List.of(Map.of("path", "/ehr_id/value"), Map.of("uuid", "c/uuid")))); - response.setRows( List.of( - new ArrayList<>(List.of("ehr-id-1", Map.of("_type", "OBSERVATION", "uuid", "12345"))), - new ArrayList<>(List.of("ehr-id-2", Map.of("_type", "SECTION", "uuid", "bla"))))); - ByteArrayOutputStream out = new ByteArrayOutputStream(); - projectService.streamResponseAsZip(List.of(response), "testFile", out); - - ZipInputStream zipInputStream = new ZipInputStream(new ByteArrayInputStream(out.toByteArray())); - ZipEntry expectedFile = zipInputStream.getNextEntry(); - Assert.assertEquals("testFile_response-one.csv", expectedFile.getName()); - } - @Test - public void getManagerExportResponseBodyTest() { - CohortDto cohortDto = CohortDto.builder() - .name("alter cohort") - .projectId(2L).build(); - projectService.getManagerExportResponseBody(cohortDto, List.of("Alter"), "approvedCoordinatorId", ExportType.json); - Mockito.verify(cohortService, Mockito.times(1)).toCohort(Mockito.any(CohortDto.class)); - } @Test public void existsTest() {