diff --git a/queue_job/README.rst b/queue_job/README.rst index 310b9b0e5e..ae1001b44c 100644 --- a/queue_job/README.rst +++ b/queue_job/README.rst @@ -23,7 +23,7 @@ Job Queue :target: https://runbot.odoo-community.org/runbot/230/15.0 :alt: Try me on Runbot -|badge1| |badge2| |badge3| |badge4| |badge5| +|badge1| |badge2| |badge3| |badge4| |badge5| This addon adds an integrated Job Queue to Odoo. @@ -385,7 +385,7 @@ promote its widespread use. Current `maintainer `__: -|maintainer-guewen| +|maintainer-guewen| This module is part of the `OCA/queue `_ project on GitHub. diff --git a/queue_job/__manifest__.py b/queue_job/__manifest__.py index 57084deccc..a26eb828fd 100644 --- a/queue_job/__manifest__.py +++ b/queue_job/__manifest__.py @@ -1,6 +1,5 @@ # License LGPL-3.0 or later (http://www.gnu.org/licenses/lgpl.html) - { "name": "Job Queue", "version": "15.0.1.1.0", @@ -23,6 +22,14 @@ "data/queue_data.xml", "data/queue_job_function_data.xml", ], + "assets": { + "web.assets_backend": [ + "/queue_job/static/lib/vis/vis-network.min.css", + "/queue_job/static/src/scss/queue_job_fields.scss", + "/queue_job/static/lib/vis/vis-network.min.js", + "/queue_job/static/src/js/queue_job_fields.js", + ], + }, "installable": True, "development_status": "Mature", "maintainers": ["guewen"], diff --git a/queue_job/controllers/main.py b/queue_job/controllers/main.py index 33b5778476..555fd3de65 100644 --- a/queue_job/controllers/main.py +++ b/queue_job/controllers/main.py @@ -3,15 +3,18 @@ # License LGPL-3.0 or later (http://www.gnu.org/licenses/lgpl.html) import logging +import random +import time import traceback from io import StringIO -from psycopg2 import OperationalError -from werkzeug.exceptions import Forbidden +from psycopg2 import OperationalError, errorcodes +from werkzeug.exceptions import BadRequest, Forbidden from odoo import SUPERUSER_ID, _, api, http, registry, tools from odoo.service.model import PG_CONCURRENCY_ERRORS_TO_RETRY +from ..delay import chain, group from ..exception import FailedJobError, NothingToDoJob, RetryableJobError from ..job import ENQUEUED, Job @@ -19,6 +22,8 @@ PG_RETRY = 5 # seconds +DEPENDS_MAX_TRIES_ON_CONCURRENCY_FAILURE = 5 + class RunJobController(http.Controller): def _try_perform_job(self, env, job): @@ -35,6 +40,35 @@ def _try_perform_job(self, env, job): env.cr.commit() _logger.debug("%s done", job) + def _enqueue_dependent_jobs(self, env, job): + tries = 0 + while True: + try: + job.enqueue_waiting() + except OperationalError as err: + # Automatically retry the typical transaction serialization + # errors + if err.pgcode not in PG_CONCURRENCY_ERRORS_TO_RETRY: + raise + if tries >= DEPENDS_MAX_TRIES_ON_CONCURRENCY_FAILURE: + _logger.info( + "%s, maximum number of tries reached to update dependencies", + errorcodes.lookup(err.pgcode), + ) + raise + wait_time = random.uniform(0.0, 2**tries) + tries += 1 + _logger.info( + "%s, retry %d/%d in %.04f sec...", + errorcodes.lookup(err.pgcode), + tries, + DEPENDS_MAX_TRIES_ON_CONCURRENCY_FAILURE, + wait_time, + ) + time.sleep(wait_time) + else: + break + @http.route("/queue_job/runjob", type="http", auth="none", save_session=False) def runjob(self, db, job_uuid, **kw): http.request.session.db = db @@ -111,6 +145,10 @@ def retry_postpone(job, message, seconds=None): buff.close() raise + _logger.debug("%s enqueue depends started", job) + self._enqueue_dependent_jobs(env, job) + _logger.debug("%s enqueue depends done", job) + return "" def _get_failure_values(self, job, traceback_txt, orig_exception): @@ -125,13 +163,35 @@ def _get_failure_values(self, job, traceback_txt, orig_exception): "exc_message": exc_message, } + # flake8: noqa: C901 @http.route("/queue_job/create_test_job", type="http", auth="user") def create_test_job( - self, priority=None, max_retries=None, channel=None, description="Test job" + self, + priority=None, + max_retries=None, + channel=None, + description="Test job", + size=1, + failure_rate=0, ): if not http.request.env.user.has_group("base.group_erp_manager"): raise Forbidden(_("Access Denied")) + if failure_rate is not None: + try: + failure_rate = float(failure_rate) + except (ValueError, TypeError): + failure_rate = 0 + + if not (0 <= failure_rate <= 1): + raise BadRequest("failure_rate must be between 0 and 1") + + if size is not None: + try: + size = int(size) + except (ValueError, TypeError): + size = 1 + if priority is not None: try: priority = int(priority) @@ -144,6 +204,35 @@ def create_test_job( except ValueError: max_retries = None + if size == 1: + return self._create_single_test_job( + priority=priority, + max_retries=max_retries, + channel=channel, + description=description, + failure_rate=failure_rate, + ) + + if size > 1: + return self._create_graph_test_jobs( + size, + priority=priority, + max_retries=max_retries, + channel=channel, + description=description, + failure_rate=failure_rate, + ) + return "" + + def _create_single_test_job( + self, + priority=None, + max_retries=None, + channel=None, + description="Test job", + size=1, + failure_rate=0, + ): delayed = ( http.request.env["queue.job"] .with_delay( @@ -152,7 +241,56 @@ def create_test_job( channel=channel, description=description, ) - ._test_job() + ._test_job(failure_rate=failure_rate) ) + return "job uuid: %s" % (delayed.db_record().uuid,) + + TEST_GRAPH_MAX_PER_GROUP = 5 - return delayed.db_record().uuid + def _create_graph_test_jobs( + self, + size, + priority=None, + max_retries=None, + channel=None, + description="Test job", + failure_rate=0, + ): + model = http.request.env["queue.job"] + current_count = 0 + + possible_grouping_methods = (chain, group) + + tails = [] # we can connect new graph chains/groups to tails + root_delayable = None + while current_count < size: + jobs_count = min( + size - current_count, random.randint(1, self.TEST_GRAPH_MAX_PER_GROUP) + ) + + jobs = [] + for __ in range(jobs_count): + current_count += 1 + jobs.append( + model.delayable( + priority=priority, + max_retries=max_retries, + channel=channel, + description="%s #%d" % (description, current_count), + )._test_job(failure_rate=failure_rate) + ) + + grouping = random.choice(possible_grouping_methods) + delayable = grouping(*jobs) + if not root_delayable: + root_delayable = delayable + else: + tail_delayable = random.choice(tails) + tail_delayable.on_done(delayable) + tails.append(delayable) + + root_delayable.delay() + + return "graph uuid: %s" % ( + list(root_delayable._head())[0]._generated_job.graph_uuid, + ) diff --git a/queue_job/delay.py b/queue_job/delay.py new file mode 100644 index 0000000000..f93fb5cc9d --- /dev/null +++ b/queue_job/delay.py @@ -0,0 +1,625 @@ +# Copyright 2019 Camptocamp +# Copyright 2019 Guewen Baconnier +# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl.html) + +import itertools +import logging +import os +import uuid +from collections import defaultdict, deque + +from .job import Job + +_logger = logging.getLogger(__name__) + + +def group(*delayables): + """Return a group of delayable to form a graph + + A group means that jobs can be executed concurrently. + A job or a group of jobs depending on a group can be executed only after + all the jobs of the group are done. + + Shortcut to :class:`~odoo.addons.queue_job.delay.DelayableGroup`. + + Example:: + + g1 = group(delayable1, delayable2) + g2 = group(delayable3, delayable4) + g1.on_done(g2) + g1.delay() + """ + return DelayableGroup(*delayables) + + +def chain(*delayables): + """Return a chain of delayable to form a graph + + A chain means that jobs must be executed sequentially. + A job or a group of jobs depending on a group can be executed only after + the last job of the chain is done. + + Shortcut to :class:`~odoo.addons.queue_job.delay.DelayableChain`. + + Example:: + + chain1 = chain(delayable1, delayable2, delayable3) + chain2 = chain(delayable4, delayable5, delayable6) + chain1.on_done(chain2) + chain1.delay() + """ + return DelayableChain(*delayables) + + +class Graph: + """Acyclic directed graph holding vertices of any hashable type + + This graph is not specifically designed to hold :class:`~Delayable` + instances, although ultimately it is used for this purpose. + """ + + __slots__ = "_graph" + + def __init__(self, graph=None): + if graph: + self._graph = graph + else: + self._graph = {} + + def add_vertex(self, vertex): + """Add a vertex + + Has no effect if called several times with the same vertex + """ + self._graph.setdefault(vertex, set()) + + def add_edge(self, parent, child): + """Add an edge between a parent and a child vertex + + Has no effect if called several times with the same pair of vertices + """ + self.add_vertex(child) + self._graph.setdefault(parent, set()).add(child) + + def vertices(self): + """Return the vertices (nodes) of the graph""" + return set(self._graph) + + def edges(self): + """Return the edges (links) of the graph""" + links = [] + for vertex, neighbours in self._graph.items(): + for neighbour in neighbours: + links.append((vertex, neighbour)) + return links + + # from + # https://codereview.stackexchange.com/questions/55767/finding-all-paths-from-a-given-graph + def paths(self, vertex): + """Generate the maximal cycle-free paths in graph starting at vertex. + + >>> g = {1: [2, 3], 2: [3, 4], 3: [1], 4: []} + >>> sorted(self.paths(1)) + [[1, 2, 3], [1, 2, 4], [1, 3]] + >>> sorted(self.paths(3)) + [[3, 1, 2, 4]] + """ + path = [vertex] # path traversed so far + seen = {vertex} # set of vertices in path + + def search(): + dead_end = True + for neighbour in self._graph[path[-1]]: + if neighbour not in seen: + dead_end = False + seen.add(neighbour) + path.append(neighbour) + yield from search() + path.pop() + seen.remove(neighbour) + if dead_end: + yield list(path) + + yield from search() + + def topological_sort(self): + """Yields a proposed order of nodes to respect dependencies + + The order is not unique, the result may vary, but it is guaranteed + that a node depending on another is not yielded before. + It assumes the graph has no cycle. + """ + depends_per_node = defaultdict(int) + for __, tail in self.edges(): + depends_per_node[tail] += 1 + + # the queue contains only elements for which all dependencies + # are resolved + queue = deque(self.root_vertices()) + while queue: + vertex = queue.popleft() + yield vertex + for node in self._graph[vertex]: + depends_per_node[node] -= 1 + if not depends_per_node[node]: + queue.append(node) + + def root_vertices(self): + """Returns the root vertices + + meaning they do not depend on any other job. + """ + dependency_vertices = set() + for dependencies in self._graph.values(): + dependency_vertices.update(dependencies) + return set(self._graph.keys()) - dependency_vertices + + def __repr__(self): + paths = [path for vertex in self.root_vertices() for path in self.paths(vertex)] + lines = [] + for path in paths: + lines.append(" → ".join(repr(vertex) for vertex in path)) + return "\n".join(lines) + + +class DelayableGraph(Graph): + """Directed Graph for :class:`~Delayable` dependencies + + It connects together the :class:`~Delayable`, :class:`~DelayableGroup` and + :class:`~DelayableChain` graphs, and creates then enqueued the jobs. + """ + + def _merge_graph(self, graph): + """Merge a graph in the current graph + + It takes each vertex, which can be :class:`~Delayable`, + :class:`~DelayableChain` or :class:`~DelayableGroup`, and updates the + current graph with the edges between Delayable objects (connecting + heads and tails of the groups and chains), so that at the end, the + graph contains only Delayable objects and their links. + """ + for vertex, neighbours in graph._graph.items(): + tails = vertex._tail() + for tail in tails: + # connect the tails with the heads of each node + heads = {head for n in neighbours for head in n._head()} + self._graph.setdefault(tail, set()).update(heads) + + def _connect_graphs(self): + """Visit the vertices' graphs and connect them, return the whole graph + + Build a new graph, walk the vertices and their related vertices, merge + their graph in the new one, until we have visited all the vertices + """ + graph = DelayableGraph() + graph._merge_graph(self) + + seen = set() + visit_stack = deque([self]) + while visit_stack: + current = visit_stack.popleft() + if current in seen: + continue + + vertices = current.vertices() + for vertex in vertices: + vertex_graph = vertex._graph + graph._merge_graph(vertex_graph) + visit_stack.append(vertex_graph) + + seen.add(current) + + return graph + + def _has_to_execute_directly(self, vertices): + """Used for tests to run tests directly instead of storing them + + In tests, prefer to use + :func:`odoo.addons.queue_job.tests.common.trap_jobs`. + """ + if os.getenv("TEST_QUEUE_JOB_NO_DELAY"): + _logger.warning( + "`TEST_QUEUE_JOB_NO_DELAY` env var found. NO JOB scheduled." + ) + return True + envs = {vertex.recordset.env for vertex in vertices} + for env in envs: + if env.context.get("test_queue_job_no_delay"): + _logger.warning( + "`test_queue_job_no_delay` ctx key found. NO JOB scheduled." + ) + return True + return False + + @staticmethod + def _ensure_same_graph_uuid(jobs): + """Set the same graph uuid on all jobs of the same graph""" + jobs_count = len(jobs) + if jobs_count == 0: + raise ValueError("Expecting jobs") + elif jobs_count == 1: + if jobs[0].graph_uuid: + raise ValueError( + "Job %s is a single job, it should not" + " have a graph uuid" % (jobs[0],) + ) + else: + graph_uuids = {job.graph_uuid for job in jobs if job.graph_uuid} + if len(graph_uuids) > 1: + raise ValueError("Jobs cannot have dependencies between several graphs") + elif len(graph_uuids) == 1: + graph_uuid = graph_uuids.pop() + else: + graph_uuid = str(uuid.uuid4()) + for job in jobs: + job.graph_uuid = graph_uuid + + def delay(self): + """Build the whole graph, creates jobs and delay them""" + graph = self._connect_graphs() + + vertices = graph.vertices() + + for vertex in vertices: + vertex._build_job() + + self._ensure_same_graph_uuid([vertex._generated_job for vertex in vertices]) + + if self._has_to_execute_directly(vertices): + self._execute_graph_direct(graph) + return + + for vertex, neighbour in graph.edges(): + neighbour._generated_job.add_depends({vertex._generated_job}) + + # If all the jobs of the graph have another job with the same identity, + # we do not create them. Maybe we should check that the found jobs are + # part of the same graph, but not sure it's really required... + # Also, maybe we want to check only the root jobs. + existing_mapping = {} + for vertex in vertices: + if not vertex.identity_key: + continue + generated_job = vertex._generated_job + existing = generated_job.job_record_with_same_identity_key() + if not existing: + # at least one does not exist yet, we'll delay the whole graph + existing_mapping.clear() + break + existing_mapping[vertex] = existing + + # We'll replace the generated jobs by the existing ones, so callers + # can retrieve the existing job in "_generated_job". + # existing_mapping contains something only if *all* the job with an + # identity have an existing one. + for vertex, existing in existing_mapping.items(): + vertex._generated_job = existing + return + + for vertex in vertices: + vertex._generated_job.store() + + def _execute_graph_direct(self, graph): + for delayable in graph.topological_sort(): + delayable._execute_direct() + + +class DelayableChain: + """Chain of delayables to form a graph + + Delayables can be other :class:`~Delayable`, :class:`~DelayableChain` or + :class:`~DelayableGroup` objects. + + A chain means that jobs must be executed sequentially. + A job or a group of jobs depending on a group can be executed only after + the last job of the chain is done. + + Chains can be connected to other Delayable, DelayableChain or + DelayableGroup objects by using :meth:`~done`. + + A Chain is enqueued by calling :meth:`~delay`, which delays the whole + graph. + Important: :meth:`~delay` must be called on the top-level + delayable/chain/group object of the graph. + """ + + __slots__ = ("_graph", "__head", "__tail") + + def __init__(self, *delayables): + self._graph = DelayableGraph() + iter_delayables = iter(delayables) + head = next(iter_delayables) + self.__head = head + self._graph.add_vertex(head) + for neighbour in iter_delayables: + self._graph.add_edge(head, neighbour) + head = neighbour + self.__tail = head + + def _head(self): + return self.__head._tail() + + def _tail(self): + return self.__tail._head() + + def __repr__(self): + inner_graph = "\n\t".join(repr(self._graph).split("\n")) + return "DelayableChain(\n\t{}\n)".format(inner_graph) + + def on_done(self, *delayables): + """Connects the current chain to other delayables/chains/groups + + The delayables/chains/groups passed in the parameters will be executed + when the current Chain is done. + """ + for delayable in delayables: + self._graph.add_edge(self.__tail, delayable) + return self + + def delay(self): + """Delay the whole graph""" + self._graph.delay() + + +class DelayableGroup: + """Group of delayables to form a graph + + Delayables can be other :class:`~Delayable`, :class:`~DelayableChain` or + :class:`~DelayableGroup` objects. + + A group means that jobs must be executed sequentially. + A job or a group of jobs depending on a group can be executed only after + the all the jobs of the group are done. + + Groups can be connected to other Delayable, DelayableChain or + DelayableGroup objects by using :meth:`~done`. + + A group is enqueued by calling :meth:`~delay`, which delays the whole + graph. + Important: :meth:`~delay` must be called on the top-level + delayable/chain/group object of the graph. + """ + + __slots__ = ("_graph", "_delayables") + + def __init__(self, *delayables): + self._graph = DelayableGraph() + self._delayables = set(delayables) + for delayable in delayables: + self._graph.add_vertex(delayable) + + def _head(self): + return itertools.chain.from_iterable(node._head() for node in self._delayables) + + def _tail(self): + return itertools.chain.from_iterable(node._tail() for node in self._delayables) + + def __repr__(self): + inner_graph = "\n\t".join(repr(self._graph).split("\n")) + return "DelayableGroup(\n\t{}\n)".format(inner_graph) + + def on_done(self, *delayables): + """Connects the current group to other delayables/chains/groups + + The delayables/chains/groups passed in the parameters will be executed + when the current Group is done. + """ + for parent in self._delayables: + for child in delayables: + self._graph.add_edge(parent, child) + return self + + def delay(self): + """Delay the whole graph""" + self._graph.delay() + + +class Delayable: + """Unit of a graph, one Delayable will lead to an enqueued job + + Delayables can have dependencies on each others, as well as dependencies on + :class:`~DelayableGroup` or :class:`~DelayableChain` objects. + + This class will generally not be used directly, it is used internally + by :meth:`~odoo.addons.queue_job.models.base.Base.delayable`. Look + in the base model for more details. + + Delayables can be connected to other Delayable, DelayableChain or + DelayableGroup objects by using :meth:`~done`. + + Properties of the future job can be set using the :meth:`~set` method, + which always return ``self``:: + + delayable.set(priority=15).set({"max_retries": 5, "eta": 15}).delay() + + It can be used for example to set properties dynamically. + + A Delayable is enqueued by calling :meth:`delay()`, which delays the whole + graph. + Important: :meth:`delay()` must be called on the top-level + delayable/chain/group object of the graph. + """ + + _properties = ( + "priority", + "eta", + "max_retries", + "description", + "channel", + "identity_key", + ) + __slots__ = _properties + ( + "recordset", + "_graph", + "_job_method", + "_job_args", + "_job_kwargs", + "_generated_job", + ) + + def __init__( + self, + recordset, + priority=None, + eta=None, + max_retries=None, + description=None, + channel=None, + identity_key=None, + ): + self._graph = DelayableGraph() + self._graph.add_vertex(self) + + self.recordset = recordset + + self.priority = priority + self.eta = eta + self.max_retries = max_retries + self.description = description + self.channel = channel + self.identity_key = identity_key + + self._job_method = None + self._job_args = () + self._job_kwargs = {} + + self._generated_job = None + + def _head(self): + return [self] + + def _tail(self): + return [self] + + def __repr__(self): + return "Delayable({}.{}({}, {}))".format( + self.recordset, self._job_method.__name__, self._job_args, self._job_kwargs + ) + + def __del__(self): + if not self._generated_job: + _logger.warning("Delayable %s was prepared but never delayed", self) + + def _set_from_dict(self, properties): + for key, value in properties.items(): + if key not in self._properties: + raise ValueError("No property %s" % (key,)) + setattr(self, key, value) + + def set(self, *args, **kwargs): + """Set job properties and return self + + The values can be either a dictionary and/or keywork args + """ + if args: + # args must be a dict + self._set_from_dict(*args) + self._set_from_dict(kwargs) + return self + + def on_done(self, *delayables): + """Connects the current Delayable to other delayables/chains/groups + + The delayables/chains/groups passed in the parameters will be executed + when the current Delayable is done. + """ + for child in delayables: + self._graph.add_edge(self, child) + return self + + def delay(self): + """Delay the whole graph""" + self._graph.delay() + + def _build_job(self): + if self._generated_job: + return self._generated_job + self._generated_job = Job( + self._job_method, + args=self._job_args, + kwargs=self._job_kwargs, + priority=self.priority, + max_retries=self.max_retries, + eta=self.eta, + description=self.description, + channel=self.channel, + identity_key=self.identity_key, + ) + return self._generated_job + + def _store_args(self, *args, **kwargs): + self._job_args = args + self._job_kwargs = kwargs + return self + + def __getattr__(self, name): + if name in self.__slots__: + return super().__getattr__(name) + if name in self.recordset: + raise AttributeError( + "only methods can be delayed (%s called on %s)" % (name, self.recordset) + ) + recordset_method = getattr(self.recordset, name) + self._job_method = recordset_method + return self._store_args + + def _execute_direct(self): + assert self._generated_job + self._generated_job.perform() + + +class DelayableRecordset(object): + """Allow to delay a method for a recordset (shortcut way) + + Usage:: + + delayable = DelayableRecordset(recordset, priority=20) + delayable.method(args, kwargs) + + The method call will be processed asynchronously in the job queue, with + the passed arguments. + + This class will generally not be used directly, it is used internally + by :meth:`~odoo.addons.queue_job.models.base.Base.with_delay` + """ + + __slots__ = ("delayable",) + + def __init__( + self, + recordset, + priority=None, + eta=None, + max_retries=None, + description=None, + channel=None, + identity_key=None, + ): + self.delayable = Delayable( + recordset, + priority=priority, + eta=eta, + max_retries=max_retries, + description=description, + channel=channel, + identity_key=identity_key, + ) + + @property + def recordset(self): + return self.delayable.recordset + + def __getattr__(self, name): + def _delay_delayable(*args, **kwargs): + getattr(self.delayable, name)(*args, **kwargs).delay() + return self.delayable._generated_job + + return _delay_delayable + + def __str__(self): + return "DelayableRecordset(%s%s)" % ( + self.delayable.recordset._name, + getattr(self.delayable.recordset, "_ids", ""), + ) + + __repr__ = __str__ diff --git a/queue_job/job.py b/queue_job/job.py index 794c7fb030..f824020a27 100644 --- a/queue_job/job.py +++ b/queue_job/job.py @@ -7,13 +7,16 @@ import os import sys import uuid +import weakref from datetime import datetime, timedelta +from functools import total_ordering from random import randint import odoo from .exception import FailedJobError, NoSuchJobError, RetryableJobError +WAIT_DEPENDENCIES = "wait_dependencies" PENDING = "pending" ENQUEUED = "enqueued" CANCELLED = "cancelled" @@ -22,6 +25,7 @@ FAILED = "failed" STATES = [ + (WAIT_DEPENDENCIES, "Wait Dependencies"), (PENDING, "Pending"), (ENQUEUED, "Enqueued"), (STARTED, "Started"), @@ -37,69 +41,17 @@ _logger = logging.getLogger(__name__) -class DelayableRecordset(object): - """Allow to delay a method for a recordset +# TODO remove in 15.0 or 16.0, used to keep compatibility as the +# class has been moved in 'delay'. +def DelayableRecordset(*args, **kwargs): + # prevent circular import + from .delay import DelayableRecordset as dr - Usage:: - - delayable = DelayableRecordset(recordset, priority=20) - delayable.method(args, kwargs) - - The method call will be processed asynchronously in the job queue, with - the passed arguments. - - This class will generally not be used directly, it is used internally - by :meth:`~odoo.addons.queue_job.models.base.Base.with_delay` - """ - - def __init__( - self, - recordset, - priority=None, - eta=None, - max_retries=None, - description=None, - channel=None, - identity_key=None, - ): - self.recordset = recordset - self.priority = priority - self.eta = eta - self.max_retries = max_retries - self.description = description - self.channel = channel - self.identity_key = identity_key - - def __getattr__(self, name): - if name in self.recordset: - raise AttributeError( - "only methods can be delayed ({} called on {})".format( - name, self.recordset - ) - ) - recordset_method = getattr(self.recordset, name) - - def delay(*args, **kwargs): - return Job.enqueue( - recordset_method, - args=args, - kwargs=kwargs, - priority=self.priority, - max_retries=self.max_retries, - eta=self.eta, - description=self.description, - channel=self.channel, - identity_key=self.identity_key, - ) - - return delay - - def __str__(self): - return "DelayableRecordset({}{})".format( - self.recordset._name, getattr(self.recordset, "_ids", "") - ) - - __repr__ = __str__ + _logger.debug( + "DelayableRecordset moved from the queue_job.job" + " to the queue_job.delay python module" + ) + return dr(*args, **kwargs) def identity_exact(job_): @@ -147,6 +99,7 @@ def identity_example(job_): return hasher.hexdigest() +@total_ordering class Job(object): """A Job is a task to execute. It is the in-memory representation of a job. @@ -157,6 +110,10 @@ class Job(object): Id (UUID) of the job. + .. attribute:: graph_uuid + + Shared UUID of the job's graph. Empty if the job is a single job. + .. attribute:: state State of the job, can pending, enqueued, started, done or failed. @@ -256,14 +213,26 @@ class Job(object): @classmethod def load(cls, env, job_uuid): - """Read a job from the Database""" - stored = cls.db_record_from_uuid(env, job_uuid) + """Read a single job from the Database + + Raise an error if the job is not found. + """ + stored = cls.db_records_from_uuids(env, [job_uuid]) if not stored: raise NoSuchJobError( "Job %s does no longer exist in the storage." % job_uuid ) return cls._load_from_db_record(stored) + @classmethod + def load_many(cls, env, job_uuids): + """Read jobs in batch from the Database + + Jobs not found are ignored. + """ + recordset = cls.db_records_from_uuids(env, job_uuids) + return {cls._load_from_db_record(record) for record in recordset} + @classmethod def _load_from_db_record(cls, job_db_record): stored = job_db_record @@ -307,6 +276,7 @@ def _load_from_db_record(cls, job_db_record): job_.date_cancelled = stored.date_cancelled job_.state = stored.state + job_.graph_uuid = stored.graph_uuid if stored.graph_uuid else None job_.result = stored.result if stored.result else None job_.exc_info = stored.exc_info if stored.exc_info else None job_.retry = stored.retry @@ -315,6 +285,11 @@ def _load_from_db_record(cls, job_db_record): job_.company_id = stored.company_id.id job_.identity_key = stored.identity_key job_.worker_pid = stored.worker_pid + + job_.__depends_on_uuids.update(stored.dependencies.get("depends_on", [])) + job_.__reverse_depends_on_uuids.update( + stored.dependencies.get("reverse_depends_on", []) + ) return job_ def job_record_with_same_identity_key(self): @@ -332,6 +307,7 @@ def job_record_with_same_identity_key(self): ) return existing + # TODO to deprecate (not called anymore) @classmethod def enqueue( cls, @@ -365,31 +341,41 @@ def enqueue( channel=channel, identity_key=identity_key, ) - if new_job.identity_key: - existing = new_job.job_record_with_same_identity_key() + return new_job._enqueue_job() + + # TODO to deprecate (not called anymore) + def _enqueue_job(self): + if self.identity_key: + existing = self.job_record_with_same_identity_key() if existing: _logger.debug( "a job has not been enqueued due to having " "the same identity key (%s) than job %s", - new_job.identity_key, + self.identity_key, existing.uuid, ) return Job._load_from_db_record(existing) - new_job.store() + self.store() _logger.debug( "enqueued %s:%s(*%r, **%r) with uuid: %s", - new_job.recordset, - new_job.method_name, - new_job.args, - new_job.kwargs, - new_job.uuid, + self.recordset, + self.method_name, + self.args, + self.kwargs, + self.uuid, ) - return new_job + return self @staticmethod def db_record_from_uuid(env, job_uuid): + # TODO remove in 15.0 or 16.0 + _logger.debug("deprecated, use 'db_records_from_uuids") + return Job.db_records_from_uuids(env, [job_uuid]) + + @staticmethod + def db_records_from_uuids(env, job_uuids): model = env["queue.job"].sudo() - record = model.search([("uuid", "=", job_uuid)], limit=1) + record = model.search([("uuid", "in", tuple(job_uuids))]) return record.with_env(env).sudo() def __init__( @@ -428,8 +414,6 @@ def __init__( :param identity_key: A hash to uniquely identify a job, or a function that returns this hash (the function takes the job as argument) - :param env: Odoo Environment - :type env: :class:`odoo.api.Environment` """ if args is None: args = () @@ -466,10 +450,16 @@ def __init__( self.max_retries = max_retries self._uuid = job_uuid + self.graph_uuid = None self.args = args self.kwargs = kwargs + self.__depends_on_uuids = set() + self.__reverse_depends_on_uuids = set() + self._depends_on = set() + self._reverse_depends_on = weakref.WeakSet() + self.priority = priority if self.priority is None: self.priority = DEFAULT_PRIORITY @@ -506,6 +496,17 @@ def __init__( self.channel = channel self.worker_pid = None + def add_depends(self, jobs): + if self in jobs: + raise ValueError("job cannot depend on itself") + self.__depends_on_uuids |= {j.uuid for j in jobs} + self._depends_on.update(jobs) + for parent in jobs: + parent.__reverse_depends_on_uuids.add(self.uuid) + parent._reverse_depends_on.add(self) + if any(j.state != DONE for j in jobs): + self.state = WAIT_DEPENDENCIES + def perform(self): """Execute the job. @@ -530,8 +531,41 @@ def perform(self): ) raise new_exc from err raise + return self.result + def enqueue_waiting(self): + sql = """ + UPDATE queue_job + SET state = %s + FROM ( + SELECT child.id, array_agg(parent.state) as parent_states + FROM queue_job job + JOIN LATERAL + json_array_elements_text( + job.dependencies::json->'reverse_depends_on' + ) child_deps ON true + JOIN queue_job child + ON child.graph_uuid = job.graph_uuid + AND child.uuid = child_deps + JOIN LATERAL + json_array_elements_text( + child.dependencies::json->'depends_on' + ) parent_deps ON true + JOIN queue_job parent + ON parent.graph_uuid = job.graph_uuid + AND parent.uuid = parent_deps + WHERE job.uuid = %s + GROUP BY child.id + ) jobs + WHERE + queue_job.id = jobs.id + AND %s = ALL(jobs.parent_states) + AND state = %s; + """ + self.env.cr.execute(sql, (PENDING, self.uuid, DONE, WAIT_DEPENDENCIES)) + self.env["queue.job"].invalidate_cache(["state"]) + def store(self): """Store the Job""" job_model = self.env["queue.job"] @@ -568,6 +602,7 @@ def _store_values(self, create=False): "eta": False, "identity_key": False, "worker_pid": self.worker_pid, + "graph_uuid": self.graph_uuid, } if self.date_enqueued: @@ -585,6 +620,14 @@ def _store_values(self, create=False): if self.identity_key: vals["identity_key"] = self.identity_key + dependencies = { + "depends_on": [parent.uuid for parent in self.depends_on], + "reverse_depends_on": [ + children.uuid for children in self.reverse_depends_on + ], + } + vals["dependencies"] = dependencies + if create: vals.update( { @@ -632,8 +675,24 @@ def func_string(self): all_args = ", ".join(args + kwargs) return "{}.{}({})".format(model, self.method_name, all_args) + def __eq__(self, other): + return self.uuid == other.uuid + + def __hash__(self): + return self.uuid.__hash__() + + def sorting_key(self): + return self.eta, self.priority, self.date_created, self.seq + + def __lt__(self, other): + if self.eta and not other.eta: + return True + elif not self.eta and other.eta: + return False + return self.sorting_key() < other.sorting_key() + def db_record(self): - return self.db_record_from_uuid(self.env, self.uuid) + return self.db_records_from_uuids(self.env, [self.uuid]) @property def func(self): @@ -663,6 +722,20 @@ def identity_key(self, value): self._identity_key = None self._identity_key_func = value + @property + def depends_on(self): + if not self._depends_on: + self._depends_on = Job.load_many(self.env, self.__depends_on_uuids) + return self._depends_on + + @property + def reverse_depends_on(self): + if not self._reverse_depends_on: + self._reverse_depends_on = Job.load_many( + self.env, self.__reverse_depends_on_uuids + ) + return set(self._reverse_depends_on) + @property def description(self): if self._description: @@ -717,7 +790,10 @@ def exec_time(self): return None def set_pending(self, result=None, reset_retry=True): - self.state = PENDING + if any(j.state != DONE for j in self.depends_on): + self.state = WAIT_DEPENDENCIES + else: + self.state = PENDING self.date_enqueued = None self.date_started = None self.date_done = None diff --git a/queue_job/jobrunner/channels.py b/queue_job/jobrunner/channels.py index 2d7e0a8be0..75d895156a 100644 --- a/queue_job/jobrunner/channels.py +++ b/queue_job/jobrunner/channels.py @@ -7,9 +7,9 @@ from weakref import WeakValueDictionary from ..exception import ChannelNotFound -from ..job import DONE, ENQUEUED, FAILED, PENDING, STARTED +from ..job import DONE, ENQUEUED, FAILED, PENDING, STARTED, WAIT_DEPENDENCIES -NOT_DONE = (PENDING, ENQUEUED, STARTED, FAILED) +NOT_DONE = (WAIT_DEPENDENCIES, PENDING, ENQUEUED, STARTED, FAILED) _logger = logging.getLogger(__name__) @@ -1054,6 +1054,9 @@ def notify( job.channel.set_running(job) elif state == FAILED: job.channel.set_failed(job) + elif state == WAIT_DEPENDENCIES: + # wait until all parent jobs are done + pass else: _logger.error("unexpected state %s for job %s", state, job) diff --git a/queue_job/models/base.py b/queue_job/models/base.py index a398bf85fb..d218d0d777 100644 --- a/queue_job/models/base.py +++ b/queue_job/models/base.py @@ -3,10 +3,10 @@ import functools import logging -import os from odoo import api, models +from ..delay import Delayable from ..job import DelayableRecordset _logger = logging.getLogger(__name__) @@ -32,23 +32,88 @@ def with_delay( ): """Return a ``DelayableRecordset`` - The returned instance allows to enqueue any method of the recordset's - Model. - - Usage:: + It is a shortcut for the longer form as shown below:: - self.env['res.users'].with_delay().write({'name': 'test'}) + self.with_delay(priority=20).action_done() + # is equivalent to: + self.delayable().set(priority=20).action_done().delay() ``with_delay()`` accepts job properties which specify how the job will be executed. Usage with job properties:: - delayable = env['a.model'].with_delay(priority=30, eta=60*60*5) + env['a.model'].with_delay(priority=30, eta=60*60*5).action_done() delayable.export_one_thing(the_thing_to_export) # => the job will be executed with a low priority and not before a # delay of 5 hours from now + When using :meth:``with_delay``, the final ``delay()`` is implicit. + See the documentation of :meth:``delayable`` for more details. + + :return: instance of a DelayableRecordset + :rtype: :class:`odoo.addons.queue_job.job.DelayableRecordset` + """ + return DelayableRecordset( + self, + priority=priority, + eta=eta, + max_retries=max_retries, + description=description, + channel=channel, + identity_key=identity_key, + ) + + def delayable( + self, + priority=None, + eta=None, + max_retries=None, + description=None, + channel=None, + identity_key=None, + ): + """Return a ``Delayable`` + + The returned instance allows to enqueue any method of the recordset's + Model. + + Usage:: + + delayable = self.env["res.users"].browse(10).delayable(priority=20) + delayable.do_work(name="test"}).delay() + + In this example, the ``do_work`` method will not be executed directly. + It will be executed in an asynchronous job. + + Method calls on a Delayable generally return themselves, so calls can + be chained together:: + + delayable.set(priority=15).do_work(name="test"}).delay() + + The order of the calls that build the job is not relevant, beside + the call to ``delay()`` that must happen at the very end. This is + equivalent to the example above:: + + delayable.do_work(name="test"}).set(priority=15).delay() + + Very importantly, ``delay()`` must be called on the top-most parent + of a chain of jobs, so if you have this:: + + job1 = record1.delayable().do_work() + job2 = record2.delayable().do_work() + job1.on_done(job2) + + The ``delay()`` call must be made on ``job1``, otherwise ``job2`` will + be delayed, but ``job1`` will never be. When done on ``job1``, the + ``delay()`` call will traverse the graph of jobs and delay all of + them:: + + job1.delay() + + For more details on the graph dependencies, read the documentation of + :module:`~odoo.addons.queue_job.delay`. + :param priority: Priority of the job, 0 being the higher priority. Default is 10. :param eta: Estimated Time of Arrival of the job. It will not be @@ -66,30 +131,11 @@ def with_delay( the new job will not be added. It is either a string, either a function that takes the job as argument (see :py:func:`..job.identity_exact`). - :return: instance of a DelayableRecordset - :rtype: :class:`odoo.addons.queue_job.job.DelayableRecordset` - - Note for developers: if you want to run tests or simply disable - jobs queueing for debugging purposes, you can: - - a. set the env var `TEST_QUEUE_JOB_NO_DELAY=1` - b. pass a ctx key `test_queue_job_no_delay=1` - - In tests you'll have to mute the logger like: - - @mute_logger('odoo.addons.queue_job.models.base') + the new job will not be added. + :return: instance of a Delayable + :rtype: :class:`odoo.addons.queue_job.job.Delayable` """ - if os.getenv("TEST_QUEUE_JOB_NO_DELAY"): - _logger.warning( - "`TEST_QUEUE_JOB_NO_DELAY` env var found. NO JOB scheduled." - ) - return self - if self.env.context.get("test_queue_job_no_delay"): - _logger.warning( - "`test_queue_job_no_delay` ctx key found. NO JOB scheduled." - ) - return self - return DelayableRecordset( + return Delayable( self, priority=priority, eta=eta, diff --git a/queue_job/models/queue_job.py b/queue_job/models/queue_job.py index b6f8b52a6e..c276287084 100644 --- a/queue_job/models/queue_job.py +++ b/queue_job/models/queue_job.py @@ -2,13 +2,28 @@ # License LGPL-3.0 or later (http://www.gnu.org/licenses/lgpl.html) import logging +import random from datetime import datetime, timedelta from odoo import _, api, exceptions, fields, models from odoo.osv import expression +from odoo.tools import html_escape +from odoo.addons.base_sparse_field.models.fields import Serialized + +from ..delay import Graph +from ..exception import JobError from ..fields import JobSerialized -from ..job import CANCELLED, DONE, PENDING, STATES, Job +from ..job import ( + CANCELLED, + DONE, + FAILED, + PENDING, + STARTED, + STATES, + WAIT_DEPENDENCIES, + Job, +) _logger = logging.getLogger(__name__) @@ -46,6 +61,12 @@ class QueueJob(models.Model): ) uuid = fields.Char(string="UUID", readonly=True, index=True, required=True) + graph_uuid = fields.Char( + string="Graph UUID", + readonly=True, + index=True, + help="Single shared identifier of a Graph. Empty for a single job.", + ) user_id = fields.Many2one(comodel_name="res.users", string="User ID") company_id = fields.Many2one( comodel_name="res.company", string="Company", index=True @@ -62,6 +83,10 @@ class QueueJob(models.Model): readonly=True, base_type=models.BaseModel, ) + dependencies = Serialized(readonly=True) + # dependency graph as expected by the field widget + dependency_graph = Serialized(compute="_compute_dependency_graph") + graph_jobs_count = fields.Integer(compute="_compute_graph_jobs_count") args = JobSerialized(readonly=True, base_type=tuple) kwargs = JobSerialized(readonly=True, base_type=dict) func_string = fields.Char(string="Task", readonly=True) @@ -93,7 +118,7 @@ class QueueJob(models.Model): "Retries are infinite when empty.", ) # FIXME the name of this field is very confusing - channel_method_name = fields.Char(readonly=True) + channel_method_name = fields.Char(string="Complete Method Name", readonly=True) job_function_id = fields.Many2one( comodel_name="queue.job.function", string="Job Function", @@ -122,6 +147,97 @@ def _compute_record_ids(self): for record in self: record.record_ids = record.records.ids + @api.depends("dependencies") + def _compute_dependency_graph(self): + jobs_groups = self.env["queue.job"].read_group( + [ + ( + "graph_uuid", + "in", + [uuid for uuid in self.mapped("graph_uuid") if uuid], + ) + ], + ["graph_uuid", "ids:array_agg(id)"], + ["graph_uuid"], + ) + ids_per_graph_uuid = { + group["graph_uuid"]: group["ids"] for group in jobs_groups + } + for record in self: + if not record.graph_uuid: + record.dependency_graph = {} + continue + + graph_jobs = self.browse(ids_per_graph_uuid.get(record.graph_uuid) or []) + if not graph_jobs: + record.dependency_graph = {} + continue + + graph_ids = {graph_job.uuid: graph_job.id for graph_job in graph_jobs} + graph_jobs_by_ids = {graph_job.id: graph_job for graph_job in graph_jobs} + + graph = Graph() + for graph_job in graph_jobs: + graph.add_vertex(graph_job.id) + for parent_uuid in graph_job.dependencies["depends_on"]: + parent_id = graph_ids.get(parent_uuid) + if not parent_id: + continue + graph.add_edge(parent_id, graph_job.id) + for child_uuid in graph_job.dependencies["reverse_depends_on"]: + child_id = graph_ids.get(child_uuid) + if not child_id: + continue + graph.add_edge(graph_job.id, child_id) + + record.dependency_graph = { + # list of ids + "nodes": [ + graph_jobs_by_ids[graph_id]._dependency_graph_vis_node() + for graph_id in graph.vertices() + ], + # list of tuples (from, to) + "edges": graph.edges(), + } + + def _dependency_graph_vis_node(self): + """Return the node as expected by the JobDirectedGraph widget""" + default = ("#D2E5FF", "#2B7CE9") + colors = { + DONE: ("#C2FABC", "#4AD63A"), + FAILED: ("#FB7E81", "#FA0A10"), + STARTED: ("#FFFF00", "#FFA500"), + } + return { + "id": self.id, + "title": "%s
%s" + % ( + html_escape(self.display_name), + html_escape(self.func_string), + ), + "color": colors.get(self.state, default)[0], + "border": colors.get(self.state, default)[1], + "shadow": True, + } + + def _compute_graph_jobs_count(self): + jobs_groups = self.env["queue.job"].read_group( + [ + ( + "graph_uuid", + "in", + [uuid for uuid in self.mapped("graph_uuid") if uuid], + ) + ], + ["graph_uuid"], + ["graph_uuid"], + ) + count_per_graph_uuid = { + group["graph_uuid"]: group["graph_uuid_count"] for group in jobs_groups + } + for record in self: + record.graph_jobs_count = count_per_graph_uuid.get(record.graph_uuid) or 0 + @api.model_create_multi def create(self, vals_list): if self.env.context.get("_job_edit_sentinel") is not self.EDIT_SENTINEL: @@ -176,6 +292,22 @@ def open_related_action(self): raise exceptions.UserError(_("No action available for this job")) return action + def open_graph_jobs(self): + """Return action that opens all jobs of the same graph""" + self.ensure_one() + jobs = self.env["queue.job"].search([("graph_uuid", "=", self.graph_uuid)]) + + action_jobs = self.env.ref("queue_job.action_queue_job") + action = action_jobs.read()[0] + action.update( + { + "name": _("Jobs for graph %s") % (self.graph_uuid), + "context": {}, + "domain": [("id", "in", jobs.ids)], + } + ) + return action + def _change_job_state(self, state, result=None): """Change the state of the `Job` object @@ -186,13 +318,17 @@ def _change_job_state(self, state, result=None): job_ = Job.load(record.env, record.uuid) if state == DONE: job_.set_done(result=result) + job_.store() + record.env["queue.job"].flush() + job_.enqueue_waiting() elif state == PENDING: job_.set_pending(result=result) + job_.store() elif state == CANCELLED: job_.set_cancelled(result=result) + job_.store() else: raise ValueError("State not supported: %s" % state) - job_.store() def button_done(self): result = _("Manually set to done by %s") % self.env.user.name @@ -205,7 +341,8 @@ def button_cancelled(self): return True def requeue(self): - self._change_job_state(PENDING) + jobs_to_requeue = self.filtered(lambda job_: job_.state != WAIT_DEPENDENCIES) + jobs_to_requeue._change_job_state(PENDING) return True def _message_post_on_failure(self): @@ -358,5 +495,7 @@ def related_action_open_record(self): ) return action - def _test_job(self): + def _test_job(self, failure_rate=0): _logger.info("Running test job.") + if random.random() <= failure_rate: + raise JobError("Job failed") diff --git a/queue_job/readme/USAGE.rst b/queue_job/readme/USAGE.rst index b595a9c595..f2b5fbc535 100644 --- a/queue_job/readme/USAGE.rst +++ b/queue_job/readme/USAGE.rst @@ -5,7 +5,124 @@ To use this module, you need to: Developers ~~~~~~~~~~ -**Configure default options for jobs** +Delaying jobs +------------- + +The fast way to enqueue a job for a method is to use ``with_delay()`` on a record +or model: + + +.. code-block:: python + + def button_done(self): + self.with_delay().print_confirmation_document(self.state) + self.write({"state": "done"}) + return True + +Here, the method ``print_confirmation_document()`` will be executed asynchronously +as a job. ``with_delay()`` can take several parameters to define more precisely how +the job is executed (priority, ...). + +All the arguments passed to the method being delayed are stored in the job and +passed to the method when it is executed asynchronously, including ``self``, so +the current record is maintained during the job execution (warning: the context +is not kept). + +Dependencies can be expressed between jobs. To start a graph of jobs, use ``delayable()`` +on a record or model. The following is the equivalent of ``with_delay()`` but using the +long form: + +.. code-block:: python + + def button_done(self): + delayable = self.delayable() + delayable.print_confirmation_document(self.state) + delayable.delay() + self.write({"state": "done"}) + return True + +Methods of Delayable objects return itself, so it can be used as a builder pattern, +which in some cases allow to build the jobs dynamically: + +.. code-block:: python + + def button_generate_simple_with_delayable(self): + self.ensure_one() + # Introduction of a delayable object, using a builder pattern + # allowing to chain jobs or set properties. The delay() method + # on the delayable object actually stores the delayable objects + # in the queue_job table + ( + self.delayable() + .generate_thumbnail((50, 50)) + .set(priority=30) + .set(description=_("generate xxx")) + .delay() + ) + +The simplest way to define a dependency is to use ``.on_done(job)`` on a Delayable: + +.. code-block:: python + + def button_chain_done(self): + self.ensure_one() + job1 = self.browse(1).delayable().generate_thumbnail((50, 50)) + job2 = self.browse(1).delayable().generate_thumbnail((50, 50)) + job3 = self.browse(1).delayable().generate_thumbnail((50, 50)) + # job 3 is executed when job 2 is done which is executed when job 1 is done + job1.on_done(job2.on_done(job3)).delay() + +Delayables can be chained to form more complex graphs using the ``chain()`` and +``group()`` primitives. +A chain represents a sequence of jobs to execute in order, a group represents +jobs which can be executed in parallel. Using ``chain()`` has the same effect as +using several nested ``on_done()`` but is more readable. Both can be combined to +form a graph, for instance we can group [A] of jobs, which blocks another group +[B] of jobs. When and only when all the jobs of the group [A] are executed, the +jobs of the group [B] are executed. The code would look like: + +.. code-block:: python + + from odoo.addons.queue_job.delay import group, chain + + def button_done(self): + group_a = group(self.delayable().method_foo(), self.delayable().method_bar()) + group_b = group(self.delayable().method_baz(1), self.delayable().method_baz(2)) + chain(group_a, group_b).delay() + self.write({"state": "done"}) + return True + +When a failure happens in a graph of jobs, the execution of the jobs that depend on the +failed job stops. They remain in a state ``wait_dependencies`` until their "parent" job is +successful. This can happen in two ways: either the parent job retries and is successful +on a second try, either the parent job is manually "set to done" by a user. In these two +cases, the dependency is resolved and the graph will continue to be processed. Alternatively, +the failed job and all its dependent jobs can be canceled by a user. The other jobs of the +graph that do not depend on the failed job continue their execution in any case. + +Note: ``delay()`` must be called on the delayable, chain, or group which is at the top +of the graph. In the example above, if it was called on ``group_a``, then ``group_b`` +would never be delayed (but a warning would be shown). + + +Enqueing Job Options +-------------------- + +* priority: default is 10, the closest it is to 0, the faster it will be + executed +* eta: Estimated Time of Arrival of the job. It will not be executed before this + date/time +* max_retries: default is 5, maximum number of retries before giving up and set + the job state to 'failed'. A value of 0 means infinite retries. +* description: human description of the job. If not set, description is computed + from the function doc or method name +* channel: the complete name of the channel to use to process the function. If + specified it overrides the one defined on the function +* identity_key: key uniquely identifying the job, if specified and a job with + the same key has not yet been run, the new job will not be created + +Configure default options for jobs +---------------------------------- In earlier versions, jobs could be configured using the ``@job`` decorator. This is now obsolete, they can be configured using optional ``queue.job.function`` @@ -152,3 +269,142 @@ Tip: you can do this at test case level like this Then all your tests execute the job methods synchronously without delaying any jobs. + +Testing +------- + +**Asserting enqueued jobs** + +The recommended way to test jobs, rather than running them directly and synchronously is to +split the tests in two parts: + + * one test where the job is mocked (trap jobs with ``trap_jobs()`` and the test + only verifies that the job has been delayed with the expected arguments + * one test that only calls the method of the job synchronously, to validate the + proper behavior of this method only + +Proceeding this way means that you can prove that jobs will be enqueued properly +at runtime, and it ensures your code does not have a different behavior in tests +and in production (because running your jobs synchronously may have a different +behavior as they are in the same transaction / in the middle of the method). +Additionally, it gives more control on the arguments you want to pass when +calling the job's method (synchronously, this time, in the second type of +tests), and it makes tests smaller. + +The best way to run such assertions on the enqueued jobs is to use +``odoo.addons.queue_job.tests.common.trap_jobs()``. + +A very small example (more details in ``tests/common.py``): + +.. code-block:: python + + # code + def my_job_method(self, name, count): + self.write({"name": " ".join([name] * count) + + def method_to_test(self): + count = self.env["other.model"].search_count([]) + self.with_delay(priority=15).my_job_method("Hi!", count=count) + return count + + # tests + from odoo.addons.queue_job.tests.common import trap_jobs + + # first test only check the expected behavior of the method and the proper + # enqueuing of jobs + def test_method_to_test(self): + with trap_jobs() as trap: + result = self.env["model"].method_to_test() + expected_count = 12 + + trap.assert_jobs_count(1, only=self.env["model"].my_job_method) + trap.assert_enqueued_job( + self.env["model"].my_job_method, + args=("Hi!",), + kwargs=dict(count=expected_count), + properties=dict(priority=15) + ) + self.assertEqual(result, expected_count) + + + # second test to validate the behavior of the job unitarily + def test_my_job_method(self): + record = self.env["model"].browse(1) + record.my_job_method("Hi!", count=12) + self.assertEqual(record.name, "Hi! Hi! Hi! Hi! Hi! Hi! Hi! Hi! Hi! Hi! Hi! Hi!") + +If you prefer, you can still test the whole thing in a single test, by calling +``jobs_tester.perform_enqueued_jobs()`` in your test. + +.. code-block:: python + + def test_method_to_test(self): + with trap_jobs() as trap: + result = self.env["model"].method_to_test() + expected_count = 12 + + trap.assert_jobs_count(1, only=self.env["model"].my_job_method) + trap.assert_enqueued_job( + self.env["model"].my_job_method, + args=("Hi!",), + kwargs=dict(count=expected_count), + properties=dict(priority=15) + ) + self.assertEqual(result, expected_count) + + trap.perform_enqueued_jobs() + + record = self.env["model"].browse(1) + record.my_job_method("Hi!", count=12) + self.assertEqual(record.name, "Hi! Hi! Hi! Hi! Hi! Hi! Hi! Hi! Hi! Hi! Hi! Hi!") + +**Execute jobs synchronously when running Odoo** + +When you are developing (ie: connector modules) you might want +to bypass the queue job and run your code immediately. + +To do so you can set ``TEST_QUEUE_JOB_NO_DELAY=1`` in your environment. + +.. WARNING:: Do not do this in production + +**Execute jobs synchronously in tests** + +You should use ``trap_jobs``, really, but if for any reason you could not use it, +and still need to have job methods executed synchronously in your tests, you can +do so by setting ``test_queue_job_no_delay=True`` in the context. + +Tip: you can do this at test case level like this + +.. code-block:: python + + @classmethod + def setUpClass(cls): + super().setUpClass() + cls.env = cls.env(context=dict( + cls.env.context, + test_queue_job_no_delay=True, # no jobs thanks + )) + +Then all your tests execute the job methods synchronously without delaying any +jobs. + +In tests you'll have to mute the logger like: + + @mute_logger('odoo.addons.queue_job.models.base') + +.. NOTE:: in graphs of jobs, the ``test_queue_job_no_delay`` context key must be in at + least one job's env of the graph for the whole graph to be executed synchronously + + +Tips and tricks +--------------- + +* **Idempotency** (https://www.restapitutorial.com/lessons/idempotency.html): The queue_job should be idempotent so they can be retried several times without impact on the data. +* **The job should test at the very beginning its relevance**: the moment the job will be executed is unknown by design. So the first task of a job should be to check if the related work is still relevant at the moment of the execution. + +Patterns +-------- +Through the time, two main patterns emerged: + +1. For data exposed to users, a model should store the data and the model should be the creator of the job. The job is kept hidden from the users +2. For technical data, that are not exposed to the users, it is generally alright to create directly jobs with data passed as arguments to the job, without intermediary models. diff --git a/queue_job/static/lib/vis/vis-network.min.css b/queue_job/static/lib/vis/vis-network.min.css new file mode 100644 index 0000000000..d708f173b6 --- /dev/null +++ b/queue_job/static/lib/vis/vis-network.min.css @@ -0,0 +1 @@ +.vis-overlay{position:absolute;top:0;right:0;bottom:0;left:0;z-index:10}.vis-active{box-shadow:0 0 10px #86d5f8}.vis [class*=span]{min-height:0;width:auto}div.vis-color-picker{position:absolute;top:0;left:30px;margin-top:-140px;margin-left:30px;width:310px;height:444px;z-index:1;padding:10px;border-radius:15px;background-color:#fff;display:none;box-shadow:0 0 10px 0 rgba(0,0,0,.5)}div.vis-color-picker div.vis-arrow{position:absolute;top:147px;left:5px}div.vis-color-picker div.vis-arrow:after,div.vis-color-picker div.vis-arrow:before{right:100%;top:50%;border:solid transparent;content:" ";height:0;width:0;position:absolute;pointer-events:none}div.vis-color-picker div.vis-arrow:after{border-color:hsla(0,0%,100%,0) #fff hsla(0,0%,100%,0) hsla(0,0%,100%,0);border-width:30px;margin-top:-30px}div.vis-color-picker div.vis-color{position:absolute;width:289px;height:289px;cursor:pointer}div.vis-color-picker div.vis-brightness{position:absolute;top:313px}div.vis-color-picker div.vis-opacity{position:absolute;top:350px}div.vis-color-picker div.vis-selector{position:absolute;top:137px;left:137px;width:15px;height:15px;border-radius:15px;border:1px solid #fff;background:#4c4c4c;background:-moz-linear-gradient(top,#4c4c4c 0,#595959 12%,#666 25%,#474747 39%,#2c2c2c 50%,#000 51%,#111 60%,#2b2b2b 76%,#1c1c1c 91%,#131313 100%);background:-webkit-gradient(linear,left top,left bottom,color-stop(0,#4c4c4c),color-stop(12%,#595959),color-stop(25%,#666),color-stop(39%,#474747),color-stop(50%,#2c2c2c),color-stop(51%,#000),color-stop(60%,#111),color-stop(76%,#2b2b2b),color-stop(91%,#1c1c1c),color-stop(100%,#131313));background:-webkit-linear-gradient(top,#4c4c4c,#595959 12%,#666 25%,#474747 39%,#2c2c2c 50%,#000 51%,#111 60%,#2b2b2b 76%,#1c1c1c 91%,#131313);background:-o-linear-gradient(top,#4c4c4c 0,#595959 12%,#666 25%,#474747 39%,#2c2c2c 50%,#000 51%,#111 60%,#2b2b2b 76%,#1c1c1c 91%,#131313 100%);background:-ms-linear-gradient(top,#4c4c4c 0,#595959 12%,#666 25%,#474747 39%,#2c2c2c 50%,#000 51%,#111 60%,#2b2b2b 76%,#1c1c1c 91%,#131313 100%);background:linear-gradient(180deg,#4c4c4c 0,#595959 12%,#666 25%,#474747 39%,#2c2c2c 50%,#000 51%,#111 60%,#2b2b2b 76%,#1c1c1c 91%,#131313);filter:progid:DXImageTransform.Microsoft.gradient(startColorstr="#4c4c4c",endColorstr="#131313",GradientType=0)}div.vis-color-picker div.vis-new-color{left:159px;text-align:right;padding-right:2px}div.vis-color-picker div.vis-initial-color,div.vis-color-picker div.vis-new-color{position:absolute;width:140px;height:20px;border:1px solid rgba(0,0,0,.1);border-radius:5px;top:380px;font-size:10px;color:rgba(0,0,0,.4);vertical-align:middle;line-height:20px}div.vis-color-picker div.vis-initial-color{left:10px;text-align:left;padding-left:2px}div.vis-color-picker div.vis-label{position:absolute;width:300px;left:10px}div.vis-color-picker div.vis-label.vis-brightness{top:300px}div.vis-color-picker div.vis-label.vis-opacity{top:338px}div.vis-color-picker div.vis-button{position:absolute;width:68px;height:25px;border-radius:10px;vertical-align:middle;text-align:center;line-height:25px;top:410px;border:2px solid #d9d9d9;background-color:#f7f7f7;cursor:pointer}div.vis-color-picker div.vis-button.vis-cancel{left:5px}div.vis-color-picker div.vis-button.vis-load{left:82px}div.vis-color-picker div.vis-button.vis-apply{left:159px}div.vis-color-picker div.vis-button.vis-save{left:236px}div.vis-color-picker input.vis-range{width:290px;height:20px}div.vis-configuration{position:relative;display:block;float:left;font-size:12px}div.vis-configuration-wrapper{display:block;width:700px}div.vis-configuration-wrapper:after{clear:both;content:"";display:block}div.vis-configuration.vis-config-option-container{display:block;width:495px;background-color:#fff;border:2px solid #f7f8fa;border-radius:4px;margin-top:20px;left:10px;padding-left:5px}div.vis-configuration.vis-config-button{display:block;width:495px;height:25px;vertical-align:middle;line-height:25px;background-color:#f7f8fa;border:2px solid #ceced0;border-radius:4px;margin-top:20px;left:10px;padding-left:5px;cursor:pointer;margin-bottom:30px}div.vis-configuration.vis-config-button.hover{background-color:#4588e6;border:2px solid #214373;color:#fff}div.vis-configuration.vis-config-item{display:block;float:left;width:495px;height:25px;vertical-align:middle;line-height:25px}div.vis-configuration.vis-config-item.vis-config-s2{left:10px;background-color:#f7f8fa;padding-left:5px;border-radius:3px}div.vis-configuration.vis-config-item.vis-config-s3{left:20px;background-color:#e4e9f0;padding-left:5px;border-radius:3px}div.vis-configuration.vis-config-item.vis-config-s4{left:30px;background-color:#cfd8e6;padding-left:5px;border-radius:3px}div.vis-configuration.vis-config-header{font-size:18px;font-weight:700}div.vis-configuration.vis-config-label{width:120px;height:25px;line-height:25px}div.vis-configuration.vis-config-label.vis-config-s3{width:110px}div.vis-configuration.vis-config-label.vis-config-s4{width:100px}div.vis-configuration.vis-config-colorBlock{top:1px;width:30px;height:19px;border:1px solid #444;border-radius:2px;padding:0;margin:0;cursor:pointer}input.vis-configuration.vis-config-checkbox{left:-5px}input.vis-configuration.vis-config-rangeinput{position:relative;top:-5px;width:60px;padding:1px;margin:0;pointer-events:none}input.vis-configuration.vis-config-range{-webkit-appearance:none;border:0 solid #fff;background-color:transparent;width:300px;height:20px}input.vis-configuration.vis-config-range::-webkit-slider-runnable-track{width:300px;height:5px;background:#dedede;background:-moz-linear-gradient(top,#dedede 0,#c8c8c8 99%);background:-webkit-gradient(linear,left top,left bottom,color-stop(0,#dedede),color-stop(99%,#c8c8c8));background:-webkit-linear-gradient(top,#dedede,#c8c8c8 99%);background:-o-linear-gradient(top,#dedede 0,#c8c8c8 99%);background:-ms-linear-gradient(top,#dedede 0,#c8c8c8 99%);background:linear-gradient(180deg,#dedede 0,#c8c8c8 99%);filter:progid:DXImageTransform.Microsoft.gradient(startColorstr="#dedede",endColorstr="#c8c8c8",GradientType=0);border:1px solid #999;box-shadow:0 0 3px 0 #aaa;border-radius:3px}input.vis-configuration.vis-config-range::-webkit-slider-thumb{-webkit-appearance:none;border:1px solid #14334b;height:17px;width:17px;border-radius:50%;background:#3876c2;background:-moz-linear-gradient(top,#3876c2 0,#385380 100%);background:-webkit-gradient(linear,left top,left bottom,color-stop(0,#3876c2),color-stop(100%,#385380));background:-webkit-linear-gradient(top,#3876c2,#385380);background:-o-linear-gradient(top,#3876c2 0,#385380 100%);background:-ms-linear-gradient(top,#3876c2 0,#385380 100%);background:linear-gradient(180deg,#3876c2 0,#385380);filter:progid:DXImageTransform.Microsoft.gradient(startColorstr="#3876c2",endColorstr="#385380",GradientType=0);box-shadow:0 0 1px 0 #111927;margin-top:-7px}input.vis-configuration.vis-config-range:focus{outline:none}input.vis-configuration.vis-config-range:focus::-webkit-slider-runnable-track{background:#9d9d9d;background:-moz-linear-gradient(top,#9d9d9d 0,#c8c8c8 99%);background:-webkit-gradient(linear,left top,left bottom,color-stop(0,#9d9d9d),color-stop(99%,#c8c8c8));background:-webkit-linear-gradient(top,#9d9d9d,#c8c8c8 99%);background:-o-linear-gradient(top,#9d9d9d 0,#c8c8c8 99%);background:-ms-linear-gradient(top,#9d9d9d 0,#c8c8c8 99%);background:linear-gradient(180deg,#9d9d9d 0,#c8c8c8 99%);filter:progid:DXImageTransform.Microsoft.gradient(startColorstr="#9d9d9d",endColorstr="#c8c8c8",GradientType=0)}input.vis-configuration.vis-config-range::-moz-range-track{width:300px;height:10px;background:#dedede;background:-moz-linear-gradient(top,#dedede 0,#c8c8c8 99%);background:-webkit-gradient(linear,left top,left bottom,color-stop(0,#dedede),color-stop(99%,#c8c8c8));background:-webkit-linear-gradient(top,#dedede,#c8c8c8 99%);background:-o-linear-gradient(top,#dedede 0,#c8c8c8 99%);background:-ms-linear-gradient(top,#dedede 0,#c8c8c8 99%);background:linear-gradient(180deg,#dedede 0,#c8c8c8 99%);filter:progid:DXImageTransform.Microsoft.gradient(startColorstr="#dedede",endColorstr="#c8c8c8",GradientType=0);border:1px solid #999;box-shadow:0 0 3px 0 #aaa;border-radius:3px}input.vis-configuration.vis-config-range::-moz-range-thumb{border:none;height:16px;width:16px;border-radius:50%;background:#385380}input.vis-configuration.vis-config-range:-moz-focusring{outline:1px solid #fff;outline-offset:-1px}input.vis-configuration.vis-config-range::-ms-track{width:300px;height:5px;background:transparent;border-color:transparent;border-width:6px 0;color:transparent}input.vis-configuration.vis-config-range::-ms-fill-lower{background:#777;border-radius:10px}input.vis-configuration.vis-config-range::-ms-fill-upper{background:#ddd;border-radius:10px}input.vis-configuration.vis-config-range::-ms-thumb{border:none;height:16px;width:16px;border-radius:50%;background:#385380}input.vis-configuration.vis-config-range:focus::-ms-fill-lower{background:#888}input.vis-configuration.vis-config-range:focus::-ms-fill-upper{background:#ccc}.vis-configuration-popup{position:absolute;background:rgba(57,76,89,.85);border:2px solid #f2faff;line-height:30px;height:30px;width:150px;text-align:center;color:#fff;font-size:14px;border-radius:4px;-webkit-transition:opacity .3s ease-in-out;-moz-transition:opacity .3s ease-in-out;transition:opacity .3s ease-in-out}.vis-configuration-popup:after,.vis-configuration-popup:before{left:100%;top:50%;border:solid transparent;content:" ";height:0;width:0;position:absolute;pointer-events:none}.vis-configuration-popup:after{border-color:rgba(136,183,213,0) rgba(136,183,213,0) rgba(136,183,213,0) rgba(57,76,89,.85);border-width:8px;margin-top:-8px}.vis-configuration-popup:before{border-color:rgba(194,225,245,0) rgba(194,225,245,0) rgba(194,225,245,0) #f2faff;border-width:12px;margin-top:-12px}div.vis-tooltip{position:absolute;visibility:hidden;padding:5px;white-space:nowrap;font-family:verdana;font-size:14px;color:#000;background-color:#f5f4ed;-moz-border-radius:3px;-webkit-border-radius:3px;border-radius:3px;border:1px solid #808074;box-shadow:3px 3px 10px rgba(0,0,0,.2);pointer-events:none;z-index:5}div.vis-network div.vis-navigation div.vis-button{width:34px;height:34px;-moz-border-radius:17px;border-radius:17px;position:absolute;display:inline-block;background-position:2px 2px;background-repeat:no-repeat;cursor:pointer;-webkit-touch-callout:none;-webkit-user-select:none;-khtml-user-select:none;-moz-user-select:none;-ms-user-select:none;user-select:none}div.vis-network div.vis-navigation div.vis-button:hover{box-shadow:0 0 3px 3px rgba(56,207,21,.3)}div.vis-network div.vis-navigation div.vis-button:active{box-shadow:0 0 1px 3px rgba(56,207,21,.95)}div.vis-network div.vis-navigation div.vis-button.vis-up{background-image:url("");bottom:50px;left:55px}div.vis-network div.vis-navigation div.vis-button.vis-down{background-image:url("");bottom:10px;left:55px}div.vis-network div.vis-navigation div.vis-button.vis-left{background-image:url("");bottom:10px;left:15px}div.vis-network div.vis-navigation div.vis-button.vis-right{background-image:url("");bottom:10px;left:95px}div.vis-network div.vis-navigation div.vis-button.vis-zoomIn{background-image:url("");bottom:10px;right:15px}div.vis-network div.vis-navigation div.vis-button.vis-zoomOut{background-image:url("");bottom:10px;right:55px}div.vis-network div.vis-navigation div.vis-button.vis-zoomExtends{background-image:url("");bottom:50px;right:15px}div.vis-network div.vis-manipulation{box-sizing:content-box;border:0 solid #d6d9d8;border-bottom:1px;background:#fff;background:-moz-linear-gradient(top,#fff 0,#fcfcfc 48%,#fafafa 50%,#fcfcfc 100%);background:-webkit-gradient(linear,left top,left bottom,color-stop(0,#fff),color-stop(48%,#fcfcfc),color-stop(50%,#fafafa),color-stop(100%,#fcfcfc));background:-webkit-linear-gradient(top,#fff,#fcfcfc 48%,#fafafa 50%,#fcfcfc);background:-o-linear-gradient(top,#fff 0,#fcfcfc 48%,#fafafa 50%,#fcfcfc 100%);background:-ms-linear-gradient(top,#fff 0,#fcfcfc 48%,#fafafa 50%,#fcfcfc 100%);background:linear-gradient(180deg,#fff 0,#fcfcfc 48%,#fafafa 50%,#fcfcfc);filter:progid:DXImageTransform.Microsoft.gradient(startColorstr="#ffffff",endColorstr="#fcfcfc",GradientType=0);padding-top:4px;position:absolute;left:0;top:0;width:100%;height:28px}div.vis-network button.vis-edit-mode,div.vis-network div.vis-edit-mode{position:absolute;left:0;top:5px;height:30px}div.vis-network button.vis-close{position:absolute;right:0;top:0;width:30px;height:30px;background-color:transparent;background-position:20px 3px;background-repeat:no-repeat;background-image:url("");border:none;cursor:pointer;-webkit-touch-callout:none;-webkit-user-select:none;-khtml-user-select:none;-moz-user-select:none;-ms-user-select:none;user-select:none}div.vis-network button.vis-close:hover{opacity:.6}div.vis-network div.vis-edit-mode button.vis-button,div.vis-network div.vis-manipulation button.vis-button{float:left;font-family:verdana;font-size:12px;border:none;box-sizing:content-box;-moz-border-radius:15px;border-radius:15px;background-color:transparent;background-position:0 0;background-repeat:no-repeat;height:24px;margin-left:10px;cursor:pointer;padding:0 8px;-webkit-touch-callout:none;-webkit-user-select:none;-khtml-user-select:none;-moz-user-select:none;-ms-user-select:none;user-select:none}div.vis-network div.vis-manipulation button.vis-button:hover{box-shadow:1px 1px 8px rgba(0,0,0,.2)}div.vis-network div.vis-manipulation button.vis-button:active{box-shadow:1px 1px 8px rgba(0,0,0,.5)}div.vis-network div.vis-manipulation button.vis-button.vis-back{background-image:url("")}div.vis-network div.vis-manipulation div.vis-none:hover{box-shadow:1px 1px 8px transparent;cursor:default}div.vis-network div.vis-manipulation div.vis-none:active{box-shadow:1px 1px 8px transparent}div.vis-network div.vis-manipulation div.vis-none{padding:0;line-height:23px}div.vis-network div.vis-manipulation div.notification{margin:2px;font-weight:700}div.vis-network div.vis-manipulation button.vis-button.vis-add{background-image:url("")}div.vis-network div.vis-edit-mode button.vis-button.vis-edit,div.vis-network div.vis-manipulation button.vis-button.vis-edit{background-image:url("")}div.vis-network div.vis-edit-mode button.vis-button.vis-edit.vis-edit-mode{background-color:#fcfcfc;border:1px solid #ccc}div.vis-network div.vis-manipulation button.vis-button.vis-connect{background-image:url("")}div.vis-network div.vis-manipulation button.vis-button.vis-delete{background-image:url("")}div.vis-network div.vis-edit-mode div.vis-label,div.vis-network div.vis-manipulation div.vis-label{margin:0 0 0 23px;line-height:25px}div.vis-network div.vis-manipulation div.vis-separator-line{float:left;display:inline-block;width:1px;height:21px;background-color:#bdbdbd;margin:0 7px 0 15px} \ No newline at end of file diff --git a/queue_job/static/lib/vis/vis-network.min.js b/queue_job/static/lib/vis/vis-network.min.js new file mode 100644 index 0000000000..aa1897181e --- /dev/null +++ b/queue_job/static/lib/vis/vis-network.min.js @@ -0,0 +1,27 @@ +/** + * vis-network + * https://visjs.github.io/vis-network/ + * + * A dynamic, browser-based visualization library. + * + * @version 9.0.4 + * @date 2021-03-16T05:44:27.440Z + * + * @copyright (c) 2011-2017 Almende B.V, http://almende.com + * @copyright (c) 2017-2019 visjs contributors, https://github.com/visjs + * + * @license + * vis.js is dual licensed under both + * + * 1. The Apache 2.0 License + * http://www.apache.org/licenses/LICENSE-2.0 + * + * and + * + * 2. The MIT License + * http://opensource.org/licenses/MIT + * + * vis.js may be distributed under either license. + */ +!function(t,e){"object"==typeof exports&&"undefined"!=typeof module?e(exports):"function"==typeof define&&define.amd?define(["exports"],e):e((t="undefined"!=typeof globalThis?globalThis:t||self).vis=t.vis||{})}(this,(function(t){"use strict";var e="undefined"!=typeof globalThis?globalThis:"undefined"!=typeof window?window:"undefined"!=typeof global?global:"undefined"!=typeof self?self:{};function i(t){return t&&t.__esModule&&Object.prototype.hasOwnProperty.call(t,"default")?t.default:t}function n(t,e){return t(e={exports:{}},e.exports),e.exports}var o=function(t){return t&&t.Math==Math&&t},r=o("object"==typeof globalThis&&globalThis)||o("object"==typeof window&&window)||o("object"==typeof self&&self)||o("object"==typeof e&&e)||function(){return this}()||Function("return this")(),s=function(t){try{return!!t()}catch(t){return!0}},a=!s((function(){return 7!=Object.defineProperty({},1,{get:function(){return 7}})[1]})),h={}.propertyIsEnumerable,l=Object.getOwnPropertyDescriptor,d={f:l&&!h.call({1:2},1)?function(t){var e=l(this,t);return!!e&&e.enumerable}:h},c=function(t,e){return{enumerable:!(1&t),configurable:!(2&t),writable:!(4&t),value:e}},u={}.toString,f=function(t){return u.call(t).slice(8,-1)},p="".split,v=s((function(){return!Object("z").propertyIsEnumerable(0)}))?function(t){return"String"==f(t)?p.call(t,""):Object(t)}:Object,g=function(t){if(null==t)throw TypeError("Can't call method on "+t);return t},y=function(t){return v(g(t))},m=function(t){return"object"==typeof t?null!==t:"function"==typeof t},b=function(t,e){if(!m(t))return t;var i,n;if(e&&"function"==typeof(i=t.toString)&&!m(n=i.call(t)))return n;if("function"==typeof(i=t.valueOf)&&!m(n=i.call(t)))return n;if(!e&&"function"==typeof(i=t.toString)&&!m(n=i.call(t)))return n;throw TypeError("Can't convert object to primitive value")},w={}.hasOwnProperty,k=function(t,e){return w.call(t,e)},_=r.document,x=m(_)&&m(_.createElement),E=function(t){return x?_.createElement(t):{}},O=!a&&!s((function(){return 7!=Object.defineProperty(E("div"),"a",{get:function(){return 7}}).a})),C=Object.getOwnPropertyDescriptor,S={f:a?C:function(t,e){if(t=y(t),e=b(e,!0),O)try{return C(t,e)}catch(t){}if(k(t,e))return c(!d.f.call(t,e),t[e])}},T=/#|\.prototype\./,M=function(t,e){var i=D[P(t)];return i==B||i!=I&&("function"==typeof e?s(e):!!e)},P=M.normalize=function(t){return String(t).replace(T,".").toLowerCase()},D=M.data={},I=M.NATIVE="N",B=M.POLYFILL="P",z=M,N={},A=function(t){if("function"!=typeof t)throw TypeError(String(t)+" is not a function");return t},F=function(t,e,i){if(A(t),void 0===e)return t;switch(i){case 0:return function(){return t.call(e)};case 1:return function(i){return t.call(e,i)};case 2:return function(i,n){return t.call(e,i,n)};case 3:return function(i,n,o){return t.call(e,i,n,o)}}return function(){return t.apply(e,arguments)}},j=function(t){if(!m(t))throw TypeError(String(t)+" is not an object");return t},R=Object.defineProperty,L={f:a?R:function(t,e,i){if(j(t),e=b(e,!0),j(i),O)try{return R(t,e,i)}catch(t){}if("get"in i||"set"in i)throw TypeError("Accessors not supported");return"value"in i&&(t[e]=i.value),t}},H=a?function(t,e,i){return L.f(t,e,c(1,i))}:function(t,e,i){return t[e]=i,t},W=S.f,q=function(t){var e=function(e,i,n){if(this instanceof t){switch(arguments.length){case 0:return new t;case 1:return new t(e);case 2:return new t(e,i)}return new t(e,i,n)}return t.apply(this,arguments)};return e.prototype=t.prototype,e},V=function(t,e){var i,n,o,s,a,h,l,d,c=t.target,u=t.global,f=t.stat,p=t.proto,v=u?r:f?r[c]:(r[c]||{}).prototype,g=u?N:N[c]||(N[c]={}),y=g.prototype;for(o in e)i=!z(u?o:c+(f?".":"#")+o,t.forced)&&v&&k(v,o),a=g[o],i&&(h=t.noTargetGet?(d=W(v,o))&&d.value:v[o]),s=i&&h?h:e[o],i&&typeof a==typeof s||(l=t.bind&&i?F(s,r):t.wrap&&i?q(s):p&&"function"==typeof s?F(Function.call,s):s,(t.sham||s&&s.sham||a&&a.sham)&&H(l,"sham",!0),g[o]=l,p&&(k(N,n=c+"Prototype")||H(N,n,{}),N[n][o]=s,t.real&&y&&!y[o]&&H(y,o,s)))},U=Math.ceil,Y=Math.floor,X=function(t){return isNaN(t=+t)?0:(t>0?Y:U)(t)},G=Math.min,K=function(t){return t>0?G(X(t),9007199254740991):0},Q=Math.max,$=Math.min,Z=function(t,e){var i=X(t);return i<0?Q(i+e,0):$(i,e)},J=function(t){return function(e,i,n){var o,r=y(e),s=K(r.length),a=Z(n,s);if(t&&i!=i){for(;s>a;)if((o=r[a++])!=o)return!0}else for(;s>a;a++)if((t||a in r)&&r[a]===i)return t||a||0;return!t&&-1}},tt={includes:J(!0),indexOf:J(!1)},et={},it=tt.indexOf,nt=function(t,e){var i,n=y(t),o=0,r=[];for(i in n)!k(et,i)&&k(n,i)&&r.push(i);for(;e.length>o;)k(n,i=e[o++])&&(~it(r,i)||r.push(i));return r},ot=["constructor","hasOwnProperty","isPrototypeOf","propertyIsEnumerable","toLocaleString","toString","valueOf"],rt=Object.keys||function(t){return nt(t,ot)},st={f:Object.getOwnPropertySymbols},at=function(t){return Object(g(t))},ht=Object.assign,lt=Object.defineProperty,dt=!ht||s((function(){if(a&&1!==ht({b:1},ht(lt({},"a",{enumerable:!0,get:function(){lt(this,"b",{value:3,enumerable:!1})}}),{b:2})).b)return!0;var t={},e={},i=Symbol(),n="abcdefghijklmnopqrst";return t[i]=7,n.split("").forEach((function(t){e[t]=t})),7!=ht({},t)[i]||rt(ht({},e)).join("")!=n}))?function(t,e){for(var i=at(t),n=arguments.length,o=1,r=st.f,s=d.f;n>o;)for(var h,l=v(arguments[o++]),c=r?rt(l).concat(r(l)):rt(l),u=c.length,f=0;u>f;)h=c[f++],a&&!s.call(l,h)||(i[h]=l[h]);return i}:ht;V({target:"Object",stat:!0,forced:Object.assign!==dt},{assign:dt});var ct=N.Object.assign,ut=[].slice,ft={},pt=function(t,e,i){if(!(e in ft)){for(var n=[],o=0;o=.1;)(p=+r[c++%s])>d&&(p=d),f=Math.sqrt(p*p/(1+l*l)),e+=f=a<0?-f:f,i+=l*f,!0===u?t.lineTo(e,i):t.moveTo(e,i),d-=p,u=!u}var Ot={circle:wt,dashedLine:Et,database:xt,diamond:function(t,e,i,n){t.beginPath(),t.lineTo(e,i+n),t.lineTo(e+n,i),t.lineTo(e,i-n),t.lineTo(e-n,i),t.closePath()},ellipse:_t,ellipse_vis:_t,hexagon:function(t,e,i,n){t.beginPath();var o=2*Math.PI/6;t.moveTo(e+n,i);for(var r=1;r<6;r++)t.lineTo(e+n*Math.cos(o*r),i+n*Math.sin(o*r));t.closePath()},roundRect:kt,square:function(t,e,i,n){t.beginPath(),t.rect(e-n,i-n,2*n,2*n),t.closePath()},star:function(t,e,i,n){t.beginPath(),i+=.1*(n*=.82);for(var o=0;o<10;o++){var r=o%2==0?1.3*n:.5*n;t.lineTo(e+r*Math.sin(2*o*Math.PI/10),i-r*Math.cos(2*o*Math.PI/10))}t.closePath()},triangle:function(t,e,i,n){t.beginPath(),i+=.275*(n*=1.15);var o=2*n,r=o/2,s=Math.sqrt(3)/6*o,a=Math.sqrt(o*o-r*r);t.moveTo(e,i-(a-s)),t.lineTo(e+r,i+s),t.lineTo(e-r,i+s),t.lineTo(e,i-(a-s)),t.closePath()},triangleDown:function(t,e,i,n){t.beginPath(),i-=.275*(n*=1.15);var o=2*n,r=o/2,s=Math.sqrt(3)/6*o,a=Math.sqrt(o*o-r*r);t.moveTo(e,i+(a-s)),t.lineTo(e+r,i-s),t.lineTo(e-r,i-s),t.lineTo(e,i+(a-s)),t.closePath()}};var Ct=n((function(t){function e(t){if(t)return function(t){for(var i in e.prototype)t[i]=e.prototype[i];return t}(t)}t.exports=e,e.prototype.on=e.prototype.addEventListener=function(t,e){return this._callbacks=this._callbacks||{},(this._callbacks["$"+t]=this._callbacks["$"+t]||[]).push(e),this},e.prototype.once=function(t,e){function i(){this.off(t,i),e.apply(this,arguments)}return i.fn=e,this.on(t,i),this},e.prototype.off=e.prototype.removeListener=e.prototype.removeAllListeners=e.prototype.removeEventListener=function(t,e){if(this._callbacks=this._callbacks||{},0==arguments.length)return this._callbacks={},this;var i,n=this._callbacks["$"+t];if(!n)return this;if(1==arguments.length)return delete this._callbacks["$"+t],this;for(var o=0;o=a?t?"":void 0:(n=r.charCodeAt(s))<55296||n>56319||s+1===a||(o=r.charCodeAt(s+1))<56320||o>57343?t?r.charAt(s):n:t?r.slice(s,s+2):o-56320+(n-55296<<10)+65536}},Tt={codeAt:St(!1),charAt:St(!0)},Mt="__core-js_shared__",Pt=r[Mt]||function(t,e){try{H(r,t,e)}catch(i){r[t]=e}return e}(Mt,{}),Dt=Function.toString;"function"!=typeof Pt.inspectSource&&(Pt.inspectSource=function(t){return Dt.call(t)});var It,Bt,zt,Nt=Pt.inspectSource,At=r.WeakMap,Ft="function"==typeof At&&/native code/.test(Nt(At)),jt=n((function(t){(t.exports=function(t,e){return Pt[t]||(Pt[t]=void 0!==e?e:{})})("versions",[]).push({version:"3.9.1",mode:"pure",copyright:"© 2021 Denis Pushkarev (zloirock.ru)"})})),Rt=0,Lt=Math.random(),Ht=function(t){return"Symbol("+String(void 0===t?"":t)+")_"+(++Rt+Lt).toString(36)},Wt=jt("keys"),qt=function(t){return Wt[t]||(Wt[t]=Ht(t))},Vt=r.WeakMap;if(Ft){var Ut=Pt.state||(Pt.state=new Vt),Yt=Ut.get,Xt=Ut.has,Gt=Ut.set;It=function(t,e){return e.facade=t,Gt.call(Ut,t,e),e},Bt=function(t){return Yt.call(Ut,t)||{}},zt=function(t){return Xt.call(Ut,t)}}else{var Kt=qt("state");et[Kt]=!0,It=function(t,e){return e.facade=t,H(t,Kt,e),e},Bt=function(t){return k(t,Kt)?t[Kt]:{}},zt=function(t){return k(t,Kt)}}var Qt,$t,Zt={set:It,get:Bt,has:zt,enforce:function(t){return zt(t)?Bt(t):It(t,{})},getterFor:function(t){return function(e){var i;if(!m(e)||(i=Bt(e)).type!==t)throw TypeError("Incompatible receiver, "+t+" required");return i}}},Jt=!s((function(){function t(){}return t.prototype.constructor=null,Object.getPrototypeOf(new t)!==t.prototype})),te=qt("IE_PROTO"),ee=Object.prototype,ie=Jt?Object.getPrototypeOf:function(t){return t=at(t),k(t,te)?t[te]:"function"==typeof t.constructor&&t instanceof t.constructor?t.constructor.prototype:t instanceof Object?ee:null},ne="process"==f(r.process),oe=function(t){return"function"==typeof t?t:void 0},re=function(t,e){return arguments.length<2?oe(N[t])||oe(r[t]):N[t]&&N[t][e]||r[t]&&r[t][e]},se=re("navigator","userAgent")||"",ae=r.process,he=ae&&ae.versions,le=he&&he.v8;le?$t=(Qt=le.split("."))[0]+Qt[1]:se&&(!(Qt=se.match(/Edge\/(\d+)/))||Qt[1]>=74)&&(Qt=se.match(/Chrome\/(\d+)/))&&($t=Qt[1]);var de,ce,ue,fe=$t&&+$t,pe=!!Object.getOwnPropertySymbols&&!s((function(){return!Symbol.sham&&(ne?38===fe:fe>37&&fe<41)})),ve=pe&&!Symbol.sham&&"symbol"==typeof Symbol.iterator,ge=jt("wks"),ye=r.Symbol,me=ve?ye:ye&&ye.withoutSetter||Ht,be=function(t){return k(ge,t)&&(pe||"string"==typeof ge[t])||(pe&&k(ye,t)?ge[t]=ye[t]:ge[t]=me("Symbol."+t)),ge[t]},we=be("iterator"),ke=!1;[].keys&&("next"in(ue=[].keys())?(ce=ie(ie(ue)))!==Object.prototype&&(de=ce):ke=!0);var _e=null==de||s((function(){var t={};return de[we].call(t)!==t}));_e&&(de={}),_e&&!k(de,we)&&H(de,we,(function(){return this}));var xe,Ee={IteratorPrototype:de,BUGGY_SAFARI_ITERATORS:ke},Oe=a?Object.defineProperties:function(t,e){j(t);for(var i,n=rt(e),o=n.length,r=0;o>r;)L.f(t,i=n[r++],e[i]);return t},Ce=re("document","documentElement"),Se=qt("IE_PROTO"),Te=function(){},Me=function(t){return"