175

Any call in my unit tests to either Debug.Write(line) or Console.Write(Line) simply gets skipped over while debugging and the output is never printed. Calls to these functions from within classes I'm using work fine.

I understand that unit testing is meant to be automated, but I still would like to be able to output messages from a unit test.

1

17 Answers 17

170

Try using TestContext.WriteLine() which outputs text in test results.

Example:

[TestClass]
public class UnitTest1
{
    private TestContext testContextInstance;

    /// <summary>
    /// Gets or sets the test context which provides
    /// information about and functionality for the current test run.
    /// </summary>
    public TestContext TestContext
    {
        get { return testContextInstance; }
        set { testContextInstance = value; }
    }

    [TestMethod]
    public void TestMethod1()
    {
        TestContext.WriteLine("Message...");
    }
}

The "magic" is described in MSDN:

To use TestContext, create a member and property within your test class [...] The test framework automatically sets the property, which you can then use in unit tests.

6
  • 9
    I have found (with VS2013) that this only prints something if the test is run in debug mode.
    – fusi
    Commented Sep 24, 2015 at 11:15
  • 3
    It seems that using TestContext requires VS2015 Enterprise (or premium versions of earlier editions), according to this documentation Commented Nov 24, 2015 at 21:48
  • 1
    I'm noticing that if your string has curly braces in it, the method blows up. So "_testContext.WriteLine("hello");" works but "_testContext.WriteLine("he{ll}o");" fails with "System.FormatException: Input string was not in a correct format."
    – Mike K
    Commented Jul 20, 2017 at 21:08
  • 1
    The WriteLine method in TestContext takes the same parameters that Console.WriteWriteLine takes. That means that the string is a formatting string (documented at learn.microsoft.com/en-us/dotnet/standard/base-types/…). To use a literal { in a string, you need to double it. To print your string use WriteLine("he{{ll}}o");
    – Walter
    Commented Nov 7, 2017 at 21:59
  • 2
    this was interesting... although i still don't know why they didn't add this and similar features to a base class or interface we could use from the framework to avoid confusion.
    – Niklas
    Commented Apr 16, 2019 at 5:49
166

I was also trying to get Debug or Trace or Console or TestContext to work in unit testing.

None of these methods would appear to work or show output in the output window:

    Trace.WriteLine("test trace");
    Debug.WriteLine("test debug");
    TestContext.WriteLine("test context");
    Console.WriteLine("test console");

Visual Studio 2012 and greater

(from comments) In Visual Studio 2012, there is no icon. Instead, there is a link in the test results called Output. If you click on the link, you see all of the WriteLine.

Prior to Visual Studio 2012

I then noticed in my Test Results window, after running the test, next to the little success green circle, there is another icon. I doubled clicked it. It was my test results, and it included all of the types of writelines above.

10
  • 44
    In Visual Studio 2012, there is no icon. Instead, there is a link in the test results called "Output". If you click on the link, you see all of the writelines.
    – epotter
    Commented Apr 25, 2013 at 19:20
  • 18
    geez, that "Output" link is intrinsically hard to find. If anyone is struggling to find it, it's in the Text Explorer, bottom section. When a test is selected, it shows you the result with "Elapsed time: xxx". Below that is the "Output" link.
    – kevin
    Commented May 18, 2014 at 11:00
  • 1
    I don't see an Output link either (using VS2013 Pro, XUnit, Resharper)
    – atreeon
    Commented Jul 25, 2015 at 8:24
  • 4
    In Visual Studio 2017 it is still the 'Output' link.
    – HankCa
    Commented Jan 8, 2018 at 3:20
  • 7
    There is no longer an output link in VS2022 Commented Aug 19, 2021 at 16:25
84

In Visual Studio 2017, you can see the output from test explorer.

1) In your test method, Console.WriteLine("something");

2) Run the test.

3) In Test Explorer window, click the Passed Test Method.

4) And click the "Output" link.

enter image description here

And click "Output", you can see the result of Console.Writeline(). enter image description here

7
  • 1
    Excellent. Visual Studio/C# is not my norm for developing, this is just want I needed! Thank you for posting this.
    – matt32
    Commented Jun 2, 2018 at 1:00
  • 17
    I am using 2017 Version 15.5.6 Bit O don't see the output link.
    – Mike IT
    Commented Jul 12, 2018 at 17:56
  • 1
    Does it depend on which test framework is in use? I'm using xUnit 2.4 in a .NET 4.6 test project and no "Standard Output" pane appears. Output from Console.WriteLine is not visible in the Output pane under "Tests" either.
    – Qwertie
    Commented Jun 4, 2019 at 15:00
  • For xUnit, jonzim's answer worked for me, even in a different thread spawned by the test.
    – Qwertie
    Commented Jun 4, 2019 at 15:31
  • 2
    For those didn't see "output" link. 1) You should select a test method. 2) let the Test Explorer window fill the VS window vertically otherwise the link was hidden and the scroll bar is too small to be noticed or used. Commented Jul 24, 2019 at 4:55
43

It depends on your test runner... for instance, I'm using xUnit, so in case that's what you are using, follow these instructions:

https://xunit.net/docs/capturing-output

This method groups your output with each specific unit test.

using Xunit;
using Xunit.Abstractions;

public class MyTestClass
{
    private readonly ITestOutputHelper output;

    public MyTestClass(ITestOutputHelper output)
    {
        this.output = output;
    }

    [Fact]
    public void MyTest()
    {
        var temp = "my class!";
        output.WriteLine("This is output from {0}", temp);
    }
}

There's another method listed in the link I provided for writing to your Output window, but I prefer the previous.

0
9

In VS 2019

  1. in the main VS menu bar click: View -> Test Explorer
  2. Right-click your test method in Test Explorer -> Debug
  3. Click the additional output link as seen in the screenshot below.

enter image description here

You can use:

  • Debug.WriteLine
  • Console.WriteLine
  • TestContext.WriteLine

all will log to the additional output window.

4
  • 1
    Can't see console output in OneTimeSetUp and OneTimeTearDown though.
    – Pang
    Commented Dec 29, 2020 at 8:31
  • When I right click and select debug it reruns the test in debug mode. I'm using NUnit3, what test framework are you using?
    – RonC
    Commented Feb 1, 2021 at 23:27
  • 1
    Looks like autofac while using a test adapter for MsTest automatetheplanet.com/mstest-cheat-sheet/#tab-con-6
    – Legends
    Commented Feb 2, 2021 at 8:56
  • Note that a failed assertion will abort the test, so any output you want to add to the test results, specifically in case of a failed test, must be output before the assertion. Commented Nov 2, 2023 at 14:48
5

It is indeed depending on the test runner as @jonzim mentioned. For NUnit 3 I had to use NUnit.Framework.TestContext.Progress.WriteLine() to get running output in the Output window of Visual Studio 2017.

NUnit describes how to: here

To my understanding this revolves around the added parallelization of test execution the test runners have received.

4

I think it is still actual.

You can use this NuGet package: Bitoxygen.Testing.Pane

Call the custom WriteLine method from this library. It creates a Testing pane inside the Output window and puts messages there always (during each test, it runs independently of DEBUG and TRACE flags).

To make tracing more easy I can recommend to create a base class:

[TestClass]
public abstract class BaseTest
{
    #region Properties
    public TestContext TestContext { get; set; }

    public string Class
    {
        get { return this.TestContext.FullyQualifiedTestClassName; }
    }
    public string Method
    {
        get { return this.TestContext.TestName; }
    }
    #endregion

    #region Methods
    protected virtual void Trace(string message)
    {
        System.Diagnostics.Trace.WriteLine(message);

        Output.Testing.Trace.WriteLine(message);
    }
    #endregion
}

[TestClass]
public class SomeTest : BaseTest
{
    [TestMethod]
    public void SomeTest1()
    {
        this.Trace(string.Format("Yeah: {0} and {1}", this.Class, this.Method));
    }
}
3
  • 1
    The magic is at: Output.Testing.Trace.WriteLine(message); with BitOxygen. Commented Sep 30, 2016 at 17:43
  • 1
    Does not work with the current Visual Studio version you get a error Could not load file or assembly 'Microsoft.VisualStudio.Shell.12.0,
    – quadroid
    Commented Jul 9, 2018 at 9:59
  • @Console Yes it needs some support, but I am not sure if community is interesting in this way to output. xUnit has OutputHandler and VS can show its result.
    – Maxim
    Commented Jul 6, 2020 at 15:35
3

I'm using xUnit so this is what I use:

Debugger.Log(0, "1", input);

PS: you can use Debugger.Break(); too, so you can see your log in out.

2
  • 1
    What is this "out"? Commented Jun 25, 2020 at 11:18
  • 👍 DbgView also shows the output from Debugger.Log from any execution environment including xUnit.
    – noseratio
    Commented Jun 10, 2022 at 1:13
3

I've accidentally found that DebugView (with enabled Capture > Capture Win32 option) will capture writes to System.Console.

Example test:

[Test]
public void FooTest()
{
    for (int i = 0; i < 5; i++)
        Console.WriteLine($"[{DateTime.UtcNow.ToString("G")}] Hello World");
}

enter image description here

2

If you're running tests with dotnet test, you may find that Console.Error.WriteLine produces output.

This is how I've worked around the issue with NUnit tests, which also seem to have all Debug, Trace and standard output suppressed.

1
  • 1
    I can confirm that. Tried with .NET Core 3.1 on Ubuntu MATE 20.04 (Focal Fossa). System.Console.Error.WriteLine worked, but System.Diagnostics.Debug.WriteLine didn't. Commented Jun 14, 2021 at 1:00
2

For VS2022 in the Test Explorer the output will show up in the Test Detail Summary Pane. In additional you can right-click on your test and choose the 'Open test log' or simply Control+L and a testlog will open with all your Debug.WriteLine comments.

Further when .runsettings was attached to the unit test, please ensure that the CaptureTraceOutput was set to True.

<CaptureTraceOutput>True</CaptureTraceOutput>

DateTime myExpectedDateTime = sortedLPR[0].myDateTime;
DateTime myPreviousDateTime = sortedLPR[0].myDateTime;      // 1st date element
Debug.WriteLine($"(1) myPreviousDateTime: {myPreviousDateTime}");
Console.WriteLine($"(2) myPreviousDateTime: {myPreviousDateTime}");

testlog

Please notice the order of the output: (2) before (1) - standard output before debug output.

1

Try using:

Console.WriteLine()

The call to Debug.WriteLine will only be made during when DEBUG is defined.

Other suggestions are to use: Trace.WriteLine as well, but I haven't tried this.

There is also an option (not sure if Visual Studio 2008 has it), but you can still Use Debug.WriteLine when you run the test with Test With Debuggeroption in the IDE.

1
1

Solved with the following example:

public void CheckConsoleOutput()
{
    Console.WriteLine("Hello, World!");
    Trace.WriteLine("Trace Trace the World");
    Debug.WriteLine("Debug Debug World");
    Assert.IsTrue(true);
}

After running this test, under 'Test Passed', there is the option to view the output, which will bring up the output window.

0

Are you sure you're running your unit tests in Debug? Debug.WriteLine won't be called in Release builds.

Two options to try are:

  • Trace.WriteLine(), which is built into release builds as well as debug

  • Undefine DEBUG in your build settings for the unit test

0

A different variant of the cause/solution:

My issue was that I was not getting an output because I was writing the result set from an asynchronous LINQ call to the console in a loop in an asynchronous context:

var p = _context.Payment.Where(pp => pp.applicationNumber.Trim() == "12345");

p.ForEachAsync(payment => Console.WriteLine(payment.Amount));

And so the test was not writing to the console before the console object was cleaned up by the runtime (when running only one test).

The solution was to convert the result set to a list first, so I could use the non-asynchronous version of forEach():

var p = _context.Payment.Where(pp => pp.applicationNumber.Trim() == "12345").ToList();

p.ForEachAsync(payment =>Console.WriteLine(payment.Amount));
0

Console.WriteLine won't work. Only Debug.WriteLine() or Trace.WriteLine() will work, in debug mode.

I do the following: include using System.Diagnostics in the test module. Then, use Debug.WriteLine for my output, right click on the test, and choose Debug Selected Tests. The result output will now appear in the Output window below. I use Visual Studio 2017 version 15.8.1, with the default unit test framework Visual Studio provides.

-1

Trace.WriteLine should work provided you select the correct output (the dropdown labeled with "Show output from" found in the Output window).

Not the answer you're looking for? Browse other questions tagged or ask your own question.