TransportUnit.java

  1. /*
  2.  * Copyright 2005-2025 the original author or authors.
  3.  *
  4.  * Licensed under the Apache License, Version 2.0 (the "License");
  5.  * you may not use this file except in compliance with the License.
  6.  * You may obtain a copy of the License at
  7.  *
  8.  * http://www.apache.org/licenses/LICENSE-2.0
  9.  *
  10.  * Unless required by applicable law or agreed to in writing, software
  11.  * distributed under the License is distributed on an "AS IS" BASIS,
  12.  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  13.  * See the License for the specific language governing permissions and
  14.  * limitations under the License.
  15.  */
  16. package org.openwms.common.transport;

  17. import jakarta.persistence.AttributeOverride;
  18. import jakarta.persistence.CascadeType;
  19. import jakarta.persistence.Column;
  20. import jakarta.persistence.Embedded;
  21. import jakarta.persistence.Entity;
  22. import jakarta.persistence.ForeignKey;
  23. import jakarta.persistence.JoinColumn;
  24. import jakarta.persistence.ManyToOne;
  25. import jakarta.persistence.OneToMany;
  26. import jakarta.persistence.OrderBy;
  27. import jakarta.persistence.Table;
  28. import jakarta.persistence.UniqueConstraint;
  29. import jakarta.validation.constraints.NotNull;
  30. import org.ameba.integration.jpa.ApplicationEntity;
  31. import org.ameba.integration.jpa.BaseEntity;
  32. import org.hibernate.envers.AuditOverride;
  33. import org.hibernate.envers.Audited;
  34. import org.hibernate.envers.NotAudited;
  35. import org.openwms.common.app.Default;
  36. import org.openwms.common.location.Location;
  37. import org.openwms.common.transport.barcode.Barcode;
  38. import org.openwms.common.transport.reservation.TransportUnitReservation;
  39. import org.openwms.core.units.api.Weight;
  40. import org.openwms.core.values.CoreTypeDefinitions;
  41. import org.springframework.beans.factory.annotation.Configurable;
  42. import org.springframework.util.Assert;

  43. import java.beans.ConstructorProperties;
  44. import java.io.Serializable;
  45. import java.time.LocalDateTime;
  46. import java.util.ArrayList;
  47. import java.util.HashSet;
  48. import java.util.List;
  49. import java.util.Objects;
  50. import java.util.Set;

  51. import static org.hibernate.envers.RelationTargetAuditMode.NOT_AUDITED;

  52. /**
  53.  * 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
  54.  * {@code Location}s and might carry goods or other items like {@code LoadUnit}s on top. A TransportUnit must have some kind of identifier,
  55.  * like a physical Barcode, an RFID tag or others. There might be projects where TransportUnits are solely identified by virtual identifiers
  56.  * and don't have any physical identifiers. A TransportUnit may even carry other TransportUnits.
  57.  *
  58.  * @author Heiko Scherrer
  59.  * @GlossaryTerm
  60.  */
  61. @Configurable
  62. @Audited(targetAuditMode = NOT_AUDITED)
  63. @AuditOverride(forClass = ApplicationEntity.class)
  64. @AuditOverride(forClass = BaseEntity.class)
  65. @Entity
  66. @Table(name = "COM_TRANSPORT_UNIT", uniqueConstraints = @UniqueConstraint(name = "COM_TRANSPORT_UNIT_BARCODE", columnNames = {"C_BARCODE"}))
  67. public class TransportUnit extends ApplicationEntity implements Serializable {

  68.     /** Unique natural key. */
  69.     @Embedded
  70.     @AttributeOverride(name = "value", column = @Column(name = "C_BARCODE", length = Barcode.BARCODE_LENGTH, nullable = false))
  71.     @OrderBy
  72.     private Barcode barcode;

  73.     /** Indicates whether the {@code TransportUnit} is empty or not (nullable). */
  74.     @Column(name = "C_EMPTY")
  75.     private Boolean empty;

  76.     /** A {@code TransportUnit} may belong to a group of {@code TransportUnits}. */
  77.     @Column(name = "C_GROUP_NAME")
  78.     private String groupName;

  79.     /** Date when the {@code TransportUnit} has been moved to the current {@link Location}. */
  80.     @Column(name = "C_ACTUAL_LOCATION_DATE")
  81.     private LocalDateTime actualLocationDate;

  82.     /** Weight of the {@code TransportUnit}. */
  83.     @Embedded
  84.     @AttributeOverride(name = "unitType", column = @Column(name = "C_WEIGHT_UOM", length = CoreTypeDefinitions.QUANTITY_LENGTH))
  85.     @AttributeOverride(name = "magnitude", column = @Column(name = "C_WEIGHT"))
  86.     private Weight weight = Weight.ZERO;

  87.     /** State of the {@code TransportUnit}. */
  88.     @Column(name = "C_STATE")
  89.     private String state = TransportUnitState.AVAILABLE.name();

  90.     /** The current {@link Location} of the {@code TransportUnit}. */
  91.     @ManyToOne
  92.     @JoinColumn(name = "C_ACTUAL_LOCATION", nullable = false, foreignKey = @ForeignKey(name = "COM_TU_FK_LOC_ACTUAL"))
  93.     private Location actualLocation;

  94.     /** The target {@link Location} of the {@code TransportUnit}. This property is set when a {@code TransportOrder} is started. */
  95.     @ManyToOne
  96.     @JoinColumn(name = "C_TARGET_LOCATION", foreignKey = @ForeignKey(name = "COM_TU_FK_LOC_TARGET"))
  97.     private Location targetLocation;

  98.     /** The {@link TransportUnitType} of the {@code TransportUnit}. */
  99.     @ManyToOne
  100.     @JoinColumn(name = "C_TRANSPORT_UNIT_TYPE", nullable = false, foreignKey = @ForeignKey(name = "COM_TU_FK_TUT"))
  101.     private TransportUnitType transportUnitType;

  102.     /** Owning {@code TransportUnit}. */
  103.     @ManyToOne
  104.     @JoinColumn(name = "C_PARENT", foreignKey = @ForeignKey(name = "COM_TU_FK_TU_PARENT"))
  105.     private TransportUnit parent;

  106.     /** The {@code User} who performed the last inventory action on the {@code TransportUnit}. */
  107.     @Column(name = "C_INVENTORY_USER")
  108.     private String inventoryUser;

  109.     /** Date of last inventory check. */
  110.     @Column(name = "C_INVENTORY_DATE")
  111.     private LocalDateTime inventoryDate;

  112.     /** A set of all child {@code TransportUnit}s, ordered by id. */
  113.     @OneToMany(mappedBy = "parent", cascade = {CascadeType.MERGE, CascadeType.PERSIST})
  114.     @OrderBy("actualLocationDate DESC")
  115.     private Set<TransportUnit> children = new HashSet<>();

  116.     /** A List of errors occurred on the {@code TransportUnit}. */
  117.     @OneToMany(mappedBy = "transportUnit", cascade = {CascadeType.ALL})
  118.     @NotAudited
  119.     private List<UnitError> errors = new ArrayList<>(0);

  120.     /** Tracks all active {@link TransportUnitReservation}s on this {@link TransportUnit}. */
  121.     @OneToMany(mappedBy = "transportUnit", cascade = {CascadeType.ALL})
  122.     private List<TransportUnitReservation> reservations;

  123.    
  124.     /*~ ----------------------------- constructors ------------------- */
  125.     /** Dear JPA... */
  126.     protected TransportUnit() { }

  127.     @Default
  128.     @ConstructorProperties({"barcode"})
  129.     public TransportUnit(@NotNull Barcode barcode) {
  130.         Assert.notNull(barcode, "Barcode must not be null");
  131.         this.barcode = barcode;
  132.         initInventory();
  133.     }

  134.     /**
  135.      * Create a new {@code TransportUnit} with an unique {@link Barcode}.
  136.      *
  137.      * @param barcode The unique identifier of this {@code TransportUnit} is the {@link Barcode} - must not be {@literal null}
  138.      * @param transportUnitType The {@code TransportUnitType} of this {@code TransportUnit} - must not be {@literal null}
  139.      * @param actualLocation The current {@code Location} of this {@code TransportUnit} - must not be {@literal null}
  140.      * @throws IllegalArgumentException when one of the params is {@literal null}
  141.      */
  142.     @ConstructorProperties({"barcode", "transportUnitType", "actualLocation"})
  143.     public TransportUnit(@NotNull Barcode barcode, @NotNull TransportUnitType transportUnitType, @NotNull Location actualLocation) {
  144.         Assert.notNull(barcode, "Barcode must not be null");
  145.         Assert.notNull(transportUnitType, "TransportUnitType must not be null");
  146.         Assert.notNull(actualLocation, "ActualLocation must not be null");
  147.         this.barcode = barcode;
  148.         setTransportUnitType(transportUnitType);
  149.         setActualLocation(actualLocation);
  150.         initInventory();
  151.     }

  152.     /*~ ----------------------------- methods ------------------- */
  153.     /** Required for the Mapper. */
  154.     @Override
  155.     public void setPersistentKey(String pKey) {
  156.         super.setPersistentKey(pKey);
  157.     }

  158.     /**
  159.      * Get the actual {@link Location} of the {@code TransportUnit}.
  160.      *
  161.      * @return The {@link Location} where the {@code TransportUnit} is placed on
  162.      */
  163.     public Location getActualLocation() {
  164.         return actualLocation;
  165.     }

  166.     /**
  167.      * Place the {@code TransportUnit} and all its children to a {@link Location}.
  168.      *
  169.      * @param actualLocation The new {@link Location} of the {@code TransportUnit} and all its children
  170.      * @throws IllegalArgumentException when {@code actualLocation} is {@literal null}
  171.      */
  172.     public void setActualLocation(Location actualLocation) {
  173.         Assert.notNull(actualLocation, "ActualLocation must not be null, this: " + this);
  174.         this.actualLocation = actualLocation;
  175.         this.actualLocationDate = LocalDateTime.now();
  176.         this.actualLocation.setLastMovement(this.actualLocationDate);
  177.         if (this.getChildren() != null) {
  178.             this.getChildren().forEach(child -> child.setActualLocation(actualLocation));
  179.         }
  180.     }

  181.     /**
  182.      * Initialize inventory info of the {@code TransportUnit}.
  183.      */
  184.     public void initInventory() {
  185.         setInventoryUser("init");
  186.         setInventoryDate(LocalDateTime.now());
  187.     }

  188.     /**
  189.      * Get the target {@link Location} of the {@code TransportUnit}. This property can not be {@literal null} when an active {@code
  190.      * TransportOrder} exists.
  191.      *
  192.      * @return The target location
  193.      */
  194.     public Location getTargetLocation() {
  195.         return this.targetLocation;
  196.     }

  197.     /**
  198.      * Set the target {@link Location} of the {@code TransportUnit}. Shall only be set in combination with an active {@code
  199.      * TransportOrder}.
  200.      *
  201.      * @param targetLocation The target {@link Location} where this {@code TransportUnit} shall be transported to
  202.      */
  203.     public void setTargetLocation(Location targetLocation) {
  204.         this.targetLocation = targetLocation;
  205.     }

  206.     /**
  207.      * Indicates whether the {@code TransportUnit} is empty or not.
  208.      *
  209.      * @return {@literal true} if empty, {@literal false} if not empty, {@literal null} when not defined
  210.      */
  211.     public Boolean getEmpty() {
  212.         return this.empty;
  213.     }

  214.     /**
  215.      * Marks the {@code TransportUnit} to be empty.
  216.      *
  217.      * @param empty {@literal true} to mark the {@code TransportUnit} as empty, {@literal false} to mark it as not empty and {@literal null}
  218.      * for no definition
  219.      */
  220.     public void setEmpty(Boolean empty) {
  221.         this.empty = empty;
  222.     }

  223.     /**
  224.      * Get the groupId.
  225.      *
  226.      * @return The groupId
  227.      */
  228.     public String getGroupName() {
  229.         return groupName;
  230.     }

  231.     /**
  232.      * Set the groupId.
  233.      *
  234.      * @param groupId The groupId
  235.      */
  236.     public void setGroupName(String groupId) {
  237.         this.groupName = groupId;
  238.     }

  239.     /**
  240.      * Returns the username of the User who performed the last inventory action on the {@code TransportUnit}.
  241.      *
  242.      * @return The username who did the last inventory check
  243.      */
  244.     public String getInventoryUser() {
  245.         return this.inventoryUser;
  246.     }

  247.     /**
  248.      * Set the username who performed the last inventory action on the {@code TransportUnit}.
  249.      *
  250.      * @param inventoryUser The username who did the last inventory check
  251.      */
  252.     public void setInventoryUser(String inventoryUser) {
  253.         this.inventoryUser = inventoryUser;
  254.     }

  255.     /**
  256.      * Number of {@code TransportUnit}s belonging to the {@code TransportUnit}.
  257.      *
  258.      * @return The number of all {@code TransportUnit}s belonging to this one
  259.      */
  260.     public int getNoTransportUnits() {
  261.         return this.children.size();
  262.     }

  263.     /**
  264.      * Returns the date when the {@code TransportUnit} moved to the actualLocation.
  265.      *
  266.      * @return The timestamp when the {@code TransportUnit} moved the last time
  267.      */
  268.     public LocalDateTime getActualLocationDate() {
  269.         return actualLocationDate;
  270.     }

  271.     /**
  272.      * Returns the timestamp of the last inventory check of the {@code TransportUnit}.
  273.      *
  274.      * @return The timestamp of the last inventory check of the {@code TransportUnit}.
  275.      */
  276.     public LocalDateTime getInventoryDate() {
  277.         return inventoryDate;
  278.     }

  279.     /**
  280.      * Set the timestamp of the last inventory action of the {@code TransportUnit}.
  281.      *
  282.      * @param inventoryDate The timestamp of the last inventory check
  283.      * @throws IllegalArgumentException when {@code inventoryDate} is {@literal null}
  284.      */
  285.     public void setInventoryDate(LocalDateTime inventoryDate) {
  286.         Assert.notNull(inventoryDate, () -> "InventoryDate must not be null, this: " + this);
  287.         this.inventoryDate = inventoryDate;
  288.     }

  289.     /**
  290.      * Returns the current weight of the {@code TransportUnit}.
  291.      *
  292.      * @return The current weight of the {@code TransportUnit}
  293.      */
  294.     public Weight getWeight() {
  295.         return weight;
  296.     }

  297.     /**
  298.      * Sets the current weight of the {@code TransportUnit}.
  299.      *
  300.      * @param weight The current weight of the {@code TransportUnit}
  301.      */
  302.     public void setWeight(Weight weight) {
  303.         this.weight = weight;
  304.     }

  305.     public List<UnitError> getErrors() {
  306.         return this.errors;
  307.     }

  308.     /**
  309.      * Add an error to the {@code TransportUnit}.
  310.      *
  311.      * @param error An {@link UnitError} to be added
  312.      * @return The key.
  313.      * @throws IllegalArgumentException when {@code error} is {@literal null}
  314.      */
  315.     public UnitError addError(UnitError error) {
  316.         Assert.notNull(error, () -> "Error must not be null, this: " + this);
  317.         error.setTransportUnit(this);
  318.         this.errors.add(error);
  319.         return error;
  320.     }

  321.     /**
  322.      * Checks whether this {@code TransportUnit} has one or more reservations.
  323.      *
  324.      * @return {@literal true} if so
  325.      */
  326.     public boolean hasReservations() {
  327.         return this.reservations != null && !this.reservations.isEmpty();
  328.     }

  329.     /**
  330.      * Return the state of the {@code TransportUnit}.
  331.      *
  332.      * @return The current state of the {@code TransportUnit}
  333.      */
  334.     public String getState() {
  335.         return this.state;
  336.     }

  337.     /**
  338.      * Set the state of the {@code TransportUnit}.
  339.      *
  340.      * @param state The state to set on the {@code TransportUnit}
  341.      */
  342.     public void setState(String state) {
  343.         this.state = state;
  344.     }

  345.     /**
  346.      * Return the {@link TransportUnitType} of the {@code TransportUnit}.
  347.      *
  348.      * @return The {@link TransportUnitType} the {@code TransportUnit} belongs to
  349.      */
  350.     public TransportUnitType getTransportUnitType() {
  351.         return this.transportUnitType;
  352.     }

  353.     /**
  354.      * Set the {@link TransportUnitType} of the {@code TransportUnit}.
  355.      *
  356.      * @param transportUnitType The type of the {@code TransportUnit}
  357.      */
  358.     public void setTransportUnitType(TransportUnitType transportUnitType) {
  359.         Assert.notNull(transportUnitType, () -> "TransportUnitType must not be null, this: " + this);
  360.         this.transportUnitType = transportUnitType;
  361.     }

  362.     /**
  363.      * Return the {@link Barcode} of the {@code TransportUnit}.
  364.      *
  365.      * @return The current {@link Barcode}
  366.      */
  367.     public Barcode getBarcode() {
  368.         return barcode;
  369.     }

  370.     /**
  371.      * Returns the parent {@code TransportUnit}.
  372.      *
  373.      * @return the parent.
  374.      */
  375.     public TransportUnit getParent() {
  376.         return parent;
  377.     }

  378.     /**
  379.      * Set a parent {@code TransportUnit}.
  380.      *
  381.      * @param parent The parent to set.
  382.      */
  383.     public void setParent(TransportUnit parent) {
  384.         this.parent = parent;
  385.     }

  386.     /**
  387.      * Get all child {@code TransportUnit}s.
  388.      *
  389.      * @return the transportUnits.
  390.      */
  391.     public Set<TransportUnit> getChildren() {
  392.         return this.children;
  393.     }

  394.     /**
  395.      * Add a {@code TransportUnit} to the children.
  396.      *
  397.      * @param transportUnit The {@code TransportUnit} to be added to the list of children
  398.      * @throws IllegalArgumentException when transportUnit is {@literal null}
  399.      */
  400.     public void addChild(TransportUnit transportUnit) {
  401.         Assert.notNull(transportUnit, () -> "TransportUnitType must not be null, this: " + this);
  402.         if (transportUnit.hasParent()) {
  403.             if (transportUnit.getParent().equals(this)) {

  404.                 // if this instance is already the parent, we just return
  405.                 return;
  406.             }

  407.             // disconnect post from it's current relationship
  408.             transportUnit.getParent().removeChild(transportUnit);
  409.         }

  410.         // make this instance the new parent
  411.         transportUnit.setParent(this);
  412.         this.children.add(transportUnit);
  413.     }

  414.     /**
  415.      * Checks whether this {@code TransportUnit} has a parent {@code TransportUnit} or not.
  416.      *
  417.      * @return {@code true} it has a parent, otherwise {@code false}
  418.      */
  419.     public boolean hasParent() {
  420.         return parent != null;
  421.     }

  422.     /**
  423.      * Checks whether this {@code TransportUnit} has child {@code TransportUnit}s or not.
  424.      *
  425.      * @return {@code true} it has children, otherwise {@code false}
  426.      */
  427.     public boolean hasChildren() {
  428.         return this.children != null && !this.children.isEmpty();
  429.     }

  430.     /**
  431.      * Remove a {@code TransportUnit} from the list of children.
  432.      *
  433.      * @param transportUnit The {@code TransportUnit} to be removed from the list of children
  434.      * @throws IllegalArgumentException when {@code transportUnit} is {@literal null} or not a child of this instance
  435.      */
  436.     public void removeChild(TransportUnit transportUnit) {
  437.         Assert.notNull(transportUnit, () -> "TransportUnit must not be null, this: " + this);
  438.         // make sure this is the parent before we break the relationship
  439.         if (transportUnit.parent == null || !transportUnit.parent.equals(this)) {
  440.             throw new IllegalArgumentException("Child TransportUnit not associated with this instance, this: " + this);
  441.         }
  442.         transportUnit.setParent(null);
  443.         this.children.remove(transportUnit);
  444.     }

  445.     /**
  446.      * {@inheritDoc}
  447.      *
  448.      * Return the {@link Barcode} as String.
  449.      */
  450.     @Override
  451.     public String toString() {
  452.         return barcode.toString();
  453.     }

  454.     /**
  455.      * {@inheritDoc}
  456.      *
  457.      * Uses barcode for comparison.
  458.      */
  459.     @Override
  460.     public boolean equals(Object o) {
  461.         if (this == o) return true;
  462.         if (o == null || getClass() != o.getClass()) return false;
  463.         var that = (TransportUnit) o;
  464.         return barcode.equals(that.barcode);
  465.     }

  466.     /**
  467.      * {@inheritDoc}
  468.      *
  469.      * Uses barcode for calculation.
  470.      */
  471.     @Override
  472.     public int hashCode() {
  473.         return Objects.hash(barcode);
  474.     }
  475. }