यह पता लगाना आसान है कि शेल स्क्रिप्ट कब शुरू होती है, लेकिन यह जानना हमेशा आसान नहीं होता कि यह कब रुकती है। एक स्क्रिप्ट सामान्य रूप से समाप्त हो सकती है, जैसा कि इसके लेखक ने इसे समाप्त करने का इरादा किया है, लेकिन यह एक अप्रत्याशित घातक त्रुटि के कारण भी विफल हो सकता है। कभी-कभी स्क्रिप्ट के विफल होने पर जो कुछ भी प्रगति पर था, उसके अवशेषों को संरक्षित करना फायदेमंद होता है, और दूसरी बार यह असुविधाजनक होता है। किसी भी तरह से, किसी स्क्रिप्ट के अंत का पता लगाना और उस पर कुछ पूर्व-गणना तरीके से प्रतिक्रिया करना यही कारण है कि बैश trap
निर्देश मौजूद है।
असफलता का जवाब
यहां एक उदाहरण दिया गया है कि कैसे एक स्क्रिप्ट में एक विफलता भविष्य में विफलताओं का कारण बन सकती है। मान लें कि आपने एक प्रोग्राम लिखा है जो /tmp
. में एक अस्थायी निर्देशिका बनाता है ताकि यह फ़ाइलों को एक अलग प्रारूप में वापस एक साथ बंडल करने से पहले संग्रह से हटा सके और संसाधित कर सके:
#!/usr/bin/env bash
CWD=`pwd`
TMP=${TMP:-/tmp/tmpdir}
## create tmp dir
mkdir "${TMP}"
## extract files to tmp
tar xf "${1}" --directory "${TMP}"
## move to tmpdir and run commands
pushd "${TMP}"
for IMG in *.jpg; do
mogrify -verbose -flip -flop "${IMG}"
done
tar --create --file "${1%.*}".tar *.jpg
## move back to origin
popd
## bundle with bzip2
bzip2 --compress "${TMP}"/"${1%.*}".tar \
--stdout > "${1%.*}".tbz
## clean up
/usr/bin/rm -r /tmp/tmpdir
ज्यादातर समय, स्क्रिप्ट उम्मीद के मुताबिक काम करती है। हालाँकि, यदि आप गलती से इसे अपेक्षित JPEG फ़ाइलों के बजाय PNG फ़ाइलों से भरे संग्रह पर चलाते हैं, तो यह आधे रास्ते में विफल हो जाता है। एक विफलता दूसरे की ओर ले जाती है, और अंत में, स्क्रिप्ट अस्थायी निर्देशिका को हटाने के लिए अपने अंतिम निर्देश तक पहुंचे बिना बाहर निकलती है। जब तक आप निर्देशिका को मैन्युअल रूप से हटाते हैं, आप जल्दी से ठीक हो सकते हैं, लेकिन यदि आप ऐसा करने के लिए आस-पास नहीं हैं, तो अगली बार जब स्क्रिप्ट चलती है, तो उसे अप्रत्याशित बचे हुए फ़ाइलों से भरी मौजूदा अस्थायी निर्देशिका से निपटना होगा।पी>
इसका मुकाबला करने का एक तरीका यह है कि स्क्रिप्ट की शुरुआत में एहतियाती निष्कासन जोड़कर तर्क को उलट दिया जाए और उसे दोगुना कर दिया जाए। वैध होने पर, यह संरचना के बजाय पाशविक बल पर निर्भर करता है। trap
. एक अधिक सुंदर समाधान है ।
ट्रैप के साथ सिग्नल पकड़ना
trap
कीवर्ड सिग्नल पकड़ता है जो निष्पादन के दौरान हो सकता है। यदि आपने कभी kill
. का उपयोग किया है तो आपने इनमें से किसी एक सिग्नल का उपयोग किया है या killall
कमांड, जो SIGTERM
. कहते हैं डिफ़ॉल्ट रूप से। ऐसे कई अन्य संकेत हैं जिन पर शेल प्रतिक्रिया करते हैं, और आप उनमें से अधिकांश को trap -l
के साथ देख सकते हैं। (जैसा कि "सूची" में है):
$ trap --list
1) SIGHUP 2) SIGINT 3) SIGQUIT 4) SIGILL 5) SIGTRAP
6) SIGABRT 7) SIGBUS 8) SIGFPE 9) SIGKILL 10) SIGUSR1
11) SIGSEGV 12) SIGUSR2 13) SIGPIPE 14) SIGALRM 15) SIGTERM
16) SIGSTKFLT 17) SIGCHLD 18) SIGCONT 19) SIGSTOP 20) SIGTSTP
21) SIGTTIN 22) SIGTTOU 23) SIGURG 24) SIGXCPU 25) SIGXFSZ
26) SIGVTALRM 27) SIGPROF 28) SIGWINCH 29) SIGIO 30) SIGPWR
31) SIGSYS 34) SIGRTMIN 35) SIGRTMIN+1 36) SIGRTMIN+2 37) SIGRTMIN+3
38) SIGRTMIN+4 39) SIGRTMIN+5 40) SIGRTMIN+6 41) SIGRTMIN+7 42) SIGRTMIN+8
43) SIGRTMIN+9 44) SIGRTMIN+10 45) SIGRTMIN+11 46) SIGRTMIN+12 47) SIGRTMIN+13
48) SIGRTMIN+14 49) SIGRTMIN+15 50) SIGRTMAX-14 51) SIGRTMAX-13 52) SIGRTMAX-12
53) SIGRTMAX-11 54) SIGRTMAX-10 55) SIGRTMAX-9 56) SIGRTMAX-8 57) SIGRTMAX-7
58) SIGRTMAX-6 59) SIGRTMAX-5 60) SIGRTMAX-4 61) SIGRTMAX-3 62) SIGRTMAX-2
63) SIGRTMAX-1 64) SIGRTMAX
इनमें से किसी भी संकेत का अनुमान trap
. से लगाया जा सकता है . इनके अलावा, trap
पहचानता है:
EXIT
:तब होता है जब शेल प्रक्रिया स्वयं बाहर निकल जाती हैERR
:तब होता है जब कोई आदेश (जैसे tar .) या mkdir ) या एक अंतर्निहित कमांड (जैसे पुशड या सीडी ) गैर-शून्य स्थिति के साथ पूर्ण होता हैDEBUG
:डिबग मोड का प्रतिनिधित्व करने वाला एक बूलियन
बैश में ट्रैप सेट करने के लिए, trap
. का उपयोग करें उसके बाद उन आदेशों की एक सूची, जिन्हें आप निष्पादित करना चाहते हैं, उसके बाद इसे ट्रिगर करने के लिए संकेतों की एक सूची।
उदाहरण के लिए, यह ट्रैप एक SIGINT
. का पता लगाता है , सिग्नल तब भेजा जाता है जब कोई उपयोगकर्ता Ctrl+C . दबाता है जब कोई प्रक्रिया चल रही हो:
trap "{ echo 'Terminated with Ctrl+C'; }" SIGINT
अस्थायी निर्देशिका समस्याओं वाली उदाहरण स्क्रिप्ट को SIGINT
. का पता लगाने वाले ट्रैप के साथ ठीक किया जा सकता है , त्रुटियां, और सफल निकास:
#!/usr/bin/env bash
CWD=`pwd`
TMP=${TMP:-/tmp/tmpdir}
trap \
"{ /usr/bin/rm -r "${TMP}" ; exit 255; }" \
SIGINT SIGTERM ERR EXIT
## create tmp dir
mkdir "${TMP}"
tar xf "${1}" --directory "${TMP}"
## move to tmp and run commands
pushd "${TMP}"
for IMG in *.jpg; do
mogrify -verbose -flip -flop "${IMG}"
done
tar --create --file "${1%.*}".tar *.jpg
## move back to origin
popd
## zip tar
bzip2 --compress $TMP/"${1%.*}".tar \
--stdout > "${1%.*}".tbz
जटिल कार्यों के लिए, आप trap
को सरल बना सकते हैं बैश फ़ंक्शन के साथ कथन।
ट्रैप्स इन बैश
ट्रैप यह सुनिश्चित करने के लिए उपयोगी होते हैं कि आपकी स्क्रिप्ट का अंत साफ-सुथरा हो, चाहे वे सफलतापूर्वक चले या नहीं। स्वचालित कचरा संग्रहण पर पूरी तरह से भरोसा करना कभी भी सुरक्षित नहीं होता है, इसलिए सामान्य रूप से इसमें शामिल होना एक अच्छी आदत है। अपनी स्क्रिप्ट में उनका उपयोग करके देखें, और देखें कि वे क्या कर सकते हैं!