Поиск страниц руководства, содержащих ВСЕ слова «foo», «bar» и «baz».

Поиск страниц руководства, содержащих ВСЕ слова «foo», «bar» и «baz».

Я хотел бы найти страницы руководства, содержащие ВСЕ слова «foo», «bar» и «baz».

Если возможно, я бы хотел выполнить поиск по всему тексту (а не только по названию и описанию) всех страниц руководства.

Я предполагаю, что-то вроде

man -K foo AND bar AND baz

решение1

Я реализовал скрипт, который делает именно это.

if [ $# -eq 0 ]; then
  PATTERNS=(NAME AUTHOR EXAMPLES FILES)
else
  PATTERNS=( "$@" )
fi

[ ${#PATTERNS[@]} -lt 1 ] && echo "Needs at least 1 pattern to search for" && exit 1

for i in $(find /usr/share/man/ -type f); do
  TMPOUT=$(zgrep -l "${PATTERNS[0]}" "$i")
  [ -z "$TMPOUT" ] && continue

  for c in `seq 1 $((${#PATTERNS[@]}-1))`; do
    TMPOUT=$(echo "$TMPOUT" | xargs zgrep -l "${PATTERNS[$c]}")
    [ -z "$TMPOUT" ] && break
  done

  if [ ! -z "$TMPOUT" ]; then
    #echo "$TMPOUT" # Prints the whole path
    MANNAME="$(basename "$TMPOUT")"
    man "${MANNAME%%.*}"
  fi
done

Думаю, это была пустая трата времени :(

Редактировать: Похоже на то

man -K expr1 expr2 expr3

не сработало?

Редактировать: Теперь вы можете передавать скрипты с вашими поисковыми запросами через./script foo bar

решение2

Несколько мыслей по поводу написания этого сценария:

  • Использование manpathдля получения местоположения(й) страниц руководства. Если я добавлю /home/graeme/.cabal/binк моему PATH, manpathman) найдет страницы руководства в /home/graeme/.cabal/share/man.

  • Используйте сам man для распаковки и форматирования страниц перед поиском, таким образом вы просто ищете сам текст man, а не комментарии и т. д. в необработанном файле. Использование manпотенциально будет иметь дело с несколькими форматами.

  • Сохранение отформатированных страниц во временном файле позволит избежать множественных распаковок и должно значительно ускорить процесс.

Вот (с помощью bashGNU find):

#!/bin/bash

set -f; IFS=:
trap 'rm -f "$temp"' EXIT
temp=$(mktemp --tmpdir search_man.XXXXXXXXXX)

while IFS= read -rd '' file; do
  man "$file" >"$temp" 2>/dev/null

  unset fail
  for arg; do
    if ! grep -Fq -- "$arg" "$temp"; then
      fail=true
      break
    fi
  done

  if [ -z "$fail" ]; then
    file=${file##*/}
    printf '%s\n' "${file%.gz}"
  fi
done < <(find $(manpath) -type d ! -name 'man*' -prune -o -type f -print0)

решение3

Не такой полный ответ, как у @polym, но я собирался предложить что-то вроде

while IFS= read -rd $'\0' f; do 
  zgrep -qwm1 'foo' "$f" && \
  zgrep -qwm1 'bar' "$f" && \
  zgrep -qwm1 'baz' "$f" && \
  printf '%s\n' "$f"
done < <(find /usr/share/man -name '*.gz' -print0)

Обратите внимание, что я добавил -wпереключатель (соответствие слов) к greps-, что может быть не тем, что вам нужно (вы хотите включить совпадения типафулиш и орехбар?)

решение4

Этот подход не проверен, но он довольно прост (до глупости прост), и я ожидаю, что он должен работать, даже если он неэффективен:

#!/bin/bash

if [ "$#" -eq 0 ]; then
  echo "Provide arguments to search all man pages for all arguments." >&2
  echo "Putting rare search terms first will improve performance." >&2
  exit
fi

if [ "$#" -eq 1 ]; then
  exec man -K "$@"
fi

pages=( $(man -wK "$1") )
shift
while [ "$#" -gt 1 ]; do
  pages=( $(zgrep -l "$1" "${pages[@]}") )
  shift
done
exec man "${pages[@]}"

Связанный контент