Como faço para implementar um script wrapper pstrace
no bash que altera a interface do
[sudo] strace -c -p [PID]
para
[sudo] pstrace -c -p [PROCESS-NAME]
semelhante a como
killall [PROCESS-NAME]
é usado. Com finalização e tudo.
Responder1
A partir da versão 5.15, o strace pode imprimir os nomes dos comandos/processos:
- Implementada
--decode-pids=comm
opção (e seu alias-Y
) para impressão de nomes de comandos para PIDs.
Responder2
Teu isto:
ps auxw | grep [PROCESS-NAME] | awk '{print"-p " $2}' | xargs strace
Responder3
Requisitos aparentemente complicados :-)
Em duas partes, primeiramente o pstrace
script wrapper para strace
, usado pgrep
para a operação nome-para-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[@]}"
Para a segunda parte usarei bash
a conclusão programável para completar processos por nome, coloque isso no seu ~/.bash_profile
ou similar:
# 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
Testado e usado no Linux com bash-3.xe bash-4.x. ps
as opções podem precisar de ajustes em plataformas não-Linux, também devem suportar truss
uma alteração de uma linha.
As limitações incluem:
- nenhum escape correto do processo semelhante ao thread do kernel
[names]
, isso farápgrep
com que (provavelmente) não faça o que você deseja - incompatibilidade de processos com espaço no nome ("
args
" é usado em vez de "comm
" para que/paths
possa ser usado, quando disponível)