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

NUnit.ConsoleRunner.NetCore executes test library under wrong runtime version #1380

Open
MichalMucek opened this issue Jan 11, 2024 · 31 comments
Labels
Bug High Priority Repro needed We need a short small repro for this issue.

Comments

@MichalMucek
Copy link

MichalMucek commented Jan 11, 2024

I'm trying to execute a test library using NUnit Console 3.16.3 (NuGet, .NET Tool) on Windows Server 2019 Datacenter (10.0.17763, x64). There are multiple .NET SDK versions installed on the machine - 6.0.202, 7.0.100, 7.0.309, 8.0.100.

In use are NUnit 3.13.1, NUnit3TestAdapter 4.4.2, and SpecFlow.NUnit 3.9.74.

The TargetFramework of the library is either net7.0-windows10.0.19041.0 or net8.0-windows10.0.19041.0. Each time I execute one of the libraries, .NET 6.0.23 (greatest/latest of the other available 6.0.x) is used for runtime, which results in errors of missing assemblies for the targeted framework or the selected runtime. In the case of net7.0-windows10.0.19041.0, it can reach the point of finding the tests. It is unable to do even that for net8.0-windows10.0.19041.0.

The test library executes with no issues when I set up an environment with .NET 7 or 8 only (or when I make a few tweaks in the appropriate keys of the Windows Registry). Using global.json with a specific .NET SDK version doesn't work.

InternalTrace.9572.Org.App.Specs, Version=1.0.0.0, Culture=neutral, PublicKeyToken=null (.NET 7).log
InternalTrace.9572 (.NET 7).log
Console output for net7.0-windows10.0.19041.0:

dotnet tool run dotnet-nunit .\Org.App.Specs.dll --trace=Debug
NUnit Console 3.16.3 (Release)
Copyright (c) 2022 Charlie Poole, Rob Prouse
Thursday, January 11, 2024 11:44:21 AM

Runtime Environment
   OS Version: Microsoft Windows 10.0.17763
  Runtime: .NET 6.0.23

Test Files
    .\Org.App.Specs.dll


Errors, Failures and Warnings

1) SetUp Error : Org_App_Specs_NUnitAssemblyHooks
System.IO.FileNotFoundException : Could not load file or assembly 'System.Collections.Specialized, Version=7.0.0.0, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a'. The system cannot find the file specified.
   at System.Configuration.ConfigurationElement..ctor()
   at System.Configuration.ConfigurationElementCollection..ctor()
   at BoDi.ContainerRegistrationCollection..ctor()
   at TechTalk.SpecFlow.Configuration.ConfigurationLoader.GetDefault()
   at TechTalk.SpecFlow.Infrastructure.ContainerBuilder.CreateGlobalContainer(Assembly testAssembly, IRuntimeConfigurationProvider configurationProvider)
   at TechTalk.SpecFlow.TestRunnerManager.CreateTestRunnerManager(Assembly testAssembly, IContainerBuilder containerBuilder)
   at TechTalk.SpecFlow.TestRunnerManager.GetTestRunnerManager(Assembly testAssembly, IContainerBuilder containerBuilder, Boolean createIfMissing)
   at TechTalk.SpecFlow.TestRunnerManager.OnTestRunStart(Assembly testAssembly, IContainerBuilder containerBuilder)
   at Org_App_Specs_NUnitAssemblyHooks.AssemblyInitialize() in C:\ApplicationRepo\Tests\App\Org.App.Specs\obj\Release\net7.0-windows10.0.19041.0\NUnit.AssemblyHooks.cs:line 20

Run Settings
    DisposeRunners: True
    InternalTraceLevel: Debug
    WorkDirectory: C:\net7.0-windows10.0.19041.0
    NumberOfTestWorkers: 8

Test Run Summary
  Overall result: Failed
  Test Count: 678, Passed: 0, Failed: 678, Warnings: 0, Inconclusive: 0, Skipped: 0
    Failed Tests - Failures: 0, Errors: 678, Invalid: 0
  Start time: 2024-01-11 11:44:21Z
    End time: 2024-01-11 11:44:22Z
    Duration: 0.916 seconds

Results (nunit3) saved as TestResult.xml

InternalTrace.3312.Org.App.Specs, Version=1.0.0.0, Culture=neutral, PublicKeyToken=null (.NET 8).log
InternalTrace.3312 (.NET 8).log
Console output for net8.0-windows10.0.19041.0:

dotnet tool run dotnet-nunit .\Org.App.Specs.dll --trace=Debug
NUnit Console 3.16.3 (Release)
Copyright (c) 2022 Charlie Poole, Rob Prouse
Thursday, January 11, 2024 11:44:51 AM

Runtime Environment
   OS Version: Microsoft Windows 10.0.17763
  Runtime: .NET 6.0.23

Test Files
    .\Org.App.Specs.dll


Errors, Failures and Warnings

1) Invalid : C:/net8.0-windows10.0.19041.0/Org.App.Specs.dll
Could not load type 'System.Runtime.CompilerServices.NullableContextAttribute' from assembly 'System.Runtime, Version=6.0.0.0, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a'.

Run Settings
    DisposeRunners: True
    InternalTraceLevel: Debug
    WorkDirectory: C:\net8.0-windows10.0.19041.0
    NumberOfTestWorkers: 8

Test Run Summary
  Overall result: Failed
  Test Count: 0, Passed: 0, Failed: 0, Warnings: 0, Inconclusive: 0, Skipped: 0
  Start time: 2024-01-11 11:44:51Z
    End time: 2024-01-11 11:44:52Z
    Duration: 0.327 seconds

Results (nunit3) saved as TestResult.xml
@stevenaw
Copy link
Member

Hi @MichalMucek
Unfortunately there are a few issues with the 3.16 version of the console runner. Since you have .NET8 installed could you please see if the prerelease of 3.17 works for you? That version should include .NET8 support among other fixes. https://github.com/nunit/nunit-console/releases/tag/V3.17.0-dev.3

Note that there is still a known issue related to projects requiring certain particular runtimes, with a workaround defined in #1383. If you hit the issue the workaround should work if your tests do not require a non-Windows OS.

@MichalMucek
Copy link
Author

@stevenaw Thanks for responding.
The release version of 3.17.0 produces the same issue.

I submitted #1381 not only with the intention of making the NUnit console work in environments where the .NET SDK is not installed, but also with the hope that it will be a workaround for this issue.

@OsirisTerje
Copy link
Member

@MichalMucek Have you worked through the comments in the #1374 and see if any of those might apply to you?

@MichalMucek
Copy link
Author

MichalMucek commented Mar 6, 2024

@OsirisTerje Thanks for the tip.
I've tried tweaking the nunit3-console.runtimeconfig.json file which was located at .../.nuget/packages/nunit.consolerunner.netcore/3.17.0/tools/net6.0/any. Changing runtimeOptions.framework.version from 6.0.0 to 8.0.0 helped.

@MichalMucek
Copy link
Author

Today, I installed NUnit.ConsoleRunner.NetCore 3.17.0 (.NET tool) on some other environment. There was a warning message (colored dark yellow), saying Format version is missing. This tool may not be supported in this SDK version. Contact the author of the tool. The used .NET SDK version was 8.0.200.

File nunit3-console.runtimeconfig.json was still pointing at framework version 6.0.0.

@OsirisTerje OsirisTerje added the Bug label Apr 4, 2024
@OsirisTerje OsirisTerje added this to the 3.18 milestone Apr 4, 2024
@OsirisTerje
Copy link
Member

@MichalMucek I can confirm this, so should be fixed in #1396

@CharliePoole
Copy link
Member

@OsirisTerje Is this a won't fix for version 3?

NUnit.ConsoleRunner.NetCore, executed directly or as a dotnet tool, only runs tests in-process. Since it targets .NET 6.0, that's how it runs them. Of course, the user may force it to run under .NET 7.0 or 8.0, but the code is still .NET 6.0 code.

I have some thoughts about giving the NetCore runner all the same capabilities as the standard console runner. That, however is a fairly big piece of work and seems more like a version 4 thing to me. Essentially, we would eliminate the distinction between the two runners and they would both do the same thing. In the distant future, we might even phase out the netfx runner, but I wouldn't hold my breath.

For the moment, I would suggest to @MichalMucek to do one of two things.

  1. Use dotnet test
  2. Use the standard runner, starting with the upcoming 3.18.0 release

Thoughts on this? Should we enhance the dotnet runner for 3.x?

@CharliePoole CharliePoole removed this from the 3.18.0 milestone Jul 16, 2024
@kogoel
Copy link

kogoel commented Sep 23, 2024

The latest version 3.18.2 of NUnit.ConsoleRunner.NetCore has the same behavior and taking Runtime as .net 6.0.5 even when .Net 8.0.8 is installed, with below error:
Unable to load one or more of the requested types.
Could not load file or assembly 'System.Runtime, Version=8.0.0.0, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a'. The system cannot find the file specified.

@MichalMucek
Copy link
Author

Another workaround for the issue is not having any .NET SDK installed...

We handle .NET SDK on our test environment with the .NET SDK Support Jenkins plugin. It allows us to use different .NET SDK versions without installing it. We have a test environment with 28 machines and it would be cumbersome to edit manually nunit3-console.runtimeconfig.json with every new version of the NUnit Console.

An alternative is to use a zipped (with binaries) version and perhaps some temporary additions to environment variables.

@bouchraRekhadda
Copy link

The latest version 3.18.2 of NUnit.ConsoleRunner.NetCore has the same behavior and taking Runtime as .net 6.0.5 even when .Net 8.0.8 is installed, with below error: Unable to load one or more of the requested types. Could not load file or assembly 'System.Runtime, Version=8.0.0.0, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a'. The system cannot find the file specified.

@kogoel Is your DLL self-contained ? because that typically happens when you are trying to do dotnet nunit3-console.dll --explore on a a self-contained library (reproducible with 3.18.1)

@CharliePoole
Copy link
Member

@OsirisTerje @MichalMucek @bouchraRekhadda @kogoel

This issue has had no attention since my comment three months ago asking for some overall direction for NUnit.ConsoleRunner.NetCore. I flagged it as needing discussion, which has not yet happened.

It was probably a mistake on my part to ask for that general discussion on this specific issue. I'm going to pull together what's done and what remains undone, along with the much greater part of potential functionality, which is undecided, and create a new issue to summarize all of it. I'll link to it here when I create it.

Meanwhile, I would like to refocus this issue on the original problem reported by @MichalMucek.

@CharliePoole
Copy link
Member

@MichalMucek Do you have a simple repro I can use to work this issue? Preferably one without third-party references?

@CharliePoole
Copy link
Member

@MichalMucek Also, the command-line used to execute your problem test scenario.

@OsirisTerje
Copy link
Member

OsirisTerje commented Sep 23, 2024

@CharliePoole Sorry for not coming back on your comment, but I actually don't know. The console runners are not really following SemVer, so whether it should be 3 or 4 - I don't know. I would say that IF there are breaking changes, then it is a 4. If it is just enhancements, and everything works as before for those people using it, well, then it is a 3. But, if we feel the change is a big one, we could still give it a 4, to indicate this is a big change. In this case, it may be a lot of change to the code, as you indicate, but for the user - not so much. What I think is important, is what you have been doing lately, fixing issue by issue, and then unblocking a lot of stuff. To me it looks like the proposed changes here are the same, unblocking stuff. If that can be done in small steps, so that we dont have long living feature branches - which may go stale, that would be awesome. And yes, it deserves a new issue.

@CharliePoole
Copy link
Member

@OsirisTerje My question is more fundamental than version three vs four. Once we have a specific feature in mind, it's easy to figure out if it's breaking or if it's scope is so great that we prefer to wait for V4. It would be great to have some discussion about that when we have V3 a bit cleaner.

What I don't actually know is the intended feature set for the NetCore runner. One way to do that may be to look at the limitations and then decide if it's desirable to remove them or if they are reasonable for the intended use of this runner. That's what I'll write something more about in a separate issue. Soon. :-)

BTW I recognize that you may be in the same boat as I am, i.e. not a routine user of the NetCore runner or the dotnet tool that is based on it. In that case, we need to find some people who are. @nunit team? Users?

@OsirisTerje
Copy link
Member

When I am using the console runner, I am actually using the netcore one, as I install it as a dotnet tool. That makes it very convenient to use. I rarely use any of the other console apps.

But, for the use I have been doing, I have not been bothered by the limitations there. I noticed them, but that was all. That said, it would be nice to get it more aligned with its more mature siblings, but then I am not the "routine user".

@MichalMucek
Copy link
Author

@MichalMucek Do you have a simple repro I can use to work this issue? Preferably one without third-party references?

@CharliePoole No, not at the moment. I'll try to work out something but please don't expect it soon.

@bouchraRekhadda
Copy link

NUnit.ConsoleRunner.NetCore, executed directly or as a dotnet tool, only runs tests in-process. Since it targets .NET 6.0, that's how it runs them. Of course, the user may force it to run under .NET 7.0 or 8.0, but the code is still .NET 6.0 code.

@CharliePoole how can we force it to load other runtime versions? also would it be feasible for the Nunit.ConsoleRunner.NetCore to target .NET 8 directly ?

@CharliePoole
Copy link
Member

@bouchraRekhadda

how can we force it to load other runtime versions?

Some people report success changing the delivered .runtimeconfig.json file.

also would it be feasible for the Nunit.ConsoleRunner.NetCore to target .NET 8 directly ?

Yes, I plan to make a build available for testing. It will need to be tested both by some folks targeting .NET 6.0 and earlier releases as well as those who use .NET 8.0. Ideally, we can find some people to test it both in the form of a tool and by running it directly.

@bouchraRekhadda
Copy link

bouchraRekhadda commented Sep 26, 2024

@CharliePoole

Yes, I plan to make a build available for testing. It will need to be tested both by some folks targeting .NET 6.0 and earlier releases as well as those who use .NET 8.0. Ideally, we can find some people to test it both in the form of a tool and by running it directly.

If the nupkg will be available for me to download I can test it (we are currently targeting .NET 6 and working to upgrade to .NET 8 soon)

@CharliePoole
Copy link
Member

Excellent. I'll post a link here when it's ready.

@CharliePoole CharliePoole added this to the 3.19.0 milestone Sep 30, 2024
@bouchraRekhadda
Copy link

Hello @CharliePoole do you know when will this package be available to test it ?

Regards

@CharliePoole
Copy link
Member

The release won't be until 3.19 (I'm now working on 3.18.3) but I plan to have a build for testing available in a few days.

@CharliePoole
Copy link
Member

CharliePoole commented Oct 3, 2024

@bouchraRekhadda There is a .NET 8.0 console runner avaialble on the myget feed. It's NUnit.ConsoleRunner.Net80 version 3.18.3-dev00006 UPDATED: 3.18.3-dev00009. This is an experimental build so the name may not be permanent. I expect to have an equivalent runner in 3.19 either as an update to the netcore runner or an alternative.

@CharliePoole
Copy link
Member

See #1493, which I'm hoping will resolve this issue as well.

@CharliePoole
Copy link
Member

@MichalMucek Can you try the latest dev build from our myget feed? It includes NUnit.ConsoleRunner.Net80, which I believe will solve your problem. It's an experimental build, so only available as a nuget package. The tool name is nunit-net80.

@kogoel
Copy link

kogoel commented Oct 14, 2024

@CharliePoole : Can you please confirm the nuget package in which this bug is fixed is https://www.nuget.org/packages/NUnit.ConsoleRunner.Net80
Thanks
Kohima

@CharliePoole
Copy link
Member

@kogoel

The issue is still outstanding but the 3.18.3 release contains an experimental .NET 8.0 runner, NUnit.ConsoleRunner.Net80, available on NuGet or on the GitHub release page.

https://www.nuget.org/packages/NUnit.ConsoleRunner.Net80

See also the discussions under #1492 and #1493

@kogoel
Copy link

kogoel commented Oct 15, 2024

@kogoel

The issue is still outstanding but the 3.18.3 release contains an experimental .NET 8.0 runner, NUnit.ConsoleRunner.Net80, available on NuGet or on the GitHub release page.

https://www.nuget.org/packages/NUnit.ConsoleRunner.Net80

See also the discussions under #1492 and #1493

Yes tested with the mentioned Nuget package received the error similar to earlier but now the error says, if it helps. Could not load file or assembly 'System.Security.Cryptography, Version=8.0.0.0, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a'. The system cannot find the file specified.
while earlier it was
Could not load file or assembly 'System.Runtime, Version=8.0.0.0, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a'. The system cannot find the file specified.
Thanks
Kohima

@CharliePoole CharliePoole added Repro needed We need a short small repro for this issue. and removed Waiting for Comment labels Nov 27, 2024
@CharliePoole CharliePoole removed this from the 3.19.0 milestone Dec 3, 2024
@broetchenrackete36
Copy link

This is reproducable when trying to run .NET 9 tests with the .NET 8 runner.

I created a simple NUnit3-Project targeting .NET9 and added Console.WriteLine(TimeSpan.FromSeconds(1)); in the SetUp, nothing more. Trying to run it with the 3.19.0 .NET 8 runner produces the following error:

NUnit NetCore Console Runner 3.19.0 (Release)
Copyright (c) 2022 Charlie Poole, Rob Prouse
Dienstag, 14. Januar 2025 16:33:47

Runtime Environment
   OS Version: Microsoft Windows 10.0.22631
  Runtime: .NET 8.0.11

Test Files
    nunittimespantest.dll


Errors, Failures and Warnings

1) Error : nunittimespantest.Tests.Test1
System.MissingMethodException : Method not found: 'System.TimeSpan System.TimeSpan.FromSeconds(Int64)'.
   at nunittimespantest.Tests.Setup()
   at System.RuntimeMethodHandle.InvokeMethod(Object target, Void** arguments, Signature sig, Boolean isConstructor)
   at System.Reflection.MethodBaseInvoker.InvokeWithNoArgs(Object obj, BindingFlags invokeAttr)

Run Settings
    DisposeRunners: True
    WorkDirectory: C:\Temp\nunit_timespan_test\nunittimespantest\bin\Debug\net9.0
    NumberOfTestWorkers: 24

Test Run Summary
  Overall result: Failed
  Test Count: 1, Passed: 0, Failed: 1, Warnings: 0, Inconclusive: 0, Skipped: 0
    Failed Tests - Failures: 0, Errors: 1, Invalid: 0
  Start time: 2025-01-14 15:33:47Z
    End time: 2025-01-14 15:33:47Z
    Duration: 0.165 seconds

Results (nunit3) saved as TestResult.xml

Changing the nunit3-console.runtimeconfig.json to .NET 9 and it runs fine:

NUnit NetCore Console Runner 3.19.0 (Release)
Copyright (c) 2022 Charlie Poole, Rob Prouse
Dienstag, 14. Januar 2025 16:25:23

Runtime Environment
   OS Version: Microsoft Windows 10.0.22631
  Runtime: .NET 9.0.0

Test Files
    nunittimespantest.dll

00:00:01

Run Settings
    DisposeRunners: True
    WorkDirectory: C:\Temp\nunit_timespan_test\nunittimespantest\bin\Debug\net9.0
    NumberOfTestWorkers: 24

Test Run Summary
  Overall result: Passed
  Test Count: 1, Passed: 1, Failed: 0, Warnings: 0, Inconclusive: 0, Skipped: 0
  Start time: 2025-01-14 15:25:23Z
    End time: 2025-01-14 15:25:24Z
    Duration: 0.336 seconds

Results (nunit3) saved as TestResult.xml

@mikkelbu
Copy link
Member

I've not reread the entire thread, but the issue above - System.MissingMethodException : Method not found: 'System.TimeSpan System.TimeSpan.FromSeconds(Int64)'. - is due to https://learn.microsoft.com/en-us/dotnet/core/compatibility/core-libraries/9.0/timespan-from-overloads.

So when the code is compiled then TimeSpan.AddMinutes(1) would on .NET8 (and before) resolve against TimeSpan.AddMinutes(double). On .NET9 it resolves against the new overload TimeSpan.AddMinutes(int64).

I'm not sure what the best approach is to solve this

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
Bug High Priority Repro needed We need a short small repro for this issue.
Projects
None yet
Development

No branches or pull requests

8 participants