শেল স্ক্রিপ্টে সময় নির্ধারণ করা


53

আমার কাছে একটি শেল স্ক্রিপ্ট রয়েছে যা স্ট্যান্ডার্ড ইনপুট থেকে পড়ে । বিরল পরিস্থিতিতে, ইনপুট সরবরাহের জন্য কেউ প্রস্তুত থাকবে না এবং স্ক্রিপ্টটির সময় শেষ হতে হবে । সময়সীমা শেষ হওয়ার ক্ষেত্রে স্ক্রিপ্টটিতে অবশ্যই কিছু সাফাই কোড চালানো উচিত। এটি করার সর্বোত্তম উপায় কী?

এই স্ক্রিপ্টটি অবশ্যই খুব পোর্টেবল হতে হবে , সহ বিংশ শতাব্দীর ইউনিক্স সিস্টেমের সাথে সি কম্পাইলার ছাড়া এবং এম্বেড থাকা ডিভাইসগুলিতে ব্যস্তবক্স চলমান রয়েছে, সুতরাং পার্ল, ব্যাশ, কোনও সংকলিত ভাষা এবং এমনকি সম্পূর্ণ পসিক্স ২-এ নির্ভর করা যায় না। বিশেষ করে, $PPID, read -tএবং পুরোপুরি POSIX-অনুবর্তী যাত্রীর সঙ্গের নিজলটবহর পাওয়া যায় না। একটি অস্থায়ী ফাইল লিখতেও বাদ দেওয়া হয়; স্ক্রিপ্টটি চলতে পারে এমনকি যদি সমস্ত ফাইল-সিস্টেমগুলি কেবল পঠনযোগ্যভাবে মাউন্ট হয়।

কেবল জিনিসগুলিকে আরও জটিল করে তুলতে, আমিও চাই স্ক্রিপ্টটি সময় শেষ না হলে যুক্তিসঙ্গত দ্রুত হয় be বিশেষত, আমি উইন্ডোজে স্ক্রিপ্টটিও ব্যবহার করি (মূলত সাইগউইনে), যেখানে কাঁটাচামচ এবং এক্সিকিউটিভ কম, তাই আমি তাদের ব্যবহার সর্বনিম্ন রাখতে চাই।

সংক্ষেপে, আমার আছে

trap cleanup 1 2 3 15
foo=`cat`

এবং আমি একটি সময়সীমা যুক্ত করতে চাই আমি অন্তর্নির্মিত catসঙ্গে প্রতিস্থাপন করতে পারবেন না read। সময় শেষ হওয়ার ক্ষেত্রে, আমি cleanupফাংশনটি সম্পাদন করতে চাই ।


পটভূমি: এই স্ক্রিপ্টটি কিছু 8-বিট অক্ষর মুদ্রণ করে এবং কার্সারের অবস্থানের আগে এবং পরে তুলনা করে টার্মিনালের এনকোডিং অনুমান করছে। স্ক্রিপ্ট পরীক্ষার সূচনা যে স্টাডাউট একটি সমর্থিত টার্মিনালের সাথে সংযুক্ত, তবে কখনও কখনও পরিবেশটি পড়ে থাকে (উদাহরণস্বরূপ plinkসেট করে TERM=xtermএমনকি সেট করা হয়TERM=dumb )। স্ক্রিপ্টের প্রাসঙ্গিক অংশটি এরকম দেখাচ্ছে:

text='Éé'  # UTF-8; shows up as Ãé on a latin1 terminal
csi='␛['; dsr_cpr="${csi}6n"; dsr_ok="${csi}5n"  # ␛ is an escape character
stty_save=`stty -g`
cleanup () { stty "$stty_save"; }
trap 'cleanup; exit 120' 0 1 2 3 15     # cleanup code
stty eol 0 eof n -echo                # Input will end with `0n`
# echo-n is a function that outputs its argument without a newline
echo-n "$dsr_cpr$dsr_ok"              # Ask the terminal to report the cursor position
initial_report=`tr -dc \;0123456789`  # Expect ␛[42;10R␛[0n for y=42,x=10
echo-n "$text$dsr_cpr$dsr_ok"
final_report=`tr -dc \;0123456789`
cleanup
# Compute and return initial_x - final_x

আমি কীভাবে স্ক্রিপ্টটি সংশোধন করতে পারি যাতে tr2 সেকেন্ডের পরে কোনও ইনপুট না পড়লে এটি মারা যায় এবং স্ক্রিপ্টটি cleanupকার্য সম্পাদন করে ?


উত্তর:


33

এই সম্পর্কে কি:

foo=`{ { cat 1>&3; kill 0; } | { sleep 2; kill 0; } } 3>&1`

এটি: আউটপুট-উত্পাদনকারী কমান্ড চালান এবং sleepএকই প্রক্রিয়া গ্রুপে, কেবল তাদের জন্য একটি প্রক্রিয়া গ্রুপ। যে কোনও কমান্ড প্রথমে ফিরে আসে পুরো প্রক্রিয়া গোষ্ঠীকে হত্যা করে।

যে কেউ আশ্চর্য হবে: হ্যাঁ, পাইপ ব্যবহার করা হয় না; এটি পুনর্নির্দেশগুলি ব্যবহার করে বাইপাস করা হয়েছে। এর একমাত্র উদ্দেশ্য শেলটি একই প্রক্রিয়া গ্রুপে দুটি প্রক্রিয়া চালানো।


গিলস তাঁর মন্তব্যে যেমন উল্লেখ করেছেন, এটি শেল স্ক্রিপ্টে কাজ করবে না কারণ স্ক্রিপ্ট প্রক্রিয়াটি দুটি উপ-প্রসেসের সাথে হত্যা করা হবে।

একটি উপায় a একটি পৃথক প্রক্রিয়া গ্রুপে কমান্ডটি চালিত করার জন্য একটি নতুন ইন্টারেক্টিভ শেল শুরু করা:

#!/bin/sh
foo=`sh -ic '{ cat 1>&3; kill 0; } | { sleep 2; kill 0; }' 3>&1 2>/dev/null`
[ -n "$foo" ] && echo got: "$foo" || echo timeouted

তবে এটি সহ সতর্কতা থাকতে পারে (উদাহরণস্বরূপ, যখন স্ট্ডিন কোনও টিটি নয়)? ইন্টারেক্টিভ শেলটি মারা গেলে "টার্মিনেটেড" বার্তাটি থেকে মুক্তি পাওয়ার জন্য স্ট্যাডার পুনঃনির্দেশটি রয়েছে।

পরীক্ষিত zsh, bashএবং dash। তবে বৃদ্ধদের কী হবে?

বি 9 8 ম্যাক ওএস এক্স, জিএনইউ ব্যাশ 3.2.57 বা ড্যাশ সহ লিনাক্সের সাথে কাজ করে নিম্নলিখিত পরিবর্তনের পরামর্শ দেয়:

foo=`sh -ic 'exec 3>&1 2>/dev/null; { cat 1>&3; kill 0; } | { sleep 2; kill 0; }'`

-
1. অন্যটি মানহীন setsidবলে মনে হয়।


1
আমি এটি বেশ পছন্দ করি, আমি কোনও প্রক্রিয়া গোষ্ঠীটি সেভাবে ব্যবহার করার কথা ভাবিনি। এটি বেশ সহজ এবং কোনও রেসের শর্ত নেই। আমার আরও কিছু বহনযোগ্যতার জন্য এটি পরীক্ষা করা দরকার তবে এটি বিজয়ীর মতো দেখাচ্ছে।
গিলস 'অশুভ হওয়া বন্ধ করুন'

দুর্ভাগ্যক্রমে, আমি এটি স্ক্রিপ্টে রাখার সাথে সাথে এটি দর্শনীয়ভাবে ব্যর্থ হয়: কমান্ড প্রতিস্থাপনের পাইপলাইনটি তার নিজস্ব প্রক্রিয়া গোষ্ঠীতে চালিত হয় না এবং kill 0স্ক্রিপ্টের আহ্বানকারীকেও হত্যা করে। পাইপলাইনটিকে তার নিজস্ব প্রক্রিয়া গোষ্ঠীতে জোর করার কোনও পোর্টেবল উপায় আছে কি?
গিলস 'অশুভ হওয়া বন্ধ করুন'

@ গিলস: এেক! আপাতত উপায় setprgp()ছাড়া উপায় খুঁজে পাচ্ছিল না setsid:-(
স্টাফেনি গিমেনেজ

1
আমি কৌশলটি সত্যিই পছন্দ করি, তাই আমি অনুগ্রহ প্রদান করছি। এটি লিনাক্সে কাজ করছে বলে মনে হচ্ছে, অন্য সিস্টেমে এটি পরীক্ষা করার মতো সময় আমার হাতে নেই।
গিলস 'অশুভ হওয়া বন্ধ করুন'

2
@ জ্যাক: মূল ক্ষেত্রে ক্রমটি ফিরিয়ে দেওয়া সম্ভব নয়, কারণ শুধুমাত্র প্রথম প্রক্রিয়া স্টিডিনের অ্যাক্সেস পেতে পারে।
স্টাফেন গিমেনেজ

6
me=$$
(sleep 2; kill $me >/dev/null 2>&1) & nuker=$!
# do whatever
kill $nuker >/dev/null 2>&1

আপনি ইতিমধ্যে 15 টি আটকে রয়েছেন (এর সংখ্যার সংস্করণ SIGTERM, যা যা killঅন্যথায় না বলা হয় তা প্রেরণ করে), সুতরাং আপনার ইতিমধ্যে ভাল হওয়া উচিত। এটি বলেছে, আপনি যদি প্রাক-পসিক্সের দিকে তাকিয়ে থাকেন তবে সচেতন হন যে শেল ফাংশনগুলি নাও থাকতে পারে (সেগুলি সিস্টেম ভি এর শেল থেকে এসেছে)।


এটা অত সস্তা না. আপনি যদি সিগন্যালটি আটকা পড়ে থাকেন তবে অনেকগুলি শেল (ড্যাশ, বাশ, পিডিএক্স, জেডএস ... সম্ভবত এটিটি কেএসএস বাদে সবগুলি) এড়িয়ে catযাওয়ার জন্য অপেক্ষা করার সময় এটিকে এড়িয়ে চলে ignore আমি ইতিমধ্যে সামান্য পরীক্ষা-নিরীক্ষা করেছি তবে এখনও পর্যন্ত আমি সন্তুষ্ট এমন কিছু পাইনি।
গিলস 'দুষ্ট হওয়া বন্ধ করুন'

বহনযোগ্যতার বিষয়ে: শুকরিয়া আমার সর্বত্র ফাংশন রয়েছে। আমি নিশ্চিত নই $!, আমি মনে করি আমি যেসব মেশিন ব্যবহার করি খুব কমই চাকরি নিয়ন্ত্রণ থাকে না, $!সর্বজনীন উপলব্ধ?
গিলস 23'13

@ গিলস: এইচএম, ইয়ে আমি কিছু সত্যই ঘৃণ্য এবং-নির্দিষ্ট bashজিনিস মনে করি যা আমি একবার অপব্যবহারের সাথে জড়িত ছিল -o monitor। আমি সত্যিই প্রাচীন শাঁসের বিবেচনায় ছিলাম যখন আমি এটি লিখেছিলাম (এটি v7 তে কাজ করেছিল)। এটি বলেছিল, আমি মনে করি আপনি দুটি অন্য যে কোনও একটি করতে পারেন: (১) ব্যাকগ্রাউন্ড "যাই হোক না কেন" এবং wait $!, বা (২) এছাড়াও প্রেরণ SIGCLD/ SIGCHLD... তবে পুরানো পর্যাপ্ত মেশিনগুলিতে পরবর্তীটি হয় অস্তিত্বহীন বা অ-বহনযোগ্য (পূর্ববর্তীটি সিস্টেম সিস্টেম) III / V, পরবর্তী বিএসডি, এবং ভি 7 এর কোনওটি নেই)।
গীকোসৌর

@Gilles: $!অন্তত V7 ফিরে যায়, এবং অবশ্যই একটি চেয়েও পুরনো sh-একটি যে আসলে কাজ নিয়ন্ত্রণ সম্পর্কে কিছু (জানতাম, একটি দীর্ঘ সময়ের জন্য /bin/shবাসদ উপর কাজ নিয়ন্ত্রণ করেন নি; আপনি ছিল চালানোর জন্য cshএটি পেতে - কিন্তু $!সেখানে ছিল )।
গাইকোসৌর

আমি "যাই হোক না কেন" ব্যাকগ্রাউন্ড করতে পারি না, এর আউটপুট আমার দরকার। সম্পর্কে তথ্যের জন্য ধন্যবাদ $!
গিলস 'দুষ্ট হওয়া বন্ধ করুন'

4

যদিও সংস্করণ .0.০ অনুসারে কোরটুইলে একটি সময়সীমা কমান্ড অন্তর্ভুক্ত রয়েছে যা আপনি কিছু পরিবেশের উল্লেখ করেছেন যা এটিতে থাকবে না। ভাগ্যক্রমে পিক্সেলবিট.অর্গ.এর একটি টাইমআউট স্ক্রিপ্ট লেখা আছে sh

আমি এটি আগে বেশ কয়েকবার ব্যবহার করেছি এবং এটি খুব ভালভাবে কাজ করে।

http://www.pixelbeat.org/scriptts/timeout ( দ্রষ্টব্য: নীচের স্ক্রিপ্টটি পিক্সেলবিট.অর্গ.এর একটি থেকে সামান্য পরিবর্তন করা হয়েছে, এই উত্তরের নীচের মন্তব্যগুলি দেখুন))

#!/bin/sh

# Execute a command with a timeout

# Author:
#    http://www.pixelbeat.org/
# Notes:
#    Note there is a timeout command packaged with coreutils since v7.0
#    If the timeout occurs the exit status is 124.
#    There is an asynchronous (and buggy) equivalent of this
#    script packaged with bash (under /usr/share/doc/ in my distro),
#    which I only noticed after writing this.
#    I noticed later again that there is a C equivalent of this packaged
#    with satan by Wietse Venema, and copied to forensics by Dan Farmer.
# Changes:
#    V1.0, Nov  3 2006, Initial release
#    V1.1, Nov 20 2007, Brad Greenlee <brad@footle.org>
#                       Make more portable by using the 'CHLD'
#                       signal spec rather than 17.
#    V1.3, Oct 29 2009, Ján Sáreník <jasan@x31.com>
#                       Even though this runs under dash,ksh etc.
#                       it doesn't actually timeout. So enforce bash for now.
#                       Also change exit on timeout from 128 to 124
#                       to match coreutils.
#    V2.0, Oct 30 2009, Ján Sáreník <jasan@x31.com>
#                       Rewritten to cover compatibility with other
#                       Bourne shell implementations (pdksh, dash)

if [ "$#" -lt "2" ]; then
    echo "Usage:   `basename $0` timeout_in_seconds command" >&2
    echo "Example: `basename $0` 2 sleep 3 || echo timeout" >&2
    exit 1
fi

cleanup()
{
    trap - ALRM               #reset handler to default
    kill -ALRM $a 2>/dev/null #stop timer subshell if running
    kill $! 2>/dev/null &&    #kill last job
      exit 124                #exit with 124 if it was running
}

watchit()
{
    trap "cleanup" ALRM
    sleep $1& wait
    kill -ALRM $$
}

watchit $1& a=$!         #start the timeout
shift                    #first param was timeout for sleep
trap "cleanup" ALRM INT  #cleanup after timeout
"$@" < /dev/tty & wait $!; RET=$?    #start the job wait for it and save its return value
kill -ALRM $a            #send ALRM signal to watchit
wait $a                  #wait for watchit to finish cleanup
exit $RET                #return the value

দেখে মনে হচ্ছে এটি কমান্ডের আউটপুট পুনরুদ্ধার করতে দেয় না। অন্যান্য উত্তরে গিলসের মন্তব্য "আমি যাই হোক না কেন" ব্যাকগ্রাউন্ড করতে পারি না।
স্টাফেন গিমেনেজ

আহ, আকর্ষণীয়। আমি স্ক্রিপ্টটি পরিবর্তন করেছি (যাতে এটি পিক্সেলবিটের সাথে আর মেলে না) কমান্ডটিতে / dev / stdin পুনর্নির্দেশ করতে। এটি আমার পরীক্ষায় কাজ করে বলে মনে হচ্ছে।
বাহামাত

বাশ (অদ্ভুতভাবে) বাদে স্ট্যান্ডার্ড ইনপুট থেকে কমান্ডটি পড়ার জন্য এটি কাজ করে না। </dev/stdinএকটি অপ- অপ হয়। </dev/ttyএটি টার্মিনাল থেকে পড়তে সক্ষম করবে যা আমার ব্যবহারের ক্ষেত্রে যথেষ্ট for
গিলস 'অশুভ হওয়া বন্ধ করুন'

@ গাইলস: দুর্দান্ত, আমি আপডেটটি করব।
বাহামাত

এটি যথেষ্ট বেশি প্রচেষ্টা ছাড়াই কাজ করতে পারে না: কমান্ডের আউটপুট পুনরুদ্ধার করতে হবে এবং কমান্ডটি পটভূমিতে থাকলে আমি এটি করতে পারি না।
গিলস 'অশুভ হওয়া বন্ধ করুন'

3

(আব) এর জন্য কী এনসি ব্যবহার করবেন

ভালো লেগেছে;

   $ nc -l 0 2345 | cat &  # output come from here
   $ nc -w 5 0 2345   # input come from here and times out after 5 sec

অথবা একটি একক কমান্ড লাইনে পরিণত হয়েছে;

   $ foo=`nc -l 0 2222 | nc -w 5 0 2222`

সর্বশেষ সংস্করণ, যদিও লুকোচুরি দেখতে আসলে কাজ করে যখন আমি এটি একটি লিনাক্স সিস্টেমে পরীক্ষা করি - আমার সর্বোত্তম অনুমান যে এটি কোনও সিস্টেমে কাজ করা উচিত, এবং যদি না হয় তবে আউটপুট পুনর্নির্দেশের কোনও প্রকরণ বহনযোগ্যতার জন্য সমাধান করতে পারে। এখানে সুবিধা হ'ল কোনও ব্যাকগ্রাউন্ড প্রক্রিয়া জড়িত নয়।


পর্যাপ্ত পোর্টেবল নয়। আমার ncকোনও পুরানো ইউনিক্স বক্সন নেই, না অনেক এমবেডেড লিনাক্স নেই।
গিলস 'খারাপ হয়ে যাওয়া বন্ধ করুন'

0

নিজস্ব প্রক্রিয়া গোষ্ঠীতে পাইপলাইন চালানোর আর একটি উপায় হ'ল কমান্ডটি sh -c '....'ব্যবহার করে সিউডো-টার্মিনালে চালানো script(যা স্পষ্টভাবে setsidফাংশনটি প্রয়োগ করে )।

#!/bin/sh
stty -echo -onlcr
# GNU script
foo=`script -q -c 'sh -c "{ cat 1>&3; kill 0; } | { sleep 5; kill 0; }" 3>&1 2>/dev/null' /dev/null`
# FreeBSD script
#foo=`script -q /dev/null sh -c '{ cat 1>&3; kill 0; } | { sleep 5; kill 0; }' 3>&1 2>/dev/null`
stty echo onlcr
echo "foo: $foo"


# alternative without: stty -echo -onlcr
# cr=`printf '\r'`
# foo=`script -q -c 'sh -c "{ { cat 1>&3; kill 0; } | { sleep 5; kill 0; } } 3>&1 2>/dev/null"' /dev/null | sed -e "s/${cr}$//" -ne 'p;N'`  # GNU
# foo=`script -q /dev/null sh -c '{ { cat 1>&3; kill 0; } | { sleep 5; kill 0; } } 3>&1 2>/dev/null' | sed -e "s/${cr}$//" -ne 'p;N'`  # FreeBSD
# echo "foo: $foo"

পর্যাপ্ত পোর্টেবল নয়। আমার scriptকোনও পুরানো ইউনিক্স বক্সন নেই, না অনেক এমবেডেড লিনাক্স নেই।
গিলস 17'50

0

Https://unix.stackexchange.com/a/18711 এ উত্তরটি খুব সুন্দর।

আমি একটি অনুরূপ ফলাফল অর্জন করতে চেয়েছিলাম, তবে স্পষ্টভাবে শেলটি পুনরায় না চাওয়া ছাড়া আমি বিদ্যমান শেল ফাংশনগুলি শুরু করতে চেয়েছিলাম।

ব্যাশ ব্যবহার করে নিম্নলিখিত কাজগুলি করা সম্ভব:

eval 'set -m ; ( ... ) ; '"$(set +o)"

সুতরাং ধরুন আমি ইতিমধ্যে একটি শেল ফাংশন আছে f:

f() { date ; kill 0 ; }

echo Before
eval 'set -m ; ( f ) ; '"$(set +o)"
echo After

এটি সম্পাদন করে দেখছি:

$ sh /tmp/foo.sh
Before
Mon 14 Mar 2016 17:22:41 PDT
/tmp/foo.sh: line 4: 17763 Terminated: 15          ( f )
After

-2
timeout_handler () { echo Timeout. goodbye.;  exit 1; }
trap timeout_handler ALRM
( sleep 60; kill -ALRM $$ ) &

এটি গিকাউসরের উত্তরের মতো একই কারণে কাজ করে না । আমার একটি বাহ্যিক কমান্ডের আউটপুট পেতে হবে এবং এই কমান্ডটি চলাকালীন সিগন্যালটিকে উপেক্ষা করা হবে।
গিলস 'অশুভ হওয়া বন্ধ করুন'
আমাদের সাইট ব্যবহার করে, আপনি স্বীকার করেছেন যে আপনি আমাদের কুকি নীতি এবং গোপনীয়তা নীতিটি পড়েছেন এবং বুঝতে পেরেছেন ।
Licensed under cc by-sa 3.0 with attribution required.