AccessionController.java
/*
* Copyright 2023 Global Crop Diversity Trust
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package org.gringlobal.api.v2.impl;
import java.io.EOFException;
import java.io.IOException;
import java.io.OutputStream;
import java.util.List;
import java.util.Map;
import java.util.Set;
import javax.servlet.http.HttpServletResponse;
import lombok.extern.slf4j.Slf4j;
import org.apache.commons.lang3.ArrayUtils;
import org.apache.commons.lang3.StringUtils;
import org.genesys.blocks.auditlog.model.filters.AuditLogFilter;
import org.genesys.blocks.auditlog.service.AuditTrailService;
import org.gringlobal.api.model.AccessionActionDTO;
import org.gringlobal.api.model.AccessionActionRequestDTO;
import org.gringlobal.api.model.AccessionDTO;
import org.gringlobal.api.model.AccessionDetail;
import org.gringlobal.api.model.AccessionInvGroupDTO;
import org.gringlobal.api.model.AcquisitionDTO;
import org.gringlobal.api.model.AuditLogDTO;
import org.gringlobal.api.v1.ApiBaseController;
import org.gringlobal.api.v1.FilteredPage;
import org.gringlobal.api.v1.Pagination;
import org.gringlobal.api.v2.ActionController;
import org.gringlobal.api.v2.FilteredCRUDController;
import org.gringlobal.api.v2.facade.AccessionActionApiService;
import org.gringlobal.api.v2.facade.AccessionApiService;
import org.gringlobal.api.v2.mapper.MapstructMapper;
import org.gringlobal.custom.elasticsearch.SearchException;
import org.gringlobal.model.Accession;
import org.gringlobal.model.AccessionAction;
import org.gringlobal.model.QAccession;
import org.gringlobal.model.QAccessionAction;
import org.gringlobal.model.community.AccessionMCPD;
import org.gringlobal.service.AccessionActionService;
import org.gringlobal.service.AccessionService.MapInfo;
import org.gringlobal.service.DownloadService;
import org.gringlobal.service.filter.AccessionActionFilter;
import org.gringlobal.service.filter.AccessionFilter;
import org.gringlobal.service.genesys.validator.impl.GenesysValidator;
import org.gringlobal.service.glis.impl.GlisDOIRegistrationManager;
import org.gringlobal.spring.CSVMessageConverter;
import org.gringlobal.worker.dupe.DuplicateFinder;
import org.springdoc.api.annotations.ParameterObject;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.data.domain.Page;
import org.springframework.data.domain.Pageable;
import org.springframework.http.HttpStatus;
import org.springframework.http.MediaType;
import org.springframework.http.ResponseEntity;
import org.springframework.security.access.prepost.PreAuthorize;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.PathVariable;
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.RequestMethod;
import org.springframework.web.bind.annotation.RequestParam;
import org.springframework.web.bind.annotation.RestController;
import com.fasterxml.jackson.databind.ObjectMapper;
import com.querydsl.core.types.OrderSpecifier;
import io.swagger.v3.oas.annotations.Operation;
import io.swagger.v3.oas.annotations.tags.Tag;
@RestController("accessionApi2")
@RequestMapping(AccessionController.API_URL)
@PreAuthorize("isAuthenticated()")
@Tag(name = "Accession")
@Slf4j
public class AccessionController extends FilteredCRUDController<AccessionDTO, Accession, AccessionApiService, AccessionFilter> {
/** The Constant API_URL. */
public static final String API_URL = ApiBaseController.APIv2_BASE + "/a";
@Autowired
private ObjectMapper objectMapper;
@Autowired
private AuditTrailService auditService;
@Autowired
private DownloadService downloadService;
@Autowired
private GlisDOIRegistrationManager glisDOIRegistrationManager;
@Autowired
private MapstructMapper mapper;
@Autowired
private GenesysValidator genesysValidator;
@Override
protected OrderSpecifier<?>[] defaultSort() {
return new OrderSpecifier[] { QAccession.accession.id.asc() };
}
@RestController("accessionActionApi2")
@RequestMapping(AccessionActionController.API_URL)
@PreAuthorize("isAuthenticated()")
@Tag(name = "Accession")
public static class AccessionActionController extends ActionController<AccessionActionDTO, AccessionAction, AccessionActionFilter, AccessionActionRequestDTO, AccessionActionService.AccessionActionScheduleFilter, AccessionActionApiService> {
public static final String API_URL = AccessionController.API_URL;
@Override
protected OrderSpecifier<?>[] defaultSort() {
return new OrderSpecifier[] { QAccessionAction.accessionAction.createdDate.desc() };
}
}
@Override
public AccessionDTO create(@RequestBody AccessionDTO entity) {
return super.create(entity);
}
@Override
public AccessionDTO update(@RequestBody AccessionDTO entity) {
return super.update(entity);
}
@Override
@PostMapping(value = FilteredCRUDController.ENDPOINT_LIST, produces = { MediaType.APPLICATION_JSON_VALUE, CSVMessageConverter.TEXT_CSV_VALUE })
public FilteredPage<AccessionDTO, AccessionFilter> list(@ParameterObject final Pagination page, @RequestBody AccessionFilter filter) throws SearchException, IOException {
return super.list(page, filter);
}
@Override
public FilteredPage<AccessionDTO, AccessionFilter> filter(@RequestParam(name = "f", required = false) String filterCode, @ParameterObject final Pagination page,
@RequestBody(required = false) AccessionFilter filter) throws IOException, SearchException {
return super.filter(filterCode, page, filter);
}
@PostMapping(value = "/mcpd", produces = { MediaType.APPLICATION_JSON_VALUE, CSVMessageConverter.TEXT_CSV_VALUE })
public FilteredPage<AccessionMCPD, AccessionFilter> listMcpd(@RequestParam(name = "f", required = false) String filterCode, @ParameterObject final Pagination page,
@RequestBody AccessionFilter filter) throws SearchException, IOException {
final var filterInfo = shortFilterService.processFilter(filterCode, filter, AccessionFilter.class);
Pageable pageable = ArrayUtils.isEmpty(page.getS()) ? page.toPageRequest(MAX_PAGE_SIZE, DEFAULT_PAGE_SIZE, defaultSort()) : page.toPageRequest(MAX_PAGE_SIZE, DEFAULT_PAGE_SIZE);
return new FilteredPage<>(filterInfo.filterCode, filterInfo.filter, serviceFacade.listMCPD(filterInfo.filter, pageable));
}
/**
* Retrieve accession details by id
*
* @param id the id
* @return the accession details
*/
@GetMapping(value = "/details/{id}", produces = { MediaType.APPLICATION_JSON_VALUE })
@Operation(operationId = "getDetails", description = "Retrieve accession details by ID", summary = "Details")
public AccessionDetail details(@PathVariable("id") final long id) {
return serviceFacade.getAccessionDetails(id);
}
/**
* Retrieve a list of audit logs for the specified accession
*
* @param accessionId the accessionId
* @param page the page request
* @return the list of all log entries
*/
@GetMapping("/auditlog/{id}")
public Page<AuditLogDTO> accessionAuditLogs(@PathVariable(value = "id") final Long accessionId, @ParameterObject final Pagination page) {
var filter = new AuditLogFilter();
filter.entityId().eq().add(serviceFacade.get(accessionId).getId());
filter.classname(Accession.class.getName());
// TODO create auditServiceFacade
return mapper.map(auditService.listAuditLogs(filter, page.toPageRequest(MAX_PAGE_SIZE, DEFAULT_PAGE_SIZE)), mapper::map);
}
@PostMapping(value = "/acquire", produces = { MediaType.APPLICATION_JSON_VALUE })
@Operation(operationId = "addMaterial", description = "Register new accessions and inventories", summary = "Add new material")
public AccessionInvGroupDTO acquire(@RequestBody(required = true) AcquisitionDTO acquisitionBatch) {
return serviceFacade.acquire(acquisitionBatch);
}
@PostMapping(value = "/overview/{groupBy:.+}", produces = { MediaType.APPLICATION_JSON_VALUE })
@Operation(operationId = "accessionOverview", description = "Get accession statistics", summary = "Overview")
public Map<?, ?> accessionOverview(@PathVariable(name = "groupBy", required = true) String groupBy, @RequestBody AccessionFilter filter) {
return serviceFacade.accessionOverview(groupBy, filter);
}
@RequestMapping(value = "/download-mcpd", method = RequestMethod.POST, params = { "filter" })
public void downloadMcpd(@RequestParam(name = "filter", required = true) String filter, HttpServletResponse response) throws IOException, SearchException {
var accessionFilters = objectMapper.readValue(filter, AccessionFilter.class);
final var filterInfo = shortFilterService.processFilter(null, accessionFilters, AccessionFilter.class);
// Write MCPD to the stream.
response.setContentType("application/vnd.openxmlformats-officedocument.spreadsheetml.sheet");
response.addHeader("Content-Disposition", String.format("attachment; filename=\"MCPD-%1s.xlsx\"", StringUtils.defaultIfBlank(filterInfo.filterCode, "all")));
final OutputStream outputStream = response.getOutputStream();
try {
downloadService.writeXlsxMCPD(filterInfo.filter, outputStream, filterInfo.filterCode, "/a/" + filterInfo.filterCode);
response.flushBuffer();
} catch (EOFException e) {
log.warn("Download was aborted: {}", e.getMessage());
throw e;
}
}
/**
* Update GLIS DOI Registration Service for one accession
*
* @param id Accession ID
* @return DOI Registration status
* @throws Throwable the problem error
*/
@PostMapping(value = ENDPOINT_ID + "/assign-doi")
public GlisDOIRegistrationManager.GlisDoiResponse assignDoiToAccessions(@PathVariable(name = "id") final long id) throws Throwable {
var accession = serviceFacade.get(id);
return glisDOIRegistrationManager.updateDoiRegistration(null, (AccessionFilter) new AccessionFilter().id(Set.of(accession.getId()))).get(0);
}
/**
* Update GLIS DOI Registration Service for accession by filter with propress data
*
* @param filterCode Filter code
* @param filter AccessionFilter
* @return DOI Registration progress data
* @throws Exception the problem error
*/
@PostMapping(value = "/assign-doi")
public GlisDOIRegistrationManager.AssignDoiState assignDoiToAccessions(
@RequestParam(name = "f", required = false) String filterCode,
@RequestBody(required = false) final org.gringlobal.service.filter.AccessionFilter filter
) throws Exception {
return glisDOIRegistrationManager.updateDoiRegistrationWithProgress(filterCode, filter);
}
@PostMapping("/assign-doi/cancel")
public ResponseEntity<HttpStatus> cancelDoiAssignment(@RequestParam(name = "f", required = false) String filterCode) {
return glisDOIRegistrationManager.cancelDoiAssignment(filterCode);
}
/**
* Find similar accessions by source
*
* @param source source AccessionDTO
* @return the list of similar Accessions
*/
@PostMapping(value = "/similar")
public List<DuplicateFinder.Hit<AccessionDTO>> findSimilarForUnsaved(@RequestBody final AccessionDTO source) {
return serviceFacade.findSimilar(source);
}
@PostMapping(value = "/map-info")
public MapInfo<AccessionFilter> mapInfo(
@RequestParam(name = "f", required = false) String filterCode,
@RequestBody AccessionFilter filter
) throws SearchException, IOException {
final var filterInfo = shortFilterService.processFilter(filterCode, filter, AccessionFilter.class);
return serviceFacade.mapInfo(filterInfo);
}
@PostMapping(value = "/validate")
public void validate(@RequestBody AccessionFilter filter, HttpServletResponse response) throws Exception {
genesysValidator.processAccessionGeo(filter, response);
}
}