Проблема завершения вкладки относительного пути поставщика PowerShell - PullRequest
2 голосов
/ 15 марта 2012

Я реализовал простой PowerShell NavigationCmdletProvider.

Для тех, кто не знает, это означает, что я могу создать оснастку с командлетом, который фактически является диском виртуальной файловой системы; этот диск можно подключить и перейти из PowerShell, как и в любую обычную папку. Каждое действие с диском (например, проверка, указывает ли путь на допустимый элемент, получение списка имен дочерних элементов в папке и т. Д.), Сопоставляется с методом класса .NET, унаследованным от класса NavigationCmdletProvider .

Я столкнулся с проблемой с завершением табуляции и хотел бы найти решение. Я обнаружил, что завершение табуляции дает неверные результаты при использовании относительных путей. Для абсолютных путей это работает отлично.

Для тех, кто не знает, завершение табуляции для NavigationCmdletProvider работает через PowerShell, вызывая метод GetChildNames, который переопределяется из класса NavigationCmdletProvider.

- Демонстрация выпуска -

Предположим, у меня есть провайдер TEST со следующей иерархией папок:

TEST::child1
TEST::child1\child1a
TEST::child1\child1b
TEST::child2
TEST::child2\child2a
TEST::child2\child2b
TEST::child3
TEST::child3\child3a
TEST::child3\child3b

Абсолютные пути:

Если я наберу "dir TEST::child1\" и нажму tab несколько раз, это даст мне ожидаемые результаты:

> dir TEST::child1\child1a
> dir TEST::child1\child1b

Относительные пути:

Сначала я перехожу к "TEST :: child1":

> cd TEST::child1

Затем, если я наберу "dir пробел " и несколько раз нажму tab , это даст мне неверные результаты:

> dir .\child1\child1a
> dir .\child1\child1b

Я ожидаю увидеть это вместо:

> dir .\child1a
> dir .\child1b

Это ошибка в PowerShell или я что-то не так делаю?

Вот полный, автономный код для провайдера:

[CmdletProvider("TEST", ProviderCapabilities.None)]
public class MyTestProvider : NavigationCmdletProvider
{
    private Node m_Root;
    private void ConstructTestHierarchy()
    {
        //
        // Create the nodes
        //
        Node root = new Node("");
            Node child1 = new Node("child1");
                Node child1a = new Node("child1a");
                Node child1b = new Node("child1b");
            Node child2 = new Node("child2");
                Node child2a = new Node("child2a");
                Node child2b = new Node("child2b");
            Node child3 = new Node("child3");
                Node child3a = new Node("child3a");
                Node child3b = new Node("child3b");

        //
        // Construct node hierarchy
        //
        m_Root = root;
            root.AddChild(child1);
                child1.AddChild(child1a);
                child1.AddChild(child1b);
            root.AddChild(child2);
                child2.AddChild(child2a);
                child2.AddChild(child2b);
            root.AddChild(child3);
                child3.AddChild(child3a);
                child3.AddChild(child3b);
    }

    public MyTestProvider()
    {
        ConstructTestHierarchy();
    }

    protected override bool IsValidPath(string path)
    {
        return m_Root.ItemExistsAtPath(path);
    }

    protected override bool ItemExists(string path)
    {
        return m_Root.ItemExistsAtPath(path);
    }

    protected override void GetChildNames(string path, ReturnContainers returnContainers)
    {
        var children = m_Root.GetItemAtPath(path).Children;
        foreach (var child in children)
        {
            WriteItemObject(child.Name, child.Name, true);
        }
    }
    protected override bool IsItemContainer(string path)
    {
        return true;
    }
    protected override void GetChildItems(string path, bool recurse)
    {
        var children = m_Root.GetItemAtPath(path).Children;
        foreach (var child in children)
        {
            WriteItemObject(child.Name, child.Name, true);
        }
    }
}

/// <summary>
/// This is a node used to represent a folder inside a PowerShell provider
/// </summary>
public class Node
{
    private string m_Name;
    private List<Node> m_Children;

    public string Name { get { return m_Name; } }
    public ICollection<Node> Children { get { return m_Children; } }

    public Node(string name)
    {
        m_Name = name;
        m_Children = new List<Node>();
    }

    /// <summary>
    /// Adds a node to this node's list of children
    /// </summary>
    public void AddChild(Node node)
    {
        m_Children.Add(node);
    }
    /// <summary>
    /// Test whether a string matches a wildcard string ('*' must be at end of wildcardstring)
    /// </summary>
    private bool WildcardMatch(string basestring, string wildcardstring)
    {
        //
        // If wildcardstring has no *, just do a string comparison
        //
        if (!wildcardstring.Contains('*'))
        {
            return String.Equals(basestring, wildcardstring);
        }
        else
        {
            //
            // If wildcardstring is really just '*', then any name works
            //
            if (String.Equals(wildcardstring, "*"))
                return true;

            //
            // Given the wildcardstring "abc*", we just need to test if basestring starts with "abc"
            //
            string leftOfAsterisk = wildcardstring.Split(new char[] { '*' })[0];
            return basestring.StartsWith(leftOfAsterisk);

        }
    }

    /// <summary>
    /// Recursively check if "child1\child2\child3" exists
    /// </summary>
    public bool ItemExistsAtPath(string path)
    {
        //
        // If path is self, return self
        //
        if (String.Equals(path, "")) return true;

        //
        // If path has no slashes, test if it matches the child name
        //
        if(!path.Contains(@"\"))
        {
            //
            // See if any children have this name
            //
            foreach (var child in m_Children)
            {
                if (WildcardMatch(child.Name, path))
                    return true;
            }
            return false;
        }
        else
        {
            //
            // Split the path
            //
            string[] pathChunks = path.Split(new char[] { '\\' }, StringSplitOptions.RemoveEmptyEntries);

            //
            // Take out the first chunk; this is the child we're going to search
            //
            string nextChild = pathChunks[0];

            //
            // Combine the rest of the path; this is the path we're going to provide to the child
            //
            string nextPath = String.Join(@"\", pathChunks.Skip(1).ToArray());

            //
            // Recurse into child
            //
            foreach (var child in m_Children)
            {
                if (String.Equals(child.Name, nextChild))
                    return child.ItemExistsAtPath(nextPath);
            }
            return false;
        }
    }

    /// <summary>
    /// Recursively fetch "child1\child2\child3" 
    /// </summary>
    public Node GetItemAtPath(string path)
    {
        //
        // If path is self, return self
        //
        if (String.Equals(path, "")) return this;

        //
        // If path has no slashes, test if it matches the child name
        //
        if (!path.Contains(@"\"))
        {
            //
            // See if any children have this name
            //
            foreach (var child in m_Children)
            {
                if (WildcardMatch(child.Name, path))
                    return child;
            }
            throw new ApplicationException("Child doesn't exist!");
        }
        else
        {
            //
            // Split the path
            //
            string[] pathChunks = path.Split(new char[] { '\\' }, StringSplitOptions.RemoveEmptyEntries);

            //
            // Take out the first chunk; this is the child we're going to search
            //
            string nextChild = pathChunks[0];

            //
            // Combine the rest of the path; this is the path we're going to provide to the child
            //
            string nextPath = String.Join(@"\", pathChunks.Skip(1).ToArray());

            //
            // Recurse into child
            //
            foreach (var child in m_Children)
            {
                if (String.Equals(child.Name, nextChild))
                    return child.GetItemAtPath(nextPath);
            }
            throw new ApplicationException("Child doesn't exist!");
        }
    }
}

Ответы [ 4 ]

1 голос
/ 09 ноября 2012

Не уверен, что это ошибка, Я нашел этот обходной путь , который, кажется, "делает работу". (Небольшое обновление, оказалось, что мой оригинальный код будет "выдавать ошибку" при работе на нескольких уровнях.

''' <summary>
''' Joins two strings with a provider specific path separator.
''' </summary>
''' <param name="parent">The parent segment of a path to be joined with the child.</param>
''' <param name="child">The child segment of a path to be joined with the parent.</param>
''' <returns>A string that contains the parent and child segments of the path joined by a path separator.</returns>
''' <remarks></remarks>
Protected Overrides Function MakePath(parent As String, child As String) As String
    Trace.WriteLine("::MakePath(parent:=" & parent & ",child:=" & child & ")")
    Dim res As String = MyBase.MakePath(parent, child)
    Trace.WriteLine("::MakePath(parent:=" & parent & ",child:=" & child & ") " & res)
    If parent = "." Then
        'res = ".\" & child.Split("\").Last
        If String.IsNullOrEmpty(Me.SessionState.Path.CurrentLocation.ProviderPath) Then
        res = parent & PATH_SEPARATOR & child
        Else
        res = parent & PATH_SEPARATOR & child.Substring(Me.SessionState.Path.CurrentLocation.ProviderPath.Length + 1)
        'res = parent & PATH_SEPARATOR & child.Replace(Me.SessionState.Path.CurrentLocation.ProviderPath & PATH_SEPARATOR, String.Empty)
        End If
        Trace.WriteLine("::**** TRANSFORM: " & res)
    End If
    Return res
End Function
1 голос
/ 18 ноября 2012

Вы можете обойти эту проблему, если спроектируете своего провайдера так, чтобы он ожидал ввода непустого корня при создании нового диска.Я заметил, что по завершению табуляции ошибочно предлагается полный дочерний путь вместо только дочернее имя , если свойство Root PSDriveInfo не было установлено.

Можетбыть ограничивающим для некоторых поставщиков всегда требовать непустой корень.Обходное решение выше работает хорошо, если вы не хотите, чтобы пользователи всегда вводили некоторые Root при создании нового диска.

0 голосов
/ 27 сентября 2018

Мне удалось заставить его работать с перевесом string[] ExpandPath(string path) и настройкой возможностей ProviderCapabilities.ExpandWildcards.

0 голосов
/ 07 июня 2012

Я перечислил это как ошибку поставщика PowerShell в Microsoft Connect: Проблема с относительным завершением пути (через Get-ChildNames) для NavigationCmdletProvider

Если кто-то может воспроизвести этоПожалуйста, перейдите по ссылке и скажите, потому что Microsoft, вероятно, не будет рассматривать это, если только один человек сообщит об этом.

Похоже, это исправлено в PowerShell 3.0.Я не знаю, почему Microsoft не хочет исправлять это в старых версиях, это не то, от чего может зависеть какой-либо код.

...