From b9b61cfe4a4b238de7175287b7481c35a798a582 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=C8=98tefan=20Talpalaru?= Date: Thu, 29 Apr 2021 14:20:03 +0200 Subject: [PATCH] change define prefix and update docs (#7) --- .github/workflows/ci.yml | 6 +- README.md | 2 +- config.nims | 20 +- nimdoc.out.css | 891 ++++++++++++++++++ unittest2.html | 1842 ++++++++++---------------------------- unittest2.nim | 34 +- 6 files changed, 1406 insertions(+), 1389 deletions(-) create mode 100644 nimdoc.out.css diff --git a/.github/workflows/ci.yml b/.github/workflows/ci.yml index 77c8c29..bc856f4 100644 --- a/.github/workflows/ci.yml +++ b/.github/workflows/ci.yml @@ -1,4 +1,4 @@ -name: nim-unittest2 CI +name: CI on: [push, pull_request] jobs: @@ -136,13 +136,13 @@ jobs: - name: Build the Nim compiler shell: bash run: | - git clone -b ${{ matrix.branch }} --depth 1 git://github.com/nim-lang/nim nim/ curl -O -L -s -S https://raw.githubusercontent.com/status-im/nimbus-build-system/master/scripts/build_nim.sh - env MAKE="${MAKE_CMD} -j${ncpu}" ARCH_OVERRIDE=${PLATFORM} CC=gcc QUICK_AND_DIRTY_COMPILER=1 bash build_nim.sh nim csources dist/nimble NimBinaries + env MAKE="${MAKE_CMD} -j${ncpu}" ARCH_OVERRIDE=${PLATFORM} CC=gcc QUICK_AND_DIRTY_COMPILER=1 NIM_COMMIT=${{ matrix.branch }} bash build_nim.sh nim csources dist/nimble NimBinaries echo '${{ github.workspace }}/nim/bin' >> $GITHUB_PATH - name: Run tests shell: bash run: | + nim --version nim --hints:off --verbosity:0 test diff --git a/README.md b/README.md index 7a8c147..79d8196 100644 --- a/README.md +++ b/README.md @@ -24,7 +24,7 @@ requires "unittest2" ## Usage -See [unittest2.html](https://status-im.github.io/nim-unittest2/unittest2.html) documentation generated by `nim doc unittest2.nim`. +See [unittest2.html](https://status-im.github.io/nim-unittest2/unittest2.html) documentation generated by `nim buildDocs`. Create a file that contains your unit tests: diff --git a/config.nims b/config.nims index 4238552..f4bcc5a 100644 --- a/config.nims +++ b/config.nims @@ -1,8 +1,24 @@ task test, "Run tests": + let + xmlFile = "test_results.xml" + commandStart = "nim c -r -f --threads:on --hints:off --verbosity:0 --skipParentCfg:on --skipUserCfg:on " + for f in listFiles("tests"): if f.len > 4 and f[^4..^1] == ".nim": - let cmd = "nim c -r -f --threads:on --hints:off --verbosity:0 --skipParentCfg:on --skipUserCfg:on " & f + # This should generate an XML results file. + var cmd = commandStart & f & " --xml:" & xmlFile & " --console" echo cmd exec cmd - rmFile(f[0..^5].toExe()) + doAssert fileExists xmlFile + rmFile xmlFile + + # This should not, since we disable param processing. + cmd = commandStart & "-d:unittest2DisableParamFiltering " & f & " --xml:" & xmlFile + echo cmd + exec cmd + doAssert not fileExists xmlFile + rmFile f[0..^5].toExe + +task buildDocs, "Build docs": + exec "nim doc --skipParentCfg:on --skipUserCfg:on --git.url:https://github.com/status-im/nim-unittest2 --git.commit:master --git.devel:master unittest2.nim" diff --git a/nimdoc.out.css b/nimdoc.out.css new file mode 100644 index 0000000..4ee73ea --- /dev/null +++ b/nimdoc.out.css @@ -0,0 +1,891 @@ +/* +Stylesheet for use with Docutils/rst2html. + +See http://docutils.sf.net/docs/howto/html-stylesheets.html for how to +customize this style sheet. + +Modified from Chad Skeeters' rst2html-style +https://bitbucket.org/cskeeters/rst2html-style/ + +Modified by Boyd Greenfield and narimiran +*/ + +:root { + --primary-background: #fff; + --secondary-background: ghostwhite; + --third-background: #e8e8e8; + --border: #dde; + --text: #222; + --anchor: #07b; + --anchor-focus: #607c9f; + --input-focus: #1fa0eb; + --strong: #3c3c3c; + --hint: #9A9A9A; + --nim-sprite-base64: url(""); + + --keyword: #5e8f60; + --identifier: #222; + --comment: #484a86; + --operator: #155da4; + --punctuation: black; + --other: black; + --escapeSequence: #c4891b; + --number: #252dbe; + --literal: #a4255b; + --raw-data: #a4255b; +} + +[data-theme="dark"] { + --primary-background: #171921; + --secondary-background: #1e202a; + --third-background: #2b2e3b; + --border: #0e1014; + --text: #fff; + --anchor: #8be9fd; + --anchor-focus: #8be9fd; + --input-focus: #8be9fd; + --strong: #bd93f9; + --hint: #7A7C85; + --nim-sprite-base64: url(""); + + --keyword: #ff79c6; + --identifier: #f8f8f2; + --comment: #6272a4; + --operator: #ff79c6; + --punctuation: #f8f8f2; + --other: #f8f8f2; + --escapeSequence: #bd93f9; + --number: #bd93f9; + --literal: #f1fa8c; + --raw-data: #8be9fd; +} + +.theme-switch-wrapper { + display: flex; + align-items: center; + + em { + margin-left: 10px; + font-size: 1rem; + } +} +.theme-switch { + display: inline-block; + height: 22px; + position: relative; + width: 50px; +} + +.theme-switch input { + display: none; +} + +.slider { + background-color: #ccc; + bottom: 0; + cursor: pointer; + left: 0; + position: absolute; + right: 0; + top: 0; + transition: .4s; +} + +.slider:before { + background-color: #fff; + bottom: 4px; + content: ""; + height: 13px; + left: 4px; + position: absolute; + transition: .4s; + width: 13px; +} + +input:checked + .slider { + background-color: #66bb6a; +} + +input:checked + .slider:before { + transform: translateX(26px); +} + +.slider.round { + border-radius: 17px; +} + +.slider.round:before { + border-radius: 50%; +} + +html { + font-size: 100%; + -webkit-text-size-adjust: 100%; + -ms-text-size-adjust: 100%; } + +body { + font-family: "Lato", "Helvetica Neue", "HelveticaNeue", Helvetica, Arial, sans-serif; + font-weight: 400; + font-size: 1.125em; + line-height: 1.5; + color: var(--text); + background-color: var(--primary-background); } + +/* Skeleton grid */ +.container { + position: relative; + width: 100%; + max-width: 1050px; + margin: 0 auto; + padding: 0; + box-sizing: border-box; } + +.column, +.columns { + width: 100%; + float: left; + box-sizing: border-box; + margin-left: 1%; +} + +.column:first-child, +.columns:first-child { + margin-left: 0; } + +.three.columns { + width: 19%; } + +.nine.columns { + width: 80.0%; } + +.twelve.columns { + width: 100%; + margin-left: 0; } + +@media screen and (max-width: 860px) { + .three.columns { + display: none; + } + .nine.columns { + width: 98.0%; + } + body { + font-size: 1em; + line-height: 1.35; + } +} + +cite { + font-style: italic !important; } + + +/* Nim search input */ +div#searchInputDiv { + margin-bottom: 1em; +} +input#searchInput { + width: 80%; +} + +/* + * Some custom formatting for input forms. + * This also fixes input form colors on Firefox with a dark system theme on Linux. + */ +input { + -moz-appearance: none; + background-color: var(--secondary-background); + color: var(--text); + border: 1px solid var(--border); + font-family: "Lato", "Helvetica Neue", "HelveticaNeue", Helvetica, Arial, sans-serif; + font-size: 0.9em; + padding: 6px; +} + +input:focus { + border: 1px solid var(--input-focus); + box-shadow: 0 0 3px var(--input-focus); +} + +select { + -moz-appearance: none; + background-color: var(--secondary-background); + color: var(--text); + border: 1px solid var(--border); + font-family: "Lato", "Helvetica Neue", "HelveticaNeue", Helvetica, Arial, sans-serif; + font-size: 0.9em; + padding: 6px; +} + +select:focus { + border: 1px solid var(--input-focus); + box-shadow: 0 0 3px var(--input-focus); +} + +/* Docgen styles */ +/* Links */ +a { + color: var(--anchor); + text-decoration: none; +} + +a span.Identifier { + text-decoration: underline; + text-decoration-color: #aab; +} + +a.reference-toplevel { + font-weight: bold; +} + +a.toc-backref { + text-decoration: none; + color: var(--text); } + +a.link-seesrc { + color: #607c9f; + font-size: 0.9em; + font-style: italic; } + +a:hover, +a:focus { + color: var(--anchor-focus); + text-decoration: underline; } + +a:hover span.Identifier { + color: var(--anchor); +} + + +sub, +sup { + position: relative; + font-size: 75%; + line-height: 0; + vertical-align: baseline; } + +sup { + top: -0.5em; } + +sub { + bottom: -0.25em; } + +img { + width: auto; + height: auto; + max-width: 100%; + vertical-align: middle; + border: 0; + -ms-interpolation-mode: bicubic; } + +@media print { + * { + color: black !important; + text-shadow: none !important; + background: transparent !important; + box-shadow: none !important; } + + a, + a:visited { + text-decoration: underline; } + + a[href]:after { + content: " (" attr(href) ")"; } + + abbr[title]:after { + content: " (" attr(title) ")"; } + + .ir a:after, + a[href^="javascript:"]:after, + a[href^="#"]:after { + content: ""; } + + pre, + blockquote { + border: 1px solid #999; + page-break-inside: avoid; } + + thead { + display: table-header-group; } + + tr, + img { + page-break-inside: avoid; } + + img { + max-width: 100% !important; } + + @page { + margin: 0.5cm; } + + h1 { + page-break-before: always; } + + h1.title { + page-break-before: avoid; } + + p, + h2, + h3 { + orphans: 3; + widows: 3; } + + h2, + h3 { + page-break-after: avoid; } +} + + +p { + margin-top: 0.5em; + margin-bottom: 0.5em; +} + +small { + font-size: 85%; } + +strong { + font-weight: 600; + font-size: 0.95em; + color: var(--strong); +} + +em { + font-style: italic; } + +h1 { + font-size: 1.8em; + font-weight: 400; + padding-bottom: .25em; + border-bottom: 6px solid var(--third-background); + margin-top: 2.5em; + margin-bottom: 1em; + line-height: 1.2em; } + +h1.title { + padding-bottom: 1em; + border-bottom: 0px; + font-size: 2.5em; + text-align: center; + font-weight: 900; + margin-top: 0.75em; + margin-bottom: 0em; +} + +h2 { + font-size: 1.3em; + margin-top: 2em; } + +h2.subtitle { + text-align: center; } + +h3 { + font-size: 1.125em; + font-style: italic; + margin-top: 1.5em; } + +h4 { + font-size: 1.125em; + margin-top: 1em; } + +h5 { + font-size: 1.125em; + margin-top: 0.75em; } + +h6 { + font-size: 1.1em; } + + +ul, +ol { + padding: 0; + margin-top: 0.5em; + margin-left: 0.75em; } + +ul ul, +ul ol, +ol ol, +ol ul { + margin-bottom: 0; + margin-left: 1.25em; } + +li { + list-style-type: circle; +} + +ul.simple-boot li { + list-style-type: none; + margin-left: 0em; + margin-bottom: 0.5em; +} + +ol.simple > li, ul.simple > li { + margin-bottom: 0.25em; + margin-left: 0.4em } + +ul.simple.simple-toc > li { + margin-top: 1em; +} + +ul.simple-toc { + list-style: none; + font-size: 0.9em; + margin-left: -0.3em; + margin-top: 1em; } + +ul.simple-toc > li { + list-style-type: none; +} + +ul.simple-toc-section { + list-style-type: circle; + margin-left: 1em; + color: #6c9aae; } + + +ol.arabic { + list-style: decimal; } + +ol.loweralpha { + list-style: lower-alpha; } + +ol.upperalpha { + list-style: upper-alpha; } + +ol.lowerroman { + list-style: lower-roman; } + +ol.upperroman { + list-style: upper-roman; } + +ul.auto-toc { + list-style-type: none; } + + +dl { + margin-bottom: 1.5em; } + +dt { + margin-bottom: -0.5em; + margin-left: 0.0em; } + +dd { + margin-left: 2.0em; + margin-bottom: 3.0em; + margin-top: 0.5em; } + + +hr { + margin: 2em 0; + border: 0; + border-top: 1px solid #aaa; } + +blockquote { + font-size: 0.9em; + font-style: italic; + padding-left: 0.5em; + margin-left: 0; + border-left: 5px solid #bbc; +} + +.pre { + font-family: "Source Code Pro", Monaco, Menlo, Consolas, "Courier New", monospace; + font-weight: 500; + font-size: 0.85em; + color: var(--text); + background-color: var(--third-background); + padding-left: 3px; + padding-right: 3px; + border-radius: 4px; +} + +pre { + font-family: "Source Code Pro", Monaco, Menlo, Consolas, "Courier New", monospace; + color: var(--text); + font-weight: 500; + display: inline-block; + box-sizing: border-box; + min-width: 100%; + padding: 0.5em; + margin-top: 0.5em; + margin-bottom: 0.5em; + font-size: 0.85em; + white-space: pre !important; + overflow-y: hidden; + overflow-x: visible; + background-color: var(--secondary-background); + border: 1px solid var(--border); + -webkit-border-radius: 6px; + -moz-border-radius: 6px; + border-radius: 6px; } + +.pre-scrollable { + max-height: 340px; + overflow-y: scroll; } + + +/* Nim line-numbered tables */ +.line-nums-table { + width: 100%; + table-layout: fixed; } + +table.line-nums-table { + border-radius: 4px; + border: 1px solid #cccccc; + background-color: ghostwhite; + border-collapse: separate; + margin-top: 15px; + margin-bottom: 25px; } + +.line-nums-table tbody { + border: none; } + +.line-nums-table td pre { + border: none; + background-color: transparent; } + +.line-nums-table td.blob-line-nums { + width: 28px; } + +.line-nums-table td.blob-line-nums pre { + color: #b0b0b0; + -webkit-filter: opacity(75%); + text-align: right; + border-color: transparent; + background-color: transparent; + padding-left: 0px; + margin-left: 0px; + padding-right: 0px; + margin-right: 0px; } + + +table { + max-width: 100%; + background-color: transparent; + margin-top: 0.5em; + margin-bottom: 1.5em; + border-collapse: collapse; + border-color: var(--third-background); + border-spacing: 0; + font-size: 0.9em; +} + +table th, table td { + padding: 0px 0.5em 0px; + border-color: var(--third-background); +} + +table th { + background-color: var(--third-background); + border-color: var(--third-background); + font-weight: bold; } + +table th.docinfo-name { + background-color: transparent; +} + +table tr:hover { + background-color: var(--third-background); } + + +/* rst2html default used to remove borders from tables and images */ +.borderless, table.borderless td, table.borderless th { + border: 0; } + +table.borderless td, table.borderless th { + /* Override padding for "table.docutils td" with "! important". + The right padding separates the table cells. */ + padding: 0 0.5em 0 0 !important; } + +.first { + /* Override more specific margin styles with "! important". */ + margin-top: 0 !important; } + +.last, .with-subtitle { + margin-bottom: 0 !important; } + +.hidden { + display: none; } + +blockquote.epigraph { + margin: 2em 5em; } + +dl.docutils dd { + margin-bottom: 0.5em; } + +object[type="image/svg+xml"], object[type="application/x-shockwave-flash"] { + overflow: hidden; } + + +div.figure { + margin-left: 2em; + margin-right: 2em; } + +div.footer, div.header { + clear: both; + text-align: center; + color: #666; + font-size: smaller; } + +div.footer { + padding-top: 5em; +} + +div.line-block { + display: block; + margin-top: 1em; + margin-bottom: 1em; } + +div.line-block div.line-block { + margin-top: 0; + margin-bottom: 0; + margin-left: 1.5em; } + +div.topic { + margin: 2em; } + +div.search_results { + background-color: antiquewhite; + margin: 3em; + padding: 1em; + border: 1px solid #4d4d4d; +} + +div#global-links ul { + margin-left: 0; + list-style-type: none; +} + +div#global-links > simple-boot { + margin-left: 3em; +} + +hr.docutils { + width: 75%; } + +img.align-left, .figure.align-left, object.align-left { + clear: left; + float: left; + margin-right: 1em; } + +img.align-right, .figure.align-right, object.align-right { + clear: right; + float: right; + margin-left: 1em; } + +img.align-center, .figure.align-center, object.align-center { + display: block; + margin-left: auto; + margin-right: auto; } + +.align-left { + text-align: left; } + +.align-center { + clear: both; + text-align: center; } + +.align-right { + text-align: right; } + +/* reset inner alignment in figures */ +div.align-right { + text-align: inherit; } + +p.attribution { + text-align: right; + margin-left: 50%; } + +p.caption { + font-style: italic; } + +p.credits { + font-style: italic; + font-size: smaller; } + +p.label { + white-space: nowrap; } + +p.rubric { + font-weight: bold; + font-size: larger; + color: maroon; + text-align: center; } + +p.topic-title { + font-weight: bold; } + +pre.address { + margin-bottom: 0; + margin-top: 0; + font: inherit; } + +pre.literal-block, pre.doctest-block, pre.math, pre.code { + margin-left: 2em; + margin-right: 2em; } + +pre.code .ln { + color: grey; } + +/* line numbers */ +pre.code, code { + background-color: #eeeeee; } + +pre.code .comment, code .comment { + color: #5c6576; } + +pre.code .keyword, code .keyword { + color: #3B0D06; + font-weight: bold; } + +pre.code .literal.string, code .literal.string { + color: #0c5404; } + +pre.code .name.builtin, code .name.builtin { + color: #352b84; } + +pre.code .deleted, code .deleted { + background-color: #DEB0A1; } + +pre.code .inserted, code .inserted { + background-color: #A3D289; } + +span.classifier { + font-style: oblique; } + +span.classifier-delimiter { + font-weight: bold; } + +span.option { + white-space: nowrap; } + +span.problematic { + color: #b30000; } + +span.section-subtitle { + /* font-size relative to parent (h1..h6 element) */ + font-size: 80%; } + +span.DecNumber { + color: var(--number); } + +span.BinNumber { + color: var(--number); } + +span.HexNumber { + color: var(--number); } + +span.OctNumber { + color: var(--number); } + +span.FloatNumber { + color: var(--number); } + +span.Identifier { + color: var(--identifier); } + +span.Keyword { + font-weight: 600; + color: var(--keyword); } + +span.StringLit { + color: var(--literal); } + +span.LongStringLit { + color: var(--literal); } + +span.CharLit { + color: var(--literal); } + +span.EscapeSequence { + color: var(--escapeSequence); } + +span.Operator { + color: var(--operator); } + +span.Punctuation { + color: var(--punctuation); } + +span.Comment, span.LongComment { + font-style: italic; + font-weight: 400; + color: var(--comment); } + +span.RegularExpression { + color: darkviolet; } + +span.TagStart { + color: darkviolet; } + +span.TagEnd { + color: darkviolet; } + +span.Key { + color: #252dbe; } + +span.Value { + color: #252dbe; } + +span.RawData { + color: var(--raw-data); } + +span.Assembler { + color: #252dbe; } + +span.Preprocessor { + color: #252dbe; } + +span.Directive { + color: #252dbe; } + +span.Command, span.Rule, span.Hyperlink, span.Label, span.Reference, +span.Other { + color: var(--other); } + +/* Pop type, const, proc, and iterator defs in nim def blocks */ +dt pre > span.Identifier, dt pre > span.Operator { + color: var(--identifier); + font-weight: 700; } + +dt pre > span.Keyword ~ span.Identifier, dt pre > span.Identifier ~ span.Identifier, +dt pre > span.Operator ~ span.Identifier, dt pre > span.Other ~ span.Identifier { + color: var(--identifier); + font-weight: inherit; } + +/* Nim sprite for the footer (taken from main page favicon) */ +.nim-sprite { + display: inline-block; + width: 51px; + height: 14px; + background-position: 0 0; + background-size: 51px 14px; + -webkit-filter: opacity(50%); + background-repeat: no-repeat; + background-image: var(--nim-sprite-base64); + margin-bottom: 5px; } + +span.pragmadots { + /* Position: relative frees us up to make the dots + look really nice without fucking up the layout and + causing bulging in the parent container */ + position: relative; + /* 1px down looks slightly nicer */ + top: 1px; + padding: 2px; + background-color: var(--third-background); + border-radius: 4px; + margin: 0 2px; + cursor: pointer; + font-size: 0.8em; +} + +span.pragmadots:hover { + background-color: var(--hint); +} +span.pragmawrap { + display: none; +} + +span.attachedType { + display: none; + visibility: hidden; +} \ No newline at end of file diff --git a/unittest2.html b/unittest2.html index 3b19fdb..425002a 100644 --- a/unittest2.html +++ b/unittest2.html @@ -10,6 +10,7 @@ + @@ -17,1187 +18,7 @@ unittest2 - + @@ -1212,6 +33,37 @@ function main() { event.target.parentNode.nextElementSibling.style.display = "inline"; } } + + const toggleSwitch = document.querySelector('.theme-switch input[type="checkbox"]'); + function switchTheme(e) { + if (e.target.checked) { + document.documentElement.setAttribute('data-theme', 'dark'); + localStorage.setItem('theme', 'dark'); + } else { + document.documentElement.setAttribute('data-theme', 'light'); + localStorage.setItem('theme', 'light'); + } + } + + toggleSwitch.addEventListener('change', switchTheme, false); + + + if (window.matchMedia && window.matchMedia('(prefers-color-scheme: dark)').matches) { + document.documentElement.setAttribute('data-theme', "dark"); + toggleSwitch.checked = true; + } else if (window.matchMedia && window.matchMedia('(prefers-color-scheme: light)').matches) { + document.documentElement.setAttribute('data-theme', "light"); + toggleSwitch.checked = false; + } else { + const currentTheme = localStorage.getItem('theme') ? localStorage.getItem('theme') : null; + if (currentTheme) { + document.documentElement.setAttribute('data-theme', currentTheme); + + if (currentTheme === 'dark') { + toggleSwitch.checked = true; + } + } + } } @@ -1222,6 +74,13 @@ function main() {

unittest2

+
+ +     Dark Mode +
+

Authors: Zahary Karadjov, Ștefan Talpalaru

This module implements boilerplate to make unit testing easy.

The test status and name is printed after any output or traceback.

@@ -1422,6 +287,14 @@ nim c -r test "fast_suite::mytest*" nim c -r test "auth*::" "crypto::hashing*" # Run suites starting with 'bug #' and standalone tests starting with '#' nim c -r test 'bug #*::' '::#*' +

Command line arguments

+ + + + + +
--helpPrint short help and quit
--xml:fileWrite JUnit-compatible XML report to file
--consoleWrite report to the console (default, when no other output is selected)

Command line parsing can be disabled with -d:unittest2DisableParamFiltering.

+

Running tests in parallel

To enable the threadpool-based test parallelisation, "--threads:on" needs to be passed to the compiler, along with "-d:nimtestParallel" or the NIMTEST_PARALLEL environment variable:

 nim c -r --threads:on -d:nimtestParallel testfile.nim
@@ -1463,69 +336,98 @@ NIMTEST_PARALLEL=1 nim c -r --threads:on testfile.nim

There are some imp

TestStatus = enum
   OK, FAILED, SKIPPED
+ The status of a test when it is done. +  Source +Edit
OutputLevel = enum
-  PRINT_ALL,                  ## Print as much as possible.
-  PRINT_FAILURES,             ## Print only the failed tests.
-  PRINT_NONE                  ## Print nothing.
+ PRINT_ALL, ## Print as much as possible. + PRINT_FAILURES, ## Print only the failed tests. + PRINT_NONE ## Print nothing.
+ The output verbosity of the tests. +  Source +Edit
TestResult = object
   suiteName*: string ## Name of the test suite that contains this test case.
-                   ## Can be ``nil`` if the test case is not in a suite.
-  testName*: string            ## Name of the test case
+                     ## Can be ``nil`` if the test case is not in a suite.
+  testName*: string          ## Name of the test case
   status*: TestStatus
+  duration*: Duration
 
+  Source +Edit +
OutputFormatter = ref object of RootObj
+  Source +Edit +
ConsoleOutputFormatter = ref object of OutputFormatter
-  colorOutput: bool            ## Have test results printed in color.
-                  ## Default is true for the non-js target,
-                  ## for which ``stdout`` is a tty.
-                  ## Setting the environment variable
-                  ## ``NIMTEST_COLOR`` to ``always`` or
-                  ## ``never`` changes the default for the
-                  ## non-js target to true or false respectively.
-                  ## The deprecated environment variable
-                  ## ``NIMTEST_NO_COLOR``, when set,
-                  ## changes the defualt to true, if
-                  ## ``NIMTEST_COLOR`` is undefined.
-  outputLevel: OutputLevel     ## Set the verbosity of test results.
-                         ## Default is ``PRINT_ALL``, unless
-                         ## the ``NIMTEST_OUTPUT_LVL`` environment
-                         ## variable is set for the non-js target.
+  colorOutput: bool ## Have test results printed in color.
+                    ## Default is `auto` depending on `isatty(stdout)`, or override it with
+                    ## `-d:nimUnittestColor:auto|on|off`.
+                    ## 
+                    ## Deprecated: Setting the environment variable `NIMTEST_COLOR` to `always`
+                    ## or `never` changes the default for the non-js target to true or false respectively.
+                    ## Deprecated: the environment variable `NIMTEST_NO_COLOR`, when set, changes the
+                    ## default to true, if `NIMTEST_COLOR` is undefined.
+  outputLevel: OutputLevel ## Set the verbosity of test results.
+                           ## Default is `PRINT_ALL`, or override with:
+                           ## `-d:nimUnittestOutputLevel:PRINT_ALL|PRINT_FAILURES|PRINT_NONE`.
+                           ## 
+                           ## Deprecated: the `NIMTEST_OUTPUT_LVL` environment variable is set for the non-js target.
   isInSuite: bool
   isInTest: bool
 
+  Source +Edit +
JUnitOutputFormatter = ref object of OutputFormatter
   stream: Stream
-  testErrors: seq[string]
-  testStartTime: float
-  testStackTrace: string
+  defaultSuite: JUnitSuite
+  suites: seq[JUnitSuite]
+  currentSuite: int
 
+  Source +Edit +
@@ -1533,9 +435,16 @@ The output verbosity of the tests.

Vars

-
abortOnError: bool
+
abortOnError: bool
-Set to true in order to quit immediately on fail. Default is false, unless the NIMTEST_ABORT_ON_ERROR environment variable is set for the non-js target. + +

Set to true in order to quit immediately on fail. Default is false, or override with -d:nimUnittestAbortOnError:on|off.

+

Deprecated: can also override depending on whether NIMTEST_ABORT_ON_ERROR environment variable is set.

+ +  Source +Edit
@@ -1544,9 +453,14 @@ Set to true in order to quit immediately on fail. Default is false, unless the <

Consts

-
paralleliseTests = false
+
paralleliseTests = false
+ Whether parallel test running was enabled (set at compile time). This constant might be useful in custom output formatters. +  Source +Edit
@@ -1554,60 +468,104 @@ Whether parallel test running was enabled (set at compile time). This constant m

Procs

- -
proc clearOutputFormatters() {...}{.raises: [], tags: [].}
+ +
proc addOutputFormatter(formatter: OutputFormatter) {...}{.raises: [Defect], tags: [].}
+  Source +Edit +
- -
proc addOutputFormatter(formatter: OutputFormatter) {...}{.raises: [], tags: [].}
+ +
proc resetOutputFormatters() {...}{.raises: [Defect], tags: [].}
+  Source +Edit +
-
proc newConsoleOutputFormatter(outputLevel: OutputLevel = PRINT_ALL;
-                              colorOutput = true): ConsoleOutputFormatter {...}{.
-    raises: [], tags: [].}
+
proc newConsoleOutputFormatter(outputLevel: OutputLevel = outputLevelDefault;
+                               colorOutput = true): ConsoleOutputFormatter {...}{.
+    raises: [Defect], tags: [].}
+  Source +Edit +
- -
proc defaultConsoleFormatter(): ConsoleOutputFormatter {...}{.raises: [],
+
+
proc defaultConsoleFormatter(): ConsoleOutputFormatter {...}{.raises: [Defect],
     tags: [ReadEnvEffect].}
-
- -
proc newJUnitOutputFormatter(stream: Stream): JUnitOutputFormatter {...}{.
-    raises: [Exception], tags: [WriteIOEffect].}
-
-Creates a formatter that writes report to the specified stream in JUnit format. The stream is NOT closed automatically when the test are finished, because the formatter has no way to know when all tests are finished. You should invoke formatter.close() to finalize the report. +  Source +Edit
- -
proc close(formatter: JUnitOutputFormatter) {...}{.raises: [Exception],
-    tags: [WriteIOEffect].}
+ +
proc newJUnitOutputFormatter(stream: Stream): JUnitOutputFormatter {...}{.
+    raises: [Defect], tags: [WriteIOEffect].}
-Completes the report and closes the underlying stream. + +Creates a formatter that writes report to the specified stream in JUnit format. The stream is NOT closed automatically when the test are finished, because the formatter has no way to know when all tests are finished. You should invoke formatter.close() to finalize the report. +  Source +Edit + +
+ +
proc parseParameters(args: openArray[string]) {...}{.raises: [Defect],
+    tags: [WriteIOEffect, ReadEnvEffect].}
+
+ + +  Source +Edit
-
proc checkpoint(msg: string) {...}{.raises: [], tags: [].}
+
proc checkpoint(msg: string) {...}{.raises: [Defect], tags: [].}
-Set a checkpoint identified by msg. Upon test failure all checkpoints encountered so far are printed out. Example:
checkpoint("Checkpoint A")
+
+Set a checkpoint identified by msg. Upon test failure all checkpoints encountered so far are printed out. Example:
checkpoint("Checkpoint A")
 check((42, "the Answer to life and everything") == (1, "a"))
 checkpoint("Checkpoint B")

outputs "Checkpoint A" once it fails.

+  Source +Edit
- -
proc disableParamFiltering() {...}{.raises: [], tags: [].}
+ +
proc disableParamFiltering() {...}{.deprecated: "Compile with -d:unittest2DisableParamFiltering instead",
+                               raises: [], tags: [].}
-disables filtering tests with the command line params +
+ Deprecated: Compile with -d:unittest2DisableParamFiltering instead +
+ + +  Source +Edit
@@ -1616,106 +574,209 @@ disables filtering tests with the command line params

Methods

-
method suiteStarted(formatter: OutputFormatter; suiteName: string) {...}{.base, gcsafe,
-    raises: [], tags: [].}
+
method suiteStarted(formatter: OutputFormatter; suiteName: string) {...}{.base,
+    gcsafe, raises: [Defect], tags: [].}
+  Source +Edit +
-
method testStarted(formatter: OutputFormatter; testName: string) {...}{.base, gcsafe,
-    raises: [], tags: [].}
+
method testStarted(formatter: OutputFormatter; testName: string) {...}{.base, gcsafe,
+    raises: [Defect], tags: [].}
+  Source +Edit +
-
method failureOccurred(formatter: OutputFormatter; checkpoints: seq[string];
-                      stackTrace: string) {...}{.base, gcsafe, raises: [], tags: [].}
+
method failureOccurred(formatter: OutputFormatter; checkpoints: seq[string];
+                       stackTrace: string) {...}{.base, gcsafe, raises: [Defect],
+    tags: [].}
+ stackTrace is provided only if the failure occurred due to an exception. checkpoints is never nil. +  Source +Edit
-
method testEnded(formatter: OutputFormatter; testResult: TestResult) {...}{.base, gcsafe,
-    raises: [], tags: [].}
+
method testEnded(formatter: OutputFormatter; testResult: TestResult) {...}{.base,
+    gcsafe, raises: [Defect], tags: [].}
+  Source +Edit +
-
method suiteEnded(formatter: OutputFormatter) {...}{.base, gcsafe, raises: [], tags: [].}
-
- - -
- -
method suiteStarted(formatter: ConsoleOutputFormatter; suiteName: string) {...}{.
-    raises: [IOError], tags: [WriteIOEffect].}
-
- - -
- -
method testStarted(formatter: ConsoleOutputFormatter; testName: string) {...}{.raises: [],
+
method suiteEnded(formatter: OutputFormatter) {...}{.base, gcsafe, raises: [Defect],
     tags: [].}
+  Source +Edit +
- -
method failureOccurred(formatter: ConsoleOutputFormatter; checkpoints: seq[string];
-                      stackTrace: string) {...}{.raises: [], tags: [].}
+ +
method testRunEnded(formatter: OutputFormatter) {...}{.base, gcsafe,
+    raises: [Defect], tags: [].}
+  Source +Edit + +
+ +
method suiteStarted(formatter: ConsoleOutputFormatter; suiteName: string) {...}{.
+    raises: [Defect], tags: [WriteIOEffect].}
+
+ + +  Source +Edit + +
+ +
method testStarted(formatter: ConsoleOutputFormatter; testName: string) {...}{.
+    raises: [Defect], tags: [].}
+
+ + +  Source +Edit + +
+ +
method failureOccurred(formatter: ConsoleOutputFormatter;
+                       checkpoints: seq[string]; stackTrace: string) {...}{.
+    raises: [Defect], tags: [].}
+
+ + +  Source +Edit +
-
method testEnded(formatter: ConsoleOutputFormatter; testResult: TestResult) {...}{.
-    raises: [IOError], tags: [WriteIOEffect].}
+
method testEnded(formatter: ConsoleOutputFormatter; testResult: TestResult) {...}{.
+    raises: [Defect], tags: [WriteIOEffect].}
+  Source +Edit +
-
method suiteEnded(formatter: ConsoleOutputFormatter) {...}{.raises: [], tags: [].}
+
method suiteEnded(formatter: ConsoleOutputFormatter) {...}{.raises: [Defect],
+    tags: [].}
+  Source +Edit +
-
method suiteStarted(formatter: JUnitOutputFormatter; suiteName: string) {...}{.
-    raises: [Exception, ValueError], tags: [WriteIOEffect].}
+
method suiteStarted(formatter: JUnitOutputFormatter; suiteName: string) {...}{.
+    raises: [Defect], tags: [].}
+  Source +Edit +
-
method testStarted(formatter: JUnitOutputFormatter; testName: string) {...}{.raises: [],
-    tags: [TimeEffect].}
+
method testStarted(formatter: JUnitOutputFormatter; testName: string) {...}{.
+    raises: [Defect], tags: [].}
+  Source +Edit +
-
method failureOccurred(formatter: JUnitOutputFormatter; checkpoints: seq[string];
-                      stackTrace: string) {...}{.raises: [], tags: [].}
+
method failureOccurred(formatter: JUnitOutputFormatter;
+                       checkpoints: seq[string]; stackTrace: string) {...}{.
+    raises: [Defect], tags: [].}
+ stackTrace is provided only if the failure occurred due to an exception. checkpoints is never nil. +  Source +Edit
-
method testEnded(formatter: JUnitOutputFormatter; testResult: TestResult) {...}{.
-    raises: [Exception, ValueError], tags: [TimeEffect, WriteIOEffect].}
+
method testEnded(formatter: JUnitOutputFormatter; testResult: TestResult) {...}{.
+    raises: [Defect], tags: [].}
+  Source +Edit +
-
method suiteEnded(formatter: JUnitOutputFormatter) {...}{.raises: [Exception],
+
method suiteEnded(formatter: JUnitOutputFormatter) {...}{.raises: [Defect], tags: [].}
+
+ + +  Source +Edit + +
+ +
method testRunEnded(formatter: JUnitOutputFormatter) {...}{.raises: [Defect],
     tags: [WriteIOEffect].}
+Completes the report and closes the underlying stream. +  Source +Edit
@@ -1724,33 +785,55 @@ disables filtering tests with the command line params

Macros

-
macro check(conditions: untyped): untyped
+
macro check(conditions: untyped): untyped
-Verify if a statement or a list of statements is true. A helpful error message and set checkpoints are printed out on failure (if outputLevel is not PRINT_NONE). Example:
import strutils
 
-check("AKB48".toLowerAscii() == "akb48")
+Verify if a statement or a list of statements is true. A helpful error message and set checkpoints are printed out on failure (if outputLevel is not PRINT_NONE).
+

Examples:

+
import
+  std / strutils
 
-let teams = {'A', 'K', 'B', '4', '8'}
-
-check:
-  "AKB48".toLowerAscii() == "akb48"
-  'C' in teams
+check("AKB48".toLowerAscii() == "akb48") +let teams = {'A', 'K', 'B', '4', '8'} +check: + "AKB48".toLowerAscii() == "akb48" + 'C' notin teams
+  Source +Edit
-
macro expect(exceptions: varargs[typed]; body: untyped): untyped
+
macro expect(exceptions: varargs[typed]; body: untyped): untyped
-Test if body raises an exception found in the passed exceptions. The test passes if the raised exception is part of the acceptable exceptions. Otherwise, it fails. Example:
import math, random
-proc defectiveRobot() =
-  randomize()
-  case random(1..4)
-  of 1: raise newException(OSError, "CANNOT COMPUTE!")
-  of 2: discard parseInt("Hello World!")
-  of 3: raise newException(IOError, "I can't do that Dave.")
-  else: assert 2 + 2 == 5
 
-expect IOError, OSError, ValueError, AssertionError:
-  defectiveRobot()
+Test if body raises an exception found in the passed exceptions. The test passes if the raised exception is part of the acceptable exceptions. Otherwise, it fails. +

Examples:

+
import
+  std / [math, random, strutils]
+
+proc defectiveRobot() =
+  randomize()
+  case rand(1 .. 4)
+  of 1:
+    raise newException(OSError, "CANNOT COMPUTE!")
+  of 2:
+    discard parseInt("Hello World!")
+  of 3:
+    raise newException(IOError, "I can\'t do that Dave.")
+  else:
+    assert 2 + 2 == 5
+
+when (NimMajor, NimMinor, NimPatch) < (1, 4, 0):
+  type
+    AssertionDefect = AssertionError
+expect IOError,OSError,ValueError,AssertionDefect:
+  defectiveRobot()
+  Source +Edit
@@ -1759,9 +842,10 @@ Test if body raises an exception found in the passed exceptio

Templates

-
template suite(name, body) {...}{.dirty.}
+
template suite(name, body) {...}{.dirty.}
-

Declare a test suite identified by name with optional setup and/or teardown section.

+ +

Declare a test suite identified by name with optional setup and/or teardown section.

A test suite is a series of one or more related tests sharing a common fixture (setup, teardown). The fixture is executed for EACH test.

suite "test suite for addition":
   setup:
@@ -1777,37 +861,61 @@ Test if body raises an exception found in the passed exceptio
 
[Suite] test suite for addition
   [OK] 2 + 2 = 4
   [OK] (2 + -2) != 4
+  Source +Edit
- -
template test(name, body)
+ +
template test(name: string; body: untyped)
-Define a single test case identified by name.
test "roses are red":
+
+Define a single test case identified by name.
test "roses are red":
   let roses = "red"
   check(roses == "red")

The above code outputs:

[OK] roses are red
+  Source +Edit
- -
template fail()
+ +
template fail()
+ Print out the checkpoints encountered so far and quit if abortOnError is true. Otherwise, erase the checkpoints and indicate the test has failed (change exit code and test status). This template is useful for debugging, but is otherwise mostly used internally. Example:
checkpoint("Checkpoint A")
 complicatedProcInThread()
 fail()

outputs "Checkpoint A" before quitting.

+  Source +Edit
- -
template skip()
+ +
template skip()
-Mark the test as skipped. Should be used directly in case when it is not possible to perform test for reasons depending on outer environment, or certain application logic conditions or configurations. The test code is still executed.
if not isGLConextCreated():
+
+Mark the test as skipped. Should be used directly in case when it is not possible to perform test for reasons depending on outer environment, or certain application logic conditions or configurations. The test code is still executed.
if not isGLContextCreated():
   skip()
+  Source +Edit
-
template require(conditions: untyped)
+
template require(conditions: untyped)
-Same as check except any failed test causes the program to quit immediately. Any teardown statements are not executed and the failed test output is not generated. + +Same as check except any failed test causes the program to quit immediately. Any teardown statements are not executed and the failed test output is not generated. +  Source +Edit
@@ -1820,7 +928,7 @@ Same as check except any failed test causes the program to quit imm
diff --git a/unittest2.nim b/unittest2.nim index 1f7fe90..f7da301 100644 --- a/unittest2.nim +++ b/unittest2.nim @@ -8,7 +8,7 @@ {.push raises: [Defect].} -## :Authors: Zahary Karadjov, Stefan Talpalaru +## :Authors: Zahary Karadjov, Ștefan Talpalaru ## ## This module implements boilerplate to make unit testing easy. ## @@ -62,6 +62,16 @@ ## # Run suites starting with 'bug #' and standalone tests starting with '#' ## nim c -r test 'bug #*::' '::#*' ## +## Command line arguments +## ====================== +## +## --help Print short help and quit +## --xml:file Write JUnit-compatible XML report to `file` +## --console Write report to the console (default, when no other output is +## selected) +## +## Command line parsing can be disabled with `-d:unittest2DisableParamFiltering`. +## ## Running tests in parallel ## ========================= ## @@ -118,16 +128,6 @@ ## ## suiteTeardown: ## echo "suite teardown: run once after the tests" -## -## Command line arguments -## ====================== -## -## --help Print short help and quit -## --xml:file Write JUnit-compatible XML report to `file` -## --console Write report to the console (default, when no other output is -## selected) -## -## Command line parsing can be disabled with `-d:nimtestDisableParamFiltering`. import std/[locks, macros, sets, strutils, streams, times, monotimes] @@ -136,10 +136,10 @@ when declared(stdout): const useTerminal = not defined(js) -# compile with `-d:nimtestDisableParamFiltering` to skip parsing test filters, +# compile with `-d:unittest2DisableParamFiltering` to skip parsing test filters, # `--help` and other command line options - you can manually call # `parseParameters` instead then. -const autoParseArgs = not defined(nimtestDisableParamFiltering) +const autoParseArgs = not defined(unittest2DisableParamFiltering) when useTerminal: import std/terminal @@ -252,7 +252,6 @@ var formatters {.guard: formattersLock.}: seq[OutputFormatter] testFiltersLock: Lock testsFilters {.guard: testFiltersLock.}: HashSet[string] - disabledParamFiltering: bool const outputLevelDefault = PRINT_ALL @@ -953,6 +952,9 @@ macro expect*(exceptions: varargs[typed], body: untyped): untyped = of 3: raise newException(IOError, "I can't do that Dave.") else: assert 2 + 2 == 5 + when (NimMajor, NimMinor, NimPatch) < (1, 4, 0): + type AssertionDefect = AssertionError + expect IOError, OSError, ValueError, AssertionDefect: defectiveRobot() @@ -974,5 +976,5 @@ macro expect*(exceptions: varargs[typed], body: untyped): untyped = result = getAst(expectBody(errorTypes, errorTypes.lineInfo, body)) proc disableParamFiltering* {.deprecated: - "Compile with -d:nimtestDisableParamFiltering instead".} = - discard \ No newline at end of file + "Compile with -d:unittest2DisableParamFiltering instead".} = + discard