LocationGroupController.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.NotFoundException;
import org.ameba.http.MeasuredRestController;
import org.ameba.i18n.Translator;
import org.openwms.common.SimpleLink;
import org.openwms.common.location.api.ErrorCodeTransformers;
import org.openwms.common.location.api.ErrorCodeVO;
import org.openwms.common.location.api.LocationGroupState;
import org.openwms.common.location.api.LocationGroupVO;
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.cloud.context.config.annotation.RefreshScope;
import org.springframework.context.annotation.Profile;
import org.springframework.http.ResponseEntity;
import org.springframework.transaction.annotation.Transactional;
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.RequestBody;
import org.springframework.web.bind.annotation.RequestParam;
import java.util.List;
import static java.util.Arrays.asList;
import static org.openwms.common.CommonMessageCodes.LOCATION_GROUP_NOT_FOUND;
import static org.openwms.common.location.api.LocationApiConstants.API_LOCATION_GROUP;
import static org.openwms.common.location.api.LocationApiConstants.API_LOCATION_GROUPS;
import static org.springframework.hateoas.server.mvc.WebMvcLinkBuilder.linkTo;
import static org.springframework.hateoas.server.mvc.WebMvcLinkBuilder.methodOn;
/**
* A LocationGroupController.
*
* @author Heiko Scherrer
*/
@Profile("!" + SpringProfiles.IN_MEMORY)
@Validated
@RefreshScope
@MeasuredRestController
public class LocationGroupController extends AbstractWebController {
public static final String PARENT = "parent";
private final Translator translator;
private final LocationGroupService locationGroupService;
private final LocationGroupMapper mapper;
private final ErrorCodeTransformers.GroupStateIn groupStateIn;
private final ErrorCodeTransformers.GroupStateOut groupStateOut;
LocationGroupController(Translator translator, LocationGroupService locationGroupService, LocationGroupMapper mapper,
ErrorCodeTransformers.GroupStateIn groupStateIn,
ErrorCodeTransformers.GroupStateOut groupStateOut) {
this.translator = translator;
this.locationGroupService = locationGroupService;
this.mapper = mapper;
this.groupStateIn = groupStateIn;
this.groupStateOut = groupStateOut;
}
/**
* Creates a new location group.
*
* @param vo The LocationGroupVO object containing the data for the new location group. The object must be
* validated using the annotated constraints specified in the ValidationGroups.Create interface.
* @return A ResponseEntity object containing the LocationGroupVO of the newly created location group,
* or an empty ResponseEntity if the creation failed.
*/
@PostMapping(API_LOCATION_GROUPS)
public ResponseEntity<LocationGroupVO> create(@Validated(ValidationGroups.Create.class) @Valid @RequestBody LocationGroupVO vo, HttpServletRequest req) {
var result = locationGroupService.create(vo);
var location = getLocationURIForCreatedResource(req, result.getPersistentKey());
return ResponseEntity.created(location).body(mapper.convertToVO(result));
}
@Transactional(readOnly = true)
@GetMapping(value = API_LOCATION_GROUPS, params = {"name"})
public LocationGroupVO findByName(
@RequestParam("name") String name
) {
var locationGroup = locationGroupService.findByName(name)
.orElseThrow(() -> new NotFoundException(translator, LOCATION_GROUP_NOT_FOUND, new String[]{name}, name));
var result = mapper.convertToVO(locationGroup);
if (locationGroup.hasParent()) {
result.add(new SimpleLink(linkTo(methodOn(LocationGroupController.class)
.findByName(locationGroup.getParent().getName())).withRel(PARENT)));
}
return result;
}
@Transactional(readOnly = true)
@GetMapping(value = API_LOCATION_GROUPS, params = {"names"})
public List<LocationGroupVO> findByNames(
@RequestParam("names") List<String> names
) {
var vos = mapper.convertToVO(locationGroupService.findByNames(names));
vos.forEach(lg -> {
if (lg.hasParent()) {
lg.add(new SimpleLink(linkTo(methodOn(LocationGroupController.class)
.findByName(lg.getParent())).withRel(PARENT)));
}
});
return vos;
}
@Transactional(readOnly = true)
@GetMapping(API_LOCATION_GROUPS)
public List<LocationGroupVO> findAll() {
try {
var result = mapper.convertToVO(locationGroupService.findAll());
result.forEach(lg -> {
if (lg.hasParent()) {
lg.add(new SimpleLink(linkTo(methodOn(LocationGroupController.class)
.findByName(lg.getParent())).withRel(PARENT)));
}
}
);
return result;
} catch (Exception e) {
e.printStackTrace();
throw new RuntimeException(e);
}
}
@PatchMapping(value = API_LOCATION_GROUPS, params = {"name", "op=change-state"})
public ResponseEntity<Void> changeGroupState(
@RequestParam(name = "name") String name,
@RequestParam(name = "op") String op,
@RequestBody ErrorCodeVO errorCode
) {
locationGroupService.changeGroupStates(
name,
groupStateIn.available(errorCode.getErrorCode()),
groupStateOut.available(errorCode.getErrorCode())
);
return ResponseEntity.ok().build();
}
@PatchMapping(value = API_LOCATION_GROUP + "/{pKey}", params = "op=change-state")
public ResponseEntity<Void> changeGroupState(
@PathVariable String pKey,
@RequestParam(name = "op") String op,
@RequestParam(name = "statein") LocationGroupState stateIn,
@RequestParam(name = "stateout") LocationGroupState stateOut
) {
locationGroupService.changeGroupState(pKey, stateIn, stateOut);
return ResponseEntity.ok().build();
}
@DeleteMapping(value = API_LOCATION_GROUPS + "/{pKey}")
public ResponseEntity<Void> deleteLocationGroup(@PathVariable("pKey") String pKey) {
locationGroupService.delete(pKey);
return ResponseEntity.noContent().build();
}
@PatchMapping(value = API_LOCATION_GROUPS + "/{pKey}")
public ResponseEntity<Void> modifyLocationGroup(
@PathVariable String pKey,
@RequestBody LocationGroupVO locationGroupVO
) {
locationGroupService.update(pKey, locationGroupVO);
return ResponseEntity.ok().build();
}
@GetMapping(API_LOCATION_GROUPS + "/index")
public ResponseEntity<Index> index() {
return ResponseEntity.ok(
new Index(
linkTo(methodOn(LocationGroupController.class).create(LocationGroupVO.create("FOO", "INFEED_AND_OUTFEED"), null)).withRel("location-groups-create"),
linkTo(methodOn(LocationGroupController.class).deleteLocationGroup("UUID")).withRel("location-groups-delete"),
linkTo(methodOn(LocationGroupController.class).findByName("FOO")).withRel("location-groups-findbyname"),
linkTo(methodOn(LocationGroupController.class).findByNames(asList("FOO", "BAR"))).withRel("location-groups-findbynames"),
linkTo(methodOn(LocationGroupController.class).findAll()).withRel("location-groups-findall"),
linkTo(methodOn(LocationGroupController.class).changeGroupState("UUID", "change-state", LocationGroupState.AVAILABLE, LocationGroupState.NOT_AVAILABLE)).withRel("location-groups-changestate"),
linkTo(methodOn(LocationGroupController.class).changeGroupState("FOO", "change-state", ErrorCodeVO.LOCK_STATE_IN_AND_OUT)).withRel("location-groups-changestate-with-bitmap"),
linkTo(methodOn(LocationGroupController.class).modifyLocationGroup("FOO", null)).withRel("location-groups-modify")
)
);
}
}