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

org.scalatest.time.Span.scala Maven / Gradle / Ivy

/*
 * Copyright 2001-2012 Artima, Inc.
 *
 * 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.scalatest.time

import Span.totalNanosForLongLength
import Span.totalNanosForDoubleLength
import org.scalatest.Resources

/**
 * A time span.
 *
 * 

* A Span is used to express time spans in ScalaTest, in constructs such as the * failAfter method of trait Timeouts, * the timeLimit field of trait * TimeLimitedTests, and * the timeouts of traits Eventually, * , and * AsyncAssertions. Here's an example: *

* *
 * import org.scalatest.time.Span
 * import org.scalatest.time.Millis
 * import org.scalatest.concurrent.Timeouts._
 *
 * failAfter(Span(100, Millis)) {
 *   // ...
 * }
 * 
* *

* If you prefer you can mix in or import the members of SpanSugar and place a units value after the timeout value. * Here are some examples: *

* *
 * import org.scalatest.time.SpanSugar._
 * import org.scalatest.concurrent.Timeouts._
 *
 * failAfter(100 millis) {
 *   // ...
 * }
 *
 * failAfter(1 second) {
 *   // ...
 * }
 * 
* *

* In addition to expression the numeric value with an Int or a Long, you * can also express it via a Float or Double. Here are some examples: *

* *
 * import org.scalatest.time.Span
 * import org.scalatest.time.Seconds
 * import org.scalatest.concurrent.Timeouts._
 *
 * failAfter(Span(1.5, Seconds)) {
 *   // ...
 * }
 *
 * import org.scalatest.time.SpanSugar._
 *
 * failAfter(0.8 seconds) {
 *   // ...
 * }
 * 
* *

* Internally, a Span is expressed in terms of a Long number of nanoseconds. Thus, the maximum * time span that can be represented is Long.MaxValue nanoseconds, or approximately 292 years. * Hopefully these won't be "famous last words," but 292 years should be sufficient for software testing purposes. * Any attempt to create a Span longer than Long.MaxValue nanoseconds will be met with * an IllegalArgumentException: *

* *
 * Span(Long.MaxValue, Nanoseconds) // Produces the longest possible time.Span
 * Span(Long.MaxValue, Seconds)     // Produces an IllegalArgumentException
 * 
* *

* All of class Span's constructors are private. The only way you can create a new Span is * via one of the two apply factory methods in Span's * companion object. Here is a table showing one example of each numeric type and unit value: *

* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *
* Int * * Long * * Float * * Double *
* Span(1, Nanosecond) * * Span(1L, Nanosecond) * * Span(1.0F, Nanosecond) * * Span(1.0, Nanosecond) *
* Span(100, Nanoseconds) * * Span(100L, Nanoseconds) * * Span(99.8F, Nanoseconds) * * Span(99.8, Nanoseconds) *
* Span(1, Microsecond) * * Span(1L, Microsecond) * * Span(1.0F, Microsecond) * * Span(1.0, Microsecond) *
* Span(100, Microseconds) * * Span(100L, Microseconds) * * Span(99.8F, Microseconds) * * Span(99.8, Microseconds) *
* Span(1, Millisecond) * * Span(1L, Millisecond) * * Span(1.0F, Millisecond) * * Span(1.0, Millisecond) *
* Span(100, Milliseconds) * * Span(100L, Milliseconds) * * Span(99.8F, Milliseconds) * * Span(99.8, Milliseconds) *
* Span(100, Millis) * * Span(100L, Millis) * * Span(99.8F, Millis) * * Span(99.8, Millis) *
* Span(1, Second) * * Span(1L, Second) * * Span(1.0F, Second) * * Span(1.0, Second) *
* Span(100, Seconds) * * Span(100L, Seconds) * * Span(99.8F, Seconds) * * Span(99.8, Seconds) *
* Span(1, Minute) * * Span(1L, Minute) * * Span(1.0F, Minute) * * Span(1.0, Minute) *
* Span(100, Minutes) * * Span(100L, Minutes) * * Span(99.8F, Minutes) * * Span(99.8, Minutes) *
* Span(1, Hour) * * Span(1L, Hour) * * Span(1.0F, Hour) * * Span(1.0, Hour) *
* Span(100, Hours) * * Span(100L, Hours) * * Span(99.8F, Hours) * * Span(99.8, Hours) *
* Span(1, Day) * * Span(1L, Day) * * Span(1.0F, Day) * * Span(1.0, Day) *
* Span(100, Days) * * Span(100L, Days) * * Span(99.8F, Days) * * Span(99.8, Days) *
* * @author Bill Venners */ final class Span private (totNanos: Long, lengthString: String, unitsResource: String, unitsName: String) { private[time] def this(length: Long, units: Units) { this( totalNanosForLongLength(length, units), length.toString, if (length == 1) units.singularResourceName else units.pluralResourceName, units.toString ) } private def this(length: Double, units: Units) { this( totalNanosForDoubleLength(length, units), length.toString, if (length == 1.0) units.singularResourceName else units.pluralResourceName, units.toString ) } /** * The total number of nanoseconds in this time span. * * This number will never be negative, but can be zero. */ val totalNanos: Long = totNanos // Didn't use a parametric field becuase don't know how to scaladoc that in a // private constructor /** * This time span converted to milliseconds, computed via totalNanos / 1000000, which * truncates off any leftover nanoseconds. * *

* The millisPart and nanosPart can be used, for example, when invoking * Thread.sleep. For example, given a Span named span, you could * write: *

* *
   * Thread.sleep(span.millisPart, span.nanosPart)
   * 
*/ lazy val millisPart: Long = totalNanos / 1000000 /** * The number of nanoseconds remaining when this time span is converted to milliseconds, computed via * (totalNanos % 1000000).toInt. * *

* The millisPart and nanosPart can be used, for example, when invoking * Thread.sleep. For example, given a Span named span, you could * write: *

* *
   * Thread.sleep(span.millisPart, span.nanosPart)
   * 
*/ lazy val nanosPart: Int = (totalNanos % 1000000).toInt /** * Returns a Span representing this Span scaled by the passed factor. * *

* The passed factor can be any positive number or zero, including fractional numbers. A number * greater than one will scale the Span up to a larger value. A fractional number will scale it * down to a smaller value. A factor of 1.0 will cause the exact same Span to be returned. A * factor of zero will cause Span.ZeroLength to be returned. *

* *

* If overflow occurs, Span.Max will be returned. If underflow occurs, Span.ZeroLength * will be returned. *

* * @throws IllegalArgumentException if the passed value is less than zero */ def scaledBy(factor: Double) = { require(factor >= 0.0, "factor was less than zero") factor match { case 0.0 => Span.ZeroLength case 1.0 => this case _ => val newNanos: Double = totNanos * factor newNanos match { case n if n > Long.MaxValue => Span.Max case 1 => Span(1, Nanosecond) case n if n < 1000 => if (n.longValue == n) Span(n.longValue, Nanoseconds) else Span(n, Nanoseconds) case 1000 => Span(1, Microsecond) case n if n < 1000 * 1000 => val v = n / 1000 if (v.longValue == v) Span(v.longValue, Microseconds) else Span(v, Microseconds) case 1000000 => Span(1, Millisecond) case n if n < 1000 * 1000 * 1000 => val v = n / 1000 / 1000 if (v.longValue == v) Span(v.longValue, Millis) else Span(v, Millis) case 1000000000L => Span(1, Second) case n if n < 1000L * 1000 * 1000 * 60 => val v = n / 1000 / 1000 / 1000 if (v.longValue == v) Span(v.longValue, Seconds) else Span(v, Seconds) case 60000000000L => Span(1, Minute) case n if n < 1000L * 1000 * 1000 * 60 * 60 => val v = n / 1000 / 1000 / 1000 / 60 if (v.longValue == v) Span(v.longValue, Minutes) else Span(v, Minutes) case 3600000000000L => Span(1, Hour) case n if n < 1000L * 1000 * 1000 * 60 * 60 * 24 => val v = n / 1000 / 1000 / 1000 / 60 / 60 if (v.longValue == v) Span(v.longValue, Hours) else Span(v, Hours) case 86400000000000L => Span(1, Day) case n => val v = n / 1000 / 1000 / 1000 / 60 / 60 / 24 if (v.longValue == v) Span(v.longValue, Days) else Span(v, Days) } } } /** * Returns a localized string suitable for presenting to a user that describes the time span represented * by this Span. * *

* For example, for Span(1, Millisecond), this method would return "1 millisecond". * For Span(9.99, Seconds), this method would return "9.9 seconds". *

* * @return a localized string describing this Span */ lazy val prettyString: String = Resources(unitsResource, lengthString) /** * Returns a string that looks similar to a factory method call that would have produced this Span. * *

* For example, for Span(1, Millisecond), this method would return "Span(1, Millisecond)". * For Span(9.99, Seconds), this method would return "Span(9.99, Seconds)". *

* * @return a string that looks like a factory method call that would produce this Span */ override def toString = "Span(" + lengthString + ", " + unitsName + ")" /** * Compares another object for equality. * *

* If the passed object is a Span, this method will return true only if the other * Span returns the exact same value as this Span for totalNanos. *

* * @param other the object to compare with this one for equality * * @return true if the other object is a Span with the same totalNanos value. */ override def equals(other: Any): Boolean = { other match { case that: Span => totalNanos == that.totalNanos case _ => false } } /** * Returns a hash code for this Span. * * @return a hash code based only on the totalNanos field. */ override def hashCode: Int = totalNanos.hashCode } /** * Companion object for Span that provides two factory methods for creating Span instances. * *

* The first argument to each factory method is a numeric value; the second argument is a Units value. * One factory method takes a Long, so it can be invoked with either an Int or * Long, as in: *

* *
 * import org.scalatest.time._
 *
 * Span(1, Second)
 * Span(1L, Millisecond)
 * 
* *

* The other factory method takes a Double, so it can be invoked with either a Float or * a Double: *

*
 * import org.scalatest.time._
 *
 * Span(2.5F, Seconds)
 * Span(99.9, Microseconds)
 * 
* * @author Bill Venners */ object Span { /** * Returns a Span representing the passed Long length of time in the * passed units. * *

* If the requested time span is less than zero or greater than Long.MaxValue nanoseconds, this method will throw * an IllegalArgumentException. (Note: a zero-length time span is allowed, just not a negative or * too-large time span.) *

* * @param length the length of time denominated by the passed units * @param units the units of time for the passed length * @return a Span representing the requested time span * @throws IllegalArgumentException if the requested time span is greater than Long.MaxValue * nanoseconds, the maximum time span expressible with a Span */ def apply(length: Long, units: Units): Span = new Span(length, units) /** * Returns a Span representing the passed Double length of time in the * passed units. * *

* If the requested time span is less than 0.0 or, when converted to Long number of nanoseconds, would be greater than * Long.MaxValue nanoseconds, this method will throw an IllegalArgumentException. * (Note: a zero-length time span is allowed, just not a negative or too-large time span.) *

* * @param length the length of time denominated by the passed units * @param units the units of time for the passed length * @return a Span representing the requested time span * @throws IllegalArgumentException if the requested time span, when converted to Long number of * nanoseconds, would be greater than Long.MaxValue nanoseconds, the maximum time span * expressible with a Span */ def apply(length: Double, units: Units): Span = new Span(length, units) /** * A Span with the maximum expressible value, Span(Long.MaxValue, Nanoseconds), * which is approximately 292 years. * *

* One use case for this Span value is to help convert a duration concept from a different library to * Span when that library's duration concept includes a notion of infinite durations. An infinite * duration can be converted to Span.Max. *

* * @return a Span with the maximum expressible value, Long.MaxValue nanoseconds. */ val Max: Span = new Span(Long.MaxValue.toDouble / 1000 / 1000 / 1000 / 60 / 60 / 24, Days) /** * A Span with representing a zero-length span of time. * * @return a zero-length Span. */ val ZeroLength: Span = new Span(0, Nanoseconds) private def totalNanosForLongLength(length: Long, units: Units): Long = { require(length >= 0, "length must be greater than or equal to zero, but was: " + length) val MaxMicroseconds = Long.MaxValue / 1000 val MaxMilliseconds = Long.MaxValue / 1000 / 1000 val MaxSeconds = Long.MaxValue / 1000 / 1000 / 1000 val MaxMinutes = Long.MaxValue / 1000 / 1000 / 1000 / 60 val MaxHours = Long.MaxValue / 1000 / 1000 / 1000 / 60 / 60 val MaxDays = Long.MaxValue / 1000 / 1000 / 1000 / 60 / 60 / 24 require(units != Nanosecond || length == 1, singularErrorMsg("Nanosecond")) require(units != Microsecond || length == 1, singularErrorMsg("Microsecond")) require(units != Millisecond || length == 1, singularErrorMsg("Millisecond")) require(units != Second || length == 1, singularErrorMsg("Second")) require(units != Minute || length == 1, singularErrorMsg("Minute")) require(units != Hour || length == 1, singularErrorMsg("Hour")) require(units != Day || length == 1, singularErrorMsg("Day")) require(units != Microseconds || length <= MaxMicroseconds, "Passed length, " + length + ", is larger than the largest expressible number of microseconds: Long.MaxValue / 1000") require(units != Milliseconds && units != Millis || length <= MaxMilliseconds, "Passed length, " + length + ", is larger than the largest expressible number of millieconds: Long.MaxValue / 1000 / 1000") require(units != Seconds || length <= MaxSeconds, "Passed length, " + length + ", is larger than the largest expressible number of seconds: Long.MaxValue / 1000 / 1000 / 1000") require(units != Minutes || length <= MaxMinutes, "Passed length, " + length + ", is larger than the largest expressible number of minutes: Long.MaxValue / 1000 / 1000 / 1000 / 60") require(units != Hours || length <= MaxHours, "Passed length, " + length + ", is larger than the largest expressible number of hours: Long.MaxValue / 1000 / 1000 / 1000 / 60 / 60") require(units != Days || length <= MaxDays, "Passed length, " + length + ", is larger than the largest expressible number of days: Long.MaxValue / 1000 / 1000 / 1000 / 60 / 60 / 24") units match { case Nanosecond | Nanoseconds => length case Microsecond | Microseconds => length * 1000 case Millisecond | Milliseconds | Millis => length * 1000 * 1000 case Second | Seconds => length * 1000 * 1000 * 1000 case Minute | Minutes => length * 1000 * 1000 * 1000 * 60 case Hour | Hours => length * 1000 * 1000 * 1000 * 60 * 60 case Day | Days => length * 1000 * 1000 * 1000 * 60 * 60 * 24 } } private def totalNanosForDoubleLength(length: Double, units: Units): Long = { require(length >= 0, "length must be greater than or equal to zero, but was: " + length) val MaxNanoseconds = (Long.MaxValue).toDouble val MaxMicroseconds = Long.MaxValue.toDouble / 1000 val MaxMilliseconds = Long.MaxValue.toDouble / 1000 / 1000 val MaxSeconds = Long.MaxValue.toDouble / 1000 / 1000 / 1000 val MaxMinutes = Long.MaxValue.toDouble / 1000 / 1000 / 1000 / 60 val MaxHours = Long.MaxValue.toDouble / 1000 / 1000 / 1000 / 60 / 60 val MaxDays = Long.MaxValue.toDouble / 1000 / 1000 / 1000 / 60 / 60 / 24 require(units != Nanosecond || length == 1.0, singularErrorMsg("Nanosecond")) require(units != Microsecond || length == 1.0, singularErrorMsg("Microsecond")) require(units != Millisecond || length == 1.0, singularErrorMsg("Millisecond")) require(units != Second || length == 1.0, singularErrorMsg("Second")) require(units != Minute || length == 1.0, singularErrorMsg("Minute")) require(units != Hour || length == 1.0, singularErrorMsg("Hour")) require(units != Day || length == 1.0, singularErrorMsg("Day")) require(units != Nanoseconds || length <= MaxNanoseconds, "Passed length, " + length + ", is larger than the largest expressible number of nanoseconds: Long.MaxValue") require(units != Microseconds || length <= MaxMicroseconds, "Passed length, " + length + ", is larger than the largest expressible number of microseconds: Long.MaxValue / 1000") require(units != Milliseconds && units != Millis || length <= MaxMilliseconds, "Passed length, " + length + ", is larger than the largest expressible number of millieconds: Long.MaxValue / 1000 / 1000") require(units != Seconds || length <= MaxSeconds, "Passed length, " + length + ", is larger than the largest expressible number of seconds: Long.MaxValue / 1000 / 1000 / 1000") require(units != Minutes || length <= MaxMinutes, "Passed length, " + length + ", is larger than the largest expressible number of minutes: Long.MaxValue / 1000 / 1000 / 1000 / 60") require(units != Hours || length <= MaxHours, "Passed length, " + length + ", is larger than the largest expressible number of hours: Long.MaxValue / 1000 / 1000 / 1000 / 60 / 60") require(units != Days || length <= MaxDays, "Passed length, " + length + ", is larger than the largest expressible number of days: Long.MaxValue / 1000 / 1000 / 1000 / 60 / 60 / 24") units match { case Nanosecond | Nanoseconds => length.toLong case Microsecond | Microseconds => (length * 1000).toLong case Millisecond | Milliseconds | Millis => (length * 1000 * 1000).toLong case Second | Seconds => (length * 1000 * 1000 * 1000).toLong case Minute | Minutes => (length * 1000 * 1000 * 1000 * 60).toLong case Hour | Hours => (length * 1000 * 1000 * 1000 * 60 * 60).toLong case Day | Days => (length * 1000 * 1000 * 1000 * 60 * 60 * 24).toLong } } private def singularErrorMsg(unitsString: String) = { "Singular form of " + unitsString + " (i.e., without the trailing s) can only be used with the value 1. Use " + unitsString + "s (i.e., with an s) instead." } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy