i18n: make GETTEXT_POISON a runtime option

Change the GETTEXT_POISON compile-time + runtime GIT_GETTEXT_POISON
test parameter to only be a GIT_TEST_GETTEXT_POISON=<non-empty?>
runtime parameter, to be consistent with other parameters documented
in "Running tests with special setups" in t/README.

When I added GETTEXT_POISON in bb946bba76 ("i18n: add GETTEXT_POISON
to simulate unfriendly translator", 2011-02-22) I was concerned with
ensuring that the _() function would get constant folded if NO_GETTEXT
was defined, and likewise that GETTEXT_POISON would be compiled out
unless it was defined.

But as the benchmark in my [1] shows doing a one-off runtime
getenv("GIT_TEST_[...]") is trivial, and since GETTEXT_POISON was
originally added the GIT_TEST_* env variables have become the common
idiom for turning on special test setups.

So change GETTEXT_POISON to work the same way. Now the
GETTEXT_POISON=YesPlease compile-time option is gone, and running the
tests with GIT_TEST_GETTEXT_POISON=[YesPlease|] can be toggled on/off
without recompiling.

This allows for conditionally amending tests to test with/without
poison, similar to what 859fdc0c3c ("commit-graph: define
GIT_TEST_COMMIT_GRAPH", 2018-08-29) did for GIT_TEST_COMMIT_GRAPH. Do
some of that, now we e.g. always run the t0205-gettext-poison.sh test.

I did enough there to remove the GETTEXT_POISON prerequisite, but its
inverse C_LOCALE_OUTPUT is still around, and surely some tests using
it can be converted to e.g. always set GIT_TEST_GETTEXT_POISON=.

Notes on the implementation:

 * We still compile a dedicated GETTEXT_POISON build in Travis
   CI. Perhaps this should be revisited and integrated into the
   "linux-gcc" build, see ae59a4e44f ("travis: run tests with
   GIT_TEST_SPLIT_INDEX", 2018-01-07) for prior art in that area. Then
   again maybe not, see [2].

 * We now skip a test in t0000-basic.sh under
   GIT_TEST_GETTEXT_POISON=YesPlease that wasn't skipped before. This
   test relies on C locale output, but due to an edge case in how the
   previous implementation of GETTEXT_POISON worked (reading it from
   GIT-BUILD-OPTIONS) wasn't enabling poison correctly. Now it does,
   and needs to be skipped.

 * The getenv() function is not reentrant, so out of paranoia about
   code of the form:

       printf(_("%s"), getenv("some-env"));

   call use_gettext_poison() in our early setup in git_setup_gettext()
   so we populate the "poison_requested" variable in a codepath that's
   won't suffer from that race condition.

 * We error out in the Makefile if you're still saying
   GETTEXT_POISON=YesPlease to prompt users to change their
   invocation.

 * We should not print out poisoned messages during the test
   initialization itself to keep it more readable, so the test library
   hides the variable if set in $GIT_TEST_GETTEXT_POISON_ORIG during
   setup. See [3].

See also [4] for more on the motivation behind this patch, and the
history of the GETTEXT_POISON facility.

1. https://public-inbox.org/git/871s8gd32p.fsf@evledraar.gmail.com/
2. https://public-inbox.org/git/20181102163725.GY30222@szeder.dev/
3. https://public-inbox.org/git/20181022202241.18629-2-szeder.dev@gmail.com/
4. https://public-inbox.org/git/878t2pd6yu.fsf@evledraar.gmail.com/

Signed-off-by: Ævar Arnfjörð Bjarmason <avarab@gmail.com>
Signed-off-by: Junio C Hamano <gitster@pobox.com>
This commit is contained in:
Ævar Arnfjörð Bjarmason 2018-11-08 21:15:29 +00:00 committed by Junio C Hamano
parent d582ea202b
commit 6cdccfce1e
16 changed files with 60 additions and 50 deletions

View File

@ -26,7 +26,7 @@ addons:
matrix: matrix:
include: include:
- env: jobname=GETTEXT_POISON - env: jobname=GIT_TEST_GETTEXT_POISON
os: linux os: linux
compiler: compiler:
addons: addons:

View File

@ -362,11 +362,6 @@ all::
# Define HAVE_DEV_TTY if your system can open /dev/tty to interact with the # Define HAVE_DEV_TTY if your system can open /dev/tty to interact with the
# user. # user.
# #
# Define GETTEXT_POISON if you are debugging the choice of strings marked
# for translation. In a GETTEXT_POISON build, you can turn all strings marked
# for translation into gibberish by setting the GIT_GETTEXT_POISON variable
# (to any value) in your environment.
#
# Define JSMIN to point to JavaScript minifier that functions as # Define JSMIN to point to JavaScript minifier that functions as
# a filter to have gitweb.js minified. # a filter to have gitweb.js minified.
# #
@ -1452,7 +1447,7 @@ ifdef NO_SYMLINK_HEAD
BASIC_CFLAGS += -DNO_SYMLINK_HEAD BASIC_CFLAGS += -DNO_SYMLINK_HEAD
endif endif
ifdef GETTEXT_POISON ifdef GETTEXT_POISON
BASIC_CFLAGS += -DGETTEXT_POISON $(error The GETTEXT_POISON option has been removed in favor of runtime GIT_TEST_GETTEXT_POISON. See t/README!)
endif endif
ifdef NO_GETTEXT ifdef NO_GETTEXT
BASIC_CFLAGS += -DNO_GETTEXT BASIC_CFLAGS += -DNO_GETTEXT
@ -2603,7 +2598,6 @@ ifdef GIT_TEST_CMP_USE_COPIED_CONTEXT
@echo GIT_TEST_CMP_USE_COPIED_CONTEXT=YesPlease >>$@+ @echo GIT_TEST_CMP_USE_COPIED_CONTEXT=YesPlease >>$@+
endif endif
@echo NO_GETTEXT=\''$(subst ','\'',$(subst ','\'',$(NO_GETTEXT)))'\' >>$@+ @echo NO_GETTEXT=\''$(subst ','\'',$(subst ','\'',$(NO_GETTEXT)))'\' >>$@+
@echo GETTEXT_POISON=\''$(subst ','\'',$(subst ','\'',$(GETTEXT_POISON)))'\' >>$@+
ifdef GIT_PERF_REPEAT_COUNT ifdef GIT_PERF_REPEAT_COUNT
@echo GIT_PERF_REPEAT_COUNT=\''$(subst ','\'',$(subst ','\'',$(GIT_PERF_REPEAT_COUNT)))'\' >>$@+ @echo GIT_PERF_REPEAT_COUNT=\''$(subst ','\'',$(subst ','\'',$(GIT_PERF_REPEAT_COUNT)))'\' >>$@+
endif endif

View File

@ -123,7 +123,7 @@ osx-clang|osx-gcc)
# Travis CI OS X # Travis CI OS X
export GIT_SKIP_TESTS="t9810 t9816" export GIT_SKIP_TESTS="t9810 t9816"
;; ;;
GETTEXT_POISON) GIT_TEST_GETTEXT_POISON)
export GETTEXT_POISON=YesPlease export GIT_TEST_GETTEXT_POISON=YesPlease
;; ;;
esac esac

View File

@ -7,6 +7,7 @@
#include "gettext.h" #include "gettext.h"
#include "strbuf.h" #include "strbuf.h"
#include "utf8.h" #include "utf8.h"
#include "config.h"
#ifndef NO_GETTEXT #ifndef NO_GETTEXT
# include <locale.h> # include <locale.h>
@ -46,15 +47,15 @@ const char *get_preferred_languages(void)
return NULL; return NULL;
} }
#ifdef GETTEXT_POISON
int use_gettext_poison(void) int use_gettext_poison(void)
{ {
static int poison_requested = -1; static int poison_requested = -1;
if (poison_requested == -1) if (poison_requested == -1) {
poison_requested = getenv("GIT_GETTEXT_POISON") ? 1 : 0; const char *v = getenv("GIT_TEST_GETTEXT_POISON");
poison_requested = v && strlen(v) ? 1 : 0;
}
return poison_requested; return poison_requested;
} }
#endif
#ifndef NO_GETTEXT #ifndef NO_GETTEXT
static int test_vsnprintf(const char *fmt, ...) static int test_vsnprintf(const char *fmt, ...)
@ -164,6 +165,8 @@ void git_setup_gettext(void)
if (!podir) if (!podir)
podir = p = system_path(GIT_LOCALE_PATH); podir = p = system_path(GIT_LOCALE_PATH);
use_gettext_poison(); /* getenv() reentrancy paranoia */
if (!is_directory(podir)) { if (!is_directory(podir)) {
free(p); free(p);
return; return;

View File

@ -28,12 +28,15 @@
#define FORMAT_PRESERVING(n) __attribute__((format_arg(n))) #define FORMAT_PRESERVING(n) __attribute__((format_arg(n)))
extern int use_gettext_poison(void);
#ifndef NO_GETTEXT #ifndef NO_GETTEXT
extern void git_setup_gettext(void); extern void git_setup_gettext(void);
extern int gettext_width(const char *s); extern int gettext_width(const char *s);
#else #else
static inline void git_setup_gettext(void) static inline void git_setup_gettext(void)
{ {
use_gettext_poison(); /* getenv() reentrancy paranoia */
} }
static inline int gettext_width(const char *s) static inline int gettext_width(const char *s)
{ {
@ -41,12 +44,6 @@ static inline int gettext_width(const char *s)
} }
#endif #endif
#ifdef GETTEXT_POISON
extern int use_gettext_poison(void);
#else
#define use_gettext_poison() 0
#endif
static inline FORMAT_PRESERVING(1) const char *_(const char *msgid) static inline FORMAT_PRESERVING(1) const char *_(const char *msgid)
{ {
if (!*msgid) if (!*msgid)

View File

@ -17,7 +17,7 @@ export TEXTDOMAINDIR
# First decide what scheme to use... # First decide what scheme to use...
GIT_INTERNAL_GETTEXT_SH_SCHEME=fallthrough GIT_INTERNAL_GETTEXT_SH_SCHEME=fallthrough
if test -n "$GIT_GETTEXT_POISON" if test -n "$GIT_TEST_GETTEXT_POISON"
then then
GIT_INTERNAL_GETTEXT_SH_SCHEME=poison GIT_INTERNAL_GETTEXT_SH_SCHEME=poison
elif test -n "@@USE_GETTEXT_SCHEME@@" elif test -n "@@USE_GETTEXT_SCHEME@@"

View File

@ -289,16 +289,11 @@ something in the test suite might still depend on the US English
version of the strings, e.g. to grep some error message or other version of the strings, e.g. to grep some error message or other
output. output.
To smoke out issues like these Git can be compiled with gettext poison To smoke out issues like these, Git tested with a translation mode that
support, at the top-level: emits gibberish on every call to gettext. To use it run the test suite
with it, e.g.:
make GETTEXT_POISON=YesPlease cd t && GIT_TEST_GETTEXT_POISON=YesPlease prove -j 9 ./t[0-9]*.sh
That'll give you a git which emits gibberish on every call to
gettext. It's obviously not meant to be installed, but you should run
the test suite with it:
cd t && prove -j 9 ./t[0-9]*.sh
If tests break with it you should inspect them manually and see if If tests break with it you should inspect them manually and see if
what you're translating is sane, i.e. that you're not translating what you're translating is sane, i.e. that you're not translating

View File

@ -301,6 +301,12 @@ that cannot be easily covered by a few specific test cases. These
could be enabled by running the test suite with correct GIT_TEST_ could be enabled by running the test suite with correct GIT_TEST_
environment set. environment set.
GIT_TEST_GETTEXT_POISON=<non-empty?> turns all strings marked for
translation into gibberish if non-empty (think "test -n"). Used for
spotting those tests that need to be marked with a C_LOCALE_OUTPUT
prerequisite when adding more strings for translation. See "Testing
marked strings" in po/README for details.
GIT_TEST_SPLIT_INDEX=<boolean> forces split-index mode on the whole GIT_TEST_SPLIT_INDEX=<boolean> forces split-index mode on the whole
test suite. Accept any boolean values that are accepted by git-config. test suite. Accept any boolean values that are accepted by git-config.

View File

@ -12,7 +12,7 @@ export GIT_TEXTDOMAINDIR GIT_PO_PATH
. "$GIT_BUILD_DIR"/git-sh-i18n . "$GIT_BUILD_DIR"/git-sh-i18n
if test_have_prereq GETTEXT && ! test_have_prereq GETTEXT_POISON if test_have_prereq GETTEXT && test_have_prereq C_LOCALE_OUTPUT
then then
# is_IS.UTF-8 on Solaris and FreeBSD, is_IS.utf8 on Debian # is_IS.UTF-8 on Solaris and FreeBSD, is_IS.utf8 on Debian
is_IS_locale=$(locale -a 2>/dev/null | is_IS_locale=$(locale -a 2>/dev/null |

View File

@ -274,7 +274,7 @@ test_expect_success 'pretend we have a mix of all possible results' "
EOF EOF
" "
test_expect_success 'test --verbose' ' test_expect_success C_LOCALE_OUTPUT 'test --verbose' '
test_must_fail run_sub_test_lib_test \ test_must_fail run_sub_test_lib_test \
test-verbose "test verbose" --verbose <<-\EOF && test-verbose "test verbose" --verbose <<-\EOF &&
test_expect_success "passing test" true test_expect_success "passing test" true

View File

@ -5,13 +5,15 @@
test_description='Gettext Shell poison' test_description='Gettext Shell poison'
GIT_TEST_GETTEXT_POISON=YesPlease
export GIT_TEST_GETTEXT_POISON
. ./lib-gettext.sh . ./lib-gettext.sh
test_expect_success GETTEXT_POISON 'sanity: $GIT_INTERNAL_GETTEXT_SH_SCHEME" is poison' ' test_expect_success 'sanity: $GIT_INTERNAL_GETTEXT_SH_SCHEME" is poison' '
test "$GIT_INTERNAL_GETTEXT_SH_SCHEME" = "poison" test "$GIT_INTERNAL_GETTEXT_SH_SCHEME" = "poison"
' '
test_expect_success GETTEXT_POISON 'gettext: our gettext() fallback has poison semantics' ' test_expect_success 'gettext: our gettext() fallback has poison semantics' '
printf "# GETTEXT POISON #" >expect && printf "# GETTEXT POISON #" >expect &&
gettext "test" >actual && gettext "test" >actual &&
test_cmp expect actual && test_cmp expect actual &&
@ -20,7 +22,7 @@ test_expect_success GETTEXT_POISON 'gettext: our gettext() fallback has poison s
test_cmp expect actual test_cmp expect actual
' '
test_expect_success GETTEXT_POISON 'eval_gettext: our eval_gettext() fallback has poison semantics' ' test_expect_success 'eval_gettext: our eval_gettext() fallback has poison semantics' '
printf "# GETTEXT POISON #" >expect && printf "# GETTEXT POISON #" >expect &&
eval_gettext "test" >actual && eval_gettext "test" >actual &&
test_cmp expect actual && test_cmp expect actual &&

View File

@ -77,7 +77,7 @@ test_expect_success 'rebase -n overrides config rebase.stat config' '
# "Does not point to a valid commit: invalid-ref" # "Does not point to a valid commit: invalid-ref"
# #
# NEEDSWORK: This "grep" is fine in real non-C locales, but # NEEDSWORK: This "grep" is fine in real non-C locales, but
# GETTEXT_POISON poisons the refname along with the enclosing # GIT_TEST_GETTEXT_POISON poisons the refname along with the enclosing
# error message. # error message.
test_expect_success 'rebase --onto outputs the invalid ref' ' test_expect_success 'rebase --onto outputs the invalid ref' '
test_must_fail git rebase --onto invalid-ref HEAD HEAD 2>err && test_must_fail git rebase --onto invalid-ref HEAD HEAD 2>err &&

View File

@ -254,9 +254,9 @@ test_expect_success 'checkout to detach HEAD (with advice declined)' '
test_expect_success 'checkout to detach HEAD' ' test_expect_success 'checkout to detach HEAD' '
git config advice.detachedHead true && git config advice.detachedHead true &&
git checkout -f renamer && git clean -f && git checkout -f renamer && git clean -f &&
git checkout renamer^ 2>messages && GIT_TEST_GETTEXT_POISON= git checkout renamer^ 2>messages &&
test_i18ngrep "HEAD is now at 7329388" messages && grep "HEAD is now at 7329388" messages &&
(test_line_count -gt 1 messages || test -n "$GETTEXT_POISON") && test_line_count -gt 1 messages &&
H=$(git rev-parse --verify HEAD) && H=$(git rev-parse --verify HEAD) &&
M=$(git show-ref -s --verify refs/heads/master) && M=$(git show-ref -s --verify refs/heads/master) &&
test "z$H" = "z$M" && test "z$H" = "z$M" &&

View File

@ -1697,7 +1697,8 @@ test_expect_success 'sourcing the completion script clears cached commands' '
verbose test -z "$__git_all_commands" verbose test -z "$__git_all_commands"
' '
test_expect_success !GETTEXT_POISON 'sourcing the completion script clears cached merge strategies' ' test_expect_success 'sourcing the completion script clears cached merge strategies' '
GIT_TEST_GETTEXT_POISON= &&
__git_compute_merge_strategies && __git_compute_merge_strategies &&
verbose test -n "$__git_merge_strategies" && verbose test -n "$__git_merge_strategies" &&
. "$GIT_BUILD_DIR/contrib/completion/git-completion.bash" && . "$GIT_BUILD_DIR/contrib/completion/git-completion.bash" &&

View File

@ -755,16 +755,16 @@ test_cmp_bin() {
# Use this instead of test_cmp to compare files that contain expected and # Use this instead of test_cmp to compare files that contain expected and
# actual output from git commands that can be translated. When running # actual output from git commands that can be translated. When running
# under GETTEXT_POISON this pretends that the command produced expected # under GIT_TEST_GETTEXT_POISON this pretends that the command produced expected
# results. # results.
test_i18ncmp () { test_i18ncmp () {
test -n "$GETTEXT_POISON" || test_cmp "$@" ! test_have_prereq C_LOCALE_OUTPUT || test_cmp "$@"
} }
# Use this instead of "grep expected-string actual" to see if the # Use this instead of "grep expected-string actual" to see if the
# output from a git command that can be translated either contains an # output from a git command that can be translated either contains an
# expected string, or does not contain an unwanted one. When running # expected string, or does not contain an unwanted one. When running
# under GETTEXT_POISON this pretends that the command produced expected # under GIT_TEST_GETTEXT_POISON this pretends that the command produced expected
# results. # results.
test_i18ngrep () { test_i18ngrep () {
eval "last_arg=\${$#}" eval "last_arg=\${$#}"
@ -779,7 +779,7 @@ test_i18ngrep () {
error "bug in the test script: too few parameters to test_i18ngrep" error "bug in the test script: too few parameters to test_i18ngrep"
fi fi
if test -n "$GETTEXT_POISON" if test_have_prereq !C_LOCALE_OUTPUT
then then
# pretend success # pretend success
return 0 return 0

View File

@ -95,6 +95,16 @@ PAGER=cat
TZ=UTC TZ=UTC
export LANG LC_ALL PAGER TZ export LANG LC_ALL PAGER TZ
EDITOR=: EDITOR=:
# GIT_TEST_GETTEXT_POISON should not influence git commands executed
# during initialization of test-lib and the test repo. Back it up,
# unset and then restore after initialization is finished.
if test -n "$GIT_TEST_GETTEXT_POISON"
then
GIT_TEST_GETTEXT_POISON_ORIG=$GIT_TEST_GETTEXT_POISON
unset GIT_TEST_GETTEXT_POISON
fi
# A call to "unset" with no arguments causes at least Solaris 10 # A call to "unset" with no arguments causes at least Solaris 10
# /usr/xpg4/bin/sh and /bin/ksh to bail out. So keep the unsets # /usr/xpg4/bin/sh and /bin/ksh to bail out. So keep the unsets
# deriving from the command substitution clustered with the other # deriving from the command substitution clustered with the other
@ -1104,13 +1114,15 @@ test -n "$USE_LIBPCRE1" && test_set_prereq LIBPCRE1
test -n "$USE_LIBPCRE2" && test_set_prereq LIBPCRE2 test -n "$USE_LIBPCRE2" && test_set_prereq LIBPCRE2
test -z "$NO_GETTEXT" && test_set_prereq GETTEXT test -z "$NO_GETTEXT" && test_set_prereq GETTEXT
# Can we rely on git's output in the C locale? if test -n "$GIT_TEST_GETTEXT_POISON_ORIG"
if test -n "$GETTEXT_POISON" then
GIT_TEST_GETTEXT_POISON=$GIT_TEST_GETTEXT_POISON_ORIG
unset GIT_TEST_GETTEXT_POISON_ORIG
fi
# Can we rely on git's output in the C locale?
if test -z "$GIT_TEST_GETTEXT_POISON"
then then
GIT_GETTEXT_POISON=YesPlease
export GIT_GETTEXT_POISON
test_set_prereq GETTEXT_POISON
else
test_set_prereq C_LOCALE_OUTPUT test_set_prereq C_LOCALE_OUTPUT
fi fi