All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.opentcs.data.order.OrderSequence Maven / Gradle / Ivy

There is a newer version: 6.2.0
Show newest version
/**
 * Copyright (c) The openTCS Authors.
 *
 * This program is free software and subject to the MIT license. (For details,
 * see the licensing information (LICENSE.txt) you should have received with
 * this copy of the software.)
 */
package org.opentcs.data.order;

import static java.util.Objects.requireNonNull;
import static org.opentcs.data.order.OrderSequenceHistoryCodes.SEQUENCE_COMPLETED;
import static org.opentcs.data.order.OrderSequenceHistoryCodes.SEQUENCE_CREATED;
import static org.opentcs.data.order.OrderSequenceHistoryCodes.SEQUENCE_FINISHED;
import static org.opentcs.data.order.OrderSequenceHistoryCodes.SEQUENCE_ORDER_APPENDED;
import static org.opentcs.data.order.OrderSequenceHistoryCodes.SEQUENCE_PROCESSING_VEHICLE_CHANGED;
import static org.opentcs.util.Assertions.checkArgument;
import static org.opentcs.util.Assertions.checkInRange;

import jakarta.annotation.Nonnull;
import java.io.Serializable;
import java.util.ArrayList;
import java.util.Collections;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import org.opentcs.data.ObjectHistory;
import org.opentcs.data.TCSObject;
import org.opentcs.data.TCSObjectReference;
import org.opentcs.data.model.Vehicle;

/**
 * Describes a process spanning multiple {@link TransportOrder}s which are to be executed
 * subsequently by the same {@link Vehicle}.
 * 

* The most important rules for order sequence processing are: *

*
    *
  • Only transport orders that have not yet been activated may be added to an order sequence. * Allowing them to be added at a later point of time would imply that, due to concurrency in the * kernel, a transport order might happen to be dispatched at the same time or shortly after it is * added to a sequence, regardless of if its predecessors in the sequence have already been finished * or not.
  • *
  • The intendedVehicle of a transport order being added to an order sequence must be * the same as that of the sequence itself. * If it is null in the sequence, a vehicle that will process all orders in the * sequence will be chosen automatically once the first order in the sequence is dispatched.
  • *
  • If an order sequence is marked as complete and all transport orders belonging to it * have arrived in state FINISHED or FAILED, it will be marked as * finished implicitly.
  • *
  • If a transport order belonging to an order sequence fails and the sequence's * failureFatal flag is set, all subsequent orders in the sequence will automatically be * considered (and marked as) failed, too, and the order sequence will implicitly be marked as * complete (and finished).
  • *
*/ public class OrderSequence extends TCSObject implements Serializable { /** * The type of this order sequence. * An order sequence and all transport orders it contains share the same type. */ @Nonnull private final String type; /** * Transport orders belonging to this sequence that still need to be processed. */ private final List> orders; /** * The index of the order that was last finished in the sequence. * -1 if none was finished, yet. */ private final int finishedIndex; /** * Indicates whether this order sequence is complete and will not be extended by more orders. */ private final boolean complete; /** * Indicates whether this order sequence has been processed completely. */ private final boolean finished; /** * Indicates whether the failure of one order in this sequence is fatal to all subsequent orders. */ private final boolean failureFatal; /** * The vehicle that is intended to process this order sequence. * If this sequence is free to be processed by any vehicle, this is null. */ private final TCSObjectReference intendedVehicle; /** * The vehicle processing this order sequence, or null, if no vehicle has been * assigned to it, yet. */ private final TCSObjectReference processingVehicle; /** * Creates a new OrderSequence. * * @param name This sequence's name. */ public OrderSequence(String name) { super( name, new HashMap<>(), new ObjectHistory().withEntryAppended(new ObjectHistory.Entry(SEQUENCE_CREATED)) ); this.type = OrderConstants.TYPE_NONE; this.orders = List.of(); this.finishedIndex = -1; this.complete = false; this.finished = false; this.failureFatal = false; this.intendedVehicle = null; this.processingVehicle = null; } private OrderSequence( String name, Map properties, ObjectHistory history, String type, TCSObjectReference intendedVehicle, List> orders, int finishedIndex, boolean complete, boolean failureFatal, boolean finished, TCSObjectReference processingVehicle ) { super(name, properties, history); this.type = requireNonNull(type, "type"); this.intendedVehicle = intendedVehicle; this.orders = new ArrayList<>(requireNonNull(orders, "orders")); this.finishedIndex = finishedIndex; this.complete = complete; this.failureFatal = failureFatal; this.finished = finished; this.processingVehicle = processingVehicle; } @Override public OrderSequence withProperty(String key, String value) { return new OrderSequence( getName(), propertiesWith(key, value), getHistory(), type, intendedVehicle, orders, finishedIndex, complete, failureFatal, finished, processingVehicle ); } @Override public OrderSequence withProperties(Map properties) { return new OrderSequence( getName(), properties, getHistory(), type, intendedVehicle, orders, finishedIndex, complete, failureFatal, finished, processingVehicle ); } @Override public TCSObject withHistoryEntry(ObjectHistory.Entry entry) { return new OrderSequence( getName(), getProperties(), getHistory().withEntryAppended(entry), type, intendedVehicle, orders, finishedIndex, complete, failureFatal, finished, processingVehicle ); } @Override public TCSObject withHistory(ObjectHistory history) { return new OrderSequence( getName(), getProperties(), history, type, intendedVehicle, orders, finishedIndex, complete, failureFatal, finished, processingVehicle ); } /** * Returns this order sequence's type. * * @return This order sequence's type. */ @Nonnull public String getType() { return type; } /** * Creates a copy of this object, with the given type. * * @param type The type to be set in the copy. * @return A copy of this object, differing in the given value. */ public OrderSequence withType(String type) { return new OrderSequence( getName(), getProperties(), getHistory(), type, intendedVehicle, orders, finishedIndex, complete, failureFatal, finished, processingVehicle ); } /** * Returns the list of orders making up this sequence. * * @return The list of orders making up this sequence. */ public List> getOrders() { return Collections.unmodifiableList(orders); } /** * Creates a copy of this object, with the given order. * * @param order The value to be set in the copy. * @return A copy of this object, differing in the given value. */ public OrderSequence withOrder(TCSObjectReference order) { checkArgument(!complete, "Sequence complete, cannot add order"); checkArgument(!orders.contains(order), "Sequence already contains order %s", order); return new OrderSequence( getName(), getProperties(), historyForAppendedOrder(order), type, intendedVehicle, ordersWithAppended(order), finishedIndex, complete, failureFatal, finished, processingVehicle ); } /** * Returns the next order in the sequence that hasn't been finished, yet. * * @return null if this sequence has been finished already or * currently doesn't have any unfinished orders, else the order after the one * that was last finished. */ public TCSObjectReference getNextUnfinishedOrder() { // If the whole sequence has been finished already, return null. if (finished) { return null; } // If the sequence has not been marked as finished but the last order in the // list has been, return null, too. else if (finishedIndex + 1 >= orders.size()) { return null; } // Otherwise just get the order after the one that was last finished. else { return orders.get(finishedIndex + 1); } } /** * Returns the index of the order that was last finished in the sequence, or * -1, if none was finished, yet. * * @return the index of the order that was last finished in the sequence. */ public int getFinishedIndex() { return finishedIndex; } /** * Creates a copy of this object, with the given finished index. * * @param finishedIndex The value to be set in the copy. * @return A copy of this object, differing in the given value. */ public OrderSequence withFinishedIndex(int finishedIndex) { checkInRange(finishedIndex, 0, orders.size() - 1, "finishedIndex"); return new OrderSequence( getName(), getProperties(), getHistory(), type, intendedVehicle, orders, finishedIndex, complete, failureFatal, finished, processingVehicle ); } /** * Indicates whether this order sequence is complete and will not be extended * by more orders. * * @return true if, and only if, this order sequence is complete * and will not be extended by more orders. */ public boolean isComplete() { return complete; } /** * Creates a copy of this object, with the given complete flag. * * @param complete The value to be set in the copy. * @return A copy of this object, differing in the given value. */ public OrderSequence withComplete(boolean complete) { return new OrderSequence( getName(), getProperties(), historyForComplete(complete), type, intendedVehicle, orders, finishedIndex, complete, failureFatal, finished, processingVehicle ); } /** * Indicates whether this order sequence has been processed completely. * (Note that processed completely does not necessarily mean * finished successfully; it is possible that one or more transport * orders belonging to this sequence have failed.) * * @return true if, and only if, this order sequence has been * processed completely. */ public boolean isFinished() { return finished; } /** * Creates a copy of this object, with the given finished flag. * * @param finished The value to be set in the copy. * @return A copy of this object, differing in the given value. */ public OrderSequence withFinished(boolean finished) { return new OrderSequence( getName(), getProperties(), historyForFinished(finished), type, intendedVehicle, orders, finishedIndex, complete, failureFatal, finished, processingVehicle ); } /** * Indicates whether the failure of a single order in this sequence implies * that all subsequent orders in this sequence are to be considered failed, * too. * * @return true if, and only if, the failure of an order in this * sequence implies the failure of all subsequent orders. */ public boolean isFailureFatal() { return failureFatal; } /** * Creates a copy of this object, with the given failure-fatal flag. * * @param failureFatal The value to be set in the copy. * @return A copy of this object, differing in the given value. */ public OrderSequence withFailureFatal(boolean failureFatal) { return new OrderSequence( getName(), getProperties(), getHistory(), type, intendedVehicle, orders, finishedIndex, complete, failureFatal, finished, processingVehicle ); } /** * Returns a reference to the vehicle that is intended to process this * order sequence. * * @return A reference to the vehicle that is intended to process this * order sequence. If this sequence is free to be processed by any vehicle, * null is returned. */ public TCSObjectReference getIntendedVehicle() { return intendedVehicle; } /** * Creates a copy of this object, with the given intended vehicle. * * @param intendedVehicle The value to be set in the copy. * @return A copy of this object, differing in the given value. */ public OrderSequence withIntendedVehicle(TCSObjectReference intendedVehicle) { return new OrderSequence( getName(), getProperties(), getHistory(), type, intendedVehicle, orders, finishedIndex, complete, failureFatal, finished, processingVehicle ); } /** * Returns a reference to the vehicle currently processing this sequence. * * @return A reference to the vehicle currently processing this sequence. If * this sequence has not been processed, yet, null is * returned. */ public TCSObjectReference getProcessingVehicle() { return processingVehicle; } /** * Creates a copy of this object, with the given processing vehicle. * * @param processingVehicle The value to be set in the copy. * @return A copy of this object, differing in the given value. */ public OrderSequence withProcessingVehicle(TCSObjectReference processingVehicle) { return new OrderSequence( getName(), getProperties(), historyForNewProcessingVehicle(processingVehicle), type, intendedVehicle, orders, finishedIndex, complete, failureFatal, finished, processingVehicle ); } private List> ordersWithAppended( @Nonnull TCSObjectReference order ) { List> result = new ArrayList<>(orders.size() + 1); result.addAll(orders); result.add(order); return result; } private ObjectHistory historyForNewProcessingVehicle(TCSObjectReference ref) { return getHistory().withEntryAppended( new ObjectHistory.Entry( SEQUENCE_PROCESSING_VEHICLE_CHANGED, ref == null ? "" : ref.getName() ) ); } private ObjectHistory historyForAppendedOrder(TCSObjectReference ref) { return getHistory().withEntryAppended( new ObjectHistory.Entry( SEQUENCE_ORDER_APPENDED, ref == null ? "" : ref.getName() ) ); } private ObjectHistory historyForFinished(boolean finished) { return finished ? getHistory().withEntryAppended( new ObjectHistory.Entry(SEQUENCE_FINISHED) ) : getHistory(); } private ObjectHistory historyForComplete(boolean complete) { return complete ? getHistory().withEntryAppended( new ObjectHistory.Entry(SEQUENCE_COMPLETED) ) : getHistory(); } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy