Browse Source

Handle leading **/.

Jelmer Vernooij 7 years ago
parent
commit
2b73b1c708
2 changed files with 15 additions and 8 deletions
  1. 12 7
      dulwich/ignore.py
  2. 3 1
      dulwich/tests/test_ignore.py

+ 12 - 7
dulwich/ignore.py

@@ -17,14 +17,12 @@
 # License, Version 2.0.
 #
 
-"""Parsing of gitignore files."""
+"""Parsing of gitignore files.
 
-# TODO(jelmer): Handle ignore files in subdirectories
-# TODO(jelmer): Handle ** in patterns
-# TODO(jelmer): Compile patterns
+For details for the matching rules, see https://git-scm.com/docs/gitignore
+"""
 
 import re
-import posixpath
 
 
 def translate(pat):
@@ -32,12 +30,19 @@ def translate(pat):
 
     There is no way to quote meta-characters.
 
-    Originally copied from fnmatch in Python 2.7.
+    Originally copied from fnmatch in Python 2.7, but modified for Dulwich
+    to cope with features in Git ignore patterns.
     """
 
     res = ''
 
     if not '/' in pat:
+        # If there's no slash, this is a filename-based match
+        res = '(.*\/)?'
+
+    if pat.startswith('**/'):
+        # Leading **/
+        pat = pat[2:]
         res = '(.*\/)?'
 
     if pat.startswith(b'/'):
@@ -49,7 +54,7 @@ def translate(pat):
         c = pat[i]
         i = i+1
         if c == '*':
-            if i < n and pat[i] == '*':
+            if i < n and pat[i:i+1] == '*':
                 res = res + '.*?'
                 i = i+1
             else:

+ 3 - 1
dulwich/tests/test_ignore.py

@@ -43,6 +43,8 @@ POSITIVE_MATCH_TESTS = [
     ("foo/bar/bla/blie.c", "foo/**/blie.c"),
     ("foo/bar/bla.c", "**/bla.c"),
     ("bla.c", "**/bla.c"),
+    ("foo/bar", "foo/**/bar"),
+    ("foo/bla/bar", "foo/**/bar"),
 ]
 
 NEGATIVE_MATCH_TESTS = [
@@ -61,7 +63,7 @@ TRANSLATE_TESTS = [
     ("foo.[ch]", '(.*\\/)?foo\\.[ch]\\Z(?ms)'),
     ("foo/**", 'foo\\/.*?\\Z(?ms)'),
     ("foo/**/blie.c", 'foo\\/.*?\\/blie\\.c\\Z(?ms)'),
-    ("**/bla.c", '.*?\\/bla\\.c\\Z(?ms)'),
+    ("**/bla.c", '(.*\\/)?bla\\.c\\Z(?ms)'),
 ]