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

org.cloudsimplus.core.CloudSimEntity Maven / Gradle / Ivy

/*
 * Title:        CloudSim Toolkit
 * Description:  CloudSim (Cloud Simulation) Toolkit for Modeling and Simulation of Clouds
 * Licence:      GPL - http://www.gnu.org/copyleft/gpl.html
 *
 * Copyright (c) 2009-2012, The University of Melbourne, Australia
 */
package org.cloudsimplus.core;

import lombok.EqualsAndHashCode;
import lombok.Getter;
import lombok.NonNull;
import lombok.Setter;
import org.cloudsimplus.core.events.CloudSimEvent;
import org.cloudsimplus.core.events.SimEvent;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import java.util.function.Predicate;

import static java.util.Objects.requireNonNull;
import static java.util.Objects.requireNonNullElse;

/**
 * Represents a simulation entity. An entity handles events and can
 * send events to other entities.
 *
 * @author Marcos Dias de Assuncao
 * @since CloudSim Toolkit 1.0
 */
@EqualsAndHashCode(onlyExplicitlyIncluded = true)
public abstract class CloudSimEntity implements SimEntity {
    private static final Logger LOGGER = LoggerFactory.getLogger(CloudSimEntity.class.getSimpleName());

    @Getter @EqualsAndHashCode.Include
    private long id;
    @Getter @NonNull @EqualsAndHashCode.Include
    private final Simulation simulation;

    @Getter
    private String name;

    @Getter
    private double startTime;

    @Getter
    private double shutdownTime;

    @Getter @Setter
    private State state;

    /**
     * The buffer for selected incoming events.
     */
    private SimEvent buffer;

    /**
     * Creates a new entity.
     *
     * @param simulation The CloudSimPlus instance that represents the simulation the Entity belongs to
     * @throws IllegalArgumentException when the entity name is invalid
     */
    public CloudSimEntity(@NonNull final Simulation simulation) {
        this.simulation = simulation;
        setId(-1);
        state = State.RUNNABLE;
        this.simulation.addEntity(this);
        this.startTime = -1;
        this.shutdownTime = -1;
    }

    /**
     * {@inheritDoc}.
     * It performs general initialization tasks that are common for every entity
     * and executes the specific entity startup code.
     *
     * @return {@inheritDoc}
     */
    @Override
    public final boolean start() {
        if(this.isStarted()){
            return false;
        }

        startInternal();
        this.startTime = simulation.clock();
        return true;
    }

    @Override
    public void shutdown() {
        if(this.state == State.FINISHED){
            return;
        }

        setState(State.FINISHED);
        this.shutdownTime = simulation.clock();

        /*
         * Since entities never get removed from the entities list, this can create
         * a memory leak with severe performance implications.
         * Here the finished entity is purged from that list
         * to improve performance of large-scale experiments.
         */
        ((CloudSimPlus)simulation).removeFinishedEntity(this);
    }

    /**
     * Defines the logic to be performed by the entity when the simulation starts.
     */
    protected abstract void startInternal();

    @Override
    public boolean schedule(final SimEntity dest, final double delay, final int tag, final Object data) {
        return schedule(new CloudSimEvent(delay, this, dest, tag, data));
    }

    @Override
    public boolean schedule(final double delay, final int tag, final Object data) {
        return schedule(this, delay, tag, data);
    }

    @Override
    public boolean schedule(final double delay, final int tag) {
        return schedule(this, delay, tag, null);
    }

    @Override
    public boolean schedule(final SimEntity dest, final double delay, final int tag) {
        return schedule(dest, delay, tag, null);
    }

    @Override
    public boolean schedule(final int tag, final Object data) {
        return schedule(this, 0, tag, data);
    }

    @Override
    public boolean schedule(final SimEvent evt) {
        if (canSendEvent(evt)) {
            simulation.send(evt);
            return true;
        }

        return false;
    }

    /**
     * If the simulation has finished and an  {@link CloudSimTag#SIMULATION_END}
     * message is sent, it has to be processed to enable entities to shut down.
     */
    private boolean canSendEvent(final SimEvent evt) {
        if (simulation.isRunning() || evt.getTag() == CloudSimTag.SIMULATION_END) {
            return true;
        }

        LOGGER.warn(
            "{}: {}: Cannot send events before simulation starts or after it finishes. Trying to send message {} to {}",
            getSimulation().clockStr(), this, evt.getTag(), evt.getDestination());
        return false;
    }

    /**
     * Sends an event to another entity with no delay.
     *
     * @param dest the destination entity
     * @param tag  a tag representing the type of event.
     * @param data The data to be sent with the event.
     */
    public void scheduleNow(final SimEntity dest, final int tag, final Object data) {
        schedule(dest, 0, tag, data);
    }

    /**
     * Sends an event to another entity with no attached data and no delay.
     *
     * @param dest the destination entity
     * @param tag  a user-defined number representing the type of event.
     */
    public void scheduleNow(final SimEntity dest, final int tag) {
        schedule(dest, 0, tag, null);
    }

    /**
     * Sends a high priority event to another entity with no delay.
     *
     * @param dest the destination entity
     * @param tag  a tag representing the type of event.
     * @param data The data to be sent with the event.
     */
    public void scheduleFirstNow(final SimEntity dest, final int tag, final Object data) {
        scheduleFirst(dest, 0, tag, data);
    }

    /**
     * Sends a high priority event to another entity with no attached data and no delay.
     * @param dest the destination entity
     * @param tag  a user-defined number representing the type of event.
     */
    public void scheduleFirstNow(final SimEntity dest, final int tag) {
        scheduleFirst(dest, 0, tag, null);
    }

    /**
     * Sends a high priority event to another entity and with no attached data.
     *
     * @param dest  the destination entity
     * @param delay How many seconds after the current simulation time the event should be sent
     * @param tag   a user-defined number representing the type of event.
     */
    public void scheduleFirst(final SimEntity dest, final double delay, final int tag) {
        scheduleFirst(dest, delay, tag, null);
    }

    /**
     * Sends a high priority event to another entity.
     *
     * @param dest  the destination entity
     * @param delay How many seconds after the current simulation time the event should be sent
     * @param tag   a user-defined number representing the type of event.
     * @param data  The data to be sent with the event.
     */
    public void scheduleFirst(final SimEntity dest, final double delay, final int tag, final Object data) {
        final var evt = new CloudSimEvent(delay, this, dest, tag, data);
        if (canSendEvent(evt)) {
            simulation.sendFirst(evt);
        }
    }

    /**
     * Sets the entity to be inactive for a time period.
     *
     * @param delay the time period for which the entity will be inactive
     */
    public void pause(final double delay) {
        if (delay < 0) {
            throw new IllegalArgumentException("Negative delay supplied.");
        }

        if (simulation.isRunning()) {
            simulation.pauseEntity(this, delay);
        }
    }

    /**
     * Extracts the first event matching a predicate waiting in the entity's
     * deferred queue.
     *
     * @param predicate The event selection predicate
     * @return the simulation event;
     *         or {@link SimEvent#NULL} if not found or the simulation is not running
     */
    public SimEvent selectEvent(final Predicate predicate) {
        if (simulation.isRunning()) {
            return simulation.select(this, predicate);
        }

        return SimEvent.NULL;
    }

    /**
     * Cancels the first event from the future event queue that matches a given predicate
     * and that was submitted by this entity, then removes it from the queue.
     *
     * @param predicate the event selection predicate
     * @return the removed event or {@link SimEvent#NULL} if not found
     */
    public SimEvent cancelEvent(final Predicate predicate) {
        return simulation.isRunning() ? simulation.cancel(this, predicate) : SimEvent.NULL;
    }

    /**
     * Gets the first event matching a predicate from the deferred queue, or if
     * none match, wait for a matching event to arrive.
     *
     * @param predicate The predicate to match
     * @return the simulation event;
     *         or {@link SimEvent#NULL} if not found or the simulation is not running
     */
    public SimEvent getNextEvent(final Predicate predicate) {
        if (simulation.isRunning()) {
            return selectEvent(predicate);
        }

        return SimEvent.NULL;
    }

    /**
     * Gets the first event waiting in the entity's deferred queue, or if there
     * are none, wait for an event to arrive.
     *
     * @return the simulation event;
     *         or {@link SimEvent#NULL} if not found or the simulation is not running
     */
    public SimEvent getNextEvent() {
        return getNextEvent(Simulation.ANY_EVT);
    }

    /**
     * Waits for an event matching a specific predicate. This method does not
     * check the entity's deferred queue.
     *
     * @param predicate the predicate to match
     */
    public void waitForEvent(final Predicate predicate) {
        if (simulation.isRunning()) {
            simulation.wait(this, predicate);
            state = State.WAITING;
        }
    }

    @Override
    public void run() {
        run(Double.MAX_VALUE);
    }

    public void run(final double until) {
        var evt = requireNonNullElse(buffer, getNextEvent(e -> e.getTime() <= until));

        while (evt != SimEvent.NULL) {
            processEvent(evt);
            if (state != State.RUNNABLE) {
                break;
            }

            evt = getNextEvent(e -> e.getTime() <= until);
        }

        buffer = null;
    }

    @Override
    public SimEntity setName(@NonNull final String name) throws IllegalArgumentException {
        if (name.isBlank()) {
            throw new IllegalArgumentException("Entity names cannot be empty.");
        }

        this.name = name;
        return this;
    }

    /**
     * Sets the entity id and defines its name based on such ID.
     *
     * @param id the new id
     */
    protected final void setId(final int id) {
        this.id = id;
        setAutomaticName();
    }

    /**
     * Sets an automatic generated name for the entity.
     */
    private void setAutomaticName() {
        final long id = this.id >= 0 ? this.id : this.simulation.getNumEntities();
        this.name = "%s%d".formatted(getClass().getSimpleName(), id);
    }

    /**
     * Sets the event buffer.
     *
     * @param evt the new event buffer
     */
    protected void setEventBuffer(@NonNull final SimEvent evt) {
        this.buffer = evt;
    }

    /**
     * Sends an event/message to another entity by delaying the
     * simulation time from the current time, with a tag representing the event
     * type.
     * @param dest the destination entity
     * @param delay How many seconds after the current simulation time the event should be sent.
     *              If delay is a negative number, then it will be changed to 0
     * @param tag a user-defined number representing the type of event/message
     * @param data A reference to data to be sent with the event
     */
    protected void send(final SimEntity dest, double delay, final int tag, final Object data) {
        requireNonNull(dest);
        if (dest.getId() < 0) {
            LOGGER.error("{}.send(): invalid entity id {} for {}", getName(), dest.getId(), dest);
            return;
        }

        // if delay is negative, then it doesn't make sense. So resets to 0.0
        if (delay < 0) {
            delay = 0;
        }

        if (Double.isInfinite(delay)) {
            throw new IllegalArgumentException("The specified delay is infinite value");
        }

        // only considers network delay when sending messages between different entities
        if (dest.getId() != getId()) {
            delay += getNetworkDelay(this, dest);
        }

        schedule(dest, delay, tag, data);
    }

    /**
     * Sends an event/message to another entity by delaying the
     * simulation time from the current time, with a tag representing the event
     * type.
     *  @param dest    the destination entity
     * @param delay       How many seconds after the current simulation time the event should be sent.
     *                    If delay is a negative number, then it will be changed to 0
     * @param tag a user-defined number representing the type of an
     */
    protected void send(final SimEntity dest, final double delay, final int tag) {
        send(dest, delay, tag, null);
    }

    /**
     * Sends an event/message to another entity, with a tag representing the
     * event type.
     *  @param dest    the destination entity
     * @param tag a user-defined number representing the type of an
     *                    event/message
     * @param data        A reference to data to be sent with the event
     */
    protected void sendNow(final SimEntity dest, final int tag, final Object data) {
        send(dest, 0, tag, data);
    }

    /**
     * Sends an event/message to another entity, with a tag representing the
     * event type.
     *  @param dest    the destination entity
     * @param tag a user-defined number representing the event/message type
     */
    protected void sendNow(final SimEntity dest, final int tag) {
        send(dest, 0, tag, null);
    }

    /**
     * Gets the network delay associated to the sent of a message from a given
     * source to a given destination.
     *
     * @param src source of the message
     * @param dst destination of the message
     * @return the delay to send a message from src to dst
     */
    private double getNetworkDelay(final SimEntity src, final SimEntity dst) {
        return getSimulation().getNetworkTopology().getDelay(src, dst);
    }

    @Override
    public boolean isStarted() {
        return startTime > -1;
    }

    @Override
    public boolean isAlive() {
        return state != State.FINISHED;
    }

    @Override
    public boolean isFinished() {
        return state == State.FINISHED;
    }

    @Override
    public int compareTo(final SimEntity entity) {
        return Long.compare(this.getId(), entity.getId());
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy