[MIG] cron_daylight_saving_time_resistant: Migration to 14.0

pull/2578/head
Florian da Costa 2022-02-16 11:47:32 +01:00 committed by clementmbr
parent 81a98a27e0
commit 8af4df7d56
8 changed files with 111 additions and 215 deletions

View File

@ -1,83 +0,0 @@
.. image:: https://img.shields.io/badge/license-AGPL--3-blue.png
:target: https://www.gnu.org/licenses/agpl
:alt: License: AGPL-3
===================================
Cron daylight saving time resistant
===================================
This module adjust cron to run at fixed hours, local time.
Without this module, when a daylight saving time change occur, the cron will not take
the hour change in account.
With this module, when a daylight saving time change occur, the offset (+1 or -1 hour)
will be applied.
Usage
=====
To use this module, you need to edit a cron, and check the option,
"Daylight saving time resistant".
#. Go to ...
.. image:: https://odoo-community.org/website/image/ir.attachment/5784_f2813bd/datas
:alt: Try me on Runbot
:target: https://runbot.odoo-community.org/runbot/149/9.0
Known issues / Roadmap
======================
* Write tests
* Edge cases like run every 5 minutes + dst resistant may behave
incorrectly during the time change.
Bug Tracker
===========
Bugs are tracked on `GitHub Issues
<https://github.com/OCA/server-tools/issues>`_. In case of trouble, please
check there if your issue has already been reported. If you spotted it first,
help us smash it by providing detailed and welcomed feedback.
Credits
=======
Images
------
* Odoo Community Association: `Icon <https://odoo-community.org/logo.png>`_.
Contributors
------------
* Raphaël Reverdy https://akretion.com
Do not contact contributors directly about support or help with technical issues.
Funders
-------
The development of this module has been financially supported by:
* Akretion
Maintainer
----------
.. image:: https://odoo-community.org/logo.png
:alt: Odoo Community Association
:target: https://odoo-community.org
This module is maintained by the OCA.
OCA, or the Odoo Community Association, is a nonprofit organization whose
mission is to support the collaborative development of Odoo features and
promote its widespread use.
To contribute to this module, please visit https://odoo-community.org.

View File

@ -1,9 +1,8 @@
# Copyright <2018> <Akretion>
# License AGPL-3.0 or later (https://www.gnu.org/licenses/agpl).
{
"name": "Cron daylight saving time resistant",
"summary": "Run cron on fixed hours",
"version": "9.0.1.0.0",
"version": "14.0.1.0.0",
"category": "Tools",
"website": "https://github.com/OCA/server-tools",
"author": "akretion, Odoo Community Association (OCA)",
@ -13,6 +12,6 @@
"base",
],
"data": [
"views/cron.xml",
"views/ir_cron_views.xml",
],
}

View File

@ -1,14 +1,13 @@
# © 2018 Akretion - Raphaël Reverdy
# License AGPL-3.0 or later (https://www.gnu.org/licenses/agpl).
import logging
from datetime import datetime
import pytz
from openerp import api, fields, models
from openerp.addons.base.ir.ir_cron import _intervalTypes
from openerp.osv import fields as osv_fields
from openerp.tools import DEFAULT_SERVER_DATETIME_FORMAT
from odoo import api, fields, models
from odoo.addons.base.models.ir_cron import _intervalTypes
_logger = logging.getLogger(__name__)
@ -37,55 +36,52 @@ class IrCron(models.Model):
diff_offset = after_offset - before_offset
return diff_offset
@classmethod
def _process_job(cls, job_cr, job, cron_cr):
"""Add or remove the Daylight saving offset when needed."""
super(IrCron, cls)._process_job(job_cr, job, cron_cr)
super()._process_job(job_cr, job, cron_cr)
# changing the date has to be after the super, else, e may add a hour
# to next call, and the super will no run the cron, (because now will
# be 1 hour too soon) and the date will just be incremented of 1
# hour, each hour...until the changes time really occurs...
# if need to test this, use freeze_gun lib.
if job["daylight_saving_time_resistant"]:
with api.Environment.manage():
now = osv_fields.datetime.context_timestamp(
job_cr, job["user_id"], datetime.now()
)
nextcall = osv_fields.datetime.context_timestamp(
job_cr,
job["user_id"],
datetime.strptime(
job["nextcall"], # original nextcall
DEFAULT_SERVER_DATETIME_FORMAT,
),
)
numbercall = job["numbercall"]
delta = _intervalTypes[job["interval_type"]](job["interval_number"])
diff_offset = cls._calculate_daylight_offset(
nextcall, delta, numbercall, now
)
if diff_offset and nextcall < now and numbercall:
cron_cr.execute(
"""
SELECT nextcall FROM ir_cron WHERE id = %s
""",
(job["id"],),
)
res_sql = cron_cr.fetchall()
new_nextcall = res_sql and res_sql[0][0]
new_nextcall = osv_fields.datetime.context_timestamp(
try:
cron = api.Environment(
job_cr,
job["user_id"],
datetime.strptime(
new_nextcall, # original nextcall
DEFAULT_SERVER_DATETIME_FORMAT,
),
)
new_nextcall -= diff_offset
modified_next_call = fields.Datetime.to_string(
new_nextcall.astimezone(pytz.UTC)
)
cron_cr.execute(
"UPDATE ir_cron SET nextcall=%s WHERE id=%s",
(modified_next_call, job["id"]),
{"lastcall": fields.Datetime.from_string(job["lastcall"])},
)[cls._name]
now = fields.Datetime.context_timestamp(cron, datetime.now())
# original nextcall
nextcall = fields.Datetime.context_timestamp(cron, job["nextcall"])
numbercall = job["numbercall"]
delta = _intervalTypes[job["interval_type"]](job["interval_number"])
diff_offset = cron._calculate_daylight_offset(
nextcall, delta, numbercall, now
)
if diff_offset and nextcall < now and numbercall:
cron_cr.execute(
"""
SELECT nextcall FROM ir_cron WHERE id = %s
""",
(job["id"],),
)
res_sql = cron_cr.fetchall()
new_nextcall = res_sql and res_sql[0][0]
new_nextcall = fields.Datetime.context_timestamp(
cron, new_nextcall
)
new_nextcall -= diff_offset
modified_next_call = fields.Datetime.to_string(
new_nextcall.astimezone(pytz.UTC)
)
cron_cr.execute(
"UPDATE ir_cron SET nextcall=%s WHERE id=%s",
(modified_next_call, job["id"]),
)
cron.flush()
cron.invalidate_cache()
finally:
job_cr.commit()
cron_cr.commit()

View File

@ -0,0 +1,2 @@
* Go to the menu Settings => Technical => Automation => Scheduled Actions
Then you can check the check box Daylight Saving Time Resistant

View File

@ -0,0 +1,2 @@
* Raphaël Reverdy https://akretion.com
* Florian da Costa <florian.dacosta@akretion.com>

View File

@ -0,0 +1,8 @@
This module adjust cron to run at fixed hours, local time.
Without this module, when a daylight saving time change occur, the cron will not take
the hour change in account.
With this module, when a daylight saving time change occur, the offset (+1 or -1 hour)
will be applied.

View File

@ -1,89 +1,62 @@
# © 2018 Akretion <https://akretion.com>
# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl).
from datetime import datetime, timedelta
from openerp.tests.common import TransactionCase
from pytz import timezone
from freezegun import freeze_time
import odoo
from odoo import fields
from odoo.tests.common import TransactionCase
from odoo.tools.misc import DEFAULT_SERVER_DATETIME_FORMAT
class TestDST(TransactionCase):
def test_dst(self):
"""First test, caching some data."""
cron = self.env["ir.cron"]
brux = timezone("Europe/Brussels")
ncall = -1
winter_jan_12 = datetime(2018, 1, 1, 12, 0)
winter_feb_0 = datetime(2018, 1, 2, 0, 0)
summer_june_12 = datetime(2018, 6, 15, 12, 0)
summer_sep_3 = datetime(2018, 9, 17, 3, 0)
winter_jan_next_year = datetime(2019, 2, 3, 0, 0)
tests = [
{
"nextcall": brux.localize(winter_jan_12),
"delta": timedelta(days=5),
"now": brux.localize(winter_feb_0),
"expected": timedelta(hours=0),
},
{
"nextcall": brux.localize(winter_jan_12),
"delta": timedelta(days=6 * 30),
"now": brux.localize(winter_feb_0),
"expected": timedelta(hours=1),
},
{
"nextcall": brux.localize(winter_jan_12),
"delta": timedelta(days=5),
"now": brux.localize(summer_june_12),
"expected": timedelta(hours=1),
},
{
"nextcall": brux.localize(winter_jan_12),
"delta": timedelta(days=6 * 30),
"now": brux.localize(summer_june_12),
"expected": timedelta(hours=1),
},
{
"nextcall": brux.localize(winter_jan_12),
"delta": timedelta(days=6 * 365),
"now": brux.localize(winter_jan_next_year),
"expected": timedelta(hours=0),
},
]
tests = tests + [
{
"nextcall": brux.localize(summer_june_12),
"delta": timedelta(days=5),
"now": brux.localize(winter_jan_next_year),
"expected": timedelta(hours=-1),
},
{
"nextcall": brux.localize(summer_june_12),
"delta": timedelta(days=4 * 30),
"now": brux.localize(winter_jan_next_year),
"expected": timedelta(hours=-1),
},
{
"nextcall": brux.localize(summer_june_12),
"delta": timedelta(days=5),
"now": brux.localize(summer_june_12),
"expected": timedelta(hours=0),
},
{
"nextcall": brux.localize(summer_june_12),
"delta": timedelta(days=6 * 30),
"now": brux.localize(summer_sep_3),
"expected": timedelta(hours=-1),
},
{
"nextcall": brux.localize(summer_june_12),
"delta": timedelta(days=6 * 365),
"now": brux.localize(summer_sep_3),
"expected": timedelta(hours=0),
},
]
test = tests[0]
for test in tests:
res = cron._calculate_daylight_offset(
test["nextcall"], test["delta"], ncall, test["now"]
def setUp(self):
super().setUp()
self.registry.enter_test_mode(self.env.cr)
def tearDown(self):
self.registry.leave_test_mode()
super().tearDown()
def _check_cron_date_after_run(self, cron, datetime_str):
# add 10 sec to make sure cron will run
datetime_current = datetime.strptime(
datetime_str, DEFAULT_SERVER_DATETIME_FORMAT
) + timedelta(seconds=10)
datetime_current_str = datetime_current.strftime(DEFAULT_SERVER_DATETIME_FORMAT)
with freeze_time(datetime_current_str):
cron.write(
{"nextcall": datetime_str, "daylight_saving_time_resistant": True}
)
self.assertEqual(res, test["expected"])
cron.flush()
self.env.cr.execute("SELECT * FROM ir_cron WHERE id = %s", (cron.id,))
job = self.env.cr.dictfetchall()[0]
timezone_date_orig = fields.Datetime.context_timestamp(cron, cron.nextcall)
with odoo.registry(self.env.cr.dbname).cursor() as new_cr:
registry = odoo.registry(new_cr.dbname)
registry["ir.cron"]._process_job(new_cr, job, new_cr)
day_after_date_orig = (timezone_date_orig + timedelta(days=1)).day
timezone_date_after = fields.Datetime.context_timestamp(cron, cron.nextcall)
# check the cron is really planned the next day (which mean it has run
# then check the planned hour is the same even in case of change of time
# (brussels summer time/ brussels winter time
self.assertEqual(day_after_date_orig, timezone_date_after.day)
self.assertEqual(timezone_date_orig.hour, timezone_date_after.hour)
def test_cron(self):
cron = self.env["ir.cron"].create(
{
"name": "TestCron",
"model_id": self.env.ref("base.model_res_partner").id,
"state": "code",
"code": "model.search([])",
"interval_number": 1,
"interval_type": "days",
"numbercall": -1,
"doall": False,
}
)
# from summer time to winter time
self._check_cron_date_after_run(cron, "2021-10-30 15:00:00")
# from winter time to summer time
self._check_cron_date_after_run(cron, "2021-03-27 15:00:00")

View File

@ -1,9 +1,8 @@
<?xml version="1.0" encoding="utf-8" ?>
<odoo>
<record id="ir_cron_view" model="ir.ui.view">
<field name="name">ir.cron.resist_dst</field>
<field name="model">ir.cron</field>
<field name="inherit_id" ref="base.ir_cron_view" />
<field name="inherit_id" ref="base.ir_cron_view_form" />
<field name="arch" type="xml">
<field name="doall" position="after">
<field name="daylight_saving_time_resistant" />