While stress testing `git filter-repo`, I noticed an issue with encoding; further digging led to the fixes and features in this series. See the individual commit messages for details. Changes since v5 (full range-diff below): * s/utf-8/UTF-8/, as pointed out by Torsten (in commit messages and comments) * small code cleanup pointed out by Eric * rewrap the first commit message Elijah Newren (5): t9350: fix encoding test to actually test reencoding fast-import: support 'encoding' commit header fast-export: avoid stripping encoding header if we cannot reencode fast-export: differentiate between explicitly UTF-8 and implicitly UTF-8 fast-export: do automatic reencoding of commit messages only if requested Documentation/git-fast-export.txt | 7 ++ Documentation/git-fast-import.txt | 7 ++ builtin/fast-export.c | 55 ++++++++++++-- fast-import.c | 11 ++- t/t9300-fast-import.sh | 20 +++++ t/t9350-fast-export.sh | 78 +++++++++++++++++--- t/t9350/broken-iso-8859-7-commit-message.txt | 1 + t/t9350/simple-iso-8859-7-commit-message.txt | 1 + 8 files changed, 163 insertions(+), 17 deletions(-) create mode 100644 t/t9350/broken-iso-8859-7-commit-message.txt create mode 100644 t/t9350/simple-iso-8859-7-commit-message.txt Range-diff: 1: 37a68a0ffd ! 1: b5dcdab662 t9350: fix encoding test to actually test reencoding @@ -2,13 +2,13 @@ t9350: fix encoding test to actually test reencoding - This test used an author with non-ascii characters in the name, but - no special commit message. It then grep'ed for those non-ascii - characters, but those are guaranteed to exist regardless of the - reencoding process since the reencoding only affects the commit message, - not the author or committer names. As such, the test would work even if - the re-encoding process simply stripped the commit message entirely. - Modify the test to actually check that the reencoding in utf-8 worked. + This test used an author with non-ascii characters in the name, but no + special commit message. It then grep'ed for those non-ascii characters, + but those are guaranteed to exist regardless of the reencoding process + since the reencoding only affects the commit message, not the author or + committer names. As such, the test would work even if the re-encoding + process simply stripped the commit message entirely. Modify the test to + actually check that the reencoding into UTF-8 worked. Signed-off-by: Elijah Newren <newren@xxxxxxxxx> @@ -40,7 +40,7 @@ + # The commit object, if not re-encoded, would be 240 bytes. + # Removing the "encoding iso-8859-7\n" header drops 20 bytes. + # Re-encoding the Pi character from \xF0 (\360) in iso-8859-7 -+ # to \xCF\x80 (\317\200) in utf-8 adds a byte. Check for ++ # to \xCF\x80 (\317\200) in UTF-8 adds a byte. Check for + # the expected size. + test 221 -eq "$(git cat-file -s i18n)" && + # ...and for the expected translation of bytes. 2: 3d84f4613d ! 2: af7d4e18fa fast-import: support 'encoding' commit header @@ -2,7 +2,7 @@ fast-import: support 'encoding' commit header - Since git supports commit messages with an encoding other than utf-8, + Since git supports commit messages with an encoding other than UTF-8, allow fast-import to import such commits. This may be useful for folks who do not want to reencode commit messages from an external system, and may also be useful to achieve reversible history rewrites (e.g. sha1sum 3: baa8394a3a ! 3: d5b300692a fast-export: avoid stripping encoding header if we cannot reencode @@ -3,8 +3,8 @@ fast-export: avoid stripping encoding header if we cannot reencode When fast-export encounters a commit with an 'encoding' header, it tries - to reencode in utf-8 and then drops the encoding header. However, if it - fails to reencode in utf-8 because e.g. one of the characters in the + to reencode in UTF-8 and then drops the encoding header. However, if it + fails to reencode in UTF-8 because e.g. one of the characters in the commit message was invalid in the old encoding, then we need to retain the original encoding or otherwise we lose information needed to understand all the other (valid) characters in the original commit 4: 49960164c6 ! 4: 2cef40c613 fast-export: differentiate between explicitly utf-8 and implicitly utf-8 @@ -1,11 +1,11 @@ Author: Elijah Newren <newren@xxxxxxxxx> - fast-export: differentiate between explicitly utf-8 and implicitly utf-8 + fast-export: differentiate between explicitly UTF-8 and implicitly UTF-8 The find_encoding() function returned the encoding used by a commit - message, returning a default of git_commit_encoding (usually utf-8). + message, returning a default of git_commit_encoding (usually UTF-8). Although the current code does not differentiate between a commit which - explicitly requested utf-8 and one where we just assume utf-8 because no + explicitly requested UTF-8 and one where we just assume UTF-8 because no encoding is set, it will become important when we try to preserve the encoding header. Since is_encoding_utf8() returns true when passed NULL, we can just return NULL from find_encoding() instead of returning 5: d8be4ee826 ! 5: d18f03d1bf fast-export: do automatic reencoding of commit messages only if requested @@ -62,7 +62,7 @@ + reencode_mode = REENCODE_YES; + break; + default: -+ if (arg && !strcasecmp(arg, "abort")) ++ if (!strcasecmp(arg, "abort")) + reencode_mode = REENCODE_ABORT; + else + return error("Unknown reencoding mode: %s", arg); @@ -156,7 +156,7 @@ + # The commit object, if not re-encoded, is 240 bytes. + # Removing the "encoding iso-8859-7\n" header would drops 20 + # bytes. Re-encoding the Pi character from \xF0 (\360) in -+ # iso-8859-7 to \xCF\x80 (\317\200) in utf-8 adds a byte. ++ # iso-8859-7 to \xCF\x80 (\317\200) in UTF-8 adds a byte. + # Check for the expected size... + test 240 -eq "$(git cat-file -s i18n-no-recoding)" && + # ...as well as the expected byte. -- 2.21.0.782.gd18f03d1bf