ฉันจะรันเชลล์สคริปต์เป็น daemon ภายใต้ Redhat ได้อย่างไร


12

ฉันมีเชลล์สคริปต์ซึ่งโดยพื้นฐานแล้วจะเป็นซับไฟล์หนึ่งที่มีการบันทึกซึ่งฉันพยายามเรียกใช้จากสคริปต์เริ่มต้น ฉันกำลังใช้daemonฟังก์ชั่นด้านในของ/etc/init.d/functionsการเรียกใช้เพราะ Redhat ไม่start-stop-daemonพร้อมใช้งาน เมื่อฉันเรียกสคริปต์เริ่มต้น ( /etc/init.d/script start) สคริปต์จะยังคงอยู่เบื้องหน้าแทนที่จะทำจนเสร็จและปล่อยให้กระบวนการทำงาน อะไรคือวิธีที่เหมาะสมสำหรับฉันที่จะกำจัดสคริปนี้?

สคริปต์ที่จะทำงาน:

# conf file where variables are defined
. /etc/script.conf

echo "Starting..." | logger -i
echo "Monitoring $LOG_LOCATION." | logger -i
echo "Sending to $MONITOR_HOST:$MONITOR_PORT." | logger -i

tail -n 1 -F $LOG_LOCATION |
grep WARN --line-buffered  |
/usr/bin/nc -vv $MONITOR_HOST $MONITOR_PORT 2>&1 |
logger -i

สคริปต์เริ่มต้น:

#!/bin/bash


# Source Defaults
. /etc/default/script

# Source init functions
. /etc/init.d/functions

prog=/usr/local/bin/script.sh

[ -f /etc/script.conf ] || exit 1

RETVAL=0

start()
{
    # Quit if disabled
    if ! $ENABLED; then
            echo "Service Disabled in /etc/default/script"
            exit 1
    fi

    echo "Starting $prog"

    daemon $prog

    RETVAL=$?

    return $RETVAL
}

stop ()
{
    echo -n $"Stopping $prog: "
    killproc $prog

    RETVAL=$?

    return $RETVAL
}

reload()
{
    echo "Reload command is not implemented for this service."
    return $RETVAL
}

restart()
{
    stop
    start
}

condrestart()
{
    echo "Not Implemented."
}

# See how we were called.
case "$1" in
    start)
        start
        ;;
    stop)
        stop
        ;;
    status)
        status $prog
        ;;
    restart)
        restart
        ;;
    reload)
        reload
        ;;
    condrestart)
        condrestart
        ;;
    *)
        echo $"Usage: $0 {start|stop|status|restart|condrestart|reload}"
        RETVAL=1
esac

~ 20 บรรทัดสุดท้ายของการประมวลผลด้วย bash -vx:

+ case "$1" in
+ start
+ true
+ echo 'Starting /usr/local/bin/script.sh'
Starting /usr/local/bin/script.sh
+ daemon /usr/local/bin/script.sh
+ local gotbase= force=
+ local base= user= nice= bg= pid=
+ nicelevel=0
+ '[' /usr/local/bin/script.sh '!=' /usr/local/bin/script.sh ']'
+ '[' -z '' ']'
+ base=script.sh
+ '[' -f /var/run/script.sh.pid ']'
+ '[' -n '' -a -z '' ']'
+ ulimit -S -c 0
+ '[' -n '' ']'
+ '[' color = verbose -a -z '' ']'
+ '[' -z '' ']'
+ initlog -q -c /usr/local/bin/script.sh

ดูเหมือนว่ามีประโยชน์สำหรับฉันที่คุณเรียกใช้สคริปต์ผ่านbash -vx ...และโพสต์บรรทัดสุดท้ายเพื่อให้เราสามารถเห็นสิ่งที่อยู่ในเบื้องหน้า
Hauke ​​Laging

1
ไม่ต้องกังวลเกี่ยวกับการใช้สิทธินี้และdaemonมีแพ็คเกจ RPMด้วย Btw มีเครื่องมือตรวจสอบบันทึกจำนวนมากอยู่ที่นั่น ( เริ่มต้นที่นี่ )
sr_

Hauke ​​คุณหมายถึงการใช้บรรทัดแรกของ#!/bin/bash -vx? ฉันพยายามทำสิ่งนี้ แต่มันไม่ได้สร้างผลลัพธ์เดียวกันจากสคริปต์เริ่มต้นเช่นเดียวกับถ้าฉันเรียกใช้เชลล์สคริปต์โดยตรง
bshacklett

@bshacklett คุณสามารถตรวจสอบฟังก์ชันใด ๆ ของสคริปต์เริ่มต้น (สคริปต์เชลล์ใด ๆ ) โดยเรียกใช้อย่างชัดเจนด้วยbash -vxเช่น bash -vx /etc/init.d/script start.
sr_

1
บันทึก @bshacklett Wrt ผมได้มองใกล้ที่logstash ร้านค้าสามารถป้อนบันทึกโดยตรงจาก Log4j แต่ตัวแทน logstash ยังสามารถตรวจสอบไฟล์บันทึก
sr_

คำตอบ:


2

ฉันพบสคริปต์ที่http://www.linuxforums.org/forum/programming-scripting/190279-daemon-etc-init-d-functions-does-not-return-launching-process.html#post897522ซึ่งฉันสามารถ เพื่อปรับเปลี่ยนให้เหมาะกับความต้องการของฉัน ด้วยตนเองติดตาม PID และสร้างไฟล์ PID pidofใช้ ฉันต้องแก้ไขสิ่งนี้เพื่อใช้pgrepเนื่องจากpidofไม่สามารถดู PID ของสคริปต์ของฉันได้ หลังจากปรับเปลี่ยนแล้วก็ใช้งานได้ดี * หมายเหตุ pgrep ดูเหมือนว่าจะใช้งานได้หากชื่อเต็มของสคริปต์มีความยาวไม่เกิน 15 ตัวอักษร

นี่คือสิ่งที่ฉันลงเอยด้วย:

#!/bin/bash
#
# 
#
# Start on runlevels 3, 4 and 5. Start late, kill early.
# chkconfig: 345 95 05
#
#
#!/bin/bash

# absolute path to executable binary
progpath='/usr/local/bin/script.sh'

# arguments to script
opts=''

# binary program name
prog=$(basename $progpath)

# pid file
pidfile="/var/run/${prog}.pid"

# make sure full path to executable binary is found
! [ -x $progpath ] && echo "$progpath: executable not found" && exit 1

eval_cmd() {
  local rc=$1
  if [ $rc -eq 0 ]; then
    echo '[  OK  ]'
  else
    echo '[FAILED]'
  fi
  return $rc
}

start() {
  # see if running
  local pids=$(pgrep $prog)

  if [ -n "$pids" ]; then
    echo "$prog (pid $pids) is already running"
    return 0
  fi
  printf "%-50s%s" "Starting $prog: " ''
  $progpath $opts &

  # save pid to file if you want
  echo $! > $pidfile

  # check again if running
  pgrep $prog >/dev/null 2>&1
  eval_cmd $?
}

stop() {
  # see if running
  local pids=$(pgrep $prog)

  if [ -z "$pids" ]; then
    echo "$prog not running"
    return 0
  fi
  printf "%-50s%s" "Stopping $prog: " ''
  rm -f $pidfile
  kill -9 $pids
  eval_cmd $?
}

status() {
  # see if running
  local pids=$(pgrep $prog)

  if [ -n "$pids" ]; then
    echo "$prog (pid $pids) is running"
  else
    echo "$prog is stopped"
  fi
}

case $1 in
  start)
    start
    ;;
  stop)
    stop
    ;;
  status)
    status
    ;;
  restart)
    stop
    sleep 1
    start
    ;;
  *)
    echo "Usage: $0 {start|stop|status|restart}"
    exit 1
esac

exit $?

0

ฉันไม่รู้ Redhat แต่daemon $prog &ดูแปลกสำหรับฉัน หากมีฟังก์ชั่นสำหรับ daemonizing อยู่แล้วทำไมจึงมีความจำเป็น (และมีประโยชน์) ในการทำให้ฟังก์ชั่นนี้อยู่ในพื้นหลัง? &ดังนั้นลองโดยไม่ต้อง


4
มันไม่ผิดหรอก /etc/init.d/functionsกำหนดdaemonฟังก์ชั่นที่คาดว่าอาร์กิวเมนต์จะ daemonize ตัวเองดูแลสิ่งต่าง ๆ เช่นการเปลี่ยนผู้ใช้การตั้งค่า ulimits การตรวจสอบ (ไม่สร้าง!) pidfile ... การใช้งานที่ดีที่สุดของdaemonฟังก์ชันนี้คือแทนที่ด้วยlibslack'sdaemon ;)
sr_

ฉันขอโทษ & อยู่ที่จุดหนึ่งเมื่อฉันแก้ไขปัญหา ฉันไม่ได้ตั้งใจจะรวมไว้ในโพสต์นี้
bshacklett
โดยการใช้ไซต์ของเรา หมายความว่าคุณได้อ่านและทำความเข้าใจนโยบายคุกกี้และนโยบายความเป็นส่วนตัวของเราแล้ว
Licensed under cc by-sa 3.0 with attribution required.