From 02bc747d607d9f040ecd6b455dd8007bba819de6 Mon Sep 17 00:00:00 2001 From: Guewen Baconnier Date: Thu, 29 Oct 2020 09:04:43 +0100 Subject: [PATCH] Add ddmrp_cron_actions_as_job It makes calls to "cron_actions" run in queue jobs. The jobs have an identity key with "identity_exact", meaning that only one occurence of a job for the same buffer with the same arguments (only_nfp) will be created at a time (e.g. when the state of a stock.move is changed several times in the same transaction or in a different transaction in a short timeframe). It needs https://github.com/OCA/queue/pull/274 and https://github.com/OCA/queue/pull/275 --- ddmrp_cron_actions_as_job/__init__.py | 1 + ddmrp_cron_actions_as_job/__manifest__.py | 15 +++++ .../data/queue_job_channel_data.xml | 6 ++ .../data/queue_job_function_data.xml | 7 +++ ddmrp_cron_actions_as_job/models/__init__.py | 2 + .../models/stock_buffer.py | 51 +++++++++++++++++ .../models/stock_move.py | 13 +++++ .../readme/CONTRIBUTORS.rst | 1 + .../readme/DESCRIPTION.rst | 14 +++++ ddmrp_cron_actions_as_job/tests/__init__.py | 1 + .../tests/test_cron_actions_as_job.py | 56 +++++++++++++++++++ 11 files changed, 167 insertions(+) create mode 100644 ddmrp_cron_actions_as_job/__init__.py create mode 100644 ddmrp_cron_actions_as_job/__manifest__.py create mode 100644 ddmrp_cron_actions_as_job/data/queue_job_channel_data.xml create mode 100644 ddmrp_cron_actions_as_job/data/queue_job_function_data.xml create mode 100644 ddmrp_cron_actions_as_job/models/__init__.py create mode 100644 ddmrp_cron_actions_as_job/models/stock_buffer.py create mode 100644 ddmrp_cron_actions_as_job/models/stock_move.py create mode 100644 ddmrp_cron_actions_as_job/readme/CONTRIBUTORS.rst create mode 100644 ddmrp_cron_actions_as_job/readme/DESCRIPTION.rst create mode 100644 ddmrp_cron_actions_as_job/tests/__init__.py create mode 100644 ddmrp_cron_actions_as_job/tests/test_cron_actions_as_job.py diff --git a/ddmrp_cron_actions_as_job/__init__.py b/ddmrp_cron_actions_as_job/__init__.py new file mode 100644 index 000000000..0650744f6 --- /dev/null +++ b/ddmrp_cron_actions_as_job/__init__.py @@ -0,0 +1 @@ +from . import models diff --git a/ddmrp_cron_actions_as_job/__manifest__.py b/ddmrp_cron_actions_as_job/__manifest__.py new file mode 100644 index 000000000..b8b121772 --- /dev/null +++ b/ddmrp_cron_actions_as_job/__manifest__.py @@ -0,0 +1,15 @@ +# Copyright 2020 Camptocamp (https://www.camptocamp.com) +# License LGPL-3.0 or later (https://www.gnu.org/licenses/lgpl.html). + +{ + "name": "DDMRP Buffer Calculation as job", + "version": "13.0.1.0.0", + "summary": "Run DDMRP Buffer Calculation as jobs", + "author": "Camptocamp, Odoo Community Association (OCA)", + "website": "https://github.com/OCA/ddmrp", + "category": "Warehouse Management", + "depends": ["ddmrp", "queue_job"], + "data": ["data/queue_job_channel_data.xml", "data/queue_job_function_data.xml"], + "license": "LGPL-3", + "installable": True, +} diff --git a/ddmrp_cron_actions_as_job/data/queue_job_channel_data.xml b/ddmrp_cron_actions_as_job/data/queue_job_channel_data.xml new file mode 100644 index 000000000..b3053db8e --- /dev/null +++ b/ddmrp_cron_actions_as_job/data/queue_job_channel_data.xml @@ -0,0 +1,6 @@ + + + ddmrp + + + diff --git a/ddmrp_cron_actions_as_job/data/queue_job_function_data.xml b/ddmrp_cron_actions_as_job/data/queue_job_function_data.xml new file mode 100644 index 000000000..eb90f00f3 --- /dev/null +++ b/ddmrp_cron_actions_as_job/data/queue_job_function_data.xml @@ -0,0 +1,7 @@ + + + + cron_actions + + + diff --git a/ddmrp_cron_actions_as_job/models/__init__.py b/ddmrp_cron_actions_as_job/models/__init__.py new file mode 100644 index 000000000..911aa93ea --- /dev/null +++ b/ddmrp_cron_actions_as_job/models/__init__.py @@ -0,0 +1,2 @@ +from . import stock_buffer +from . import stock_move diff --git a/ddmrp_cron_actions_as_job/models/stock_buffer.py b/ddmrp_cron_actions_as_job/models/stock_buffer.py new file mode 100644 index 000000000..202ea0d99 --- /dev/null +++ b/ddmrp_cron_actions_as_job/models/stock_buffer.py @@ -0,0 +1,51 @@ +# Copyright 2020 Camptocamp (https://www.camptocamp.com) +# License LGPL-3.0 or later (https://www.gnu.org/licenses/lgpl.html). + +from odoo import models + +from odoo.addons.queue_job.job import identity_exact + + +class Buffer(models.Model): + _inherit = "stock.buffer" + + def cron_actions_job_options(self, only_nfp=False): + return { + "identity_key": identity_exact, + "priority": 15, + "description": "DDMRP Buffer calculation ({})".format(self.display_name), + } + + def _calc_adu_job_options(self): + return { + "identity_key": identity_exact, + "priority": 15, + "description": "DDMRP Buffer ADU calculation ({})".format( + self.display_name + ), + } + + def _register_hook(self): + self._patch_method( + "cron_actions", + self._patch_job_auto_delay( + "cron_actions", context_key="auto_delay_ddmrp_cron_actions" + ), + ) + self._patch_method( + "_calc_adu", + self._patch_job_auto_delay( + "_calc_adu", context_key="auto_delay_ddmrp_calc_adu" + ), + ) + return super()._register_hook() + + def cron_ddmrp(self, automatic=False): + return super( + Buffer, self.with_context(auto_delay_ddmrp_cron_actions=True) + ).cron_ddmrp(automatic=automatic) + + def cron_ddmrp_adu(self, automatic=False): + return super( + Buffer, self.with_context(auto_delay_ddmrp_calc_adu=True) + ).cron_ddmrp_adu(automatic=automatic) diff --git a/ddmrp_cron_actions_as_job/models/stock_move.py b/ddmrp_cron_actions_as_job/models/stock_move.py new file mode 100644 index 000000000..01abd2455 --- /dev/null +++ b/ddmrp_cron_actions_as_job/models/stock_move.py @@ -0,0 +1,13 @@ +# Copyright 2020 Camptocamp +# License LGPL-3.0 or later (https://www.gnu.org/licenses/lgpl.html). + +from odoo import models + + +class StockMove(models.Model): + _inherit = "stock.move" + + def _update_ddmrp_nfp(self): + return super( + StockMove, self.with_context(auto_delay_ddmrp_cron_actions=True) + )._update_ddmrp_nfp() diff --git a/ddmrp_cron_actions_as_job/readme/CONTRIBUTORS.rst b/ddmrp_cron_actions_as_job/readme/CONTRIBUTORS.rst new file mode 100644 index 000000000..48286263c --- /dev/null +++ b/ddmrp_cron_actions_as_job/readme/CONTRIBUTORS.rst @@ -0,0 +1 @@ +* Guewen Baconnier diff --git a/ddmrp_cron_actions_as_job/readme/DESCRIPTION.rst b/ddmrp_cron_actions_as_job/readme/DESCRIPTION.rst new file mode 100644 index 000000000..8b98240b8 --- /dev/null +++ b/ddmrp_cron_actions_as_job/readme/DESCRIPTION.rst @@ -0,0 +1,14 @@ +DDMRP Buffer calculations are now run with Queue Jobs. + +When auto-update of NFP is active, each time the state of a stock move changes, +a new computation is triggered, but thanks to identity keys on jobs, only one +job at a time is generated for the same buffer. + +The ``.cron_actions`` method is automatically delayed when the +context contains ``auto_delay_ddmrp_cron_actions=True``. + +The scheduled action for buffers ADU computation also generates jobs instead +of recomputing all the buffers at once. + +The ``._calc_adu`` method is automatically delayed when the +context contains ``auto_delay_ddmrp_calc_adu=True``. diff --git a/ddmrp_cron_actions_as_job/tests/__init__.py b/ddmrp_cron_actions_as_job/tests/__init__.py new file mode 100644 index 000000000..48aece491 --- /dev/null +++ b/ddmrp_cron_actions_as_job/tests/__init__.py @@ -0,0 +1 @@ +from . import test_cron_actions_as_job diff --git a/ddmrp_cron_actions_as_job/tests/test_cron_actions_as_job.py b/ddmrp_cron_actions_as_job/tests/test_cron_actions_as_job.py new file mode 100644 index 000000000..1d0397a8c --- /dev/null +++ b/ddmrp_cron_actions_as_job/tests/test_cron_actions_as_job.py @@ -0,0 +1,56 @@ +# Copyright 2020 Camptocamp (https://www.camptocamp.com) +# License LGPL-3.0 or later (https://www.gnu.org/licenses/lgpl.html). + + +from odoo.tests import tagged + +from odoo.addons.ddmrp.tests.common import TestDdmrpCommon +from odoo.addons.queue_job.job import identity_exact +from odoo.addons.queue_job.tests.common import mock_with_delay + + +@tagged("post_install", "-at_install") +class TestDdmrpCronActionsAsJob(TestDdmrpCommon): + def test_cron_actions_delay_job(self): + context = dict(self.env.context, auto_delay_ddmrp_cron_actions=True) + del context["test_queue_job_no_delay"] + buffer_a = self.buffer_a.with_context(context) + + with mock_with_delay() as (delayable_cls, delayable): + buffer_a.cron_actions(only_nfp=True) + + # check 'with_delay()' part: + self.assertEqual(delayable_cls.call_count, 1) + # arguments passed in 'with_delay()' + delay_args, delay_kwargs = delayable_cls.call_args + self.assertEqual(delay_args, (self.buffer_a,)) + self.assertEqual(delay_kwargs.get("priority"), 15) + self.assertEqual(delay_kwargs.get("identity_key"), identity_exact) + + # check what's passed to the job method 'cron_actions' + self.assertEqual(delayable.cron_actions.call_count, 1) + delay_args, delay_kwargs = delayable.cron_actions.call_args + self.assertEqual(delay_args, ()) + self.assertDictEqual(delay_kwargs, {"only_nfp": True}) + + def test_calc_adu_delay_job(self): + context = dict(self.env.context, auto_delay_ddmrp_calc_adu=True) + del context["test_queue_job_no_delay"] + buffer_a = self.buffer_a.with_context(context) + + with mock_with_delay() as (delayable_cls, delayable): + buffer_a._calc_adu() + + # check 'with_delay()' part: + self.assertEqual(delayable_cls.call_count, 1) + # arguments passed in 'with_delay()' + delay_args, delay_kwargs = delayable_cls.call_args + self.assertEqual(delay_args, (self.buffer_a,)) + self.assertEqual(delay_kwargs.get("priority"), 15) + self.assertEqual(delay_kwargs.get("identity_key"), identity_exact) + + # check what's passed to the job method '_calc_adu' + self.assertEqual(delayable._calc_adu.call_count, 1) + delay_args, delay_kwargs = delayable._calc_adu.call_args + self.assertEqual(delay_args, ()) + self.assertDictEqual(delay_kwargs, {})