5

ファイルとフォルダーをその存続期間全体にわたって追跡しようとしています (それらは移動または名前変更されている可能性があります)。検索を行ったところ、FileSystemWatcherがファイルとフォルダーを追跡する最も一般的な方法であることがわかりましたが、アプリケーションが常に実行されている場合と実行されていない場合があるため、これはうまくいきません。ID を使用してフォルダーを追跡することにしました。

このスタック投稿への回答から、ID からファイルを追跡する方法を見つけました。この回答のアプローチ B に基づいて、ファイル ID を正常に取得できます。

検索中に、 FSCTL_GET_OBJECT_IDを使用して解決策を見つけたと述べているこのスタック投稿を見つけました。この関数の使用方法を理解するためにかなりの時間を費やしましたが、頭を包むことはできません。C# 内からネイティブ Windows 関数を呼び出した経験は基本的にありません。

誰かがこれを正しい方向に押し進めることができますか? 明らかな何かが欠けているに違いないと感じています。

C# がファイル/フォルダー ID にアクセスできない理由はありますか? ファイル/フォルダーの追跡は一般的ではありませんか?

編集、コードの追加:

        static uint returnVal;

    //Working example to get File ID
    public static string GetFileId(string path)
    {
        WinAPI.BY_HANDLE_FILE_INFORMATION objectFileInfo = new WinAPI.BY_HANDLE_FILE_INFORMATION();

        FileInfo fi = new FileInfo(path);
        FileStream fs = fi.Open(FileMode.Open, FileAccess.Read, FileShare.ReadWrite);

        WinAPI.GetFileInformationByHandle(fs.Handle, out objectFileInfo);

        fs.Close();

        ulong fileIndex = ((ulong)objectFileInfo.FileIndexHigh << 32) + (ulong)objectFileInfo.FileIndexLow;

        return fileIndex.ToString();
    }

    public static string GetFolderId(string path)
    {
        //Get a handle on the given folder
        IntPtr cFile = WinAPI.CreateFile(
            path,
            WinAPI.GENERIC_READ,
            FileShare.Read,
            IntPtr.Zero,
            (FileMode)WinAPI.OPEN_EXISTING,
            WinAPI.FILE_FLAG_BACKUP_SEMANTICS,
            IntPtr.Zero);

        Console.WriteLine(path);
        Console.WriteLine(cFile);

        if ((int)cFile != -1)
        {
            int cFileSize = Marshal.SizeOf(typeof(IntPtr));
            Console.WriteLine("cFile size = {0}", cFileSize);

            IntPtr cFileBlob = Marshal.AllocHGlobal(cFileSize);
            uint numBytesRead = 0;

            WinAPI.DeviceIoControl(cFile, WinAPI.FSCTL_GET_OBJECT_ID, IntPtr.Zero, 0, cFileBlob, (uint)cFileSize, ref numBytesRead, IntPtr.Zero);

            if (returnVal == 0)
            {
                Console.WriteLine(Marshal.GetLastWin32Error()); // Returning error 87 here
            }
        }

        //Should be returning the ID from the folder.
        return String.Empty;
    }

    public static void Main(string[] args)
    {
    Console.WriteLine(GetFileId(@"C:\Users\Matt\Desktop\TestDocument.txt"));
    Console.WriteLine(GetFolderId(@"C:\Users\Matt\Desktop"));
    }

}

class WinAPI
{
    // Win32 constants for accessing files.
    internal const int GENERIC_READ = unchecked((int)0x80000000);
    internal const int FILE_FLAG_BACKUP_SEMANTICS = unchecked((int)0x02000000);
    internal const int OPEN_EXISTING = unchecked((int)3);
    internal const int FSCTL_GET_OBJECT_ID = 0x0009009c;
    internal const int FSCTL_CREATE_OR_GET_OBJECT_ID = 0x000900c0;

    [DllImport("kernel32.dll", SetLastError = true)]
    public static extern bool DeviceIoControl(IntPtr hDevice, uint dwIoControlCode, IntPtr lpInBuffer, uint nInBufferSize, [Out] IntPtr lpOutBuffer, uint nOutBufferSize, ref uint lpBytesReturned, IntPtr lpOverlapped);

    [DllImport("kernel32.dll", SetLastError = true)]
    public static extern bool GetFileInformationByHandle(IntPtr hFile, out BY_HANDLE_FILE_INFORMATION lpFileInformation);

    [DllImport("kernel32.dll", SetLastError = true)]
    public static extern IntPtr CreateFile(
       String fileName,
       int dwDesiredAccess,
       System.IO.FileShare dwShareMode,
       IntPtr securityAttrs_MustBeZero,
       System.IO.FileMode dwCreationDisposition,
       int dwFlagsAndAttributes,
       IntPtr hTemplateFile_MustBeZero);

    public struct BY_HANDLE_FILE_INFORMATION
    {
        public uint FileAttributes;
        public FILETIME CreationTime;
        public FILETIME LastAccessTime;
        public FILETIME LastWriteTime;
        public uint VolumeSerialNumber;
        public uint FileSizeHigh;
        public uint FileSizeLow;
        public uint NumberOfLinks;
        public uint FileIndexHigh;
        public uint FileIndexLow;
    }
}

MSDN の投稿によると、無効なパラメーターである DeviceIoControl からエラー "87" が返されます (レピュテ​​ーションの制限により、これ以上リンクを投稿できません)。

4

1 に答える 1