CINXE.COM
Scalactic
<!DOCTYPE html> <html> <head> <meta http-equiv="content-type" content="text/html;charset=utf-8" /> <title>Scalactic</title> <link rel="Stylesheet" href="/assets/stylesheets/main.css" type="text/css" media="screen"/> <!-- Global site tag (gtag.js) - Google Analytics --> <script async src="https://www.googletagmanager.com/gtag/js?id=G-19BZD1XMRV"></script> <script defer src='/assets/javascripts/gtag.js'></script> </head> <body> <div class="Container"> <!-- Top of scalactic.org 660 x 60 [async] --> <script id="adsArtimaScript" type="text/javascript" src="https://www.artima.com/assets/javascripts/4ca150665e51d0b1c3890ca1b891c507-ads.js?product=Scalactic"></script> <div id="Header"> <div id="Nav"> <span><em>Home</em></span> | <a href="/quick_start">Quick Start</a> | <a href="/install">Install</a> | <a href="/user_guide">User Guide</a> | <a href="/scaladoc">Scaladoc</a> | <a href="/supersafe">SuperSafe</a> | <a href="/about">About</a> </div> <!-- Nav --> </div> <!-- Header --> <div id="body"> <div class="homePageMessage"> <img src="/assets/images/scalacticHeading.gif" width="700" height="200" alt="Scalactic: Simply Productive"/> </div> <p style="align: center"> <span style="font-size: 150%">Latest Release - <a href="http://www.scalatest.org/release_notes/3.2.19">Scalactic 3.2.19</a>!</span> </p> <div style="text-align: left"> <p> The Scalactic library is focused on constructs related to quality that are useful in both production code and tests. Although <a href="http://scalatest.org">ScalaTest</a> is tightly integrated with Scalactic, you can use Scalactic with any Scala project and any test framework. The Scalactic library has no dependencies other than Scala itself. </p> <h1>The <code>===</code> and <code>!==</code> operators</h1> <p> Scalactic provides a powerful <code>===</code> operator (and its complement, <code>!==</code>) that allows you to </p> <ul> <li>Customize equality for a type outside its class</li> <li>Get compiler errors for suspicious equality comparisons</li> <li>Compare numeric values for equality with a tolerance</li> <li>Normalize values before comparing them for equality</li> </ul> <p> For example, you can compare strings for equality after being normalized by forcing them to lowercase by customizing <a href='/scaladoc/3.2.19/org/scalactic/Equality.html'><code>Equality</code></a> explicitly, like this: </p> <pre class="scala"> import org.scalactic._ import TripleEquals._ import StringNormalizations._ import Explicitly._ (<span class="quotedString">"Hello"</span> === <span class="quotedString">"hello"</span>) (after being lowerCased) <span class="lineComment">// true</span> </pre> <p> You can also define implicit <a href='/scaladoc/3.2.19/org/scalactic/Normalization.html'><code>Normalization</code></a> strategies for types and access them by invoking <code>norm</code> methods: </p> <pre class="scala"> import NormMethods._ <span class="reserved">implicit</span> <span class="reserved">val</span> strNormalization = lowerCased <span class="quotedString">"WHISPER"</span>.norm <span class="lineComment">// "whisper"</span> <span class="quotedString">"WHISPER"</span>.norm === <span class="quotedString">"whisper"</span> <span class="lineComment">// true</span> </pre> <p> The “<code>after being lowerCased</code>” syntax shown previously is provided by Scalactic' <a href='/scaladoc/3.2.19/org/scalactic/Explicitly.html'><code>Explicitly</code></a> DSL, which allows you to specify <code>Equality</code> explicitly. You can also define custom <code>Equality</code>s implicitly: </p> <pre class="scala"> <span class="reserved">implicit</span> <span class="reserved">val</span> strEquality = decided by defaultEquality[<span class="type">String</span>] afterBeing lowerCased <span class="quotedString">"Hello"</span> === <span class="quotedString">"hello"</span> <span class="lineComment">// true</span> <span class="quotedString">"normalized"</span> === <span class="quotedString">"NORMALIZED"</span> <span class="lineComment">// true</span> </pre> <p> You can compare numeric values for equality with a <a href='/scaladoc/3.2.19/org/scalactic/Tolerance.html'><code>Tolerance</code></a>, like this: </p> <pre class="scala"> import Tolerance._ <span class="literal">2.00001</span> === <span class="literal">2.0</span> +- <span class="literal">0.01</span> <span class="lineComment">// true</span> </pre> <p> Or you could use <a href='/scaladoc/3.2.19/org/scalactic/TolerantNumerics.html'><code>TolerantNumerics</code></a> define an implicit <code>Equality[Double]</code> that compares <code>Double</code>s with a tolerance: </p> <pre class="scala"> import TolerantNumerics._ <span class="reserved">implicit</span> <span class="reserved">val</span> dblEquality = tolerantDoubleEquality(<span class="literal">0.01</span>) <span class="literal">2.00001</span> === <span class="literal">2.0</span> <span class="lineComment">// true</span> </pre> <p> A compiler error for an equality comparison that would always yield false looks like: </p> <pre class="scala"> import TypeCheckedTripleEquals._ <span class="type">Some</span>(<span class="quotedString">"hi"</span>) === <span class="quotedString">"hi"</span> error: types Some[String] and String do not adhere to the type constraint selected for the === and !== operators; the missing implicit parameter is of type org.scalactic.Constraint[Some[String],String] Some("hi") === "hi" ^ </pre> <h1><code>Or</code> and <code>Every</code></h1> <p> Scalactic provides an “<code>Either</code> with attitude” named <a href='/scaladoc/3.2.19/org/scalactic/Or.html'><code>Or</code></a>, designed for functional error handling. <code>Or</code> gives you more convenient chaining of <code>map</code> and <code>flatMap</code> calls (and <code>for</code> expressions) than <code>Either</code> and, when, combined with <a href='/scaladoc/3.2.19/org/scalactic/Every.html'><code>Every</code></a>, enables you to accumulate errors. <code>Every</code> is an ordered collection of one or more elements. An <code>Or</code> is either <a href='/scaladoc/3.2.19/org/scalactic/Good.html'><code>Good</code></a> or <a href='/scaladoc/3.2.19/org/scalactic/Bad.html'><code>Bad</code></a>. An <code>Every</code> is either <a href='/scaladoc/3.2.19/org/scalactic/One.html'<code>One</code></a> or <a href='/scaladoc/3.2.19/org/scalactic/Many.html'><code>Many</code></a>. Here's an example of accumulating errors with <code>Or</code> and <code>Every</code>: </p> <pre class="scala"> import org.scalactic._ import Accumulation._ <span class="reserved">case</span> <span class="reserved">class</span> <span class="type">Person</span>(name: <span class="type">String</span>, age: <span class="type">Int</span>) <span class="reserved">def</span> parseName(input: <span class="type">String</span>): <span class="type">String</span> <span class="type">Or</span> <span class="type">One[ErrorMessage]</span> = { <span class="reserved">val</span> trimmed = input.trim <span class="reserved">if</span> (!trimmed.isEmpty) <span class="type">Good</span>(trimmed) <span class="reserved">else</span> <span class="type">Bad</span>(<span class="type">One</span>(s<span class="quotedString">""""${input}" is not a valid name"""</span>)) } <span class="reserved">def</span> parseAge(input: <span class="type">String</span>): <span class="type">Int</span> <span class="type">Or</span> <span class="type">One[ErrorMessage]</span> = { <span class="reserved">try</span> { <span class="reserved">val</span> age = input.trim.toInt <span class="reserved">if</span> (age >= <span class="literal">0</span>) <span class="type">Good</span>(age) <span class="reserved">else</span> <span class="type">Bad</span>(<span class="type">One</span>(s<span class="quotedString">""""${age}" is not a valid age"""</span>)) } <span class="reserved">catch</span> { <span class="reserved">case</span> _: <span class="type">NumberFormatException</span> => <span class="type">Bad</span>(<span class="type">One</span>(s<span class="quotedString">""""${input}" is not a valid integer"""</span>)) } } <span class="reserved">def</span> parsePerson(inputName: <span class="type">String</span>, inputAge: <span class="type">String</span>): <span class="type">Person</span> <span class="type">Or</span> <span class="type">Every[ErrorMessage]</span> = { <span class="reserved">val</span> name = parseName(inputName) <span class="reserved">val</span> age = parseAge(inputAge) withGood(name, age) { <span class="type">Person</span>(_, _) } } </pre> <p> Here are some examples of <code>parsePerson</code> in action: </p> <pre class="scala"> parsePerson(<span class="quotedString">"Bridget Jones"</span>, <span class="quotedString">"29"</span>) <span class="lineComment">// Result: Good(Person(Bridget Jones,29))</span> <br />parsePerson(<span class="quotedString">"Bridget Jones"</span>, <span class="quotedString">""</span>) <span class="lineComment">// Result: Bad(One("" is not a valid integer))</span> <br />parsePerson(<span class="quotedString">"Bridget Jones"</span>, <span class="quotedString">"-29"</span>) <span class="lineComment">// Result: Bad(One("-29" is not a valid age))</span> <br />parsePerson(<span class="quotedString">""</span>, <span class="quotedString">""</span>) <span class="lineComment">// Result: Bad(Many("" is not a valid name, "" is not a valid integer))</span> </pre> <p> <code>Or</code> offers several other ways to accumulate errors besides the <code>withGood</code> methods shown in the example above. See the <a href='/scaladoc/3.2.19/org/scalactic/Or.html'>documentation for <code>Or</code></a> for more information. </p> <h1><code>Requirements</code> and <code>Snapshots</code></h1> <p> Scalactic includes a <a href='/scaladoc/3.2.19/org/scalactic/Requirements.html'><code>Requirements</code></a> trait that offers <code>require</code>, <code>requireState</code>, and <code>requireNonNull</code> methods for checking pre-conditions that give descriptive error messages extracted via a macro. Here are some examples:</p> </p> <pre class="scala"> <span class="reserved">val</span> a = <span class="literal">-1</span> require(a >= <span class="literal">0</span>) <span class="lineComment">// throws IllegalArgumentException: -1 was not greater than or equal to 0</span> requireState(a >= <span class="literal">0</span>) <span class="lineComment">// throws IllegalStateException: -1 was not greater than or equal to 0</span> <span class="reserved">val</span> b: <span class="type">String</span> = <span class="reserved">null</span> requireNonNull(a, b) <span class="lineComment">// throws NullPointerException: b was null</span> </pre> <p> Trait <a href='/scaladoc/3.2.19/org/scalactic/Snapshots.html'><code>Snapshots</code></a> offers a <code>snap</code> method that can help you make debug and log messages that include information about the values of variables: </p> <pre class="scala"> <span class="reserved">val</span> a = <span class="literal">1</span> <span class="reserved">val</span> b = <span class="literal">'2'</span> <span class="reserved">val</span> c = <span class="literal">"3"</span> snap(a, b, c) <span class="lineComment">// Result: a was 1, b was '2', c was "3"</span> </pre> <h1>And more (but not much more)...</h1> <p> Scalactic also includes a <a href='/scaladoc/3.2.19/org/scalactic/TimesOnInt.html'><code>TimesOnInt</code></a> trait that allows you to perform side-effecting loops a specified number of times, like this: </p> <pre class="scala"> import TimesOnInt._ <span class="literal">3</span> times println(<span class="quotedString">"hello "</span>) <span class="lineComment">// Output: hello hello hello</span> </pre> <p> You can also define an alternate <code>String</code> forms for types using <a href='/scaladoc/3.2.19/org/scalactic/Prettifier.html'><code>Prettifier</code></a>s and create extractors for <code>Throwable</code>s via the <a href='/scaladoc/3.2.19/org/scalactic/Catcher.html'><code>Catcher</code></a> factory. </p> <p> And that's it: Scalactic is a small, very focused library. Why not give it a try? Just visit the <a href="/quick_start">Quick Start</a> page.</p> </div> <!-- body --> <div style="font-size: 66%; margin-top: 60px"> <p> Scalactic is brought to you by Bill Venners, with contributions from several other folks. It is sponsored by Artima, Inc.<br /> ScalaTest is free, open-source software released under the <a href="http://www.apache.org/licenses/LICENSE-2.0.html">Apache 2.0 license</a>. </p> <p> Copyright © 2009-2025 Artima, Inc. All Rights Reserved. </p> <p> <a href="http://www.artima.com" class="no_link_hover"> <img src="/assets/images/artima100Black.png" style="margin-top: 2px" width="100" height="38" alt="artima"/> </a> </p> </div> <!-- style="..." --> </div> <!-- Container --> </body> </html>