a008c1
#!/bin/bash
a008c1
a008c1
# If using normal root, avoid changing anything.
a008c1
if [ -z "$RPM_BUILD_ROOT" -o "$RPM_BUILD_ROOT" = "/" ]; then
a008c1
  exit 0
a008c1
fi
a008c1
a008c1
exclude_files=""
a008c1
exclude_files_from=""
a008c1
exclude_shebangs=""
a008c1
exclude_shebangs_from=""
a008c1
a008c1
usage() {
a008c1
  local verbose=$1 && shift
a008c1
  local outfile=$1 && shift
a008c1
  local status=$1 && shift
a008c1
a008c1
  (
a008c1
    echo 'usage: brp-mangle-shebangs [--files <regexp>] [--files-from <file>] [--shebangs <regexp>] [--shebangs-from <file>]'
a008c1
    if [ "${verbose}" == "yes" ]; then
a008c1
      echo '  --files: extended regexp of files to ignore'
a008c1
      echo '  --files-from: file containing a list of extended regexps of files to ignore'
a008c1
      echo '  --shebangs: extended regexp of shebangs to ignore'
a008c1
      echo '  --shebangs-from: file containing a list of extended regexps of shebangs to ignore'
a008c1
    fi
a008c1
  ) >>${outfile}
a008c1
  exit ${status}
a008c1
}
a008c1
a008c1
while [ $# -gt 0 ] ; do
a008c1
  case "$1" in
a008c1
    --files)
a008c1
      exclude_files="${2}"
a008c1
      shift
a008c1
      ;;
a008c1
    --files=*)
a008c1
      exclude_files="${1##--files=}"
a008c1
      ;;
a008c1
    --files-from)
a008c1
      exclude_files_from="${2}"
a008c1
      shift
a008c1
      ;;
a008c1
    --files-from=*)
a008c1
      exclude_files_from="${1##--files-from=}"
a008c1
      ;;
a008c1
    --shebangs)
a008c1
      exclude_shebangs="${2}"
a008c1
      shift
a008c1
      ;;
a008c1
    --shebangs=*)
a008c1
      exclude_shebangs="${1##--shebangs=}"
a008c1
      ;;
a008c1
    --shebangs-from)
a008c1
      exclude_shebangs_from="${2}"
a008c1
      shift
a008c1
      ;;
a008c1
    --shebangs-from=*)
a008c1
      exclude_shebangs_from="${1##--shebangs-from=}"
a008c1
      ;;
a008c1
    --help|--usage|"-?"|-h)
a008c1
      usage yes /dev/stdout 0
a008c1
      ;;
a008c1
    *)
a008c1
      echo "Unknown option \"${1}\"" 1>&2
a008c1
      usage no /dev/stderr 1
a008c1
      ;;
a008c1
  esac
a008c1
  shift
a008c1
done
a008c1
a008c1
cd "$RPM_BUILD_ROOT"
a008c1
32d5e8
# Large packages such as kernel can have thousands of executable files.
32d5e8
# We take care to not fork/exec thousands of "file"s and "grep"s,
32d5e8
# but run just two of them.
32d5e8
# (Take care to exclude filenames which would mangle "file" output).
32d5e8
find -executable -type f ! -path '*:*' ! -path $'*\n*' \
32d5e8
| file -N --mime-type -f - \
32d5e8
| grep -P ".+(?=: text/)" \
32d5e8
| {
a008c1
fail=0
32d5e8
while IFS= read -r line; do
32d5e8
  f=${line%%:*}
a008c1
a008c1
  # Remove the dot
a008c1
  path="${f#.}"
a008c1
a008c1
  if [ -n "$exclude_files" ]; then
a008c1
    echo "$path" | grep -q -E "$exclude_files" && continue
a008c1
  fi
a008c1
  if [ -n "$exclude_files_from" ]; then
a008c1
    echo "$path" | grep -q -E -f "$exclude_files_from" && continue
a008c1
  fi
a008c1
a008c1
8ca6f7
  if ! read shebang_line < "$f"; then
8ca6f7
    echo >&2 "*** WARNING: Cannot read the first line from $f, removing executable bit"
8ca6f7
    ts=$(stat -c %y "$f")
8ca6f7
    chmod -x "$f"
8ca6f7
    touch -d "$ts" "$f"
8ca6f7
    continue
8ca6f7
  fi
8ca6f7
32d5e8
  orig_shebang="${shebang_line#\#!}"
32d5e8
  if [ "$orig_shebang" = "$shebang_line" ]; then
32d5e8
    echo >&2 "*** WARNING: $f is executable but has no shebang, removing executable bit"
32d5e8
    ts=$(stat -c %y "$f")
32d5e8
    chmod -x "$f"
32d5e8
    touch -d "$ts" "$f"
32d5e8
    continue
a008c1
  fi
a008c1
32d5e8
  # Trim spaces
32d5e8
  while shebang="${orig_shebang//  / }"; [ "$shebang" != "$orig_shebang" ]; do
32d5e8
    orig_shebang="$shebang"
32d5e8
  done
32d5e8
  # Treat "#! /path/to " as "#!/path/to"
32d5e8
  orig_shebang="${orig_shebang# }"
32d5e8
32d5e8
  shebang="$orig_shebang"
32d5e8
a008c1
  if [ -z "$shebang" ]; then
32d5e8
    echo >&2 "*** WARNING: $f is executable but has empty shebang, removing executable bit"
32d5e8
    ts=$(stat -c %y "$f")
a008c1
    chmod -x "$f"
a008c1
    touch -d "$ts" "$f"
a008c1
    continue
32d5e8
  fi
32d5e8
  if [ -n "${shebang##/*}" ]; then
a008c1
    echo >&2 "*** ERROR: $f has shebang which doesn't start with '/' ($shebang)"
a008c1
    fail=1
a008c1
    continue
a008c1
  fi
a008c1
a008c1
  if ! { echo "$shebang" | grep -q -P "^/(?:usr/)?(?:bin|sbin)/"; }; then
a008c1
    continue
a008c1
  fi
a008c1
a008c1
  # Replace "special" env shebang:
a008c1
  # /whatsoever/env /whatever/foo → /whatever/foo
a008c1
  shebang=$(echo "$shebang" | sed -r -e 's@^(.+)/env /(.+)$@/\2@')
a008c1
  # /whatsoever/env foo → /whatsoever/foo
a008c1
  shebang=$(echo "$shebang" | sed -r -e 's@^(.+/)env (.+)$@\1\2@')
a008c1
a008c1
  # Replace python3 with the desired Python 3 shebang,
a008c1
  # if passed as an non-empty environment variable PYTHON3
a008c1
  if [ -n "${PYTHON3:+x}" ]; then
a008c1
    shebang=$(echo "$shebang" | sed -r -e "s@/usr/bin/python3(\s|$)@${PYTHON3}\1@")
a008c1
  fi
a008c1
a008c1
  # Replace ambiguous python with python2
a008c1
  py_shebang=$(echo "$shebang" | sed -r -e 's@/usr/bin/python(\s|$)@/usr/bin/python2\1@')
a008c1
a008c1
  if [ "$shebang" != "$py_shebang" ]; then
a008c1
    echo >&2 "*** ERROR: ambiguous python shebang in $path: #!$orig_shebang. Change it to python3 (or python2) explicitly."
a008c1
    fail=1
a008c1
  elif [ "#!$shebang" != "#!$orig_shebang" ]; then
a008c1
    echo "mangling shebang in $path from $orig_shebang to #!$shebang"
32d5e8
    ts=$(stat -c %y "$f")
32d5e8
    sed -i -e "1c #!$shebang" "$f"
32d5e8
    touch -d "$ts" "$f"
a008c1
  fi
a008c1
32d5e8
done
a008c1
a008c1
exit $fail
32d5e8
}