CitationController.java
/*
* Copyright 2024 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.genesys.server.api.v2.impl;
import io.swagger.v3.oas.annotations.Operation;
import io.swagger.v3.oas.annotations.tags.Tag;
import org.apache.commons.lang3.ArrayUtils;
import org.genesys.server.api.ApiBaseController;
import org.genesys.server.api.FilteredPage;
import org.genesys.server.api.Pagination;
import org.genesys.server.api.v2.EntityUuidAndVersion;
import org.genesys.server.api.v2.FilteredCRUDController;
import org.genesys.server.api.v2.facade.CitationApiService;
import org.genesys.server.api.v2.model.bib.CitationDTO;
import org.genesys.server.exception.InvalidApiUsageException;
import org.genesys.server.exception.SearchException;
import org.genesys.server.model.bib.Citation;
import org.genesys.server.model.bib.QCitation;
import org.genesys.server.model.filters.CitationFilter;
import org.genesys.server.service.PartnerService;
import org.genesys.server.service.worker.bib.CitationRisConverter;
import org.genesys.server.service.worker.bib.CitationStyle;
import org.genesys.server.service.worker.dupe.DuplicateFinder;
import org.springdoc.api.annotations.ParameterObject;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.data.domain.Pageable;
import org.springframework.http.MediaType;
import org.springframework.security.access.prepost.PreAuthorize;
import org.springframework.web.bind.annotation.DeleteMapping;
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.RequestParam;
import org.springframework.web.bind.annotation.RestController;
import com.querydsl.core.types.OrderSpecifier;
import java.io.IOException;
import java.io.StringReader;
import java.io.StringWriter;
import java.util.List;
import java.util.Map;
import java.util.UUID;
import java.util.stream.Collectors;
import javax.validation.Valid;
@RestController("citationApi2")
@PreAuthorize("isAuthenticated()")
@RequestMapping(CitationController.CONTROLLER_URL)
@Tag(name = "Citation")
public class CitationController extends FilteredCRUDController<CitationDTO, Citation, CitationApiService, CitationFilter> {
/** The Constant CONTROLLER_URL. */
public static final String CONTROLLER_URL = ApiBaseController.APIv2_BASE + "/citation";
@Autowired
CitationStyle citationStyle;
@Autowired
private CitationRisConverter risConverter;
@Autowired
private PartnerService partnerService;
@Override
protected OrderSpecifier<?>[] defaultSort() {
return new OrderSpecifier[] { QCitation.citation.lastModifiedDate.desc() };
}
@Override
@Operation(hidden = true)
public CitationDTO get(long id) {
throw new InvalidApiUsageException("Not applicable");
}
@Operation(description = "Get record by UUID", summary = "Get")
@GetMapping("/{uuid:\\w{8}\\-\\w{4}\\-.{22}}")
public CitationDTO get(@PathVariable(value = "uuid") UUID uuid) {
return serviceFacade.get(uuid);
}
@Override
@Operation(hidden = true)
public CitationDTO remove(@PathVariable("id") final long id) {
throw new InvalidApiUsageException("Not applicable");
}
@Operation(description = "Delete record by UUID", summary = "Delete citation")
@DeleteMapping("/{uuid:\\w{8}\\-\\w{4}\\-.{22}}")
public CitationDTO remove(@PathVariable(value = "uuid") UUID uuid) {
return serviceFacade.remove(serviceFacade.get(uuid));
}
@Operation(description = "List records I can access", summary = "List only my records")
@PostMapping(value = "/list-mine", produces = MediaType.APPLICATION_JSON_VALUE)
public FilteredPage<CitationDTO, CitationFilter> listMyCitations(@RequestParam(name = "f", required = false) String filterCode, @ParameterObject final Pagination page,
@RequestBody(required = false) CitationFilter filter) throws IOException, SearchException {
var filterInfo = processFilter(filterCode, filter, filterType);
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.listMyCitations(filterInfo.filter, pageable));
}
@Operation(description = "Find similar citations", summary = "Get similar")
@PostMapping(value = "/similar", produces = MediaType.APPLICATION_JSON_VALUE)
public List<DuplicateFinder.Hit<CitationDTO>> findSimilar(@RequestBody CitationDTO citation) {
return serviceFacade.findSimilar(citation, null);
}
/**
* Generate citation texts in different styles
*
* @param citation thing to cite
* @return Map of citations by style
*/
@Operation(description = "Generate citations in all available styles", summary = "Generate citation")
@PostMapping(value = "/cite", produces = MediaType.APPLICATION_JSON_VALUE)
public Map<String, String> cite(@RequestBody CitationDTO citation) {
return Map.of("apa", citationStyle.toAPA(citation));
}
@Operation(description = "Read citation data from RIS format", summary = "Parse RIS")
@PostMapping(value = "/parse", produces = MediaType.APPLICATION_JSON_VALUE, consumes = MediaType.TEXT_PLAIN_VALUE)
public List<CitationDTO> parseCitations(@RequestBody String input) throws IOException {
return risConverter.readCitations(new StringReader(input)).collect(Collectors.toList());
}
@Operation(description = "Export citation data inf RIS format", summary = "Export as RIS")
@PostMapping(value = "/export", produces = MediaType.TEXT_PLAIN_VALUE, consumes = MediaType.APPLICATION_JSON_VALUE)
public String exportCitations(@RequestBody CitationFilter filter) throws IOException, SearchException {
var citations = serviceFacade.listMyCitations(filter, Pageable.unpaged()).getContent();
var writer = new StringWriter();
for (CitationDTO citation : citations) {
risConverter.writeCitation(citation, writer);
}
return writer.toString();
}
@PostMapping(value = "/approve", produces = MediaType.APPLICATION_JSON_VALUE)
public CitationDTO approveCitation(@RequestBody @Valid EntityUuidAndVersion ref) {
return serviceFacade.get(serviceFacade.approveCitation(ref));
}
@PostMapping(value = "/reject", produces = MediaType.APPLICATION_JSON_VALUE)
public CitationDTO rejectCitation(@RequestBody @Valid EntityUuidAndVersion ref) {
return serviceFacade.get(serviceFacade.rejectCitation(ref));
}
@PostMapping(value = "/for-review", produces = MediaType.APPLICATION_JSON_VALUE)
public CitationDTO reviewCitation(@RequestBody @Valid EntityUuidAndVersion ref) {
return serviceFacade.get(serviceFacade.reviewCitation(ref));
}
@Operation(description = "Get all collections by owner", summary = "Get collections by owner")
@GetMapping("/owner/{uuid:\\w{8}\\-\\w{4}\\-.{22}}/collections")
public List<String> getCollections(@PathVariable(value = "uuid") UUID owner) {
return serviceFacade.getCollections(partnerService.get(owner));
}
@Operation(description = "Get all published collections by filter", summary = "Get published collections by filter")
@PostMapping("/collections")
public List<String> getCollections(@RequestBody(required = false) CitationFilter filter) {
return serviceFacade.getCollections(filter);
}
}