Powershell для разрешения целевого пути соединения
В PowerShell мне нужно решить целевой путь соединения (symlink).
например, скажем, у меня есть соединение c:\someJunction
, целью которого является c:\temp\target
Я попробовал варианты $junc = Get-Item c:\someJunction
, но смог получить c:\someJunction
Как найти целевой путь соединения в этом примере c:\temp\target
заданного перехода?
Ответы
Ответ 1
Вы можете получить путь, выполнив следующие действия:
Get-ChildItem -Path C:\someJunction
Изменить для поиска пути, а не содержимого папки
Add-Type -MemberDefinition @"
private const int FILE_SHARE_READ = 1;
private const int FILE_SHARE_WRITE = 2;
private const int CREATION_DISPOSITION_OPEN_EXISTING = 3;
private const int FILE_FLAG_BACKUP_SEMANTICS = 0x02000000;
[DllImport("kernel32.dll", EntryPoint = "GetFinalPathNameByHandleW", CharSet = CharSet.Unicode, SetLastError = true)]
public static extern int GetFinalPathNameByHandle(IntPtr handle, [In, Out] StringBuilder path, int bufLen, int flags);
[DllImport("kernel32.dll", EntryPoint = "CreateFileW", CharSet = CharSet.Unicode, SetLastError = true)]
public static extern SafeFileHandle CreateFile(string lpFileName, int dwDesiredAccess, int dwShareMode,
IntPtr SecurityAttributes, int dwCreationDisposition, int dwFlagsAndAttributes, IntPtr hTemplateFile);
public static string GetSymbolicLinkTarget(System.IO.DirectoryInfo symlink)
{
SafeFileHandle directoryHandle = CreateFile(symlink.FullName, 0, 2, System.IntPtr.Zero, CREATION_DISPOSITION_OPEN_EXISTING, FILE_FLAG_BACKUP_SEMANTICS, System.IntPtr.Zero);
if(directoryHandle.IsInvalid)
throw new Win32Exception(Marshal.GetLastWin32Error());
StringBuilder path = new StringBuilder(512);
int size = GetFinalPathNameByHandle(directoryHandle.DangerousGetHandle(), path, path.Capacity, 0);
if (size<0)
throw new Win32Exception(Marshal.GetLastWin32Error());
// The remarks section of GetFinalPathNameByHandle mentions the return being prefixed with "\\?\"
// More information about "\\?\" here -> http://msdn.microsoft.com/en-us/library/aa365247(v=VS.85).aspx
if (path[0] == '\\' && path[1] == '\\' && path[2] == '?' && path[3] == '\\')
return path.ToString().Substring(4);
else
return path.ToString();
}
"@ -Name Win32 -NameSpace System -UsingNamespace System.Text,Microsoft.Win32.SafeHandles,System.ComponentModel
$dir = Get-Item D:\1
[System.Win32]::GetSymbolicLinkTarget($dir)
Ответ 2
New-Item, Remove-Item и Get-ChildItem были улучшены для поддержки создания и управления символическими ссылками. Параметр -ItemType для New-Item принимает новое значение, SymbolicLink. Теперь вы можете создавать символические ссылки в одну строку, выполнив командлет New-Item.
Что нового в Windows PowerShell v5
Я проверил поддержку символических ссылок на моем компьютере с Windows 7, он работает нормально.
PS> New-Item -Type SymbolicLink -Target C:\ -Name TestSymlink
Directory: C:\Users\skokhanovskiy\Desktop
Mode LastWriteTime Length Name
---- ------------- ------ ----
d----l 06.09.2016 18:27 TestSymlink
Получить цель символической ссылки так же просто, как создать ее.
> Get-Item .\TestSymlink | Select-Object -ExpandProperty Target
C:\
Ответ 3
Это делает трюк с меньшим количеством работы и работает даже для соединений на удаленном сервере:
fsutil reparsepoint query "M:\Junc"
Если вы хотите только имя цели:
fsutil reparsepoint query "M:\Junc" | where-object { $_ -imatch 'Print Name:' } | foreach-object { $_ -replace 'Print Name\:\s*','' }
так
function Get_JunctionTarget($p_path)
{
fsutil reparsepoint query $p_path | where-object { $_ -imatch 'Print Name:' } | foreach-object { $_ -replace 'Print Name\:\s*','' }
}
Кроме того, приведенный ниже код представляет собой небольшую модификацию кода, описанного выше Джошем. Его можно поместить в файл, который читается несколько раз, и он правильно управляет ведущим \\?\
в случае сетевого диска:
function Global:Get_UNCPath($l_dir)
{
if( ( ([System.Management.Automation.PSTypeName]'System.Win32').Type -eq $null) -or ([system.win32].getmethod('GetSymbolicLinkTarget') -eq $null) )
{
Add-Type -MemberDefinition @"
private const int CREATION_DISPOSITION_OPEN_EXISTING = 3;
private const int FILE_FLAG_BACKUP_SEMANTICS = 0x02000000;
[DllImport("kernel32.dll", EntryPoint = "GetFinalPathNameByHandleW", CharSet = CharSet.Unicode, SetLastError = true)]
public static extern int GetFinalPathNameByHandle(IntPtr handle, [In, Out] StringBuilder path, int bufLen, int flags);
[DllImport("kernel32.dll", EntryPoint = "CreateFileW", CharSet = CharSet.Unicode, SetLastError = true)]
public static extern SafeFileHandle CreateFile(string lpFileName, int dwDesiredAccess, int dwShareMode,
IntPtr SecurityAttributes, int dwCreationDisposition, int dwFlagsAndAttributes, IntPtr hTemplateFile);
public static string GetSymbolicLinkTarget(System.IO.DirectoryInfo symlink)
{
SafeFileHandle directoryHandle = CreateFile(symlink.FullName, 0, 2, System.IntPtr.Zero, CREATION_DISPOSITION_OPEN_EXISTING, FILE_FLAG_BACKUP_SEMANTICS, System.IntPtr.Zero);
if(directoryHandle.IsInvalid)
{
throw new Win32Exception(Marshal.GetLastWin32Error());
}
StringBuilder path = new StringBuilder(512);
int size = GetFinalPathNameByHandle(directoryHandle.DangerousGetHandle(), path, path.Capacity, 0);
if (size<0)
{
throw new Win32Exception(Marshal.GetLastWin32Error());
}
// The remarks section of GetFinalPathNameByHandle mentions the return being prefixed with "\\?\"
// More information about "\\?\" here -> http://msdn.microsoft.com/en-us/library/aa365247(v=VS.85).aspx
string sPath = path.ToString();
if( sPath.Length>8 && sPath.Substring(0,8) == @"\\?\UNC\" )
{
return @"\" + sPath.Substring(7);
}
else if( sPath.Length>4 && sPath.Substring(0,4) == @"\\?\" )
{
return sPath.Substring(4);
}
else
{
return sPath;
}
}
"@ -Name Win32 -NameSpace System -UsingNamespace System.Text,Microsoft.Win32.SafeHandles,System.ComponentModel
}
[System.Win32]::GetSymbolicLinkTarget($l_dir)
}
и учитывая вышеприведенную функцию Get_UNCPath
, мы можем улучшить функцию Get_JunctionTarget
следующим образом:
function Global:Get_JunctionTarget([string]$p_path)
{
$l_target = fsutil reparsepoint query $p_path | where-object { $_ -imatch 'Print Name\:' } | foreach-object { $_ -replace 'Print Name\:\s*','' }
if( $l_target -imatch "(^[A-Z])\:\\" )
{
$l_drive = $matches[1]
$l_uncPath = Get_UncPath $p_path
if( $l_uncPath -imatch "(^\\\\[^\\]*\\)" )
{
$l_machine = $matches[1]
$l_target = $l_target -replace "^$l_drive\:","$l_machine$l_drive$"
}
}
$l_target
}
Ответ 4
В результате мы используем эту функцию
function Get-SymlinkTargetDirectory {
[cmdletbinding()]
param(
[string]$SymlinkDir
)
$basePath = Split-Path $SymlinkDir
$folder = Split-Path -leaf $SymlinkDir
$dir = cmd /c dir /a:l $basePath | Select-String $folder
$dir = $dir -join ' '
$regx = $folder + '\ *\[(.*?)\]'
$Matches = $null
$found = $dir -match $regx
if ($found) {
if ($Matches[1]) {
Return $Matches[1]
}
}
Return ''
}
Ответ 5
По крайней мере, в PSv5 так просто, как этот, перечислить все цели некоторых ссылок dirs (или далее вниз по одной) и получить их как объекты и красиво отформатировать (например, все каталоги *~
самом деле являются переходами):
C:\Jaspersoft> ls | select name, target
Name Target
---- ------
apache-websrv~ {C:\Program Files (x86)\Apache24\}
jasperreports-server-cp-6.3.0 {}
jasperreports-server-cp~ {C:\Jaspersoft\jasperreports-server-cp-6.3.0}
jr-srv-cp~ {C:\Jaspersoft\jasperreports-server-cp~}
за одну ссылку:
C:\Jaspersoft> ls . apache-websrv~ | select name, target
Name Target
---- ------
apache-websrv~ {C:\Program Files (x86)\Apache24\}
или (просто получить Target as String для соединения C:\Jaspersoft\apache-websrv~
):
> ls C:\Jaspersoft apache-websrv~ | %{$_.target}
C:\Program Files (x86)\Apache24\
Стандартный ls
будет выглядеть следующим образом:
C:\Jaspersoft> ls
Verzeichnis: C:\Jaspersoft
Mode LastWriteTime Length Name
---- ------------- ------ ----
d----l 01.04.2019 15:05 apache-websrv~
d----- 02.04.2019 10:30 jasperreports-server-cp-6.3.0
d----l 05.10.2018 15:19 jasperreports-server-cp~
d----l 12.02.2019 11:46 jr-srv-cp~
(Другие ответы также содержали это в некотором роде, но не так легко были видны/понятны)