https://github.com/ngarbezza/testy
A minimal Javascript testing framework, for educational purposes. Live at npm at @pmoo/testy.
https://github.com/ngarbezza/testy
hacktoberfest javascript js learning oop tdd teaching test-driven-development testing unit-testing
Last synced: 13 days ago
JSON representation
A minimal Javascript testing framework, for educational purposes. Live at npm at @pmoo/testy.
- Host: GitHub
- URL: https://github.com/ngarbezza/testy
- Owner: ngarbezza
- License: mit
- Created: 2018-10-15T05:22:12.000Z (over 6 years ago)
- Default Branch: main
- Last Pushed: 2025-04-07T01:21:44.000Z (19 days ago)
- Last Synced: 2025-04-13T01:05:46.624Z (13 days ago)
- Topics: hacktoberfest, javascript, js, learning, oop, tdd, teaching, test-driven-development, testing, unit-testing
- Language: JavaScript
- Homepage: https://github.com/ngarbezza/testy/
- Size: 1.9 MB
- Stars: 32
- Watchers: 5
- Forks: 13
- Open Issues: 35
-
Metadata Files:
- Readme: README.md
- Changelog: CHANGELOG.md
- Contributing: CONTRIBUTING.md
- Funding: .github/FUNDING.yml
- License: LICENSE
- Code of conduct: CODE_OF_CONDUCT.md
Awesome Lists containing this project
README
# Testy

\
[](https://codeclimate.com/github/ngarbezza/testy)
[](https://codeclimate.com/github/ngarbezza/testy)
[](https://codeclimate.com/github/ngarbezza/testy)
[](https://dashboard.stryker-mutator.io/reports/github.com/ngarbezza/testy/main)
\
[](https://sonarcloud.io/summary/new_code?id=ngarbezza_testy)
[](https://sonarcloud.io/summary/new_code?id=ngarbezza_testy)
[](https://sonarcloud.io/summary/new_code?id=ngarbezza_testy)
[](https://sonarcloud.io/summary/new_code?id=ngarbezza_testy)
\




\


\


[](#contributors)
A very simple JS testing framework, for educational purposes. Live at npm at [@pmoo/testy](https://www.npmjs.com/package/@pmoo/testy).
:arrow_right: [Documentación en español aquí](README_es.md)
:construction_worker: [Contributing guidelines](CONTRIBUTING.md)## Sponsors
## Getting started
`npm install --save-dev @pmoo/testy` (if you use [npm](https://www.npmjs.com/)) \
`yarn add --dev @pmoo/testy` (if you use [yarn](https://classic.yarnpkg.com/en/))**Supported Node versions**: 18.x or higher (versions with active and security support
listed [here](https://endoflife.date/nodejs))## Usage
### Writing test suites
A test suite is a file ending `_test.js` that looks like this:
```javascript
// my_test.js
import { suite, test, assert } from "@pmoo/testy";suite("a boring test suite", () => {
test("42 is 42, not surprising", () => {
assert.that(42).isEqualTo(42);
});
});
```A test suite represents a grouping of tests, and it is implemented as a function call to `suite` passing a name and a
zero-argument function, which is the suite body.A test is implemented as a function call to `test()`, passing a name and the test body as a zero-argument function.
Inside the test you can call different assertions that are documented in detail later on.
### Running Testy
You can run an individual test file using:
```sh
npx testy my_test.js
```Or, you can run it without arguments to run all the tests (by default it looks on a `tests` folder located in the project
root):```sh
npx testy
```You can also add it as the `test` script for npm/yarn in your `package.json`:
```json
{
...
"scripts": {
"test": "npx testy"
},
...
}
```And then run the tests using `npm test` or `yarn test`.
### Configuring Testy
Testy will look for a `.testyrc.json` configuration file in the project root directory. You can use this configuration
as a template (values here are the defaults):```json
{
"directory": "./tests", // directory including your test files
"filter": ".*_test.js$", // which convention to use to recognize test files
"language": "en", // language of the output messages. "es" and "en" supported for now
"failFast": false, // enable/disable fail fast mode (stop as soon as a failed test appears)
"randomOrder": false // enable/disable execution of tests in random order
"timeoutMs": 1000 // sets the per-test timeout in milliseconds
"language": "en", // language of the output messages. "es", "it" and "en" supported for now
}
```You can also pass a configuration through the console when running tests by adding these available options after your
test file path:- `-f` or `--fail-fast` to enable fail fast mode.
- `-r` or `--randomize` to enable the execution of tests in random order.
- `-l xx` or `--language xx` where `xx` must be either `es` for Spanish, `en` for English or `it` for Italian.These console parameters can be sent in any order and combined as you want.
These are all the configuration parameters you can set. Feel free to change it according to your needs.
When declaring this configuration, every test suite under the `tests` directory (matching files ending with
`*test.js`) will be executed.### Examples and available assertions
There must be at least one assertion on the test to be valid. These are all the supported assertion types:
- Boolean assertions:
- `assert.that(boolean).isTrue()` or `assert.isTrue(boolean)`. It does a strict comparison against `true`
(`object === true`)
- `assert.that(boolean).isFalse()` or `assert.isFalse(boolean)`. It does a strict comparison against `false`
(`object === false`)
- Equality assertions:
- `assert.that(actual).isEqualTo(expected)` or `assert.areEqual(actual, expected)`.
- `assert.that(actual).isNotEqualTo(expected)` or `assert.areNotEqual(actual, expected)`
- Equality assertions use a deep object comparison (based on Node's `assert` module) and fail if objects under
comparison have circular references.
- Equality criteria on non-primitive objects can be specified:
- Passing an extra two-arg comparator function to `isEqualTo(expected, criteria)` or
`areEqual(actual, expected, criteria)`
- Passing a method name that `actual` object understands: `isEqualTo(expected, 'myEqMessage')` or
`areEqual(actual, expected, 'myEqMessage')`
- By default, if `actual` has an `equals` method it will be used.
- If we compare `undefined` with `undefined` using `isEqualTo()`, it will make the test fail. For explicit
check for `undefined`, use the `isUndefined()`/`isNotUndefined()` assertions documented above.
- Identity assertions:
- `assert.that(actual).isIdenticalTo(expected)` or `assert.areIdentical(actual, expected)`
- `assert.that(actual).isNotIdenticalTo(expected)` or `assert.areNotIdentical(actual, expected)`
- Identity assertions check if two references point to the same object using the `===` operator.
- Check for `undefined` presence/absence:
- `assert.that(aValue).isUndefined()` or `assert.isUndefined(aValue)`
- `assert.that(aValue).isNotUndefined()` or `assert.isNotUndefined(aValue)`
- Check for `null` presence/absence:
- `assert.that(aValue).isNull()` or `assert.isNull(aValue)`
- `assert.that(aValue).isNotNull()` or `assert.isNotNull(aValue)`
- Exception testing:
- `assert.that(() => { ... }).raises(error)` or with regex `.raises(/part of message/)`
- `assert.that(() => { ... }).doesNotRaise(error)`
- `assert.that(() => { ... }).doesNotRaiseAnyErrors()`
- Numeric assertions:
- Comparators:
- `assert.that(aNumber).isGreaterThan(anotherNumber)`
- `assert.that(aNumber).isLessThan(anotherNumber)`
- `assert.that(aNumber).isGreaterThanOrEqualTo(anotherNumber)`
- `assert.that(aNumber).isLessThanOrEqualTo(anotherNumber)`
- Rounding:
- `assert.that(aNumber).isNearTo(anotherNumber)`. There's a second optional argument that indicates the number
of digits to be used for precision. Default is `4`.
- String assertions:
- `assert.that(string).matches(regexOrString)` or `assert.isMatching(string, regexOrString)`
- Array inclusion:
- `assert.that(collection).includes(object)`
- `assert.that(collection).doesNotInclude(object)`
- `assert.that(collection).includesExactly(...objects)`
- Emptiness
- `assert.that(collection).isEmpty()` or `assert.isEmpty(collection)`
- `assert.that(collection).isNotEmpty()` or `assert.isNotEmpty(collection)`
- the collection under test can be an `Array`, a `String` or a `Set`Please take a look at the `tests` folder, you'll find examples of each possible test you can write. Testy is
self-tested.### Running testy globally
If you don't have a NPM project you can install Testy globally using `npm install -g testy` and then run
`testy `.### Other features
- **Running code before/after every test**: just like many testing frameworks have, there is a way to execute some code
before or after each test in a suite using the `before()` and `after()` functions, respectively. You can use only one
definition of `before()` and `after()` per suite, and they always receive a function as argument. Example:```javascript
import { suite, test, assert, before, after } from '@pmoo/testy';suite('using the before() and after() helpers', () => {
let answer;before(() => {
answer = 42;
});test('checking the answer', () => {
assert.that(answer).isEqualTo(42);
});after(() => {
answer = undefined;
});
});
```- **Support for pending tests**: if a test has no body, it will be reported as `[WIP]` and it won't be considered a
failure.
- **Support for skipped tests**: you can skip a test by adding `.skip()` after test declaration, it will be reported
as `[SKIP]` and it won't be executed nor considered a failure.```javascript
import { suite, test, assert } from '@pmoo/testy';suite('I am a suite with a skipped test', () => {
test('I am a skipped test', async () => {
assert.that(1).isEqualTo(1);
}).skip();
});
```- **Support for exclusive tests**: you can mark a test as exclusive for a run by adding `.only()` after test
declaration. If there's at least one test marked as exclusive for run in a suite, only the tests marked as exclusive
for run will be executed and the rest will be skipped.```javascript
import { suite, test, assert } from '@pmoo/testy';suite('I am a suite with an exclusive test', () => {
test('I am an exclusive for run test', async () => {
assert.that(1).isEqualTo(1);
}).only();});
```- **Support for skipped suites**: you can skip a test suite by adding `.skip()` after suite declaration. All of the
suite's test will be reported as `[SKIP]` and won't be executed nor considered a failure.- **Support for asynchronous tests**: if the code you are testing has `async` logic, you can `await` inside the test
definition and make assertions later. You can also use it on `before()` and `after()` declarations. Example:```javascript
import { suite, test, assert, before } from '@pmoo/testy';const promiseOne = async () => Promise.resolve(42);
const promiseTwo = async () => Promise.resolve(21);suite('using async & await', () => {
let answerOne;before(async () => {
answerOne = await promiseOne();
});test('comparing results from promises', async () => {
const answerTwo = await promiseTwo();
assert.that(answerOne).isEqualTo(42);
assert.that(answerTwo).isEqualTo(21);
});
});
```- **Fail-Fast mode**: if enabled, it stops execution in the first test that fails (or has an error). Remaining tests
will be marked as skipped.
- **Run tests and suites in random order**: a good test suite does not depend on a particular order. Enabling this
setting is a good way to ensure that.
- **Strict check for assertions**: if a test does not evaluate any assertion while it is executed, the result is
considered an error. Basically, a test with no assertion is considered a "bad" test.
- **Explicitly failing or marking a test as pending**: there's a possibility of marking a test as failed or pending,
for example:```javascript
import { suite, test, fail, pending } from "@pmoo/testy";suite("marking tests as failed and pending", () => {
test("marking as failed", () => fail.with("should not be here"));test("marking as pending", () =>
pending.dueTo("did not have time to finish"));
});
```The output includes the messages provided:
```text
[FAIL] marking as failed
=> should not be here
[WIP] marking as pending
=> did not have time to finish
```## Why?
Why another testing tool? The main reason is that we want to keep simplicity, something it's hard to see in the main
testing tools out there.- **Zero dependencies:** right now, this project does not depend on any npm package, making the tool easy to install,
and fast: essential to have immediate feedback while doing TDD. This is also good for installing on places where the
internet connection is not good and we don't want to download hundreds of libraries.
- **Understandable object-oriented code:** we want to use this tool for teaching, so eventually we'll look at the code
during lessons, and students should be able to see what is going on, and even contributing at it, with no dark magic
involved. Also, we try to follow good OO practices.
- **Unique set of features:** we are not following any specification nor trying to copy behavior from other approaches
(like the "xUnit" or "xSpec" way).["Design Principles Behind Smalltalk"](https://www.cs.virginia.edu/~evans/cs655/readings/smalltalk.html) is a source of
inspiration for this work. We try to follow the same principles here.## Contributing
Please take a look at the [Contributing section](CONTRIBUTING.md).
## Contributors
Thanks goes to these wonderful people ([emoji key](https://allcontributors.org/docs/en/emoji-key)):
Facundo Javier Gelatti
⚠️ 💻
Tomer Ben-Rachel
⚠️ 💻
Abraão Duarte
💻
adico
💻 ⚠️
Askar Imran
💻 ⚠️
Nigel Yong
💻
Chelsie Ng
💻
Pablo T
⚠️ 💻
Francisco Jaimes Freyre
💻 ⚠️ 📖
giovannipessiva
🌍
Abhishek Khatri
💻
Ignacio Robledo
💻 ⚠️
Marco Ellwanger
💻 ⚠️
María Belén Amat
💻 ⚠️ 🌍 📖
This project follows the [all-contributors](https://github.com/all-contributors/all-contributors) specification.
Contributions of any kind welcome!