GNU/Linux >> Znalost Linux >  >> Linux

Jak určíte skutečný příkaz, který do vás vstupuje?

roura se objeví jako položka v seznamu otevřených deskriptorů souborů vašeho procesu:

 % ls -l /proc/PID/fd
 lr-x------ 1 xyz xyz 64 Feb 11 08:05 0 -> pipe:[124149866]
 lrwx------ 1 xyz xyz 64 Feb 11 08:05 1 -> /dev/pts/2
 lrwx------ 1 xyz xyz 64 Feb 11 08:05 2 -> /dev/pts/2
 lr-x------ 1 xyz xyz 64 Feb 11 08:05 10 -> /tmp/foo.sh

můžete také použít něco jako:

 % lsof -p PID
 sh      29890 xyz  cwd    DIR   0,44    4096  77712070 /tmp
 sh      29890 xyz  rtd    DIR   0,44    4096  74368803 /
 sh      29890 xyz  txt    REG   0,44   83888  77597729 /bin/dash
 sh      29890 xyz  mem    REG   0,44 1405508  79888619 /lib/tls/i686/cmov/libc-2.11.1.so
 sh      29890 xyz  mem    REG   0,44  113964  79874782 /lib/ld-2.11.1.so
 sh      29890 xyz    0r  FIFO    0,6         124149866 pipe
 sh      29890 xyz    1u   CHR  136,2                 4 /dev/pts/2
 sh      29890 xyz    2u   CHR  136,2                 4 /dev/pts/2
 sh      29890 xyz   10r   REG   0,44      66  77712115 /tmp/foo.sh

tak, než máte inode potrubí :), nyní můžete hledat každý jiný proces pod /proc/ pro tu trubku. pak budete mít příkaz, který vám bude zaslán:

 % lsof | grep 124149866 
 cat     29889 xyz    1w  FIFO                0,6          124149866 pipe
 sh      29890 xyz    0r  FIFO                0,6          124149866 pipe

v tomto příkladu cat přesměrováno do oddělení sh . v /proc/29889 můžete najít soubor s názvem cmdline což vám říká, jak se přesně jmenovalo:

 % cat /proc/29889/cmdline
 cat/dev/zero%  

pole příkazového řádku jsou oddělena NUL, takže to vypadá trochu ošklivě :)


Akira navrhl použít lsof .

Zde je návod, jak jej můžete naskriptovat:

whatpipe2.sh

#!/bin/bash

pid=$$
pgid=$(ps -o pgid= -p $pid)
lsofout=$(lsof -g $pgid)
pipenode=$(echo "$lsofout" | awk '$5 == "0r" { print $9 }')
otherpids=$(echo "$lsofout" | awk '$5 == "1w" { print $2 }')
for pid in $otherpids; do
    if cmd=$(ps -o cmd= -p $pid 2>/dev/null); then
        echo "$cmd"
        break
    fi
done

Spuštění:

$ tail -f /var/log/messages | ./whatpipe2.sh
tail -f /var/log/messages
^C

Dalším způsobem je použití skupin procesů.

whatpipe1.sh

#!/bin/bash    

pid=$$
# ps output is nasty, can (and usually does) start with spaces
# to handle this, I don't quote the "if test $_pgrp = $pgrp" line below
pgrp=$(ps -o pgrp= -p $pid)
psout=$(ps -o pgrp= -o pid= -o cmd=)
echo "$psout" | while read _pgrp _pid _cmd; do
    if test $_pgrp = $pgrp; then
        if test $_pid != $pid; then
            case $_cmd in
            ps*)
                # don't print the "ps" we ran to get this info
                # XXX but this actually means we exclude any "ps" command :-(
                ;;
            *)
                echo "$_cmd"
                ;;
            esac
        fi
    fi
done

Spuštění:

$ tail -f /var/log/messages | ./whatpipe1.sh
tail -f /var/log/messages
^C

Všimněte si, že oba fungují pouze v případě, že příkaz na levé straně roury běží dostatečně dlouho na ps vidět to. Řekl jste, že jej používáte s tail -f , takže pochybuji, že se jedná o problém.

$ sleep 0 | ./whatpipe1.sh 

$ sleep 1 | ./whatpipe1.sh
sleep 1

Linux
  1. Jak používat příkaz basename?

  2. Jak spustit příkaz, který zahrnuje přesměrování nebo potrubí pomocí Sudo?

  3. Jak určit kódování znaků, které terminál používá v programu C/c++?

  1. Linux – Jak nabootovat Puppy Linux Slacko do příkazového řádku?

  2. Jak zaznamenáte každý zadaný příkaz?

  3. příkaz tr - jak nahradit řetězec \n skutečným novým řádkem (\n)

  1. Jak nakonfigurovat příkazový řádek v Linuxu tak, aby zobrazoval aktuální adresář?

  2. Jak můžete vidět skutečný pevný odkaz pomocí ls?

  3. Jak používáte regulární výrazy s příkazem cp v Linuxu?