LocationGroup.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.persistence.CascadeType;
import jakarta.persistence.Column;
import jakarta.persistence.Embedded;
import jakarta.persistence.Entity;
import jakarta.persistence.EnumType;
import jakarta.persistence.Enumerated;
import jakarta.persistence.FetchType;
import jakarta.persistence.ForeignKey;
import jakarta.persistence.JoinColumn;
import jakarta.persistence.ManyToOne;
import jakarta.persistence.OneToMany;
import jakarta.persistence.Table;
import jakarta.persistence.UniqueConstraint;
import jakarta.validation.constraints.NotBlank;
import jakarta.validation.constraints.NotNull;
import jakarta.validation.constraints.Size;
import org.ameba.exception.ServiceLayerException;
import org.openwms.common.StateChangeException;
import org.openwms.common.account.Account;
import org.openwms.common.location.api.LocationGroupMode;
import org.openwms.common.location.api.LocationGroupState;
import org.springframework.util.Assert;

import java.io.Serializable;
import java.util.HashSet;
import java.util.Set;

/**
 * A LocationGroup is a logical group of {@code Location}s with same characteristics.
 *
 * @author Heiko Scherrer
 * @GlossaryTerm
 * @see org.openwms.common.location.Location
 */
@Entity
@Table(name = "COM_LOCATION_GROUP", uniqueConstraints =
    @UniqueConstraint(name = "UC_LG_NAME", columnNames = "C_NAME")
)
public class LocationGroup extends Target implements Serializable {

    /** Unique identifier of a {@code LocationGroup}. */
    @Column(name = "C_NAME", nullable = false, length = LENGTH_NAME)
    @NotBlank
    @Size(min = 1, max = LENGTH_NAME)
    private String name;
    /** Length of the name field; used for telegram mapping and for column definition. */
    public static final int LENGTH_NAME = 255;

    /** The LocationGroup might be assigned to an {@link Account}. */
    @ManyToOne(fetch = FetchType.LAZY)
    @JoinColumn(name = "C_ACCOUNT", referencedColumnName = "C_IDENTIFIER", foreignKey = @ForeignKey(name = "FK_LG_ACC"))
    private Account account;

    /** Description of the {@code LocationGroup}. */
    @Column(name = "C_DESCRIPTION")
    private String description;

    /** A type can be assigned to a {@code LocationGroup}. */
    @Column(name = "C_GROUP_TYPE")
    private String groupType;

    /** Is the {@code LocationGroup} included in the calculation of {@code TransportUnit}s. */
    @Column(name = "C_GROUP_COUNTING_ACTIVE")
    private boolean locationGroupCountingActive = true;

    /** The operation mode is controlled by the subsystem and defines the physical mode a {@code LocationGroup} is currently able to operate in. */
    @Column(name = "C_OP_MODE")
    @NotBlank
    private String operationMode = LocationGroupMode.INFEED_AND_OUTFEED;

    /** State of infeed, controlled by the subsystem only. */
    @Column(name = "C_GROUP_STATE_IN")
    @Enumerated(EnumType.STRING)
    @NotNull
    private LocationGroupState groupStateIn = LocationGroupState.AVAILABLE;

    /** References the {@code LocationGroup} that locked this {@code LocationGroup} for infeed. */
    @ManyToOne(fetch = FetchType.LAZY)
    @JoinColumn(name = "C_IN_LOCKER", foreignKey = @ForeignKey(name = "FK_LG_LG_INLOCKER"))
    private LocationGroup stateInLocker;

    /** State of outfeed. */
    @Column(name = "C_GROUP_STATE_OUT")
    @Enumerated(EnumType.STRING)
    @NotNull
    private LocationGroupState groupStateOut = LocationGroupState.AVAILABLE;

    /** References the {@code LocationGroup} that locked this {@code LocationGroup} for outfeed. */
    @ManyToOne(fetch = FetchType.LAZY)
    @JoinColumn(name = "C_OUT_LOCKER", foreignKey = @ForeignKey(name = "FK_LG_LG_OUTLOCKER"))
    private LocationGroup stateOutLocker;

    /** Maximum fill level of the {@code LocationGroup}. */
    @Column(name = "C_MAX_FILL_LEVEL")
    private float maxFillLevel = 0;

    /** The subsystem like a PLC, that manages this {@code LocationGroup}. */
    @Embedded
    private Subsystem subsystem;

    /* ------------------- collection mapping ------------------- */
    /** Parent {@code LocationGroup}. */
    @ManyToOne(fetch = FetchType.LAZY)
    @JoinColumn(name = "C_PARENT", foreignKey = @ForeignKey(name = "FK_LG_LG_PARENT"))
    private LocationGroup parent;

    /** Child {@code LocationGroup}s. */
    @OneToMany(mappedBy = "parent", cascade = {CascadeType.ALL})
    private Set<LocationGroup> locationGroups = new HashSet<>();

    /** Child {@link Location}s. */
    @OneToMany(mappedBy = "locationGroup")
    private Set<Location> locations = new HashSet<>();

    /*~ ----------------------------- constructors ------------------- */
    /** Dear JPA... */
    protected LocationGroup() { }

    /**
     * Create a new {@code LocationGroup} with a unique name.
     *
     * @param name The name of the {@code LocationGroup} must not be {@literal null}
     */
    public LocationGroup(@NotBlank String name) {
        Assert.hasText(name, "Creation of LocationGroup with name null");
        this.name = name;
    }

    /*~ ----------------------------- methods ------------------- */

    /**
     * Returns the name of the {@code LocationGroup}.
     *
     * @return The name of the {@code LocationGroup}
     */
    public String getName() {
        return name;
    }

    /**
     * Return the {@link Account} this {@code LocationGroup} is assigned to.
     *
     * @return The {@link Account}
     */
    public Account getAccount() {
        return account;
    }

    /**
     * Sets the {@link Account} for this {@code LocationGroup}.
     *
     * @param account The {@link Account} to set for this {@code LocationGroup}
     */
    public void setAccount(Account account) {
        this.account = account;
    }

    /**
     * Check whether infeed is allowed for the {@code LocationGroup}.
     *
     * @return {@literal true} if allowed, otherwise {@literal false}.
     */
    public boolean isInfeedAllowed() {
        return (getGroupStateIn() == LocationGroupState.AVAILABLE);
    }

    /**
     * Check whether infeed of the {@code LocationGroup} is blocked.
     *
     * @return {@literal true} if blocked, otherwise {@literal false}.
     */
    public boolean isInfeedBlocked() {
        return !isInfeedAllowed();
    }

    /**
     * Check whether outfeed is allowed for the {@code LocationGroup}.
     *
     * @return {@literal true} if allowed, otherwise {@literal false}.
     */
    public boolean isOutfeedAllowed() {
        return (getGroupStateOut() == LocationGroupState.AVAILABLE);
    }

    /**
     * Check whether outfeed of the {@code LocationGroup} is blocked.
     *
     * @return {@literal true} if blocked, otherwise {@literal false}.
     */
    public boolean isOutfeedBlocked() {
        return !isOutfeedAllowed();
    }

    /**
     * Get the current operation mode this {@code LocationGroup} operates in.
     *
     * @return The operational mode
     */
    public String getOperationMode() {
        return operationMode;
    }

    /**
     * Set the current operation mode this {@code LocationGroup} can operate in.
     *
     * @param operationMode The mode as an extensible String
     * @see LocationGroupMode
     */
    public void setOperationMode(@NotBlank String operationMode) {
        this.operationMode = operationMode;
        this.locationGroups.forEach(lg -> lg.setOperationMode(operationMode));
    }

    /**
     * Returns the infeed state of the {@code LocationGroup}.
     *
     * @return The state of infeed
     */
    public LocationGroupState getGroupStateIn() {
        return this.groupStateIn;
    }

    /**
     * Change the infeed state of the {@code LocationGroup}.
     *
     * @param newGroupStateIn The state to set
     */
    public void changeGroupStateIn(LocationGroupState newGroupStateIn) {
        if (stateInLocker != null && stateInLocker != this) {
            throw new StateChangeException("The LocationGroup's state is blocked by any other LocationGroup and cannot be changed");
        }
        groupStateIn = newGroupStateIn;
        locationGroups.forEach(lg -> lg.changeGroupStateIn(newGroupStateIn, this));
    }

    /**
     * Change the infeed state of the {@code LocationGroup}.
     *
     * @param newGroupStateIn The state to set
     * @param lockLG The {@code LocationGroup} that wants to lock/unlock this {@code LocationGroup}.
     */
    private void changeGroupStateIn(LocationGroupState newGroupStateIn, LocationGroup lockLG) {
        if (groupStateIn == LocationGroupState.NOT_AVAILABLE && newGroupStateIn == LocationGroupState.AVAILABLE) {

            // unlock
            stateInLocker = null;
        }
        if (groupStateIn == LocationGroupState.AVAILABLE && newGroupStateIn == LocationGroupState.NOT_AVAILABLE) {

            // lock
            stateInLocker = lockLG;
        }
        groupStateIn = newGroupStateIn;
        locationGroups.forEach(lg -> lg.changeGroupStateIn(newGroupStateIn, lockLG));
    }

    /**
     * Return the outfeed state of the {@code LocationGroup}.
     *
     * @return The state of outfeed
     */
    public LocationGroupState getGroupStateOut() {
        return groupStateOut;
    }

    /**
     * Change the outfeed state of the {@code LocationGroup}.
     *
     * @param newGroupStateOut The state to set
     */
    public void changeGroupStateOut(LocationGroupState newGroupStateOut) {
        if (stateOutLocker != null && stateOutLocker != this) {
            throw new StateChangeException("The LocationGroup's state is blocked by any other LocationGroup and cannot be changed");
        }
        groupStateOut = newGroupStateOut;
        locationGroups.forEach(lg -> lg.changeGroupStateOut(newGroupStateOut, this));
    }

    /**
     * Set the outfeed state of the {@code LocationGroup}.
     *
     * @param gStateOut The state to set
     * @param lockLg The {@code LocationGroup} that wants to lock/unlock this {@code LocationGroup}.
     */
    void changeGroupStateOut(LocationGroupState gStateOut, LocationGroup lockLg) {
        if (this.groupStateOut == LocationGroupState.NOT_AVAILABLE && gStateOut == LocationGroupState.AVAILABLE && (this.stateOutLocker == null || this.stateOutLocker.equals(lockLg))) {
            this.groupStateOut = gStateOut;
            this.stateOutLocker = null;
            for (LocationGroup child : locationGroups) {
                child.changeGroupStateOut(gStateOut, lockLg);
            }
        }
        if (this.groupStateOut == LocationGroupState.AVAILABLE && gStateOut == LocationGroupState.NOT_AVAILABLE && (this.stateOutLocker == null || this.stateOutLocker.equals(lockLg))) {
            this.groupStateOut = gStateOut;
            this.stateOutLocker = lockLg;
            for (LocationGroup child : locationGroups) {
                child.changeGroupStateOut(gStateOut, lockLg);
            }
        }
    }

    /**
     * Returns the count of all sub {@link Location}s.
     *
     * @return The count of {@link Location}s belonging to this {@code LocationGroup}
     */
    public int getNoLocations() {
        return this.locations != null ? this.locations.size() : 0;
    }

    /**
     * Returns the maximum fill level of the {@code LocationGroup}.<br> The maximum fill level defines how many {@link Location}s of the
     * {@code LocationGroup} can be occupied by {@code TransportUnit}s. <p> The maximum fill level is a value between 0 and 1 and represents
     * a percentage value. </p>
     *
     * @return The maximum fill level
     */
    public float getMaxFillLevel() {
        return this.maxFillLevel;
    }

    /**
     * Set the maximum fill level for the {@code LocationGroup}. <p> Pass a value between 0 and 1.<br> For example maxFillLevel = 0.85
     * means: 85% of all {@link Location}s can be occupied. </p>
     *
     * @param maxFillLevel The maximum fill level
     */
    public void setMaxFillLevel(float maxFillLevel) {
        this.maxFillLevel = maxFillLevel;
    }

    /**
     * Returns the type of the {@code LocationGroup}.
     *
     * @return The type of the {@code LocationGroup}
     */
    public String getGroupType() {
        return this.groupType;
    }

    /**
     * Set the type for the {@code LocationGroup}.
     *
     * @param groupType The type of the {@code LocationGroup}
     */
    public void setGroupType(String groupType) {
        this.groupType = groupType;
    }

    /**
     * Returns the description text.
     *
     * @return The Description as String
     */
    public String getDescription() {
        return this.description;
    }

    /**
     * Set the description text.
     *
     * @param description The String to set as description text
     */
    public void setDescription(String description) {
        this.description = description;
    }

    /**
     * Returns the parent {@code LocationGroup}.
     *
     * @return The parent {@code LocationGroup}
     */
    public LocationGroup getParent() {
        return this.parent;
    }

    /**
     * Sets the parent {@code LocationGroup} of this {@code LocationGroup}.
     *
     * @param parent The parent {@code LocationGroup} to set
     */
    public void setParent(LocationGroup parent) {
        this.parent = parent;
    }

    /**
     * Return all child {@code LocationGroup}.
     *
     * @return A set of all {@code LocationGroup} having this one as parent
     */
    public Set<LocationGroup> getLocationGroups() {
        return locationGroups;
    }

    /**
     * Check whether this {@code LocationGroup} has {@code LocationGroup}s as children.
     *
     * @return {@literal true} if {@code LocationGroup}s are assigned, otherwise {@literal false}
     */
    public boolean hasLocationGroups() {
        return locationGroups != null && !locationGroups.isEmpty();
    }

    /**
     * Sets the child {@code LocationGroup}.
     *
     * @param locationGroups the set of LocationGroups to set
     */
    public void setLocationGroups(Set<LocationGroup> locationGroups) {
        this.locationGroups = locationGroups;
    }

    /**
     * Add a {@code LocationGroup} to the list of children.
     *
     * @param locationGroup The {@code LocationGroup} to be added as a child
     * @return {@literal true} if the {@code LocationGroup} was new in the collection of {@code LocationGroup}s, otherwise {@literal false}
     */
    public boolean addLocationGroup(LocationGroup locationGroup) {
        if (locationGroup == null) {
            throw new IllegalArgumentException("LocationGroup to be added is null");
        }
        if (locationGroup.parent != null) {
            locationGroup.parent.removeLocationGroup(locationGroup);
        }
        locationGroup.parent = this;
        locationGroup.changeGroupStateIn(groupStateIn, this);
        locationGroup.changeGroupStateOut(groupStateOut, this);
        return locationGroups.add(locationGroup);
    }

    /**
     * Remove a {@code LocationGroup} from the list of children.
     *
     * @param locationGroup The {@code LocationGroup} to be removed from the list of children
     * @return {@literal true} if the {@code LocationGroup} was found and could be removed, otherwise {@literal false}
     */
    public boolean removeLocationGroup(@NotNull LocationGroup locationGroup) {
        Assert.notNull(locationGroup, () -> "LocationGroup to remove is null. this: " + this);
        locationGroup.parent = null;
        return locationGroups.remove(locationGroup);
    }

    /**
     * Return all {@link Location}s.
     *
     * @return {@link Location}s
     */
    public Set<Location> getLocations() {
        return locations;
    }

    /**
     * Check whether this {@code LocationGroup} has {@code Location}s assigned.
     *
     * @return {@literal true} if {@code Location}s are assigned, otherwise {@literal false}
     */
    public boolean hasLocations() {
        return locations != null && !locations.isEmpty();
    }

    /**
     * Add a {@link Location} to the list of children.
     *
     * @param location The {@link Location} to be added as child
     * @return {@literal true} if the {@link Location} was new in the collection of {@link Location}s, otherwise {@literal false}
     */
    public boolean addLocation(Location location) {
        Assert.notNull(location, () -> "Location to be added to LocationGroup is null. this: " + this);
        location.setLocationGroup(this);
        return locations.add(location);
    }

    /**
     * Remove a {@link Location} from the list of children.
     *
     * @param location The {@link Location} to be removed from the list of children
     * @return {@literal true} if the {@link Location} was found and could be removed, otherwise {@literal false}
     */
    public boolean removeLocation(Location location) {
        Assert.notNull(location, () -> "Location to remove from LocationGroup is null. this: " + this);
        location.unsetLocationGroup();
        return locations.remove(location);
    }

    /**
     * Returns the locationGroupCountingActive.
     *
     * @return The locationGroupCountingActive
     */
    public boolean isLocationGroupCountingActive() {
        return locationGroupCountingActive;
    }

    /**
     * Set the locationGroupCountingActive.
     *
     * @param locationGroupCountingActive The locationGroupCountingActive to set
     */
    public void setLocationGroupCountingActive(boolean locationGroupCountingActive) {
        this.locationGroupCountingActive = locationGroupCountingActive;
    }

    /**
     * {@inheritDoc}
     *
     * @see java.lang.Object#hashCode()
     */
    @Override
    public int hashCode() {
        final int prime = 31;
        int result = 111;
        result = prime * result + ((name == null) ? 0 : name.hashCode());
        return result;
    }

    /**
     * {@inheritDoc}
     *
     * @see java.lang.Object#equals(java.lang.Object)
     */
    @Override
    public boolean equals(Object obj) {
        if (this == obj) {
            return true;
        }
        if (!(obj instanceof LocationGroup)) {
            return false;
        }
        LocationGroup other = (LocationGroup) obj;
        if (name == null) {
            if (other.name != null) {
                return false;
            }
        } else if (!name.equals(other.name)) {
            return false;
        }
        return true;
    }

    /**
     * Return the name of the {@code LocationGroup} as String.
     *
     * @return The name
     */
    @Override
    public String toString() {
        return getName();
    }

    /**
     * Tries to change the {@code groupStateIn} and {@code groupStateOut} of the {@code LocationGroup}. A state change is only allowed when
     * the parent {@code LocationGroup}s state is not blocked.
     *
     * @param stateIn The new groupStateIn to set, or {@literal null}
     * @param stateOut The new groupStateOut to set, or {@literal null}
     */
    public void changeState(LocationGroupState stateIn, LocationGroupState stateOut) {
        if (groupStateIn != stateIn && stateIn != null) {
            // GroupStateIn changed
            if (parent != null && parent.getGroupStateIn() == LocationGroupState.NOT_AVAILABLE && groupStateIn == LocationGroupState.AVAILABLE) {
                throw new ServiceLayerException("Not allowed to change GroupStateIn, parent locationGroup is not available");
            }
            changeGroupStateIn(stateIn, this);
        }
        if (groupStateOut != stateOut && stateOut != null) {
            // GroupStateOut changed
            if (parent != null && parent.getGroupStateOut() == LocationGroupState.NOT_AVAILABLE && groupStateOut == LocationGroupState.AVAILABLE) {
                throw new ServiceLayerException("Not allowed to change GroupStateOut, parent locationGroup is not available");
            }
            changeGroupStateOut(stateOut, this);
        }
    }

    /**
     * Whether this LocationGroup has a parent LocationGroup or not.
     *
     * @return {@literal true} If it has a parent
     */
    public boolean hasParent() {
        return parent != null;
    }
}