LocationController.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.location;

import jakarta.servlet.http.HttpServletRequest;
import jakarta.validation.Valid;
import org.ameba.exception.BusinessRuntimeException;
import org.ameba.exception.NotFoundException;
import org.ameba.http.MeasuredRestController;
import org.ameba.i18n.Translator;
import org.openwms.common.location.api.ErrorCodeVO;
import org.openwms.common.location.api.LocationVO;
import org.openwms.common.location.api.LockMode;
import org.openwms.common.location.api.LockType;
import org.openwms.common.location.api.ValidationGroups;
import org.openwms.core.SpringProfiles;
import org.openwms.core.http.AbstractWebController;
import org.openwms.core.http.Index;
import org.springframework.context.annotation.Profile;
import org.springframework.http.HttpHeaders;
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.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.io.Serializable;
import java.util.List;
import java.util.function.BiConsumer;

import static java.util.Arrays.asList;
import static org.openwms.common.CommonMessageCodes.LOCATION_ID_INVALID;
import static org.openwms.common.CommonMessageCodes.LOCATION_NOT_FOUND_BY_ERP_CODE;
import static org.openwms.common.CommonMessageCodes.LOCATION_NOT_FOUND_BY_ID;
import static org.openwms.common.CommonMessageCodes.LOCATION_NOT_FOUND_BY_PLC_CODE;
import static org.openwms.common.CommonMessageCodes.LOCK_MODE_UNSUPPORTED;
import static org.openwms.common.CommonMessageCodes.LOCK_TYPE_UNSUPPORTED;
import static org.openwms.common.location.api.LocationApiConstants.API_LOCATION;
import static org.openwms.common.location.api.LocationApiConstants.API_LOCATIONS;
import static org.springframework.hateoas.server.mvc.WebMvcLinkBuilder.linkTo;
import static org.springframework.hateoas.server.mvc.WebMvcLinkBuilder.methodOn;

/**
 * A LocationController.
 *
 * @author Heiko Scherrer
 */
@Profile("!" + SpringProfiles.IN_MEMORY)
@Validated
@MeasuredRestController
public class LocationController extends AbstractWebController {

    private final LocationMapper mapper;
    private final Translator translator;
    private final LocationService locationService;
    private final LocationRemovalManager locationRemovalManager;

    LocationController(LocationService locationService, LocationMapper mapper, Translator translator, LocationRemovalManager locationRemovalManager) {
        this.locationService = locationService;
        this.mapper = mapper;
        this.translator = translator;
        this.locationRemovalManager = locationRemovalManager;
    }

    @PostMapping(value = API_LOCATIONS)
    @Validated(ValidationGroups.Create.class)
    public ResponseEntity<LocationVO> createLocation(@Valid @RequestBody LocationVO location, HttpServletRequest req) {
        var created = locationService.create(mapper.convertVO(location));
        var result = mapper.convertToVO(created);
        addSelfLink(result);
        return ResponseEntity
                .created(super.getLocationURIForCreatedResource(req, created.getPersistentKey()))
                .header(HttpHeaders.CONTENT_TYPE, LocationVO.MEDIA_TYPE)
                .body(result);
    }

    @PutMapping(value = API_LOCATIONS)
    @Validated(ValidationGroups.Update.class)
    public ResponseEntity<LocationVO> updateLocation(@Valid @RequestBody LocationVO location) {
        var updated = locationService.save(mapper.convertVO(location));
        var result = mapper.convertToVO(updated);
        addSelfLink(result);
        return ResponseEntity.status(HttpStatus.OK).header(HttpHeaders.CONTENT_TYPE, LocationVO.MEDIA_TYPE).body(result);
    }

    @DeleteMapping(value = API_LOCATIONS + "/{pKey}")
    public ResponseEntity<Void> deleteLocation(@PathVariable("pKey") String pKey) {
        locationRemovalManager.tryDelete(pKey);
        return ResponseEntity.noContent().build();
    }

    @GetMapping(value = API_LOCATIONS + "/{pKey}")
    public ResponseEntity<LocationVO> findByPKey(@PathVariable("pKey") String pKey) {
        var location = locationService.findByPKey(pKey);
        var result = mapper.convertToVO(location);
        addSelfLink(result);
        return ResponseEntity.status(HttpStatus.OK).header(HttpHeaders.CONTENT_TYPE, LocationVO.MEDIA_TYPE).body(result);
    }

    private void addSelfLink(LocationVO result) {
        result.add(linkTo(methodOn(LocationController.class).findByPKey(result.getpKey())).withRel("location-findbypkey"));
    }

    @GetMapping(value = API_LOCATIONS, params = {"locationId"})
    public ResponseEntity<LocationVO> findByCoordinate(@RequestParam("locationId") String locationId) {
        if (!LocationPK.isValid(locationId)) {
            // here we need to throw an NFE because Feign needs to cast it into an Optional. IAE won't work!
            throw new NotFoundException(translator, LOCATION_ID_INVALID, new String[]{locationId}, locationId);
        }
        var location = locationService.findByLocationPk(LocationPK.fromString(locationId))
                .orElseThrow(() -> new NotFoundException(
                        translator,
                        LOCATION_NOT_FOUND_BY_ID,
                        new String[]{locationId},
                        locationId
                ));
        var result = mapper.convertToVO(location);
        addSelfLink(result);
        return ResponseEntity.status(HttpStatus.OK).header(HttpHeaders.CONTENT_TYPE, LocationVO.MEDIA_TYPE).body(result);
    }

    @GetMapping(value = API_LOCATIONS, params = {"erpCode"})
    public ResponseEntity<LocationVO> findByErpCode(@RequestParam("erpCode") String erpCode) {
        var location = locationService.findByErpCode(erpCode).orElseThrow(() -> locationNotFound(erpCode));
        var result = mapper.convertToVO(location);
        addSelfLink(result);
        return ResponseEntity.status(HttpStatus.OK).header(HttpHeaders.CONTENT_TYPE, LocationVO.MEDIA_TYPE).body(result);
    }

    @GetMapping(value = API_LOCATIONS, params = {"plcCode"})
    public ResponseEntity<LocationVO> findByPlcCode(@RequestParam("plcCode") String plcCode) {
        var location = locationService.findByPlcCode(plcCode)
                .orElseThrow(() -> new NotFoundException(
                        translator,
                        LOCATION_NOT_FOUND_BY_PLC_CODE,
                        new String[]{plcCode},
                        plcCode
                ));
        var result = mapper.convertToVO(location);
        addSelfLink(result);
        return ResponseEntity.status(HttpStatus.OK).header(HttpHeaders.CONTENT_TYPE, LocationVO.MEDIA_TYPE).body(result);
    }

    @GetMapping(value = API_LOCATIONS, params = {"locationGroupNames"})
    public ResponseEntity<List<LocationVO>> findForLocationGroups(
            @RequestParam("locationGroupNames") List<String> locationGroupNames) {
        var locations = locationService.findAllOf(locationGroupNames);
        var result = mapper.convertToVO(locations);
        result.forEach(this::addSelfLink);
        return ResponseEntity.status(HttpStatus.OK).header(HttpHeaders.CONTENT_TYPE, LocationVO.MEDIA_TYPE).body(result);
    }

    @PatchMapping(value = API_LOCATION + "/{pKey}", params = "op=change-state")
    public ResponseEntity<Void> changeState(
            @PathVariable(name = "pKey") String pKey,
            @RequestParam(name = "op") String op,
            @RequestBody ErrorCodeVO errorCode
    ) {
        locationService.changeState(pKey, errorCode);
        return ResponseEntity.noContent().build();
    }

    @PatchMapping(value = API_LOCATION, params = {"locationId", "op=change-state"})
    public ResponseEntity<Void> changeState(
            @RequestParam(name = "locationId") String locationId,
            @RequestBody ErrorCodeVO errorCode
    ) {
        locationService.changeState(LocationPK.fromString(locationId), errorCode);
        return ResponseEntity.noContent().build();
    }

    /**
     * Change the current {@code mode} a {@code Location}, identified by {@code erpCode}.
     *
     * @param erpCode The ERP code of the Location
     * @param type The type of lock to apply to the Location
     * @param mode The mode to apply to the Locations lock
     */
    @PostMapping(path = API_LOCATIONS , params = {"erpCode", "type", "mode"})
    public ResponseEntity<Void> changeState(
            @RequestParam("erpCode") String erpCode,
            @RequestParam("type") LockType type,
            @RequestParam("mode") LockMode mode,
            @RequestParam(value = "plcState", required = false) Integer plcState
    ) {
        var location = locationService.findByErpCode(erpCode).orElseThrow(() -> locationNotFound(erpCode));
        if (type == LockType.ALLOCATION_LOCK) {
            changeLocation(
                    mode,
                    location,
                    plcState,
                    (l, code) -> locationService.changeState(l.getPersistentKey(), code)
            );
        } else {
            unsupportedOperation(type);
        }
        return ResponseEntity.noContent().build();
    }

    @GetMapping(API_LOCATIONS)
    public ResponseEntity<List<LocationVO>> findByCoordinate(
            @RequestParam(value = "area", required = false, defaultValue = "%") String area,
            @RequestParam(value = "aisle", required = false, defaultValue = "%") String aisle,
            @RequestParam(value = "x", required = false, defaultValue = "%") String x,
            @RequestParam(value = "y", required = false, defaultValue = "%") String y,
            @RequestParam(value = "z", required = false, defaultValue = "%") String z
    ) {
        var pk = LocationPK.of(area, aisle, x, y, z);
        var result = mapper.convertToVO(locationService.findLocations(pk));
        result.forEach(this::addSelfLink);
        return result.isEmpty()
                ? ResponseEntity.notFound().build()
                : ResponseEntity.status(HttpStatus.OK).header(HttpHeaders.CONTENT_TYPE, LocationVO.MEDIA_TYPE).body(result);
    }

    @GetMapping(API_LOCATIONS + "/index")
    public ResponseEntity<Index> index() {
        return ResponseEntity.ok(
                new Index(
                        linkTo(methodOn(LocationController.class).createLocation(new LocationVO("locationId"), null)).withRel("location-create"),
                        linkTo(methodOn(LocationController.class).updateLocation(new LocationVO("locationId"))).withRel("location-updatelocation"),
                        linkTo(methodOn(LocationController.class).deleteLocation("pKey")).withRel("location-deletelocation"),
                        linkTo(methodOn(LocationController.class).findByPKey("pKey")).withRel("location-findbypkey"),
                        linkTo(methodOn(LocationController.class).findByCoordinate("AREA/AISLE/X/Y/Z")).withRel("location-findbycoordinate"),
                        linkTo(methodOn(LocationController.class).findByCoordinate("area", "aisle", "x", "y", "z")).withRel("location-findbycoordinate-wc"),
                        linkTo(methodOn(LocationController.class).findByErpCode("ERP_CODE")).withRel("location-findbyerpcode"),
                        linkTo(methodOn(LocationController.class).findByPlcCode("PLC_CODE")).withRel("location-findbyplccode"),
                        linkTo(methodOn(LocationController.class).findForLocationGroups(asList("LG1", "LG2"))).withRel("location-forlocationgroup"),
                        linkTo(methodOn(LocationController.class).changeState("pKey", "change-state", ErrorCodeVO.LOCK_STATE_IN_AND_OUT)).withRel("location-changestate")
                )
        );
    }

    private void changeLocation(LockMode mode, Target target, Integer plcState, BiConsumer<Target, ErrorCodeVO> fnc) {
        ErrorCodeVO state;
        switch (mode) {
            case IN -> {
                state = ErrorCodeVO.LOCK_STATE_IN;
                state.setPlcState(plcState);
                fnc.accept(target, state);
            }
            case OUT -> {
                state = ErrorCodeVO.LOCK_STATE_OUT;
                state.setPlcState(plcState);
                fnc.accept(target, state);
            }
            case IN_AND_OUT -> {
                state = ErrorCodeVO.LOCK_STATE_IN_AND_OUT;
                state.setPlcState(plcState);
                fnc.accept(target, state);
            }
            case NONE -> {
                state = ErrorCodeVO.UNLOCK_STATE_IN_AND_OUT;
                state.setPlcState(plcState);
                fnc.accept(target, state);
            }
            default -> unsupportedOperation(mode);
        }
    }

    private NotFoundException locationNotFound(String erpCode) {
        return new NotFoundException(translator, LOCATION_NOT_FOUND_BY_ERP_CODE, new String[]{erpCode}, erpCode);
    }

    private void unsupportedOperation(LockMode mode) {
        throw new BusinessRuntimeException(translator, LOCK_MODE_UNSUPPORTED, new Serializable[]{mode}, mode);
    }

    private void unsupportedOperation(LockType type) {
        throw new BusinessRuntimeException(translator, LOCK_TYPE_UNSUPPORTED, new Serializable[]{type}, type);
    }
}