Bash-скрипты, руководство в 11 частях — страница 8 из 19


Для того, чтобы включить в скрипте отслеживание сигналов Linux, используется команда trap. Если скрипт получает сигнал, указанный при вызове этой команды, он обрабатывает его самостоятельно, при этом оболочка такой сигнал обрабатывать не будет.


Команда trap позволяет скрипту реагировать на сигналы, в противном случае их обработка выполняется оболочкой без его участия.


Рассмотрим пример, в котором показано, как при вызове команды trap задаётся код, который надо выполнить, и список сигналов, разделённых пробелами, которые мы хотим перехватить. В данном случае это всего один сигнал:


#!/bin/bash

trap "echo ' Trapped Ctrl-C'" SIGINT

echo This is a test script

count=1

while [ $count -le 10 ]

do

echo "Loop #$count"

sleep 1

count=$(( $count + 1 ))

done

Команда trap, использованная в этом примере, выводит текстовое сообщение всякий раз, когда она обнаруживает сигнал SIGINT, который можно сгенерировать, нажав Ctrl + C на клавиатуре.


Перехват сигналов

Каждый раз, когда вы нажимаете клавиши CTRL + C, скрипт выполняет команду echo, указанную при вызове trace вместо того, чтобы позволить оболочке завершит его работу.

Перехват сигнала выхода из скрипта


Перехватить сигнал выхода из скрипта можно, использовав при вызове команды trap имя сигнала EXIT:


#!/bin/bash

trap "echo Goodbye..." EXIT

count=1

while [ $count -le 5 ]

do

echo "Loop #$count"

sleep 1

count=$(( $count + 1 ))

done

Перехват сигнала выхода из скрипта

При выходе из скрипта, будь то нормальное завершение его работы или завершение, вызванное сигналом SIGINT, сработает перехват и оболочка исполнит команду echo.

Модификация перехваченных сигналов и отмена перехвата


Для модификации перехваченных скриптом сигналов можно выполнить команду trap с новыми параметрами:


#!/bin/bash

trap "echo 'Ctrl-C is trapped.'" SIGINT

count=1

while [ $count -le 5 ]

do

echo "Loop #$count"

sleep 1

count=$(( $count + 1 ))

done

trap "echo ' I modified the trap!'" SIGINT

count=1

while [ $count -le 5 ]

do

echo "Second Loop #$count"

sleep 1

count=$(( $count + 1 ))

done

Модификация перехвата сигналов

После модификации сигналы будут обрабатываться по-новому.


Перехват сигналов можно и отменить, для этого достаточно выполнить команду trap, передав ей двойное тире и имя сигнала:


#!/bin/bash

trap "echo 'Ctrl-C is trapped.'" SIGINT

count=1

while [ $count -le 5 ]

do

echo "Loop #$count"

sleep 1

count=$(( $count + 1 ))

done

trap -- SIGINT

echo "I just removed the trap"

count=1

while [ $count -le 5 ]

do

echo "Second Loop #$count"

sleep 1

count=$(( $count + 1 ))

done

Если скрипт получит сигнал до отмены перехвата, он обработает его так, как задано в действующей команде trap.


Запустим скрипт:


$ ./myscript

И нажмём CTRL + C на клавиатуре.


Сигнал, перехваченный до отмены перехвата

Первое нажатие CTRL + C пришлось на момент исполнения скрипта, когда перехват сигнала был в силе, поэтому скрипт исполнил назначенную сигналу команду echo. После того, как исполнение дошло до команды отмены перехвата, команда CTRL + C сработала обычным образом, завершив работу скрипта.

Выполнение сценариев командной строки в фоновом режиме


Иногда bash-скриптам требуется немало времени для выполнения некоей задачи. При этом вам может понадобиться возможность нормально работать в командной строке, не дожидаясь завершения скрипта. Реализовать это не так уж и сложно.


Если вы видели список процессов, выводимый командой ps, вы могли заметить процессы, которые выполняются в фоне и не привязаны к терминалу.


Напишем такой скрипт:


#!/bin/bash

count=1

while [ $count -le 10 ]

do

sleep 1

count=$(( $count + 1 ))

done

Запустим его, указав после имени символ амперсанда (&):


$ ./myscipt &

Это приведёт к тому, что он будет запущен как фоновый процесс.


Запуск скрипта в фоновом режиме

Скрипт будет запущен в фоновом процессе, в терминал выведется его идентификатор, а когда его выполнение завершится, вы увидите сообщение об этом.


Обратите внимание на то, что хотя скрипт выполняется в фоне, он продолжает использовать терминал для вывода сообщений в STDOUT и STDERR, то есть, выводимый им текст или сообщения об ошибках можно будет увидеть в терминале.


Список процессов

При таком подходе, если выйти из терминала, скрипт, выполняющийся в фоне, так же завершит работу.

Что если нужно, чтобы скрипт продолжал работать и после закрытия терминала?

Выполнение скриптов, не завершающих работу при закрытии терминала


Скрипты можно выполнять в фоновых процессах даже после выхода из терминальной сессии. Для этого можно воспользоваться командой nohup. Эта команда позволяет запустить программу, блокируя сигналы SIGHUP, отправляемые процессу. В результате процесс будет исполняться даже при выходе из терминала, в котором он был запущен.


Применим эту методику при запуске нашего скрипта:


nohup ./myscript &

Вот что будет выведено в терминал.


Команда nohup

Команда nohup отвязывает процесс от терминала. Это означает, что процесс потеряет ссылки на STDOUT и STDERR. Для того, чтобы не потерять данные, выводимые скриптом, nohup автоматически перенаправляет сообщения, поступающие в STDOUT и в STDERR, в файл nohup.out.

Обратите внимание на то, что при запуске нескольких скриптов из одной и той же директории то, что они выводят, попадёт в один файл nohup.out.

Просмотр заданий


Команда jobs позволяет просматривать текущие задания, которые выполняются в оболочке. Напишем такой скрипт:


#!/bin/bash

count=1

while [ $count -le 10 ]

do

echo "Loop #$count"

sleep 10

count=$(( $count + 1 ))

done

Запустим его:


$ ./myscript

И временно остановим комбинацией клавиш CTRL + Z.


Запуск и приостановка скрипта

Запустим тот же скрипт в фоновом режиме, при этом перенаправим вывод скрипта в файл так, чтобы он ничего не выводил на экране:


$ ./myscript > outfile &

Выполнив теперь команду jobs, мы увидим сведения как о приостановленном скрипте, так и о том, который работает в фоне.


Получение сведений о скриптах

Ключ -l при вызове команды jobs указывает на то, что нам нужны сведения об ID процессов.

Перезапуск приостановленных заданий


Для того, чтобы перезапустить скрипт в фоновом режиме, можно воспользоваться командой bg.

Запустим скрипт:


$ ./myscript

Нажмём CTRL + Z, что временно остановит его выполнение. Выполним следующую команду:


$ bg

Команда bg

Теперь скрипт выполняется в фоновом режиме.


Если у вас имеется несколько приостановленных заданий, для перезапуска конкретного задания команде bg можно передать его номер.


Для перезапуска задания в обычном режиме воспользуйтесь командой fg:


$ fg 1

Планирование запуска скриптов


Linux предоставляет пару способов запуска bash-скриптов в заданное время. Это команда at и планировщик заданий cron.


Вызов команды at выглядит так:


at [-f filename] time

Эта команда распознаёт множество форматов указания времени.


   • Стандартный, с указанием часов и минут, например — 10:15.

   • С использованием индикаторов AM/PM, до или после полудня, например — 10:15PM.

   • С использованием специальных имён, таких, как now, noon, midnight.


В дополнение к возможности указания времени запуска задания, команде at можно передать и дату, используя один из поддерживаемых ей форматов.


   • Стандартный формат указания даты, при котором дата записывается по шаблонам MMDDYY, MM/DD/YY, или DD.MM.YY.

   • Текстовое представление даты, например, Jul 4 или Dec 25, при этом год можно указать, а можно обойтись и без него.

   • Запись вида now + 25 minutes.

   • Запись вида 10:15PM tomorrow.

   • Запись вида 10:15 + 7 days.


Не будем углубляться в эту тему, рассмотрим простой вариант использования команды:


$ at -f ./myscript now

Планирование заданий с использованием команды at

Ключ -M при вызове at используется для отправки того, что выведет скрипт, по электронной почте, если система соответствующим образом настроена. Если отправка электронного письма невозможна, этот ключ просто подавит вывод.


Для того чтобы посмотреть список заданий, ожидающих выполнения, можно воспользоваться командой atq:


$ atq

Список заданий, ожидающих выполнения

Удаление заданий, ожидающих выполнения


Удалить задание, ожидающее выполнения, позволяет команда atrm. При её вызове указывают номер задания:


$ atrm 18

Удаление задания

Запуск скриптов по расписанию


Планирование однократного запуска скриптов с использованием команды at способно облегчить жизнь во многих ситуациях. Но как быть, если нужно, чтобы скрипт выполнялся в одно и то же время ежедневно, или раз в неделю, или раз в месяц?


В Linux имеется утилита crontab, позволяющая планировать запуск скриптов, которые нужно выполнять регулярно.