Click here to Skip to main content
15,867,973 members
Articles / Programming Languages / C#

Accessing alternative data-streams of files on an NTFS volume

Rate me:
Please Sign up or sign in to vote.
4.85/5 (53 votes)
15 Aug 2016CPOL4 min read 500.1K   5.4K   132   86
A pair of classes to encapsulate access to NTFS alternative data streams.

Introduction

Since NT 3.1, the NTFS file system has supported multiple data-streams for files. There has never been built-in support for viewing or manipulating these additional streams, but the Windows API functions include support for them with a special file syntax: Filename.ext:StreamName. Even Win9x machines can access the alternative data streams of files on any NTFS volume they have access to, e.g., through a mapped drive. Because the Scripting.FileSystemObject and many other libraries call the CreateFile API behind the scenes, even scripts have been able to access alternative streams quite easily (although enumerating the existing streams has always been tricky).

In .NET, however, it seems someone decided to add some checking to the format of filenames. If you attempt to open a FileStream on an alternative stream, you will get a "Path Format not supported" exception. I have been unable to find any class in the CLR that provides support for alternative data streams, so I decided to roll my own.

Update

I originally wrote this code in 2002, targeting v1 of the .NET Framework. Looking at the code now, it seems quite messy, and has several bugs and problems which were mentioned in the comments. I have since completely re-written the code for .NET v3.5, and (hopefully) fixed the bugs.

The new code is not compatible with the original version. However, I have included a sample compatibility wrapper which maps the old API to the new API. You can find these files under the "other/Compatibility wrapper" folder in the download.

Bugs / Issues Fixed

  • The code now uses the FileSystemInfo class rather than the FileInfo class. This allows you to access alternate data streams attached to folders as well as files. (Suggested by Dan Elebash.)
  • The code now uses SafeFileHandle instead of IntPtr for the file handle. (Suggested by Moomansun.)
  • The code is now 64-bit compatible. (Reported by John SMith 5634552745.)
  • The code now correctly calls BackupRead with the bAbort parameter set to true after enumerating the streams. (Reported by scooter_jsm.)
  • The number of global memory allocations required to read the streams from a file has been reduced. (Suggested by scooter_jsm.)
  • v2.1: ValidateStreamName now accepts stream names which contain characters with ASCII codes between 1 and 31. (Reported by Andy Missico.)
  • v2.1: The code will now attempt to map standard IO errors to the equivalent .NET Exception type. (Suggested by Andy Missico.)

Using the Classes

The AlternateDataStreamInfo class represents the details of an individual stream, and provides methods to create, open, or delete the stream.

The static FileSystem class provides methods to retrieve the list of streams for a file, retrieve a specific stream from a file, determine whether a stream exists, and delete a specific stream.

All methods on the FileSystem class offer overloads which accept either a path or a FileSystemInfo object. The overloads which accept a FileSystemInfo object can also be invoked as extension methods.

Example:

C#
using System;
using System.Drawing;
using System.Drawing.Imaging;
using System.IO;
using Trinet.Core.IO.Ntfs;

...

FileInfo file = new FileInfo(path);

// List the additional streams for a file:
foreach (AlternateDataStreamInfo s in file.ListAlternateDataStreams())
{
    Console.WriteLine("{0} - {1} bytes", s.Name, s.Size);
}

// Read the "Zone.Identifier" stream, if it exists:
if (file.AlternateDataStreamExists("Zone.Identifier"))
{
    Console.WriteLine("Found zone identifier stream:");
    
    AlternateDataStreamInfo s = 
       file.GetAlternateDataStream("Zone.Identifier",
                                   FileMode.Open);
    using (TextReader reader = s.OpenText())
    {
        Console.WriteLine(reader.ReadToEnd());
    }
    
    // Delete the stream:
    s.Delete();
}
else
{
    Console.WriteLine("No zone identifier stream found.");
}

// Alternative method to delete the stream:
file.DeleteAlternateDataStream("Zone.Identifier");

Files Included

  • The Trinet.Core.IO.Ntfs folder contains the source code.
  • The doc folder contains the documentation and FxCop project.
  • The bin folder contains the compiled assembly.
  • The other folder contains the compatibility wrapper, and a sample to recursively delete the "Zone.Identifier" stream from all files in a given path.

References

If you want more information on NTFS programming, or the C++ code I based this on, see Dino Esposito's MSDN article from March 2000: http://msdn.microsoft.com/en-us/library/ms810604.aspx [^].

History

  • v1 - 1 August 2002 - Initial release, targeting .NET 1.0.
  • v2 - 16 September 2008 - Re-written to target .NET 3.5; major changes include:
    • Replaced FileInfo with FileSystemInfo.
    • Replaced IntPtr with SafeFileHandle.
    • Made the P/Invoke calls 64-bit compatible.
    • Made the code abort the BackupRead operation after enumerating the streams.
    • Reduced the number of global memory allocations required.
  • v2.1 - 28 July 2010 - Minor fixes suggested by Andy Missico:
    • Changed ValidateStreamName to accept stream names containing characters with ASCII codes between 1 and 31 (see msdn.microsoft.com/en-us/library/aa365247).
    • Added mapping of standard IO errors to the correct .NET exceptions.
  • v2.2 - 15 August 2016 - Fix for bug reported by Member 12683101, where zero-length alternate streams prevented subsequent streams from being enumerated.

License

This article, along with any associated source code and files, is licensed under The Code Project Open License (CPOL)


Written By
Software Developer CodeProject
United Kingdom United Kingdom
I started writing code when I was 8, with my trusty ZX Spectrum and a subscription to "Input" magazine. Spent many a happy hour in the school's computer labs with the BBC Micros and our two DOS PCs.

After a brief detour into the world of Maths, I found my way back into programming during my degree via free copies of Delphi and Visual C++ given away with computing magazines.

I went straight from my degree into my first programming job, at Trinet Ltd. Eleven years later, the company merged to become ArcomIT. Three years after that, our project manager left to set up Nevalee Business Solutions, and took me with him. Since then, we've taken on four more members of staff, and more work than you can shake a stick at. Smile | :)

Between writing custom code to integrate with Visma Business, developing web portals to streamline operations for a large multi-national customer, and maintaining RedAtlas, our general aviation airport management system, there's certainly never a dull day in the office!

Outside of work, I enjoy real ale and decent books, and when I get the chance I "tinkle the ivories" on my Technics organ.

Comments and Discussions

 
GeneralRe: another question Pin
#realJSOP11-Feb-11 1:51
mve#realJSOP11-Feb-11 1:51 
GeneralFileInfo question Pin
Brendan Chong16-Sep-09 7:33
Brendan Chong16-Sep-09 7:33 
GeneralRe: FileInfo question Pin
Richard Deeming16-Sep-09 7:49
mveRichard Deeming16-Sep-09 7:49 
GeneralEdit File summary Pin
Joe Rozario29-Jun-09 23:55
Joe Rozario29-Jun-09 23:55 
GeneralRe: Edit File summary Pin
Richard Deeming30-Jun-09 1:42
mveRichard Deeming30-Jun-09 1:42 
GeneralRe: Edit File summary Pin
Joe Rozario1-Jul-09 1:33
Joe Rozario1-Jul-09 1:33 
GeneralPlz simple sample code [modified] Pin
moris200069-Jun-09 18:55
moris200069-Jun-09 18:55 
GeneralRe: Plz simple sample code Pin
Richard Deeming10-Jun-09 1:20
mveRichard Deeming10-Jun-09 1:20 
GeneralRe: Plz simple sample code Pin
moris2000610-Jun-09 3:28
moris2000610-Jun-09 3:28 
GeneralBUG: Does not work in Vista SP1 Pin
ini18328-Apr-09 20:48
ini18328-Apr-09 20:48 
GeneralRe: BUG: Does not work in Vista SP1 Pin
Richard Deeming29-Apr-09 1:21
mveRichard Deeming29-Apr-09 1:21 
GeneralRe: BUG: Does not work in Vista SP1 Pin
ini18329-Apr-09 2:19
ini18329-Apr-09 2:19 
GeneralVery nice Pin
Sergey Morenko25-Sep-08 12:51
professionalSergey Morenko25-Sep-08 12:51 
GeneralBug: BackupRead() documentation points to bug in this code. Pin
StCroixSkipper16-Sep-08 7:40
StCroixSkipper16-Sep-08 7:40 
GeneralRe: Bug: BackupRead() documentation points to bug in this code. Pin
Richard Deeming19-Sep-08 8:33
mveRichard Deeming19-Sep-08 8:33 
GeneralNot for 64-bit Pin
John SMith 563455274511-Apr-07 17:56
John SMith 563455274511-Apr-07 17:56 
NewsRe: Not for 64-bit Pin
unRheal21-May-09 9:19
unRheal21-May-09 9:19 
GeneralUpdated NTFS.dll to .NET 2.0 Pin
Moomansun23-Aug-06 14:01
Moomansun23-Aug-06 14:01 
GeneralAttach ADS to folder Pin
Dan Elebash30-Jul-04 11:57
Dan Elebash30-Jul-04 11:57 
GeneralRe: Attach ADS to folder Pin
Richard Deeming5-Aug-04 4:53
mveRichard Deeming5-Aug-04 4:53 
Generalreturns null stream when open subStream within ntfs main stream Pin
rajesh patel14-Jan-04 12:08
rajesh patel14-Jan-04 12:08 
GeneralRe: returns null stream when open subStream within ntfs main stream Pin
Richard Deeming19-Jan-04 7:56
mveRichard Deeming19-Jan-04 7:56 
GeneralRe: returns null stream when open subStream within ntfs main stream Pin
Anonymous26-Jan-04 9:49
Anonymous26-Jan-04 9:49 
GeneralBug Fix Pin
jfos14-Jan-04 4:32
jfos14-Jan-04 4:32 
GeneralNTFS limitation Pin
Warren Stevens1-Aug-02 6:58
Warren Stevens1-Aug-02 6:58 

General General    News News    Suggestion Suggestion    Question Question    Bug Bug    Answer Answer    Joke Joke    Praise Praise    Rant Rant    Admin Admin   

Use Ctrl+Left/Right to switch messages, Ctrl+Up/Down to switch threads, Ctrl+Shift+Left/Right to switch pages.