TransportUnitController.java
- /*
- * Copyright 2005-2025 the original author or authors.
- *
- * 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.openwms.common.transport;
- import jakarta.servlet.http.HttpServletRequest;
- import jakarta.validation.Valid;
- import jakarta.validation.constraints.NotBlank;
- import jakarta.validation.constraints.NotEmpty;
- import org.ameba.LoggingCategories;
- import org.ameba.exception.NotFoundException;
- import org.ameba.exception.ResourceExistsException;
- import org.ameba.http.MeasuredRestController;
- import org.ameba.http.Response;
- import org.ameba.i18n.Translator;
- import org.openwms.common.SimpleLink;
- import org.openwms.common.StateChangeException;
- import org.openwms.common.location.LocationController;
- import org.openwms.common.transport.api.TransportApiConstants;
- import org.openwms.common.transport.api.TransportUnitVO;
- import org.openwms.common.transport.api.ValidationGroups;
- import org.openwms.common.transport.barcode.BarcodeGenerator;
- import org.openwms.core.SpringProfiles;
- import org.openwms.core.http.AbstractWebController;
- import org.openwms.core.http.Index;
- import org.slf4j.Logger;
- import org.slf4j.LoggerFactory;
- import org.springframework.context.annotation.Profile;
- import org.springframework.http.HttpStatus;
- import org.springframework.http.ResponseEntity;
- import org.springframework.validation.annotation.Validated;
- import org.springframework.web.bind.annotation.DeleteMapping;
- import org.springframework.web.bind.annotation.ExceptionHandler;
- import org.springframework.web.bind.annotation.GetMapping;
- import org.springframework.web.bind.annotation.PatchMapping;
- import org.springframework.web.bind.annotation.PathVariable;
- import org.springframework.web.bind.annotation.PostMapping;
- import org.springframework.web.bind.annotation.PutMapping;
- import org.springframework.web.bind.annotation.RequestBody;
- import org.springframework.web.bind.annotation.RequestParam;
- import java.util.List;
- import static java.util.Arrays.asList;
- import static org.openwms.common.CommonMessageCodes.TU_BARCODE_MISSING;
- import static org.openwms.common.CommonMessageCodes.TU_EXISTS;
- import static org.openwms.common.transport.api.TransportApiConstants.API_TRANSPORT_UNIT;
- import static org.openwms.common.transport.api.TransportApiConstants.API_TRANSPORT_UNITS;
- import static org.openwms.common.transport.api.TransportUnitVO.MEDIA_TYPE;
- import static org.springframework.hateoas.server.mvc.WebMvcLinkBuilder.linkTo;
- import static org.springframework.hateoas.server.mvc.WebMvcLinkBuilder.methodOn;
- /**
- * A TransportUnitController.
- *
- * @author Heiko Scherrer
- */
- @Profile("!" + SpringProfiles.IN_MEMORY)
- @Validated
- @MeasuredRestController
- public class TransportUnitController extends AbstractWebController {
- private static final Logger EXC_LOGGER = LoggerFactory.getLogger(LoggingCategories.PRESENTATION_LAYER_EXCEPTION);
- private final TransportUnitMapper mapper;
- private final Translator translator;
- private final BarcodeGenerator barcodeGenerator;
- private final TransportUnitService service;
- TransportUnitController(TransportUnitMapper mapper, Translator translator, BarcodeGenerator barcodeGenerator, TransportUnitService service) {
- this.mapper = mapper;
- this.translator = translator;
- this.barcodeGenerator = barcodeGenerator;
- this.service = service;
- }
- @ExceptionHandler({ StateChangeException.class })
- private ResponseEntity<?> handleStateChangeException(StateChangeException e) {
- EXC_LOGGER.error("[P] Presentation Layer Exception: {}", e.getLocalizedMessage(), e);
- return new ResponseEntity<>(Response.newBuilder().withMessage(e.getMessage()).withMessageKey(e.getMessageKey())
- .withHttpStatus(HttpStatus.CONFLICT.toString()).build(), HttpStatus.CONFLICT);
- }
- @GetMapping(value = API_TRANSPORT_UNITS + "/{pKey}", produces = MEDIA_TYPE)
- public ResponseEntity<TransportUnitVO> findTransportUnitByPKey(
- @PathVariable("pKey") String pKey
- ) {
- return ResponseEntity.ok(
- convertAndLinks(service.findByPKey(pKey))
- );
- }
- @GetMapping(value = API_TRANSPORT_UNITS, params = {"bk"}, produces = MEDIA_TYPE)
- public ResponseEntity<TransportUnitVO> findTransportUnit(
- @RequestParam("bk") String transportUnitBK
- ) {
- return ResponseEntity.ok(
- convertAndLinks(service.findByBarcode(transportUnitBK))
- );
- }
- @GetMapping(value = API_TRANSPORT_UNITS, produces = MEDIA_TYPE)
- public ResponseEntity<List<TransportUnitVO>> findAll() {
- return ResponseEntity.ok(
- convertAndLinks(service.findAll())
- );
- }
- @GetMapping(value = API_TRANSPORT_UNITS, params = {"bks"}, produces = MEDIA_TYPE)
- public ResponseEntity<List<TransportUnitVO>> findTransportUnits(
- @RequestParam("bks") @NotEmpty List<String> barcodes
- ) {
- return ResponseEntity.ok(
- convertAndLinks(service.findByBarcodes(barcodes.stream().map(barcodeGenerator::convert).toList()))
- );
- }
- @GetMapping(value = API_TRANSPORT_UNITS, params = {"actualLocation"}, produces = MEDIA_TYPE)
- public ResponseEntity<List<TransportUnitVO>> findTransportUnitsOn(
- @RequestParam("actualLocation") String actualLocation
- ) {
- return ResponseEntity.ok(
- convertAndLinks(service.findOnLocation(actualLocation))
- );
- }
- @PostMapping(value = API_TRANSPORT_UNITS, params = {"bk"})
- public ResponseEntity<Void> createTU(
- @RequestParam("bk") String transportUnitBK,
- @Validated(ValidationGroups.TransportUnit.Create.class) @RequestBody TransportUnitVO tu,
- @RequestParam(value = "strict", required = false) Boolean strict,
- HttpServletRequest req
- ) {
- if (Boolean.TRUE.equals(strict)) {
- // check if already exists ...
- try {
- service.findByBarcode(transportUnitBK);
- throw new ResourceExistsException(translator.translate(TU_EXISTS, transportUnitBK), TU_EXISTS, transportUnitBK);
- } catch (NotFoundException nfe) {
- // that's fine we just cast the exception thrown by the service
- }
- }
- var created = service.create(transportUnitBK, tu.getTransportUnitType().getType(), tu.getActualLocation().getLocationId(), strict);
- return ResponseEntity.created(getLocationURIForCreatedResource(req, created.getPersistentKey())).build();
- }
- @PostMapping(API_TRANSPORT_UNITS + "/synchronize")
- public void synchronizeTU() {
- service.synchronizeTransportUnits();
- }
- @DeleteMapping(value = API_TRANSPORT_UNITS + "/{pKey}")
- public ResponseEntity<Void> deleteTU(@PathVariable("pKey") String pKey) {
- service.delete(pKey);
- return ResponseEntity.noContent().build();
- }
- @PostMapping(value = API_TRANSPORT_UNITS, params = {"actualLocation", "tut"}, produces = MEDIA_TYPE)
- public ResponseEntity<TransportUnitVO> createTU(
- @RequestParam(value = "bk", required = false) String transportUnitBK,
- @RequestParam("actualLocation") String actualLocation,
- @RequestParam("tut") String tut,
- @RequestParam(value = "strict", required = false) Boolean strict,
- HttpServletRequest req
- ) {
- if (Boolean.TRUE.equals(strict)) {
- if (transportUnitBK == null || transportUnitBK.isEmpty()) {
- throw new IllegalArgumentException(translator.translate(TU_BARCODE_MISSING));
- }
- // check if already exists ...
- try {
- service.findByBarcode(transportUnitBK);
- throw new ResourceExistsException(translator.translate(TU_EXISTS, transportUnitBK), TU_EXISTS, transportUnitBK);
- } catch (NotFoundException nfe) {
- // that's fine we just cast the exception thrown by the service
- }
- }
- var created = transportUnitBK == null
- ? service.createNew(tut, actualLocation)
- : service.create(transportUnitBK, tut, actualLocation, strict);
- return ResponseEntity
- .created(getLocationURIForCreatedResource(req, created.getPersistentKey()))
- .body(convertAndLinks(created))
- ;
- }
- @Validated(ValidationGroups.TransportUnit.Update.class)
- @PutMapping(value = API_TRANSPORT_UNITS, params = {"bk"}, produces = MEDIA_TYPE)
- public ResponseEntity<TransportUnitVO> updateTU(
- @RequestParam("bk") String transportUnitBK,
- @Valid @RequestBody TransportUnitVO tu
- ) {
- return ResponseEntity.ok(
- convertAndLinks(service.update(barcodeGenerator.convert(transportUnitBK), mapper.convert(tu)))
- );
- }
- @PatchMapping(value = API_TRANSPORT_UNITS, params = {"bk", "newLocation"}, produces = MEDIA_TYPE)
- public ResponseEntity<TransportUnitVO> moveTU(
- @RequestParam("bk") String transportUnitBK,
- @RequestParam("newLocation") String newLocation
- ) {
- return ResponseEntity.ok(
- convertAndLinks(service.moveTransportUnit(barcodeGenerator.convert(transportUnitBK), newLocation))
- );
- }
- @PostMapping(value = API_TRANSPORT_UNIT + "/error", params = {"bk", "errorCode"}, produces = MEDIA_TYPE)
- public ResponseEntity<Void> addErrorToTransportUnit(
- @RequestParam("bk") String transportUnitBK,
- @RequestParam(value = "errorCode") String errorCode
- ) {
- service.addError(transportUnitBK, UnitError.newBuilder()
- .errorNo(errorCode)
- .build()
- );
- return ResponseEntity.noContent().build();
- }
- @GetMapping(API_TRANSPORT_UNITS + "/index")
- public ResponseEntity<Index> index() {
- return ResponseEntity.ok(
- new Index(
- linkTo(methodOn(TransportUnitController.class).createTU("{transportUnitBK}", null, true, null)).withRel("transport-unit-createtuwithbody"),
- linkTo(methodOn(TransportUnitController.class).createTU("{transportUnitBK}", "{actualLocation}", "{transportUnitType}", true, null)).withRel("transport-unit-createtuwithparams"),
- linkTo(methodOn(TransportUnitController.class).deleteTU("{pKey}")).withRel("transport-unit-deletebypkey"),
- linkTo(methodOn(TransportUnitController.class).findTransportUnitByPKey("1")).withRel("transport-unit-findbypkey"),
- linkTo(methodOn(TransportUnitController.class).findTransportUnit("{transportUnitBK}")).withRel("transport-unit-findbybarcode"),
- linkTo(methodOn(TransportUnitController.class).findTransportUnits(asList("{transportUnitBK-1}", "{transportUnitBK-n}"))).withRel("transport-unit-findbybarcodes"),
- linkTo(methodOn(TransportUnitController.class).findTransportUnitsOn("{actualLocation.locationId}")).withRel("transport-unit-findonlocation"),
- linkTo(methodOn(TransportUnitController.class).blockTransportUnit("{transportUnitBK}")).withRel("transport-unit-block"),
- linkTo(methodOn(TransportUnitController.class).unblockTransportUnit("{transportUnitBK}")).withRel("transport-unit-unblock"),
- linkTo(methodOn(TransportUnitController.class).qcTransportUnit("{transportUnitBK}")).withRel("transport-unit-qc")
- )
- );
- }
- @PostMapping(value = TransportApiConstants.API_TRANSPORT_UNITS + "/block", params = {"bk"})
- public ResponseEntity<Void> blockTransportUnit(@NotBlank @RequestParam("bk") String transportUnitBK) {
- service.setState(transportUnitBK, TransportUnitState.BLOCKED.name());
- return ResponseEntity.noContent().build();
- }
- @PostMapping(value = TransportApiConstants.API_TRANSPORT_UNITS + "/available", params = {"bk"})
- public ResponseEntity<Void> unblockTransportUnit(@NotBlank @RequestParam("bk") String transportUnitBK) {
- service.setState(transportUnitBK, TransportUnitState.AVAILABLE.name());
- return ResponseEntity.noContent().build();
- }
- @PostMapping(value = TransportApiConstants.API_TRANSPORT_UNITS + "/quality-check", params = {"bk"})
- public ResponseEntity<Void> qcTransportUnit(@NotBlank @RequestParam("bk") String transportUnitBK) {
- service.setState(transportUnitBK, TransportUnitState.QUALITY_CHECK.name());
- return ResponseEntity.noContent().build();
- }
- @PostMapping(value = TransportApiConstants.API_TRANSPORT_UNITS, params = {"bk", "state"})
- public ResponseEntity<Void> changeState(
- @NotBlank @RequestParam("bk") String transportUnitBK,
- @NotBlank @RequestParam("state") String newState) {
- service.setState(transportUnitBK, newState);
- return ResponseEntity.noContent().build();
- }
- private TransportUnitVO addLinks(TransportUnitVO result) {
- result.add(
- new SimpleLink(linkTo(methodOn(TransportUnitController.class).findTransportUnitByPKey(result.getpKey())).withSelfRel()),
- new SimpleLink(linkTo(methodOn(TransportUnitTypeController.class).findTransportUnitType(result.getTransportUnitType().getType())).withRel("transport-unit-type"))
- );
- if (result.getActualLocation() != null) {
- result.add(
- new SimpleLink(linkTo(methodOn(LocationController.class).findByCoordinate(result.getActualLocation().getLocationId())).withRel("actual-location"))
- );
- }
- return result;
- }
- private TransportUnitVO convertAndLinks(TransportUnit entity) {
- return addLinks(
- mapper.convertToVO(entity)
- );
- }
- private List<TransportUnitVO> convertAndLinks(List<TransportUnit> entities) {
- return entities.stream()
- .map(mapper::convertToVO)
- .map(this::addLinks)
- .toList();
- }
- }