ฉันจะทราบได้อย่างไรว่าคำสั่งกำลังทำงานอยู่หรือรอการป้อนข้อมูลจากผู้ใช้


14

ในบรรทัดคำสั่งฉันพิมพ์คำสั่งแล้วกด Enter มันไม่ส่งออกอะไรเลย ฉันจะบอกได้อย่างไรว่ามันกำลังทำงานอยู่และยังไม่ได้ส่งออกหรือกำลังขอให้ผู้ใช้ป้อนข้อมูล?


ถ้ามันกำลังรออยู่คุณจะไม่ได้รับPS1พรอมต์
Prvt_Yadav

1. โปรดบอกเราว่าโปรแกรมใด (เงียบ) และเราสามารถให้คำแนะนำที่แม่นยำยิ่งขึ้นเกี่ยวกับสิ่งที่คาดหวังและวิธีตรวจสอบ 2. คุณต้องการที่จะได้รับการแจ้งเตือนเมื่อโปรแกรมขอในที่สุดสำหรับการป้อนข้อมูลหรือเสร็จสิ้น (เพื่อให้มีสิ่งที่เขียนไปยังหน้าต่างเทอร์มินัล?
sudodus

หากคาดว่าจะมีการป้อนข้อมูลฉันจะถือว่าคุณได้รับพร้อมท์กับข้อความที่อยู่ข้างหน้ามันเพื่อขอให้ป้อนข้อมูล
Rinzwind

5
@ Rinzwind - นั่นเป็นข้อสมมติฐานที่แย่ ตัวอย่างแรกที่จำได้คือcatคำสั่ง เพียงพิมพ์catด้วยตัวเองและมันจะรออินพุตจาก stdin แต่ไม่แสดงพรอมต์ คำสั่งอื่น ๆหลายตัวทำงานในลักษณะเดียวกันเนื่องจากคาดว่าอินพุตจาก stdin หรือไฟล์ แต่ไม่ได้แยกความแตกต่างระหว่างแหล่งอินพุตที่แตกต่างกัน (เทอร์มินัลโต้ตอบ, ไพพ์, ไฟล์ ... )
Dave Sherohman

คำตอบ:


15

มีหลายวิธี:

  1. ลองส่งสัญญาณการสิ้นสุดของอินพุต : หากไม่มีสิทธิ์ superuser มันเป็นการยากที่จะรู้ว่าเกิดอะไรขึ้นภายใต้ประทุน สิ่งที่สามารถทำได้คือการกด+Ctrl dเทอร์มินัลและยูทิลิตีในโหมด canonicalส่งข้อความที่มีอยู่ทั้งหมดไปยังread()syscall เมื่อได้รับสัญญาณ EOT ที่ถูกผูกไว้กับคีย์ผสมนี้และหากไม่มีอินพุต - read()จะส่งคืนสถานะการออกเชิงลบซึ่งยูทิลิตี้ส่วนใหญ่ยอมรับว่าเป็นสัญญาณออก ดังนั้นหากยูทิลิตี้กำลังรอการป้อนข้อมูลมันจะออกเมื่อได้รับการรวมกันที่สำคัญ มิฉะนั้นยูทิลิตีอาจกำลังทำงานหรือเขียนไม่ถูกต้อง

  2. การสอดแนมบน syscalls : หากคุณมีสิทธิ์ superuser คุณสามารถเรียกใช้straceในเทอร์มินัลอื่นเพื่อดูสิ่งที่กำลังทำอยู่ เพื่อที่คุณจะต้องค้นหา PID ของโปรแกรม ยกตัวอย่างเช่นในแท็บวิ่งสถานีอื่นpgrep -f firefoxซึ่งอาจ 1234 sudo strace -f -p 1234เป็นตัวอย่างแล้ว หากเอาต์พุตที่คุณเห็นติดอยู่บนread()syscall หมายความว่าคำสั่งอาจกำลังรออินพุต มิฉะนั้นถ้าคุณเห็น syscalls ทำงานอยู่คำสั่งจะทำอย่างอื่น ดูคำถามที่เกี่ยวข้องสำหรับการใช้งานของstraceยังจะคิดออกว่าคำสั่งทำงานนานออกจาก

  3. ใช้วิธีการของตัวเองของคำสั่ง : เหนือสิ่งอื่นใดสาธารณูปโภคเช่นการddใช้สัญญาณ ตัวอย่างเช่นหากคุณใช้kill -USR1 1234(โดยที่ 1234 เป็น PID ของddคำสั่งที่กำลังรัน) จะพิมพ์เพื่อ stdout จำนวนไบต์ที่ประมวลผลในปัจจุบัน แน่นอนว่าสิ่งนี้จำเป็นต้องรู้เกี่ยวกับพฤติกรรมของคำสั่งตั้งแต่แรก สองวิธีข้างต้นนั้นกว้างกว่าและไม่ต้องการความรู้เชิงลึกเกี่ยวกับพฤติกรรมของคำสั่งแต่ละอัน (แม้ว่าจะเป็นการดีที่สุดที่จะรู้ว่าคุณกำลังใช้งานอะไรอยู่ - ไม่เช่นนั้นคุณอาจเสี่ยงต่อการรันคำสั่งซึ่งอาจสร้างความเสียหาย)


+1 ขอบคุณสำหรับstraceวิธีการ :-) แต่วิธีการที่ง่ายกว่านั้นก็มีประโยชน์ (ทั่วไปหรือเฉพาะสำหรับแต่ละโปรแกรม) บางคนทำงานโดยไม่มีสิทธิ์ superuser ตัวอย่าง: ตรวจสอบว่าddกำลังทำอะไรอยู่และตรวจสอบสาเหตุที่grep --color asdfรออย่างเงียบ ๆ
sudodus

@sudodus อาเตือนที่ดีเกี่ยวกับddฉันจะเพิ่มที่
Sergiy Kolodyazhnyy

คุณไม่จำเป็นต้องมีสิทธิ์ superuser ในการดีบักกระบวนการที่ผู้ใช้ของคุณเป็นเจ้าของ ดีถ้าคุณยังไม่ได้ตั้งค่าระบบอย่างเหมาะสม
Ruslan

6

จะทราบได้อย่างไรว่าโปรแกรมทำงานหรือต้องการให้ผู้ใช้ป้อนข้อมูล

ขึ้นอยู่กับโปรแกรมและวิธีที่คุณเรียกใช้

  • บ่อยครั้ง แต่ไม่เสมอไปจะมีพรอมต์แสดงว่าโปรแกรมกำลังขอให้ป้อนข้อมูล

  • หากคุณไม่แน่ใจคุณสามารถตรวจสอบว่ากระบวนการของโปรแกรมไม่ว่างหรือไม่

    • ใช้ CPU - ใช้topหรือhtop

    • อ่านหรือเขียน - ใช้ sudo iotop -o

  • และเมื่อโปรแกรมเสร็จสิ้นคุณจะเห็นพรอมต์ของเชลล์

Shellscript running

ฉันมี shellscript ที่ตรวจสอบว่าโปรแกรมกำลังทำงานอยู่หรือไม่และตอนนี้ฉันได้เพิ่มตัวเลือก-sเพื่อให้ทำงานได้sudo strace -f -p <PID>(ตามคำตอบของ Sergiy Kolodyazhnyy) เมื่อพบ ...

shellscript ใช้

  • ps -ef เพื่อค้นหาโปรแกรมส่วนใหญ่
  • systemctl is-active --quiet เพื่อค้นหาบางโปรแกรม
  • และถ้าคุณต้องการstraceในxtermหน้าต่าง

    ติดตั้งxtermหากคุณต้องการใช้straceเพื่อดูกิจกรรมของโปรแกรม

การใช้

$ ./running
Usage:    ./running <program-name>
          ./running <part of program name>
Examples: ./running firefox
          ./running term                     # part of program name
          ./running dbus
          ./running 'dbus-daemon --session'  # words with quotes
          ./running -v term                  # verbose output
          ./running -s term                  # strace checks activity

คุณสามารถติดตั้ง shellscript runningลงในไดเรกทอรีในPATHถ้าคุณต้องการเข้าถึงได้ง่าย

รหัส shellscript

#!/bin/bash

# date        sign     comment
# 2019-02-14  sudodus  version 1.0

verbose=false
strace=false
if [ "$1" == "-v" ]
then
 verbose=true
 shift
fi
if [ "$1" == "-s" ]
then
 strace=true
 shift
fi

if [ $# -ne 1 ]
then
 echo "Usage:    $0 <program-name>
          $0 <part of program name>
Examples: $0 firefox
          $0 term                     # part of program name
          $0 dbus
          $0 'dbus-daemon --session'  # words with quotes
          $0 -v term                  # verbose output
          $0 -s term                  # strace checks activity"
 exit
fi

inversvid="\0033[7m"
resetvid="\0033[0m"
redback="\0033[1;37;41m"
greenback="\0033[1;37;42m"
blueback="\0033[1;37;44m"

runn=false
#tmpfil=$(mktemp)
tmpdir=$(mktemp -d)
tmpfil="$tmpdir/tmpfil"
vtfile="$tmpdir/vtfile"
vthead="$tmpdir/vthead"

# check by systemctl

systemctl is-active --quiet "$1"
if [ $? -eq 0 ]
then
 echo "systemctl is-active:"
 runn=true
fi

# check by ps

ps -ef | tr -s ' ' ' ' | cut -d ' ' -f 8- | grep "$1" | grep -vE -e "$0 *$1" -e "$0 *.* *$1" -e "grep $1" | sort -u > "$tmpfil"
#cat "$tmpfil"
if $verbose || $strace
then
 ps -ef |head -n1 > "$vthead"
 ps -ef | grep "$1" | grep -vE -e "$0 *.* *$1" -e "grep $1" | sort -u > "$vtfile"
fi

tmpstr=$(head -n1 "$tmpfil")
#echo "tmpstr=$tmpstr"
tmpess=$(grep -om1 "$1" "$tmpfil")
#echo "tmpess=$tmpess"
if [ "$tmpstr" == "$1" ] || [ "${tmpstr##*/}" == "$1" ] || [ "${1##*/}" == "${0##*/}" ] || [ "$tmpess" == "$1" ]
then
 echo "ps -ef: active:"
 runn=true
 if $verbose
 then
  cat "$vthead" "$vtfile"
 fi
elif test -s "$tmpfil"
then
 if $runn
 then
  echo "----- consider also ------------------------------------------------------------"
  if $verbose
  then
   cat "$vthead" "$vtfile"
  else
   cat "$tmpfil"
  fi
  echo "--------------------------------------------------------------------------------"
 else
  echo "----- try with: ----------------------------------------------------------------"
  if $verbose
  then
   cat "$vthead" "$vtfile"
  else
   cat "$tmpfil"
  fi
  echo "--------------------------------------------------------------------------------"
 fi
fi

if $runn
then
 echo -en "$greenback '$1"
 if [ "$tmpstr" != "$tmpess" ]
 then
  echo -n " ..."
 fi
 echo -e "' is running $resetvid"

 if $strace
 then
  which xterm
  if [ $? -eq 0 ]
  then
   pid=$(head -n1 "$vtfile" | sed 's/^ *//' | tr -s ' ' '\t' | cut -f 2)
   echo "checking pid=$pid; quit with 'ctrl + c' in the xterm window"
   xterm -title "'strace' checks '$1'" 2> /dev/null -e sudo strace -f -p $pid
  else
   echo "Please install 'xterm' for this function to work"
   exit
  fi
 fi
else
 inpath=$(which "$1")
 if [ "$inpath" == "" ]
 then
  echo -e "$redback no path found to '$1' $resetvid"
 else
  echo -e "$blueback '$1' is not running $resetvid"
 fi
fi
rm -r "$tmpdir"

การสาธิต

การตรวจสอบหน้าต่างเทอร์มินัลใน Lubuntu (LXTerminal เริ่มเป็นx-terminal-emulatorและgnome-terminalหน้าต่างแบบกำหนดเอง)

$ running -v -s term 
----- try with: ----------------------------------------------------------------
UID        PID  PPID  C STIME TTY          TIME CMD
sudodus   2087  1384  0 13:33 ?        00:00:00 x-terminal-emulator
sudodus   2108  1269  0 13:33 ?        00:00:17 /usr/lib/gnome-terminal/gnome-terminal-server
--------------------------------------------------------------------------------
 no path found to 'term' 

$ running -v -s x-terminal-emulator
ps -ef: active:
UID        PID  PPID  C STIME TTY          TIME CMD
sudodus   2087  1384  0 13:33 ?        00:00:00 x-terminal-emulator
 'x-terminal-emulator' is running 
/usr/bin/xterm
checking pid=2087; quit with 'ctrl + c' in the xterm window

มีจำนวนมากของกิจกรรมเร็วที่สุดเท่าที่เคอร์เซอร์อยู่ในหน้าต่าง terminal

ป้อนคำอธิบายรูปภาพที่นี่

เริ่มต้นgrep(กำลังรออินพุตจาก/dev/stdin)

$ grep -i --color 'hello'
asdf
Hello World    
Hello World

ตรวจสอบมัน

$ running -s grep
ps -ef: active:
 'grep ...' is running 
/usr/bin/xterm
checking pid=14982; quit with 'ctrl + c' in the xterm window

มีกิจกรรมไม่มากนักและคุณสามารถระบุได้ว่าเกิดอะไรขึ้น

ป้อนคำอธิบายรูปภาพที่นี่


พูดถึงดีถึงiotopแม้ว่าการใช้งาน CPU อาจไม่จำเป็นต้องเป็นตัวบ่งชี้หากกระบวนการไม่ว่าง โปรแกรมที่เขียนด้วย C และปรับให้เหมาะสมอาจใช้ CPU น้อยที่สุด ตัวบ่งชี้บางตัวที่ฉันเขียนใน Python กำหนดการทำงานซ้ำเพื่อให้ทำงานดังนั้นอาจใช้ CPU เพื่ออัปเดตเมนูตัวบ่งชี้ในช่วงเวลาสั้น ๆ จากนั้นก็นั่งอยู่ตรงนั้น
Sergiy Kolodyazhnyy

@SergiyKolodyazhnyy ใช่คุณพูดถูก straceวิธีการจะดีกว่า แต่อาจจะไม่จำเป็นหรือไม่สามารถใช้ได้
sudodus

ตกลง ฉันไม่คิดว่ามันติดตั้งไว้ล่วงหน้ากับ Ubuntu และอาจ overkill
Sergiy Kolodyazhnyy

1

ไม่แน่ใจว่าคุณยังต้องการสิ่งนี้หรือไม่ แต่ยังมีเคล็ดลับที่มีประโยชน์ให้ทราบ: หากโปรแกรมดูเหมือนว่าจะออกโดยไม่มีเอาต์พุตใด ๆ คุณสามารถตรวจสอบว่ามันกำลังทำงานอยู่เบื้องหลังหรือไม่

ps -efa | grep "program_name"

ไชโย!


1

หากคุณใช้เชลล์ในเทอร์มินัลตัวอย่างเช่นเทอร์มินัลอีมูเลเตอร์หรือเซสชัน ssh ทั่วไปเชลล์ของคุณเปิดใช้งานการควบคุมงานได้อย่างแน่นอน สิ่งนี้ทำให้การตอบคำถามของคุณเป็นเรื่องง่ายสุด ๆ ในกรณีส่วนใหญ่

พิมพ์Ctrl+Zเพื่อหยุดกระบวนการและbgดำเนินการต่อในพื้นหลังจากนั้นพิมพ์บรรทัดว่างลงในเชลล์เพื่อตรวจสอบว่าโปรแกรมหยุดทำงานโดยสัญญาณหรือไม่

หากกระบวนการพยายามอ่านจากเทอร์มินัลกระบวนการจะรับSIGTTINสัญญาณทันทีและจะถูกระงับ (เมื่อเปิดใช้งานการควบคุมงานระบบจะอนุญาตให้อ่านเพียงหนึ่งกระบวนการในแต่ละครั้งจากเทอร์มินัล) เชลล์จะรายงานสิ่งนี้ จากนั้นคุณสามารถพิมพ์fgเพื่อทำกระบวนการต่อในเบื้องหน้าแล้วพิมพ์อินพุตที่โปรแกรมอ่านตามปกติ

mp@ubuntu:~$ sleep 30 # a program that is not reading from the terminal
^Z
[1]+  Stopped                 sleep 30
mp@ubuntu:~$ bg
[1]+ sleep 30 &
mp@ubuntu:~$ 
mp@ubuntu:~$ 


mp@ubuntu:~$ cat - # a program that is reading from the terminal
^Z
[1]+  Stopped                 cat -
mp@ubuntu:~$ bg
[1]+ cat - &
mp@ubuntu:~$ 
[1]+  Stopped                 cat -
mp@ubuntu:~$ jobs -l
[1]+  3503 Stopped (tty input)     cat -
mp@ubuntu:~$ fg
cat -
hi
hi

บางโปรแกรมเช่นโปรแกรมแก้ไขจะดักจับหรือไม่สนใจสัญญาณที่สร้างโดยCtrl+Zหรือทำให้เทอร์มินัลเข้าสู่โหมดที่อักขระควบคุมไม่ได้สร้างสัญญาณ คุณจะต้องใช้เทคนิคขั้นสูงในกรณีนี้เช่นการใช้straceเพื่อดูว่ากระบวนการที่จะทำread, select, pollฯลฯ

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