Stopwatch.java

// Generated by delombok at Mon Jan 06 07:19:11 UTC 2025
package de.larssh.utils.time;

import static de.larssh.utils.function.ThrowingConsumer.throwing;
import static java.util.Collections.synchronizedList;
import static java.util.Collections.unmodifiableList;
import java.time.Duration;
import java.time.Instant;
import java.time.temporal.TemporalAccessor;
import java.time.temporal.TemporalField;
import java.util.Comparator;
import java.util.LinkedList;
import java.util.List;
import java.util.Objects;
import java.util.Optional;
import java.util.function.Consumer;
import java.util.stream.Stream;
import de.larssh.utils.Nullables;
import edu.umd.cs.findbugs.annotations.Nullable;
import edu.umd.cs.findbugs.annotations.SuppressFBWarnings;

/**
 * Implementation of a synchronized stopwatch starting at {@link Instant#now()}.
 * {@link #sinceStart()} can be used to retrieve the duration since the
 * stopwatch started.
 *
 * <p>
 * {@link #checkpoint(String)} adds a new checkpoint. It might be used for split
 * or lap times. {@link #sinceLast()} can be used to retrieve the duration since
 * the last checkpoint was created.
 */
public class Stopwatch {
	/**
	 * List of checkpoints
	 */
	private final List<Checkpoint> checkpoints = synchronizedList(new LinkedList<>());
	/**
	 * Instant at the stopwatches start
	 */
	private final Instant startInstant = Instant.now();
	/**
	 * Object used for locking
	 */
	private final Object lock = new Object();

	/**
	 * Adds a new checkpoint referenced by {@code name}.
	 *
	 * <p>
	 * {@code name} does not need to be unique. Multiple checkpoints with the same
	 * name might exist.
	 *
	 * @param name name to reference the checkpoint
	 * @return the new checkpoint
	 */
	public Checkpoint checkpoint(final String name) {
		synchronized (lock) {
			final Checkpoint checkpoint = new Checkpoint(this, name, Instant.now(), getLastCheckpoint());
			checkpoints.add(checkpoint);
			return checkpoint;
		}
	}

	/**
	 * List of checkpoints
	 *
	 * @return the list of checkpoints
	 */
	public List<Checkpoint> getCheckpoints() {
		return unmodifiableList(checkpoints);
	}

	/**
	 * The last created {@link Checkpoint} or empty if no checkpoint was created
	 *
	 * @return the last created checkpoint
	 */
	public Optional<Checkpoint> getLastCheckpoint() {
		return checkpoints.isEmpty() ? Optional.empty() : Optional.of(checkpoints.get(checkpoints.size() - 1));
	}

	/**
	 * Instant of the last created {@link Checkpoint} or the stopwatches starting
	 * time if no checkpoint was created
	 *
	 * @return the last created instant or the stopwatches starting time
	 */
	public Instant getLastInstant() {
		return checkpoints.isEmpty() ? getStartInstant() : checkpoints.get(checkpoints.size() - 1).getInstant();
	}

	/**
	 * Duration since the last created {@link Checkpoint} or the stopwatches
	 * starting time if no checkpoint was created
	 *
	 * @return the duration since last checkpoints instant or the stopwatches
	 *         starting time
	 */
	public Duration sinceLast() {
		return Duration.between(getLastInstant(), Instant.now());
	}

	/**
	 * Duration since the stopwatches start
	 *
	 * @return the duration since the stopwatches start
	 */
	public Duration sinceStart() {
		return Duration.between(getStartInstant(), Instant.now());
	}

	/**
	 * Waits for {@code duration} using {@code Thread#sleep(long)}. If reached, it
	 * times out at {@code timeoutSinceStart} after the {@link Stopwatch} start.
	 *
	 * @param duration          duration to wait
	 * @param timeoutSinceStart timeout duration since the {@link Stopwatch} start
	 * @return {@code true} if the timeout was not reached and {@code false} if the
	 *         timeout has been reached
	 * @throws InterruptedException if any thread has interrupted the current thread
	 */
	@SuppressWarnings("unused")
	@SuppressFBWarnings(value = "MDM_THREAD_YIELD", justification = "This is really intended to sleep for a specified duration.")
	public boolean waitFor(final Duration duration, final Duration timeoutSinceStart) throws InterruptedException {
		return waitFor(duration, timeoutSinceStart, throwing(waiting -> Thread.sleep(Nullables.orElseThrow(waiting).toMillis())));
	}

	/**
	 * Waits for {@code duration} using {@code wait}. If reached, it times out at
	 * {@code timeoutSinceStart} after the {@link Stopwatch} start.
	 *
	 * <p>
	 * Note: The {@link Duration} given to {@code wait} might be less than
	 * {@code duration} to handle {@code timeoutSinceStart} more precisely.
	 *
	 * @param duration          duration to wait
	 * @param timeoutSinceStart timeout duration since the {@link Stopwatch} start
	 * @param wait              method to use for waiting
	 * @return {@code true} if the timeout was not reached and {@code false} if the
	 *         timeout has been reached
	 */
	@SuppressFBWarnings(value = "PRMC_POSSIBLY_REDUNDANT_METHOD_CALLS", justification = "no redundant calls, because Instant.now() might return different values")
	public boolean waitFor(final Duration duration, final Duration timeoutSinceStart, final Consumer<Duration> wait) {
		if (duration.isNegative()) {
			throw new IllegalArgumentException(String.format("Parameter \"duration\" must not be negative, but is %s.", duration));
		}
		if (timeoutSinceStart.isNegative()) {
			throw new IllegalArgumentException(String.format("Parameter \"timeoutSinceStart\" must not be negative, but is %s.", timeoutSinceStart));
		}
		final Instant timeout = getStartInstant().plus(timeoutSinceStart);
		final Duration maxWaiting = Duration.between(Instant.now(), timeout);
		final Duration actualWaiting = maxWaiting.compareTo(duration) > 0 ? duration : maxWaiting;
		if (!actualWaiting.isNegative() && !actualWaiting.isZero()) {
			wait.accept(actualWaiting);
		}
		return !Instant.now().isAfter(timeout);
	}

	/**
	 * Sequential {@code Stream} with the checkpoints as its source
	 *
	 * @return a sequential {@code Stream} over the checkpoints
	 */
	public Stream<Checkpoint> stream() {
		return checkpoints.stream();
	}


	/**
	 * Implementation of a stopwatches checkpoint. It might be used for split or lap
	 * times.
	 */
	public static class Checkpoint implements Comparable<Checkpoint>, TemporalAccessor {
		/**
		 * Comparator of {@link Checkpoint}
		 */
		private static final Comparator<Checkpoint> COMPARATOR = Comparator.comparing(Checkpoint::getInstant).thenComparing(Comparator.comparing(checkpoint -> checkpoint.getPreviousCheckpoint().map(Checkpoint::getInstant).orElse(Instant.MIN)));
		/**
		 * Stopwatch
		 */
		private final Stopwatch stopwatch;
		/**
		 * Checkpoints name as reference
		 */
		private final String name;
		/**
		 * Instant
		 */
		private final Instant instant;
		/**
		 * Previous checkpoint or empty if this is the stopwatches first checkpoint
		 */
		private final Optional<Checkpoint> previousCheckpoint;

		/**
		 * {@inheritDoc}
		 */
		@Override
		public int compareTo(@Nullable final Checkpoint object) {
			return Objects.compare(this, object, COMPARATOR);
		}

		/**
		 * {@inheritDoc}
		 */
		@Override
		public long getLong(@Nullable final TemporalField field) {
			return getInstant().getLong(field);
		}

		/**
		 * Instant of the previous {@link Checkpoint} or the stopwatches starting time
		 * if this is the first checkpoint
		 *
		 * @return the previous checkpoints instant or the stopwatches starting time
		 */
		public Instant getPreviousInstant() {
			return getPreviousCheckpoint().map(Checkpoint::getInstant).orElseGet(getStopwatch()::getStartInstant);
		}

		/**
		 * {@inheritDoc}
		 */
		@Override
		public boolean isSupported(@Nullable final TemporalField field) {
			return getInstant().isSupported(field);
		}

		/**
		 * Duration between this checkpoint and the previous {@link Checkpoint} or the
		 * stopwatches starting time if this is the first checkpoint
		 *
		 * @return the duration between this checkpoint and previous checkpoints instant
		 *         or the stopwatches starting time
		 */
		public Duration sincePrevious() {
			return Duration.between(getPreviousInstant(), getInstant());
		}

		/**
		 * Duration between this checkpoint and the stopwatches start
		 *
		 * @return the duration between this checkpoint and the stopwatches start
		 */
		public Duration sinceStart() {
			return Duration.between(getStopwatch().getStartInstant(), getInstant());
		}

		/**
		 * Stopwatch
		 *
		 * @return the stopwatch
		 */
		@java.lang.SuppressWarnings("all")
		@edu.umd.cs.findbugs.annotations.SuppressFBWarnings(justification = "generated code")
		@lombok.Generated
		public Stopwatch getStopwatch() {
			return this.stopwatch;
		}

		/**
		 * Checkpoints name as reference
		 *
		 * @return the checkpoints name
		 */
		@java.lang.SuppressWarnings("all")
		@edu.umd.cs.findbugs.annotations.SuppressFBWarnings(justification = "generated code")
		@lombok.Generated
		public String getName() {
			return this.name;
		}

		/**
		 * Instant
		 *
		 * @return the instant
		 */
		@java.lang.SuppressWarnings("all")
		@edu.umd.cs.findbugs.annotations.SuppressFBWarnings(justification = "generated code")
		@lombok.Generated
		public Instant getInstant() {
			return this.instant;
		}

		/**
		 * Previous checkpoint or empty if this is the stopwatches first checkpoint
		 *
		 * @return the previous checkpoint
		 */
		@java.lang.SuppressWarnings("all")
		@edu.umd.cs.findbugs.annotations.SuppressFBWarnings(justification = "generated code")
		@lombok.Generated
		public Optional<Checkpoint> getPreviousCheckpoint() {
			return this.previousCheckpoint;
		}

		@edu.umd.cs.findbugs.annotations.NonNull
		@java.lang.Override
		@java.lang.SuppressWarnings("all")
		@edu.umd.cs.findbugs.annotations.SuppressFBWarnings(justification = "generated code")
		@lombok.Generated
		public java.lang.String toString() {
			return "Stopwatch.Checkpoint(instant=" + this.getInstant() + ", name=" + this.getName() + ", previousCheckpoint=" + this.getPreviousCheckpoint() + ", stopwatch=" + this.getStopwatch() + ")";
		}

		@java.lang.Override
		@java.lang.SuppressWarnings("all")
		@edu.umd.cs.findbugs.annotations.SuppressFBWarnings(justification = "generated code")
		@lombok.Generated
		public boolean equals(@edu.umd.cs.findbugs.annotations.Nullable final java.lang.Object o) {
			if (o == this) return true;
			if (!(o instanceof Stopwatch.Checkpoint)) return false;
			final Stopwatch.Checkpoint other = (Stopwatch.Checkpoint) o;
			if (!other.canEqual((java.lang.Object) this)) return false;
			final java.lang.Object this$name = this.getName();
			final java.lang.Object other$name = other.getName();
			if (this$name == null ? other$name != null : !this$name.equals(other$name)) return false;
			final java.lang.Object this$instant = this.getInstant();
			final java.lang.Object other$instant = other.getInstant();
			if (this$instant == null ? other$instant != null : !this$instant.equals(other$instant)) return false;
			final java.lang.Object this$previousCheckpoint = this.getPreviousCheckpoint();
			final java.lang.Object other$previousCheckpoint = other.getPreviousCheckpoint();
			if (this$previousCheckpoint == null ? other$previousCheckpoint != null : !this$previousCheckpoint.equals(other$previousCheckpoint)) return false;
			return true;
		}

		@java.lang.SuppressWarnings("all")
		@edu.umd.cs.findbugs.annotations.SuppressFBWarnings(justification = "generated code")
		@lombok.Generated
		protected boolean canEqual(@edu.umd.cs.findbugs.annotations.Nullable final java.lang.Object other) {
			return other instanceof Stopwatch.Checkpoint;
		}

		@java.lang.Override
		@java.lang.SuppressWarnings("all")
		@edu.umd.cs.findbugs.annotations.SuppressFBWarnings(justification = "generated code")
		@lombok.Generated
		public int hashCode() {
			final int PRIME = 59;
			int result = 1;
			final java.lang.Object $name = this.getName();
			result = result * PRIME + ($name == null ? 43 : $name.hashCode());
			final java.lang.Object $instant = this.getInstant();
			result = result * PRIME + ($instant == null ? 43 : $instant.hashCode());
			final java.lang.Object $previousCheckpoint = this.getPreviousCheckpoint();
			result = result * PRIME + ($previousCheckpoint == null ? 43 : $previousCheckpoint.hashCode());
			return result;
		}

		/**
		 * Creates a new {@code Checkpoint} instance.
		 *
		 * @param stopwatch Stopwatch
		 * @param name Checkpoints name as reference
		 * @param instant Instant
		 * @param previousCheckpoint Previous checkpoint or empty if this is the stopwatches first checkpoint
		 */
		@java.lang.SuppressWarnings("all")
		@edu.umd.cs.findbugs.annotations.SuppressFBWarnings(justification = "generated code")
		@lombok.Generated
		protected Checkpoint(final Stopwatch stopwatch, final String name, final Instant instant, final Optional<Checkpoint> previousCheckpoint) {
			this.stopwatch = stopwatch;
			this.name = name;
			this.instant = instant;
			this.previousCheckpoint = previousCheckpoint;
		}
	}

	/**
	 * Instant at the stopwatches start
	 *
	 * @return the instant at the stopwatches start time
	 */
	@java.lang.SuppressWarnings("all")
	@edu.umd.cs.findbugs.annotations.SuppressFBWarnings(justification = "generated code")
	@lombok.Generated
	public Instant getStartInstant() {
		return this.startInstant;
	}

	@edu.umd.cs.findbugs.annotations.NonNull
	@java.lang.Override
	@java.lang.SuppressWarnings("all")
	@edu.umd.cs.findbugs.annotations.SuppressFBWarnings(justification = "generated code")
	@lombok.Generated
	public java.lang.String toString() {
		return "Stopwatch(checkpoints=" + this.getCheckpoints() + ", startInstant=" + this.getStartInstant() + ")";
	}

	@java.lang.SuppressWarnings("all")
	@edu.umd.cs.findbugs.annotations.SuppressFBWarnings(justification = "generated code")
	@lombok.Generated
	public Stopwatch() {
	}
}