Виконайте скрипт Python через crontab


90

Я намагаюся виконати скрипт python, використовуючи Linux crontab. Я хочу запускати цей сценарій кожні 10 хвилин.

Я знайшов багато рішень, і жодне з них не працює. Наприклад: відредагуйте анакрон у /etc/cron.d або використовуйте crontab -e. Я розміщую цей рядок у кінці файлу, але це нічого не змінює. Чи потрібно перезапускати будь-яку службу?

*/2 * * * * /usr/bin/python /home/souza/Documets/Listener/listener.py

Який файл потрібно редагувати, щоб це налаштувати?

Спасибі заздалегідь


Ось сценарій.

#!/usr/bin/python
# -*- coding: iso-8859-15 -*-

import json
import os
import pycurl
import sys
import cStringIO

if __name__ == "__main__":

    name_server_standart = "Server created by script %d"
    json_file_standart = "{ \"server\" : {  \"name\" : \"%s\", \"imageRef\" : \"%s\", \"flavorRef\" : \"%s\" } }"

    curl_auth_token = pycurl.Curl()

    gettoken = cStringIO.StringIO()

    curl_auth_token.setopt(pycurl.URL, "http://192.168.100.241:8774/v1.1")
    curl_auth_token.setopt(pycurl.POST, 1)
    curl_auth_token.setopt(pycurl.HTTPHEADER, ["X-Auth-User: cpca", 
                          "X-Auth-Key: 438ac2d9-689f-4c50-9d00-c2883cfd38d0"])

    curl_auth_token.setopt(pycurl.HEADERFUNCTION, gettoken.write)
    curl_auth_token.perform()
    chg = gettoken.getvalue()

    auth_token = chg[chg.find("X-Auth-Token: ")+len("X-Auth-Token: ") : chg.find("X-Server-Management-Url:")-1]

    token = "X-Auth-Token: {0}".format(auth_token)
    curl_auth_token.close()

    #----------------------------

    getter = cStringIO.StringIO()
    curl_hab_image = pycurl.Curl()
    curl_hab_image.setopt(pycurl.URL, "http://192.168.100.241:8774/v1.1/nuvemcpca/images/7")
    curl_hab_image.setopt(pycurl.HTTPGET, 1) #tirei essa linha e funcionou, nao sei porque
    curl_hab_image.setopt(pycurl.HTTPHEADER, [token])

    curl_hab_image.setopt(pycurl.WRITEFUNCTION, getter.write)
    #curl_list.setopt(pycurl.VERBOSE, 1)
    curl_hab_image.perform()
    curl_hab_image.close()

    getter = cStringIO.StringIO()

    curl_list = pycurl.Curl()
    curl_list.setopt(pycurl.URL, "http://192.168.100.241:8774/v1.1/nuvemcpca/servers/detail")
    curl_list.setopt(pycurl.HTTPGET, 1) #tirei essa linha e funcionou, nao sei porque
    curl_list.setopt(pycurl.HTTPHEADER, [token])

    curl_list.setopt(pycurl.WRITEFUNCTION, getter.write)
    #curl_list.setopt(pycurl.VERBOSE, 1)
    curl_list.perform()
    curl_list.close()

    #----------------------------

    resp = getter.getvalue()    

    con = int(resp.count("status"))

    s = json.loads(resp)

    lst = []

    for i in range(con):
        lst.append(s['servers'][i]['status'])

    for j in range(len(lst)):
        actual = lst.pop()
        print actual

        if actual != "ACTIVE" and actual != "BUILD" and actual != "REBOOT" and actual != "RESIZE":

            print "Entra no If"

            f = file('counter', 'r+w')

            num = 0
            for line in f:
                num = line

            content = int(num)+1    

            ins = str(content)

            f.seek(0)
            f.write(ins)
            f.truncate()
            f.close()

            print "Contador"

            json_file = file('json_file_create_server.json','r+w')

            name_server_final = name_server_standart % content
            path_to_image = "http://192.168.100.241:8774/v1.1/nuvemcpca/images/7"
            path_to_flavor = "http://192.168.100.241:8774/v1.1/nuvemcpca/flavors/1"

            new_json_file_content = json_file_standart % (name_server_final, path_to_image, path_to_flavor)

            json_file.seek(0)
            json_file.write(new_json_file_content)
            json_file.truncate()
            json_file.close()

            print "Json File"

            fil = file("json_file_create_server.json")
            siz = os.path.getsize("json_file_create_server.json")

            cont_size = "Content-Length: %d" % siz
            cont_type = "Content-Type: application/json"
            accept = "Accept: application/json"

            c_create_servers = pycurl.Curl()

            logger = cStringIO.StringIO()

            c_create_servers.setopt(pycurl.URL, "http://192.168.100.241:8774/v1.1/nuvemcpca/servers")

            c_create_servers.setopt(pycurl.HTTPHEADER, [token, cont_type, accept, cont_size])

            c_create_servers.setopt(pycurl.POST, 1)

            c_create_servers.setopt(pycurl.INFILE, fil)

            c_create_servers.setopt(pycurl.INFILESIZE, siz)

            c_create_servers.setopt(pycurl.WRITEFUNCTION, logger.write)

            print "Teste perform"

            c_create_servers.perform()

            print logger.getvalue()

            c_create_servers.close()

Коли ви говорите "це нічого не змінює". Чи відображається помилка, чи не працює? Яка поведінка?
Рауль Маренго,

чи навмисне "Документи" замість "Документи"?
Рауль Маренго,

Просто нічого не відбувається. :(
guisantogui

Це виходить за рамки питання, але що, як ви очікуєте, зробить ваш сценарій "listener.py"? Чи робить він щось, що може свідчити про те, що він запущений? Зробіть ps -ef | grep 'crond' у вашому командному рядку, щоб перевірити, чи запущений cron.
Рауль Маренго,

Ні, цей скрипт надсилає кілька команд cURL на інший комп’ютер. Коли я виконую "ps -f | grep 'crond'", він повертає це: "souza 4736 3947 0 14:01 очок / 1 00:00:00 grep --color = auto crond"
guisantogui

Відповіді:


131

Просто скористайтесь crontab -eі дотримуйтесь підручника тут.

Подивіться на пункт 3 керівництво щодо того, як вказати частоту.

Виходячи з ваших вимог, це має бути ефективно:

*/10 * * * * /usr/bin/python script.py

1
Я слідую цьому посібнику, але коли я зберігаю файл, з'являється повідомлення: "/tmp/crontab.JTQ0My/crontab":22: помилки помилок у файлі crontab, не вдається встановити. Хочете повторити те саме редагування? (y / n) "якщо я вводжу" y ", я повернувся до редагування файлу. І якщо я вводжу" n ", файл не зберігається. Я додаю цей рядок в останньому рядку файлу:" / 1 * * * * / usr / bin / python script.py "
guisantogui

@guisantogui у підручнику є пункт, який пояснює, що використання "/ 1" може підтримуватися не всіма операційними системами. На якій операційній системі ви використовуєте це?
Рауль Маренго,

3
@guisantogui щойно помітив, що вам не вистачає "*" перед "/"
Рауль Маренго,

інший спосіб - це додати декларацію env у ваш script.py. Перегляньте мої коментарі до прийнятого рішення за адресою: stackoverflow.com/questions/25633737/python-crontab-and-paths
Кетцалькоатль

Що робити, якщо ви хочете виконати script.pyєдиний у даному каталозі?
Шубхем А.

65

Помістіть свій сценарій у файл, foo.pyпочинаючи з

#!/usr/bin/python

потім надайте дозвіл на виконання цього сценарію за допомогою

chmod a+x foo.py

і використовувати повний шлях до вашого foo.pyфайлу у вашому crontab.

Див. Документацію execve (2), яка обробляє шебанг


1
@Tomer Якщо це shсценарії оболонки POSIX, тоді так. Якщо вони використовують нестандартні можливості , специфічні для ksh, zshабо bashто вони повинні бути виконані з допомогою цієї конкретної оболонки.
триплеє

25

Як ви вже згадували, це нічого не змінює ,

По-перше, вам слід перенаправити як stdin, так і stderr із виконання crontab, як показано нижче:

*/2 * * * * /usr/bin/python /home/souza/Documets/Listener/listener.py > /tmp/listener.log 2>&1

тоді ви можете переглянути файл /tmp/listener.log щоб побачити, чи сценарій виконаний так, як ви очікували.

По-друге, вгадайте, що ви маєте на увазі щось змінити , переглядаючи файли, створені вашою програмою:

f = file('counter', 'r+w')
json_file = file('json_file_create_server.json','r+w')

завдання crontab вище не створить цей файл у каталозі /home/souza/Documets/Listener, оскільки завдання cron не виконується в цьому каталозі, і ви використовуєте відносний шлях у програмі. Отже, щоб створити цей файл у каталозі /home/souza/Documets/Listener, наступна робота cron зробить трюк:

*/2 * * * * cd /home/souza/Documets/Listener && /usr/bin/python listener.py > /tmp/listener.log 2>&1

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


що означає 2> & 1?
Мохідін бен Мохаммед

1
@MohideenibnMohammed перенаправляє повідомлення про помилки ( stderr) у видимий командний рядок ( stdout)
Юха Унтінен

Цю відповідь слід використовувати, якщо використовуєте відносні шляхи.
DaReal
Використовуючи наш веб-сайт, ви визнаєте, що прочитали та зрозуміли наші Політику щодо файлів cookie та Політику конфіденційності.
Licensed under cc by-sa 3.0 with attribution required.