LocationGroupServiceImpl.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.CommonMessageCodes;
import org.openwms.common.account.AccountService;
import org.openwms.common.location.Location;
import org.openwms.common.location.LocationGroup;
import org.openwms.common.location.LocationGroupService;
import org.openwms.common.location.LocationRemovalManager;
import org.openwms.common.location.api.LocationGroupState;
import org.openwms.common.location.api.LocationGroupVO;
import org.openwms.common.location.api.ValidationGroups;
import org.openwms.common.location.api.events.LocationGroupEvent;
import org.openwms.common.location.events.DeletionFailedEvent;
import org.openwms.core.listener.RemovalNotAllowedException;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.context.ApplicationContext;
import org.springframework.validation.annotation.Validated;

import java.util.ArrayList;
import java.util.List;
import java.util.Optional;
import java.util.stream.Collectors;
import java.util.stream.Stream;

import static org.openwms.common.CommonMessageCodes.LOCATION_GROUP_EXISTS;
import static org.openwms.common.CommonMessageCodes.LOCATION_GROUP_NOT_FOUND;
import static org.openwms.common.CommonMessageCodes.LOCATION_GROUP_NOT_FOUND_BY_PKEY;

/**
 * A LocationGroupServiceImpl is a Spring managed transactional Service that operates on {@link LocationGroup} entities and spans the
 * tx boundary.
 *
 * @author Heiko Scherrer
 */
@Validated
@TxService
class LocationGroupServiceImpl implements LocationGroupService {

    private static final Logger LOGGER = LoggerFactory.getLogger(LocationGroupServiceImpl.class);
    private final ApplicationContext ctx;
    private final Translator translator;
    private final LocationGroupRepository repository;
    private final AccountService accountService;
    private final LocationRemovalManager locationRemovalManager;

    LocationGroupServiceImpl(ApplicationContext ctx, Translator translator, LocationGroupRepository repository, AccountService accountService, LocationRemovalManager locationRemovalManager) {
        this.ctx = ctx;
        this.translator = translator;
        this.repository = repository;
        this.accountService = accountService;
        this.locationRemovalManager = locationRemovalManager;
    }

    /**
     * {@inheritDoc}
     */
    @Override
    @Measured
    public @NotNull LocationGroup create(@NotNull @Validated(ValidationGroups.Create.class) @Valid LocationGroupVO vo) {
        if (LOGGER.isDebugEnabled()) {
            LOGGER.debug("Create a LocationGroup with the VO [{}]", vo.allFieldsToString());
        }
        var eo = createLocationGroup(vo);
        var savedEo = repository.save(eo);
        if (LOGGER.isInfoEnabled()) {
            LOGGER.info("Request to create a new LocationGroup [{}]", savedEo);
        }
        ctx.publishEvent(LocationGroupEvent.of(savedEo, LocationGroupEvent.LocationGroupEventType.CREATED));
        return savedEo;
    }

    private LocationGroup createLocationGroup(LocationGroupVO vo) {
        var eoOpt = repository.findByName(vo.getName());
        if (eoOpt.isPresent()) {
            throw new ResourceExistsException(translator, LOCATION_GROUP_EXISTS, new String[]{vo.getName()}, vo.getName());
        }
        var eo = new LocationGroup(vo.getName());
        if (vo.getAccountId() != null && !vo.getAccountId().isEmpty()) {
            var accountOpt = accountService.findByIdentifier(vo.getAccountId());
            if (accountOpt.isEmpty()) {
                throw new NotFoundException(translator, CommonMessageCodes.ACCOUNT_NOT_FOUND_BY_ID, new String[]{vo.getAccountId()}, vo.getAccountId());
            }
            eo.setAccount(accountOpt.get());
        }
        eo.setGroupType(vo.getGroupType());
        if (vo.getParent() != null && !vo.getParent().isEmpty()) {
            var parentOpt = repository.findByName(vo.getParent());
            if (parentOpt.isEmpty()) {
                throw new NotFoundException(translator, CommonMessageCodes.LOCATION_GROUP_NOT_FOUND, new String[]{vo.getParent()}, vo.getParent());
            }
            eo.setParent(parentOpt.get());
        }
        eo.setOperationMode(vo.getOperationMode());

        if (vo.getGroupStateIn() != null) {
            eo.changeGroupStateIn(vo.getGroupStateIn());
        }

        if (vo.getGroupStateOut() != null) {
            eo.changeGroupStateOut(vo.getGroupStateOut());
        }
        if (vo.getChildren() != null && !vo.getChildren().isEmpty()) {
            eo.setLocationGroups(vo.getChildren().stream().map(this::createLocationGroup).collect(Collectors.toSet()));
        }
        return eo;
    }

    /**
     * {@inheritDoc}
     */
    @Override
    @Measured
    public void changeGroupState(@NotBlank String pKey, @NotNull LocationGroupState stateIn, @NotNull LocationGroupState stateOut) {
        var locationGroup = findInternalByPKey(pKey);
        locationGroup.changeState(stateIn, stateOut);
        ctx.publishEvent(LocationGroupEvent.of(locationGroup, LocationGroupEvent.LocationGroupEventType.STATE_CHANGE));
    }

    /**
     * {@inheritDoc}
     */
    @Override
    @Measured
    public void changeGroupStates(@NotBlank String name, Optional<LocationGroupState> stateIn, Optional<LocationGroupState> stateOut) {
        var locationGroup = findByNameOrThrowInternal(name);
        stateIn.ifPresent(locationGroup::changeGroupStateIn);
        stateOut.ifPresent(locationGroup::changeGroupStateOut);
        if (stateIn.isPresent() || stateOut.isPresent()) {
            ctx.publishEvent(LocationGroupEvent.of(locationGroup, LocationGroupEvent.LocationGroupEventType.STATE_CHANGE));
        }
    }

    /**
     * {@inheritDoc}
     */
    @Override
    @Measured
    public void changeOperationMode(@NotBlank String name, @NotBlank String mode) {
        var locationGroup = findByNameOrThrowInternal(name);
        locationGroup.setOperationMode(mode);
        repository.save(locationGroup);
    }

    /**
     * {@inheritDoc}
     */
    @Override
    @Measured
    public Optional<LocationGroup> findByName(@NotBlank String name) {
        return repository.findByName(name);
    }

    /**
     * {@inheritDoc}
     */
    @Override
    @Measured
    public @NotNull LocationGroup findByNameOrThrow(@NotBlank String name) {
        return findByNameOrThrowInternal(name);
    }

    private LocationGroup findByNameOrThrowInternal(String name) {
        return repository.findByName(name).orElseThrow(() -> new NotFoundException(
                translator, LOCATION_GROUP_NOT_FOUND, new String[]{name}, name
        ));
    }

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

    /**
     * {@inheritDoc}
     */
    @Override
    @Measured
    public @NotNull List<LocationGroup> findByNames(@NotEmpty List<String> locationGroupNames) {
        var result = repository.findByNameIn(locationGroupNames);
        return result == null ? new ArrayList<>(0) : result;
    }

    /**
     * {@inheritDoc}
     */
    @Override
    @Measured
    public void delete(@NotBlank String pKey) {
        var locationGroup = repository.findBypKey(pKey).orElseThrow(() -> new NotFoundException(
                translator, LOCATION_GROUP_NOT_FOUND_BY_PKEY, new String[]{pKey}, pKey
        ));
        delete(locationGroup);
    }

    private Stream<String> getLocationKeys(LocationGroup locationGroup) {
        var result = locationGroup.getLocations().stream().map(Location::getPersistentKey);
        if (locationGroup.hasLocationGroups()) {
            for (var group : locationGroup.getLocationGroups()) {
                result = Stream.concat(result, getLocationKeys(group));
            }
        }
        return result;
    }

    private void delete(LocationGroup locationGroup) {
        LOGGER.info("Going to delete LocationGroup [{}]", locationGroup.getName());
        var allLocationKeys = getLocationKeys(locationGroup).toList();

        // first check all Locations of all LocationGroups if they're allowed to be deleted.
        // Check within the service if any TU is booked on the Locations and then ask foreign services if deletion is okay
        var allowedToDelete = locationRemovalManager.allowedToDelete(allLocationKeys);

        if (!allowedToDelete) {
           throw new RemovalNotAllowedException("At least one Location is not allowed to be deleted, therefore the LocationGroup [%s] cannot be deleted".formatted(locationGroup.getName()));
        }

        // Go and mark all for deletion...
        locationRemovalManager.markForDeletion(allLocationKeys);

        // Finally delete the LocationGroups...
        try {
            deleteOnlyGroups(locationGroup);
        } catch (Exception e) {
            LOGGER.error(e.getMessage(), e);
            LOGGER.debug("Deletion of LocationGroup [{}] went wrong, rolling back Location deletion", locationGroup.getPersistentKey());
            // if any failure occurs, send a persistent async message to release the deletion for everyone
            for (var pKey : allLocationKeys) {
                ctx.publishEvent(new DeletionFailedEvent(pKey));
            }
        }
    }

    private void deleteOnlyGroups(LocationGroup locationGroup) {
        if (locationGroup.hasLocationGroups()) {
            for (var group : locationGroup.getLocationGroups()) {
                deleteOnlyGroups(group);
            }
        }
        if (locationGroup.hasLocations()) {
            locationRemovalManager.deleteAll(locationGroup.getLocations());
        }
        repository.delete(locationGroup);
        LOGGER.debug("LocationGroup deleted [{}]", locationGroup.getPersistentKey());
    }

    /**
     * {@inheritDoc}
     */
    @Override
    @Measured
    public @NotNull LocationGroup update(@NotBlank String pKey, @NotNull LocationGroupVO locationGroupVO) {
        var locationGroup = findInternalByPKey(pKey);
        if (locationGroupVO.getDescription() != null && !locationGroupVO.getDescription().equals(locationGroup.getDescription())) {
            LOGGER.debug("Description of LocationGroup changes from [{}] to [{}]", locationGroup.getDescription(), locationGroupVO.getDescription());
            locationGroup.setDescription(locationGroupVO.getDescription());
            locationGroup = repository.save(locationGroup);
            ctx.publishEvent(LocationGroupEvent.of(locationGroup, LocationGroupEvent.LocationGroupEventType.CHANGED));
        }
        if (locationGroupVO.hasParent() && !locationGroupVO.getParent().equals(locationGroup.getParent().getName())) {
            var newParent = findByNameOrThrowInternal(locationGroupVO.getParent());
            LOGGER.debug("Parent of LocationGroup changes from [{}] to [{}]", locationGroup.getParent(), newParent);
            locationGroup.setParent(newParent);
            locationGroup = repository.save(locationGroup);
            ctx.publishEvent(LocationGroupEvent.of(locationGroup, LocationGroupEvent.LocationGroupEventType.CHANGED));
        }
        return locationGroup;
    }

    private LocationGroup findInternalByPKey(String pKey) {
        return repository.findBypKey(pKey).orElseThrow(() -> new NotFoundException(
                translator, LOCATION_GROUP_NOT_FOUND_BY_PKEY, new String[]{pKey}, pKey
        ));
    }
}