TransportUnit.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;
import jakarta.persistence.AttributeOverride;
import jakarta.persistence.CascadeType;
import jakarta.persistence.Column;
import jakarta.persistence.Embedded;
import jakarta.persistence.Entity;
import jakarta.persistence.ForeignKey;
import jakarta.persistence.JoinColumn;
import jakarta.persistence.ManyToOne;
import jakarta.persistence.OneToMany;
import jakarta.persistence.OrderBy;
import jakarta.persistence.Table;
import jakarta.persistence.UniqueConstraint;
import jakarta.validation.constraints.NotNull;
import org.ameba.integration.jpa.ApplicationEntity;
import org.ameba.integration.jpa.BaseEntity;
import org.hibernate.envers.AuditOverride;
import org.hibernate.envers.Audited;
import org.hibernate.envers.NotAudited;
import org.openwms.common.app.Default;
import org.openwms.common.location.Location;
import org.openwms.common.transport.barcode.Barcode;
import org.openwms.common.transport.reservation.TransportUnitReservation;
import org.openwms.core.units.api.Weight;
import org.openwms.core.values.CoreTypeDefinitions;
import org.springframework.beans.factory.annotation.Configurable;
import org.springframework.util.Assert;
import java.beans.ConstructorProperties;
import java.io.Serializable;
import java.time.LocalDateTime;
import java.util.ArrayList;
import java.util.HashSet;
import java.util.List;
import java.util.Objects;
import java.util.Set;
import static org.hibernate.envers.RelationTargetAuditMode.NOT_AUDITED;
/**
* A TransportUnit is a physical item like a box, a toad, a bin, a pallet etc., used as a container that is moved between warehouse
* {@code Location}s and might carry goods or other items like {@code LoadUnit}s on top. A TransportUnit must have some kind of identifier,
* like a physical Barcode, an RFID tag or others. There might be projects where TransportUnits are solely identified by virtual identifiers
* and don't have any physical identifiers. A TransportUnit may even carry other TransportUnits.
*
* @author Heiko Scherrer
* @GlossaryTerm
*/
@Configurable
@Audited(targetAuditMode = NOT_AUDITED)
@AuditOverride(forClass = ApplicationEntity.class)
@AuditOverride(forClass = BaseEntity.class)
@Entity
@Table(name = "COM_TRANSPORT_UNIT", uniqueConstraints = @UniqueConstraint(name = "COM_TRANSPORT_UNIT_BARCODE", columnNames = {"C_BARCODE"}))
public class TransportUnit extends ApplicationEntity implements Serializable {
/** Unique natural key. */
@Embedded
@AttributeOverride(name = "value", column = @Column(name = "C_BARCODE", length = Barcode.BARCODE_LENGTH, nullable = false))
@OrderBy
private Barcode barcode;
/** Indicates whether the {@code TransportUnit} is empty or not (nullable). */
@Column(name = "C_EMPTY")
private Boolean empty;
/** A {@code TransportUnit} may belong to a group of {@code TransportUnits}. */
@Column(name = "C_GROUP_NAME")
private String groupName;
/** Date when the {@code TransportUnit} has been moved to the current {@link Location}. */
@Column(name = "C_ACTUAL_LOCATION_DATE")
private LocalDateTime actualLocationDate;
/** Weight of the {@code TransportUnit}. */
@Embedded
@AttributeOverride(name = "unitType", column = @Column(name = "C_WEIGHT_UOM", length = CoreTypeDefinitions.QUANTITY_LENGTH))
@AttributeOverride(name = "magnitude", column = @Column(name = "C_WEIGHT"))
private Weight weight = Weight.ZERO;
/** State of the {@code TransportUnit}. */
@Column(name = "C_STATE")
private String state = TransportUnitState.AVAILABLE.name();
/** The current {@link Location} of the {@code TransportUnit}. */
@ManyToOne
@JoinColumn(name = "C_ACTUAL_LOCATION", nullable = false, foreignKey = @ForeignKey(name = "COM_TU_FK_LOC_ACTUAL"))
private Location actualLocation;
/** The target {@link Location} of the {@code TransportUnit}. This property is set when a {@code TransportOrder} is started. */
@ManyToOne
@JoinColumn(name = "C_TARGET_LOCATION", foreignKey = @ForeignKey(name = "COM_TU_FK_LOC_TARGET"))
private Location targetLocation;
/** The {@link TransportUnitType} of the {@code TransportUnit}. */
@ManyToOne
@JoinColumn(name = "C_TRANSPORT_UNIT_TYPE", nullable = false, foreignKey = @ForeignKey(name = "COM_TU_FK_TUT"))
private TransportUnitType transportUnitType;
/** Owning {@code TransportUnit}. */
@ManyToOne
@JoinColumn(name = "C_PARENT", foreignKey = @ForeignKey(name = "COM_TU_FK_TU_PARENT"))
private TransportUnit parent;
/** The {@code User} who performed the last inventory action on the {@code TransportUnit}. */
@Column(name = "C_INVENTORY_USER")
private String inventoryUser;
/** Date of last inventory check. */
@Column(name = "C_INVENTORY_DATE")
private LocalDateTime inventoryDate;
/** A set of all child {@code TransportUnit}s, ordered by id. */
@OneToMany(mappedBy = "parent", cascade = {CascadeType.MERGE, CascadeType.PERSIST})
@OrderBy("actualLocationDate DESC")
private Set<TransportUnit> children = new HashSet<>();
/** A List of errors occurred on the {@code TransportUnit}. */
@OneToMany(mappedBy = "transportUnit", cascade = {CascadeType.ALL})
@NotAudited
private List<UnitError> errors = new ArrayList<>(0);
/** Tracks all active {@link TransportUnitReservation}s on this {@link TransportUnit}. */
@OneToMany(mappedBy = "transportUnit", cascade = {CascadeType.ALL})
private List<TransportUnitReservation> reservations;
/*~ ----------------------------- constructors ------------------- */
/** Dear JPA... */
protected TransportUnit() { }
@Default
@ConstructorProperties({"barcode"})
public TransportUnit(@NotNull Barcode barcode) {
Assert.notNull(barcode, "Barcode must not be null");
this.barcode = barcode;
initInventory();
}
/**
* Create a new {@code TransportUnit} with an unique {@link Barcode}.
*
* @param barcode The unique identifier of this {@code TransportUnit} is the {@link Barcode} - must not be {@literal null}
* @param transportUnitType The {@code TransportUnitType} of this {@code TransportUnit} - must not be {@literal null}
* @param actualLocation The current {@code Location} of this {@code TransportUnit} - must not be {@literal null}
* @throws IllegalArgumentException when one of the params is {@literal null}
*/
@ConstructorProperties({"barcode", "transportUnitType", "actualLocation"})
public TransportUnit(@NotNull Barcode barcode, @NotNull TransportUnitType transportUnitType, @NotNull Location actualLocation) {
Assert.notNull(barcode, "Barcode must not be null");
Assert.notNull(transportUnitType, "TransportUnitType must not be null");
Assert.notNull(actualLocation, "ActualLocation must not be null");
this.barcode = barcode;
setTransportUnitType(transportUnitType);
setActualLocation(actualLocation);
initInventory();
}
/*~ ----------------------------- methods ------------------- */
/** Required for the Mapper. */
@Override
public void setPersistentKey(String pKey) {
super.setPersistentKey(pKey);
}
/**
* Get the actual {@link Location} of the {@code TransportUnit}.
*
* @return The {@link Location} where the {@code TransportUnit} is placed on
*/
public Location getActualLocation() {
return actualLocation;
}
/**
* Place the {@code TransportUnit} and all its children to a {@link Location}.
*
* @param actualLocation The new {@link Location} of the {@code TransportUnit} and all its children
* @throws IllegalArgumentException when {@code actualLocation} is {@literal null}
*/
public void setActualLocation(Location actualLocation) {
Assert.notNull(actualLocation, "ActualLocation must not be null, this: " + this);
this.actualLocation = actualLocation;
this.actualLocationDate = LocalDateTime.now();
this.actualLocation.setLastMovement(this.actualLocationDate);
if (this.getChildren() != null) {
this.getChildren().forEach(child -> child.setActualLocation(actualLocation));
}
}
/**
* Initialize inventory info of the {@code TransportUnit}.
*/
public void initInventory() {
setInventoryUser("init");
setInventoryDate(LocalDateTime.now());
}
/**
* Get the target {@link Location} of the {@code TransportUnit}. This property can not be {@literal null} when an active {@code
* TransportOrder} exists.
*
* @return The target location
*/
public Location getTargetLocation() {
return this.targetLocation;
}
/**
* Set the target {@link Location} of the {@code TransportUnit}. Shall only be set in combination with an active {@code
* TransportOrder}.
*
* @param targetLocation The target {@link Location} where this {@code TransportUnit} shall be transported to
*/
public void setTargetLocation(Location targetLocation) {
this.targetLocation = targetLocation;
}
/**
* Indicates whether the {@code TransportUnit} is empty or not.
*
* @return {@literal true} if empty, {@literal false} if not empty, {@literal null} when not defined
*/
public Boolean getEmpty() {
return this.empty;
}
/**
* Marks the {@code TransportUnit} to be empty.
*
* @param empty {@literal true} to mark the {@code TransportUnit} as empty, {@literal false} to mark it as not empty and {@literal null}
* for no definition
*/
public void setEmpty(Boolean empty) {
this.empty = empty;
}
/**
* Get the groupId.
*
* @return The groupId
*/
public String getGroupName() {
return groupName;
}
/**
* Set the groupId.
*
* @param groupId The groupId
*/
public void setGroupName(String groupId) {
this.groupName = groupId;
}
/**
* Returns the username of the User who performed the last inventory action on the {@code TransportUnit}.
*
* @return The username who did the last inventory check
*/
public String getInventoryUser() {
return this.inventoryUser;
}
/**
* Set the username who performed the last inventory action on the {@code TransportUnit}.
*
* @param inventoryUser The username who did the last inventory check
*/
public void setInventoryUser(String inventoryUser) {
this.inventoryUser = inventoryUser;
}
/**
* Number of {@code TransportUnit}s belonging to the {@code TransportUnit}.
*
* @return The number of all {@code TransportUnit}s belonging to this one
*/
public int getNoTransportUnits() {
return this.children.size();
}
/**
* Returns the date when the {@code TransportUnit} moved to the actualLocation.
*
* @return The timestamp when the {@code TransportUnit} moved the last time
*/
public LocalDateTime getActualLocationDate() {
return actualLocationDate;
}
/**
* Returns the timestamp of the last inventory check of the {@code TransportUnit}.
*
* @return The timestamp of the last inventory check of the {@code TransportUnit}.
*/
public LocalDateTime getInventoryDate() {
return inventoryDate;
}
/**
* Set the timestamp of the last inventory action of the {@code TransportUnit}.
*
* @param inventoryDate The timestamp of the last inventory check
* @throws IllegalArgumentException when {@code inventoryDate} is {@literal null}
*/
public void setInventoryDate(LocalDateTime inventoryDate) {
Assert.notNull(inventoryDate, () -> "InventoryDate must not be null, this: " + this);
this.inventoryDate = inventoryDate;
}
/**
* Returns the current weight of the {@code TransportUnit}.
*
* @return The current weight of the {@code TransportUnit}
*/
public Weight getWeight() {
return weight;
}
/**
* Sets the current weight of the {@code TransportUnit}.
*
* @param weight The current weight of the {@code TransportUnit}
*/
public void setWeight(Weight weight) {
this.weight = weight;
}
public List<UnitError> getErrors() {
return this.errors;
}
/**
* Add an error to the {@code TransportUnit}.
*
* @param error An {@link UnitError} to be added
* @return The key.
* @throws IllegalArgumentException when {@code error} is {@literal null}
*/
public UnitError addError(UnitError error) {
Assert.notNull(error, () -> "Error must not be null, this: " + this);
error.setTransportUnit(this);
this.errors.add(error);
return error;
}
/**
* Checks whether this {@code TransportUnit} has one or more reservations.
*
* @return {@literal true} if so
*/
public boolean hasReservations() {
return this.reservations != null && !this.reservations.isEmpty();
}
/**
* Return the state of the {@code TransportUnit}.
*
* @return The current state of the {@code TransportUnit}
*/
public String getState() {
return this.state;
}
/**
* Set the state of the {@code TransportUnit}.
*
* @param state The state to set on the {@code TransportUnit}
*/
public void setState(String state) {
this.state = state;
}
/**
* Return the {@link TransportUnitType} of the {@code TransportUnit}.
*
* @return The {@link TransportUnitType} the {@code TransportUnit} belongs to
*/
public TransportUnitType getTransportUnitType() {
return this.transportUnitType;
}
/**
* Set the {@link TransportUnitType} of the {@code TransportUnit}.
*
* @param transportUnitType The type of the {@code TransportUnit}
*/
public void setTransportUnitType(TransportUnitType transportUnitType) {
Assert.notNull(transportUnitType, () -> "TransportUnitType must not be null, this: " + this);
this.transportUnitType = transportUnitType;
}
/**
* Return the {@link Barcode} of the {@code TransportUnit}.
*
* @return The current {@link Barcode}
*/
public Barcode getBarcode() {
return barcode;
}
/**
* Returns the parent {@code TransportUnit}.
*
* @return the parent.
*/
public TransportUnit getParent() {
return parent;
}
/**
* Set a parent {@code TransportUnit}.
*
* @param parent The parent to set.
*/
public void setParent(TransportUnit parent) {
this.parent = parent;
}
/**
* Get all child {@code TransportUnit}s.
*
* @return the transportUnits.
*/
public Set<TransportUnit> getChildren() {
return this.children;
}
/**
* Add a {@code TransportUnit} to the children.
*
* @param transportUnit The {@code TransportUnit} to be added to the list of children
* @throws IllegalArgumentException when transportUnit is {@literal null}
*/
public void addChild(TransportUnit transportUnit) {
Assert.notNull(transportUnit, () -> "TransportUnitType must not be null, this: " + this);
if (transportUnit.hasParent()) {
if (transportUnit.getParent().equals(this)) {
// if this instance is already the parent, we just return
return;
}
// disconnect post from it's current relationship
transportUnit.getParent().removeChild(transportUnit);
}
// make this instance the new parent
transportUnit.setParent(this);
this.children.add(transportUnit);
}
/**
* Checks whether this {@code TransportUnit} has a parent {@code TransportUnit} or not.
*
* @return {@code true} it has a parent, otherwise {@code false}
*/
public boolean hasParent() {
return parent != null;
}
/**
* Checks whether this {@code TransportUnit} has child {@code TransportUnit}s or not.
*
* @return {@code true} it has children, otherwise {@code false}
*/
public boolean hasChildren() {
return this.children != null && !this.children.isEmpty();
}
/**
* Remove a {@code TransportUnit} from the list of children.
*
* @param transportUnit The {@code TransportUnit} to be removed from the list of children
* @throws IllegalArgumentException when {@code transportUnit} is {@literal null} or not a child of this instance
*/
public void removeChild(TransportUnit transportUnit) {
Assert.notNull(transportUnit, () -> "TransportUnit must not be null, this: " + this);
// make sure this is the parent before we break the relationship
if (transportUnit.parent == null || !transportUnit.parent.equals(this)) {
throw new IllegalArgumentException("Child TransportUnit not associated with this instance, this: " + this);
}
transportUnit.setParent(null);
this.children.remove(transportUnit);
}
/**
* {@inheritDoc}
*
* Return the {@link Barcode} as String.
*/
@Override
public String toString() {
return barcode.toString();
}
/**
* {@inheritDoc}
*
* Uses barcode for comparison.
*/
@Override
public boolean equals(Object o) {
if (this == o) return true;
if (o == null || getClass() != o.getClass()) return false;
TransportUnit that = (TransportUnit) o;
return barcode.equals(that.barcode);
}
/**
* {@inheritDoc}
*
* Uses barcode for calculation.
*/
@Override
public int hashCode() {
return Objects.hash(barcode);
}
}