В .NET, Как получить цель символической ссылки (или Reparse Point)? - PullRequest
8 голосов
/ 20 февраля 2010

В .NET я думаю, что могу определить, является ли файл символической ссылкой, вызвав System.IO.File.GetAttributes () и проверив наличие бита ReparsePoint. вот так:

var a = System.IO.File.GetAttributes(fileName);
if ((a & FileAttributes.ReparsePoint) != 0)
{
    // it's a symlink
}

Как я могу получить цель символической ссылки, в этом случае?


пс: я знаю, как создать символическую ссылку. Требуется P / Invoke:

[Interop.DllImport("kernel32.dll", EntryPoint="CreateSymbolicLinkW", CharSet=Interop.CharSet.Unicode)] 
public static extern int CreateSymbolicLink(string lpSymlinkFileName, string lpTargetFileName, int dwFlags); 

Ответы [ 3 ]

11 голосов
/ 03 ноября 2015

Основываясь на ответе, который упомянул GetFinalPathNameByHandle, вот код C #, который делает это (поскольку все остальные ответы были просто указателями):

Использование

var path = NativeMethods.GetFinalPathName(@"c:\link");

Код:

public static class NativeMethods
{
    private static readonly IntPtr INVALID_HANDLE_VALUE = new IntPtr(-1);

    private const uint FILE_READ_EA = 0x0008;
    private const uint FILE_FLAG_BACKUP_SEMANTICS = 0x2000000;

    [DllImport("Kernel32.dll", SetLastError = true, CharSet = CharSet.Auto)]
    static extern uint GetFinalPathNameByHandle(IntPtr hFile, [MarshalAs(UnmanagedType.LPTStr)] StringBuilder lpszFilePath, uint cchFilePath, uint dwFlags);

    [DllImport("kernel32.dll", SetLastError = true)]
    [return: MarshalAs(UnmanagedType.Bool)]
    static extern bool CloseHandle(IntPtr hObject);

    [DllImport("kernel32.dll", CharSet = CharSet.Auto, SetLastError = true)]
    public static extern IntPtr CreateFile(
            [MarshalAs(UnmanagedType.LPTStr)] string filename,
            [MarshalAs(UnmanagedType.U4)] uint access,
            [MarshalAs(UnmanagedType.U4)] FileShare share,
            IntPtr securityAttributes, // optional SECURITY_ATTRIBUTES struct or IntPtr.Zero
            [MarshalAs(UnmanagedType.U4)] FileMode creationDisposition,
            [MarshalAs(UnmanagedType.U4)] uint flagsAndAttributes,
            IntPtr templateFile);

    public static string GetFinalPathName(string path)
    {
        var h = CreateFile(path, 
            FILE_READ_EA, 
            FileShare.ReadWrite | FileShare.Delete, 
            IntPtr.Zero, 
            FileMode.Open, 
            FILE_FLAG_BACKUP_SEMANTICS,
            IntPtr.Zero);
        if (h == INVALID_HANDLE_VALUE)
            throw new Win32Exception();

        try
        {
            var sb = new StringBuilder(1024);
            var res = GetFinalPathNameByHandle(h, sb, 1024, 0);
            if (res == 0)
                throw new Win32Exception();

            return sb.ToString();
        }
        finally
        {
            CloseHandle(h);
        }
    }
}
9 голосов
/ 20 февраля 2010

Вы должны использовать DeviceIoControl () и отправить управляющий код FSCTL_GET_REPARSE_POINT. Детали использования P / Invoke и API довольно грубые, но он действительно хорошо гуглит .

4 голосов
/ 20 февраля 2010

Откройте файл с помощью CreateFile, а затем передайте дескриптор GetFinalPathNameByHandle .

...