LocationRemovalManagerImpl.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.constraints.NotBlank;
import jakarta.validation.constraints.NotNull;
import org.ameba.annotation.Measured;
import org.ameba.annotation.TxService;
import org.ameba.exception.NotFoundException;
import org.ameba.i18n.Translator;
import org.openwms.common.location.Location;
import org.openwms.common.location.LocationRemovalManager;
import org.openwms.common.location.api.events.LocationEvent;
import org.openwms.common.location.events.DeletionFailedEvent;
import org.openwms.common.location.impl.registration.RegistrationService;
import org.openwms.common.location.impl.registration.ReplicaRegistry;
import org.openwms.core.listener.RemovalNotAllowedException;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.context.ApplicationEventPublisher;
import org.springframework.http.HttpEntity;
import org.springframework.http.HttpHeaders;
import org.springframework.http.HttpMethod;
import org.springframework.web.client.RestTemplate;
import java.util.ArrayList;
import java.util.Collection;
import java.util.List;
import static java.lang.String.format;
import static org.openwms.common.CommonMessageCodes.LOCATION_NOT_FOUND_BY_PKEY;
/**
* A LocationRemovalManagerImpl.
*
* @author Heiko Scherrer
*/
@TxService
class LocationRemovalManagerImpl implements LocationRemovalManager {
private static final Logger LOGGER = LoggerFactory.getLogger(LocationRemovalManagerImpl.class);
private final ApplicationEventPublisher eventPublisher;
private final Translator translator;
private final RestTemplate aLoadBalanced;
private final LocationRepository repository;
private final RegistrationService registrationService;
LocationRemovalManagerImpl(ApplicationEventPublisher eventPublisher, Translator translator, RestTemplate aLoadBalanced,
LocationRepository repository, RegistrationService registrationService) {
this.eventPublisher = eventPublisher;
this.translator = translator;
this.aLoadBalanced = aLoadBalanced;
this.repository = repository;
this.registrationService = registrationService;
}
/**
* {@inheritDoc}
*/
@Override
@Measured
public void tryDelete(@NotBlank String pKey) {
var locationOpt = repository.findBypKey(pKey);
if (locationOpt.isEmpty()) {
LOGGER.warn("Location with pKey [{}] shall be deleted but it does not exist", pKey);
throw new NotFoundException(translator, LOCATION_NOT_FOUND_BY_PKEY, new String[]{pKey}, pKey);
}
try {
deleteInternal(locationOpt.get());
} catch (Exception e) {
// if any failure occurs, send a persistent async message to release the deletion for everyone
eventPublisher.publishEvent(new DeletionFailedEvent(locationOpt.get().getPersistentKey()));
throw new RemovalNotAllowedException(e.getMessage());
}
}
/**
* {@inheritDoc}
*/
@Override
@Measured
public void deleteAll(@NotNull Collection<Location> locations) {
repository.deleteAll(locations);
}
private void deleteInternal(Location location) {
verifyNoTransportUnitsOn(location);
var registeredServices = registrationService.getAllRegistered();
if (registeredServices.isEmpty()) {
LOGGER.info("No services registered that are requested for removal");
}
// first ask all services and call the requestRemovalEndpoint
// might throw RemovalNotAllowedException and exit
for (var srv : registeredServices) {
askForRemoval(srv, List.of(location.getPersistentKey()));
}
// if all are fine then call the removeEndpoint to mark the Location as deleted and not be visible in the foreign service
for (var srv : registeredServices) {
removeLocation(srv, List.of(location.getPersistentKey()));
}
deleteLocation(location);
}
private void verifyNoTransportUnitsOn(Location location) {
if (repository.doesTUonLocationExists(location.getPk())) {
throw new RemovalNotAllowedException("TransportUnit exist on Location [%s]".formatted(location.getLocationId()));
}
LOGGER.info("No TransportUnits exist on Location [{}]", location.getLocationId());
}
private void verifyNoTransportUnitsOn(Collection<String> pKeys) {
if (repository.doesTUonLocationExists(pKeys)) {
throw new RemovalNotAllowedException("TransportUnit exist on one or more Locations");
}
}
private void deleteLocation(Location location) {
verifyNoTransportUnitsOn(location);
// if ALL still agree, then send a persistent async message to commit the deletion after the transaction
eventPublisher.publishEvent(LocationEvent.of(location, LocationEvent.LocationEventType.DELETED));
// and finally delete the Location
repository.delete(location);
}
/**
* {@inheritDoc}
*
* At first the implementation checks if any {@code TransportUnit} is booked onto one of the {@link Location}s and as second all parties
* of interest are asked if deletion in the current moment is okay for them, too.
*/
@Override
@Measured
public boolean allowedToDelete(@NotNull Collection<String> pKeys) {
verifyNoTransportUnitsOn(pKeys);
var registeredServices = registrationService.getAllRegistered();
try {
// first ask all services and call the requestRemovalEndpoint
// might throw RemovalNotAllowedException and exit
for (var srv : registeredServices) {
askForRemoval(srv, pKeys);
}
LOGGER.debug("It's allowed to delete the given Locations");
return true;
} catch (Exception e) {
LOGGER.warn(e.getMessage(), e);
LOGGER.debug("It's NOT allowed to delete the given Locations");
return false;
}
}
/**
* {@inheritDoc}
*
* Call interesing parties to mark the {@link Location}s for deletion.
*/
@Override
@Measured
public void markForDeletion(@NotNull Collection<String> pKeys) {
var registeredServices = registrationService.getAllRegistered();
try {
for (var srv : registeredServices) {
removeLocation(srv, pKeys);
}
LOGGER.debug("All Locations marked for deletion");
} catch (Exception e) {
// if any failure occurs, send a persistent async message to release the deletion for everyone
LOGGER.debug("Location cannot be marked for deletion, sending events to rollback");
for (var pKey : pKeys) {
eventPublisher.publishEvent(new DeletionFailedEvent(pKey));
}
throw e;
}
}
private void askForRemoval(ReplicaRegistry srv, Collection<String> pKeys) {
var headers = new HttpHeaders();
boolean result;
var endpoint = "http://" + srv.getApplicationName() + srv.getRequestRemovalEndpoint();
if (LOGGER.isDebugEnabled()) {
LOGGER.debug("Request for removal to the service with API [{}]", endpoint);
}
try {
var response = aLoadBalanced.exchange(
endpoint,
HttpMethod.POST,
new HttpEntity<List<String>>(new ArrayList<>(pKeys), headers),
Boolean.class
);
result = response.getBody() != null && response.getBody();
if (result) {
LOGGER.info("Service [{}] allows to remove all Locations", srv.getApplicationName());
} else {
LOGGER.info("Service [{}] does not allow to remove Locations", srv.getApplicationName());
}
} catch (Exception e) {
LOGGER.error(e.getMessage(), e);
throw new RemovalNotAllowedException("Exception. Removal of Locations is declined by service [%s]".formatted(srv.getApplicationName()));
}
if (!result) {
throw new RemovalNotAllowedException("Removal of Locations has been declined by service [%s]".formatted(srv.getApplicationName()));
}
}
private void removeLocation(ReplicaRegistry srv, Collection<String> pKeys) {
var headers = new HttpHeaders();
var endpoint = "http://" + srv.getApplicationName() + srv.getRemovalEndpoint();
if (LOGGER.isDebugEnabled()) {
LOGGER.debug("Ask for removal the service with API [{}]", endpoint);
}
try {
aLoadBalanced.exchange(
endpoint,
HttpMethod.DELETE,
new HttpEntity<List<String>>(new ArrayList<>(pKeys), headers),
Void.class
);
LOGGER.info("Service [{}] removed Locations", srv.getApplicationName());
} catch (Exception e) {
throw new RemovalNotAllowedException(format("Exception. Removal of Locations is declined by service [%s]", srv.getApplicationName()));
}
}
}