LocationServiceImpl.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.impl;
import jakarta.validation.Valid;
import jakarta.validation.constraints.NotBlank;
import jakarta.validation.constraints.NotEmpty;
import jakarta.validation.constraints.NotNull;
import org.ameba.annotation.Measured;
import org.ameba.annotation.TxService;
import org.ameba.exception.NotFoundException;
import org.ameba.exception.ResourceExistsException;
import org.ameba.i18n.Translator;
import org.openwms.common.location.Location;
import org.openwms.common.location.LocationMapper;
import org.openwms.common.location.LocationPK;
import org.openwms.common.location.LocationService;
import org.openwms.common.location.api.ErrorCodeTransformers;
import org.openwms.common.location.api.ErrorCodeVO;
import org.openwms.common.location.api.events.LocationEvent;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.context.ApplicationContext;
import org.springframework.transaction.annotation.Transactional;
import org.springframework.validation.annotation.Validated;
import java.io.Serializable;
import java.util.Collections;
import java.util.List;
import java.util.Optional;
import static org.openwms.common.CommonMessageCodes.LOCATION_ID_EXISTS;
import static org.openwms.common.CommonMessageCodes.LOCATION_ID_INVALID;
import static org.openwms.common.CommonMessageCodes.LOCATION_NOT_FOUND_BY_ID;
import static org.openwms.common.CommonMessageCodes.LOCATION_NOT_FOUND_BY_PKEY;
/**
* A LocationServiceImpl is a Spring managed transactional Service that operates on {@link Location} entities and spans the tx boundary.
*
* @author Heiko Scherrer
*/
@Validated
@TxService
class LocationServiceImpl implements LocationService {
private static final Logger LOGGER = LoggerFactory.getLogger(LocationServiceImpl.class);
private final Translator translator;
private final LocationMapper locationMapper;
private final LocationRepository repository;
private final ErrorCodeTransformers.LocationStateIn stateInTransformer;
private final ErrorCodeTransformers.LocationStateOut stateOutTransformer;
private final ApplicationContext ctx;
LocationServiceImpl(Translator translator, LocationMapper locationMapper, LocationRepository repository,
ErrorCodeTransformers.LocationStateIn stateInTransformer, ErrorCodeTransformers.LocationStateOut stateOutTransformer,
ApplicationContext ctx) {
this.translator = translator;
this.locationMapper = locationMapper;
this.repository = repository;
this.stateInTransformer = stateInTransformer;
this.stateOutTransformer = stateOutTransformer;
this.ctx = ctx;
}
/**
* {@inheritDoc}
*/
@Override
@Measured
public @NotNull Location create(@NotNull @Valid Location location) {
var locationOpt = repository.findByLocationId(location.getLocationId());
if (location.hasLocationId() && locationOpt.isPresent()) {
throw new ResourceExistsException(translator, LOCATION_ID_EXISTS,
new Serializable[]{location.getLocationId()},
location.getLocationId());
}
var created = repository.save(location);
ctx.publishEvent(LocationEvent.of(created, LocationEvent.LocationEventType.CREATED));
return created;
}
/**
* {@inheritDoc}
*/
@Override
@Measured
public @NotNull Location findByPKey(@NotBlank String pKey) {
return findInternal(pKey);
}
private Location findInternal(String pKey) {
return repository
.findBypKey(pKey)
.orElseThrow(() -> new NotFoundException(translator, LOCATION_NOT_FOUND_BY_PKEY, new String[]{pKey}, pKey));
}
/**
* {@inheritDoc}
*/
@Override
@Measured
@Transactional(readOnly = true)
public Optional<Location> findByLocationPk(@NotNull LocationPK locationId) {
return repository.findByLocationId(locationId);
}
private Location findByLocationPkOrThrow(LocationPK locationId) {
return repository.findByLocationId(locationId).orElseThrow(() -> new NotFoundException(
translator,
LOCATION_NOT_FOUND_BY_ID,
new Object[]{locationId},
locationId
));
}
/**
* {@inheritDoc}
*/
@Override
@Measured
@Transactional(readOnly = true)
public Optional<Location> findByPlcCode(@NotBlank String plcCode) {
return repository.findByPlcCode(plcCode);
}
/**
* {@inheritDoc}
*/
@Override
@Measured
public Optional<Location> findByLocationId(@NotBlank String locationId) {
return findByLocationIdInternal(locationId);
}
private Optional<Location> findByLocationIdInternal(String locationId) {
if (!LocationPK.isValid(locationId)) {
throw new IllegalArgumentException(translator.translate(LOCATION_ID_INVALID, locationId));
}
return repository.findByLocationId(LocationPK.fromString(locationId));
}
/**
* {@inheritDoc}
*/
@Override
@Measured
public @NotNull Location findByLocationIdOrThrow(@NotBlank String locationId) {
return findByLocationIdInternal(locationId).orElseThrow(() -> new NotFoundException(
translator,
LOCATION_NOT_FOUND_BY_ID,
new String[]{locationId},
locationId
));
}
/**
* {@inheritDoc}
*/
@Override
@Measured
public @NotNull List<Location> findAllOf(@NotEmpty List<String> locationGroupNames) {
return locationGroupNames.size() == 1
? repository.findByLocationGroup_Name(locationGroupNames.get(0))
: repository.findByLocationGroup_Name(locationGroupNames);
}
/**
* {@inheritDoc}
*/
@Override
@Measured
public void changeState(@NotBlank String pKey, @NotNull ErrorCodeVO errorCode) {
var location = findInternal(pKey);
changeStateInternal(location, errorCode);
}
private void changeStateInternal(Location location, ErrorCodeVO errorCode) {
boolean changed = false;
if (Optional.ofNullable(errorCode.getPlcState()).isPresent() && errorCode.getPlcState() != location.getPlcState()) {
location.setPlcState(errorCode.getPlcState());
LOGGER.info("PLC state of location [{}] has been updated to [{}]", location.getLocationId(), errorCode.getPlcState());
changed = true;
}
var infeedAvailable = stateInTransformer.available(errorCode.getErrorCode());
if (infeedAvailable.isPresent() &&
//location.getLocationGroup().isInfeedAllowed() &&
!infeedAvailable.get().equals(location.isInfeedActive())) {
location.setInfeed(infeedAvailable.get());
LOGGER.info("Incoming active of location [{}] has been updated to [{}]", location.getLocationId(), infeedAvailable.get());
changed = true;
}
var outfeedAvailable = stateOutTransformer.available(errorCode.getErrorCode());
if (outfeedAvailable.isPresent() &&
//location.getLocationGroup().isOutfeedAllowed() &&
!outfeedAvailable.get().equals(location.isOutfeedActive())) {
location.setOutfeed(outfeedAvailable.get());
LOGGER.info("Outgoing active of location [{}] has been updated to [{}]", location.getLocationId(), outfeedAvailable.get());
changed = true;
}
if (changed) {
// don't send twice only if one has changed
ctx.publishEvent(LocationEvent.of(location, LocationEvent.LocationEventType.STATE_CHANGE));
}
}
/**
* {@inheritDoc}
*/
@Override
@Measured
public void changeState(@NotNull LocationPK locationId, @NotNull ErrorCodeVO errorCode) {
var location = findByLocationPkOrThrow(locationId);
changeStateInternal(location, errorCode);
}
/**
* {@inheritDoc}
*/
@Override
@Measured
public @NotNull List<Location> findLocations(@NotNull LocationPK locationPK) {
var result = repository.findByLocationIdContaining(locationPK);
return result == null ? Collections.emptyList() : result;
}
/**
* {@inheritDoc}
*/
@Override
@Measured
public Optional<Location> findByErpCode(@NotBlank String erpCode) {
return repository.findByErpCode(erpCode);
}
/**
* {@inheritDoc}
*/
@Override
@Measured
public @NotNull Location save(@NotNull Location location) {
var existing = findInternal(location.getPersistentKey());
var modified = locationMapper.copyForUpdate(location, existing);
if (LOGGER.isDebugEnabled()) {
LOGGER.debug("Saving Location [{}]", modified);
}
return repository.save(modified);
}
}