TestableIO.System.IO.Abstractions.Wrappers 21.1.3

Prefix Reserved
dotnet add package TestableIO.System.IO.Abstractions.Wrappers --version 21.1.3                
NuGet\Install-Package TestableIO.System.IO.Abstractions.Wrappers -Version 21.1.3                
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.Wrappers" Version="21.1.3" />                
For projects that support PackageReference, copy this XML node into the project file to reference the package.
paket add TestableIO.System.IO.Abstractions.Wrappers --version 21.1.3                
#r "nuget: TestableIO.System.IO.Abstractions.Wrappers, 21.1.3"                
#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.Wrappers as a Cake Addin
#addin nuget:?package=TestableIO.System.IO.Abstractions.Wrappers&version=21.1.3

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

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 (19)

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

Package Downloads
System.IO.Abstractions

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.

FluentStorage

FluentStorage, originally known as Storage.NET, is a polycloud .NET cloud storage library to interface with multiple cloud providers from a single unified interface. Provides Blob storage (AWS S3, GCP, FTP, SFTP, Azure Blob/File/Event Hub/Data Lake) and Messaging (AWS SQS, Azure Queue/ServiceBus). Supports .NET 5+ and .NET Standard 2.0+. Pure C#. MIT license. Commercial use allowed.

stryker

All stryker mutation test logic is contained in this library. This package does not include a runner. Use this package if you want to extend stryker with your own runner.

AppMotor.Core

The core APIs for the AppMotor framework.

GitHub repositories (8)

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

Repository Stars
Kareadita/Kavita
Kavita is a fast, feature rich, cross platform reading server. Built with the goal of being a full solution for all your reading needs. Setup your own server and share your reading collection with your friends and family.
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.1.3 41,994 11/8/2024
21.1.2 3,451 11/8/2024
21.1.1 4,581 11/7/2024
21.0.29 1,421,417 7/25/2024
21.0.26 272,363 7/13/2024
21.0.22 284,055 6/22/2024
21.0.2 1,722,367 3/17/2024
20.0.34 48,095 3/15/2024
20.0.28 144,044 3/9/2024
20.0.15 1,457,998 1/22/2024
20.0.4 839,997 12/5/2023
20.0.1 2,115 12/5/2023
19.2.91 262,088 12/5/2023
19.2.87 692,718 11/16/2023
19.2.69 2,384,749 8/29/2023
19.2.67 45,348 8/25/2023
19.2.66 2,444 8/25/2023
19.2.64 99,909 8/22/2023
19.2.63 2,970 8/22/2023
19.2.61 19,073 8/21/2023
19.2.51 295,170 7/31/2023
19.2.50 5,969 7/31/2023
19.2.29 1,936,642 5/17/2023
19.2.26 65,047 5/12/2023
19.2.25 3,216 5/12/2023
19.2.22 199,250 5/4/2023
19.2.18 259,566 4/24/2023
19.2.17 12,267 4/23/2023
19.2.16 138,404 4/19/2023
19.2.15 369,220 4/18/2023
19.2.13 2,989 4/18/2023
19.2.12 6,321 4/18/2023
19.2.11 82,944 4/13/2023
19.2.9 62,528 4/11/2023
19.2.8 7,683 4/11/2023
19.2.4 1,216,704 3/13/2023
19.2.1 393,680 3/2/2023
19.1.18 419,411 2/14/2023
19.1.14 185,616 1/31/2023
19.1.13 403,434 1/24/2023
19.1.5 2,008,189 12/19/2022
19.1.1 89,448 12/13/2022
19.0.1 208,757 12/8/2022
18.0.1 465,830 11/28/2022
17.2.26 276,663 11/18/2022 17.2.26 is deprecated.