tldr 0.10.1 → 1.0.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
checksums.yaml CHANGED
@@ -1,7 +1,7 @@
1
1
  ---
2
2
  SHA256:
3
- metadata.gz: 29759fb49b03f109abea0bc89b863a8094b965784e6827772dae2e148b6b38fd
4
- data.tar.gz: f511164ea7cfb2602fe242dbd0cb3cfa36fee656d89a38cede3eb09ae73ffaef
3
+ metadata.gz: e62dd10f12e7e2f2e2a691e0679207f4f3231b916b54b6fc51acbe51cc899918
4
+ data.tar.gz: 2d29248e49b135cc840e3ad249f067ee0a4b706a730889c806c3385f0791f54c
5
5
  SHA512:
6
- metadata.gz: 4a2760b9496cf063b295c08661004e24def945d219b45869ac731ab856b6370138d3bb53549ea7612110bf7ed8f9f565400413ef7b8c1084b2dce96d916aea86
7
- data.tar.gz: 47d2fa8f8c3a32a4ee6eb730d097960d404f9eef5f406b07bf599049e391614f6b095be5222c181bb2d35cbf527f8bae86829edf8758cddccb49c0e4dc438967
6
+ metadata.gz: 3a02db5e6c8c42a2d507a5b6ff99867060fd902bf94ef980546d052c535577db4174a6e3ab05e7146876ca4fc7f992030f40daf425a96c3d82e1439fd27b980c
7
+ data.tar.gz: d580e9a4176c182da622ff826e2d7df64e9ed52f71efe933a22f6dff32d4e8ed60953e5062fccd23f8a2e24db4f0ed26ecaa00144dd5261728fa8b804e8f5c5b
data/CHANGELOG.md CHANGED
@@ -1,3 +1,24 @@
1
+ ## unreleased
2
+
3
+ ## [1.0.0]
4
+
5
+ * **BREAKING** you know how the whole point of TLDR is that it aborts your test
6
+ run after 1.8s? Yeah, well, it doesn't anymore. Use `--timeout` to enable it
7
+ * **BREAKING** replace the `--no-dotfile` flag and has been replaced by a
8
+ `--[no-]config PATH` flag. To skip loading the YAML file, use `--no-config`.
9
+ To set the file, use `--config FILE` option
10
+ * **BREAKING** Remove `assert_include?` and `refute_include?` in favor of
11
+ Minitest-compatible `assert_includes` and `refute_includes`.
12
+ * **BREAKING** Rename `TLDR::Assertions::MinitestCompatibility` to `TLDR::MinitestCompatibility` and remove `assert_send`, which [nobody uses](https://github.com/minitest/minitest/issues/668)
13
+ * **BREAKING** Replace `no_emoji` YAML option with `emoji` option. Disable emoji output by default. Add `--emoji` flag for enabling it.
14
+ * Add `--[no-]timeout TIMEOUT` flag and `timeout` YAML option. To enable the
15
+ TLDR Classic™ default of 1.8s, specify `--timeout` from the CLI or `timeout: true`
16
+ in YAML. To specify a custom timeout of 42.3 seconds, flag `--timeout 42.3` or
17
+ `timeout: 42.3` in YAML
18
+ * Add `require "tldr/autorun"`, which adds an `at_exit` hook so that tests can be
19
+ run from the command line (still supports CLI args and YAML options) by running `ruby path/to/test.rb` (see [its test](/tests/autorun_test.rb))
20
+ * Fix custom reporters by looking them up only after helpers have a chance to run. [#15](https://github.com/tendersearls/tldr/issues/15)
21
+
1
22
  ## [0.10.1]
2
23
 
3
24
  * Fix 3.4 / Prism [#17](https://github.com/tendersearls/tldr/pull/17)
data/README.md CHANGED
@@ -1,114 +1,51 @@
1
- # TLDR - for people who don't have time for slow tests
2
-
3
- Okay, you might need to sit down for this:
4
-
5
- **tl;dr, TLDR is a Ruby test framework that stops running your tests after 1.8 seconds.**
6
-
7
- We initially meant this as a joke [while
8
- pairin'](https://www.youtube.com/live/bmi-SWeH4MA?si=p5g1j1FQZrbYEOCg&t=63), but
9
- in addition to being funny, it was also a pretty good idea. So we fleshed out
10
- `tldr` to be a full-featured, mostly
11
- [Minitest-compatible](#minitest-compatibility), and dare-we-say pleasant test
12
- framework for Ruby.
13
-
14
- The "big idea" here is TLDR is designed for users to run the `tldr` command
15
- repeatedly as they work—as opposed to only running the tests for whatever is
16
- being worked on. Even if the suite runs over the 1.8 second time limit. Because
17
- TLDR shuffles and runs in parallel and is guaranteed to take less than two
18
- seconds,
19
- **you'll actually wind up running _all_ of your tests quite often as you work**,
20
- catching any problems much earlier than if you had waited until the end of the
21
- day to push your work and let a continuous integration server run the full
22
- suite.
23
-
24
- Some stuff you might like:
25
-
26
- * A CLI that can specify tests by line number(s) (e.g. `foo.rb:5 bar.rb:3:10`)
27
- and by names or patterns (e.g. `--name test_fail,test_error --name "/_\d/"`)
28
- * Everything is **parallel by default**, and seems pretty darn fast; TLDR
29
- also provides [several escape hatches to sequester tests that aren't thread-safe](#parallel-by-default-is-nice-in-theory-but-half-my-tests-are-failing-wat)
30
- * Surprisingly delightful color diff output when two things fail to equal one
31
- another, care of [@mcmire's super_diff gem](https://github.com/mcmire/super_diff)
32
- * By default, the CLI will prepend your most-recently-edited test file to the
33
- front of your suite so its tests will run first. The test you worked on most recently
34
- is the one you most likely want to ensure runs, so TLDR runs it first (see the
35
- `--prepend` option for how to control this behavior)
36
- * And, of course, our signature feature: your test suite will never grow into
37
- a glacially slow, soul-sucking albatross around your neck, because **after 1.8
38
- seconds, it stops running your tests**, with a report on what it _was_ able to
39
- run and where your slowest tests are
40
-
41
- Some stuff you might _not_ like:
42
-
43
- * The thought of switching Ruby test frameworks in 2023
44
- * That bit about your test suite exploding after 1.8 seconds
45
-
46
- ## Install
47
-
48
- Either `gem install tldr` or add it to your Gemfile:
1
+ # TLDR - Ruby tests for people who prioritize fast feedback
49
2
 
50
- ```
51
- gem "tldr"
52
- ```
3
+ TLDR is a suspiciously-delightful testing framework for Ruby.
53
4
 
54
- ## Usage
5
+ As a test library, TLDR is largely [API-compatible with Minitest](#minitest-compatibility). As a test runner, TLDR boasts a few features RSpec's CLI still doesn't have.
55
6
 
56
- Here's what a test looks like:
7
+ The library, command line interface, and every decision in-between was prioritized to maximize productivity by promote fast feedback loops. Some highlights:
57
8
 
58
- ```ruby
59
- class MathTest < TLDR
60
- def test_adding
61
- assert_equal 1 + 1, 2
62
- end
63
- end
64
- ```
9
+ * Numerous ways to run specific tests: by path (`foo_test.rb`), line number (`foo_test.rb:13`), name (`--name test_foo`), or regex pattern (`-n /_foo_/`)
10
+ * Parallel test execution by default, as well as [controls for serial execution of thread-unsafe tests](#parallel-by-default-is-nice-in-theory-but-half-my-tests-are-failing-wat)
11
+ * Continuously run [after every file change](#running-tests-continuously-with---watch) with `--watch`
12
+ * An optional timer to [enforce your tests never get slow](#enforcing-a-testing---timeout) with `--timeout`
13
+ * A `--fail-fast` flag that aborts the run as soon as a failure is encountered
14
+ * Running your most-recently-edited test before all the others (see `--prepend`)
15
+ * Delightful diffs when assertions fail, care of [super_diff](https://github.com/splitwise/super_diff)
65
16
 
66
- A TLDR subclass defines its tests with instance methods that begin with
67
- `test_`. They can define `setup` and/or `teardown` methods which will run before
68
- and after each test, respectively.
17
+ We hope you'll give it a try!
69
18
 
70
- If you place your tests in `test/**/*_test.rb` (and/or `test/**/test_*.rb`)
71
- files, the `tldr` executable will find them automatically. And if you define a
72
- `test/helper.rb` file, it will be loaded prior to your tests.
19
+ ## Getting started
73
20
 
74
- Running the CLI is pretty straightforward:
21
+ You can either read the rest of this README and learn about TLDR passively, or you can just clone this [TLDR demo repo](https://github/searls/tldr_demo) and work through its README as you play with its tests and run them with various options.
75
22
 
76
- ```
77
- $ tldr
78
- ```
79
-
80
- You can, of course, also just run a specific test file or glob:
23
+ **JUST IN CASE YOU'RE ALREADY SKIMMING THIS**, I said stop reading and [clone this interactive repo](https://github/searls/tldr_demo) if you're a hands-on learner.
81
24
 
82
- ```
83
- $ tldr test/this/one/in/particular.rb
84
- ```
25
+ ### Install
85
26
 
86
- Or specify the line numbers of tests to run by appending them after a `:`
27
+ You know the drill. `gem install tldr` or add it to your Gemfile:
87
28
 
88
29
  ```
89
- $ tldr test/fixture/line_number.rb:3:10
30
+ gem "tldr"
90
31
  ```
91
32
 
92
- And filter which tests run by name or pattern with one or more `--name` or `-n`
93
- flags:
33
+ ### Project setup
94
34
 
95
- ```
96
- $ tldr --name FooTest#test_foo -n test_bar,test_baz -n /_qux/
97
- ```
35
+ By default, TLDR expects your tests to be in `test/` with filenames that match `test_*.rb` or `*_test.rb` and will require a `test/helper.rb` if you define one.
98
36
 
99
- (The above will translate to this array of name filters internally:
100
- `["FooTest#test_foo", "test_bar", "test_baz", "/_qux/"]`.)
37
+ ## Configuring TLDR
101
38
 
102
- ### Options
103
-
104
- Here is the full list of CLI options:
39
+ ### CLI Options
105
40
 
106
41
  ```
107
42
  $ tldr --help
108
43
  Usage: tldr [options] some_tests/**/*.rb some/path.rb:13 ...
44
+ -t, --[no-]timeout [TIMEOUT] Timeout (in seconds) before timer aborts the run (Default: 1.8)
45
+ --watch Run your tests continuously on file save (requires 'fswatch' to be installed)
109
46
  --fail-fast Stop running tests as soon as one fails
110
- -s, --seed SEED Seed for randomization
111
47
  --[no-]parallel Parallelize tests (Default: true)
48
+ -s, --seed SEED Random seed for test order (setting --seed disables parallelization by default)
112
49
  -n, --name PATTERN One or more names or /patterns/ of tests to run (like: foo_test, /test_foo.*/, Foo#foo_test)
113
50
  --exclude-name PATTERN One or more names or /patterns/ NOT to run
114
51
  --exclude-path PATH One or more paths NOT to run (like: foo.rb, "test/bar/**", baz.rb:3)
@@ -117,96 +54,93 @@ Usage: tldr [options] some_tests/**/*.rb some/path.rb:13 ...
117
54
  --prepend PATH Prepend one or more paths to run before the rest (Default: most recently modified test)
118
55
  --no-prepend Don't prepend any tests before the rest of the suite
119
56
  -l, --load-path PATH Add one or more paths to the $LOAD_PATH (Default: ["lib", "test"])
120
- -r, --reporter REPORTER Set a custom reporter class (Default: "TLDR::Reporters::Default")
121
57
  --base-path PATH Change the working directory for all relative paths (Default: current working directory)
122
- --no-dotfile Disable loading .tldr.yml dotfile
123
- --no-emoji Disable emoji in the output
58
+ -c, --[no-]config PATH The YAML configuration file to load (Default: '.tldr.yml')
59
+ -r, --reporter REPORTER Set a custom reporter class (Default: "TLDR::Reporters::Default")
60
+ --[no-]emoji Enable emoji output for the default reporter (Default: false)
61
+ --[no-]warnings Print Ruby warnings (Default: true)
124
62
  -v, --verbose Print stack traces for errors
63
+ --yes-i-know Suppress TLDR report when suite runs beyond any configured --timeout
125
64
  --print-interrupted-test-backtraces
126
- Print stack traces for interrupted tests
127
- --[no-]warnings Print Ruby warnings (Default: true)
128
- --watch Run your tests continuously on file save (requires 'fswatch' to be installed)
129
- --yes-i-know Suppress TLDR report when suite runs over 1.8s
130
- --i-am-being-watched [INTERNAL] Signals to tldr it is being invoked under --watch mode
131
- --comment COMMENT [INTERNAL] No-op; used for multi-line execution instructions
65
+ Print stack traces of tests interrupted after a timeout
132
66
  ```
133
67
 
134
- After being parsed, all the CLI options are converted into a
135
- [TLDR::Config](/lib/tldr/value/config.rb) object.
136
-
137
68
  ### Setting defaults in .tldr.yml
138
69
 
139
- The `tldr` CLI will look for a `.tldr.yml` file in your project root (your
140
- working directory or whatever `--base-path` you set), which can contain values
141
- for any properties on [TLDR::Config](/lib/tldr/value/config.rb) (with the
142
- exception of `--base-path` itself).
70
+ The `tldr` CLI will look for a `.tldr.yml` that can set the same set of options supported by the CLI file in the root of your project. You can specify a custom YAML location with `--config some/path.yml` if you need to.
143
71
 
144
- Any values found in the dotfile will override TLDR's built-in values, but can
145
- still be specified by the `tldr` CLI or a `TLDR::Config` object passed to
146
- [TLDR::Run.at_exit!](#running-tests-without-the-cli).
72
+ Any options found in the dotfile will override TLDR's defaults, but can still be overridden by the `tldr` CLI or a `TLDR::Config` object passed to [TLDR::Run.at_exit!](#running-tests-without-the-cli).
147
73
 
148
- Here's an [example project](/example/c) that specifies a `.tldr.yml` file as
149
- well as some [internal tests](/tests/dotfile_test.rb) demonstrating its behavior.
74
+ Here's an [example project](/example/c) that specifies a `.tldr.yml` file as well as some [internal tests](/tests/dotfile_test.rb) demonstrating its behavior.
150
75
 
151
- ### Minitest compatibility
76
+ ## Writing your tests
152
77
 
153
- Tests you write with tldr are designed to be mostly-compatible with
154
- [Minitest](https://github.com/minitest/minitest) tests. Some notes:
155
-
156
- * `setup` and `teardown` hook methods should work as you expect. (We even threw
157
- in [an `around` hook](https://github.com/splattael/minitest-around) as a bonus!)
158
- * All of Minitest's assertions (e.g. `assert`, `assert_equals`) are provided,
159
- with these caveats:
160
- * To retain the `expected, actual` argument ordering, `tldr` defines
161
- `assert_include?(element, container)` instead of
162
- `assert_includes(container, element)`
163
- * If you want to maximize compatibility and mix in `assert_includes` and the
164
- deprecated `assert_send`, just `include
165
- TLDR::Assertions::MinitestCompatibility` into the `TLDR` base class or
166
- individual test classesJust set it
78
+ If you've ever seen a [Minitest](https://github.com/minitest/minitest?tab=readme-ov-file#synopsis-) test, then you already know how to write TLDR tests. Rather than document how to write tests, this section just highlights the ways TLDR tests differ from Minitest tests.
167
79
 
168
- ### Running tests continuously with --watch
80
+ First, instead of inheriting from `Minitest::Test`, your test classes must subclass `TLDR` instead:
169
81
 
170
- The `tldr` CLI includes a `--watch` option which will watch for changes in any
171
- of the configured load paths (`["test", "lib"]` by default) and then execute
172
- your tests each time a file is changed. To keep the output up-to-date and easy
173
- to scan, it will also clear your console before each run.
82
+ ```ruby
83
+ class MyTest < TLDR
84
+ def test_looks_familiar
85
+ assert true
86
+ end
87
+ end
88
+ ```
174
89
 
175
- Note that this feature requires you have
176
- [fswatch](https://github.com/emcrisostomo/fswatch) installed and on your `PATH`
90
+ Second, if your tests depend on a test helper, so long as you name it `test/helper.rb`, it will automatically be required by TLDR, so you don't need to add `require "helper"` at the top of each test. If you want to name the helper something else, you can do so with the `--helper` option:
177
91
 
178
- Here's what that might look like:
92
+ ```
93
+ tldr --helper test/test_helper.rb
94
+ ```
179
95
 
180
- ![tldr-watch](https://github.com/tendersearls/tldr/assets/79303/364f0e52-5596-49ce-a470-5eaeddd11f03)
96
+ Third, TLDR offers fewer features:
181
97
 
182
- ### Running TLDR with Rake
98
+ * No built-in mock library ([use mocktail](https://justin.searls.co/posts/a-real-world-mocktail-example-test/), maybe!)
99
+ * No "spec" API
100
+ * No benchmark tool
101
+ * No test bisect script
183
102
 
184
- TLDR ships with a [very](lib/tldr/rake.rb) minimal rake task that simply shells
185
- out to the `tldr` CLI. If you want to run TLDR with Rake, you can configure
186
- the test run by setting flags on an env var named `TLDR_OPTS` or else in
187
- the [.tldr.yml](#setting-defaults-in-tldryml).
103
+ And that's it! You now know how to write TLDR tests.
188
104
 
189
- Here's an example Rakefile:
105
+ ## Running your tests
190
106
 
191
- ```ruby
192
- require "standard/rake"
193
- require "tldr/rake"
107
+ Because TLDR ships with a CLI, it offers a _lot_ of ways to run your tests.
108
+
109
+ ### Running your tests
110
+
111
+ Once installed, running all your tests is just five keystrokes away:
194
112
 
195
- task default: [:tldr, "standard:fix"]
196
113
  ```
114
+ tldr
115
+ ```
116
+
117
+ This assumes your tests are stored in `test/`. It will also add `lib/` to Ruby's load paths and require `test/helper.rb` before your tests, if it exists.
118
+
119
+ ### Running TLDR with Rake
120
+
121
+ TLDR ships with a minimal [rake task](lib/tldr/rake.rb) that simply shells out to the `tldr` CLI by default. If you want to run TLDR with Rake, you can configure the task by setting flags on an env var named `TLDR_OPTS` or in a [.tldr.yml file](#setting-defaults-in-tldryml).
197
122
 
198
- You could then run the task with:
123
+ All your Rakefile needs is `require "tldr/rake` and you can run the task individually like this:
124
+
125
+ ```
126
+ $ rake tldr
199
127
 
128
+ # Or, with options in TLDR_OPTS
129
+ $ TLDR_OPTS="--no-parallel" rake tldr
200
130
  ```
201
- $ TLDR_OPTS="--no-parallel" bundle exec rake tldr
131
+
132
+ Here's an example Rakefile that runs both TLDR and [Standard Ruby](https://github.com/standardrb/standard) as the default task:
133
+
134
+ ```ruby
135
+ require "standard/rake"
136
+ require "tldr/rake"
137
+
138
+ task default: ["tldr", "standard:fix"]
202
139
  ```
203
140
 
204
- One reason you'd want to invoke TLDR with Rake is because you have multiple
205
- test suites that you want to be able to conveniently run separately ([this
206
- talk](https://blog.testdouble.com/talks/2014-05-25-breaking-up-with-your-test-suite/)
207
- discussed a few reasons why this can be useful).
141
+ One reason you'd want to invoke TLDR with Rake is because you have multiple test suites that you want to be able to conveniently run separately ([this talk](https://blog.testdouble.com/talks/2014-05-25-breaking-up-with-your-test-suite/) discussed a few reasons why this can be useful).
208
142
 
209
- To create a custom TLDR Rake test, just instantiate `TLDR::Task` like this:
143
+ To create a custom TLDR Rake task, you can instantiate `TLDR::Task` like this, which allows you to configure [TLDR::Config](/lib/tldr/value/config.rb) in code:
210
144
 
211
145
  ```ruby
212
146
  require "tldr/rake"
@@ -218,20 +152,27 @@ TLDR::Task.new(name: :safe_tests, config: TLDR::Config.new(
218
152
  ))
219
153
  ```
220
154
 
221
- The above will create a second Rake task named `safe_tests` running a different
222
- set of tests than the default `tldr` task. Here's [an
223
- example](/example/b/Rakefile).
155
+ The above will create a second Rake task named `safe_tests` running a different set of tests than the default `tldr` task. Here's [an example](/example/b/Rakefile) from TLDR's test suite.
156
+
157
+ ### Running tests continuously with --watch
158
+
159
+ The `tldr` CLI includes a `--watch` option that will watch for changes in any of the configured load paths (`["test", "lib"]` by default) and then execute your tests each time a file is changed. To keep the output up-to-date and easy
160
+ to scan, it will also clear your console before each run.
224
161
 
225
- ### Running tests without the CLI
162
+ Note that this feature requires you have [fswatch](https://github.com/emcrisostomo/fswatch) installed and on your `PATH`
226
163
 
227
- If you'd rather use TLDR by running Ruby files instead of the `tldr` CLI
228
- (similar to `require "minitest/autorun"`), here's how to do it!
164
+ Here's what that might look like with the `--emoji` flag enabled:
165
+
166
+ ![tldr-watch](https://github.com/tendersearls/tldr/assets/79303/364f0e52-5596-49ce-a470-5eaeddd11f03)
167
+
168
+ ### Running tests programmatically
169
+
170
+ If you'd rather use TLDR by running Ruby files instead of the `tldr` CLI, you can simply `require "tldr/autorun"` (just like `require "minitest/autorun"`).
229
171
 
230
172
  Given a file `test/some_test.rb`:
231
173
 
232
174
  ```ruby
233
- require "tldr"
234
- TLDR::Run.at_exit! TLDR::Config.new(no_emoji: true)
175
+ require "tldr/autorun"
235
176
 
236
177
  class SomeTest < TLDR
237
178
  def test_truth
@@ -240,112 +181,137 @@ class SomeTest < TLDR
240
181
  end
241
182
  ```
242
183
 
243
- You could run the test with:
184
+ You can then run the test by passing `ruby` the file:
244
185
 
245
186
  ```
246
187
  $ ruby test/some_test.rb
247
188
  ```
248
189
 
249
- To maximize control and to avoid running code accidentally (and _unlike_ the
250
- `tldr` CLI), running `at_exit!` will not set default values to the `paths`,
251
- `helper`, `load_paths`, and `prepend_paths` config properties. You'll have to
252
- pass any values you want to set on a [Config object](/lib/tldr/value/config.rb)
253
- and pass it to `at_exit!`.
190
+ Any CLI options you add will still be parsed, as well (e.g. `ruby my_test.rb --emoji` will work).
191
+
192
+ If you want to be explicit about setting the `Kernel.at_exit` hook, or if you want to configure TLDR with code, you can invoke `TLDR.at_exit!` directly:
193
+
194
+ ```ruby
195
+ require "tldr"
196
+ TLDR::Run.at_exit! TLDR::Config.new(emoji: true)
197
+ ```
254
198
 
255
- To avoid running multiple suites accidentally, if `TLDR::Run.at_exit!` is
256
- encountered multiple times, only the first hook will be registered. If the
257
- `tldr` CLI is running and encounters a call to `at_exit!`, it will be ignored.
199
+ ## Failing with style
258
200
 
259
- #### Setting up the load path
201
+ ### Failing fast and first
260
202
 
261
- By default, the `tldr` CLI adds `test` and `lib` directories to the load path
262
- for you, but when running TLDR from a Ruby script, it doesn't set those up for
263
- you.
203
+ If we just want to know if the build passes, we want to know as fast as possible. Ever see a test fail and then sit around waiting for the whole suite to finish running anyway? Why wait? Turn on `--fail-fast` and abort the test run the instant a failure is encountered:
264
204
 
265
- If you want to require code in `test/` or `lib/` without using
266
- `require_relative`, you'll need to add those directories to the load path. You
267
- can do this programmatically by prepending the path to `$LOAD_PATH`, like
268
- this:
205
+ ```
206
+ tldr --fail-fast
207
+ ```
269
208
 
270
- ```ruby
271
- $LOAD_PATH.unshift "test"
209
+ Additionally, you might notice the top of each run will show you a command you can use to execute the same run, like this:
272
210
 
273
- require "tldr"
274
- TLDR::Run.at_exit! TLDR::Config.new(no_emoji: true)
211
+ ```
212
+ Command: bundle exec tldr --fail-fast --prepend "test/calculator_test.rb"
213
+ ```
214
+
215
+ That's because TLDR will look at the file system and move your most-recently-edited test file to the front of the queue with `--prepend`. When used in conjunction with `--fail-fast`, you'll fail _extra fast_, because the most likely test to fail is the one you're actively working on.
216
+
217
+ ### Enforcing a testing --timeout
218
+
219
+ We initially developed TLDR because we wanted a test runner that supported suite-wide time limits as a first-class feature. When test suites become slow, people run them (_much_) less often. And once a developer gets in the habit of only running tests occasionally, it's not long before they only run them before push, and then only the ones they're immediately working on, and then they just wait for CI to run them in a pull request. And if you don't run all your tests very often, you don't feel any pain when you make your tests (or the code its testing) slower.
220
+
221
+ Each time you write code and don't run your tests, you're making an assumption that whatever code you just wrote _works flawlessly_. If that assumption is correct, you saved however much time it takes to run your tests. But every time that assumption is incorrect, you've just extended the amount of time before discovering that you broke something. If that's a few minutes later, that might only cost a few minutes of rework. **If you only run tests once or twice a day, you might have to undo hours of work to fix it.**
222
+
223
+ We originally came up with the idea of TLDR [on a livestream](https://www.youtube.com/live/bmi-SWeH4MA?si=p5g1j1FQZrbYEOCg&t=63), joking that only an unconfigurable 1.8 second time limit would prevent test suites from ballooning in duration over time. As of 1.0, we've made the timeout configurable, but we still think it's a good idea to enable it with the `--timeout` option:
224
+
225
+ ```
226
+ tldr --timeout
227
+ ```
228
+
229
+ Not only does a timeout keep us running the whole suite frequently (it'll never take more than 1.8 seconds, after all), but even if the suite begins to exceed our self-imposed timeout, TLDR's random test order and parallel execution means that—so long as you keep running that partial suite frequently—you'll still be running ALL your tests _many more times_ than if they waited until some arbitrary checkpoint to run them.
230
+
231
+ When enabled, `--timeout` will set the timer to 1.8 seconds. But you can set whatever time limit you like. The right value is depends on your individual capacity for paying attention. Basically, "however long you're willing to wait without caving and running your tests less often."
232
+
233
+ Examples:
234
+
235
+ ```
236
+ # A badass 200ms timeout
237
+ tldr --timeout 0.2
275
238
 
276
- require "helper"
239
+ # A miserable-sounding 20 second timeout
240
+ tldr --timeout 20
277
241
  ```
278
242
 
279
- Or by using Ruby's `-I` flag to include it:
243
+ If we've won you over towards this way of working, we suggest creating a [.tldr.yml file](#setting-defaults-in-tldryml) in the root of your project and specifying your desired timeout.
244
+
245
+ For TLDR Classic™ and a 1.8s timeout:
280
246
 
247
+ ```yaml
248
+ # .tldr.yml
249
+ timeout: true
281
250
  ```
282
- $ ruby -Itest test/some_test.rb
251
+
252
+ Or any number of seconds you like:
253
+
254
+ ```yaml
255
+ # .tldr.yml
256
+ timeout: 0.01
283
257
  ```
284
258
 
259
+ And if you're running with the timeout enabled this way, you can still disable it for any given test run by adding the `--no-timeout` flag.
260
+
285
261
  ## Questions you might be asking
286
262
 
287
263
  TLDR is very similar to Minitest in API, but different in enough ways that you
288
264
  probably have some questions.
289
265
 
266
+ ## Wait, isn't this the one that blows up after 1.8 seconds?
267
+
268
+ The `tldr` gem was initially developed and released after we did a [lighthearted pairing session]() and imagined a Ruby test runner with a CLI and an unorthodox, unconfigurable rule: a hard-and-fast 1.8 second time limit on every test suite.
269
+
270
+ In the 18 months since, and to our utter surprise, TLDR did not immediately rise to the top of the charts and dominate the Ruby testing world.
271
+
272
+ While we still contend the mandatory time limit was a Very Good Idea (even if your tests are necessarily slower than 1.8 seconds), we believe there is a very remote, almost certainly wrong possibility that it was slowing adoption of this otherwise very capable test runner. As a result, **as of 1.0.0, the 1.8s timeout is disabled by default, and can be re-enabled (and even set to a specific value) with the `--timeout` option.
273
+
274
+ ### Minitest compatibility
275
+
276
+ Tests you write with tldr are designed to be mostly-compatible with [Minitest](https://github.com/minitest/minitest) tests.
277
+
278
+ Details:
279
+
280
+ * We [implemented all](/lib/tldr/assertions.rb) of Minitest's built-in assertions (e.g. `assert`, `assert_equals`)
281
+ * `setup` and `teardown` hook methods should work as you expect. (We even threw in [our own `around` hook](https://github.com/search?q=repo%3Atendersearls/tldr%20%3Aaround&type=code), free of charge!)
282
+ * If you need anything else from Minitest as you port tests to TLDR, try `include TLDR::MinitestCompatibility`, and if that doesn't do the trick [add whatever you need in a pull request](/lib/tldr/minitest_compatibility.rb)
283
+
290
284
  ### Parallel-by-default is nice in theory but half my tests are failing. Wat?
291
285
 
292
- **Read this before you add `--no-parallel` because some tests are failing when
293
- you run `tldr`.**
294
-
295
- The vast majority of test suites in the wild are not parallelized and the vast
296
- majority of _those_ will only parallelize by forking processes as opposed to
297
- using a thread pool. We wanted to encourage more people to save time (after all,
298
- you only get 1.8 seconds here) by making your test suite run as fast as it can,
299
- so your tests run in parallel threads by default.
300
-
301
- If you're writing new code and tests with TLDR and dutifully running `tldr`
302
- constantly for fast feedback, odds are that this will help you catch thread
303
- safety issues early—this is a good thing, because it gives you a chance to
304
- address them before they're too hard to fix! But maybe you're porting an
305
- existing test suite to TLDR and running in parallel for the first time, or maybe
306
- you need to test something that simply _can't_ be exercised in a thread-safe
307
- way. For those cases, TLDR's goal is to give you some tools to prevent you from
308
- giving up and adding `--no-parallel` to your entire test suite and **slowing
309
- everything down for the sake of a few tests**.
310
-
311
- So, when you see a test that is failing when run in parallel with the rest of your
312
- suite, here is what we recommend doing, in priority order:
313
-
314
- 1. Figure out a way to redesign the test (or the code under test) to be
315
- thread-safe. Modern versions of Ruby provide a number of tools to make this
316
- easier than it used to be, and it may be as simple as making an instance
317
- variable thread-local
318
- 2. If the problem is that a subset of your tests depend on the same resource,
319
- try using [TLDR.run_these_together!](lib/tldr/parallel_controls.rb) class to
320
- group the tests together. This will ensure that those tests run in the same
321
- thread in sequence (here's a [simple
322
- example](/tests/fixture/run_these_together.rb))
323
- 3. For tests that affect process-wide resources like setting the system clock or
324
- changing the process's working directory (i.e. `Dir.chdir`), you can sequester
325
- them to run sequentially _after_ all parallel tests in your suite have run with
326
- [TLDR.dont_run_these_in_parallel!](lib/tldr/parallel_controls.rb), which takes
327
- the same arguments as `run_these_together!`
328
- ([example](/tests/fixture/dont_run_these_in_parallel.rb))
329
- 4. Give up and make the whole suite `--no-parallel`. If you find that you need
330
- to resort to this, you might save some keystrokes by adding `parallel: false` in
331
- a [.tldr.yml](#setting-defaults-in-tldryml) file
332
-
333
- We have a couple other ideas of ways to incorporate non-thread-safe tests into
334
- your suite without slowing down the rest of your tests, so stay tuned!
335
-
336
- ### How will I run all my tests in CI without the time bomb going off?
337
-
338
- TLDR will run all your tests in CI without the time bomb going off. If
339
- `tldr` is run in a non-interactive shell and a `CI` environment variable is set
340
- (as it is on virtually every CI service), then the bomb will be defused.
286
+ **Read this before you add `--no-parallel` because some tests are failing when you run `tldr`.**
341
287
 
342
- ### What if I already have another `tldr` executable on my path?
288
+ The vast majority of test suites in the wild are run sequentially even though they'd work perfectly fine in parallel. Why? Because test runners tend not to enable it by default. Moreover, when they do, they usually rely on forking processes, which is slower and more resource-intensive than using threads or [Ractors](https://docs.ruby-lang.org/en/3.4/ractor_md.html). For this reason, TLDR is optimistic by default and will multi-thread your test suite.
289
+
290
+ So, what do you do when you run into a situation where a test has good reason not to run in parallel? Either because of resource contention or because the order of its test cases actually matters? Here's what we'd do:
291
+
292
+ 1. Start by challenging the assumption that the test can't be run in a thread-safe way (it may be as simple as changing a globally-edited instance variable to a [thread-local](https://docs.ruby-lang.org/en/master/Thread.html#class-Thread-label-Fiber-local+vs.+Thread-local))
293
+ 2. If the problem is that a subset of your tests depend on the same resource, try using [TLDR.run_these_together!](/lib/tldr/parallel_controls.rb) class to group the tests together. This will ensure that those tests run in the same thread in sequence (here's a [simple example](/tests/fixture/run_these_together.rb))
294
+ 3. For tests that affect process-wide resources like setting the system clock or changing the process's working directory (i.e. `Dir.chdir`), you can sequester them to run sequentially _after_ all parallel tests in your suite have run with [TLDR.dont_run_these_in_parallel!](lib/tldr/parallel_controls.rb), which takes the same arguments as `run_these_together!` ([example](/tests/fixture/dont_run_these_in_parallel.rb))
295
+ 4. Give up and make the whole suite `--no-parallel`. If you find that you need to resort to this, you might save some keystrokes by adding `parallel: false` in a [.tldr.yml](#setting-defaults-in-tldryml) file
296
+
297
+ ### Any help porting from Minitest?
298
+
299
+ If you're currently using Minitest, you can take a stab at dropping your dependency on the minitest gem and replace references to `Minitest::Test` dynamically in a test helper, like I did for my [todo_or_die gem](https://github.com/searls/todo_or_die/blob/b50beb0166307d901393435594508f5142976c93/test/helper.rb#L7-L16):
300
+
301
+ ```
302
+ require "tldr"
303
+ if defined?(Minitest::Test)
304
+ TLDR::MinitestTestBackup = Minitest::Test
305
+ Minitest.send(:remove_const, "Test")
306
+ end
307
+ module Minitest
308
+ class Test < TLDR
309
+ include TLDR::MinitestCompatibility
310
+ end
311
+ end
312
+ ```
343
313
 
344
- There's a [command-line utility named tldr](https://tldr.sh) that might conflict
345
- with this gem's binary in your PATH. If that's the case you could change your
346
- path, invoke `bundle exec tldr`, run [with Rake](#running-tldr-with-rake), or
347
- use the `tldt` ("too long; didn't test") executable alias that ships with this
348
- gem.
314
+ This probably won't work for complex projects, but it might for simple ones!
349
315
 
350
316
  ### Is there a plugin system?
351
317
 
@@ -353,21 +319,14 @@ There is not.
353
319
 
354
320
  Currently, the only pluggable aspect of TLDR are reporters, which can be set
355
321
  with the `--reporter` command line option. It can be set to any fully-qualified
356
- class name that extends from
322
+ class name that responds to the same methods defined in
357
323
  [TLDR::Reporters::Base](/lib/tldr/reporters/base.rb).
358
324
 
359
- ### I know my tests are over 1.8s, how do I suppress the huge output?
360
-
361
- Plenty of test suites are over 1.8s and having TLDR repeatedly print out the
362
- huge summary at the end of each test run can be distracting and make it harder
363
- to spot test failures. If you know your test suite is too slow, you can simply
364
- add the `--yes-i-know` flag
325
+ If you define a custom reporter, be sure to require it from your test helper, so TLDR can instantiate it!
365
326
 
366
- ### What about mocking?
327
+ ### What if I already have another `tldr` executable on my path?
367
328
 
368
- TLDR is laser-focused on running tests, so it doesn't provide a built-in mocking
369
- facility. Might we interest you in a refreshing
370
- [mocktail](https://github.com/testdouble/mocktail), instead?
329
+ There's a [command-line utility named tldr](https://tldr.sh) that might conflict with this gem's executable in your PATH. If that's the case, you could either change your path, invoke `bundle exec tldr`, run [with Rake](#running-tldr-with-rake), or use the `tldt` ("too long; didn't test") executable alias that ships with this gem.
371
330
 
372
331
  ## Contributing to TLDR
373
332
 
data/exe/tldr CHANGED
@@ -1,5 +1,7 @@
1
1
  #!/usr/bin/env ruby
2
2
 
3
+ $LOAD_PATH.unshift("#{__dir__}/../lib")
4
+
3
5
  require "tldr"
4
6
 
5
7
  TLDR::Run.cli(ARGV)
data/exe/tldt CHANGED
@@ -1,5 +1,7 @@
1
1
  #!/usr/bin/env ruby
2
2
 
3
+ $LOAD_PATH.unshift("#{__dir__}/../lib")
4
+
3
5
  require "tldr"
4
6
 
5
7
  TLDR::Run.cli(ARGV)