shell启动/停止python脚本

时间:2013-07-16 20:27:58

标签: python linux shell

我有一个简单的python脚本,我需要启动和停止,我需要使用start.sh和stop.sh脚本来完成它。

我有start.sh:

#!/bin/sh

script='/path/to/my/script.py'
echo 'starting $script with nohup'

nohup /usr/bin/python $script &

和stop.sh

#!/bin/sh

PID=$(ps aux | grep "/path/to/my/script.py" | awk '{print $2}')
echo "killing $PID"
kill -15 $PID

我主要关注stop.sh脚本。我认为这是找到pid的合适方式,但我不会对它下注太多。 start.sh成功启动它。当我运行stop.sh时,我无法再通过"ps aux | grep 'myscript.py'"找到该进程,但控制台输出:

killing 25052
25058
./stop.sh: 5: kill: No such process

因此它似乎有效并且给出了“没有这样的过程”的错误。

这实际上是一个错误吗?我是否以理智的方式接近这个?还有其他我应该注意的事情吗?

编辑 - 我实际上最终得到了这样的结果: start.sh

#!/bin/bash
ENVT=$1
COMPONENTS=$2


TARGETS=("/home/user/project/modules/script1.py" "/home/user/project/modules/script2.py")
for target in "${TARGETS[@]}"
do
      PID=$(ps aux | grep -v grep | grep $target | awk '{print $2}')
      echo $PID
      if [[ -z "$PID" ]]
      then
              echo "starting $target with nohup for env't: $ENVT"
              nohup python $target $ENVT $COMPONENTS &
      fi
done

stop.sh

#!/bin/bash
ENVT=$1

TARGETS=("/home/user/project/modules/script1.py" "/home/user/project/modules/script2.py")
for target in "${TARGETS[@]}"
do
     pkill -f $target
     echo "killing process $target"
done

5 个答案:

答案 0 :(得分:4)

这是因为ps aux |grep SOMETHING也找到grep SOMETHING进程,因为SOMETHING匹配。执行完grep后,无法找到它。

添加一行:ps aux | grep -v grep | grep YOURSCRIPT

其中-v表示排除。更多man grep

答案 1 :(得分:2)

init-type脚本对此很有用。这与我使用的非常相似。您将pid存储在一个文件中,当您想检查它是否正在运行时,请查看/ proc文件系统。

#!/bin/bash

script_home=/path/to/my
script_name="$script_home/script.py"
pid_file="$script_home/script.pid"

# returns a boolean and optionally the pid
running() {
    local status=false
    if [[ -f $pid_file ]]; then
        # check to see it corresponds to the running script
        local pid=$(< "$pid_file")
        local cmdline=/proc/$pid/cmdline
        # you may need to adjust the regexp in the grep command
        if [[ -f $cmdline ]] && grep -q "$script_name" $cmdline; then
            status="true $pid"
        fi
    fi
    echo $status
}

start() {
    echo "starting $script_name"
    nohup "$script_name" &
    echo $! > "$pid_file"
}

stop() {
    # `kill -0 pid` returns successfully if the pid is running, but does not
    # actually kill it.
    kill -0 $1 && kill $1
    rm "$pid_file"
    echo "stopped"
}

read running pid < <(running)

case $1 in 
    start)
        if $running; then
            echo "$script_name is already running with PID $pid"
        else
            start
        fi
        ;;
    stop)
        stop $pid
        ;;
    restart)
        stop $pid
        start
        ;;
    status)
        if $running; then
            echo "$script_name is running with PID $pid"
        else
            echo "$script_name is not running"
        fi
        ;;
    *)  echo "usage: $0 <start|stop|restart|status>"
        exit
        ;;
esac

答案 2 :(得分:1)

ps aux | grep“/path/to/my/script.py”

将返回script.py实例的pid以及grep的此实例。这可能就是为什么你没有这样的过程:当你开始杀死grep时,它已经死了。

答案 3 :(得分:1)

"correct"方法可能是让您的脚本将其pid写入/ var / run中的文件,并在您终止脚本时清除它。如果无法更改脚本,请查看start-stop-daemon

如果您想继续使用类似grep的方法,请查看proctools。它们内置于大多数GNU / Linux机器上,并且可以在BSD上轻松获得,包括OS X:

pkill -f /path/to/my/script.py

答案 4 :(得分:0)

我目前没有打开unix框,所以我无法对此进行测试,但获得这个想法应该相当简单。

start.sh:

if [ -e ./temp ]
then
  pid=`cat temp`
  echo "Process already exists; $pid"
else
  script='/path/to/my/script.py'
  echo 'starting $script with nohup'
  nohup /usr/bin/python $script &
  echo $! > temp
fi

stop.sh:

if [ -e ./temp ]
then
  pid=`cat temp`
  echo "killing $pid"
  kill -15 $PID
  rm temp
else
  echo "Process not started"
fi

试试这个。