TransportUnitServiceImpl.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.impl;

import jakarta.validation.Valid;
import jakarta.validation.Validator;
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.exception.ServiceLayerException;
import org.ameba.i18n.Translator;
import org.openwms.common.CommonMessageCodes;
import org.openwms.common.StateChangeException;
import org.openwms.common.location.Location;
import org.openwms.common.location.LocationPK;
import org.openwms.common.location.LocationService;
import org.openwms.common.transport.TransportUnit;
import org.openwms.common.transport.TransportUnitMapper;
import org.openwms.common.transport.TransportUnitService;
import org.openwms.common.transport.TransportUnitType;
import org.openwms.common.transport.UnitError;
import org.openwms.common.transport.api.ValidationGroups;
import org.openwms.common.transport.api.commands.TUCommand;
import org.openwms.common.transport.api.messages.TransportUnitMO;
import org.openwms.common.transport.barcode.Barcode;
import org.openwms.common.transport.barcode.BarcodeGenerator;
import org.openwms.common.transport.events.TransportUnitEvent;
import org.openwms.common.transport.spi.NotApprovedException;
import org.openwms.common.transport.spi.TransportUnitMoveApproval;
import org.openwms.common.transport.spi.TransportUnitStateChangeApproval;
import org.openwms.core.exception.IllegalConfigurationValueException;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.context.ApplicationEventPublisher;
import org.springframework.context.event.EventListener;
import org.springframework.transaction.annotation.Propagation;
import org.springframework.transaction.annotation.Transactional;
import org.springframework.util.Assert;
import org.springframework.validation.annotation.Validated;

import java.io.Serializable;
import java.util.ArrayList;
import java.util.List;
import java.util.function.Supplier;

import static java.lang.String.format;
import static org.ameba.system.ValidationUtil.validate;
import static org.openwms.common.transport.api.commands.TUCommand.Type.REMOVING;

/**
 * A TransportUnitServiceImpl is a Spring managed bean that deals with TransportUnits.
 *
 * @author Heiko Scherrer
 */
@Validated
@TxService
class TransportUnitServiceImpl implements TransportUnitService {

    private static final Logger LOGGER = LoggerFactory.getLogger(TransportUnitServiceImpl.class);
    private static final String NO_LOCATION_SET = "The actualLocation must be given in order to create a TransportUnit";
    private static final String NO_BARCODE = "The barcode must be given in order to create a TransportUnit";
    private static final String NO_TRANSPORT_UNIT_TYPE = "The transportUnitType must be given in order to create a TransportUnit";

    private final ApplicationEventPublisher publisher;
    private final Validator validator;
    private final Translator translator;
    private final BarcodeGenerator barcodeGenerator;
    private final TransportUnitMapper mapper;
    private final TransportUnitRepository repository;
    private final TransportUnitTypeRepository transportUnitTypeRepository;
    private final TransportUnitStateChangeApproval stateChangeApproval;
    private final TransportUnitMoveApproval moveApproval;
    private final LocationService locationService;
    private final String deleteTransportUnitMode;

    @SuppressWarnings("squid:S107")
    TransportUnitServiceImpl(ApplicationEventPublisher publisher, Validator validator, Translator translator,
            BarcodeGenerator barcodeGenerator, TransportUnitMapper mapper, TransportUnitRepository repository,
            TransportUnitTypeRepository transportUnitTypeRepository,
            @Autowired(required = false) TransportUnitStateChangeApproval stateChangeApproval,
            @Autowired(required = false) TransportUnitMoveApproval moveApproval,
            LocationService locationService, @Value("${owms.common.delete-transport-unit-mode}") String deleteTransportUnitMode) {
        this.publisher = publisher;
        this.validator = validator;
        this.translator = translator;
        this.barcodeGenerator = barcodeGenerator;
        this.mapper = mapper;
        this.repository = repository;
        this.transportUnitTypeRepository = transportUnitTypeRepository;
        this.stateChangeApproval = stateChangeApproval;
        this.moveApproval = moveApproval;
        this.locationService = locationService;
        this.deleteTransportUnitMode = deleteTransportUnitMode;
    }

    /**
     * {@inheritDoc}
     */
    @Override
    @Measured
    public @NotNull TransportUnit create(
            @NotBlank String transportUnitBK, @NotNull TransportUnitType transportUnitType,
            @NotNull LocationPK actualLocation, Boolean strict) {
        Assert.notNull(transportUnitBK, NO_BARCODE);
        Assert.notNull(transportUnitType, NO_TRANSPORT_UNIT_TYPE);
        Assert.notNull(actualLocation, NO_LOCATION_SET);
        return createInternal(
                barcodeGenerator.convert(transportUnitBK),
                transportUnitType.getType(),
                strict,
                () -> locationService.findByLocationPk(actualLocation)
                        .orElseThrow(() -> new NotFoundException(format("No Location with locationPk [%s] found", actualLocation)))
        );
    }

    /**
     * {@inheritDoc}
     */
    @Override
    @Measured
    public @NotNull TransportUnit create(
            @NotBlank String transportUnitBK, @NotBlank String transportUnitType,
            @NotBlank String actualLocation, Boolean strict) {
        Assert.notNull(actualLocation, NO_LOCATION_SET);

        return createInternal(
                barcodeGenerator.convert(transportUnitBK),
                transportUnitType,
                strict,
                () -> locationService.findByLocationId(actualLocation)
                        .orElseThrow(() -> new NotFoundException(format("No Location with actual location [%s] found", actualLocation)))
        );
    }

    /**
     * {@inheritDoc}
     */
    @Override
    @Measured
    public @NotNull TransportUnit createNew(@NotBlank String transportUnitType, @NotBlank String actualLocation) {
        var nextBarcode = barcodeGenerator.generate(transportUnitType, actualLocation);
        return createInternal(
                nextBarcode,
                transportUnitType,
                false,
                () -> locationService.findByLocationId(actualLocation)
                        .orElseThrow(() -> new NotFoundException(format("No Location with actual location [%s] found", actualLocation)))
        );
    }

    private TransportUnit createInternal(Barcode barcode, String transportUnitType, Boolean strict, Supplier<Location> locationResolver) {
        Assert.notNull(barcode, NO_BARCODE);
        Assert.hasText(transportUnitType, NO_TRANSPORT_UNIT_TYPE);

        var optTransportUnit = repository.findByBarcode(barcode);
        if (strict == null || Boolean.FALSE.equals(strict)) {
            if (optTransportUnit.isPresent()) {
                if (LOGGER.isDebugEnabled()) {
                    LOGGER.debug("TransportUnit with Barcode [{}] already exists, silently returning the existing one", barcode);
                }
                return optTransportUnit.get();
            }
        } else {
            optTransportUnit.ifPresent(tu -> {
                throw new ResourceExistsException(format("TransportUnit with id [%s] already exists", barcode));
            });
        }

        var actualLocation = locationResolver.get();
        var type = transportUnitTypeRepository.findByType(transportUnitType).orElseThrow(() -> new NotFoundException(format("TransportUnitType [%s] not found", transportUnitType)));
        if (LOGGER.isDebugEnabled()) {
            LOGGER.debug("Creating a TransportUnit with Barcode [{}] of Type [{}] on Location [{}]", barcode, transportUnitType, actualLocation);
        }
        var transportUnit = repository.save(new TransportUnit(barcode, type, actualLocation));
        publisher.publishEvent(
                TransportUnitEvent.newBuilder()
                        .tu(transportUnit)
                        .type(TransportUnitEvent.TransportUnitEventType.CREATED)
                        .build()
        );
        return transportUnit;
    }

    /**
     * {@inheritDoc}
     */
    @Override
    @Validated(ValidationGroups.TransportUnit.Update.class)
    @Measured
    public @NotNull TransportUnit update(@NotNull Barcode barcode, final @Valid @NotNull TransportUnit tu) {
        if (!barcode.equals(tu.getBarcode())) {
            throw new ServiceLayerException("Mismatch between Barcode and tu.Barcode in API");
        }
        var existing = findByBarcodeInternal(barcode);
        var updated = new TransportUnit(barcode);
        updated.setTransportUnitType(existing.getTransportUnitType());
        mapper.copy(existing, updated);
        if (tu.getActualLocation() !=  null && tu.getActualLocation().isNew()) {
            moveInternal(existing, this.locationService.findByLocationPk(tu.getActualLocation().getLocationId())
                    .orElseThrow(() -> new NotFoundException(format("Location [%s] not found", tu.getActualLocation()))));
        }
        var saved = repository.save(existing);
        publisher.publishEvent(TransportUnitEvent.newBuilder()
                .tu(saved)
                .type(TransportUnitEvent.TransportUnitEventType.CHANGED)
                .build()
        );
        return saved;
    }

    private void approveMove(TransportUnit transportUnit, Location newLocation) {
        if (moveApproval == null) {
            return;
        }
        try {
            moveApproval.approve(transportUnit, newLocation);
        } catch (NotApprovedException nae) {
            LOGGER.error(nae.getMessage(), nae);
            throw new StateChangeException("Not allowed to move the TransportUnit [%s] to [%s]".formatted(transportUnit.getBarcode(), newLocation));
        }
    }

    private TransportUnit moveInternal(TransportUnit transportUnit, Location target) {
        if (transportUnit.getActualLocation().getLocationId().equals(target.getLocationId())) {
            LOGGER.debug("TransportUnit [{}] is already booked on Location [{}]", transportUnit.getBarcode(), target.getLocationId());
            return transportUnit;
        }
        approveMove(transportUnit, target);
        transportUnit.setActualLocation(target);
        if (LOGGER.isInfoEnabled()) {
            LOGGER.info("Moving TransportUnit with barcode [{}] from Location [{}] to Location [{}]", transportUnit.getBarcode(),
                    transportUnit.getActualLocation(), target.getLocationId());
        }
        var saved = repository.save(transportUnit);
        publisher.publishEvent(
                TransportUnitEvent.newBuilder()
                        .tu(saved)
                        .type(TransportUnitEvent.TransportUnitEventType.MOVED)
                        .previousLocation(transportUnit.getActualLocation())
                        .actualLocation(transportUnit.getActualLocation())
                        .build()
        );
        return saved;
    }

    /**
     * {@inheritDoc}
     */
    @SuppressWarnings("javasecurity:S5145")
    @Override
    @Measured
    public @NotNull TransportUnit moveTransportUnit(@NotNull Barcode barcode, @NotBlank String targetLocation) {
        var transportUnit = findByBarcodeInternal(barcode);
        var target = LocationPK.isValid(targetLocation)
                ? locationService.findByLocationPk(LocationPK.fromString(targetLocation))
                .orElseThrow(() -> new NotFoundException(format("No Location with LocationPk [%s] found", LocationPK.fromString(targetLocation))))
                : locationService.findByErpCode(targetLocation).orElseGet(() -> locationService.findByPlcCode(targetLocation)
                .orElseThrow(() -> new NotFoundException(format("No Location with LocationPk [%s] found", LocationPK.fromString(targetLocation)))));
        return moveInternal(transportUnit, target);
    }

    /**
     * {@inheritDoc}
     */
    @Override
    @Measured
    public void deleteTransportUnits(@NotNull List<TransportUnit> transportUnits) {
        if (!transportUnits.isEmpty()) {
            var tus = new ArrayList<>(transportUnits);
            tus.sort((o1, o2) -> {
                if (o1.hasChildren() && o2.hasChildren() ||
                        !o1.hasChildren() && !o2.hasChildren()) {
                    return 0;
                } else if (!o1.hasChildren() && o2.hasChildren()) {
                    return -1;
                } else {
                    return 1;
                }
            });
            tus.forEach(this::delete);
        }
    }

    /**
     * {@inheritDoc}
     */
    @Override
    @Measured
    public void delete(String pKey) {
        var existing = this.findByPKeyInternal(pKey);
        this.delete(existing);
    }

    /* we expect that the calling service spans the TX here, because the EL advice may come differently in the chain. */
    @Transactional(propagation = Propagation.MANDATORY)
    @EventListener
    public void onEvent(TUCommand command) {
        if (command.getType() == TUCommand.Type.REMOVE) {
            validate(validator, command, ValidationGroups.TransportUnit.Remove.class);
            if (LOGGER.isDebugEnabled()) {
                LOGGER.debug("Got command to REMOVE TransportUnit with pKey [{}]", command.getTransportUnit().getpKey());
            }
            deleteDefinitely(command.getTransportUnit().getpKey());
        }
    }

    private void deleteDefinitely(String pKey) {
        repository.findByPKey(pKey).ifPresent(tu -> {
            repository.delete(tu);
            publisher.publishEvent(
                    TransportUnitEvent.newBuilder()
                            .tu(tu)
                            .type(TransportUnitEvent.TransportUnitEventType.DELETED)
                            .build());
        });
    }

    private void delete(TransportUnit transportUnit) {
        if ("strict".equalsIgnoreCase(deleteTransportUnitMode)) {
            deleteDefinitely(transportUnit.getPersistentKey());
        } else if ("on-accept".equalsIgnoreCase(deleteTransportUnitMode)) {
            var mo = TransportUnitMO.newBuilder()
                    .withPKey(transportUnit.getPersistentKey())
                    .withBarcode(transportUnit.getBarcode().getValue())
                    .build();
            publisher.publishEvent(TUCommand.newBuilder(REMOVING)
                    .withTransportUnit(mo)
                    .build()
            );
        } else {
            throw new IllegalConfigurationValueException(format("Configuration value [owms.common.delete-transport-unit-mode] is configured with invalid value [%s]", deleteTransportUnitMode));
        }
    }

    /**
     * {@inheritDoc}
     */
    @Override
    @Measured
    public @NotNull TransportUnit findByBarcode(@NotBlank String transportUnitBK) {
        return findByBarcodeInternal(barcodeGenerator.convert(transportUnitBK));
    }

    private TransportUnit findByBarcodeInternal(Barcode barcode) {
        return repository.findByBarcode(barcode)
                .orElseThrow(() -> new NotFoundException(translator, CommonMessageCodes.TU_BARCODE_NOT_FOUND, new Serializable[]{barcode}, barcode));
    }

    /**
     * {@inheritDoc}
     */
    @Override
    @Measured
    public @NotNull List<TransportUnit> findByBarcodes(@NotEmpty List<Barcode> barcodes) {
        var tus = repository.findByBarcodeIn(barcodes);
        return tus == null ? new ArrayList<>(0) : tus;
    }

    /**
     * {@inheritDoc}
     */
    @Override
    @Measured
    public @NotNull List<TransportUnit> findOnLocation(@NotBlank String actualLocation) {
        Assert.hasText(actualLocation, NO_LOCATION_SET);
        var location = locationService.findByLocationIdOrThrow(actualLocation);
        return repository.findByActualLocationOrderByActualLocationDate(location);
    }

    /**
     * {@inheritDoc}
     */
    @Override
    @Measured
    public @NotNull TransportUnit findByPKey(@NotBlank String pKey) {
        return this.findByPKeyInternal(pKey);
    }

    private TransportUnit findByPKeyInternal(String pKey) {
        return repository.findByPKey(pKey).orElseThrow(() -> new NotFoundException(format("No TransportUnit with pKey [%s] found", pKey)));
    }

    /**
     * {@inheritDoc}
     */
    @Override
    @Measured
    public void addError(@NotBlank String transportUnitBK, @NotNull UnitError unitError) {
        var tu = this.findByBarcodeInternal(barcodeGenerator.convert(transportUnitBK));
        tu.addError(unitError);
    }

    /**
     * {@inheritDoc}
     */
    @Override
    @Measured
    public @NotNull List<TransportUnit> findAll() {
        return repository.findAll();
    }

    /**
     * {@inheritDoc}
     */
    @Override
    @Measured
    public TransportUnit changeTarget(@NotNull Barcode barcode, @NotBlank String targetLocation) {
        var transportUnit = findByBarcodeInternal(barcode);
        var location = locationService.findByLocationIdOrThrow(targetLocation);
        transportUnit.setTargetLocation(location);
        var saved = repository.save(transportUnit);
        publisher.publishEvent(TransportUnitEvent.newBuilder()
                .tu(saved)
                .type(TransportUnitEvent.TransportUnitEventType.CHANGED).build()
        );
        return saved;
    }

    /**
     * {@inheritDoc}
     */
    @Override
    @Measured
    public void setState(@NotBlank String transportUnitBK, @NotBlank String newState) {
        var transportUnit = findByBarcodeInternal(barcodeGenerator.convert(transportUnitBK));
        approveStateChange(transportUnit, newState);
        LOGGER.debug("Setting state of TransportUnit [{}] to [{}]", transportUnitBK, newState);
        transportUnit.setState(newState);
        publisher.publishEvent(TransportUnitEvent.newBuilder()
                .tu(transportUnit)
                .type(TransportUnitEvent.TransportUnitEventType.STATE_CHANGE).build()
        );
        repository.save(transportUnit);
    }

    private void approveStateChange(TransportUnit transportUnit, String newState) {
        if (stateChangeApproval == null) {
            return;
        }
        try {
            stateChangeApproval.approve(transportUnit, newState);
        } catch (NotApprovedException nae) {
            LOGGER.error(nae.getMessage(), nae);
            throw new StateChangeException("Not allowed to change the state of TransportUnit [%s] to [%s]".formatted(transportUnit.getBarcode(), newState));
        }
    }

    /**
     * {@inheritDoc}
     */
    @Override
    @Measured
    public void synchronizeTransportUnits() {
        var all = repository.findAll();
        all.forEach(tu -> publisher.publishEvent(
                TransportUnitEvent.newBuilder()
                        .tu(tu)
                        .type(TransportUnitEvent.TransportUnitEventType.CREATED)
                        .build()
        ));
    }
}