From 97b0723722d8ebd3cbf5828c871da19f3d471187 Mon Sep 17 00:00:00 2001 From: Joren Van Onder <jov@odoo.com> Date: Wed, 11 Feb 2015 10:18:10 +0100 Subject: [PATCH] [ADD] pos_cache: add pos_cache ported to v9 from odoo-extra Originally written by Nicolas Seinlet <nse@odoo.com>. Provides pos.config with a product.product cache. This adds a ported and refactored pos_cache from odoo-extra: - Adapt to changes in web and point_of_sale since 8.0 - Refactor the load_server_data function, so it isn't copy pasted but cleanly patched into the load_server_data super. - Make the cron inactive by default. - Also remove retrying the write 5 times, when it fails we'll just have to wait until the cron triggers again (an hour from then). - Store the product.product fields and domain on pos_cache. When cache is requested from the pos with a different field or domain it'll automatically recompute. Necessary for when someone would install a module that modifies these things in the pos after a cache has been generated. - Have multiple caches per pos_config per user - Make the module adhere to the Odoo Coding Guidelines. --- addons/pos_cache/__init__.py | 3 + addons/pos_cache/__openerp__.py | 25 +++++ addons/pos_cache/data/pos_cache_data.xml | 16 +++ addons/pos_cache/doc/cache.rst | 13 +++ addons/pos_cache/models/__init__.py | 3 + addons/pos_cache/models/pos_cache.py | 97 +++++++++++++++++++ addons/pos_cache/security/ir.model.access.csv | 2 + addons/pos_cache/static/src/js/pos_cache.js | 43 ++++++++ .../pos_cache/views/pos_cache_templates.xml | 10 ++ addons/pos_cache/views/pos_cache_views.xml | 18 ++++ 10 files changed, 230 insertions(+) create mode 100644 addons/pos_cache/__init__.py create mode 100644 addons/pos_cache/__openerp__.py create mode 100644 addons/pos_cache/data/pos_cache_data.xml create mode 100644 addons/pos_cache/doc/cache.rst create mode 100644 addons/pos_cache/models/__init__.py create mode 100644 addons/pos_cache/models/pos_cache.py create mode 100644 addons/pos_cache/security/ir.model.access.csv create mode 100644 addons/pos_cache/static/src/js/pos_cache.js create mode 100644 addons/pos_cache/views/pos_cache_templates.xml create mode 100644 addons/pos_cache/views/pos_cache_views.xml diff --git a/addons/pos_cache/__init__.py b/addons/pos_cache/__init__.py new file mode 100644 index 000000000000..48f417bbd932 --- /dev/null +++ b/addons/pos_cache/__init__.py @@ -0,0 +1,3 @@ +# -*- coding: utf-8 -*- +# Part of Odoo. See LICENSE file for full copyright and licensing details. +import models diff --git a/addons/pos_cache/__openerp__.py b/addons/pos_cache/__openerp__.py new file mode 100644 index 000000000000..708ae46108b9 --- /dev/null +++ b/addons/pos_cache/__openerp__.py @@ -0,0 +1,25 @@ +# -*- coding: utf-8 -*- +# Part of Odoo. See LICENSE file for full copyright and licensing details. +{ + 'name': "pos_cache", + + 'summary': """ + Enable a cache on products for a lower POS loading time.""", + + 'description': """ +This creates a product cache per POS config. It drastically lowers the +time it takes to load a POS session with a lot of products. + """, + + 'author': "Odoo", + 'website': "https://www.odoo.com/page/point-of-sale", + 'category': 'Point Of Sale', + 'version': '1.0', + 'depends': ['point_of_sale'], + 'data': [ + 'data/pos_cache_data.xml', + 'security/ir.model.access.csv', + 'views/pos_cache_views.xml', + 'views/pos_cache_templates.xml', + ] +} diff --git a/addons/pos_cache/data/pos_cache_data.xml b/addons/pos_cache/data/pos_cache_data.xml new file mode 100644 index 000000000000..709e6edea356 --- /dev/null +++ b/addons/pos_cache/data/pos_cache_data.xml @@ -0,0 +1,16 @@ +<?xml version="1.0" encoding="utf-8"?> +<odoo> + <data> + <record model="ir.cron" id="refresh_pos_cache_cron"> + <field name="name">Refresh POS Cache</field> + <field name="active" eval="False"/> + <field name="interval_number">1</field> + <field name="interval_type">hours</field> + <field name="numbercall">-1</field> + <field name="doall" eval="False"/> + <field name="model" eval="'pos.cache'"/> + <field name="function" eval="'refresh_all_caches'"/> + <field name="args" eval="'()'"/> + </record> + </data> +</odoo> diff --git a/addons/pos_cache/doc/cache.rst b/addons/pos_cache/doc/cache.rst new file mode 100644 index 000000000000..b01c635ef38d --- /dev/null +++ b/addons/pos_cache/doc/cache.rst @@ -0,0 +1,13 @@ +POS Cache ++++++++++ + +This module enables a cache for the products in the pos configs. Each POS Config has his own Cache. + +The Cache is updated every hour by a cron. + +============ +Compute user +============ + +As it's a bad practice to use the admin in a multi-company configuration, a field permit to force a user to compute +the cache. A badly chosen user can result in wrong taxes in POS in a multi-company environment. diff --git a/addons/pos_cache/models/__init__.py b/addons/pos_cache/models/__init__.py new file mode 100644 index 000000000000..8c6b3129856a --- /dev/null +++ b/addons/pos_cache/models/__init__.py @@ -0,0 +1,3 @@ +# -*- coding: utf-8 -*- +# Part of Odoo. See LICENSE file for full copyright and licensing details. +import pos_cache diff --git a/addons/pos_cache/models/pos_cache.py b/addons/pos_cache/models/pos_cache.py new file mode 100644 index 000000000000..0ae91b8bfeb7 --- /dev/null +++ b/addons/pos_cache/models/pos_cache.py @@ -0,0 +1,97 @@ +# -*- coding: utf-8 -*- +# Part of Odoo. See LICENSE file for full copyright and licensing details. +from ast import literal_eval +import cPickle + +from openerp import models, fields, api + + +class pos_cache(models.Model): + _name = 'pos.cache' + + cache = fields.Binary() + product_domain = fields.Text(required=True) + product_fields = fields.Text(required=True) + + config_id = fields.Many2one('pos.config', ondelete='cascade', required=True) + compute_user_id = fields.Many2one('res.users', 'Cache compute user', required=True) + + @api.model + def refresh_all_caches(self): + self.env['pos.cache'].search([]).refresh_cache() + + @api.one + def refresh_cache(self): + products = self.env['product.product'].search(self.get_product_domain()) + prod_ctx = products.with_context(pricelist=self.config_id.pricelist_id.id, display_default_code=False) + prod_ctx = prod_ctx.sudo(self.compute_user_id.id) + res = prod_ctx.read(self.get_product_fields()) + datas = { + 'cache': cPickle.dumps(res, protocol=cPickle.HIGHEST_PROTOCOL), + } + + self.write(datas) + + @api.model + def get_product_domain(self): + return literal_eval(self.product_domain) + + @api.model + def get_product_fields(self): + return literal_eval(self.product_fields) + + @api.model + def get_cache(self, domain, fields): + if not self.cache or domain != self.get_product_domain() or fields != self.get_product_fields(): + self.product_domain = str(domain) + self.product_fields = str(fields) + self.refresh_cache() + + return cPickle.loads(self.cache) + + +class pos_config(models.Model): + _inherit = 'pos.config' + + @api.one + @api.depends('cache_ids') + def _get_oldest_cache_time(self): + pos_cache = self.env['pos.cache'] + oldest_cache = pos_cache.search([('config_id', '=', self.id)], order='write_date', limit=1) + if oldest_cache: + self.oldest_cache_time = oldest_cache.write_date + + # Use a related model to avoid the load of the cache when the pos load his config + cache_ids = fields.One2many('pos.cache', 'config_id') + oldest_cache_time = fields.Datetime(compute='_get_oldest_cache_time', string='Oldest cache time', readonly=True) + + def _get_cache_for_user(self): + pos_cache = self.env['pos.cache'] + cache_for_user = pos_cache.search([('id', 'in', self.cache_ids.ids), ('compute_user_id', '=', self.env.uid)]) + + if cache_for_user: + return cache_for_user[0] + else: + return None + + @api.multi + def get_products_from_cache(self, fields, domain): + cache_for_user = self._get_cache_for_user() + + if cache_for_user: + return cache_for_user.get_cache(domain, fields) + else: + pos_cache = self.env['pos.cache'] + pos_cache.create({ + 'config_id': self.id, + 'product_domain': str(domain), + 'product_fields': str(fields), + 'compute_user_id': self.env.uid + }) + new_cache = self._get_cache_for_user() + return new_cache.get_cache(domain, fields) + + @api.one + def delete_cache(self): + # throw away the old caches + self.cache_ids.unlink() diff --git a/addons/pos_cache/security/ir.model.access.csv b/addons/pos_cache/security/ir.model.access.csv new file mode 100644 index 000000000000..1a2651865f9f --- /dev/null +++ b/addons/pos_cache/security/ir.model.access.csv @@ -0,0 +1,2 @@ +id,name,model_id:id,group_id:id,perm_read,perm_write,perm_create,perm_unlink +access_pos_cache,pos.cache,model_pos_cache,point_of_sale.group_pos_user,1,1,1,1 diff --git a/addons/pos_cache/static/src/js/pos_cache.js b/addons/pos_cache/static/src/js/pos_cache.js new file mode 100644 index 000000000000..29d5e3bfbc49 --- /dev/null +++ b/addons/pos_cache/static/src/js/pos_cache.js @@ -0,0 +1,43 @@ +odoo.define('pos_cache.pos_cache', function (require) { + "use strict"; + var core = require('web.core'); + var models = require('point_of_sale.models'); + var Model = require('web.DataModel'); + var _t = core._t; + + var posmodel_super = models.PosModel.prototype; + models.PosModel = models.PosModel.extend({ + load_server_data: function () { + var self = this; + + var product_index = _.findIndex(this.models, function (model) { + return model.model === "product.product"; + }); + + // Give both the fields and domain to pos_cache in the + // backend. This way we don't have to hardcode these + // values in the backend and they automatically stay in + // sync with whatever is defined (and maybe extended by + // other modules) in js. + var product_model = this.models[product_index]; + var product_fields = product_model.fields; + var product_domain = product_model.domain; + + // We don't want to load product.product the normal + // uncached way, so get rid of it. + if (product_index !== -1) { + this.models.splice(product_index, 1); + } + + return posmodel_super.load_server_data.apply(this, arguments).then(function () { + var records = new Model('pos.config').call('get_products_from_cache', + [self.pos_session.config_id[0], product_fields, product_domain]); + + self.chrome.loading_message(_t('Loading') + ' product.product', 1); + return records.then(function (product) { + self.db.add_products(product); + }); + }); + }, + }); +}); diff --git a/addons/pos_cache/views/pos_cache_templates.xml b/addons/pos_cache/views/pos_cache_templates.xml new file mode 100644 index 000000000000..6866bd2a24ff --- /dev/null +++ b/addons/pos_cache/views/pos_cache_templates.xml @@ -0,0 +1,10 @@ +<?xml version="1.0" encoding="utf-8"?> +<odoo> + <data> + <template id="assets" inherit_id="point_of_sale.assets"> + <xpath expr="." position="inside"> + <script type="text/javascript" src="/pos_cache/static/src/js/pos_cache.js"></script> + </xpath> + </template> + </data> +</odoo> diff --git a/addons/pos_cache/views/pos_cache_views.xml b/addons/pos_cache/views/pos_cache_views.xml new file mode 100644 index 000000000000..522d9437a0b1 --- /dev/null +++ b/addons/pos_cache/views/pos_cache_views.xml @@ -0,0 +1,18 @@ +<?xml version="1.0"?> +<odoo> + <data> + <record model="ir.ui.view" id="view_pos_config_form"> + <field name="name">pos.config.form.view</field> + <field name="model">pos.config</field> + <field name="inherit_id" ref="point_of_sale.view_pos_config_form" /> + <field name="arch" type="xml"> + <xpath expr="//button[@name='set_deprecate']" position="after"> + <button name='delete_cache' type="object" string="Recompute cache"/> + </xpath> + <xpath expr="//field[@name='sequence_id']" position="after"> + <field name='oldest_cache_time'/> + </xpath> + </field> + </record> + </data> +</odoo> -- GitLab