org.scalatest.prop.Configuration.scala Maven / Gradle / Ivy
/*
* Copyright 2001-2011 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.prop
import org.scalacheck.Test.Params
/**
* Trait providing methods and classes used to configure property checks provided by the
* the forAll
methods of trait GeneratorDrivenPropertyChecks
(for ScalaTest-style
* property checks) and the check
methods of trait Checkers
(for ScalaCheck-style property checks).
*
* @author Bill Venners
*/
trait Configuration {
/**
* Configuration object for property checks.
*
*
* The default values for the parameters are:
*
*
*
*
*
* minSuccessful
*
*
* 100
*
*
*
*
* maxDiscarded
*
*
* 500
*
*
*
*
* minSize
*
*
* 0
*
*
*
*
* maxSize
*
*
* 100
*
*
*
*
* workers
*
*
* 1
*
*
*
*
* @param minSuccessful the minimum number of successful property evaluations required for the property to pass.
* @param maxDiscarded the maximum number of discarded property evaluations allowed during a property check
* @param minSize the minimum size parameter to provide to ScalaCheck, which it will use when generating objects for which size matters (such as strings or lists).
* @param maxSize the maximum size parameter to provide to ScalaCheck, which it will use when generating objects for which size matters (such as strings or lists).
* @param workers specifies the number of worker threads to use during property evaluation
* @throws IllegalArgumentException if the specified minSuccessful
value is less than or equal to zero,
* the specified maxDiscarded
value is less than zero,
* the specified minSize
value is less than zero,
* the specified maxSize
value is less than zero,
* the specified minSize
is greater than the specified or default value of maxSize
, or
* the specified workers
value is less than or equal to zero.
*
* @author Bill Venners
*/
case class PropertyCheckConfig(
minSuccessful: Int = 100,
maxDiscarded: Int = 500,
minSize: Int = 0,
maxSize: Int = 100,
workers: Int = 1
) {
require(minSuccessful > 0, "minSuccessful had value " + minSuccessful + ", but must be greater than zero")
require(maxDiscarded >= 0, "maxDiscarded had value " + maxDiscarded + ", but must be greater than or equal to zero")
require(minSize >= 0, "minSize had value " + minSize + ", but must be greater than or equal to zero")
require(maxSize >= 0, "maxSize had value " + maxSize + ", but must be greater than or equal to zero")
require(minSize <= maxSize, "minSize had value " + minSize + ", which must be less than or equal to maxSize, which had value " + maxSize)
require(workers > 0, "workers had value " + workers + ", but must be greater than zero")
}
/**
* Abstract class defining a family of configuration parameters for property checks.
*
*
* The subclasses of this abstract class are used to pass configuration information to
* the forAll
methods of traits PropertyChecks
(for ScalaTest-style
* property checks) and Checkers
(for ScalaCheck-style property checks).
*
*
* @author Bill Venners
*/
sealed abstract class PropertyCheckConfigParam
/**
* A PropertyCheckConfigParam
that specifies the minimum number of successful
* property evaluations required for the property to pass.
*
* @throws IllegalArgumentException if specified value
is less than or equal to zero.
*
* @author Bill Venners
*/
case class MinSuccessful(value: Int) extends PropertyCheckConfigParam {
require(value > 0)
}
/**
* A PropertyCheckConfigParam
that specifies the maximum number of discarded
* property evaluations allowed during property evaluation.
*
*
* In GeneratorDrivenPropertyChecks
, a property evaluation is discarded if it throws
* DiscardedEvaluationException
, which is produce by whenever
clause that
* evaluates to false. For example, consider this ScalaTest property check:
*
*
*
* // forAll defined in GeneratorDrivenPropertyChecks
* forAll { (n: Int) =>
* whenever (n > 0) {
* doubleIt(n) should equal (n * 2)
* }
* }
*
*
*
*
* In the above code, whenever a non-positive n
is passed, the property function will complete abruptly
* with DiscardedEvaluationException
.
*
*
*
* Similarly, in Checkers
, a property evaluation is discarded if the expression to the left
* of ScalaCheck's ==>
operator is false. Here's an example:
*
*
*
* // forAll defined in Checkers
* forAll { (n: Int) =>
* (n > 0) ==> doubleIt(n) == (n * 2)
* }
*
*
*
*
* For either kind of property check, MaxDiscarded
indicates the maximum number of discarded
* evaluations that will be allowed. As soon as one past this number of evaluations indicates it needs to be discarded,
* the property check will fail.
*
*
* @throws IllegalArgumentException if specified value
is less than zero.
*
* @author Bill Venners
*/
case class MaxDiscarded(value: Int) extends PropertyCheckConfigParam {
require(value >= 0)
}
/**
* A PropertyCheckConfigParam
that specifies the minimum size parameter to
* provide to ScalaCheck, which it will use when generating objects for which size matters (such as
* strings or lists).
*
* @throws IllegalArgumentException if specified value
is less than zero.
*
* @author Bill Venners
*/
case class MinSize(value: Int) extends PropertyCheckConfigParam {
require(value >= 0)
}
/**
* A PropertyCheckConfigParam
that specifies the maximum size parameter to
* provide to ScalaCheck, which it will use when generating objects for which size matters (such as
* strings or lists).
*
*
* Note that the maximum size should be greater than or equal to the minimum size. This requirement is
* enforced by the PropertyCheckConfig
constructor and the forAll
methods of
* traits PropertyChecks
and Checkers
. In other words, it is enforced at the point
* both a maximum and minimum size are provided together.
*
*
* @throws IllegalArgumentException if specified value
is less than zero.
*
* @author Bill Venners
*/
case class MaxSize(value: Int) extends PropertyCheckConfigParam {
require(value >= 0)
}
/**
* A PropertyCheckConfigParam
that specifies the number of worker threads
* to use when evaluating a property.
*
* @throws IllegalArgumentException if specified value
is less than or equal to zero.
*
* @author Bill Venners
*/
case class Workers(value: Int) extends PropertyCheckConfigParam {
require(value > 0)
}
/**
* Returns a MinSuccessful
property check configuration parameter containing the passed value, which specifies the minimum number of successful
* property evaluations required for the property to pass.
*
* @throws IllegalArgumentException if specified value
is less than or equal to zero.
*/
def minSuccessful(value: Int): MinSuccessful = new MinSuccessful(value)
/**
* Returns a MaxDiscarded
property check configuration parameter containing the passed value, which specifies the maximum number of discarded
* property evaluations allowed during property evaluation.
*
* @throws IllegalArgumentException if specified value
is less than zero.
*/
def maxDiscarded(value: Int): MaxDiscarded = new MaxDiscarded(value)
/**
* Returns a MinSize
property check configuration parameter containing the passed value, which specifies the minimum size parameter to
* provide to ScalaCheck, which it will use when generating objects for which size matters (such as
* strings or lists).
*
* @throws IllegalArgumentException if specified value
is less than zero.
*/
def minSize(value: Int): MinSize = new MinSize(value)
/**
* Returns a MaxSize
property check configuration parameter containing the passed value, which specifies the maximum size parameter to
* provide to ScalaCheck, which it will use when generating objects for which size matters (such as
* strings or lists).
*
*
* Note that the maximum size should be greater than or equal to the minimum size. This requirement is
* enforced by the PropertyCheckConfig
constructor and the forAll
methods of
* traits PropertyChecks
and Checkers
. In other words, it is enforced at the point
* both a maximum and minimum size are provided together.
*
*
* @throws IllegalArgumentException if specified value
is less than zero.
*/
def maxSize(value: Int): MaxSize = new MaxSize(value)
/**
* Returns a Workers
property check configuration parameter containing the passed value, which specifies the number of worker threads
* to use when evaluating a property.
*
* @throws IllegalArgumentException if specified value
is less than or equal to zero.
*/
def workers(value: Int): Workers = new Workers(value)
private[prop] def getParams(
configParams: Seq[PropertyCheckConfigParam],
config: PropertyCheckConfig
): Params = {
var minSuccessful = -1
var maxDiscarded = -1
var minSize = -1
var maxSize = -1
var workers = -1
var minSuccessfulTotalFound = 0
var maxDiscardedTotalFound = 0
var minSizeTotalFound = 0
var maxSizeTotalFound = 0
var workersTotalFound = 0
for (configParam <- configParams) {
configParam match {
case param: MinSuccessful =>
minSuccessful = param.value
minSuccessfulTotalFound += 1
case param: MaxDiscarded =>
maxDiscarded = param.value
maxDiscardedTotalFound += 1
case param: MinSize =>
minSize = param.value
minSizeTotalFound += 1
case param: MaxSize =>
maxSize = param.value
maxSizeTotalFound += 1
case param: Workers =>
workers = param.value
workersTotalFound += 1
}
}
if (minSuccessfulTotalFound > 1)
throw new IllegalArgumentException("can pass at most MinSuccessful config parameters, but " + minSuccessfulTotalFound + " were passed")
if (maxDiscardedTotalFound > 1)
throw new IllegalArgumentException("can pass at most MaxDiscarded config parameters, but " + maxDiscardedTotalFound + " were passed")
if (minSizeTotalFound > 1)
throw new IllegalArgumentException("can pass at most MinSize config parameters, but " + minSizeTotalFound + " were passed")
if (maxSizeTotalFound > 1)
throw new IllegalArgumentException("can pass at most MaxSize config parameters, but " + maxSizeTotalFound + " were passed")
if (workersTotalFound > 1)
throw new IllegalArgumentException("can pass at most Workers config parameters, but " + workersTotalFound + " were passed")
// Adding one to maxDiscarded, because I think it is easier to understand that maxDiscarded means the maximum number of times
// allowed that discarding will occur and the property can still pass. One more discarded evaluation than this number and the property will fail
// because of it. ScalaCheck fails at exactly maxDiscardedTests.
Params(
if (minSuccessful != -1) minSuccessful else config.minSuccessful,
(if (maxDiscarded != -1) maxDiscarded else config.maxDiscarded) + 1,
if (minSize != -1) minSize else config.minSize,
if (maxSize != -1) maxSize else config.maxSize,
Params().rng,
if (workers != -1) workers else config.workers,
Params().testCallback
)
}
/**
* Implicit PropertyCheckConfig
value providing default configuration values.
*/
implicit val generatorDrivenConfig = PropertyCheckConfig()
}
/**
* Companion object that facilitates the importing of Configuration
members as
* an alternative to mixing it in. One use case is to import Configuration
members so you can use
* them in the Scala interpreter.
*/
object Configuration extends Configuration