From 9679b10eb10de56d2da43e13099d863bcd132659 Mon Sep 17 00:00:00 2001 From: David Leen Date: Mon, 22 Aug 2022 15:09:00 -0700 Subject: [PATCH] Fix issue with spaces in filename when using custom prefix Stash (Bitbucket) uses `src://` and `dst://` instead of `a` and `b` Add a new handler for handling custom prefixes which allows for spaces. --- tests/samples/git_filenames_with_spaces_prefix.diff | 7 +++++++ tests/test_parser.py | 12 ++++++++++++ unidiff/constants.py | 2 ++ unidiff/patch.py | 5 ++++- 4 files changed, 25 insertions(+), 1 deletion(-) create mode 100644 tests/samples/git_filenames_with_spaces_prefix.diff diff --git a/tests/samples/git_filenames_with_spaces_prefix.diff b/tests/samples/git_filenames_with_spaces_prefix.diff new file mode 100644 index 0000000..7d95744 --- /dev/null +++ b/tests/samples/git_filenames_with_spaces_prefix.diff @@ -0,0 +1,7 @@ +diff --git src://foo bar/baz dst://foo bar/baz +new file mode 100644 +index 00000000000..0a72e5064c8 +--- /dev/null ++++ dst://foo bar/baz +@@ -0,0 +1,1 @@ ++blah diff --git a/tests/test_parser.py b/tests/test_parser.py index 9eaa722..221e5fa 100644 --- a/tests/test_parser.py +++ b/tests/test_parser.py @@ -325,6 +325,18 @@ def test_parse_filename_with_spaces(self): self.assertTrue(res[0].is_added_file) self.assertEqual(res[0].path, 'has spaces/t.sql') + def test_parse_filename_prefix_with_spaces(self): + filename = os.path.join(self.samples_dir, 'samples/git_filenames_with_spaces_prefix.diff') + with open(filename) as f: + res = PatchSet(f) + + self.assertEqual(len(res), 1) + + self.assertEqual(res[0].source_file, '/dev/null') + self.assertEqual(res[0].target_file, 'dst://foo bar/baz') + self.assertTrue(res[0].is_added_file) + self.assertEqual(res[0].path, 'dst://foo bar/baz') + def test_deleted_file(self): filename = os.path.join(self.samples_dir, 'samples/git_delete.diff') with open(filename) as f: diff --git a/unidiff/constants.py b/unidiff/constants.py index 72073b7..c206bc7 100644 --- a/unidiff/constants.py +++ b/unidiff/constants.py @@ -38,6 +38,8 @@ # check diff git line for git renamed files support RE_DIFF_GIT_HEADER = re.compile( r'^diff --git (?Pa/[^\t\n]+) (?Pb/[^\t\n]+)') +RE_DIFF_GIT_HEADER_URI_LIKE = re.compile( + r'^diff --git (?P.*://[^\t\n]+) (?P.*://[^\t\n]+)') RE_DIFF_GIT_HEADER_NO_PREFIX = re.compile( r'^diff --git (?P[^\t\n]+) (?P[^\t\n]+)') diff --git a/unidiff/patch.py b/unidiff/patch.py index ff671dc..10c4d06 100644 --- a/unidiff/patch.py +++ b/unidiff/patch.py @@ -40,6 +40,7 @@ LINE_VALUE_NO_NEWLINE, RE_DIFF_GIT_DELETED_FILE, RE_DIFF_GIT_HEADER, + RE_DIFF_GIT_HEADER_URI_LIKE, RE_DIFF_GIT_HEADER_NO_PREFIX, RE_DIFF_GIT_NEW_FILE, RE_HUNK_BODY_LINE, @@ -479,7 +480,9 @@ def _parse(self, diff, encoding, metadata_only): line = line.decode(encoding) # check for a git file rename - is_diff_git_header = RE_DIFF_GIT_HEADER.match(line) or RE_DIFF_GIT_HEADER_NO_PREFIX.match(line) + is_diff_git_header = RE_DIFF_GIT_HEADER.match(line) or \ + RE_DIFF_GIT_HEADER_URI_LIKE.match(line) or \ + RE_DIFF_GIT_HEADER_NO_PREFIX.match(line) if is_diff_git_header: patch_info = PatchInfo() source_file = is_diff_git_header.group('source')