mirror of https://github.com/interlegis/sigi.git
Sesóstris Vieira
11 months ago
15 changed files with 574 additions and 12 deletions
@ -0,0 +1,90 @@ |
|||||
|
from django_extensions.management.jobs import BaseJob |
||||
|
from django_extensions.management.jobs import get_jobs |
||||
|
from django.utils import timezone |
||||
|
from django.utils.formats import localize |
||||
|
from django.utils.translation import gettext as _ |
||||
|
from sigi.apps.utils.models import Cronjob, JobSchedule |
||||
|
|
||||
|
WHEN_SETS = { |
||||
|
"daily": "0 0 * * *", |
||||
|
"hourly": "0 * * * *", |
||||
|
"monthly": "0 0 1 * *", |
||||
|
"weekly": "0 0 * * 0", |
||||
|
"yearly": "0 0 1 1 *", |
||||
|
"minutely": "* * * * *", |
||||
|
} |
||||
|
|
||||
|
|
||||
|
class Job(BaseJob): |
||||
|
help = "Controlador de cronjobs do SIGI." |
||||
|
|
||||
|
def execute(self): |
||||
|
print("Rodando controlador de jobs...") |
||||
|
self.remove_old_jobs() |
||||
|
self.sync_new_jobs() |
||||
|
self.run_scheduled() |
||||
|
self.schedule_jobs() |
||||
|
|
||||
|
def remove_old_jobs(self): |
||||
|
"""Remover das tabelas os jobs que foram removidos do código""" |
||||
|
print("\tRemover das tabelas os jobs que foram removidos do código...") |
||||
|
all_jobs = get_jobs() |
||||
|
excludes = Cronjob.objects.all() |
||||
|
for app_name, job_name in all_jobs.keys(): |
||||
|
excludes = excludes.exclude(app_name=app_name, job_name=job_name) |
||||
|
print("\t\t", excludes.delete()) |
||||
|
|
||||
|
def sync_new_jobs(self): |
||||
|
""" |
||||
|
Atualizar a tabela de JOBS com os novos JOBS que tenham sido criados |
||||
|
""" |
||||
|
print( |
||||
|
"\tAtualizar a tabela de JOBS com os novos JOBS que tenham " |
||||
|
"sido criados..." |
||||
|
) |
||||
|
all_jobs = get_jobs() |
||||
|
for (app_name, job_name), JobClass in all_jobs.items(): |
||||
|
if app_name == "sigi.apps.utils" and job_name == "job_controller": |
||||
|
# Ignorar job_controller |
||||
|
continue |
||||
|
try: |
||||
|
job = Cronjob.objects.get(app_name=app_name, job_name=job_name) |
||||
|
except Cronjob.DoesNotExist: |
||||
|
# Inserir o JOB na tabela de JOBS # |
||||
|
job_obj = JobClass() |
||||
|
if job_obj.when in WHEN_SETS: |
||||
|
expressao_cron = WHEN_SETS[job_obj.when] |
||||
|
else: |
||||
|
expressao_cron = WHEN_SETS["daily"] # Default |
||||
|
job = Cronjob( |
||||
|
app_name=app_name, |
||||
|
job_name=job_name, |
||||
|
expressao_cron=expressao_cron, |
||||
|
) |
||||
|
job.save() |
||||
|
print(f"\t\tNovo job encontrado: {job_name}: {job_obj.help}") |
||||
|
|
||||
|
def run_scheduled(self): |
||||
|
"""Executa os jobs que estão agendados""" |
||||
|
print("\tExecutar os jobs que estão agendados...") |
||||
|
for sched in JobSchedule.objects.filter( |
||||
|
status=JobSchedule.STATUS_AGENDADO |
||||
|
): |
||||
|
agora = timezone.localtime() |
||||
|
if sched.iniciar <= agora: |
||||
|
sched.run_job() |
||||
|
|
||||
|
def schedule_jobs(self): |
||||
|
"""Criar agenda para próxima execução""" |
||||
|
print("\tCriar agenda para próxima execução...") |
||||
|
for job in Cronjob.objects.exclude( |
||||
|
jobschedule__status__in=[ |
||||
|
JobSchedule.STATUS_AGENDADO, |
||||
|
JobSchedule.STATUS_EXECUTANDO, |
||||
|
] |
||||
|
): |
||||
|
sched = job.next_schedule() |
||||
|
print( |
||||
|
f"\t\tAgendado job {sched.job.job_name} " |
||||
|
f"para {localize(sched.iniciar)}" |
||||
|
) |
@ -0,0 +1,45 @@ |
|||||
|
# Generated by Django 4.2.7 on 2024-02-26 12:53 |
||||
|
|
||||
|
from django.db import migrations, models |
||||
|
import django.db.models.deletion |
||||
|
|
||||
|
|
||||
|
class Migration(migrations.Migration): |
||||
|
|
||||
|
dependencies = [ |
||||
|
('utils', '0001_initial'), |
||||
|
] |
||||
|
|
||||
|
operations = [ |
||||
|
migrations.CreateModel( |
||||
|
name='Cronjob', |
||||
|
fields=[ |
||||
|
('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')), |
||||
|
('app_name', models.CharField(editable=False, max_length=100, verbose_name='app')), |
||||
|
('job_name', models.CharField(editable=False, max_length=100, verbose_name='job')), |
||||
|
('expressao_cron', models.CharField(default='* * * * *', help_text="Usar expressoões no formato padrão de CRON: 'minute hour day month day-of-week'. Mais detalhes: <a href='https://help.ubuntu.com/community/CronHowto'>CronHowTo</a>", max_length=100, verbose_name='expressão CRON')), |
||||
|
], |
||||
|
options={ |
||||
|
'verbose_name': 'Cron job', |
||||
|
'verbose_name_plural': 'Cron jobs', |
||||
|
'ordering': ('app_name', 'job_name'), |
||||
|
}, |
||||
|
), |
||||
|
migrations.CreateModel( |
||||
|
name='JobSchedule', |
||||
|
fields=[ |
||||
|
('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')), |
||||
|
('iniciar', models.DateTimeField(verbose_name='Iniciar em')), |
||||
|
('iniciado', models.DateTimeField(blank=True, null=True, verbose_name='Iniciado em')), |
||||
|
('status', models.CharField(choices=[('A', 'Agendado'), ('E', 'Executando'), ('C', 'Concluído')], default='A', max_length=1, verbose_name='estado')), |
||||
|
('tempo_gasto', models.DurationField(blank=True, editable=False, null=True, verbose_name='tempo gasto')), |
||||
|
('resultado', models.TextField(blank=True, editable=False, verbose_name='resultado da execução')), |
||||
|
('job', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, to='utils.cronjob', verbose_name='Cron job')), |
||||
|
], |
||||
|
options={ |
||||
|
'verbose_name': 'Agenda de execução', |
||||
|
'verbose_name_plural': 'Agenda de execuções', |
||||
|
'ordering': ('iniciar',), |
||||
|
}, |
||||
|
), |
||||
|
] |
@ -0,0 +1,33 @@ |
|||||
|
{% extends "admin/change_form.html" %} |
||||
|
{% load static i18n %} |
||||
|
|
||||
|
{% block extrastyle %} |
||||
|
{{ block.super }} |
||||
|
<style type="text/css"> |
||||
|
.display { |
||||
|
flex: 1; |
||||
|
} |
||||
|
h1 { |
||||
|
font-size: 2.5rem; |
||||
|
} |
||||
|
h2 { |
||||
|
font-size: 2rem; |
||||
|
} |
||||
|
h3 { |
||||
|
font-size: 1.5rem; |
||||
|
} |
||||
|
</style> |
||||
|
{% endblock %} |
||||
|
|
||||
|
{% block after_field_sets %} |
||||
|
<fieldset class="module aligned "> |
||||
|
<div class="form-row field-resultado"> |
||||
|
<div class="input-field"> |
||||
|
<div class="readonly-label"><label>Resultado:</label></div> |
||||
|
<div class="readonly"> |
||||
|
<pre>{{ original.resultado }}</pre> |
||||
|
</div> |
||||
|
</div> |
||||
|
</div> |
||||
|
</fieldset> |
||||
|
{% endblock %} |
Loading…
Reference in new issue