InventoryViabilityController.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.gringlobal.api.v2.impl;
import java.io.IOException;
import java.util.LinkedHashMap;
import java.util.List;
import java.util.Map;
import java.util.Set;
import javax.validation.Valid;
import javax.validation.constraints.NotNull;
import javax.validation.constraints.Size;
import org.apache.commons.lang3.ArrayUtils;
import org.gringlobal.api.model.CooperatorInfo;
import org.gringlobal.api.model.InventoryActionDTO;
import org.gringlobal.api.model.InventoryViabilityActionDTO;
import org.gringlobal.api.model.InventoryViabilityActionRequestDTO;
import org.gringlobal.api.model.InventoryViabilityDTO;
import org.gringlobal.api.model.InventoryViabilityDataDTO;
import org.gringlobal.api.model.InventoryViabilityDetailsDTO;
import org.gringlobal.api.model.OrderRequestDTO;
import org.gringlobal.api.model.SiteInfo;
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.InventoryActionApiService;
import org.gringlobal.api.v2.facade.InventoryViabilityActionApiService;
import org.gringlobal.api.v2.facade.InventoryViabilityApiService;
import org.gringlobal.api.v2.facade.InventoryViabilityDataApiService;
import org.gringlobal.custom.elasticsearch.SearchException;
import org.gringlobal.model.AbstractAction.ActionState;
import org.gringlobal.model.InventoryViability;
import org.gringlobal.model.InventoryViabilityAction;
import org.gringlobal.model.InventoryViabilityData;
import org.gringlobal.model.QInventoryAction;
import org.gringlobal.model.QInventoryViability;
import org.gringlobal.model.QInventoryViabilityAction;
import org.gringlobal.model.QInventoryViabilityData;
import org.gringlobal.model.community.CommunityCodeValues;
import org.gringlobal.service.CRUDService;
import org.gringlobal.service.InventoryViabilityActionService.InventoryViabilityActionScheduleFilter;
import org.gringlobal.service.filter.InventoryActionFilter;
import org.gringlobal.service.filter.InventoryFilter;
import org.gringlobal.service.filter.InventoryViabilityActionFilter;
import org.gringlobal.service.filter.InventoryViabilityDataFilter;
import org.gringlobal.service.filter.InventoryViabilityFilter;
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.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.RestController;
import org.springframework.web.servlet.mvc.method.annotation.StreamingResponseBody;
import com.querydsl.core.types.OrderSpecifier;
import io.swagger.v3.oas.annotations.Operation;
import io.swagger.v3.oas.annotations.media.Content;
import io.swagger.v3.oas.annotations.media.Schema;
import io.swagger.v3.oas.annotations.responses.ApiResponse;
import io.swagger.v3.oas.annotations.responses.ApiResponses;
import io.swagger.v3.oas.annotations.tags.Tag;
import lombok.NoArgsConstructor;
import lombok.experimental.SuperBuilder;
@RestController("inventoryViabilityApi2")
@RequestMapping(InventoryViabilityController.API_URL)
@PreAuthorize("isAuthenticated()")
@Tag(name = InventoryViabilityController.API_TAG)
public class InventoryViabilityController extends FilteredCRUDController<InventoryViabilityDTO, InventoryViability, InventoryViabilityApiService, InventoryViabilityFilter> {
/** The Constant API_URL. */
public static final String API_URL = InventoryController.API_URL + "/viability";
public static final String API_TAG = "InventoryViability";
@Autowired
private InventoryActionApiService inventoryActionService;
@NoArgsConstructor
@SuperBuilder
public static class ViabilityOrderRequest {
public SiteInfo site;
@NotNull
public CooperatorInfo cooperator;
@NotNull
@Size(min = 1)
public Map<Long, Long> inventories;
}
@Override
protected OrderSpecifier<?>[] defaultSort() {
return new OrderSpecifier[] { QInventoryViability.inventoryViability.testedDate.desc() };
}
@PostMapping("/scheduled")
public FilteredPage<InventoryActionDTO, InventoryFilter> listScheduledInventories(@ParameterObject final Pagination page, @RequestBody(required = false) final InventoryFilter inventoryFilter) throws SearchException, IOException {
InventoryFilter normalizedFilter = shortFilterService.normalizeFilter(inventoryFilter, InventoryFilter.class);
InventoryActionFilter actionFilter = (InventoryActionFilter) new InventoryActionFilter()
.inventory(normalizedFilter)
.actionNameCode(Set.of(CommunityCodeValues.INVENTORY_ACTION_VABILITYTEST.value))
.states(Set.of(ActionState.PENDING, ActionState.SCHEDULED));
Pageable pageable = ArrayUtils.isEmpty(page.getS()) ? page.toPageRequest(MAX_PAGE_SIZE, DEFAULT_PAGE_SIZE, QInventoryAction.inventoryAction.notBeforeDate.asc().nullsFirst()) : page.toPageRequest(MAX_PAGE_SIZE, DEFAULT_PAGE_SIZE);
return new FilteredPage<>(normalizedFilter, inventoryActionService.listActions(actionFilter, pageable));
}
@PostMapping("/order")
public OrderRequestDTO orderViabilityTest(@RequestBody @Valid ViabilityOrderRequest viabilityOrderRequest) {
return serviceFacade.orderViabilityTest(viabilityOrderRequest);
}
/**
* Calculate results across all replicates.
*
* @param id the id of InventoryViability
* @return object of InventoryViability (not persisted) with calculated results
*/
@PostMapping("/{id:\\d+}/calculate")
public InventoryViabilityDetailsDTO calculate(@PathVariable("id") final long id, @RequestBody(required = false) List<Integer> selectedReplicationNumbers) {
return serviceFacade.calculateResult(id, selectedReplicationNumbers);
}
/**
* Returns fully loaded viability + data records
* @param id InventoryViability#id
* @return fully loaded viability
*/
@GetMapping(ENDPOINT_ID + "/details")
public InventoryViabilityDetailsDTO getDetails(@PathVariable("id") final long id) {
return serviceFacade.loadDetails(id);
}
/**
* Print the same label for specified <code>ids</code>.
*/
@PostMapping(value = "/generate-labels")
@Operation(method = "generateLabels", summary = "Generate replicate labels for selected viability test IDs", description = "Generates a separate label for each replicate in the test. Use the same label configuration for each record.")
@ApiResponses(value = @ApiResponse(responseCode = "200", content = @Content(mediaType = MediaType.TEXT_PLAIN_VALUE, schema = @Schema(type = "string"))))
public ResponseEntity<StreamingResponseBody> generateLabels(
@ParameterObject CRUDService.LabelConfig labelConfig,
@RequestBody List<Long> ids
) {
return serviceFacade.generateLabels(labelConfig, ids);
}
/**
* Print the different labels for <code>ids</code>.
*/
@PostMapping(value = "/generate-labels-custom")
@Operation(method = "generateLabelCustom", summary = "Generate replicate labels for selected viability test IDs", description = "Generates a separate label for each replicate in the test with different label configuration for each record. Map keys are record IDs.")
@ApiResponses(value = @ApiResponse(responseCode = "200", content = @Content(mediaType = MediaType.TEXT_PLAIN_VALUE, schema = @Schema(type = "string"))))
public ResponseEntity<StreamingResponseBody> generateLabelsAdvanced(@RequestBody LinkedHashMap<Long, CRUDService.LabelConfig> labelConfig) {
return serviceFacade.generateLabels(labelConfig);
}
@RestController("inventoryViabilityActionApi2")
@RequestMapping(InventoryViabilityActionController.API_URL)
@PreAuthorize("isAuthenticated()")
@Tag(name = InventoryViabilityController.API_TAG)
public static class InventoryViabilityActionController extends ActionController<InventoryViabilityActionDTO, InventoryViabilityAction, InventoryViabilityActionFilter, InventoryViabilityActionRequestDTO, InventoryViabilityActionScheduleFilter, InventoryViabilityActionApiService> {
public static final String API_URL = InventoryViabilityController.API_URL;
@Override
protected OrderSpecifier<?>[] defaultSort() {
return new OrderSpecifier[] { QInventoryViabilityAction.inventoryViabilityAction.startedDate.desc() };
}
}
@RestController("inventoryViabilityDataApi2")
@RequestMapping(InventoryViabilityDataController.API_URL)
@PreAuthorize("isAuthenticated()")
@Tag(name = InventoryViabilityController.API_TAG)
public static class InventoryViabilityDataController extends FilteredCRUDController<InventoryViabilityDataDTO, InventoryViabilityData, InventoryViabilityDataApiService, InventoryViabilityDataFilter> {
public static final String API_URL = InventoryViabilityController.API_URL + "/data";
@Override
protected OrderSpecifier<?>[] defaultSort() {
return new OrderSpecifier[] { QInventoryViabilityData.inventoryViabilityData.countDate.desc() };
}
}
}