Merge pull request #418 from ForgeFlow/15.0-mig-rma_operating_unit

[15.0][mig]-rma_operating_unit
This commit is contained in:
Lois Rilo
2023-07-06 10:55:26 +02:00
committed by GitHub
14 changed files with 486 additions and 0 deletions

View File

@@ -0,0 +1,34 @@
.. image:: https://img.shields.io/badge/license-LGPLv3-blue.svg
:target: https://www.gnu.org/licenses/lgpl.html
:alt: License: LGPL-3
========================
RMA with Operating Units
========================
This module introduces the following features:
* Adds the Operating Unit (OU) to the RMA order.
* Users can only view and manage the rma orders associated to their operating
unit.
Usage
=====
* The default operating unit of the RMA comes from the default operating unit
of the user
Contributors
------------
* Aaron Henriquez <ahenriquez@forgeflow.com>
* Beñat Jimenez <benat.jimenez@forgeflow.com>
* Juany Davila <juany.davila@forgeflow.com>
Maintainer
----------
This module is maintained by ForgeFlow.

View File

@@ -0,0 +1 @@
from . import models

View File

@@ -0,0 +1,17 @@
# Copyright 2017-23 ForgeFlow S.L.
# License LGPL-3.0 or later (https://www.gnu.org/licenses/lgpl.html).
{
"name": "Operating Unit in RMA Groups",
"version": "15.0.1.0.0",
"author": "ForgeFlow",
"license": "LGPL-3",
"website": "https://github.com/ForgeFlow/stock-rma",
"category": "Operating Units",
"depends": ["rma", "stock_operating_unit"],
"data": [
"security/rma_security.xml",
"views/rma_order_view.xml",
"views/rma_order_line_view.xml",
],
}

View File

@@ -0,0 +1,2 @@
from . import rma_order
from . import rma_order_line

View File

@@ -0,0 +1,35 @@
# Copyright 2017-23 ForgeFlow S.L.
# License LGPL-3.0 or later (https://www.gnu.org/licenses/lgpl.html).
from odoo import _, api, fields, models
from odoo.exceptions import ValidationError
class RmaOrder(models.Model):
_inherit = "rma.order"
@api.constrains("rma_line_ids", "rma_line_ids.operating_unit_id")
def _check_operating_unit(self):
for rma in self:
bad_lines = rma.rma_line_ids.filtered(
lambda l: l.operating_unit_id != rma.operating_unit_id
)
if bad_lines:
raise ValidationError(
_(
"The operating unit of the rma lines have to match the"
" one of the group"
)
)
return True
@api.model
def _default_operating_unit(self):
return self.env.user.default_operating_unit_id
operating_unit_id = fields.Many2one(
comodel_name="operating.unit",
string="Operating Unit",
default=_default_operating_unit,
)

View File

@@ -0,0 +1,21 @@
# Copyright 2017-23 ForgeFlow S.L.
# License LGPL-3.0 or later (https://www.gnu.org/licenses/lgpl.html).
from odoo import api, fields, models
class RmaOrderLine(models.Model):
_inherit = "rma.order.line"
@api.model
def _default_operating_unit(self):
if self.rma_id.operating_unit_id:
return self.rma_id.operating_unit_id.id
return self.env.user.default_operating_unit_id
operating_unit_id = fields.Many2one(
comodel_name="operating.unit",
string="Operating Unit",
default=_default_operating_unit,
)

View File

@@ -0,0 +1,16 @@
<?xml version="1.0" encoding="utf-8" ?>
<!-- Copyright 2017-19 ForgeFlow S.L.
License LGPL-3.0 or later (http://www.gnu.org/licenses/lgpl-3.0) -->
<odoo noupdate="0">
<record id="ir_rule_rma_allowed_operating_units" model="ir.rule">
<field name="model_id" ref="rma.model_rma_order" />
<field name="domain_force">['|', ('operating_unit_id', '=', False),
('operating_unit_id', 'in', [g.id for g in user.operating_unit_ids])]</field>
<field name="name">RMA from allowed operating units</field>
<field name="global" eval="True" />
<field eval="0" name="perm_unlink" />
<field eval="0" name="perm_write" />
<field eval="1" name="perm_read" />
<field eval="0" name="perm_create" />
</record>
</odoo>

View File

@@ -0,0 +1 @@
from . import test_rma_operating_unit

View File

@@ -0,0 +1,138 @@
# Copyright 2017-23 ForgeFlow S.L.
# License LGPL-3.0 or later (https://www.gnu.org/licenses/lgpl.html).
from odoo import exceptions
from odoo.tests import common
class TestRmaOperatingUnit(common.TransactionCase):
def setUp(self):
super(TestRmaOperatingUnit, self).setUp()
self.res_users_model = self.env["res.users"]
self.rma_model = self.env["rma.order"]
self.rma_line_model = self.env["rma.order.line"]
self.company = self.env.ref("base.main_company")
self.partner = self.env.ref("base.res_partner_1")
self.grp_rma_manager = self.env.ref("rma.group_rma_manager")
self.grp_ou = self.env.ref("operating_unit.group_multi_operating_unit")
self.grp_stock = self.env.ref("stock.group_stock_manager")
self.product = self.env.ref("product.product_product_12")
# Main Operating Unit
self.main_OU = self.env.ref("operating_unit.main_operating_unit")
# B2C Operating Unit
self.b2c_OU = self.env.ref("operating_unit.b2c_operating_unit")
# Users
self.user1 = self._create_user(
"user_1",
[self.grp_rma_manager, self.grp_ou, self.grp_stock],
self.company,
[self.main_OU, self.b2c_OU],
)
self.user2 = self._create_user(
"user_2",
[self.grp_rma_manager, self.grp_ou, self.grp_stock],
self.company,
[self.b2c_OU],
)
self.user3 = self._create_user(
"user_3",
[self.grp_rma_manager, self.grp_ou, self.grp_stock],
self.company,
[self.main_OU, self.b2c_OU],
)
# RMA Orders
self.rma_order1 = self._create_rma(self.user1.id, self.main_OU)
self.rma_order2 = self._create_rma(self.user2.id, self.b2c_OU)
self.rma_order3 = self._create_rma(self.user3.id)
def _create_user(self, login, groups, company, operating_units):
"""Creates a user."""
group_ids = [group.id for group in groups]
user = self.res_users_model.create(
{
"name": login,
"login": login,
"password": "demo",
"email": "example@yourcompany.com",
"company_id": company.id,
"company_ids": [(4, company.id)],
"operating_unit_ids": [(4, ou.id) for ou in operating_units],
"groups_id": [(6, 0, group_ids)],
}
)
return user
def _create_rma(self, uid, operating_unit=False):
"""Creates an RMA"""
if not operating_unit:
operating_unit = (
self.rma_model.sudo().with_user(uid)._default_operating_unit()
)
rma_order = (
self.rma_model.sudo()
.with_user(uid)
.create(
{
"operating_unit_id": operating_unit.id,
"partner_id": self.partner.id,
}
)
)
return rma_order
def _create_rma_line(self, rma, uid, operating_unit):
"""Creates an RMA"""
rma_order_line = (
self.rma_line_model.sudo()
.with_user(uid)
.create(
{
"operating_unit_id": operating_unit.id,
"operation_id": self.env.ref(
"rma.rma_operation_supplier_replace"
).id,
"rma_id": rma.id,
"partner_id": self.partner.id,
"in_route_id": 1,
"out_route_id": 1,
"in_warehouse_id": 1,
"out_warehouse_id": 1,
"location_id": 1,
"receipt_policy": "ordered",
"delivery_policy": "ordered",
"name": self.product.name,
"product_id": self.product.id,
"uom_id": self.product.uom_id.id,
}
)
)
return rma_order_line
def test_security(self):
# User 2 is only assigned to Operating Unit B2C, and cannot
# access RMA of Main Operating Unit.
record = (
self.rma_model.sudo()
.with_user(self.user2.id)
.search(
[
("id", "=", self.rma_order1.id),
("operating_unit_id", "=", self.main_OU.id),
]
)
)
self.assertEqual(
record.ids,
[],
"User 2 should not have access to " "OU %s." % self.main_OU.name,
)
def test_constraint(self):
# RMA group should contain rma lines for the same OU
with self.assertRaises(exceptions.ValidationError):
self._create_rma_line(self.rma_order1, self.user1.id, self.main_OU)
self._create_rma_line(self.rma_order1, self.user1.id, self.b2c_OU)
self.rma_order1._check_operating_unit()

View File

@@ -0,0 +1,84 @@
<?xml version="1.0" ?>
<!-- Copyright 2017-23 ForgeFlow S.L.
License LGPL-3.0 or later (http://www.gnu.org/licenses/lgpl-3.0) -->
<odoo>
<record id="view_rma_line_tree" model="ir.ui.view">
<field name="name">rma.order.line.tree</field>
<field name="model">rma.order.line</field>
<field name="inherit_id" ref="rma.view_rma_line_tree" />
<field name="arch" type="xml">
<field name="state" position="after">
<field
name="operating_unit_id"
groups="operating_unit.group_multi_operating_unit"
/>
</field>
</field>
</record>
<record id="view_rma_line_supplier_tree" model="ir.ui.view">
<field name="name">rma.order.line.supplier.tree</field>
<field name="model">rma.order.line</field>
<field name="inherit_id" ref="rma.view_rma_line_supplier_tree" />
<field name="arch" type="xml">
<field name="state" position="after">
<field
name="operating_unit_id"
groups="operating_unit.group_multi_operating_unit"
/>
</field>
</field>
</record>
<record id="view_rma_line_supplier_form" model="ir.ui.view">
<field name="name">rma.order.line.supplier.form</field>
<field name="model">rma.order.line</field>
<field name="inherit_id" ref="rma.view_rma_line_supplier_form" />
<field name="arch" type="xml">
<field name="origin" position="after">
<field
name="operating_unit_id"
groups="operating_unit.group_multi_operating_unit"
/>
</field>
<field name="operation_id" position="attributes">
<attribute
name="domain"
>[('type','=','supplier'),('out_warehouse_id.operating_unit_id', '=', operating_unit_id)]</attribute>
</field>
</field>
</record>
<record id="view_rma_line_form" model="ir.ui.view">
<field name="name">rma.order.line.form</field>
<field name="model">rma.order.line</field>
<field name="inherit_id" ref="rma.view_rma_line_form" />
<field name="arch" type="xml">
<field name="origin" position="after">
<field
name="operating_unit_id"
groups="operating_unit.group_multi_operating_unit"
/>
</field>
<field name="operation_id" position="attributes">
<attribute
name="domain"
>[('type','=','customer'),('in_warehouse_id.operating_unit_id', '=', operating_unit_id)]</attribute>
</field>
</field>
</record>
<record id="view_rma_rma_line_filter" model="ir.ui.view">
<field name="name">rma.order.line.select</field>
<field name="model">rma.order.line</field>
<field name="inherit_id" ref="rma.view_rma_rma_line_filter" />
<field name="arch" type="xml">
<field name="partner_id" position="after">
<field
name="operating_unit_id"
groups="operating_unit.group_multi_operating_unit"
/>
</field>
</field>
</record>
</odoo>

View File

@@ -0,0 +1,85 @@
<?xml version="1.0" ?>
<!-- Copyright 2017-23 ForgeFlow S.L.
License LGPL-3.0 or later (http://www.gnu.org/licenses/lgpl-3.0) -->
<odoo>
<record id="view_rma_tree" model="ir.ui.view">
<field name="name">rma.order.tree</field>
<field name="model">rma.order</field>
<field name="inherit_id" ref="rma.view_rma_tree" />
<field name="arch" type="xml">
<field name="date_rma" position="after">
<field
name="operating_unit_id"
groups="operating_unit.group_multi_operating_unit"
/>
</field>
</field>
</record>
<record id="view_rma_supplier_tree" model="ir.ui.view">
<field name="name">rma.order.supplier.tree</field>
<field name="model">rma.order</field>
<field name="inherit_id" ref="rma.view_rma_supplier_tree" />
<field name="arch" type="xml">
<field name="date_rma" position="after">
<field
name="operating_unit_id"
groups="operating_unit.group_multi_operating_unit"
/>
</field>
</field>
</record>
<record id="view_rma_supplier_form" model="ir.ui.view">
<field name="name">rma.order.supplier.form</field>
<field name="model">rma.order</field>
<field name="inherit_id" ref="rma.view_rma_supplier_form" />
<field name="arch" type="xml">
<group name="comments" position="before">
<group name="company">
<field name="company_id" invisible="True" />
<field
name="operating_unit_id"
domain="[('company_id','=', company_id)]"
options="{'no_create': True}"
groups="operating_unit.group_multi_operating_unit"
/>
</group>
</group>
</field>
</record>
<record id="view_rma_form" model="ir.ui.view">
<field name="name">rma.order.form</field>
<field name="model">rma.order</field>
<field name="inherit_id" ref="rma.view_rma_form" />
<field name="arch" type="xml">
<group name="comments" position="before">
<group name="company">
<field name="company_id" invisible="True" />
<field
name="operating_unit_id"
domain="[('company_id','=', company_id)]"
options="{'no_create': True}"
groups="operating_unit.group_multi_operating_unit"
/>
</group>
</group>
</field>
</record>
<record id="view_rma_rma_filter" model="ir.ui.view">
<field name="name">rma.order.select</field>
<field name="model">rma.order</field>
<field name="inherit_id" ref="rma.view_rma_rma_filter" />
<field name="arch" type="xml">
<field name="partner_id" position="after">
<field
name="operating_unit_id"
groups="operating_unit.group_multi_operating_unit"
/>
</field>
</field>
</record>
</odoo>

View File

@@ -0,0 +1,45 @@
# Copyright 2017-23 ForgeFlow S.L.
# License LGPL-3.0 or later (https://www.gnu.org/licenses/lgpl.html).
from odoo import _, models
from odoo.exceptions import ValidationError
class RmaAddStockMove(models.TransientModel):
_inherit = "rma_add_stock_move"
_description = "Wizard to add rma lines from pickings"
def _prepare_rma_line_from_stock_move(self, sm, lot=False):
res = super(RmaAddStockMove, self)._prepare_rma_line_from_stock_move(sm, lot)
if self.env.context.get("customer"):
operation = (
sm.product_id.rma_customer_operation_id
or sm.product_id.categ_id.rma_customer_operation_id
)
else:
operation = (
sm.product_id.rma_supplier_operation_id
or sm.product_id.categ_id.rma_supplier_operation_id
)
if not operation:
operation = self.env["rma.operation"].search(
[("type", "=", self.rma_id.type)], limit=1
)
if not operation:
raise ValidationError(_("Please define an operation first"))
if not operation.in_warehouse_id or not operation.out_warehouse_id:
warehouse = self.env["stock.warehouse"].search(
[
("company_id", "=", self.rma_id.company_id.id),
("lot_rma_id", "!=", False),
("operating_unit_id", "=", self.rma_id.operating_unit_id.id),
],
limit=1,
)
if not warehouse:
raise ValidationError(
_("Please define a warehouse with a default RMA location")
)
res.update(warehouse_id=warehouse.id)
return res

View File

@@ -0,0 +1 @@
../../../../rma_operating_unit

View File

@@ -0,0 +1,6 @@
import setuptools
setuptools.setup(
setup_requires=['setuptools-odoo'],
odoo_addon=True,
)