TestableIO.System.IO.Abstractions.TestingHelpers 21.2.1

Prefix Reserved
dotnet add package TestableIO.System.IO.Abstractions.TestingHelpers --version 21.2.1                
NuGet\Install-Package TestableIO.System.IO.Abstractions.TestingHelpers -Version 21.2.1                
This command is intended to be used within the Package Manager Console in Visual Studio, as it uses the NuGet module's version of Install-Package.
<PackageReference Include="TestableIO.System.IO.Abstractions.TestingHelpers" Version="21.2.1" />                
For projects that support PackageReference, copy this XML node into the project file to reference the package.
paket add TestableIO.System.IO.Abstractions.TestingHelpers --version 21.2.1                
#r "nuget: TestableIO.System.IO.Abstractions.TestingHelpers, 21.2.1"                
#r directive can be used in F# Interactive and Polyglot Notebooks. Copy this into the interactive tool or source code of the script to reference the package.
// Install TestableIO.System.IO.Abstractions.TestingHelpers as a Cake Addin
#addin nuget:?package=TestableIO.System.IO.Abstractions.TestingHelpers&version=21.2.1

// Install TestableIO.System.IO.Abstractions.TestingHelpers as a Cake Tool
#tool nuget:?package=TestableIO.System.IO.Abstractions.TestingHelpers&version=21.2.1                

System.IO.Abstractions NuGet Continuous Integration Codacy Badge Renovate enabled FOSSA Status

At the core of the library is IFileSystem and FileSystem. Instead of calling methods like File.ReadAllText directly, use IFileSystem.File.ReadAllText. We have exactly the same API, except that ours is injectable and testable.

Usage

dotnet add package TestableIO.System.IO.Abstractions.Wrappers

Note: This NuGet package is also published as System.IO.Abstractions but we suggest to use the prefix to make clear that this is not an official .NET package.

public class MyComponent
{
    readonly IFileSystem fileSystem;

    // <summary>Create MyComponent with the given fileSystem implementation</summary>
    public MyComponent(IFileSystem fileSystem)
    {
        this.fileSystem = fileSystem;
    }
    /// <summary>Create MyComponent</summary>
    public MyComponent() : this(
        fileSystem: new FileSystem() //use default implementation which calls System.IO
    )
    {
    }

    public void Validate()
    {
        foreach (var textFile in fileSystem.Directory.GetFiles(@"c:\", "*.txt", SearchOption.TopDirectoryOnly))
        {
            var text = fileSystem.File.ReadAllText(textFile);
            if (text != "Testing is awesome.")
                throw new NotSupportedException("We can't go on together. It's not me, it's you.");
        }
    }
}

Test helpers

The library also ships with a series of test helpers to save you from having to mock out every call, for basic scenarios. They are not a complete copy of a real-life file system, but they'll get you most of the way there.

dotnet add package TestableIO.System.IO.Abstractions.TestingHelpers

Note: This NuGet package is also published as System.IO.Abstractions.TestingHelpers but we suggest to use the prefix to make clear that this is not an official .NET package.

[Test]
public void MyComponent_Validate_ShouldThrowNotSupportedExceptionIfTestingIsNotAwesome()
{
    // Arrange
    var fileSystem = new MockFileSystem(new Dictionary<string, MockFileData>
    {
        { @"c:\myfile.txt", new MockFileData("Testing is meh.") },
        { @"c:\demo\jQuery.js", new MockFileData("some js") },
        { @"c:\demo\image.gif", new MockFileData(new byte[] { 0x12, 0x34, 0x56, 0xd2 }) }
    });
    var component = new MyComponent(fileSystem);

    try
    {
        // Act
        component.Validate();
    }
    catch (NotSupportedException ex)
    {
        // Assert
        Assert.That(ex.Message, Is.EqualTo("We can't go on together. It's not me, it's you."));
        return;
    }

    Assert.Fail("The expected exception was not thrown.");
}

We even support casting from the .NET Framework's untestable types to our testable wrappers:

FileInfo SomeApiMethodThatReturnsFileInfo()
{
    return new FileInfo("a");
}

void MyFancyMethod()
{
    var testableFileInfo = (FileInfoBase)SomeApiMethodThatReturnsFileInfo();
    ...
}

Mock support

Since version 4.0 the top-level APIs expose interfaces instead of abstract base classes (these still exist, though), allowing you to completely mock the file system. Here's a small example, using Moq:

[Test]
public void Test1()
{
    var watcher = Mock.Of<IFileSystemWatcher>();
    var file = Mock.Of<IFile>();

    Mock.Get(file).Setup(f => f.Exists(It.IsAny<string>())).Returns(true);
    Mock.Get(file).Setup(f => f.ReadAllText(It.IsAny<string>())).Throws<OutOfMemoryException>();

    var unitUnderTest = new SomeClassUsingFileSystemWatcher(watcher, file);

    Assert.Throws<OutOfMemoryException>(() => {
        Mock.Get(watcher).Raise(w => w.Created += null, new System.IO.FileSystemEventArgs(System.IO.WatcherChangeTypes.Created, @"C:\Some\Directory", "Some.File"));
    });

    Mock.Get(file).Verify(f => f.Exists(It.IsAny<string>()), Times.Once);

    Assert.True(unitUnderTest.FileWasCreated);
}

public class SomeClassUsingFileSystemWatcher
{
    private readonly IFileSystemWatcher _watcher;
    private readonly IFile _file;

    public bool FileWasCreated { get; private set; }

    public SomeClassUsingFileSystemWatcher(IFileSystemWatcher watcher, IFile file)
    {
        this._file = file;
        this._watcher = watcher;
        this._watcher.Created += Watcher_Created;
    }

    private void Watcher_Created(object sender, System.IO.FileSystemEventArgs e)
    {
        FileWasCreated = true;

        if(_file.Exists(e.FullPath))
        {
            var text = _file.ReadAllText(e.FullPath);
        }
    }
}
Product Compatible and additional computed target framework versions.
.NET net5.0 was computed.  net5.0-windows was computed.  net6.0 is compatible.  net6.0-android was computed.  net6.0-ios was computed.  net6.0-maccatalyst was computed.  net6.0-macos was computed.  net6.0-tvos was computed.  net6.0-windows was computed.  net7.0 is compatible.  net7.0-android was computed.  net7.0-ios was computed.  net7.0-maccatalyst was computed.  net7.0-macos was computed.  net7.0-tvos was computed.  net7.0-windows was computed.  net8.0 is compatible.  net8.0-android was computed.  net8.0-browser was computed.  net8.0-ios was computed.  net8.0-maccatalyst was computed.  net8.0-macos was computed.  net8.0-tvos was computed.  net8.0-windows was computed. 
.NET Core netcoreapp2.0 was computed.  netcoreapp2.1 was computed.  netcoreapp2.2 was computed.  netcoreapp3.0 was computed.  netcoreapp3.1 was computed. 
.NET Standard netstandard2.0 is compatible.  netstandard2.1 is compatible. 
.NET Framework net461 was computed.  net462 is compatible.  net463 was computed.  net47 was computed.  net471 was computed.  net472 was computed.  net48 was computed.  net481 was computed. 
MonoAndroid monoandroid was computed. 
MonoMac monomac was computed. 
MonoTouch monotouch was computed. 
Tizen tizen40 was computed.  tizen60 was computed. 
Xamarin.iOS xamarinios was computed. 
Xamarin.Mac xamarinmac was computed. 
Xamarin.TVOS xamarintvos was computed. 
Xamarin.WatchOS xamarinwatchos was computed. 
Compatible target framework(s)
Included target framework(s) (in package)
Learn more about Target Frameworks and .NET Standard.

NuGet packages (8)

Showing the top 5 NuGet packages that depend on TestableIO.System.IO.Abstractions.TestingHelpers:

Package Downloads
System.IO.Abstractions.TestingHelpers

A set of pre-built mocks to help when testing file system interactions.

Reo.Core.Testing

Package Description

Reo.Core.IntegrationTesting

Package Description

FastMoq.Core

Easy and fast extension of the famous Moq mocking framework for mocking and auto injection of classes when testing.

SlugEnt.ResourceHealthChecker.FileSystem

File System Resource Health Checker

GitHub repositories (10)

Showing the top 5 popular GitHub repositories that depend on TestableIO.System.IO.Abstractions.TestingHelpers:

Repository Stars
gui-cs/Terminal.Gui
Cross Platform Terminal UI toolkit for .NET
stryker-mutator/stryker-net
Mutation testing for .NET core and .NET framework!
NethermindEth/nethermind
A robust execution client for Ethereum node operators.
recyclarr/recyclarr
Automatically sync TRaSH Guides to your Sonarr and Radarr instances
prom3theu5/aspirational-manifests
Handle deployments of .NET Aspire AppHost Projects
Version Downloads Last updated
21.2.1 6,116 12/28/2024
21.1.7 60,099 12/3/2024
21.1.3 137,749 11/8/2024
21.1.2 5,102 11/8/2024
21.1.1 3,008 11/7/2024
21.0.29 568,960 7/25/2024
21.0.26 104,060 7/13/2024
21.0.22 114,638 6/22/2024
21.0.2 622,635 3/17/2024
20.0.34 24,919 3/15/2024
20.0.28 37,602 3/9/2024
20.0.15 486,389 1/22/2024
20.0.4 276,823 12/5/2023
20.0.1 1,070 12/5/2023
19.2.91 177,251 12/5/2023
19.2.87 178,901 11/16/2023
19.2.69 519,035 8/29/2023
19.2.67 22,897 8/25/2023
19.2.66 1,146 8/25/2023
19.2.64 16,251 8/22/2023
19.2.63 1,372 8/22/2023
19.2.61 10,222 8/21/2023
19.2.51 101,853 7/31/2023
19.2.50 2,377 7/31/2023
19.2.29 525,142 5/17/2023
19.2.26 36,250 5/12/2023
19.2.25 1,291 5/12/2023
19.2.22 98,313 5/4/2023
19.2.18 72,977 4/24/2023
19.2.17 7,547 4/23/2023
19.2.16 18,175 4/19/2023
19.2.15 7,870 4/18/2023
19.2.13 1,322 4/18/2023
19.2.12 4,201 4/18/2023
19.2.11 28,974 4/13/2023
19.2.9 23,648 4/11/2023
19.2.8 2,199 4/11/2023
19.2.4 230,841 3/13/2023
19.2.1 90,521 3/2/2023
19.1.18 160,703 2/14/2023
19.1.14 57,988 1/31/2023
19.1.13 98,437 1/24/2023
19.1.5 231,344 12/19/2022
19.1.1 38,527 12/13/2022
19.0.1 54,692 12/8/2022
18.0.1 77,513 11/28/2022