Location.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.AttributeOverride;
import jakarta.persistence.CascadeType;
import jakarta.persistence.Column;
import jakarta.persistence.Convert;
import jakarta.persistence.Embedded;
import jakarta.persistence.Entity;
import jakarta.persistence.ForeignKey;
import jakarta.persistence.JoinColumn;
import jakarta.persistence.JoinTable;
import jakarta.persistence.ManyToOne;
import jakarta.persistence.OneToMany;
import jakarta.persistence.Table;
import jakarta.persistence.UniqueConstraint;
import jakarta.validation.constraints.NotNull;
import jakarta.validation.constraints.Size;
import org.openwms.common.account.Account;
import org.openwms.common.app.Default;
import org.springframework.util.Assert;

import java.io.Serializable;
import java.math.BigDecimal;
import java.time.LocalDateTime;
import java.util.Arrays;
import java.util.HashSet;
import java.util.List;
import java.util.Objects;
import java.util.Set;

import static java.lang.String.format;
import static org.openwms.common.location.StringListConverter.STRING_LIST_LENGTH;

/**
 * A Location, represents a physical or virtual place in a warehouse. Could be something like a storage location in the stock or a conveyor
 * location. Even error locations can be represented with the Location. Multiple Locations with same characteristics are grouped to a
 * {@link LocationGroup}.
 *
 * @author Heiko Scherrer
 * @GlossaryTerm
 * @see org.openwms.common.location.LocationGroup
 */
@Entity
@Table(name = Location.TABLE, uniqueConstraints = {
        @UniqueConstraint(name = "UC_LOC_ID", columnNames = {"C_AREA", "C_AISLE", "C_X", "C_Y", "C_Z"}),
        @UniqueConstraint(name = "UC_LOC_PLC_CODE", columnNames = "C_PLC_CODE"),
        @UniqueConstraint(name = "UC_LOC_ERP_CODE", columnNames = "C_ERP_CODE")
})
public class Location extends Target implements Serializable {

    /** Table name. */
    public static final String TABLE = "COM_LOCATION";
    private static final String CREATION_OF_LOCATION_WITH_LOCATION_ID_NULL = "Creation of Location with locationId null";

    /** Unique natural key. */
    @Embedded
    @NotNull
    @AttributeOverride(name = "area", column = @Column(name = "C_AREA"))
    @AttributeOverride(name = "aisle", column = @Column(name = "C_AISLE"))
    @AttributeOverride(name = "x", column = @Column(name = "C_X"))
    @AttributeOverride(name = "y", column = @Column(name = "C_Y"))
    @AttributeOverride(name = "z", column = @Column(name = "C_Z"))
    private LocationPK locationId;

    /** The {@code Location} might be assigned to an {@link Account}. */
    @ManyToOne
    @JoinColumn(name = "C_ACCOUNT", referencedColumnName = "C_IDENTIFIER", foreignKey = @ForeignKey(name = "FK_LOC_ACC"))
    private Account account;

    /** PLC code of the {@code Location}. */
    @Column(name = "C_PLC_CODE")
    private String plcCode;

    /** ERP code of the {@code Location}. */
    @Column(name = "C_ERP_CODE", unique = true)
    private String erpCode;

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

    /** Sort order index used by strategies for putaway, or picking. */
    @Column(name = "C_SORT")
    private Integer sortOrder;

    /** Might be assigned to a particular zone in stock. */
    @Column(name = "C_STOCK_ZONE")
    private String stockZone;

    /** A {@code Location} can be assigned to a particular labels. */
    @Column(name="C_LABELS", length = STRING_LIST_LENGTH)
    @Convert(converter = StringListConverter.class)
    @Size(max = STRING_LIST_LENGTH)
    private List<String> labels;

    /** Maximum number of {@code TransportUnit}s allowed on the {@code Location}. */
    @Column(name = "C_NO_MAX_TRANSPORT_UNITS")
    private int noMaxTransportUnits = DEF_MAX_TU;
    /** Default value of {@link #noMaxTransportUnits}. */
    public static final int DEF_MAX_TU = 1;

    /** Maximum allowed weight on the {@code Location}. */
    @Column(name = "C_MAXIMUM_WEIGHT")
    private BigDecimal maximumWeight;

    /**
     * Date of last movement. When a {@code TransportUnit} is moving to or away from the {@code Location}, {@code lastMovement} is updated.
     * This is useful to get the history of {@code TransportUnit}s as well as for inventory calculation.
     */
    @Column(name = "C_LAST_MOVEMENT")
    private LocalDateTime lastMovement;

    /**
     * Shall the {@code Location} be included in the calculation of {@code TransportUnit}s of the parent {@link LocationGroup}.
     * <ul>
     *     <li>{@literal true} : {@code Location} is included in calculation of {@code TransportUnit}s.</li>
     *     <li>{@literal false}: {@code Location} is not included in calculation of {@code TransportUnit}s.</li>
     * </ul>
     */
    @Column(name = "C_LG_COUNTING_ACTIVE")
    private Boolean locationGroupCountingActive = DEF_LG_COUNTING_ACTIVE;
    /** Default value of {@link #locationGroupCountingActive}. */
    public static final boolean DEF_LG_COUNTING_ACTIVE = false;

    /**
     * Signals the incoming state of the {@code Location}.
     * {@code Location}s which are blocked for incoming movements do not accept {@code TransportUnit}s.
     * <ul>
     *     <li>{@literal true} : {@code Location} is ready to pick up {@code TransportUnit}s.</li>
     *     <li>{@literal false}: {@code Location} is locked, and cannot pick up {@code TransportUnit}s.</li>
     * </ul>
     */
    @Column(name = "C_INCOMING_ACTIVE")
    private boolean incomingActive = DEF_INCOMING_ACTIVE;
    /** Default value of {@link #incomingActive}. */
    public static final boolean DEF_INCOMING_ACTIVE = true;

    /**
     * Signals the outgoing state of the {@code Location}.
     * {@code Location}s which are blocked for outgoing do not accept to move {@code TransportUnit}s away.
     * <ul>
     *     <li>{@literal true} : {@code Location} is enabled for outgoing {@code TransportUnit}s.</li>
     *     <li>{@literal false}: {@code Location} is locked, {@code TransportUnit}s can't leave the {@code Location}.</li>
     * </ul>
     */
    @Column(name = "C_OUTGOING_ACTIVE")
    private boolean outgoingActive = DEF_OUTGOING_ACTIVE;
    /** Default value of {@link #outgoingActive}. */
    public static final boolean DEF_OUTGOING_ACTIVE = true;

    /**
     * The PLC is able to change the state of a {@code Location}. This property stores the last state, received from the PLC.
     * <ul>
     *     <li>0 : No PLC error, everything okay</li>
     *     <li>&lt; 0: Not defined</li>
     *     <li>&gt; 0: Some defined error code</li>
     * </ul>
     */
    @Column(name = "C_PLC_STATE")
    private int plcState = DEF_PLC_STATE;
    /** Default value of {@link #plcState}. */
    public static final int DEF_PLC_STATE = 0;

    /**
     * Determines whether the {@code Location} is considered in the allocation procedure.
     * <ul>
     *     <li>{@literal true} : The {@code Location} is considered in storage calculation by an allocation procedure.</li>
     *     <li>{@literal false} : The {@code Location} is not considered in the allocation process.</li>
     * </ul>
     */
    @Column(name = "C_CONSIDERED_IN_ALLOCATION")
    private Boolean consideredInAllocation = DEF_CONSIDERED_IN_ALLOCATION;
    /** Default value of {@link #consideredInAllocation}. */
    public static final boolean DEF_CONSIDERED_IN_ALLOCATION = true;

    /** The {@link LocationType} the {@code Location} belongs to. */
    @ManyToOne
    @JoinColumn(name = "C_LOCATION_TYPE", foreignKey = @ForeignKey(name = "FK_LOC_LT"))
    private LocationType locationType;

    /** Some group the {@code Location} belongs to. */
    @Column(name = "C_GROUP")
    private String group;

    /** The {@code Location} may be classified, like 'hazardous'. */
    @Column(name = "C_CLASSIFICATION")
    @Size(max = 255)
    private String classification;

    /** The {@link LocationGroup} the {@code Location} belongs to. */
    @ManyToOne
    @JoinColumn(name = "C_LOCATION_GROUP", foreignKey = @ForeignKey(name = "FK_LOC_LG"))
    private LocationGroup locationGroup;

    /** Stored {@link Message}s on the {@code Location}. */
    @OneToMany(cascade = {CascadeType.ALL})
    @JoinTable(name = "COM_LOCATION_MESSAGE",
            uniqueConstraints = @UniqueConstraint(name = "UC_LOCM_ID", columnNames = "C_MESSAGE_ID"),
            joinColumns = @JoinColumn(name = "C_LOCATION_ID", foreignKey = @ForeignKey(name = "FK_LOCM_LOCPK")),
            inverseJoinColumns = @JoinColumn(name = "C_MESSAGE_ID", foreignKey = @ForeignKey(name = "FK_LOCM_MSGPK"))
    )
    private Set<Message> messages = new HashSet<>();

    /*~ ----------------------------- constructors ------------------- */

    /**
     * Create a new Location with the business key.
     *
     * @param locationId The unique natural key of the Location
     */
    protected Location(LocationPK locationId) {
        Assert.notNull(locationId, CREATION_OF_LOCATION_WITH_LOCATION_ID_NULL);
        this.locationId = locationId;
    }

    /**
     * Create a new Location.
     *
     * @param locationId The unique natural key of the Location
     * @param locationGroup The LocationGroup the Location belongs to
     */
    @Default
    Location(LocationPK locationId, Account account, LocationGroup locationGroup, LocationType locationType, String erpCode,
            String plcCode, Integer sortOrder, String stockZone) {
        Assert.notNull(locationId, CREATION_OF_LOCATION_WITH_LOCATION_ID_NULL);
        this.locationId = locationId;
        this.account = account;
        this.locationGroup = locationGroup;
        this.locationType = locationType;
        this.erpCode = erpCode;
        this.plcCode = plcCode;
        this.sortOrder = sortOrder;
        this.stockZone = stockZone;
    }

    /** Dear JPA... */
    protected Location() { }

    /**
     * Create a new Location with the business key.
     *
     * @param locationId The unique natural key of the Location
     * @return The Location
     */
    public static Location create(LocationPK locationId) {
        return new Location(locationId);
    }
    /*~ ----------------------------- methods ------------------- */
    /** Required for the Mapper. */
    @Override
    public void setPersistentKey(String pKey) {
        super.setPersistentKey(pKey);
    }

    /**
     * Check if the Location has a {@code locationId} set.
     *
     * @return {@literal true} if so
     */
    public boolean hasLocationId() {
        return locationId != null;
    }

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

    /**
     * Get the ERP Code of the Location.
     *
     * @return The ERP code
     */
    public String getErpCode() {
        return erpCode;
    }

    /**
     * Get the PLC Code of the Location.
     *
     * @return The PLC code
     */
    public String getPlcCode() {
        return plcCode;
    }

    /**
     * Add a new {@link Message} to this Location.
     *
     * @param message The {@link Message} to be added
     * @return {@literal true} if the {@link Message} is new in the collection of messages, otherwise {@literal false}
     */
    public boolean addMessage(Message message) {
        Assert.notNull(message, "null passed to addMessage, this: " + this);
        return this.messages.add(message);
    }

    /**
     * Determine whether the Location is considered during allocation.
     *
     * @return {@literal true} when considered in allocation, otherwise {@literal false}
     */
    public boolean isConsideredInAllocation() {
        return this.consideredInAllocation;
    }

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

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

    /**
     * Get the sortOrder.
     *
     * @return A sequence number
     */
    public Integer getSortOrder() {
        return sortOrder;
    }

    /**
     * Returns the stockZone.
     *
     * @return As string
     */
    public String getStockZone() {
        return stockZone;
    }

    /**
     * Returns the list of Strings set as labels for the Location.
     *
     * @return A list of Strings or an empty list
     */
    public List<String> getLabels() {
        return labels;
    }

    /**
     * Set a list of labels to the Location.
     *
     * @param labels A comma-separated list of labels
     */
    public void setLabels(List<String> labels) {
        this.labels = labels;
    }

    /**
     * Determine whether incoming mode is activated and {@code TransportUnit}s can be put on this Location.
     *
     * @return {@literal true} when incoming mode is activated, otherwise {@literal false}
     */
    public boolean isInfeedActive() {
        return this.incomingActive;
    }

    /**
     * Set the incoming mode of this Location.
     *
     * @param infeedActive {@literal true} means Infeed movements are possible, {@literal false} means Infeed movements are blocked
     */
    public void setInfeed(boolean infeedActive) {
        this.incomingActive = infeedActive;
    }

    /**
     * Check whether infeed is blocked and moving {@code TransportUnit}s to here is forbidden.
     *
     * @return {@literal true} is blocked, otherwise {@literal false}
     */
    public boolean isInfeedBlocked() {
        return !this.incomingActive;
    }

    /**
     * Return the date when the Location was updated the last time.
     *
     * @return Timestamp of the last update
     */
    public LocalDateTime getLastMovement() {
        return this.lastMovement;
    }

    /**
     * Change the date when a TransportUnit was put or left the Location the last time.
     *
     * @param lastMovement The date of change.
     */
    public void setLastMovement(LocalDateTime lastMovement) {
        this.lastMovement = lastMovement;
    }

    /**
     * Return the {@link LocationGroup} where the Location belongs to.
     *
     * @return The {@link LocationGroup} of the Location
     */
    public LocationGroup getLocationGroup() {
        return this.locationGroup;
    }

    /**
     * Determine whether the Location is part of the parent {@link LocationGroup}s calculation procedure of {@code TransportUnit}s.
     *
     * @return {@literal true} if calculation is activated, otherwise {@literal false}
     */
    public boolean isLocationGroupCountingActive() {
        return this.locationGroupCountingActive;
    }

    /**
     * Returns the locationId (natural key) of the Location.
     *
     * @return The locationId
     */
    public LocationPK getLocationId() {
        return this.locationId;
    }

    /**
     * Returns the type of Location.
     *
     * @return The type
     */
    public LocationType getLocationType() {
        return this.locationType;
    }

    public void setLocationType(LocationType locationType) {
        if (this.locationType != null && !this.locationType.equals(locationType)) {
            throw new IllegalArgumentException(format("LocationType of Location [%s] is already defined and can't be changed", locationType));
        }
        this.locationType = locationType;
    }

    /**
     * Returns the group the Location belongs to.
     *
     * @return The group as String
     */
    public String getGroup() {
        return group;
    }

    /**
     * Returns the classification of the Location.
     *
     * @return As a String
     */
    public String getClassification() {
        return classification;
    }

    /**
     * Set the classification.
     *
     * @param classification As an arbitrary String
     */
    public void setClassification(String classification) {
        this.classification = classification;
    }

    /**
     * Return the maximum allowed weight on the Location.
     *
     * @return The maximum allowed weight
     */
    public BigDecimal getMaximumWeight() {
        return this.maximumWeight;
    }

    /**
     * Returns an unmodifiable Set of {@link Message}s stored for the Location.
     *
     * @return An unmodifiable Set
     */
    public Set<Message> getMessages() {
        return new HashSet<>(messages);
    }

    /**
     * Returns the maximum number of {@code TransportUnit}s allowed on the Location.
     *
     * @return The maximum number of {@code TransportUnit}s
     */
    public int getNoMaxTransportUnits() {
        return noMaxTransportUnits;
    }

    /**
     * Determine whether outgoing mode is activated and {@code TransportUnit}s can leave this Location.
     *
     * @return {@literal true} when outgoing mode is activated, otherwise {@literal false}
     */
    public boolean isOutfeedActive() {
        return this.outgoingActive;
    }

    /**
     * Check whether outfeed is blocked and moving {@code TransportUnit}s from here is forbidden.
     *
     * @return {@literal true} is blocked, otherwise {@literal false}
     */
    public boolean isOutfeedBlocked() {
        return !this.outgoingActive;
    }

    /**
     * Set the outfeed mode of this Location.
     *
     * @param outfeedActive {@literal true} means Outfeed movements are possible, {@literal false} means Outfeed movements are blocked
     */
    public void setOutfeed(boolean outfeedActive) {
        this.outgoingActive = outfeedActive;
    }


    /**
     * Return the current set plc state.
     *
     * @return the plc state
     */
    public int getPlcState() {
        return plcState;
    }

    /**
     * Set the plc state.
     *
     * @param plcState the plc state
     */
    public void setPlcState(int plcState) {
        this.plcState = plcState;
    }

    /**
     * Remove one or more {@link Message}s from this Location.
     *
     * @param msgs An array of {@link Message}s to be removed
     * @return {@literal true} if the {@link Message}s were found and removed, otherwise {@literal false}
     * @throws IllegalArgumentException when messages is {@literal null}
     */
    public boolean removeMessages(Message... msgs) {
        Assert.notNull(msgs, () -> "null passed to removeMessages, this: " + this);
        return this.messages.removeAll(Arrays.asList(msgs));
    }

    /**
     * Add this {@code Location} to the {@literal locationGroup}. When the argument is {@literal null} an existing {@link LocationGroup} is
     * removed from the {@code Location}.
     *
     * @param locationGroup The {@link LocationGroup} to be assigned
     */
    void setLocationGroup(LocationGroup locationGroup) {
        Assert.notNull(locationGroup, () -> "Not allowed to call location#setLocationGroup with null argument, this: " + this);
        if (this.locationGroup != null) {
            this.locationGroup.removeLocation(this);
        }
        this.setLocationGroupCountingActive(locationGroup.isLocationGroupCountingActive());
        this.locationGroup = locationGroup;
    }

    /**
     * Define whether or not the Location shall be considered in counting {@code TransportUnit}s of the parent {@link LocationGroup}.
     *
     * @param locationGroupCountingActive {@literal true} if considered, otherwise {@literal false}
     */
    public void setLocationGroupCountingActive(boolean locationGroupCountingActive) {
        this.locationGroupCountingActive = locationGroupCountingActive;
    }

    /**
     * Checks whether this {@code Location} belongs to a {@code LocationGroup}.
     *
     * @return {@literal true} if it belongs to a {@code LocationGroup}, otherwise {@literal false}
     */
    public boolean belongsToLocationGroup() {
        return locationGroup != null;
    }

    /**
     * Checks whether this {@code Location} belongs NOT to a {@code LocationGroup}.
     *
     * @return {@literal true} if it does not belong to a {@code LocationGroup}, otherwise {@literal false}
     */
    public boolean belongsNotToLocationGroup() {
        return !belongsToLocationGroup();
    }

    /**
     * Set the locationGroup to {@literal null}.
     */
    void unsetLocationGroup() {
        this.locationGroup = null;
    }

    /**
     * {@inheritDoc}
     * <p>
     * Only use the unique natural key for comparison.
     */
    @Override
    public boolean equals(Object o) {
        if (this == o) return true;
        if (o == null || getClass() != o.getClass()) return false;
        Location location = (Location) o;
        return Objects.equals(locationId, location.locationId);
    }

    /**
     * {@inheritDoc}
     * <p>
     * Only use the unique natural key for hashCode calculation.
     */
    @Override
    public int hashCode() {
        return Objects.hash(locationId);
    }

    /**
     * Return the {@link LocationPK} as String.
     *
     * @return String locationId
     * @see LocationPK#toString()
     */
    @Override
    public String toString() {
        return locationId.toString();
    }


    public static final class LocationBuilder {
        private final Location target;

        private LocationBuilder(Location target) {
            this.target = target;
        }

        public static LocationBuilder aLocation(Location target) {
            return new LocationBuilder(target);
        }

        public LocationBuilder withAccount(Account account) {
            this.target.account = account;
            return this;
        }

        public LocationBuilder withPlcCode(String plcCode) {
            this.target.plcCode = plcCode;
            return this;
        }

        public LocationBuilder withErpCode(String erpCode) {
            this.target.erpCode = erpCode;
            return this;
        }

        public LocationBuilder withDescription(String description) {
            this.target.description = description;
            return this;
        }

        public LocationBuilder withSortOrder(Integer sortOrder) {
            this.target.sortOrder = sortOrder;
            return this;
        }

        public LocationBuilder withStockZone(String stockZone) {
            this.target.stockZone = stockZone;
            return this;
        }

        public LocationBuilder withLabels(List<String> labels) {
            this.target.labels = labels;
            return this;
        }

        public LocationBuilder withNoMaxTransportUnits(int noMaxTransportUnits) {
            this.target.noMaxTransportUnits = noMaxTransportUnits;
            return this;
        }

        public LocationBuilder withMaximumWeight(BigDecimal maximumWeight) {
            this.target.maximumWeight = maximumWeight;
            return this;
        }

        public LocationBuilder withLastMovement(LocalDateTime lastMovement) {
            this.target.lastMovement = lastMovement;
            return this;
        }

        public LocationBuilder withLocationGroupCountingActive(boolean locationGroupCountingActive) {
            this.target.locationGroupCountingActive = locationGroupCountingActive;
            return this;
        }

        public LocationBuilder withIncomingActive(boolean incomingActive) {
            this.target.incomingActive = incomingActive;
            return this;
        }

        public LocationBuilder withOutgoingActive(boolean outgoingActive) {
            this.target.outgoingActive = outgoingActive;
            return this;
        }

        public LocationBuilder withPlcState(int plcState) {
            this.target.plcState = plcState;
            return this;
        }

        public LocationBuilder withConsideredInAllocation(boolean consideredInAllocation) {
            this.target.consideredInAllocation = consideredInAllocation;
            return this;
        }

        public LocationBuilder withLocationType(LocationType locationType) {
            this.target.locationType = locationType;
            return this;
        }

        public LocationBuilder withGroup(String group) {
            this.target.group = group;
            return this;
        }

        public LocationBuilder withClassification(String classification) {
            this.target.classification = classification;
            return this;
        }

        public LocationBuilder withLocationGroup(LocationGroup locationGroup) {
            this.target.locationGroup = locationGroup;
            return this;
        }

        public LocationBuilder withMessages(Set<Message> messages) {
            this.target.messages = messages;
            return this;
        }

        public Location build() {
            return target;
        }
    }
}