diff options
| author | Sebastian Thiel <byronimo@gmail.com> | 2015-12-13 13:10:33 +0100 | 
|---|---|---|
| committer | Sebastian Thiel <byronimo@gmail.com> | 2015-12-13 13:10:33 +0100 | 
| commit | d82a6c5ed9108be5802a03c38f728a07da57438e (patch) | |
| tree | 22476136137e46bb0a08fea8cabcaa805f7c024b /git/objects/tree.py | |
| parent | a1a59764096cef4048507cb50f0303f48b87a242 (diff) | |
| download | gitpython-d82a6c5ed9108be5802a03c38f728a07da57438e.tar.gz | |
fix(tree): tree item sort now uses git-style
Previously it was possible to generate trees which didn't
appear legit to git as gitpython's sorting was a simple alpha-numeric
sort. Git uses one that minimizes literal string comparisons though,
and thus behaves slightly differently sometimes.
Fixes #369
Diffstat (limited to 'git/objects/tree.py')
| -rw-r--r-- | git/objects/tree.py | 50 | 
1 files changed, 49 insertions, 1 deletions
| diff --git a/git/objects/tree.py b/git/objects/tree.py index f9bee01e..24c88ee0 100644 --- a/git/objects/tree.py +++ b/git/objects/tree.py @@ -18,9 +18,53 @@ from .fun import (      tree_to_stream  ) +from gitdb.utils.compat import PY3 + +if PY3: +    cmp = lambda a, b: (a > b) - (a < b) +  __all__ = ("TreeModifier", "Tree") +def git_cmp(t1, t2): +    a, b = t1[2], t2[2] +    len_a, len_b = len(a), len(b) +    min_len = min(len_a, len_b) +    min_cmp = cmp(a[:min_len], b[:min_len]) + +    if min_cmp: +        return min_cmp + +    # return len_a - len_b +    return len_b - len_a + +if PY3: +    # taken from https://wiki.python.org/moin/HowTo/Sorting#The_Old_Way_Using_the_cmp_Parameter +    class CmpToKey(object): +        __slots__ = 'obj' + +        def __init__(self, obj, *args): +            self.obj = obj + +        def __lt__(self, other): +            return git_cmp(self.obj, other.obj) < 0 + +        def __gt__(self, other): +            return git_cmp(self.obj, other.obj) > 0 + +        def __eq__(self, other): +            return git_cmp(self.obj, other.obj) == 0 + +        def __le__(self, other): +            return git_cmp(self.obj, other.obj) <= 0 + +        def __ge__(self, other): +            return git_cmp(self.obj, other.obj) >= 0 + +        def __ne__(self, other): +            return git_cmp(self.obj, other.obj) != 0 + +  class TreeModifier(object):      """A utility class providing methods to alter the underlying cache in a list-like fashion. @@ -47,7 +91,10 @@ class TreeModifier(object):          It may be called several times, but be aware that each call will cause          a sort operation          :return self:""" -        self._cache.sort(key=lambda t: t[2])    # sort by name +        if PY3: +            self._cache.sort(key=CmpToKey) +        else: +            self._cache.sort(cmp=git_cmp)          return self      #} END interface @@ -286,3 +333,4 @@ class Tree(IndexObject, diff.Diffable, util.Traversable, util.Serializable):  # finalize map definition  Tree._map_id_to_type[Tree.tree_id] = Tree +# | 
