TestableIO.System.IO.Abstractions 21.2.1

Prefix Reserved
dotnet add package TestableIO.System.IO.Abstractions --version 21.2.1                
NuGet\Install-Package TestableIO.System.IO.Abstractions -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" 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 --version 21.2.1                
#r "nuget: TestableIO.System.IO.Abstractions, 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 as a Cake Addin
#addin nuget:?package=TestableIO.System.IO.Abstractions&version=21.2.1

// Install TestableIO.System.IO.Abstractions as a Cake Tool
#tool nuget:?package=TestableIO.System.IO.Abstractions&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.  net9.0 was computed.  net9.0-android was computed.  net9.0-browser was computed.  net9.0-ios was computed.  net9.0-maccatalyst was computed.  net9.0-macos was computed.  net9.0-tvos was computed.  net9.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.
  • .NETFramework 4.6.2

    • No dependencies.
  • .NETStandard 2.0

    • No dependencies.
  • .NETStandard 2.1

    • No dependencies.
  • net6.0

    • No dependencies.
  • net7.0

    • No dependencies.
  • net8.0

    • No dependencies.

NuGet packages (24)

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

Package Downloads
System.IO.Abstractions

A set of abstractions to help make file system interactions testable.

TestableIO.System.IO.Abstractions.Wrappers

A set of abstractions to help make file system interactions testable.

TestableIO.System.IO.Abstractions.TestingHelpers

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

Testably.Abstractions.Interface

Interfaces for `Testably.Abstractions` to write testable code by abstracting away system dependencies.

TestableIO.System.IO.Abstractions.Extensions

Convenience functionalities on top of System.IO.Abstractions

GitHub repositories (5)

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

Repository Stars
recyclarr/recyclarr
Automatically sync TRaSH Guides to your Sonarr and Radarr instances
ChaosRecipeEnhancer/ChaosRecipeEnhancer
🟡📈 Streamline your Chaos Recipe gains!
octgn/OCTGN
Online Card and Tabletop Gaming Network
bottlenoselabs/c2cs
Generate C# bindings from a C header.
ZarehD/AspNetStatic
Transform ASP.NET Core into a static site generator.
Version Downloads Last updated
21.2.1 105,122 12/28/2024
21.1.7 229,840 12/3/2024
21.1.3 456,613 11/8/2024
21.1.2 7,868 11/8/2024 21.1.2 is deprecated because it has critical bugs.
21.1.1 7,032 11/7/2024
21.0.29 2,093,397 7/25/2024
21.0.26 331,715 7/13/2024
21.0.22 365,164 6/22/2024
21.0.2 1,961,101 3/17/2024
20.0.34 56,425 3/15/2024
20.0.28 156,147 3/9/2024
20.0.15 1,592,599 1/22/2024
20.0.4 1,657,412 12/5/2023
20.0.1 2,528 12/5/2023
19.2.91 280,466 12/5/2023
19.2.87 725,481 11/16/2023
19.2.69 2,553,069 8/29/2023
19.2.67 48,340 8/25/2023
19.2.66 2,907 8/25/2023
19.2.64 116,235 8/22/2023
19.2.63 3,426 8/22/2023
19.2.61 20,685 8/21/2023
19.2.51 311,641 7/31/2023
19.2.50 6,563 7/31/2023
19.2.29 2,013,951 5/17/2023
19.2.26 72,484 5/12/2023
19.2.25 3,699 5/12/2023
19.2.22 204,969 5/4/2023
19.2.18 263,311 4/24/2023
19.2.17 18,737 4/23/2023
19.2.16 140,142 4/19/2023
19.2.15 375,083 4/18/2023
19.2.13 3,478 4/18/2023
19.2.12 7,098 4/18/2023
19.2.11 98,484 4/13/2023
19.2.9 67,723 4/11/2023
19.2.8 8,159 4/11/2023
19.2.4 1,284,357 3/13/2023
19.2.1 405,381 3/2/2023
19.1.18 430,677 2/14/2023
19.1.14 197,560 1/31/2023
19.1.13 413,073 1/24/2023
19.1.5 2,213,553 12/19/2022
19.1.1 99,390 12/13/2022
19.0.1 222,461 12/8/2022
18.0.1 480,946 11/28/2022
17.2.26 310,045 11/18/2022 17.2.26 is deprecated.