Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Enable unit tests to run in parallel #3007

Open
tig opened this issue Nov 23, 2023 · 7 comments
Open

Enable unit tests to run in parallel #3007

tig opened this issue Nov 23, 2023 · 7 comments
Labels
testing Issues related to testing
Milestone

Comments

@tig
Copy link
Collaborator

tig commented Nov 23, 2023

Running the unit tests on a fast machine can take upwards of 3 minutes. Runs in Github Action can take even longer. This is annoying and slows development.

The reason the unit tests are forced to run serially ("parallelizeTestCollections": false,) is primarily because Application is a singleton.

However, over time I've worked hard to make as many unit tests (and code) independent of Application.

We should create another xunit test project (Low-level UnitTests?) that has "parallelizeTestCollections": true, and move any test that is independent of Application (or other statics) there.

@tig tig added the testing Issues related to testing label Nov 23, 2023
@tig
Copy link
Collaborator Author

tig commented Jan 20, 2024

From this issue, via @dodexahedron:

We should consider leveraging Nunit as part of addressing THIS Issue.

As I'm working on Color, I'm also re-working most of the existing tests. Some are just getting unrolled into parameterized tests, some are getting expanded, some are getting re-worked, some are getting removed, and some are getting replaced, as appropriate - a lot of the same sort of work I've been doing for TGD, just XUnit-flavored over here.

Thank you. I strongly suspect the work you are doing will serve as a great "how to do it RIGHT" for everyone else (including me). Thank you. Thank you.

  • Would you mind terribly if I included NUnit as well, if or when something would be significantly better or easier? That comes like 1/4 from personal preference and majority just from the fact that it's way more robust and expressive. No worries if you'd rather I not.

I have no real love of xUnit. I used NUnit in another project and it was fine too. I'd like @BDisp and @tznind to chime in on this. I'm NOT a fan of using TWO test frameworks if we can avoid it. So, if you really think NUnit is signficantly better, and it makes your life significantly better, I would support a transition. But @BDisp and @tznind have to fully support this too because it will be a LOT of work.

  • Do you have a naming convention you're married to for test methods you'd like me to stick to? I try to be pretty descriptive with them, and thus far I've been mostly following along with what existing tests look like, with the exception of I usually don't also include the name of the class itself in the test method, since the test fixture already has that.
  • Be clear, concise, and complete. Clear means human readable and descriptive of the theory. Concise means terse (no class-name). Complete means no surprises pop when you actually look at the test code.
  • Use underscores where you'd use a space. Use camel-case for names of classes/methods/properties that are camel-cased.

What else?

  • Do you mind if, when it is important to do so, that I make something that was private be internal, for test purposes? I think that situation should typically be pretty rare, but just throwing it out there.

I actually encourage it. However, please add XML docs to any such element and have the docs say why it's internal (e.g. "This method is internal to support unit testing.). I wish there was a way of declaring a member "Make this internal for unit testing, but for god's sake don't let it be used anywhere else but this class."

All sounds good to me and are pretty much how I operate anyway, so yay.

As for the test framework, it's not terribly important to me. TGD uses NUnit, and part of the work I've been doing there is actually updating the code from NUnit 2-era assertions to NUnit 4, which is current.

There are various libraries meant to work with multiple frameworks which can add in nice features without changing it all. One example would be FluentAssertions which basically just let's you use fluent-style method chains to make more expressive assertion statements, which is a similar idea to what NUnit has natively, but even takes the concept further and is of course an optional add-on (thus non-breaking to existing tests). And then libraries like the one I used for those combinatorial tests also exist in various forms. That one was pretty commonly.mentioned and recommended around the net, which is why I picked that one. That and it's an active project, which is also good.

In the end, a test harness is just code, and the framework is just syntactic sugar to reduce boilerplate.

However (and keep in mind I'm not pushing for a change, I'm just delivering information)....

Here are some key features native to NUnit that are just damn nice:

  • Test ordering by simple Order attributes. Useful when you want to ensure that more basic, common core code is tested first, so time isn't wasted running a bunch of tests that will ultimately fail because some key dependency is broken.
  • Pre-conditions using the Assume statement (rather than Assert. These are for the setup phase of a test fixture or test method, when you want to guarantee or prove a specific set of starting conditions for the actual test to be considered valid. Failing these gives the test case an Indeterminate result, which is more informative that something other than the code under test is causing the problem.
  • More types of assertions built-in
  • Parallel test execution with control granularity all the way down to the individual generated cases if parameterized tests.
  • Combinatorial, pairwise, and sequential options for generation of test cases for parameterized tests. Sequential is handy and really cuts down on the need to write test case generator methods. The other two exist in a limited form in the extension I pulled in for xUnit.
  • Much much much much MUCH better documentation. My god, the xUnit "documentation" is sorely lacking.
  • Automatic expansion of enum parameters to a test case for each defined enum value, via this: [Values] (yep that's literally it), and that on top of the other rich parameterization goodness.
  • Nice informational attributes for indicating what type or method a given test fixture or test method is testing. This is handy for informational purposes and some tools can also make use of it.
  • Attributes to control which test methods or test fixtures are selected by the test runner based on the platform of the machine running the test (so you can test platform-specific code without messy conditional compilation or extra code in tests to handle that stuff)
  • Threading control, such as forcing a test or fixture to run on a single thread, such as when testing certain things around statics.
  • Tons more, but I'm feeling self-conscious like I'm advertising for it or something lol. Just have a look at the UnitTests project in TGD. I make use of everything I've mentioned here all over the place and more. 😅

Though one thing I will say that is intended somewhat as a directed point in support of switching: It's mostly a Find/Replace job to switch, as everything in the existing tests is there with the same syntax but just different attribute names.

Haha but enough about that. I'm gonna get back to work on Color and its tests. 😅

(This message clearly brought to you by NUnit.) 😆

@dodexahedron
Copy link
Collaborator

dodexahedron commented Jan 22, 2024

Some of this has been mentioned, but I'm just putting more words and detail to stuff.

These are some options available:

Split up the test project

First idea, which is independent of whether test framework changes or not, is to split the test project up into 2 projects - one for anything that actually requires firing up an Application, and the other for everything else.

Pros:

  • Anything requiring Application is more of an integration test, rather than a unit test, so that makes sense for separation/organization of things.
  • Different configuration for the test runner can be used in each, keeping each from being impacted by requirements of the other, such as parallelization.
  • If one test framework suits one better for any reason, each could use a different one.

Cons:

  • An additional project file.
  • Potential confusion as to where a test fixture belongs or where one was placed
    • Attributes in both XUnit and NUnit can help with that (Trait with a typeof in XUnit and TestOf in NUnit)
  • Potential for arbitrary placement of a test fixture because something lies in a gray area.
    • Addressable via policy, such as if something is in a gray area, default to placing it in one of them. Final decisions can be made during reviews.

Multiple test runner configurations

Second idea is also independent of any others and thus can be used with or without any other strategy, regardless of framework.

Different configurations can be used within a single test project in any framework through use of things such as the --filter argument to dotnet test among other ways of selecting subsets of tests to run. With XUnit, that would be easiest to achieve by using Traits, such as the "Category" Trait I've applied to the new tests I've written. Trait is kind of a generic catch-all attribute for whatever arbitrary classification you want to give a test. NUnit's direct equivalent is Property, though things like Category are also formal attributes in NUnit.

Using this option, test classes or methods that require Application setup and teardown could be marked as such, or those that don't could be marked, or both. Then, either test runner configuration files (such as different copies of the xunit.runner.json file), --filter arguments, or other means can be used to run each subset of tests with the desired/required settings.

This is a very flexible option and isn't all-or-nothing, and I think it's a prudent strategy to implement no matter which framework is in use or what other ideas we choose to implement.

Pros:

  • Very flexible
  • As granular as we want it to be, down to the individual test method.
  • Easy to do
  • Subsets do not have to be exclusive - Filters or configurations can include or exclude anything based on any filterable criteria, so there could be an "everything" configuration, plus any other subsets that are deemed advantageous.
    • Examples: Everything, Parallel/NonParallel, Platform-Specific, Serialization, Core (for things like base types and extension methods that nearly everything depends on), etc

Cons (these are super-minor, but I wanted to come up with something):

  • Mostly just have to come up with at least a crude strategy to adhere to.
  • Users running tests would have to be sure they're running the appropriate tests when working on something. The default configuration being "everything" makes that a non-issue.

@tig
Copy link
Collaborator Author

tig commented Jan 22, 2024

Love.

How about

  • I get over using two frameworks, at least in the short term.
  • You intro a PR for this issue
    • Rename "UnitTests" project to IntegrationTests
    • Add new project based on NUnit using all your mad skills and knowledge on the subject named "UnitTests"
    • Move a select subset of the old unit tests to this new project illustrating best practices.

Then, we can all start migrating tests.

??

@dodexahedron
Copy link
Collaborator

Hahaha

Well... I'm trying to take it slow.

As I mentioned in the other issue, tests are just code and anything doable in one framework is doable in another or even without a test framework, if one writes the necessary code.

For right now, and for the PR I'll put in for the Color stuff, I'm still using XUnit. Fewer changes are still a nice thing to strive for when possible.

But yes, if we want to split anything up, vis-a-vis project or framework, that action plan sounds great.

I'd offer that, if we do use NUnit for stuff, I'll be happy to shoulder most or all of the burden for conversion, since I catalyzed that initiative. 😅

@dodexahedron
Copy link
Collaborator

And also, another key is I don't want to distract from work on the actual library itself. Hence why I'm happy to lighten the load if any changes to framework are settled on.

@tig
Copy link
Collaborator Author

tig commented Jan 22, 2024

I'd offer that, if we do use NUnit for stuff, I'll be happy to shoulder most or all of the burden for conversion, since I catalyzed that initiative. 😅

Oops. I mis-typed above. I wrote xunit, but mean nunit. I've edited it to be correct.

@dodexahedron
Copy link
Collaborator

Huh. I didn't even notice that and my brain filled in "NUnit" the first time, anyway. 😅

@tig tig moved this to 📋 v2 BETA Backlog in Terminal.Gui V2 Beta May 12, 2024
@tig tig added this to the V2 Beta milestone May 25, 2024
@tig tig moved this to 🆕 Not Triaged in Terminal.Gui V2 Initial Release Jul 11, 2024
@tig tig modified the milestones: V2 Beta, V2 Release Aug 6, 2024
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
testing Issues related to testing
Projects
Status: 🆕 Not Triaged
Status: 📋 Approved - Need Owner
Development

No branches or pull requests

2 participants