On 7/4/24 09:17, Collin Funk wrote:
As long as a
Python version ≥ 3.7 is installed everything should be fine on that
end:

Yes, though sometimes Python is misinstalled.

When I run "sh -x ./gnulib-tool --list", the last thing it does is:

exec /home/eggert/src/gnu/gnulib/./gnulib-tool.py --list

and when I run "sh -x /home/eggert/src/gnu/gnulib/./gnulib-tool.py --list", the last thing it does is:

exec python3 /home/eggert/src/gnu/gnulib-savannah/./.gnulib-tool.py --list

so there should be not much going on other than running Python. Perhaps the bug reporter could try running these "sh -x" commands and letting is know what happens.


PS. There's a lot of machinery in those shell scripts for the minor benefit of letting gnulib-tool be a symlink in your $PATH to the real gnulib-tool. How about if we drop support for this? That'd simplify startup quite a bit (if we're lucky it'll even fix the reporter's bug or at least make it easier to diagnose), and there is a better way to get the benefits of that minor feature that doesn't involve so much problematic shell rigamarole.

Something like the attached patch, perhaps? I haven't installed it.


PPS. Why do we have both gnulib-tool.py and .gnulib-tool.py? Is this commented in the source code?
From 6e51dd21e8b62eb5dba5d2164d418fe22ee572a9 Mon Sep 17 00:00:00 2001
From: Paul Eggert <egg...@cs.ucla.edu>
Date: Thu, 4 Jul 2024 13:19:23 +0200
Subject: [PATCH] gnulib-tool: simplify startup
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit

* gnulib-tool, gnulib-tool.py (func_readlink, func_gnulib_dir): Remove.
In main program, use shell substitution rather than external
programs, to reduce dependencies on external programs, to
simplify, and to speed things up a little.
* doc/gnulib-tool.texi (Invoking gnulib-tool): This means that we
no longer support the trick of putting a symlink to gnulib-tool
somewhere in your PATH, but it’s just as easy to put gnulib-tool
in your PATH so document that.
---
 ChangeLog            |  12 +++++
 doc/gnulib-tool.texi |   6 +--
 gnulib-tool          | 113 ++++++-------------------------------------
 gnulib-tool.py       |  97 +++----------------------------------
 4 files changed, 35 insertions(+), 193 deletions(-)

diff --git a/ChangeLog b/ChangeLog
index b19c2a482a..d9ddf46ce5 100644
--- a/ChangeLog
+++ b/ChangeLog
@@ -1,3 +1,15 @@
+2024-07-04  Paul Eggert  <egg...@cs.ucla.edu>
+
+	gnulib-tool: simplify startup
+	* gnulib-tool, gnulib-tool.py (func_readlink, func_gnulib_dir): Remove.
+	In main program, use shell substitution rather than external
+	programs, to reduce dependencies on external programs, to
+	simplify, and to speed things up a little.
+	* doc/gnulib-tool.texi (Invoking gnulib-tool): This means that we
+	no longer support the trick of putting a symlink to gnulib-tool
+	somewhere in your PATH, but it’s just as easy to put gnulib-tool
+	in your PATH so document that.
+
 2024-07-03  Collin Funk  <collin.fu...@gmail.com>
 
 	gitlog-to-changelog: Add a new --commit-timezone option.
diff --git a/doc/gnulib-tool.texi b/doc/gnulib-tool.texi
index 0f059e1287..a3b74830ef 100644
--- a/doc/gnulib-tool.texi
+++ b/doc/gnulib-tool.texi
@@ -24,11 +24,9 @@ simplifies the management of source files, @file{Makefile.am}s and
 contained in the @code{PATH} variable.  It needs to be run directly in
 the directory that contains the Gnulib source code.  You can do this
 either by specifying the absolute filename of @file{gnulib-tool}, or
-you can also use a symbolic link from a place inside your @code{PATH}
-to the @file{gnulib-tool} file of your preferred and most up-to-date
-Gnulib checkout, like this:
+by putting the Gnulib source code directory in your @env{PATH}, like this:
 @smallexample
-$ ln -s $HOME/gnu/src/gnulib.git/gnulib-tool $HOME/bin/gnulib-tool
+$ export PATH=$HOME/gnu/src/gnulib.git:$PATH
 @end smallexample
 
 Run @samp{gnulib-tool --help} for information.  To get familiar with
diff --git a/gnulib-tool b/gnulib-tool
index 789fe916a8..0737945b12 100755
--- a/gnulib-tool
+++ b/gnulib-tool
@@ -50,110 +50,21 @@ func_fatal_error ()
   func_exit 1
 }
 
-# func_readlink SYMLINK
-# outputs the target of the given symlink.
-if (type readlink) > /dev/null 2>&1; then
-  func_readlink ()
-  {
-    # Use the readlink program from GNU coreutils.
-    readlink "$1"
-  }
-else
-  func_readlink ()
-  {
-    # Use two sed invocations. A single sed -n -e 's,^.* -> \(.*\)$,\1,p'
-    # would do the wrong thing if the link target contains " -> ".
-    LC_ALL=C ls -l "$1" | sed -e 's, -> ,#%%#,' | sed -n -e 's,^.*#%%#\(.*\)$,\1,p'
-  }
-fi
-
-# func_gnulib_dir
-# locates the directory where the gnulib repository lives
-# Input:
-# - progname                 name of this program
-# Sets variables
-# - self_abspathname         absolute pathname of gnulib-tool
-# - gnulib_dir               absolute pathname of gnulib repository
-func_gnulib_dir ()
-{
-  case "$progname" in
-    /* | ?:*) self_abspathname="$progname" ;;
-    */*) self_abspathname=`pwd`/"$progname" ;;
-    *)
-      # Look in $PATH.
-      # Iterate through the elements of $PATH.
-      # We use IFS=: instead of
-      #   for d in `echo ":$PATH:" | sed -e 's/:::*/:.:/g' | sed -e 's/:/ /g'`
-      # because the latter does not work when some PATH element contains spaces.
-      # We use a canonicalized $pathx instead of $PATH, because empty PATH
-      # elements are by definition equivalent to '.', however field splitting
-      # according to IFS=: loses empty fields in many shells:
-      #   - /bin/sh on OSF/1 and Solaris loses all empty fields (at the
-      #     beginning, at the end, and in the middle),
-      #   - /bin/sh on IRIX and /bin/ksh on IRIX and OSF/1 lose empty fields
-      #     at the beginning and at the end,
-      #   - GNU bash, /bin/sh on AIX and HP-UX, and /bin/ksh on AIX, HP-UX,
-      #     Solaris lose empty fields at the end.
-      # The 'case' statement is an optimization, to avoid evaluating the
-      # explicit canonicalization command when $PATH contains no empty fields.
-      self_abspathname=
-      if test "$PATH_SEPARATOR" = ";"; then
-        # On Windows, programs are searched in "." before $PATH.
-        pathx=".;$PATH"
-      else
-        # On Unix, we have to convert empty PATH elements to ".".
-        pathx="$PATH"
-        case :$PATH: in
-          *::*)
-            pathx=`echo ":$PATH:" | sed -e 's/:::*/:.:/g' -e 's/^://' -e 's/:\$//'`
-            ;;
-        esac
-      fi
-      saved_IFS="$IFS"
-      IFS="$PATH_SEPARATOR"
-      for d in $pathx; do
-        IFS="$saved_IFS"
-        test -z "$d" && d=.
-        if test -x "$d/$progname" && test ! -d "$d/$progname"; then
-          self_abspathname="$d/$progname"
-          break
-        fi
-      done
-      IFS="$saved_IFS"
-      if test -z "$self_abspathname"; then
-        func_fatal_error "could not locate the gnulib-tool program - how did you invoke it?"
-      fi
-      ;;
-  esac
-  while test -h "$self_abspathname"; do
-    # Resolve symbolic link.
-    linkval=`func_readlink "$self_abspathname"`
-    test -n "$linkval" || break
-    case "$linkval" in
-      /* | ?:* ) self_abspathname="$linkval" ;;
-      * ) self_abspathname=`echo "$self_abspathname" | sed -e 's,/[^/]*$,,'`/"$linkval" ;;
-    esac
-  done
-  gnulib_dir=`echo "$self_abspathname" | sed -e 's,/[^/]*$,,'`
-}
-
-func_gnulib_dir
-
 case "$GNULIB_TOOL_IMPL" in
   '')
     # Use the Python implementation if a suitable Python version is found
     # in $PATH. This is the same Python version test as in gnulib-tool.py.
     if (python3 -c 'import sys; sys.exit(not sys.version_info >= (3,7))') 2>/dev/null; then
-      exec "$gnulib_dir/gnulib-tool.py" "$@"
+      exec "$progname.py" "$@"
     else
       echo "gnulib-tool: warning: python3 not found or too old, using the slow shell-based implementation" 1>&2
-      exec "$gnulib_dir/gnulib-tool.sh" "$@"
+      exec "$progname.sh" "$@"
     fi
     ;;
   sh)
-    exec "$gnulib_dir/gnulib-tool.sh" "$@" ;;
+    exec "$progname.sh" "$@" ;;
   py)
-    exec "$gnulib_dir/gnulib-tool.py" "$@" ;;
+    exec "$progname.py" "$@" ;;
   sh+py)
     case " $* " in
       *" --import"* | *" --add-import"* | *" --remove-import"* | *" --update"* | *" --copy-file"*)
@@ -183,10 +94,14 @@ case "$GNULIB_TOOL_IMPL" in
           func_exit 1
         }
         # Execute gnulib-tool.py in the clone directory.
-        (cd "$tmp" && "$gnulib_dir/gnulib-tool.py" "$@" >"$tmp-py-out" 2>"$tmp-py-err")
+        (case $progname in
+           /*) abs_progname=$progname ;;
+           *)  abs_progname=$PWD/$progname ;;
+         esac
+         cd "$tmp" && "$abs_progname.py" "$@" >"$tmp-py-out" 2>"$tmp-py-err")
         pyrc=$?
         # Execute gnulib-tool.sh in the current directory.
-        "$gnulib_dir/gnulib-tool.sh" "$@" >"$tmp-sh-out" 2>"$tmp-sh-err"
+        "$progname.sh" "$@" >"$tmp-sh-out" 2>"$tmp-sh-err"
         shrc=$?
         if test $shrc != 0; then
           if test $pyrc = 0; then
@@ -233,10 +148,10 @@ case "$GNULIB_TOOL_IMPL" in
         # Find another directory name.
         tmp="$dir-glpy$$"
         # Execute gnulib-tool.py, creating a different directory.
-        "$gnulib_dir/gnulib-tool.py" "$@" --dir="$tmp" >"$tmp-py-out" 2>"$tmp-py-err"
+        "$progname.py" "$@" --dir="$tmp" >"$tmp-py-out" 2>"$tmp-py-err"
         pyrc=$?
         # Execute gnulib-tool.sh, creating the intended directory.
-        "$gnulib_dir/gnulib-tool.sh" "$@" >"$tmp-sh-out" 2>"$tmp-sh-err"
+        "$progname.sh" "$@" >"$tmp-sh-out" 2>"$tmp-sh-err"
         shrc=$?
         if test $shrc != 0; then
           if test $pyrc = 0; then
@@ -274,10 +189,10 @@ case "$GNULIB_TOOL_IMPL" in
         # A gnulib-tool invocation that produces only output, no files.
         tmp="glpy$$"
         # Execute gnulib-tool.py.
-        "$gnulib_dir/gnulib-tool.py" "$@" >"$tmp-py-out" 2>"$tmp-py-err"
+        "$progname.py" "$@" >"$tmp-py-out" 2>"$tmp-py-err"
         pyrc=$?
         # Execute gnulib-tool.sh.
-        "$gnulib_dir/gnulib-tool.sh" "$@" >"$tmp-sh-out" 2>"$tmp-sh-err"
+        "$progname.sh" "$@" >"$tmp-sh-out" 2>"$tmp-sh-err"
         shrc=$?
         if test $shrc != 0; then
           if test $pyrc = 0; then
diff --git a/gnulib-tool.py b/gnulib-tool.py
index 52389dcd78..6de6625d55 100755
--- a/gnulib-tool.py
+++ b/gnulib-tool.py
@@ -41,95 +41,6 @@
   func_exit 1
 }
 
-# func_readlink SYMLINK
-# outputs the target of the given symlink.
-if (type readlink) > /dev/null 2>&1; then
-  func_readlink ()
-  {
-    # Use the readlink program from GNU coreutils.
-    readlink "$1"
-  }
-else
-  func_readlink ()
-  {
-    # Use two sed invocations. A single sed -n -e 's,^.* -> \(.*\)$,\1,p'
-    # would do the wrong thing if the link target contains " -> ".
-    LC_ALL=C ls -l "$1" | sed -e 's, -> ,#%%#,' | sed -n -e 's,^.*#%%#\(.*\)$,\1,p'
-  }
-fi
-
-# func_gnulib_dir
-# locates the directory where the gnulib repository lives
-# Input:
-# - progname                 name of this program
-# Sets variables
-# - self_abspathname         absolute pathname of gnulib-tool
-# - gnulib_dir               absolute pathname of gnulib repository
-func_gnulib_dir ()
-{
-  case "$progname" in
-    /* | ?:*) self_abspathname="$progname" ;;
-    */*) self_abspathname=`pwd`/"$progname" ;;
-    *)
-      # Look in $PATH.
-      # Iterate through the elements of $PATH.
-      # We use IFS=: instead of
-      #   for d in `echo ":$PATH:" | sed -e 's/:::*/:.:/g' | sed -e 's/:/ /g'`
-      # because the latter does not work when some PATH element contains spaces.
-      # We use a canonicalized $pathx instead of $PATH, because empty PATH
-      # elements are by definition equivalent to '.', however field splitting
-      # according to IFS=: loses empty fields in many shells:
-      #   - /bin/sh on OSF/1 and Solaris loses all empty fields (at the
-      #     beginning, at the end, and in the middle),
-      #   - /bin/sh on IRIX and /bin/ksh on IRIX and OSF/1 lose empty fields
-      #     at the beginning and at the end,
-      #   - GNU bash, /bin/sh on AIX and HP-UX, and /bin/ksh on AIX, HP-UX,
-      #     Solaris lose empty fields at the end.
-      # The 'case' statement is an optimization, to avoid evaluating the
-      # explicit canonicalization command when $PATH contains no empty fields.
-      self_abspathname=
-      if test "$PATH_SEPARATOR" = ";"; then
-        # On Windows, programs are searched in "." before $PATH.
-        pathx=".;$PATH"
-      else
-        # On Unix, we have to convert empty PATH elements to ".".
-        pathx="$PATH"
-        case :$PATH: in
-          *::*)
-            pathx=`echo ":$PATH:" | sed -e 's/:::*/:.:/g' -e 's/^://' -e 's/:\$//'`
-            ;;
-        esac
-      fi
-      saved_IFS="$IFS"
-      IFS="$PATH_SEPARATOR"
-      for d in $pathx; do
-        IFS="$saved_IFS"
-        test -z "$d" && d=.
-        if test -x "$d/$progname" && test ! -d "$d/$progname"; then
-          self_abspathname="$d/$progname"
-          break
-        fi
-      done
-      IFS="$saved_IFS"
-      if test -z "$self_abspathname"; then
-        func_fatal_error "could not locate the gnulib-tool program - how did you invoke it?"
-      fi
-      ;;
-  esac
-  while test -h "$self_abspathname"; do
-    # Resolve symbolic link.
-    linkval=`func_readlink "$self_abspathname"`
-    test -n "$linkval" || break
-    case "$linkval" in
-      /* | ?:* ) self_abspathname="$linkval" ;;
-      * ) self_abspathname=`echo "$self_abspathname" | sed -e 's,/[^/]*$,,'`/"$linkval" ;;
-    esac
-  done
-  gnulib_dir=`echo "$self_abspathname" | sed -e 's,/[^/]*$,,'`
-}
-
-func_gnulib_dir
-
 # Check the Python version.
 if (python3 -c 'import sys; sys.exit(not sys.version_info >= (3,7))') 2>/dev/null; then
   :
@@ -157,4 +68,10 @@
 profiler_args=
 # For profiling, cf. <https://docs.python.org/3/library/profile.html>.
 #profiler_args="-m cProfile -s tottime"
-exec python3 $profiler_args "$gnulib_dir/.gnulib-tool.py" "$@"
+
+case $progname in
+  */*) dotprogname=${progname%/*}/.${progname##*/} ;;
+  *)   dotprogname=.${progname} ;;
+esac
+
+exec python3 $profiler_args "$dotprogname" "$@"
-- 
2.34.1

Reply via email to