Лаконичность лежит в основе ОС Linux и предшествовавших ей версий Unix, и она оказала самое серьезное влияние на развитие Linux. Но иногда чрезмерная лаконичность способна довести системного администратора до сумасшествия. Типичным примером может служить формат представления системной даты в команде date, показанный ниже:
usage: date [[[[[cc]yy]mm]dd]hh]mm[.ss]
Трудно даже просто пересчитать все эти квадратные скобки, не говоря уже о том, чтобы определить, что нужно вводить, а что нет. Объясним: вы можете ввести только минуты, или минуты и секунды, или часы, минуты и секунды, или месяц, плюс все перечисленное перед этим, или вы можете добавить год и даже век. Чистое сумасшествие! Вместо утомительных попыток выяснить, что и в каком порядке вводить, попробуйте воспользоваться приведенным в листинге 6.4 сценарием, который предложит ввести соответствующие значения и затем сконструирует компактную строку с датой. Это верный способ сохранить психическое здоровье.
Листинг 6.4.Сценарий setdate
··#!/bin/bash
··# setdate — дружественный интерфейс к команде date.
··# Команда date предлагает формат ввода: [[[[[cc]yy]mm]dd]hh]mm[.ss]
··# Чтобы обеспечить максимум удобств, эта функция просит ввести конкретную
··#·· дату, показывая значение по умолчанию в квадратных скобках [], исходя
··#·· из текущей даты и времени.
··. library.sh # Source our library of bash functions to get echon().
··askvalue()
··{
····# $1 = имя поля, $2 = значение по умолчанию, $3 = максимальное значение,
····# $4 = требуемая длина в символах/цифрах
····echon "$1 [$2]: "
····read answer
····if [${answer:=$2} −gt $3]; then
······echo "$0: $1 $answer is invalid"
······exit 0
····elif ["$(($(echo $answer | wc −c) — 1))" −lt $4]; then
······echo "$0: $1 $answer is too short: please specify $4 digits"
······exit 0
····fi
····eval $1=$answer # Загрузить в заданную переменную указанное значение.
··}
··eval $(date "+nyear=%Y nmon=%m nday=%d nhr=%H nmin=%M")
··askvalue year $nyear 3000 4
··askvalue month $nmon 12 2
··askvalue day $nday 31 2
··askvalue hour $nhr 24 2
··askvalue minute $nmin 59 2
··squished="$year$month$day$hour$minute"
··# Или, если сценарий предполагается использовать в Linux:
··# squished="$month$day$hour$minute$year"
··# Да, в системах Linux и OS X/BSD используются разные форматы.
··# Так лучше?
··echo "Setting date to $squished. You might need to enter your sudo password: "
··sudo date $squished
··exit 0
Чтобы максимально уменьшить размер сценария, мы использовали функцию eval
, решив сразу две задачи. Во-первых, эта строка получает текущие дату и время, используя строку формата команды date. Во-вторых, она записывает полученные значения в переменные nyear, nmon, nday, nhr и nmin, которые затем используются простой функцией askvalue()
, запрашивающей и проверяющей введенные значения. Использование функции eval для присваивания значений переменным также решает любые потенциальные проблемы со сменой дат или другими изменениями, которые могут произойти между вызовами функции askvalue(), что нарушило бы непротиворечивость данных в сценарии. Например, если askvalue получит месяц и день в 23:59:59, а часы и минуты в 0:00:02, системная дата фактически будет установлена на сутки назад — совершенно нежелательный результат.
Нам также нужно гарантировать использование строки с датой правильного формата, потому что, например, в OS X и в Linux он различается. По умолчанию данный сценарий использует формат даты, принятый в OS X, но в строке с комментарием
приводится строка с форматом для Linux.
Вот одна из малозаметных проблем, возникающих при работе с командой date. Если в ответ на запросы сценария ввести точное время, а затем потратить несколько мгновений на ввод пароля для sudo, системное время будет на пару секунд отставать от текущего. Возможно, это совсем не проблема, но одна из причин, почему системы, подключенные к сети, должны использовать утилиты NTP (Network Time Protocol — сетевой протокол службы времени) для синхронизации с официальным сервером времени. Знакомство с механизмом синхронизации времени по сети в системах Linux и Unix можно начать с чтения страницы справочного руководства timed(8).
Читать дальше