From f9b04aeb240ca55640284e5734f58775529add8c Mon Sep 17 00:00:00 2001 From: GuillemCForgeFlow Date: Mon, 2 Oct 2023 16:29:29 +0200 Subject: [PATCH] [ADD]product_harmonized_system_heading New module to have a better structure on H.S. Codes. If installed, H.S. Codes will require from a H.S. Heading to be created first. --- product_harmonized_system_heading/__init__.py | 2 + .../__manifest__.py | 24 +++++++ product_harmonized_system_heading/hooks.py | 33 ++++++++++ .../models/__init__.py | 2 + .../models/hs_code.py | 41 ++++++++++++ .../models/hs_code_heading.py | 48 ++++++++++++++ .../readme/CONTRIBUTORS.rst | 1 + .../readme/DESCRIPTION.rst | 4 ++ .../readme/USAGE.rst | 6 ++ .../security/ir.model.access.csv | 3 + .../security/security.xml | 22 +++++++ .../tests/__init__.py | 1 + .../test_product_harmonized_system_heading.py | 66 +++++++++++++++++++ .../views/hs_code_heading_views.xml | 51 ++++++++++++++ .../views/hs_code_views.xml | 49 ++++++++++++++ .../addons/product_harmonized_system_heading | 1 + .../setup.py | 6 ++ 17 files changed, 360 insertions(+) create mode 100644 product_harmonized_system_heading/__init__.py create mode 100644 product_harmonized_system_heading/__manifest__.py create mode 100644 product_harmonized_system_heading/hooks.py create mode 100644 product_harmonized_system_heading/models/__init__.py create mode 100644 product_harmonized_system_heading/models/hs_code.py create mode 100644 product_harmonized_system_heading/models/hs_code_heading.py create mode 100644 product_harmonized_system_heading/readme/CONTRIBUTORS.rst create mode 100644 product_harmonized_system_heading/readme/DESCRIPTION.rst create mode 100644 product_harmonized_system_heading/readme/USAGE.rst create mode 100644 product_harmonized_system_heading/security/ir.model.access.csv create mode 100644 product_harmonized_system_heading/security/security.xml create mode 100644 product_harmonized_system_heading/tests/__init__.py create mode 100644 product_harmonized_system_heading/tests/test_product_harmonized_system_heading.py create mode 100644 product_harmonized_system_heading/views/hs_code_heading_views.xml create mode 100644 product_harmonized_system_heading/views/hs_code_views.xml create mode 120000 setup/product_harmonized_system_heading/odoo/addons/product_harmonized_system_heading create mode 100644 setup/product_harmonized_system_heading/setup.py diff --git a/product_harmonized_system_heading/__init__.py b/product_harmonized_system_heading/__init__.py new file mode 100644 index 00000000..cc6b6354 --- /dev/null +++ b/product_harmonized_system_heading/__init__.py @@ -0,0 +1,2 @@ +from . import models +from .hooks import post_init_hook diff --git a/product_harmonized_system_heading/__manifest__.py b/product_harmonized_system_heading/__manifest__.py new file mode 100644 index 00000000..b20f6705 --- /dev/null +++ b/product_harmonized_system_heading/__manifest__.py @@ -0,0 +1,24 @@ +# Copyright 2023 ForgeFlow S.L. (https://www.forgeflow.com) +# License AGPL-3.0 or later (https://www.gnu.org/licenses/agpl.html). +{ + "name": "Product Harmonized System Code Heading", + "summary": """ + Adds a heading model for the Product Harmonized System Codes (H.S. Codes). + """, + "version": "13.0.1.0.0", + "category": "Reporting", + "license": "AGPL-3", + "website": "https://github.com/OCA/intrastat-extrastat", + "author": "ForgeFlow, Odoo Community Association (OCA)", + "depends": ["product_harmonized_system"], + "data": [ + "security/security.xml", + "security/ir.model.access.csv", + "views/hs_code_heading_views.xml", + "views/hs_code_views.xml", + ], + "maintainers": ["GuillemCForgeFlow"], + "installable": True, + "application": False, + "post_init_hook": "post_init_hook", +} diff --git a/product_harmonized_system_heading/hooks.py b/product_harmonized_system_heading/hooks.py new file mode 100644 index 00000000..896a9c89 --- /dev/null +++ b/product_harmonized_system_heading/hooks.py @@ -0,0 +1,33 @@ +# Copyright 2023 ForgeFlow S.L. (https://www.forgeflow.com) +# License AGPL-3.0 or later (https://www.gnu.org/licenses/agpl.html). +import logging + +from odoo import SUPERUSER_ID, api + +_logger = logging.getLogger(__name__) + + +def post_init_hook(cr, registry): + _logger.info("Creating default H.S. Code Headings for H.S. Codes in the System.") + env = api.Environment(cr, SUPERUSER_ID, {}) + HSCode = env["hs.code"].with_context(active_test=False) + HSCodeHeading = env["hs.code.heading"].with_context(active_test=False) + existing_hs_code_heading_codes = HSCodeHeading.search([]).mapped("code") + to_create_dict = dict() + hs_code_ids = HSCode.search([]) + for hs_code in hs_code_ids: + vals = False + heading_code = hs_code.local_code[:4] + if heading_code in existing_hs_code_heading_codes: + continue + if heading_code not in to_create_dict.keys(): + vals = {"code": heading_code} + to_create_dict[heading_code] = [hs_code.id] + else: + to_create_dict[heading_code].append(hs_code.id) + vals_list = list() + for code, rel_hs_code_ids in to_create_dict.items(): + vals = {"code": code, "hs_code_ids": [(6, 0, rel_hs_code_ids)]} + vals_list.append(vals) + _logger.info("Creating %s H.S. Code Headings" % len(vals_list)) + HSCodeHeading.create(vals_list) diff --git a/product_harmonized_system_heading/models/__init__.py b/product_harmonized_system_heading/models/__init__.py new file mode 100644 index 00000000..4f6ba15e --- /dev/null +++ b/product_harmonized_system_heading/models/__init__.py @@ -0,0 +1,2 @@ +from . import hs_code_heading +from . import hs_code diff --git a/product_harmonized_system_heading/models/hs_code.py b/product_harmonized_system_heading/models/hs_code.py new file mode 100644 index 00000000..db6e0620 --- /dev/null +++ b/product_harmonized_system_heading/models/hs_code.py @@ -0,0 +1,41 @@ +# Copyright 2023 ForgeFlow S.L. (https://www.forgeflow.com) +# License AGPL-3.0 or later (https://www.gnu.org/licenses/agpl.html). +from odoo import _, api, fields, models +from odoo.exceptions import ValidationError +from odoo.tools import config + + +class HSCode(models.Model): + _inherit = "hs.code" + + hs_code_heading_id = fields.Many2one(comodel_name="hs.code.heading", readonly=True) + + @api.model_create_multi + def create(self, vals_list): + if not config["test_enable"]: + HSCodeHeading = self.env["hs.code.heading"] + for vals in vals_list: + local_code = vals.get("local_code", "") + heading_code = local_code[:4] + heading = HSCodeHeading.search([("code", "=", heading_code)], limit=1) + if not heading: + msg = ( + "There is no H.S. Code Heading with the following Code: %s." + "\nYou first need to create the Heading." % heading_code + ) + raise ValidationError(_(msg)) + vals["hs_code_heading_id"] = heading.id + return super().create(vals_list) + + def _get_name(self): + """ + Overwrite method from the `product_harmonized_system` module in order to set the + Heading Description in the H.S. Code name + """ + self.ensure_one() + name = self.local_code + if self.hs_code_heading_id.description: + name += " " + self.hs_code_heading_id.description + if self.description: + name += ": " + self.description + return len(name) > 55 and name[:55] + "..." or name diff --git a/product_harmonized_system_heading/models/hs_code_heading.py b/product_harmonized_system_heading/models/hs_code_heading.py new file mode 100644 index 00000000..7ddb3bbf --- /dev/null +++ b/product_harmonized_system_heading/models/hs_code_heading.py @@ -0,0 +1,48 @@ +# Copyright 2023 ForgeFlow S.L. (https://www.forgeflow.com) +# License AGPL-3.0 or later (https://www.gnu.org/licenses/agpl.html). +from odoo import fields, models + + +class HSCodeHeading(models.Model): + _name = "hs.code.heading" + _description = "H.S. Code Heading" + + # General fields + active = fields.Boolean(default=True) + code = fields.Char( + required=True, + size=4, + help="The H.S. Code Heading can only have 4 digits, as this define the Heading.", + ) + description = fields.Text() + + # Relational fields + hs_code_ids = fields.One2many( + comodel_name="hs.code", + inverse_name="hs_code_heading_id", + string="H.S. Codes", + readonly=True, + help="The related H.S. Codes using the Heading.", + ) + + _sql_constraints = [ + ( + "code_uniq", + "UNIQUE(code)", + "There is already an existing H.S. Code Heading with this Code.", + ) + ] + + def _get_name(self): + self.ensure_one() + name = self.code + if self.description: + name += " " + self.description + return len(name) > 55 and name[:55] + "..." or name + + def name_get(self): + res = [] + for rec in self: + name = rec._get_name() + res.append((rec.id, name)) + return res diff --git a/product_harmonized_system_heading/readme/CONTRIBUTORS.rst b/product_harmonized_system_heading/readme/CONTRIBUTORS.rst new file mode 100644 index 00000000..163379ac --- /dev/null +++ b/product_harmonized_system_heading/readme/CONTRIBUTORS.rst @@ -0,0 +1 @@ +* Guillem Casassas diff --git a/product_harmonized_system_heading/readme/DESCRIPTION.rst b/product_harmonized_system_heading/readme/DESCRIPTION.rst new file mode 100644 index 00000000..f642cb82 --- /dev/null +++ b/product_harmonized_system_heading/readme/DESCRIPTION.rst @@ -0,0 +1,4 @@ +This module adds the possibility to assign Headings to the H.S. Codes in order to have +a better structure. + +The H.S. Code Heading has a unique Code and a possible Description. diff --git a/product_harmonized_system_heading/readme/USAGE.rst b/product_harmonized_system_heading/readme/USAGE.rst new file mode 100644 index 00000000..2008a142 --- /dev/null +++ b/product_harmonized_system_heading/readme/USAGE.rst @@ -0,0 +1,6 @@ +In order to assign the H.S. Code Heading to a new H.S. Code, we first need to have the +Heading created, if there is none when creating the Code, the user will see an error +demanding to create the Heading first. + +The relation is not editable from the User prespective as we just to need to ensure that +the Headings are created before creating the H.S. Codes. diff --git a/product_harmonized_system_heading/security/ir.model.access.csv b/product_harmonized_system_heading/security/ir.model.access.csv new file mode 100644 index 00000000..ce78aad1 --- /dev/null +++ b/product_harmonized_system_heading/security/ir.model.access.csv @@ -0,0 +1,3 @@ +id,name,model_id:id,group_id:id,perm_read,perm_write,perm_create,perm_unlink +access_hs_code_heading_user,access_hs_code_heading_user,model_hs_code_heading,,1,0,0,0 +access_hs_code_heading_manager,access_hs_code_heading_manager,model_hs_code_heading,product_harmonized_system_heading.group_product_harmonized_system_heading_manager,1,1,1,1 diff --git a/product_harmonized_system_heading/security/security.xml b/product_harmonized_system_heading/security/security.xml new file mode 100644 index 00000000..84988407 --- /dev/null +++ b/product_harmonized_system_heading/security/security.xml @@ -0,0 +1,22 @@ + + + + H.S. Code + 20 + + + H.S. Code Heading Manager + The user will have manager access to H.S. Code Heading. + + + + + diff --git a/product_harmonized_system_heading/tests/__init__.py b/product_harmonized_system_heading/tests/__init__.py new file mode 100644 index 00000000..3d640cda --- /dev/null +++ b/product_harmonized_system_heading/tests/__init__.py @@ -0,0 +1 @@ +from . import test_product_harmonized_system_heading diff --git a/product_harmonized_system_heading/tests/test_product_harmonized_system_heading.py b/product_harmonized_system_heading/tests/test_product_harmonized_system_heading.py new file mode 100644 index 00000000..d01400c6 --- /dev/null +++ b/product_harmonized_system_heading/tests/test_product_harmonized_system_heading.py @@ -0,0 +1,66 @@ +# Copyright 2023 ForgeFlow S.L. (https://www.forgeflow.com) +# License AGPL-3.0 or later (https://www.gnu.org/licenses/agpl.html). +from odoo.exceptions import ValidationError +from odoo.tests.common import SavepointCase + + +class TestProductHarmonizedSystemHeading(SavepointCase): + @classmethod + def setUpClass(cls): + super().setUpClass() + # Models + cls.hs_code_model = cls.env["hs.code"] + cls.hs_code_heading_model = cls.env["hs.code.heading"] + + @classmethod + def _create_hs_code_heading(cls, code="1234", description=False): + return cls.hs_code_heading_model.create( + {"code": code, "description": description} + ) + + @classmethod + def _create_hs_code(cls, local_code="123456789", heading=False, description=False): + return cls.hs_code_model.create( + { + "local_code": local_code, + "hs_code_heading_id": heading.id if heading else heading, + "description": description, + } + ) + + def test_01_check_validation_error_if_no_heading(self): + """ + Check that a Validation Error is raised when trying to create a new H.S. Code + with a Heading which is still not created in the system. + """ + with self.assertRaises(ValidationError): + self._create_hs_code() + + def test_02_check_hs_code_creation(self): + """ + Check that the H.S. Code is correctly created if there is a related Heading. + """ + heading = self._create_hs_code_heading(code="5678") + code = self._create_hs_code(local_code="56789123", heading=heading) + self.assertTrue(code) + + def test_03_check_correct_hs_code_name_get_value(self): + """ + Check that the H.S. Code display name is correctly set based on the changes. + """ + heading_description = "Heading Description" + code_description = "Code Description" + heading = self._create_hs_code_heading(description=heading_description) + code = self._create_hs_code(heading=heading, description=code_description) + self.assertEqual( + code.name_get()[0][1], + "123456789 %s: %s" % (heading_description, code_description), + ) + + def test_04_check_correct_hs_code_heading_name_get_value(self): + """ + Check that the H.S. Code display name is correctly set based on the changes. + """ + heading_description = "Heading Description 2" + heading = self._create_hs_code_heading(description=heading_description) + self.assertEqual(heading.name_get()[0][1], "1234 %s" % heading_description) diff --git a/product_harmonized_system_heading/views/hs_code_heading_views.xml b/product_harmonized_system_heading/views/hs_code_heading_views.xml new file mode 100644 index 00000000..3b56136f --- /dev/null +++ b/product_harmonized_system_heading/views/hs_code_heading_views.xml @@ -0,0 +1,51 @@ + + + + + hs.code.heading.form + hs.code.heading + +
+ + + + + + + + + +
+
+
+ + hs.code.heading.tree + hs.code.heading + + + + + + + + + + hs.code.heading.search.view + hs.code.heading + + + + + + + + + H.S. Code Heading + ir.actions.act_window + hs.code.heading + tree,form + +
diff --git a/product_harmonized_system_heading/views/hs_code_views.xml b/product_harmonized_system_heading/views/hs_code_views.xml new file mode 100644 index 00000000..7fda1d83 --- /dev/null +++ b/product_harmonized_system_heading/views/hs_code_views.xml @@ -0,0 +1,49 @@ + + + + + hs.code.search - product_harmonized_system_heading + hs.code + + + + + + + + + + + + + hs.code.tree - product_harmonized_system_heading + hs.code + + + + + + + + + hs.code.form - product_harmonized_system_heading + hs.code + + + + + + + + diff --git a/setup/product_harmonized_system_heading/odoo/addons/product_harmonized_system_heading b/setup/product_harmonized_system_heading/odoo/addons/product_harmonized_system_heading new file mode 120000 index 00000000..64470d0b --- /dev/null +++ b/setup/product_harmonized_system_heading/odoo/addons/product_harmonized_system_heading @@ -0,0 +1 @@ +../../../../product_harmonized_system_heading \ No newline at end of file diff --git a/setup/product_harmonized_system_heading/setup.py b/setup/product_harmonized_system_heading/setup.py new file mode 100644 index 00000000..28c57bb6 --- /dev/null +++ b/setup/product_harmonized_system_heading/setup.py @@ -0,0 +1,6 @@ +import setuptools + +setuptools.setup( + setup_requires=['setuptools-odoo'], + odoo_addon=True, +)