Implemented Watch mode filter by filename and by test name #1530 #3372
Add this suggestion to a batch that can be applied as a single commit.
This suggestion is invalid because no changes were made to the code.
Suggestions cannot be applied while the pull request is closed.
Suggestions cannot be applied while viewing a subset of changes.
Only one suggestion per line can be applied in a batch.
Add this suggestion to a batch that can be applied as a single commit.
Applying suggestions on deleted lines is not supported.
You must change the existing code in this line in order to create a valid suggestion.
Outdated suggestions cannot be applied.
This suggestion has been applied or marked resolved.
Suggestions cannot be applied from pending reviews.
Suggestions cannot be applied on multi-line comments.
Suggestions cannot be applied while the pull request is queued to merge.
Suggestion cannot be applied right now. Please check back later.
Hi!
This pull request implements
Watch mode filter by filename and by test name
from issue #1530Watch this short video for a demonstration of the feature:
Watch_Mode_Demo.mov
Overview
Summary of new feature
(taken from the docs)
Filter tests while watching
You may also filter tests while watching by using the cli. For example, after running
$ npx ava --watch
You will see a prompt like this :
So, to run only tests numbered like
You can type
t
and press enter, then typefoo\d+
and press enter.This will then run all tests that match that pattern.
Afterwards you can use the
r
command to run the matched tests again,or
a
command to run all tests.Code Overview
In the contributing guidelines, you say that you prefer shorter pull requests, but this is a bit long. So, to help speed things along, I'm going to provide an overview for each part that I added or changed.
I add a new variable called watchModeSkipTests
https://github.com/mmulet/ava/blob/78a3768fbe3360930351374f854f9cc7a46fda51/lib/watcher.js#L104
which is an instance of WatchModeSkipTests lib/watch-mode-skip-tests.js, this class handles the regex for the skipping, and most of the code will be passing this object around, populating its fields, etc.
There is one particular part of this file that I want to draw attention to.
#OnDataGenerator
Continue down the
plan
function until you come to the const _class;This class handles the prompt for commands at watch mode. It does this by listening to stdin's data event in an asynchronous generator.
I switched to using an asynchronous generator rather than just plain old stdin.on('data') because some new commands (namely p, and t the ones where you enter regex), also need to listen to 'data'. By this I mean that they have to listen for data while processing the command. In other words, without the generator you would need an explicit state machine which can be hard to debug and keep track of.
To see what I mean, look at listenForCommand:
Note that the
for await
waits on #data, not on this.#onDataGenerator()! What this means is that nested inside the #onCommand function we can await more lines of input, like this:Then when you are done processing your command, ie, the this.#onCommand returns, you can run the next loop (by getting the next this.#data) and everything works as it should.
And, that's how it works.
It's not too complicated, but maybe a bit unusual. So, I just wanted to document my choices and communicate clearly about why I made them.
The rest of the addition to the plan function is as you would expect, it asks the user for the command and prompts for regex filters if necessary.
Oh the places you'll go
As I mentioned before, the rest of the code is mostly passing around the watchModeSkipTests.
instructions
which is yieldedruntimeOptions
variable. Which is the passed to api.runwatchModeSkipTestsData
because only the Data gets passed to the worker, not the class itself.Since we are branching, let's choose a path
To the worker
(https://github.com/mmulet/ava/blob/78a3768fbe3360930351374f854f9cc7a46fda51/lib/worker/base.js#L75) that we need to pass the Data to the Runner
It is inside the runner that we use the watchModeSkipTests to actually , well, skip tests. The logic is simple, on every chain check if we should skip the test, and if we should, we skip!
Another Side, Another Story
Meanwhile, we have a second task, not just skip a test from running, we also want to prevent its result from being shown at all (this is a filter, the whole idea is we have too many tests to sift through).
And there you have it, a complete tour of the changes!
About the tests
One thing you'll notice while reviewing the tests is that I added try catch blocks on each and every one . I did this because whenever a test would fail, the entire test would just hang and end in a timeout (presumably because it throws an error, and the this.done function never gets run).
I tried this with a simple example, and it also had the same result:
Your contributing guidelines say not to include unrelated code changes in the pull request, so I'm considering this error as out of scope for now. But, now you know why the try catch blocks are there (the CLI will still report a
t.$someFunc()
failure as a failure, it just won't report exceptions as errors,throw $someError
will be caught)Last but not least
I'm going to submit this pull request for the bounty on https://oss.issuehunt.io/r/avajs/ava/issues/1530, but any extra tips (you
can use GitHub Sponsor page) would be very welcome!
IssueHunt Summary
Referenced issues
This pull request has been submitted to: