Important: This documentation covers Yarn 1 (Classic).
For Yarn 2+ docs and migration guide, see yarnpkg.com.

Package detail

sass-true

oddbird41.3kBSD-3-Clause8.1.0TypeScript support: included

Unit testing for Sass.

unit-test, test, sass, libsass, TDD, eyeglass-module

readme

True

License

  1. To make true; shape, adjust, place, etc., exactly or accurately:

    True the wheels of a bicycle after striking a pothole.

  2. To make even, symmetrical, level, etc. (often followed by up):

    True up the sides of a door.

  3. To test your Sass code; debug, perfect, etc. (often using True):

    True your sweet plugin before you deploy.

True is a unit-testing tool for Sass code. All of the tests are written in plain Sass, and can be compiled using Dart Sass – but we also provide integration with JavaScript test runners (e.g. Mocha or Jest), for extra features and improved reporting.

Install

In command line:

npm install --save-dev sass-true

True requires Dart Sass v1.45.0 or higher, so install it if you haven't already:

npm install --save-dev sass-embedded # or `sass`

Import in your test directory, like any other Sass file:

@use 'pkg:sass-true' as *;

If you are not using the Sass Node.js package importer, you may need to include the full path name:

// This is only an example, your path may be different
@use '../node_modules/sass-true' as *;

Or if you are using the JavaScript test runner integration:

@use 'true' as *;

One Setting

$terminal-output (boolean), defaults to true

  • true will show detailed information in the terminal for debugging failed assertions or reporting final results. This is the default, and best for compiling without a JavaScript test runner.
  • false will turn off all terminal output from Sass, though Mocha/Jest will continue to use the terminal for reporting.

If you are still using @import rather than @use, there is an import path available - which retains the legacy prefixed $true-terminal-output variable name:

// Your path may be different
@import '../node_modules/sass-true/sass/true';

Usage

True is based on common JS-testing patterns, allowing both a test-module/test syntax, and the newer describe/it for defining the structure:

@include test-module('Zip [function]') {
  @include test('Zips multiple lists into a single multi-dimensional list') {
    // Assert the expected results
    @include assert-equal(zip(a b c, 1 2 3), (a 1, b 2, c 3));
  }
}

This is the same as…

@include describe('Zip [function]') {
  @include it('Zips multiple lists into a single multi-dimensional list') {
    // Assert the expected results
    @include assert-equal(zip(a b c, 1 2 3), (a 1, b 2, c 3));
  }
}

Sass is able to compare values internally, meaning function-output and variable values can easily be compared and reported during Sass compilation.

CSS output tests, on the other hand, have to be compared after compilation is complete. You can do that by hand if you want (git diff is helpful for noticing changes), or you can use a test runner such as Mocha or Jest.

Output tests fit the same structure, but assertions take a slightly different form, with an outer assert mixin, and a matching pair of output and expect to contain the output-values.

// Test CSS output from mixins
@include it('Outputs a font size and line height based on keyword') {
  @include assert {
    @include output {
      @include font-size('large');
    }

    @include expect {
      font-size: 2rem;
      line-height: 3rem;
    }
  }
}

You can optionally show a summary report in CSS and/or the command line, after the tests have completed:

@include report;

See the full documentation online for more details. See CHANGELOG.md when upgrading from an older version of True.

Using Mocha, Jest, or other JS test runners

  1. Install true via npm:

    npm install --save-dev sass-true
  2. [Optional] Install Dart Sass (sass-embedded or sass), if not already installed.

    npm install --save-dev sass-embedded # or `sass`
  3. Write some Sass tests in test/test.scss (see above).

  4. Write a shim JS test file in test/sass.test.js:

    const path = require('node:path');
    const sassTrue = require('sass-true');
    
    const sassFile = path.join(__dirname, 'test.scss');
    sassTrue.runSass({ describe, it }, sassFile);
  5. Run Mocha/Jest, and see your Sass tests reported in the command line.

Note: Because of differences between Jest globals and Node globals, Dart Sass often errors when trying to compile in a Jest environment (e.g. J.getInterceptor$ax(...).map$1$1 is not a function). This can usually be fixed by installing jest-environment-node-single-context and setting testEnvironment: 'jest-environment-node-single-context' in jest.config.js. See possible related issues.

Note: Jest does not watch for changes in Sass files by default. To use jest --watch with True, add "scss" to your moduleFileExtensions setting.

You can call runSass more than once, if you have multiple Sass test files you want to run separately.

The first argument is an object with required describe and it options, and optional sass, contextLines and sourceType options.

Any JS test runner with equivalents to Mocha's or Jest's describe and it should be usable in the same way: just pass your test runner's describe and it equivalents in the first argument to runSass.

The sass option is an optional string name of a Dart Sass implementation installed in the current environment (e.g. 'embedded-sass' or 'sass'), or a Dart Sass implementation instance itself. If none is provided, True will attempt to detect which implementation is available, starting with sass-embedded.

If True can't parse the CSS output, it'll give you some context lines of CSS as part of the error message. This context will likely be helpful in understanding the parse failure. By default it provides up to 10 lines of context; if you need more, you can provide a numeric contextLines option: the maximum number of context lines to provide.

The second argument is a string representing either the path to a source Sass file (passed through to Sass' compile function), or a string of source Sass (passed through to Sass' compileString function). By default it is expected to be a path, and sass.compile is used. To pass in a source string (and use sass.compileString), add sourceType: 'string' to your options passed in as the first argument to runSass.

The third (optional) argument to runSass accepts the same options that Sass' compile or compileString expect (e.g. importers, loadPaths, or style), and these are passed directly through to Sass.

By default, True makes two modifications to these options. First, True's sass path is added to the loadPaths option, so @use 'true'; works in your Sass test file. Second, if Dart Sass v1.71 or greater is installed, importers is set to an array containing the Node.js package importer, which supports pkg: imports to resolve @use and @import for external modules installed via npm or Yarn. If importers is set (even as an empty array importers: []), it will override this default importer.

Note: True requires the default Sass 'expanded' output style, and will not work if { style: 'compressed' } is used in the third argument to runSass.

Custom Importers

If you use tilde notation (e.g. @use '~accoutrement/sass/tools') or another method for importing Sass files from node_modules, you'll need to tell runSass how to handle that. That will require writing a custom importer and passing it into the configuration for runSass:

const path = require('node:path');
const { pathToFileURL } = require('node:url');
const sassTrue = require('sass-true');

const importers = [
  {
    findFileUrl(url) {
      if (!url.startsWith('~')) {
        return null;
      }
      return new URL(
        pathToFileURL(path.resolve('node_modules', url.substring(1))),
      );
    },
  },
];

const sassFile = path.join(__dirname, 'test.scss');
sassTrue.runSass({ describe, it }, sassFile, { importers });

At OddBird, we love contributing to the languages & tools developers rely on. We're currently working on polyfills for new Popover & Anchor Positioning functionality, as well as CSS specifications for functions, mixins, and responsive typography. Help us keep this work sustainable and centered on your needs as a developer! We display sponsor logos and avatars on our website.

Sponsor OddBird's OSS Work

changelog

True Changelog

8.1.0 (10/02/24)

  • FEATURE: If True sass option is not specified, True will automatically attempt to use embedded-sass, then sass. #290
  • INTERNAL: Add sass and sass-embedded as optional peer-dependencies.
  • INTERNAL: Update dependencies

8.0.0 (02/23/24)

  • FEATURE: Add True sass option (string or Sass implementation instance, defaults to 'sass') to allow using either sass or embedded-sass.
  • FEATURE: Add the Node.js package importer to the Sass importers option by default, if Dart Sass v1.71 or later is available. Users can opt out by providing their own importers option, e.g. { importers: [] }.
  • BREAKING: Drop support for node < 18
  • INTERNAL: Remove sass as a peer-dependency.
  • INTERNAL: Update dependencies

7.0.1 (01/04/24)

  • FEATURE: Validate runSass arguments and warn if using v6 API.
  • DOCUMENTATION: Add note that { style: 'compressed' } is not supported.
  • DOCUMENTATION: Add note about possible Jest error and workaround.
  • INTERNAL: Update dependencies

7.0.0 (12/14/22)

  • FEATURE: contains() checks multiple block with matching selectors. #243
  • BREAKING: Upgrade to newer Sass API
    • Add True sourceType option (path [default] or string)
    • Reverse order of expected arguments to runSass: 1) True options, 2) source path (or string), 3) optional Sass options
    • Note that some of the Sass options have changed. For example, includePaths is now loadPaths, outputStyle is now style, importer is now importers, etc. See the Dart Sass documentation for more details.
  • BREAKING: Require sass (>=1.45.0) as a peer-dependency, removing True sass option
  • BREAKING: Drop support for node < 14.15.0
  • INTERNAL: Use both Jest and Mocha for internal testing
  • INTERNAL: Remove documentation from npm package
  • INTERNAL: Update dependencies

Migrating from v6

  • runSass arguments have changed:

v6:

const path = require('path');
const sass = require('node-sass');
const sassTrue = require('sass-true');

const sassFile = path.join(__dirname, 'test.scss');
sassTrue.runSass(
  // Sass options [required]
  { file: sassFile, includePaths: ['node_modules'] },
  // True options [required]
  { describe, it, sass },
);

const sassString = `
h1 {
  font-size: 40px;
}`;
sassTrue.runSass(
  // Sass options [required]
  {
    data: sassString,
    includePaths: ['node_modules'],
  },
  // True options [required]
  { describe, it, sass },
);

v7:

const path = require('path');
const sassTrue = require('sass-true');

const sassFile = path.join(__dirname, 'test.scss');
sassTrue.runSass(
  // True options [required]
  { describe, it },
  // Sass source (path) [required]
  sassFile,
  // Sass options [optional]
  { loadPaths: ['node_modules'] },
);

const sassString = `
h1 {
  font-size: 40px;
}`;
sassTrue.runSass(
  // True options [required]
  { describe, it, sourceType: 'string' },
  // Sass source (string) [required]
  sassString,
  // Sass options [optional]
  { loadPaths: ['node_modules'] },
);

7.0.0-beta.0 (09/16/22)

  • BREAKING: Upgrade to newer Sass API
    • Add True sourceType option (path [default] or string)
    • Reverse order of expected arguments to runSass: 1) True options, 2) source path (or string), 3) optional Sass options
  • BREAKING: Require sass as a peer-dependency, removing True sass option
  • BREAKING: Drop support for node < 14.15.0
  • INTERNAL: Use both Jest and Mocha for internal testing
  • INTERNAL: Update dependencies

6.1.0 (03/02/22)

  • No changes since v6.1.0-beta.1

6.1.0-beta.1 (02/24/22)

  • FEATURE: Clearer formatting of failing test diffs #210
  • INTERNAL: Limit files included in npm package #189
  • INTERNAL: Convert JS to TypeScript and bundle type definitions #212 -- thanks to @robertmaier for the initial PR #206
  • INTERNAL: Remove documentation static-site from True repository
  • INTERNAL: Use Jest for internal testing (replaces Mocha)
  • INTERNAL: Switch from Travis CI to GitHub Actions for CI
  • INTERNAL: Update dependencies

6.0.1 (10/16/20)

  • Remove eyeglass specific-version requirement.
  • Update documentation

6.0.0 (07/22/20)

  • BREAKING: Switch to Dart Sass with Sass module system, dropping support for Node Sass.
  • BREAKING: Drop support for node < 10
  • BREAKING: Rename $true-terminal-output setting to $terminal-output when importing as a module (with @use). Projects not using Sass modules can still @import '<path>/sass-true/sass/true' and access the setting as $true-terminal-output
  • FEATURE: Added _index.scss at the project root, for simpler import path: @use '<path>/sass-true'
  • FEATURE: New sass/_throw.scss module provides:
    • error() function & mixin for establishing "catchable" errors
    • global $catch-errors toggles how error() output is handled
  • FEATURE: Support testing content properties which include a curly brace.
  • Update dependencies

5.0.0 (06/03/19)

  • BREAKING: Update API for runSass, which now accepts two arguments: a sassOptions object and a trueOptions object.
  • BREAKING: Drop support for node < 8
  • Add docs and testing for usage with Jest #135
  • Add sass option to runSass for passing a different Sass implementation than node-sass #137
  • Remove node-sass from peerDependencies
  • Fix deprecated use of assert.fail #138
  • Update dev dependencies

4.0.0 (04/09/18)

  • BREAKING: Move node-sass to peerDependencies
  • Update dependencies
  • Add JS coverage reporting

3.1.0 (03/06/18)

  • NEW: Add contains() mixin for more minute output comparisons. Works the same as expect(), but doesn't require a complete match.
  • Update docs

3.0.2 (10/6/17)

  • Dependency updates

3.0.1 (9/13/17)

  • Update docs

3.0.0 (8/26/17)

  • Update dependencies & release

3.0.0-beta.1 (6/1/17)

  • Added describe and it mixins, as alias for test-module and test respectively.
  • Added $inspect argument to assert-equal and assert-unequal mixins, for comparing inspect($assert) == inspect($expected) instead of $assert == $expected. This helps with several of the equality edge-cases listed below (rounding and units).
  • BREAKING: No more Ruby gem or Ruby CLI
  • BREAKING: No more bower package
  • BREAKING: Removes special-handling of equality, in favor of allowing Sass to determine the best comparisons. There are a few edge-cases to be aware of:
    • In some versions of Sass, manipulated numbers and colors are compared without rounding, so 1/3 != 0.333333 and lighten(#246, 15%) != #356a9f. Use the $inspect argument to compare rounded output values.
    • In all versions of Sass, unitless numbers are considered comparable to all units, so 1 == 1x where x represents any unit. Use the $inspect argument to compare output values with units.
    • Lists compare both values and delimiter, so (one two three) != (one, two, three). This can be particularly confusing for single-item lists, which still have a delimiter assigned, even though it is not used.

2.2.2 (4/11/17)

  • assert-true returns false on empty strings and lists
  • assert-false returns true on empty strings and lists
  • Module/Test/Assertion stats are included in reports

2.2.1 (2/7/17)

  • Output CSS context around Mocha parsing errors.
  • Added $fail-on-error argument to report() mixin. Set to true if you need the Sass compiler to fail on broken tests.
  • Fix bug with assert-false causing it to fail on null values.
  • Allow unquoted descriptions and test/module names.
  • Fix bug throwing off test-count and reporting.

2.1.4 (12/22/16)

  • Fix default assertion messages
  • Upgrade dependencies

2.0.2 (5/13/15)

  • Fixes debug inspector.

2.0.1 (5/9/15)

  • Improve internal logic, and namespace private functions behind _true-*.
  • Add assert(), input, and expect mixins for testing CSS output.
  • Support for LibSass.
  • Add Mocha JS integration. — Create NPM package.
  • Simplify output options down to single $true-terminal-output setting.
  • Add eyeglass support.

1.0.1 (10/18/14)

  • LibSass 3.0 compatability.

1.0.0 (10/3/14)

  • Add command-line interface: true-cli <path-to-file>
  • Use -s flag for silent output
  • Check for unit differences between numbers.
  • Add assertion-failure details to css output.

0.2.0 (7/15/14)

  • Simplified reporting in both terminal and CSS.
  • Remove default-module-output, $default-test-output and $default-final-output. Replace them with $true settings map: (output: css, summary: terminal css). output handles test/module output, summary handles final output. Assertions are always output to the terminal if they fail.
  • Update to use Sass map variables.
  • Add report function and report mixin, for reporting final results.
  • Only register as a compass extension if compass is present. Compass is no longer an explicit dependency.
  • Adjust the output styles to work with Sass 3.4 and have more visual consistency.

0.1.5 (6/10/13)

  • Append actual results to custom failure messages.

0.1.4 (6/9/13)

  • Null result is considered a failure.
  • Allow output to be turned off for certain modules/tests/assertions.

0.1.3 (6/7/13)

  • Nest assertions within test() {} named tests.
  • Cleaner css output.

0.1.2 (6/7/13)

  • Use nesting for modules with test-module() {}
  • Added failure message argument to all assertions.

0.1.1 (6/6/13)

  • Fix bug in lib/true.rb compass plugin registration.

0.1.0 (6/6/13)

  • assert-true(), assert-false(), assert-equal(), and assert-unequal().
  • pass() and fail() for tracking and reporting individual results.
  • start-test-module() and report-test-result() for module results.
  • Includes tests of the testing tools!