Как создать работу cron с помощью Bash автоматически без интерактивного редактора?

326

Есть ли у crontab аргумент для создания заданий cron без использования редактора (crontab -e). Если да, то каким будет код, создающий cronjob из Bash script?

  • 0
    Возможный дубликат Как я могу программно создать новую работу cron?
  • 1
    К сожалению, большинство топовых ответов здесь просто показывают, как модифицировать crontab - хотя и достаточно безопасными способами - но я думаю, что в целом это неправильный подход. Лучше, безопаснее и проще поместить файл в {{cron.d}}, и есть (в настоящее время) ответы с низким голосованием, объясняющие, как это сделать, если вы посмотрите вниз.
Теги:
cron

18 ответов

426
Лучший ответ

Вы можете добавить к crontab следующим образом:

#write out current crontab
crontab -l > mycron
#echo new cron into cron file
echo "00 09 * * 1-5 echo hello" >> mycron
#install new cron file
crontab mycron
rm mycron

Объяснение линии Cron

* * * * * "command to be executed"
- - - - -
| | | | |
| | | | ----- Day of week (0 - 7) (Sunday=0 or 7)
| | | ------- Month (1 - 12)
| | --------- Day of month (1 - 31)
| ----------- Hour (0 - 23)
------------- Minute (0 - 59)

Источник nixCraft.

  • 14
    Вы должны использовать tempfile или mktemp
  • 0
    @ Dogbane У меня есть сомнения. Почему здесь используется «rm mycron» ??
Показать ещё 12 комментариев
257

Вы можете сделать это на лету

crontab -l | { cat; echo "0 0 0 0 0 some entry"; } | crontab -

crontab -l перечисляет текущие задания crontab, cat печатает его, echo печатает новую команду, а crontab - добавляет все напечатанные материалы в файл crontab. Вы можете увидеть эффект, выполнив новый crontab -l.

  • 3
    спасибо, не могли бы вы объяснить синтаксис или указать URL, который его объясняет?
  • 8
    crontab -l выводит список текущих заданий crontab, cat печатает его, echo печатает новую команду, а crontab - добавляет все напечатанные материалы в файл crontab. Вы можете увидеть эффект, выполнив новый crontab -l
Показать ещё 5 комментариев
58

Этот более короткий не требует временного файла, он невосприимчив к нескольким вставкам и позволяет изменять расписание существующей записи.

Скажите, что у вас есть такие возможности:

croncmd="/home/me/myfunction myargs > /home/me/myfunction.log 2>&1"
cronjob="0 */15 * * * $croncmd"

Чтобы добавить его в crontab без дублирования:

( crontab -l | grep -v -F "$croncmd" ; echo "$cronjob" ) | crontab -

Чтобы удалить его из crontab независимо от его текущего расписания:

( crontab -l | grep -v -F "$croncmd" ) | crontab -

Примечания:

  • grep -F буквально соответствует строке, так как мы не хотим интерпретировать ее как регулярное выражение
  • Мы также игнорируем планирование времени и только ищем команду. Сюда; расписание может быть изменено без риска добавления новой строки в crontab
  • 0
    это не работает
  • 0
    @ Moeseth: вы можете уточнить?
Показать ещё 9 комментариев
36

Спасибо всем за вашу помощь. Объединив то, что я нашел здесь и в другом месте, я придумал следующее:

Код

command="php $INSTALL/indefero/scripts/gitcron.php"
job="0 0 * * 0 $command"
cat <(fgrep -i -v "$command" <(crontab -l)) <(echo "$job") | crontab -

Я не мог понять, как устранить необходимость в двух переменных, не повторяя себя.

command - это, очевидно, команда, которую я хочу запланировать. job принимает $command и добавляет данные планирования. Мне нужны обе переменные отдельно в строке кода, которая выполняет эту работу.

Подробнее

  • В ответ на duckyflip я использую эту небольшую перенаправляемую вещь (<(*command*)), чтобы вывести вывод crontab -l на вход для команды fgrep.
  • fgrep затем отфильтровывает любые совпадения параметра $command (-v), нечувствительного к регистру (-i).
  • Опять же, небольшое перенаправление thingy (<(*command*)) используется для возврата результата на вход для команды cat.
  • Команда cat также получает echo "$job" (самостоятельно объяснительную), опять же, используя команду перенаправления (<(*command*)).
  • Таким образом, отфильтрованный вывод из crontab -l и простой echo "$job", объединенный, передаются по каналам ('|') на crontab -, чтобы, наконец, быть записаны.
  • И все они жили долго и счастливо!

В двух словах:

Эта строка кода отфильтровывает любые задания cron, которые соответствуют команде, а затем записывает оставшиеся задания cron с новым, эффективно действуя как функция "добавить" или "обновить". Чтобы использовать это, все, что вам нужно сделать, - это изменить значения переменных command и job.

  • 4
    Для удобства других читателей преимущество этого подхода состоит в том, что вы можете запускать его несколько раз, не беспокоясь о дублировании записей в crontab (в отличие от всех других решений). Это из-за fgrep -v
  • 4
    Если вы предпочитаете традиционный способ передачи данных слева направо, замените последнюю строку на: crontab -l | fgrep -i -v "$command" | { cat; echo "$job"; } | crontab -l
22

EDIT (фиксированная перезапись):

cat <(crontab -l) <(echo "1 2 3 4 5 scripty.sh") | crontab -
  • 0
    Это заменит любое содержимое crontab
  • 1
    @TheBonsai О, я вижу, я исправил это сейчас, так что он должен добавить новую команду к существующему содержимому crontab
Показать ещё 1 комментарий
9

Было много хороших ответов относительно использования crontab, но не упоминалось о более простом методе, таком как использование cron.

Использование cron позволит использовать системные файлы и каталоги, расположенные в /etc/crontab, /etc/cron.daily,weekly,hourly или /etc/cron.d/:

cat > /etc/cron.d/<job> << EOF
SHELL=/bin/bash 
PATH=/sbin:/bin:/usr/sbin:/usr/bin 
MAILTO=root HOME=/  
01 * * * * <user> <command>
EOF

В этом примере выше мы создали файл в /etc/cron.d/, предоставили переменные среды для успешного выполнения команды, а также предоставили user для команды и саму command. Этот файл не должен быть исполняемым, а имя должно содержать только буквенно-цифровые символы и дефисы (подробнее см. Ниже).

Чтобы дать исчерпывающий ответ, давайте посмотрим на различия между crontab и cron/crond:

crontab -- maintain tables for driving cron for individual users

Для тех, кто хочет запустить задание в контексте своего пользователя в системе, использование crontab может иметь смысл.

cron -- daemon to execute scheduled commands

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

Небольшая выдержка из справочных страниц дает вам несколько примеров того, что нужно делать, а чего нет:

/etc/crontab и файлы в /etc/cron.d должны принадлежать пользователю root и не должны быть group- или доступны для записи другим способом. В отличие от области спула, файлы в /etc/cron.d или файлы в /etc/cron.hourly,/etc/cron.daily,/etc/cron.weekly и /etc/cron.monthly также могут быть символические ссылки, при условии, что как символическая ссылка, так и файл, на который она указывает, принадлежат пользователю root. Файлы в /etc/cron.d не должны быть исполняемыми, в то время как файлы в /etc/cron.hourly,/etc/cron.daily,/etc/cron.weekly и /etc/cron.monthly делают, как они запускаются частями выполнения (см. run-parts (8) для получения дополнительной информации).

Источник: http://manpages.ubuntu.com/manpages/trusty/man8/cron.8.html

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

7

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

__cron="1 2 3 4 5 /root/bin/backup.sh"
cat <(crontab -l) |grep -v "${__cron}" <(echo "${__cron}")

Это работает только если вы используете BASH. Я не знаю правильный синтаксис DASH (sh).

Обновление: это не работает, если у пользователя еще нет crontab. Более надежный способ будет:

(crontab -l ; echo "1 2 3 4 5 /root/bin/backup.sh") | sort - | uniq - | crontab - 

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

echo "1 2 3 4 5 /root/bin/backup.sh" |sudo tee /etc/crond.d/backup

Нашел те в другом вопросе ТАК.

  • 1
    Я получаю syntax error near unexpected token `( Есть мысли?
  • 0
    Использование sh вместо bash может быть?
Показать ещё 5 комментариев
5

Вариант, который редактирует только crontab, если искомая строка не найдена:

CMD="/sbin/modprobe fcpci"
JOB="@reboot $CMD"
TMPC="mycron"
grep "$CMD" -q <(crontab -l) || (crontab -l>"$TMPC"; echo "$JOB">>"$TMPC"; crontab "$TMPC")
4

Для приятного быстрого и грязного создания/замены crontab с помощью BASH script я использовал эту нотацию:

crontab <<EOF
00 09 * * 1-5 echo hello
EOF
3

Если вы используете Vixie Cron, например. в большинстве дистрибутивов Linux вы можете просто поместить файл в /etc/cron.d с помощью отдельного cronjob.

Это работает только для root. Если ваша система поддерживает это, вы должны увидеть несколько примеров. (Обратите внимание на имя пользователя, включенное в строку, в том же синтаксисе, что и старый /etc/crontab )

Это печальная ошибка в cron, что нет способа справиться с этим как обычный пользователь, и что так много реализаций cron никак не могут справиться с этим.

1

Bash скрипт для добавления задания cron без интерактивного редактора. Ниже код помогает добавить cronjob, используя файлы linux.

#!/bin/bash

cron_path=/var/spool/cron/crontabs/root

#cron job to run every 10 min.
echo "*/10 * * * * command to be executed" >> $cron_path

#cron job to run every 1 hour.
echo "0 */1 * * * command to be executed" >> $cron_path
  • 0
    Я знаю, что это было долгое время, но это все еще единственный хороший ответ, так как все наиболее проголосовавшие удаляют старые кроны вместо того, чтобы просто добавлять новый.
1
CRON="1 2 3 4 5 /root/bin/backup.sh" 
cat < (crontab -l) |grep -v "${CRON}" < (echo "${CRON}")

добавить параметр -w в команду grep точно, без параметра -w добавление "тестирования" cronjob приводит к удалению задания cron "testing123"

script функция добавления/удаления cronjobs. нет записей дублирования:

cronjob_editor () {         
# usage: cronjob_editor '<interval>' '<command>' <add|remove>

if [[ -z "$1" ]] ;then printf " no interval specified\n" ;fi
if [[ -z "$2" ]] ;then printf " no command specified\n" ;fi
if [[ -z "$3" ]] ;then printf " no action specified\n" ;fi

if [[ "$3" == add ]] ;then
    # add cronjob, no duplication:
    ( crontab -l | grep -v -F -w "$2" ; echo "$1 $2" ) | crontab -
elif [[ "$3" == remove ]] ;then
    # remove cronjob:
    ( crontab -l | grep -v -F -w "$2" ) | crontab -
fi 
} 
cronjob_editor "$1" "$2" "$3"

:

$ ./cronjob_editor.sh '*/10 * * * *' 'echo "this is a test" > export_file' add
$ crontab  -l
$ */10 * * * * echo "this is a test" > export_file
  • 0
    Если в crontab одна и та же команда дважды (выполняется в разное время), команда delete удалит обе строки.
1

Вот функция bash для добавления команды в crontab без дублирования

function addtocrontab () {
  local frequency=$1
  local command=$2
  local job="$frequency $command"
  cat <(fgrep -i -v "$command" <(crontab -l)) <(echo "$job") | crontab -
}
addtocrontab "0 0 1 * *" "echo hello"
0

Итак, в Debian, Ubuntu и многих подобных дистрибутивах на основе Debian...

Существует механизм конкатенации задач cron, который берет файл конфигурации, объединяет их и добавляет к работающей службе cron.

Вы можете поместить файл в /etc/cron.d/somefilename, где somefilename - это то, что вы хотите.

sudo echo "0,15,30,45 * * * * ntpdate -u time.nist.gov" >> /etc/cron.d/vmclocksync

Давай разберем это:

sudo - потому что вам нужны повышенные права для изменения настроек cron в каталоге /etc

echo - средство для создания вывода на стандартный вывод. printf, кот... будет работать так же

"- используйте двойную кавычку в начале строки, вы профессионал

0,15,30,45 * * * * - стандартное расписание запуска cron, оно запускается каждые 15 минут

ntpdate -u time.nist.gov - фактическая команда, которую я хочу выполнить

"- потому что мои первые двойные кавычки нуждаются в приятеле, чтобы закрыть выводимую строку

>> - двойное перенаправление добавляется вместо перезаписи *

/etc/cron.d/vmclocksync - vmclocksync - это выбранное мной имя файла, оно находится в /etc/cron.d/


* если бы мы использовали перенаправление>, мы могли бы гарантировать, что у нас была только одна запись задачи. Но мы рискуем уничтожить любые другие правила в существующем файле. Вы можете решить для себя, является ли возможным уничтожение с> правильным или возможные дубликаты с >> для вас. В качестве альтернативы, вы можете сделать что-то замысловатое или задействованное, чтобы проверить, существует ли имя файла, есть ли в нем что-нибудь и добавляете ли вы какой-либо duplicate--, но у меня есть дела, которые я не могу сделать для ты прямо сейчас.

0

Мое предпочтительное решение этого было бы это:

(crontab -l | grep . ; echo -e "0 4 * * * myscript\n") | crontab -

Это обеспечит правильную обработку новой пустой строки внизу. Чтобы избежать проблем с crontab, вы обычно должны заканчивать файл crontab пустой новой строкой. И скрипт выше гарантирует, что сначала он удаляет все пустые строки с помощью "grep". части, а затем добавьте новую пустую строку в конце с "\n" в конце скрипта. Это также предотвратит появление пустой строки над новой командой, если существующий файл crontab заканчивается пустой строкой.

0

script, чтобы добавить cronjob. проверить повторяющиеся записи, используемые выражения * > "

cronjob_creator () {         
# usage: cronjob_creator '<interval>' '<command>'

  if [[ -z $1 ]] ;then
    printf " no interval specified\n"
elif [[ -z $2 ]] ;then
    printf " no command specified\n"
else
    CRONIN="/tmp/cti_tmp"
    crontab -l | grep -vw "$1 $2" > "$CRONIN"
    echo "$1 $2" >> $CRONIN
    crontab "$CRONIN"
    rm $CRONIN
fi
}

:

$ ./cronjob_creator.sh '*/10 * * * *' 'echo "this is a test" > export_file'
$ crontab  -l
$ */10 * * * * echo "this is a test" > export_file

источник: мой мозг;)

0

Нет, в crontab нет возможности изменять файлы cron.

Вы должны: взять текущий cron файл (crontab -l > newfile), изменить его и поместить новый файл на место (crontab newfile).

Если вы знакомы с perl, вы можете использовать этот модуль Config:: Crontab.

LLP, Andrea

-4

Возможно, вы можете изменить редактор по умолчанию на ed и использовать heredoc для редактирования.

EDITOR=ed
export EDITOR

crontab -e << EOF
> a
> * * * * * Myscript
> * * * * * AnotherScript
> * * * * * MoreScript
> .
> w
> q
> EOF

Обратите внимание, что ведущий в этом коде означает, что нажата клавиша return/enter, чтобы создать новую строку.

a означает APPEND, чтобы он ничего не перезаписывал.

. означает, что вы закончили редактирование.

w означает WRITE изменения.

q означает QUIT или exit ed.

вы можете проверить это

crontab -l

Вы также можете удалить запись.

EDITOR=ed
export EDITOR

crontab -e << EOF
> /Myscript/
> d
> .
> w
> q
> EOF

Это приведет к удалению записи crontab с Myscript в ней.

d означает удаление шаблона внутри //.

Не проверять его снова

crontab -l

Это решение работает внутри script тоже меньше : -)

  • 0
    downvote для сценариев редактора

Ещё вопросы

Сообщество Overcoder
Наверх
Меню