How to prevent "ps" reporting its own process?

22,261

Solution 1

My answer is a variation on the typical answer for searching for "foobar" in a ps listing. The argument of "-A" "ps" is more portable than "aux", I believe, but this change is irrelevant to the answer. The typical answer looks like this:

$ ps -A -ww | grep [f]oobar

Instead I use this pattern:

$ ps -A -ww | grep [^]]foobar

The main advantage is that it's easier to write scripts based on this patterns because you simply concatenate a static string [^]] with whatever pattern you are looking for. You don't need to strip off the first letter of the string then insert it between the square braces and then concatenate the that back together again. When scripting in shell it's easier to simply stick [^]] in front of the pattern you were lookginfor looking for. String slicing in Bash is an ugly thing, so my variation avoids that. This variation says show the lines where the pattern matches WITHOUT a leading right-square-bracket ]. Since the search pattern to exclude a square bracket actually adds the square bracket to the pattern then it will never match itself.

So you could write a portable psgrep command as follows. Here, I make some allowance for differences between Linux, OS X BSD, and others. This adds the column headers from ps, provides a more custom ps format that suits my needs betters, and displays processes listing extra, extra wide so that none of the command-line arguments are missed. Well, most are not missed. Java being Java, it often does things in the worst possible way, so you some java services will run past the maximum allowed length of arguments that the process table will keep track of. I believe this is 1024 characters. The command-lone length allowed to start a process is much longer, but the kernel process table doesn't bother to keep track of anything over 1K in length. Once the command is started the command-name and argument list isn't needed against, so what gets stored in the process table is just informational.

psgrep ()
{
    pattern=[^]]${1};
    case "$(uname -s)" in
        Darwin)
            ps -A -ww -o pid,ppid,nice,pri,pcpu,pmem,etime,user,wchan,stat,command | grep -i -e "^[[:space:]]*PID" -e ${pattern}
        ;;
        Linux)
            ps -A -ww -o pid,ppid,tid,nice,pri,pcpu,pmem,etime,user,wchan:20,stat,command | grep -i -e "^[[:space:]]*PID" -e ${pattern}
        ;;
        *)  # other UNIX flavors get a minimalist version.
            ps -A -ww | grep -i -e ${pattern}
        ;;
    esac
}

Solution 2

+1 for @jamzed terse answer, however the OP might need some explanation:

ps | grep "[d]jango"

Using that regex you are launching a process which its ps string will not match itself, since the regexp matches "django" and not "[d]jango". That way you'll exclude the process that has the string "[d]jango" which in this case is grep; The same can be applied to pgrep, egrep, awk, sed, etc... whichever command you used to define the regex.

From man 7 regex

   A bracket expression is a list of characters enclosed in "[]".  It nor‐
   mally matches any single character from the list (but see  below).   If
   the  list  begins  with  '^',  it matches any single character (but see
   below) not from the rest of the list.  If two characters  in  the  list
   are  separated  by '-', this is shorthand for the full range of charac‐
   ters between those two (inclusive) in the collating sequence, for exam‐
   ple,  "[0-9]" in ASCII matches any decimal digit.  It is illegal(!) for
   two ranges to share an endpoint, for example, "a-c-e".  Ranges are very
   collating-sequence-dependent,  and portable programs should avoid rely‐
   ing on them.

Solution 3

ps | grep [d]jango

ps | grep d[j]ango

...

ps | grep djang[o]

Solution 4

Use pgrep instead: pgrep -lf django

Solution 5

Oh wait, this works:

ps | grep django | grep -v grep
Share:
22,261

Related videos on Youtube

Steve Bennett
Author by

Steve Bennett

Updated on September 18, 2022

Comments

  • Steve Bennett
    Steve Bennett over 1 year
    $ ps | grep django
    28006 ttys004    0:01.12 /usr/bin/python bin/django celeryd --beat
    51393 ttys005    0:01.45 /usr/bin/python bin/django celeryd -l INFO
    51472 ttys005    0:01.29 /usr/bin/python bin/django celeryd -l INFO
    51510 ttys005    0:01.89 /usr/bin/python bin/django celeryd -l INFO
    51801 ttys005    0:01.83 /usr/bin/python bin/django celeryd -l INFO
    53470 ttys005    0:03.97 /usr/bin/python bin/django celeryd -l INFO
    53780 ttys005    0:00.00 grep django
    

    Is there a way to prevent the last process (that is, the grep that was started at the same time as my ps command) being reported?

    (I started trying to come up with a regex that would match the literal but not match itself, but that seemed, um, not the right approach...)

  • Steve Bennett
    Steve Bennett about 12 years
    Cool. I'm actually pretty comfortable with regexs but couldn't immediately think of a way to prevent the regexp matching itself. Enclosing a letter in square brackets makes perfect sense. (Including something like [^!] would also work...)
  • Steve Bennett
    Steve Bennett about 12 years
    As usual I forgot to mention the platform (OS X in this case). Presumably pgrep works on various linuxes.
  • user
    user about 12 years
    Only if the process command line doesn't legitimately include grep, which you cannot count on in the general case.
  • ash
    ash about 12 years
    That's nice and crafty.
  • plat_winz
    plat_winz almost 11 years
    still shows grep on mine...
  • Steve Bennett
    Steve Bennett over 10 years
    Yep, that works for me on OS X.
  • Asclepius
    Asclepius over 9 years
    Doesn't work so well on Linux.
  • Neromancer
    Neromancer over 9 years
    For the 'ps' specific case, I use '[ ]' at the front of the process name I am searching for. Then I don't need to parse the process name specially for the regex, but it still matches.
  • A.D.
    A.D. almost 9 years
    Add space if you need to grep one char: ps aux| grep "[Z] "
  • deltab
    deltab about 8 years
    The manual states "The running pgrep or pkill process will never report itself as a match.", and indeed I've not seen it do so.
  • tripleee
    tripleee about 8 years
    More generally, the options to ps are notoriously non-portable, so without information about which platform this is for, this answer isn't very helpful. Furthermore, this is obviously not adequate when you are not sure that the process you are looking for is not a process leader (i.e. this may help if your target is a daemon, but not generally otherwise).
  • SOUser
    SOUser over 6 years
    @hmontoliu It does not work for example: ps aux | grep [s]cript1. Could you help to comment about the solution ?
  • SOUser
    SOUser over 6 years
    @jamzed It does not work for example: ps aux | grep [s]cript1 or ps aux | grep [s]cript2. The grep line is still shown. Could you help to comment about the solution ?
  • SOUser
    SOUser over 6 years
    @jamzed My fault. It seems the line is shown because of the previous searches...
  • SOUser
    SOUser over 6 years
    @hmontoliu My fault. It seems the line is shown because of the previous searches...
  • Læti
    Læti over 5 years
    Disadvantage is that this will actually match one char more (in front) than the original pattern. For exemple, this will never match the PID. And can be a bit misleading when used with grep --colour.
  • Kelly
    Kelly about 5 years
    I've just been dealing with a problem where I thought pgrep was matching itself. Turns out it was matching the name of the bash script file I was running it from. Adding -x fixed it, then it does an exact match on the command name.
  • Steve Bennett
    Steve Bennett almost 3 years
    What is watch meant to do in this context?
  • pt1997
    pt1997 almost 3 years
    it is just used as a demo process, running in the background, to search for.