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

Show fscheck num tests and distribution on test pass #514

Open
wants to merge 14 commits into
base: main
Choose a base branch
from

Conversation

rynoV
Copy link
Contributor

@rynoV rynoV commented Dec 17, 2024

Previously for passing fscheck properties, the number of tests passed wouldn't be shown, and there was no option to show the test case distribution (e.g. percentage trivial tests, etc)

Now quietOnSuccess config can be set to false to show this info. New maxRejected config is also accepted

More generally, tests are enabled to show any message on pass via the PassWithMessage exception. This is pretty hacky so this is more of a proof of concept. The alternative was changing the core test types to optionally return a message, which seemed like too big of a change to make on my own.

@farlee2121
Copy link
Collaborator

farlee2121 commented Dec 29, 2024

I feel unsure about using pass exceptions. If we do go that route, I think it'd be better to give TestResult.Passed an optional message rather than have two union cases for passed tests. Multiple pass cases feels like it'll lead to easy mistakes. The same apply for multiple pass printers: It'd be better to have one pass printer with an optional message parameter

A notable con to some PassException is that we wouldn't be able to hide it, and then it would be available for other assertions. The message might add temptation to use the PassException even when execution should not be shortcutted

What are some alternative solutions?

  • OPT: Using the logger to print the message
    • I'm trying to deglobalize the logger, and the property test methods don't have a ready way to compose the logger without user involvement
  • OPT: As you mentioned, Return an optional message from test methods
    • This is a high-impact change. It'd also be a bit roundabout since none of the Check methods appear to return the test data directly.
  • OPT: Call Check.One directly
  • OPT: Create an FsCheck assertion that returns test distribution data
  • Others?

Some questions that might help navigate these choices:

  • How often do you want this distribution data?
  • What does it help you do?
  • Is there another way to achieve the same top-level value you need?

@rynoV
Copy link
Contributor Author

rynoV commented Jan 6, 2025

I really only need to see the distribution as a debug tool, I only need to see the distribution in the console during development, so using the logger if possible sounds good. I'm not sure of the best way to handle the logger. Currently my tests are creating the Expecto logger at startup and passing it down through all the tests, wrapped in the logger API required by the app code.

I updated the PR with a solution that would work for my use case, which is for the property test config to include a logger field, and if that is set it is used when tests pass, but not tied to this

@farlee2121
Copy link
Collaborator

Such a logger on FsCheckConfig breaks Information Hiding.
Callers have to know about the internals of the library to understand what the parameter will do.
Even if they understand what it does, it only has a very specific application.

Do you need this diagnostic information often?

@farlee2121 farlee2121 self-requested a review January 6, 2025 23:20
@rynoV
Copy link
Contributor Author

rynoV commented Jan 7, 2025

Ideally every property test would show at least the number of tests that passed, so that's frequent but more of a "nice to have". Test distribution is less frequently needed.

How about adding the TestResult to FsCheckConfig.finishedTest? Then I could use my logger as needed, and it would enable the user to handle each of the different test result types.

Or if it's not desirable to expose FsCheck's type directly, the minimum I would want is the number of tests int and stamps (int * string list) seq. Or maybe we make an Expecto.FsCheckTestResult that mirrors fscheck's TestResult?

@farlee2121
Copy link
Collaborator

I like this line of thought. I think you're on to a good idea.

Adding TestResult to the callback would make sense, but we're haunted by the fact that FsCheckConfig actually lives in the core Expecto library. The core library would have to add a reference to FsCheck to return the TestResult.
I've looked at fully isolating FsCheck into the Expecto.FsCheck package, but it's not a small lift.

For now, we probably need to look at returning our own data structure.
The question is: what should that data structure look like?

Ideally we want minimal maintenance to keep that data in sync with what's available from FsCheck.

I notice TestData is available on every TestResult case. TestData would cover your usecase without having to recreate the whole TestResult type.

That would leave us with something like this

type TestDataProxy = {
  Labels: Set<string>
  NumberOfShrinks: int
  NumberOfTests: int
  Stamps: seq<int * list<string>>
}

It would be good to know if the test passed, failed, or was exhausted, but TestData might be good enough to see if people want more. If so, it might be time to bite the bullet and figure out how to move FsCheckConfig to the Expecto.FsCheck package.

Thoughts?

@rynoV
Copy link
Contributor Author

rynoV commented Jan 13, 2025

Seems good to me, let me know when you get a chance to look at the changes. I can also add to the readme a recommendation of something like:

    let testLogger = Expecto.Logging.Log.create "MyTests"

    let numTests i =
        if i = 1 then "1 test" else sprintf "%i tests" i

    let stampsToString s =
        let entry (p, xs) =
            sprintf "%A%s %s" p "%" (String.concat ", " xs)

        match Seq.map entry s |> Seq.toList with
        | [] -> ""
        | [ x ] -> sprintf " (%s)\n" x
        | xs -> sprintf "%s\n" (String.concat "\n" xs)

    let defaultFsCheckConfig =
        { FsCheckConfig.defaultConfig with
            finishedTest =
                fun _ _ testData ->
                    async {
                        testLogger.log
                            Logging.LogLevel.Info
                            (Logging.Message.eventX
                                $"Passed {numTests testData.NumberOfTests}.\n{stampsToString testData.Stamps}")
                        |> Async.RunSynchronously
                    } }

    let myTest = propertyTestWithConfig defaultFsCheckConfig "My test" ...

Or we could add this as an alternative base config, taking the logger as an argument?

@farlee2121
Copy link
Collaborator

I think this one is almost ready, there's still some noise left over from all the revisions: QuietOnSuccess, MaxRejected, readme text related to those two plus replay

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

2 participants