Как реализовать скрипт-обертку pstrace
в bash, который изменяет интерфейс
[sudo] strace -c -p [PID]
к
[sudo] pstrace -c -p [PROCESS-NAME]
похоже на то, как
killall [PROCESS-NAME]
используется. С завершением и всем остальным.
решение1
Начиная с версии 5.15, strace может выводить имена команд/процессов:
- Реализована
--decode-pids=comm
опция (и ее псевдоним-Y
) для печати имен команд для PID.
решение2
Твое это:
ps auxw | grep [PROCESS-NAME] | awk '{print"-p " $2}' | xargs strace
решение3
Обманчиво сложные требования :-)
Состоит из двух частей, во-первых, pstrace
скрипт-оболочка для strace
, который используется pgrep
для операции преобразования имени в PID.
#!/bin/bash
IFS=$' \t\n'
# process the arguments to find "-p procname", only support one instance though
for ((nn=1; nn<=$#; nn++)); do
if [ "${!nn}" = "-p" ]; then
:
elif [ "$prev" = "-p" ]; then
pname="${!nn}"
else
args+=( "${!nn}" ) # just copy
fi
prev="${!nn}"
done
pids=()
if [ -n "$pname" ]; then
# skip this shell's PID, which pgrep -f will match
# note the use of exec to avoid picking up a matching subshell too
# uncomment && printf for pid/pname list
while read pp pname; do
[ "$pp" != "$$" ] && pids+=($pp) # && printf "%6i %s\n" "$pp" "$pname"
done < <(exec pgrep -l -f "${pname}")
fi
npids=${#pids[*]}
if [ $npids -eq 0 ]; then
echo "No PIDs to trace."; exit 2
elif [ $npids -eq 1 ]; then
args=( "${args[@]}" -p ${pids[0]} )
elif [ $npids -le 32 ]; then
read -p "$npids PIDS found, enter Y to proceed: " yy
[ "$yy" != "Y" ] && echo "Cancelled..." && exit 1
args=( "${args[@]}" ${pids[@]/#/-p } )
else
echo "Too many PIDs to trace: $npids (max 32)."; exit 2
fi
strace "${args[@]}"
Во второй части я буду использовать bash
программируемое завершение для завершения процессов по имени, вставьте это в свой ~/.bash_profile
или аналогичный:
# process-name patterns to ignore
PROCIGNORE=( "^\[", "^-bash" )
_c8n_listprocs ()
{
local cur prv ignore IFS nn mm
prv=${COMP_WORDS[COMP_CWORD-1]}
cur=${COMP_WORDS[COMP_CWORD]}
case "$prv" in
'-p')
IFS=$'\n' COMPREPLY=( $(ps axwwo "args") ) IFS=$' \t\n'
COMPREPLY=(${COMPREPLY[*]// */}) # remove arguments
ignore="0" # ps header
for ((nn=1; nn<${#COMPREPLY[*]}; nn++)); do
# filter by (partially) typed name in cur
# use " =~ ^$cur " for prefix match, without ^ it's substr match
[[ -n "$cur" && ! "${COMPREPLY[$nn]}" =~ $cur ]] && {
ignore="$nn $ignore"
} || {
# skip names matching PROCIGNORE[]
for ((mm=0; mm<${#PROCIGNORE[*]}; mm++)); do
[[ "${COMPREPLY[$nn]}" =~ ${PROCIGNORE[$mm]} ]] &&
ignore="$nn $ignore"
done
}
done
# remove unwanted, in reverse index order
for nn in $ignore; do unset COMPREPLY[$nn]; done
;;
*) COMPREPLY=()
;;
esac
}
complete -F _c8n_listprocs pstrace
Протестировано и используется на Linux с bash-3.x и bash-4.x. ps
Возможно, потребуется настройка параметров на платформах, отличных от Linux. Также должна поддерживаться truss
изменением одной строки.
Ограничения включают в себя:
- нет корректного экранирования потока ядра, как процесса
[names]
, это приведетpgrep
к тому, что (вероятно) не будет сделано то, что вы хотите - несоответствие процессов с пробелом в имени (
args
" используется вместо "comm
", поэтому его/paths
можно использовать, где это возможно)