Browse Source
Merge pull request #208 from guewen/8.0-add-partner_changeset
Merge pull request #208 from guewen/8.0-add-partner_changeset
[8.0] Add addon partner_changesetpull/288/head
Daniel Reis
9 years ago
committed by
GitHub
25 changed files with 2691 additions and 0 deletions
-
171partner_changeset/README.rst
-
4partner_changeset/__init__.py
-
23partner_changeset/__openerp__.py
-
51partner_changeset/demo/changeset_field_rule.xml
-
313partner_changeset/i18n/fr.po
-
289partner_changeset/i18n/partner_changeset.pot
-
7partner_changeset/models/__init__.py
-
144partner_changeset/models/changeset_field_rule.py
-
50partner_changeset/models/res_partner.py
-
514partner_changeset/models/res_partner_changeset.py
-
10partner_changeset/security/ir.model.access.csv
-
29partner_changeset/security/security.xml
-
BINpartner_changeset/static/description/icon.png
-
BINpartner_changeset/static/src/img/changeset.png
-
BINpartner_changeset/static/src/img/rules.png
-
6partner_changeset/tests/__init__.py
-
77partner_changeset/tests/common.py
-
95partner_changeset/tests/test_changeset_field_rule.py
-
279partner_changeset/tests/test_changeset_field_type.py
-
246partner_changeset/tests/test_changeset_flow.py
-
110partner_changeset/tests/test_changeset_origin.py
-
63partner_changeset/views/changeset_field_rule_views.xml
-
10partner_changeset/views/menu.xml
-
159partner_changeset/views/res_partner_changeset_views.xml
-
41partner_changeset/views/res_partner_views.xml
@ -0,0 +1,171 @@ |
|||
.. image:: https://img.shields.io/badge/licence-AGPL--3-blue.svg |
|||
:target: http://www.gnu.org/licenses/agpl-3.0-standalone.html |
|||
:alt: License: AGPL-3 |
|||
|
|||
================== |
|||
Partner Changesets |
|||
================== |
|||
|
|||
This module extends the functionality of partners. It allows to create |
|||
changesets that must be validated when a partner is modified instead of direct |
|||
modifications. Rules allow to configure which field must be validated. |
|||
|
|||
Configuration |
|||
============= |
|||
|
|||
Access Rights |
|||
------------- |
|||
|
|||
The changesets rules must be edited by users with the group ``Changesets |
|||
Configuration``. The changesets can be applied or canceled only by users |
|||
with the group ``Changesets Validations`` |
|||
|
|||
Changesets Rules |
|||
---------------- |
|||
|
|||
The changesets rules can be configured in ``Sales > Configuration > |
|||
Partner Changesets > Fields Rules``. For each partner field, an |
|||
action can be defined: |
|||
|
|||
* Auto: the changes made on this field are always applied |
|||
* Validate: the changes made on this field must be manually confirmed by |
|||
a 'Changesets User' user |
|||
* Never: the changes made on this field are always refused |
|||
|
|||
In any case, all the changes made by the users are always applied |
|||
directly on the users, but a 'validated' changeset is created for the |
|||
history. |
|||
|
|||
The supported fields are: |
|||
|
|||
* Char |
|||
* Text |
|||
* Date |
|||
* Datetime |
|||
* Integer |
|||
* Float |
|||
* Boolean |
|||
* Many2one |
|||
|
|||
Rules can be global (no source model) or configured by source model. |
|||
Rules by source model have the priority. If a field is not configured |
|||
for the source model, it will use the global rule (if existing). |
|||
|
|||
If a field has no rule, it is written to the partner without changeset. |
|||
|
|||
Usage |
|||
===== |
|||
|
|||
General case |
|||
------------ |
|||
|
|||
The first step is to create the changeset rules, once that done, writes on |
|||
partners will be created as changesets. |
|||
|
|||
Finding changesets |
|||
------------------ |
|||
|
|||
A menu lists all the changesets in ``Sales > Configuration > Partner |
|||
Changesets > Changesets``. |
|||
|
|||
However, it is more convenient to access them directly from the |
|||
partners. Pending changesets can be accessed directly from the top right |
|||
of the partners' view. A new filter on the partners shows the partners |
|||
having at least one pending changeset. |
|||
|
|||
Handling changesets |
|||
------------------- |
|||
|
|||
A changeset shows the list of the changes made on a partner. Some of the |
|||
changes may be 'Pending', some 'Accepted' or 'Rejected' according to the |
|||
changeset rules. The only changes that need an action from the user are |
|||
'Pending' changes. When a change is accepted, the value is written on |
|||
the user. |
|||
|
|||
The changes view shows the name of the partner's field, the Origin value |
|||
and the New value alongside the state of the change. By clicking on the |
|||
change in some cases a more detailed view is displayed, for instance, |
|||
links for relations can be clicked on. |
|||
|
|||
A button on a changeset allows to apply or reject all the changes at |
|||
once. |
|||
|
|||
Custom source rules in your addon |
|||
--------------------------------- |
|||
|
|||
Addons wanting to create changeset with their own rules should pass the |
|||
following keys in the context when they write on the partner: |
|||
|
|||
* ``__changeset_rules_source_model``: name of the model which asks for |
|||
the change |
|||
* ``__changeset_rules_source_id``: id of the record which asks for the |
|||
change |
|||
|
|||
Also, they should extend the selection in |
|||
``ChangesetFieldRule._domain_source_models`` to add their model (the |
|||
same that is passed in ``__changeset_rules_source_model``). |
|||
|
|||
The source is used for the application of the rules, allowing to have a |
|||
different rule for a different source. It is also stored on the changeset for |
|||
information. |
|||
|
|||
Screenshot: |
|||
----------- |
|||
|
|||
* Configuration of rules |
|||
|
|||
.. image:: partner_changeset/static/src/img/rules.png |
|||
|
|||
* Changeset waiting for validation |
|||
|
|||
.. image:: partner_changeset/static/src/img/changeset.png |
|||
|
|||
|
|||
.. image:: https://odoo-community.org/website/image/ir.attachment/5784_f2813bd/datas |
|||
:alt: Try me on Runbot |
|||
:target: https://runbot.odoo-community.org/runbot/134/8.0 |
|||
|
|||
Known issues / Roadmap |
|||
====================== |
|||
|
|||
* Only a subset of the type of fields is actually supported |
|||
|
|||
Bug Tracker |
|||
=========== |
|||
|
|||
Bugs are tracked on `GitHub Issues |
|||
<https://github.com/OCA/partner-contact/issues>`_. In case of trouble, please |
|||
check there if your issue has already been reported. If you spotted it first, |
|||
help us smashing it by providing a detailed and welcomed `feedback |
|||
<https://github.com/OCA/ |
|||
partner-contact/issues/new?body=module:%20 |
|||
partner_changeset%0Aversion:%20 |
|||
8.0%0A%0A**Steps%20to%20reproduce**%0A-%20...%0A%0A**Current%20behavior**%0A%0A**Expected%20behavior**>`_. |
|||
|
|||
Credits |
|||
======= |
|||
|
|||
Images |
|||
------ |
|||
|
|||
* Odoo Community Association: `Icon <https://github.com/OCA/maintainer-tools/blob/master/template/module/static/description/icon.svg>`_. |
|||
|
|||
Contributors |
|||
------------ |
|||
|
|||
* Guewen Baconnier <guewen.baconnier@camptocamp.com> |
|||
|
|||
Maintainer |
|||
---------- |
|||
|
|||
.. image:: https://odoo-community.org/logo.png |
|||
:alt: Odoo Community Association |
|||
:target: https://odoo-community.org |
|||
|
|||
This module is maintained by the OCA. |
|||
|
|||
OCA, or the Odoo Community Association, is a nonprofit organization whose |
|||
mission is to support the collaborative development of Odoo features and |
|||
promote its widespread use. |
|||
|
|||
To contribute to this module, please visit https://odoo-community.org. |
@ -0,0 +1,4 @@ |
|||
# -*- coding: utf-8 -*- |
|||
# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl.html). |
|||
|
|||
from . import models |
@ -0,0 +1,23 @@ |
|||
# -*- coding: utf-8 -*- |
|||
# © 2015 Camptocamp SA |
|||
# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl.html). |
|||
|
|||
{'name': 'Partner Changesets', |
|||
'version': '8.0.1.0.0', |
|||
'author': 'Camptocamp, Odoo Community Association (OCA)', |
|||
'license': 'AGPL-3', |
|||
'category': 'Sales Management', |
|||
'depends': ['base', |
|||
], |
|||
'website': 'http://www.camptocamp.com', |
|||
'data': ['security/security.xml', |
|||
'security/ir.model.access.csv', |
|||
'views/menu.xml', |
|||
'views/res_partner_changeset_views.xml', |
|||
'views/changeset_field_rule_views.xml', |
|||
'views/res_partner_views.xml', |
|||
], |
|||
'demo': ['demo/changeset_field_rule.xml', |
|||
], |
|||
'installable': True, |
|||
} |
@ -0,0 +1,51 @@ |
|||
<?xml version="1.0" encoding="utf-8"?> |
|||
<openerp> |
|||
<data noupdate="1"> |
|||
|
|||
<record model="changeset.field.rule" id="changeset_field_rule_name"> |
|||
<field name="field_id" ref="base.field_res_partner_name"/> |
|||
<field name="action">auto</field> |
|||
</record> |
|||
|
|||
<record model="changeset.field.rule" id="changeset_field_rule_street"> |
|||
<field name="field_id" ref="base.field_res_partner_street"/> |
|||
<field name="action">validate</field> |
|||
</record> |
|||
|
|||
<record model="changeset.field.rule" id="changeset_field_rule_street2"> |
|||
<field name="field_id" ref="base.field_res_partner_street2"/> |
|||
<field name="action">validate</field> |
|||
</record> |
|||
|
|||
<record model="changeset.field.rule" id="changeset_field_rule_zip"> |
|||
<field name="field_id" ref="base.field_res_partner_zip"/> |
|||
<field name="action">validate</field> |
|||
</record> |
|||
|
|||
<record model="changeset.field.rule" id="changeset_field_rule_city"> |
|||
<field name="field_id" ref="base.field_res_partner_city"/> |
|||
<field name="action">validate</field> |
|||
</record> |
|||
|
|||
<record model="changeset.field.rule" id="changeset_field_rule_email"> |
|||
<field name="field_id" ref="base.field_res_partner_email"/> |
|||
<field name="action">never</field> |
|||
</record> |
|||
|
|||
<record model="changeset.field.rule" id="changeset_field_rule_ref"> |
|||
<field name="field_id" ref="base.field_res_partner_ref"/> |
|||
<field name="action">validate</field> |
|||
</record> |
|||
|
|||
<record model="changeset.field.rule" id="changeset_field_rule_country_id"> |
|||
<field name="field_id" ref="base.field_res_partner_country_id"/> |
|||
<field name="action">validate</field> |
|||
</record> |
|||
|
|||
<record model="changeset.field.rule" id="changeset_field_rule_credit_limit"> |
|||
<field name="field_id" ref="base.field_res_partner_credit_limit"/> |
|||
<field name="action">validate</field> |
|||
</record> |
|||
|
|||
</data> |
|||
</openerp> |
@ -0,0 +1,313 @@ |
|||
# Translation of Odoo Server. |
|||
# This file contains the translation of the following modules: |
|||
# * partner_changeset |
|||
# |
|||
msgid "" |
|||
msgstr "" |
|||
"Project-Id-Version: Odoo Server 8.0\n" |
|||
"Report-Msgid-Bugs-To: \n" |
|||
"POT-Creation-Date: 2015-11-25 08:31+0000\n" |
|||
"PO-Revision-Date: 2015-09-18 14:48+0000\n" |
|||
"Last-Translator: <>\n" |
|||
"Language-Team: \n" |
|||
"Language: \n" |
|||
"MIME-Version: 1.0\n" |
|||
"Content-Type: text/plain; charset=UTF-8\n" |
|||
"Content-Transfer-Encoding: \n" |
|||
"Plural-Forms: \n" |
|||
|
|||
#. module: partner_changeset |
|||
#: sql_constraint:changeset.field.rule:0 |
|||
msgid "A rule already exists for this field." |
|||
msgstr "Une règle existe déjà pour ce champ." |
|||
|
|||
#. module: partner_changeset |
|||
#: selection:res.partner.changeset.change,state:0 |
|||
msgid "Accepted" |
|||
msgstr "Accepté" |
|||
|
|||
#. module: partner_changeset |
|||
#: field:changeset.field.rule,action:0 |
|||
msgid "Action" |
|||
msgstr "Action" |
|||
|
|||
#. module: partner_changeset |
|||
#: view:res.partner.changeset:partner_changeset.view_res_partner_changeset_form |
|||
#: view:res.partner.changeset.change:partner_changeset.view_res_partner_changeset_change_form |
|||
msgid "Apply" |
|||
msgstr "Appliquer" |
|||
|
|||
#. module: partner_changeset |
|||
#: view:res.partner.changeset:partner_changeset.view_res_partner_changeset_form |
|||
msgid "Apply pending changes" |
|||
msgstr "Appliquer les modifications en attente" |
|||
|
|||
#. module: partner_changeset |
|||
#: help:changeset.field.rule,action:0 |
|||
msgid "" |
|||
"Auto: always apply a change.\n" |
|||
"Validate: manually applied by an administrator.\n" |
|||
"Never: change never applied." |
|||
msgstr "" |
|||
"Auto: toujours appliquer une modification.\n" |
|||
"Validation: modification manuellement appliquée par un administrateur.\n" |
|||
"Jamais: modification jamais appliquée." |
|||
|
|||
#. module: partner_changeset |
|||
#: view:res.partner.changeset:partner_changeset.view_res_partner_changeset_form |
|||
#: view:res.partner:partner_changeset.res_partner_view_buttons |
|||
#: field:res.partner.changeset,change_ids:0 |
|||
msgid "Changes" |
|||
msgstr "Modifications" |
|||
|
|||
#. module: partner_changeset |
|||
#: field:res.partner.changeset.change,changeset_id:0 |
|||
msgid "Changeset" |
|||
msgstr "Jeu de modifications" |
|||
|
|||
#. module: partner_changeset |
|||
#: model:res.groups,name:partner_changeset.group_changeset_manager |
|||
msgid "Changeset Configuration" |
|||
msgstr "Configuration des modifications" |
|||
|
|||
#. module: partner_changeset |
|||
#: model:ir.model,name:partner_changeset.model_changeset_field_rule |
|||
msgid "Changeset Field Rules" |
|||
msgstr "Règles de modifications par champ" |
|||
|
|||
#. module: partner_changeset |
|||
#: view:changeset.field.rule:partner_changeset.view_changeset_field_rule_form |
|||
#: view:changeset.field.rule:partner_changeset.view_changeset_field_rule_search |
|||
#: view:changeset.field.rule:partner_changeset.view_changeset_field_rule_tree |
|||
#: model:ir.actions.act_window,name:partner_changeset.action_changeset_field_rule_view |
|||
msgid "Changeset Fields Rules" |
|||
msgstr "Règles de modifications par champ" |
|||
|
|||
#. module: partner_changeset |
|||
#: model:ir.ui.menu,name:partner_changeset.menu_res_partner_changeset |
|||
#: field:res.partner,changeset_ids:0 |
|||
msgid "Changesets" |
|||
msgstr "Modifications" |
|||
|
|||
#. module: partner_changeset |
|||
#: model:res.groups,name:partner_changeset.group_changeset_user |
|||
msgid "Changesets Validations " |
|||
msgstr "Validations des modifications" |
|||
|
|||
#. module: partner_changeset |
|||
#: field:changeset.field.rule,create_uid:0 |
|||
#: field:res.partner.changeset,create_uid:0 |
|||
#: field:res.partner.changeset.change,create_uid:0 |
|||
msgid "Created by" |
|||
msgstr "Créé par" |
|||
|
|||
#. module: partner_changeset |
|||
#: field:changeset.field.rule,create_date:0 |
|||
#: field:res.partner.changeset,create_date:0 |
|||
#: field:res.partner.changeset.change,create_date:0 |
|||
msgid "Created on" |
|||
msgstr "Créé le" |
|||
|
|||
#. module: partner_changeset |
|||
#: field:res.partner.changeset,date:0 |
|||
msgid "Date" |
|||
msgstr "Date" |
|||
|
|||
#. module: partner_changeset |
|||
#: view:res.partner.changeset:partner_changeset.view_res_partner_changeset_search |
|||
#: selection:res.partner.changeset,state:0 |
|||
msgid "Done" |
|||
msgstr "Fait" |
|||
|
|||
#. module: partner_changeset |
|||
#: field:changeset.field.rule,field_id:0 |
|||
#: field:res.partner.changeset.change,field_id:0 |
|||
msgid "Field" |
|||
msgstr "Champ" |
|||
|
|||
#. module: partner_changeset |
|||
#: model:ir.ui.menu,name:partner_changeset.menu_changeset_field_rule |
|||
msgid "Field Rules" |
|||
msgstr "Règle par champ" |
|||
|
|||
#. module: partner_changeset |
|||
#: view:res.partner.changeset:partner_changeset.view_res_partner_changeset_search |
|||
msgid "Group By" |
|||
msgstr "Grouper par" |
|||
|
|||
#. module: partner_changeset |
|||
#: field:changeset.field.rule,id:0 field:res.partner.changeset,id:0 |
|||
#: field:res.partner.changeset.change,id:0 |
|||
msgid "ID" |
|||
msgstr "Id." |
|||
|
|||
#. module: partner_changeset |
|||
#: help:changeset.field.rule,source_model_id:0 |
|||
msgid "" |
|||
"If a source model is defined, the rule will be applied only when the change " |
|||
"is made from this origin. Rules without source model are global and applies " |
|||
"to all backends.\n" |
|||
"Rules with a source model have precedence over global rules, but if a field " |
|||
"has no rule with a source model, the global rule is used." |
|||
msgstr "" |
|||
"Si un modèle source est défini, la règle ne sera appliquée que lorsque le " |
|||
"changement vient de cette source. Les règles sans modèle source sont " |
|||
"globales et sont appliquées à toutes les sources.\n " |
|||
"Les règles avec une souce ont la précédence sur les règles globales. Si un " |
|||
"champ n'a pas de règle avec source, la règle globale est utilisée." |
|||
|
|||
#. module: partner_changeset |
|||
#: field:changeset.field.rule,write_uid:0 |
|||
#: field:res.partner.changeset,write_uid:0 |
|||
#: field:res.partner.changeset.change,write_uid:0 |
|||
msgid "Last Updated by" |
|||
msgstr "" |
|||
|
|||
#. module: partner_changeset |
|||
#: field:changeset.field.rule,write_date:0 |
|||
#: field:res.partner.changeset,write_date:0 |
|||
#: field:res.partner.changeset.change,write_date:0 |
|||
msgid "Last Updated on" |
|||
msgstr "" |
|||
|
|||
#. module: partner_changeset |
|||
#: field:res.partner.changeset.change,new_value_boolean:0 |
|||
#: field:res.partner.changeset.change,new_value_char:0 |
|||
#: field:res.partner.changeset.change,new_value_date:0 |
|||
#: field:res.partner.changeset.change,new_value_datetime:0 |
|||
#: field:res.partner.changeset.change,new_value_float:0 |
|||
#: field:res.partner.changeset.change,new_value_integer:0 |
|||
#: field:res.partner.changeset.change,new_value_reference:0 |
|||
#: field:res.partner.changeset.change,new_value_text:0 |
|||
msgid "New" |
|||
msgstr "Nouveau" |
|||
|
|||
#. module: partner_changeset |
|||
#: field:res.partner.changeset,note:0 |
|||
msgid "Note" |
|||
msgstr "Note" |
|||
|
|||
#. module: partner_changeset |
|||
#: field:res.partner.changeset.change,old_value_boolean:0 |
|||
#: field:res.partner.changeset.change,old_value_char:0 |
|||
#: field:res.partner.changeset.change,old_value_date:0 |
|||
#: field:res.partner.changeset.change,old_value_datetime:0 |
|||
#: field:res.partner.changeset.change,old_value_float:0 |
|||
#: field:res.partner.changeset.change,old_value_integer:0 |
|||
#: field:res.partner.changeset.change,old_value_reference:0 |
|||
#: field:res.partner.changeset.change,old_value_text:0 |
|||
msgid "Old" |
|||
msgstr "Ancien" |
|||
|
|||
#. module: partner_changeset |
|||
#: model:ir.model,name:partner_changeset.model_res_partner |
|||
#: view:res.partner.changeset:partner_changeset.view_res_partner_changeset_search |
|||
#: field:res.partner.changeset,partner_id:0 |
|||
msgid "Partner" |
|||
msgstr "Partenaire" |
|||
|
|||
#. module: partner_changeset |
|||
#: model:ir.actions.act_window,name:partner_changeset.action_res_partner_changeset_view |
|||
#: model:ir.model,name:partner_changeset.model_res_partner_changeset |
|||
#: view:res.partner.changeset:partner_changeset.view_res_partner_changeset_form |
|||
#: view:res.partner.changeset:partner_changeset.view_res_partner_changeset_search |
|||
#: view:res.partner.changeset:partner_changeset.view_res_partner_changeset_tree |
|||
msgid "Partner Changeset" |
|||
msgstr "Modifications de partenaire" |
|||
|
|||
#. module: partner_changeset |
|||
#: model:ir.model,name:partner_changeset.model_res_partner_changeset_change |
|||
#: view:res.partner.changeset:partner_changeset.view_res_partner_changeset_form |
|||
#: view:res.partner.changeset.change:partner_changeset.view_res_partner_changeset_change_form |
|||
msgid "Partner Changeset Change" |
|||
msgstr "Modification de partenaire" |
|||
|
|||
#. module: partner_changeset |
|||
#: model:ir.ui.menu,name:partner_changeset.menu_changeset |
|||
msgid "Partner Changesets" |
|||
msgstr "Modifications de partenaire" |
|||
|
|||
#. module: partner_changeset |
|||
#: view:res.partner.changeset:partner_changeset.view_res_partner_changeset_search |
|||
#: selection:res.partner.changeset,state:0 |
|||
#: selection:res.partner.changeset.change,state:0 |
|||
msgid "Pending" |
|||
msgstr "En attente" |
|||
|
|||
#. module: partner_changeset |
|||
#: view:res.partner:partner_changeset.view_res_partner_filter |
|||
msgid "Pending Changesets" |
|||
msgstr "Modifications en attente" |
|||
|
|||
#. module: partner_changeset |
|||
#: view:res.partner.changeset:partner_changeset.view_res_partner_changeset_form |
|||
msgid "Previous" |
|||
msgstr "Précédent" |
|||
|
|||
#. module: partner_changeset |
|||
#: view:res.partner.changeset:partner_changeset.view_res_partner_changeset_form |
|||
#: view:res.partner.changeset.change:partner_changeset.view_res_partner_changeset_change_form |
|||
msgid "Reject" |
|||
msgstr "Rejeter" |
|||
|
|||
#. module: partner_changeset |
|||
#: view:res.partner.changeset:partner_changeset.view_res_partner_changeset_form |
|||
msgid "Reject pending changes" |
|||
msgstr "Rejeter les modifications en attente" |
|||
|
|||
#. module: partner_changeset |
|||
#: selection:res.partner.changeset.change,state:0 |
|||
msgid "Rejected" |
|||
msgstr "Rejeté" |
|||
|
|||
#. module: partner_changeset |
|||
#: field:changeset.field.rule,source_model_id:0 |
|||
msgid "Source Model" |
|||
msgstr "Modèle source" |
|||
|
|||
#. module: partner_changeset |
|||
#: field:res.partner.changeset,source:0 |
|||
msgid "Source of the change" |
|||
msgstr "Source du changement" |
|||
|
|||
#. module: partner_changeset |
|||
#: view:res.partner.changeset:partner_changeset.view_res_partner_changeset_search |
|||
#: field:res.partner.changeset,state:0 |
|||
#: field:res.partner.changeset.change,state:0 |
|||
msgid "State" |
|||
msgstr "État" |
|||
|
|||
#. module: partner_changeset |
|||
#: model:res.groups,comment:partner_changeset.group_changeset_user |
|||
msgid "The user will be able to apply or reject changesets." |
|||
msgstr "L'utilisateur pourra appliquer ou rejeter les jeux de modifications." |
|||
|
|||
#. module: partner_changeset |
|||
#: model:res.groups,comment:partner_changeset.group_changeset_manager |
|||
msgid "" |
|||
"The user will have an access to the configuration of the changeset rules." |
|||
msgstr "" |
|||
"L'utilisateur aura accès à la configuration des règles de jeux de " |
|||
"modifications." |
|||
|
|||
#. module: partner_changeset |
|||
#: code:addons/partner_changeset/models/res_partner_changeset.py:418 |
|||
#, python-format |
|||
msgid "" |
|||
"This change cannot be applied because a previous changeset for the same " |
|||
"partner is pending.\n" |
|||
"Apply all the anterior changesets before applying this one." |
|||
msgstr "" |
|||
"The changement ne peux pas être appliqué car un précédent jeu de " |
|||
"modification pour le même partenaire est toujours en attente.\n" |
|||
"Il est nécessaire d'appliquer tous les précédents jeux de modification avant " |
|||
"d'applique celui-ci." |
|||
|
|||
#. module: partner_changeset |
|||
#: code:addons/partner_changeset/models/res_partner_changeset.py:433 |
|||
#, python-format |
|||
msgid "This change has already be applied." |
|||
msgstr "The changement est déjà appliqué." |
|||
|
|||
#~ msgid "Model" |
|||
#~ msgstr "Modèle" |
@ -0,0 +1,289 @@ |
|||
# Translation of Odoo Server. |
|||
# This file contains the translation of the following modules: |
|||
# * partner_changeset |
|||
# |
|||
msgid "" |
|||
msgstr "" |
|||
"Project-Id-Version: Odoo Server 8.0\n" |
|||
"Report-Msgid-Bugs-To: \n" |
|||
"POT-Creation-Date: 2015-11-25 08:31+0000\n" |
|||
"PO-Revision-Date: 2015-11-25 08:31+0000\n" |
|||
"Last-Translator: <>\n" |
|||
"Language-Team: \n" |
|||
"MIME-Version: 1.0\n" |
|||
"Content-Type: text/plain; charset=UTF-8\n" |
|||
"Content-Transfer-Encoding: \n" |
|||
"Plural-Forms: \n" |
|||
|
|||
#. module: partner_changeset |
|||
#: sql_constraint:changeset.field.rule:0 |
|||
msgid "A rule already exists for this field." |
|||
msgstr "" |
|||
|
|||
#. module: partner_changeset |
|||
#: selection:res.partner.changeset.change,state:0 |
|||
msgid "Accepted" |
|||
msgstr "" |
|||
|
|||
#. module: partner_changeset |
|||
#: field:changeset.field.rule,action:0 |
|||
msgid "Action" |
|||
msgstr "" |
|||
|
|||
#. module: partner_changeset |
|||
#: view:res.partner.changeset:partner_changeset.view_res_partner_changeset_form |
|||
#: view:res.partner.changeset.change:partner_changeset.view_res_partner_changeset_change_form |
|||
msgid "Apply" |
|||
msgstr "" |
|||
|
|||
#. module: partner_changeset |
|||
#: view:res.partner.changeset:partner_changeset.view_res_partner_changeset_form |
|||
msgid "Apply pending changes" |
|||
msgstr "" |
|||
|
|||
#. module: partner_changeset |
|||
#: help:changeset.field.rule,action:0 |
|||
msgid "Auto: always apply a change.\n" |
|||
"Validate: manually applied by an administrator.\n" |
|||
"Never: change never applied." |
|||
msgstr "" |
|||
|
|||
#. module: partner_changeset |
|||
#: view:res.partner.changeset:partner_changeset.view_res_partner_changeset_form |
|||
#: view:res.partner:partner_changeset.res_partner_view_buttons |
|||
#: field:res.partner.changeset,change_ids:0 |
|||
msgid "Changes" |
|||
msgstr "" |
|||
|
|||
#. module: partner_changeset |
|||
#: field:res.partner.changeset.change,changeset_id:0 |
|||
msgid "Changeset" |
|||
msgstr "" |
|||
|
|||
#. module: partner_changeset |
|||
#: model:res.groups,name:partner_changeset.group_changeset_manager |
|||
msgid "Changeset Configuration" |
|||
msgstr "" |
|||
|
|||
#. module: partner_changeset |
|||
#: model:ir.model,name:partner_changeset.model_changeset_field_rule |
|||
msgid "Changeset Field Rules" |
|||
msgstr "" |
|||
|
|||
#. module: partner_changeset |
|||
#: view:changeset.field.rule:partner_changeset.view_changeset_field_rule_form |
|||
#: view:changeset.field.rule:partner_changeset.view_changeset_field_rule_search |
|||
#: view:changeset.field.rule:partner_changeset.view_changeset_field_rule_tree |
|||
#: model:ir.actions.act_window,name:partner_changeset.action_changeset_field_rule_view |
|||
msgid "Changeset Fields Rules" |
|||
msgstr "" |
|||
|
|||
#. module: partner_changeset |
|||
#: model:ir.ui.menu,name:partner_changeset.menu_res_partner_changeset |
|||
#: field:res.partner,changeset_ids:0 |
|||
msgid "Changesets" |
|||
msgstr "" |
|||
|
|||
#. module: partner_changeset |
|||
#: model:res.groups,name:partner_changeset.group_changeset_user |
|||
msgid "Changesets Validations " |
|||
msgstr "" |
|||
|
|||
#. module: partner_changeset |
|||
#: field:changeset.field.rule,create_uid:0 |
|||
#: field:res.partner.changeset,create_uid:0 |
|||
#: field:res.partner.changeset.change,create_uid:0 |
|||
msgid "Created by" |
|||
msgstr "" |
|||
|
|||
#. module: partner_changeset |
|||
#: field:changeset.field.rule,create_date:0 |
|||
#: field:res.partner.changeset,create_date:0 |
|||
#: field:res.partner.changeset.change,create_date:0 |
|||
msgid "Created on" |
|||
msgstr "" |
|||
|
|||
#. module: partner_changeset |
|||
#: field:res.partner.changeset,date:0 |
|||
msgid "Date" |
|||
msgstr "" |
|||
|
|||
#. module: partner_changeset |
|||
#: view:res.partner.changeset:partner_changeset.view_res_partner_changeset_search |
|||
#: selection:res.partner.changeset,state:0 |
|||
msgid "Done" |
|||
msgstr "" |
|||
|
|||
#. module: partner_changeset |
|||
#: field:changeset.field.rule,field_id:0 |
|||
#: field:res.partner.changeset.change,field_id:0 |
|||
msgid "Field" |
|||
msgstr "" |
|||
|
|||
#. module: partner_changeset |
|||
#: model:ir.ui.menu,name:partner_changeset.menu_changeset_field_rule |
|||
msgid "Field Rules" |
|||
msgstr "" |
|||
|
|||
#. module: partner_changeset |
|||
#: view:res.partner.changeset:partner_changeset.view_res_partner_changeset_search |
|||
msgid "Group By" |
|||
msgstr "" |
|||
|
|||
#. module: partner_changeset |
|||
#: field:changeset.field.rule,id:0 |
|||
#: field:res.partner.changeset,id:0 |
|||
#: field:res.partner.changeset.change,id:0 |
|||
msgid "ID" |
|||
msgstr "" |
|||
|
|||
#. module: partner_changeset |
|||
#: help:changeset.field.rule,source_model_id:0 |
|||
msgid "If a source model is defined, the rule will be applied only when the change is made from this origin. Rules without source model are global and applies to all backends.\n" |
|||
"Rules with a source model have precedence over global rules, but if a field has no rule with a source model, the global rule is used." |
|||
msgstr "" |
|||
|
|||
#. module: partner_changeset |
|||
#: field:changeset.field.rule,write_uid:0 |
|||
#: field:res.partner.changeset,write_uid:0 |
|||
#: field:res.partner.changeset.change,write_uid:0 |
|||
msgid "Last Updated by" |
|||
msgstr "" |
|||
|
|||
#. module: partner_changeset |
|||
#: field:changeset.field.rule,write_date:0 |
|||
#: field:res.partner.changeset,write_date:0 |
|||
#: field:res.partner.changeset.change,write_date:0 |
|||
msgid "Last Updated on" |
|||
msgstr "" |
|||
|
|||
#. module: partner_changeset |
|||
#: field:res.partner.changeset.change,new_value_boolean:0 |
|||
#: field:res.partner.changeset.change,new_value_char:0 |
|||
#: field:res.partner.changeset.change,new_value_date:0 |
|||
#: field:res.partner.changeset.change,new_value_datetime:0 |
|||
#: field:res.partner.changeset.change,new_value_float:0 |
|||
#: field:res.partner.changeset.change,new_value_integer:0 |
|||
#: field:res.partner.changeset.change,new_value_reference:0 |
|||
#: field:res.partner.changeset.change,new_value_text:0 |
|||
msgid "New" |
|||
msgstr "" |
|||
|
|||
#. module: partner_changeset |
|||
#: field:res.partner.changeset,note:0 |
|||
msgid "Note" |
|||
msgstr "" |
|||
|
|||
#. module: partner_changeset |
|||
#: field:res.partner.changeset.change,old_value_boolean:0 |
|||
#: field:res.partner.changeset.change,old_value_char:0 |
|||
#: field:res.partner.changeset.change,old_value_date:0 |
|||
#: field:res.partner.changeset.change,old_value_datetime:0 |
|||
#: field:res.partner.changeset.change,old_value_float:0 |
|||
#: field:res.partner.changeset.change,old_value_integer:0 |
|||
#: field:res.partner.changeset.change,old_value_reference:0 |
|||
#: field:res.partner.changeset.change,old_value_text:0 |
|||
msgid "Old" |
|||
msgstr "" |
|||
|
|||
#. module: partner_changeset |
|||
#: model:ir.model,name:partner_changeset.model_res_partner |
|||
#: view:res.partner.changeset:partner_changeset.view_res_partner_changeset_search |
|||
#: field:res.partner.changeset,partner_id:0 |
|||
msgid "Partner" |
|||
msgstr "" |
|||
|
|||
#. module: partner_changeset |
|||
#: model:ir.actions.act_window,name:partner_changeset.action_res_partner_changeset_view |
|||
#: model:ir.model,name:partner_changeset.model_res_partner_changeset |
|||
#: view:res.partner.changeset:partner_changeset.view_res_partner_changeset_form |
|||
#: view:res.partner.changeset:partner_changeset.view_res_partner_changeset_search |
|||
#: view:res.partner.changeset:partner_changeset.view_res_partner_changeset_tree |
|||
msgid "Partner Changeset" |
|||
msgstr "" |
|||
|
|||
#. module: partner_changeset |
|||
#: model:ir.model,name:partner_changeset.model_res_partner_changeset_change |
|||
#: view:res.partner.changeset:partner_changeset.view_res_partner_changeset_form |
|||
#: view:res.partner.changeset.change:partner_changeset.view_res_partner_changeset_change_form |
|||
msgid "Partner Changeset Change" |
|||
msgstr "" |
|||
|
|||
#. module: partner_changeset |
|||
#: model:ir.ui.menu,name:partner_changeset.menu_changeset |
|||
msgid "Partner Changesets" |
|||
msgstr "" |
|||
|
|||
#. module: partner_changeset |
|||
#: view:res.partner.changeset:partner_changeset.view_res_partner_changeset_search |
|||
#: selection:res.partner.changeset,state:0 |
|||
#: selection:res.partner.changeset.change,state:0 |
|||
msgid "Pending" |
|||
msgstr "" |
|||
|
|||
#. module: partner_changeset |
|||
#: view:res.partner:partner_changeset.view_res_partner_filter |
|||
msgid "Pending Changesets" |
|||
msgstr "" |
|||
|
|||
#. module: partner_changeset |
|||
#: view:res.partner.changeset:partner_changeset.view_res_partner_changeset_form |
|||
msgid "Previous" |
|||
msgstr "" |
|||
|
|||
#. module: partner_changeset |
|||
#: view:res.partner.changeset:partner_changeset.view_res_partner_changeset_form |
|||
#: view:res.partner.changeset.change:partner_changeset.view_res_partner_changeset_change_form |
|||
msgid "Reject" |
|||
msgstr "" |
|||
|
|||
#. module: partner_changeset |
|||
#: view:res.partner.changeset:partner_changeset.view_res_partner_changeset_form |
|||
msgid "Reject pending changes" |
|||
msgstr "" |
|||
|
|||
#. module: partner_changeset |
|||
#: selection:res.partner.changeset.change,state:0 |
|||
msgid "Rejected" |
|||
msgstr "" |
|||
|
|||
#. module: partner_changeset |
|||
#: field:changeset.field.rule,source_model_id:0 |
|||
msgid "Source Model" |
|||
msgstr "" |
|||
|
|||
#. module: partner_changeset |
|||
#: field:res.partner.changeset,source:0 |
|||
msgid "Source of the change" |
|||
msgstr "" |
|||
|
|||
#. module: partner_changeset |
|||
#: view:res.partner.changeset:partner_changeset.view_res_partner_changeset_search |
|||
#: field:res.partner.changeset,state:0 |
|||
#: field:res.partner.changeset.change,state:0 |
|||
msgid "State" |
|||
msgstr "" |
|||
|
|||
#. module: partner_changeset |
|||
#: model:res.groups,comment:partner_changeset.group_changeset_user |
|||
msgid "The user will be able to apply or reject changesets." |
|||
msgstr "" |
|||
|
|||
#. module: partner_changeset |
|||
#: model:res.groups,comment:partner_changeset.group_changeset_manager |
|||
msgid "The user will have an access to the configuration of the changeset rules." |
|||
msgstr "" |
|||
|
|||
#. module: partner_changeset |
|||
#: code:addons/partner_changeset/models/res_partner_changeset.py:418 |
|||
#, python-format |
|||
msgid "This change cannot be applied because a previous changeset for the same partner is pending.\n" |
|||
"Apply all the anterior changesets before applying this one." |
|||
msgstr "" |
|||
|
|||
#. module: partner_changeset |
|||
#: code:addons/partner_changeset/models/res_partner_changeset.py:433 |
|||
#, python-format |
|||
msgid "This change has already be applied." |
|||
msgstr "" |
|||
|
@ -0,0 +1,7 @@ |
|||
# -*- coding: utf-8 -*- |
|||
# © 2015 Camptocamp SA |
|||
# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl.html). |
|||
|
|||
from . import res_partner |
|||
from . import res_partner_changeset |
|||
from . import changeset_field_rule |
@ -0,0 +1,144 @@ |
|||
# -*- coding: utf-8 -*- |
|||
# © 2015 Camptocamp SA |
|||
# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl.html). |
|||
|
|||
from openerp import models, fields, api |
|||
from openerp.tools.cache import ormcache |
|||
|
|||
|
|||
class ChangesetFieldRule(models.Model): |
|||
_name = 'changeset.field.rule' |
|||
_description = 'Changeset Field Rules' |
|||
_rec_name = 'field_id' |
|||
|
|||
field_id = fields.Many2one( |
|||
comodel_name='ir.model.fields', |
|||
string='Field', |
|||
domain="[('model_id.model', '=', 'res.partner'), " |
|||
" ('ttype', 'in', ('char', 'selection', 'date', 'datetime', " |
|||
" 'float', 'integer', 'text', 'boolean', " |
|||
" 'many2one')), " |
|||
" ('readonly', '=', False)]", |
|||
ondelete='cascade', |
|||
required=True, |
|||
) |
|||
action = fields.Selection( |
|||
selection='_selection_action', |
|||
string='Action', |
|||
required=True, |
|||
help="Auto: always apply a change.\n" |
|||
"Validate: manually applied by an administrator.\n" |
|||
"Never: change never applied.", |
|||
) |
|||
source_model_id = fields.Many2one( |
|||
comodel_name='ir.model', |
|||
string='Source Model', |
|||
ondelete='cascade', |
|||
domain=lambda self: [('id', 'in', self._domain_source_models().ids)], |
|||
help="If a source model is defined, the rule will be applied only " |
|||
"when the change is made from this origin. " |
|||
"Rules without source model are global and applies to all " |
|||
"backends.\n" |
|||
"Rules with a source model have precedence over global rules, " |
|||
"but if a field has no rule with a source model, the global rule " |
|||
"is used." |
|||
) |
|||
|
|||
_sql_constraints = [ |
|||
('model_field_uniq', |
|||
'unique (source_model_id, field_id)', |
|||
'A rule already exists for this field.'), |
|||
] |
|||
|
|||
@api.model |
|||
def _domain_source_models(self): |
|||
""" Returns the models for which we can define rules. |
|||
|
|||
Example for submodules (replace by the xmlid of the model): |
|||
|
|||
:: |
|||
models = super(ChangesetFieldRule, self)._domain_source_models() |
|||
return models | self.env.ref('base.model_res_users') |
|||
|
|||
Rules without model are global and apply for all models. |
|||
|
|||
""" |
|||
return self.env.ref('base.model_res_users') |
|||
|
|||
@api.model |
|||
def _selection_action(self): |
|||
return [('auto', 'Auto'), |
|||
('validate', 'Validate'), |
|||
('never', 'Never'), |
|||
] |
|||
|
|||
@ormcache(skiparg=1) |
|||
@api.model |
|||
def _get_rules(self, source_model_name): |
|||
""" Cache rules |
|||
|
|||
Keep only the id of the rules, because if we keep the recordsets |
|||
in the ormcache, we won't be able to browse them once their |
|||
cursor is closed. |
|||
|
|||
The public method ``get_rules`` return the rules with the recordsets |
|||
when called. |
|||
|
|||
""" |
|||
model_rules = self.search( |
|||
['|', ('source_model_id.model', '=', source_model_name), |
|||
('source_model_id', '=', False)], |
|||
# using 'ASC' means that 'NULLS LAST' is the default |
|||
order='source_model_id ASC', |
|||
) |
|||
# model's rules have precedence over global ones so we iterate |
|||
# over rules which have a source model first, then we complete |
|||
# them with the global rules |
|||
result = {} |
|||
for rule in model_rules: |
|||
# we already have a model's rule |
|||
if result.get(rule.field_id.name): |
|||
continue |
|||
result[rule.field_id.name] = rule.id |
|||
return result |
|||
|
|||
@api.model |
|||
def get_rules(self, source_model_name): |
|||
""" Return the rules for a model |
|||
|
|||
When a model is specified, it will return the rules for this |
|||
model. Fields that have no rule for this model will use the |
|||
global rules (those without source). |
|||
|
|||
The source model is the model which ask for a change, it will be |
|||
for instance ``res.users``, ``lefac.backend`` or |
|||
``magellan.backend``. |
|||
|
|||
The second argument (``source_model_name``) is optional but |
|||
cannot be an optional keyword argument otherwise it would not be |
|||
in the key for the cache. The callers have to pass ``None`` if |
|||
they want only global rules. |
|||
""" |
|||
rules = {} |
|||
cached_rules = self._get_rules(source_model_name) |
|||
for field, rule_id in cached_rules.iteritems(): |
|||
rules[field] = self.browse(rule_id) |
|||
return rules |
|||
|
|||
@api.model |
|||
def create(self, vals): |
|||
record = super(ChangesetFieldRule, self).create(vals) |
|||
self.clear_caches() |
|||
return record |
|||
|
|||
@api.multi |
|||
def write(self, vals): |
|||
result = super(ChangesetFieldRule, self).write(vals) |
|||
self.clear_caches() |
|||
return result |
|||
|
|||
@api.multi |
|||
def unlink(self): |
|||
result = super(ChangesetFieldRule, self).unlink() |
|||
self.clear_caches() |
|||
return result |
@ -0,0 +1,50 @@ |
|||
# -*- coding: utf-8 -*- |
|||
# © 2015 Camptocamp SA |
|||
# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl.html). |
|||
|
|||
from openerp import models, fields, api |
|||
|
|||
|
|||
class ResPartner(models.Model): |
|||
_inherit = 'res.partner' |
|||
|
|||
changeset_ids = fields.One2many(comodel_name='res.partner.changeset', |
|||
inverse_name='partner_id', |
|||
string='Changesets', |
|||
readonly=True) |
|||
count_pending_changesets = fields.Integer( |
|||
string='Changes', |
|||
compute='_count_pending_changesets', |
|||
search='_search_count_pending_changesets') |
|||
|
|||
@api.one |
|||
@api.depends('changeset_ids', 'changeset_ids.state') |
|||
def _count_pending_changesets(self): |
|||
changesets = self.changeset_ids.filtered( |
|||
lambda rev: rev.state == 'draft' and rev.partner_id == self |
|||
) |
|||
self.count_pending_changesets = len(changesets) |
|||
|
|||
@api.multi |
|||
def write(self, values): |
|||
if self.env.context.get('__no_changeset'): |
|||
return super(ResPartner, self).write(values) |
|||
else: |
|||
changeset_model = self.env['res.partner.changeset'] |
|||
for record in self: |
|||
local_values = changeset_model.add_changeset(record, values) |
|||
super(ResPartner, record).write(local_values) |
|||
return True |
|||
|
|||
def _search_count_pending_changesets(self, operator, value): |
|||
if operator not in ('=', '!=', '<', '<=', '>', '>=', 'in', 'not in'): |
|||
return [] |
|||
query = ("SELECT p.id " |
|||
"FROM res_partner p " |
|||
"INNER JOIN res_partner_changeset r ON r.partner_id = p.id " |
|||
"WHERE r.state = 'draft' " |
|||
"GROUP BY p.id " |
|||
"HAVING COUNT(r.id) %s %%s ") % operator |
|||
self.env.cr.execute(query, (value,)) |
|||
ids = [row[0] for row in self.env.cr.fetchall()] |
|||
return [('id', 'in', ids)] |
@ -0,0 +1,514 @@ |
|||
# -*- coding: utf-8 -*- |
|||
# © 2015 Camptocamp SA |
|||
# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl.html). |
|||
|
|||
from itertools import groupby |
|||
from lxml import etree |
|||
from operator import attrgetter |
|||
|
|||
from openerp import models, fields, api, exceptions, _ |
|||
from openerp.osv.orm import setup_modifiers |
|||
|
|||
# sentinel object to be sure that no empty value was passed to |
|||
# ResPartnerChangesetChange._value_for_changeset |
|||
_NO_VALUE = object() |
|||
|
|||
|
|||
class ResPartnerChangeset(models.Model): |
|||
_name = 'res.partner.changeset' |
|||
_description = 'Partner Changeset' |
|||
_order = 'date desc' |
|||
_rec_name = 'date' |
|||
|
|||
partner_id = fields.Many2one(comodel_name='res.partner', |
|||
string='Partner', |
|||
select=True, |
|||
required=True, |
|||
readonly=True, |
|||
ondelete='cascade') |
|||
change_ids = fields.One2many(comodel_name='res.partner.changeset.change', |
|||
inverse_name='changeset_id', |
|||
string='Changes', |
|||
readonly=True) |
|||
date = fields.Datetime(default=fields.Datetime.now, |
|||
select=True, |
|||
readonly=True) |
|||
state = fields.Selection( |
|||
compute='_compute_state', |
|||
selection=[('draft', 'Pending'), |
|||
('done', 'Done')], |
|||
string='State', |
|||
store=True, |
|||
) |
|||
note = fields.Text() |
|||
source = fields.Reference( |
|||
string='Source of the change', |
|||
selection='_reference_models', |
|||
readonly=True, |
|||
) |
|||
|
|||
@api.model |
|||
def _reference_models(self): |
|||
models = self.env['ir.model'].search([]) |
|||
return [(model.model, model.name) for model in models] |
|||
|
|||
@api.one |
|||
@api.depends('change_ids', 'change_ids.state') |
|||
def _compute_state(self): |
|||
if all(change.state in ('done', 'cancel') for change |
|||
in self.mapped('change_ids')): |
|||
self.state = 'done' |
|||
else: |
|||
self.state = 'draft' |
|||
|
|||
@api.multi |
|||
def apply(self): |
|||
self.mapped('change_ids').apply() |
|||
|
|||
@api.multi |
|||
def cancel(self): |
|||
self.mapped('change_ids').cancel() |
|||
|
|||
@api.multi |
|||
def add_changeset(self, record, values): |
|||
""" Add a changeset on a partner |
|||
|
|||
By default, when a partner is modified by a user or by the |
|||
system, the the changeset will follow the rules configured for |
|||
the 'Users' / global rules. |
|||
|
|||
A caller should pass the following keys in the context: |
|||
|
|||
* ``__changeset_rules_source_model``: name of the model which |
|||
asks for the change |
|||
* ``__changeset_rules_source_id``: id of the record which asks |
|||
for the change |
|||
|
|||
When the source model and id are not defined, the current user |
|||
is considered as the origin of the change. |
|||
|
|||
Should be called before the execution of ``write`` on the record |
|||
so we can keep track of the existing value and also because the |
|||
returned values should be used for ``write`` as some of the |
|||
values may have been removed. |
|||
|
|||
:param values: the values being written on the partner |
|||
:type values: dict |
|||
|
|||
:returns: dict of values that should be wrote on the partner |
|||
(fields with a 'Validate' or 'Never' rule are excluded) |
|||
|
|||
""" |
|||
record.ensure_one() |
|||
|
|||
source_model = self.env.context.get('__changeset_rules_source_model') |
|||
source_id = self.env.context.get('__changeset_rules_source_id') |
|||
if not source_model: |
|||
# if the changes source is not defined, log the user who |
|||
# made the change |
|||
source_model = 'res.users' |
|||
if not source_id: |
|||
source_id = self.env.uid |
|||
if source_model and source_id: |
|||
source = '%s,%s' % (source_model, source_id) |
|||
else: |
|||
source = False |
|||
|
|||
change_model = self.env['res.partner.changeset.change'] |
|||
write_values = values.copy() |
|||
changes = [] |
|||
rules = self.env['changeset.field.rule'].get_rules( |
|||
source_model_name=source_model, |
|||
) |
|||
for field in values: |
|||
rule = rules.get(field) |
|||
if not rule: |
|||
continue |
|||
if field in values: |
|||
if not change_model._has_field_changed(record, field, |
|||
values[field]): |
|||
continue |
|||
change, pop_value = change_model._prepare_changeset_change( |
|||
record, rule, field, values[field] |
|||
) |
|||
if pop_value: |
|||
write_values.pop(field) |
|||
changes.append(change) |
|||
if changes: |
|||
self.env['res.partner.changeset'].create({ |
|||
'partner_id': record.id, |
|||
'change_ids': [(0, 0, vals) for vals in changes], |
|||
'date': fields.Datetime.now(), |
|||
'source': source, |
|||
}) |
|||
return write_values |
|||
|
|||
|
|||
class ResPartnerChangesetChange(models.Model): |
|||
""" Store the change of one field for one changeset on one partner |
|||
|
|||
This model is composed of 3 sets of fields: |
|||
|
|||
* 'origin' |
|||
* 'old' |
|||
* 'new' |
|||
|
|||
The 'new' fields contain the value that needs to be validated. |
|||
The 'old' field copies the actual value of the partner when the |
|||
change is either applied either canceled. This field is used as a storage |
|||
place but never shown by itself. |
|||
The 'origin' fields is a related field towards the actual values of |
|||
the partner until the change is either applied either canceled, past |
|||
that it shows the 'old' value. |
|||
The reason behind this is that the values may change on a partner between |
|||
the moment when the changeset is created and when it is applied. |
|||
|
|||
On the views, we show the origin fields which represent the actual |
|||
partner values or the old values and we show the new fields. |
|||
|
|||
The 'origin' and 'new_value_display' are displayed on |
|||
the tree view where we need a unique of field, the other fields are |
|||
displayed on the form view so we benefit from their widgets. |
|||
|
|||
""" |
|||
_name = 'res.partner.changeset.change' |
|||
_description = 'Partner Changeset Change' |
|||
_rec_name = 'field_id' |
|||
|
|||
changeset_id = fields.Many2one(comodel_name='res.partner.changeset', |
|||
required=True, |
|||
string='Changeset', |
|||
ondelete='cascade', |
|||
readonly=True) |
|||
field_id = fields.Many2one(comodel_name='ir.model.fields', |
|||
string='Field', |
|||
required=True, |
|||
readonly=True) |
|||
field_type = fields.Selection(related='field_id.ttype', |
|||
string='Field Type', |
|||
readonly=True) |
|||
|
|||
origin_value_display = fields.Char( |
|||
string='Previous', |
|||
compute='_compute_value_display', |
|||
) |
|||
new_value_display = fields.Char( |
|||
string='New', |
|||
compute='_compute_value_display', |
|||
) |
|||
|
|||
# Fields showing the origin partner's value or the 'old' value if |
|||
# the change is applied or canceled. |
|||
origin_value_char = fields.Char(compute='_compute_origin_values', |
|||
string='Previous', |
|||
readonly=True) |
|||
origin_value_date = fields.Date(compute='_compute_origin_values', |
|||
string='Previous', |
|||
readonly=True) |
|||
origin_value_datetime = fields.Datetime(compute='_compute_origin_values', |
|||
string='Previous', |
|||
readonly=True) |
|||
origin_value_float = fields.Float(compute='_compute_origin_values', |
|||
string='Previous', |
|||
readonly=True) |
|||
origin_value_integer = fields.Integer(compute='_compute_origin_values', |
|||
string='Previous', |
|||
readonly=True) |
|||
origin_value_text = fields.Text(compute='_compute_origin_values', |
|||
string='Previous', |
|||
readonly=True) |
|||
origin_value_boolean = fields.Boolean(compute='_compute_origin_values', |
|||
string='Previous', |
|||
readonly=True) |
|||
origin_value_reference = fields.Reference( |
|||
compute='_compute_origin_values', |
|||
string='Previous', |
|||
selection='_reference_models', |
|||
readonly=True, |
|||
) |
|||
|
|||
# Fields storing the previous partner's values (saved when the |
|||
# changeset is applied) |
|||
old_value_char = fields.Char(string='Old', |
|||
readonly=True) |
|||
old_value_date = fields.Date(string='Old', |
|||
readonly=True) |
|||
old_value_datetime = fields.Datetime(string='Old', |
|||
readonly=True) |
|||
old_value_float = fields.Float(string='Old', |
|||
readonly=True) |
|||
old_value_integer = fields.Integer(string='Old', |
|||
readonly=True) |
|||
old_value_text = fields.Text(string='Old', |
|||
readonly=True) |
|||
old_value_boolean = fields.Boolean(string='Old', |
|||
readonly=True) |
|||
old_value_reference = fields.Reference(string='Old', |
|||
selection='_reference_models', |
|||
readonly=True) |
|||
|
|||
# Fields storing the value applied on the partner |
|||
new_value_char = fields.Char(string='New', |
|||
readonly=True) |
|||
new_value_date = fields.Date(string='New', |
|||
readonly=True) |
|||
new_value_datetime = fields.Datetime(string='New', |
|||
readonly=True) |
|||
new_value_float = fields.Float(string='New', |
|||
readonly=True) |
|||
new_value_integer = fields.Integer(string='New', |
|||
readonly=True) |
|||
new_value_text = fields.Text(string='New', |
|||
readonly=True) |
|||
new_value_boolean = fields.Boolean(string='New', |
|||
readonly=True) |
|||
new_value_reference = fields.Reference(string='New', |
|||
selection='_reference_models', |
|||
readonly=True) |
|||
|
|||
state = fields.Selection( |
|||
selection=[('draft', 'Pending'), |
|||
('done', 'Accepted'), |
|||
('cancel', 'Rejected'), |
|||
], |
|||
required=True, |
|||
default='draft', |
|||
readonly=True, |
|||
) |
|||
|
|||
@api.model |
|||
def _reference_models(self): |
|||
models = self.env['ir.model'].search([]) |
|||
return [(model.model, model.name) for model in models] |
|||
|
|||
_suffix_to_types = { |
|||
'char': ('char', 'selection'), |
|||
'date': ('date',), |
|||
'datetime': ('datetime',), |
|||
'float': ('float',), |
|||
'integer': ('integer',), |
|||
'text': ('text',), |
|||
'boolean': ('boolean',), |
|||
'reference': ('many2one',), |
|||
} |
|||
|
|||
_type_to_suffix = {ftype: suffix |
|||
for suffix, ftypes in _suffix_to_types.iteritems() |
|||
for ftype in ftypes} |
|||
|
|||
_origin_value_fields = ['origin_value_%s' % suffix |
|||
for suffix in _suffix_to_types] |
|||
_old_value_fields = ['old_value_%s' % suffix |
|||
for suffix in _suffix_to_types] |
|||
_new_value_fields = ['new_value_%s' % suffix |
|||
for suffix in _suffix_to_types] |
|||
_value_fields = (_origin_value_fields + |
|||
_old_value_fields + |
|||
_new_value_fields) |
|||
|
|||
@api.one |
|||
@api.depends('changeset_id.partner_id.*') |
|||
def _compute_origin_values(self): |
|||
field_name = self.get_field_for_type(self.field_id, 'origin') |
|||
if self.state == 'draft': |
|||
value = self.changeset_id.partner_id[self.field_id.name] |
|||
else: |
|||
old_field = self.get_field_for_type(self.field_id, 'old') |
|||
value = self[old_field] |
|||
setattr(self, field_name, value) |
|||
|
|||
@api.one |
|||
@api.depends(lambda self: self._value_fields) |
|||
def _compute_value_display(self): |
|||
for prefix in ('origin', 'new'): |
|||
value = getattr(self, 'get_%s_value' % prefix)() |
|||
if self.field_id.ttype == 'many2one' and value: |
|||
value = value.display_name |
|||
setattr(self, '%s_value_display' % prefix, value) |
|||
|
|||
@api.model |
|||
def get_field_for_type(self, field, prefix): |
|||
assert prefix in ('origin', 'old', 'new') |
|||
field_type = self._type_to_suffix.get(field.ttype) |
|||
if not field_type: |
|||
raise NotImplementedError( |
|||
'field type %s is not supported' % field_type |
|||
) |
|||
return '%s_value_%s' % (prefix, field_type) |
|||
|
|||
@api.multi |
|||
def get_origin_value(self): |
|||
self.ensure_one() |
|||
field_name = self.get_field_for_type(self.field_id, 'origin') |
|||
return self[field_name] |
|||
|
|||
@api.multi |
|||
def get_new_value(self): |
|||
self.ensure_one() |
|||
field_name = self.get_field_for_type(self.field_id, 'new') |
|||
return self[field_name] |
|||
|
|||
@api.multi |
|||
def set_old_value(self): |
|||
""" Copy the value of the partner to the 'old' field """ |
|||
for change in self: |
|||
# copy the existing partner's value for the history |
|||
old_value_for_write = self._value_for_changeset( |
|||
change.changeset_id.partner_id, |
|||
change.field_id.name |
|||
) |
|||
old_field_name = self.get_field_for_type(change.field_id, 'old') |
|||
change.write({old_field_name: old_value_for_write}) |
|||
|
|||
@api.multi |
|||
def apply(self): |
|||
""" Apply the change on the changeset's partner |
|||
|
|||
It is optimized thus that it makes only one write on the partner |
|||
per changeset if many changes are applied at once. |
|||
""" |
|||
changes_ok = self.browse() |
|||
key = attrgetter('changeset_id') |
|||
for changeset, changes in groupby(self.sorted(key=key), key=key): |
|||
values = {} |
|||
partner = changeset.partner_id |
|||
for change in changes: |
|||
if change.state in ('cancel', 'done'): |
|||
continue |
|||
|
|||
field = change.field_id |
|||
value_for_write = change._convert_value_for_write( |
|||
change.get_new_value() |
|||
) |
|||
values[field.name] = value_for_write |
|||
|
|||
change.set_old_value() |
|||
|
|||
changes_ok |= change |
|||
|
|||
if not values: |
|||
continue |
|||
|
|||
previous_changesets = self.env['res.partner.changeset'].search( |
|||
[('date', '<', changeset.date), |
|||
('state', '=', 'draft'), |
|||
('partner_id', '=', changeset.partner_id.id), |
|||
], |
|||
limit=1, |
|||
) |
|||
if previous_changesets: |
|||
raise exceptions.Warning( |
|||
_('This change cannot be applied because a previous ' |
|||
'changeset for the same partner is pending.\n' |
|||
'Apply all the anterior changesets before applying ' |
|||
'this one.') |
|||
) |
|||
|
|||
partner.with_context(__no_changeset=True).write(values) |
|||
|
|||
changes_ok.write({'state': 'done'}) |
|||
|
|||
@api.multi |
|||
def cancel(self): |
|||
""" Reject the change """ |
|||
if any(change.state == 'done' for change in self): |
|||
raise exceptions.Warning( |
|||
_('This change has already be applied.') |
|||
) |
|||
self.set_old_value() |
|||
self.write({'state': 'cancel'}) |
|||
|
|||
@api.model |
|||
def _has_field_changed(self, record, field, value): |
|||
field_def = record._fields[field] |
|||
current_value = field_def.convert_to_write(record[field]) |
|||
if not (current_value or value): |
|||
return False |
|||
return current_value != value |
|||
|
|||
@api.multi |
|||
def _convert_value_for_write(self, value): |
|||
if not value: |
|||
return value |
|||
model = self.env[self.field_id.model_id.model] |
|||
model_field_def = model._fields[self.field_id.name] |
|||
return model_field_def.convert_to_write(value) |
|||
|
|||
@api.model |
|||
def _value_for_changeset(self, record, field_name, value=_NO_VALUE): |
|||
""" Return a value from the record ready to write in a changeset field |
|||
|
|||
:param record: modified record |
|||
:param field_name: name of the modified field |
|||
:param value: if no value is given, it is read from the record |
|||
""" |
|||
field_def = record._fields[field_name] |
|||
if value is _NO_VALUE: |
|||
# when the value is read from the record, we need to prepare |
|||
# it for the write (e.g. extract .id from a many2one record) |
|||
value = field_def.convert_to_write(record[field_name]) |
|||
if field_def.type == 'many2one': |
|||
# store as 'reference' |
|||
comodel = field_def.comodel_name |
|||
return "%s,%s" % (comodel, value) if value else False |
|||
else: |
|||
return value |
|||
|
|||
@api.multi |
|||
def _prepare_changeset_change(self, record, rule, field_name, value): |
|||
""" Prepare data for a changeset change |
|||
|
|||
It returns a dict of the values to write on the changeset change |
|||
and a boolean that indicates if the value should be popped out |
|||
of the values to write on the model. |
|||
|
|||
:returns: dict of values, boolean |
|||
""" |
|||
new_field_name = self.get_field_for_type(rule.field_id, 'new') |
|||
new_value = self._value_for_changeset(record, field_name, value=value) |
|||
change = { |
|||
new_field_name: new_value, |
|||
'field_id': rule.field_id.id, |
|||
} |
|||
if rule.action == 'auto': |
|||
change['state'] = 'done' |
|||
pop_value = False |
|||
elif rule.action == 'validate': |
|||
change['state'] = 'draft' |
|||
pop_value = True # change to apply manually |
|||
elif rule.action == 'never': |
|||
change['state'] = 'cancel' |
|||
pop_value = True # change never applied |
|||
|
|||
if change['state'] in ('cancel', 'done'): |
|||
# Normally the 'old' value is set when we use the 'apply' |
|||
# button, but since we short circuit the 'apply', we |
|||
# directly set the 'old' value here |
|||
old_field_name = self.get_field_for_type(rule.field_id, 'old') |
|||
# get values ready to write as expected by the changeset |
|||
# (for instance, a many2one is written in a reference |
|||
# field) |
|||
origin_value = self._value_for_changeset(record, field_name) |
|||
change[old_field_name] = origin_value |
|||
|
|||
return change, pop_value |
|||
|
|||
def fields_view_get(self, *args, **kwargs): |
|||
_super = super(ResPartnerChangesetChange, self) |
|||
result = _super.fields_view_get(*args, **kwargs) |
|||
if result['type'] != 'form': |
|||
return |
|||
doc = etree.XML(result['arch']) |
|||
for suffix, ftypes in self._suffix_to_types.iteritems(): |
|||
for prefix in ('origin', 'old', 'new'): |
|||
field_name = '%s_value_%s' % (prefix, suffix) |
|||
field_nodes = doc.xpath("//field[@name='%s']" % field_name) |
|||
for node in field_nodes: |
|||
node.set( |
|||
'attrs', |
|||
"{'invisible': " |
|||
"[('field_type', 'not in', %s)]}" % (ftypes,) |
|||
) |
|||
setup_modifiers(node) |
|||
result['arch'] = etree.tostring(doc) |
|||
return result |
@ -0,0 +1,10 @@ |
|||
id,name,model_id:id,group_id:id,perm_read,perm_write,perm_create,perm_unlink |
|||
access_view_changeset_field_rule_partner_manager,changeset field rules for partner managers,model_changeset_field_rule,base.group_partner_manager,1,0,0,0 |
|||
access_view_changeset_field_rule_user,changeset field rules for changeset users,model_changeset_field_rule,group_changeset_user,1,0,0,0 |
|||
access_view_changeset_field_rule_manager,changeset field rules for changeset managers,model_changeset_field_rule,group_changeset_user,1,1,1,1 |
|||
access_view_res_partner_changeset_partner_manager,changeset for partner managers,model_res_partner_changeset,base.group_partner_manager,1,0,1,0 |
|||
access_view_res_partner_changeset_change_partner_manager,changeset change for partner managers,model_res_partner_changeset_change,base.group_partner_manager,1,0,1,0 |
|||
access_view_res_partner_changeset_user,changeset for changeset users,model_res_partner_changeset,group_changeset_user,1,1,1,0 |
|||
access_view_res_partner_changeset_change_user,changeset change for changeset users,model_res_partner_changeset_change,group_changeset_user,1,1,1,0 |
|||
access_view_res_partner_changeset_manager,changeset for changeset managers,model_res_partner_changeset,group_changeset_manager,1,1,1,1 |
|||
access_view_res_partner_changeset_change_manager,changeset change for changeset managers,model_res_partner_changeset_change,group_changeset_manager,1,1,1,1 |
@ -0,0 +1,29 @@ |
|||
<?xml version="1.0" encoding="utf-8"?> |
|||
<openerp> |
|||
<data> |
|||
|
|||
<record id="group_changeset_manager" model="res.groups"> |
|||
<field name="name">Changeset Configuration</field> |
|||
<field name="comment">The user will have an access to the configuration of the changeset rules.</field> |
|||
</record> |
|||
|
|||
<record id="group_changeset_user" model="res.groups"> |
|||
<field name="name">Changesets Validations </field> |
|||
<field name="comment">The user will be able to apply or reject changesets.</field> |
|||
</record> |
|||
|
|||
</data> |
|||
|
|||
<data noupdate="1"> |
|||
|
|||
<record id="group_changeset_manager" model="res.groups"> |
|||
<field name="users" eval="[(4, ref('base.user_root'))]"/> |
|||
</record> |
|||
|
|||
<record id="group_changeset_user" model="res.groups"> |
|||
<field name="implied_ids" eval="[(4, ref('group_changeset_manager'))]"/> |
|||
<field name="users" eval="[(4, ref('base.user_root'))]"/> |
|||
</record> |
|||
|
|||
</data> |
|||
</openerp> |
After Width: 128 | Height: 128 | Size: 9.2 KiB |
After Width: 1252 | Height: 577 | Size: 85 KiB |
After Width: 996 | Height: 557 | Size: 60 KiB |
@ -0,0 +1,6 @@ |
|||
# -*- coding: utf-8 -*- |
|||
|
|||
from . import test_changeset_flow |
|||
from . import test_changeset_field_type |
|||
from . import test_changeset_origin |
|||
from . import test_changeset_field_rule |
@ -0,0 +1,77 @@ |
|||
# -*- coding: utf-8 -*- |
|||
# © 2015 Camptocamp SA |
|||
# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl.html). |
|||
|
|||
|
|||
class ChangesetMixin(object): |
|||
|
|||
def assert_changeset(self, partner, expected_source, expected_changes): |
|||
""" Check if a changeset has been created according to expected values |
|||
|
|||
The partner should have no prior changeset than the one created in the |
|||
test (so it has exactly 1 changeset). |
|||
|
|||
The expected changes are tuples with (field, origin_value, |
|||
new_value, state) |
|||
|
|||
:param partner: record of partner having a changeset |
|||
:param expected_changes: contains tuples with the changes |
|||
:type expected_changes: list(tuple)) |
|||
""" |
|||
changeset = self.env['res.partner.changeset'].search( |
|||
[('partner_id', '=', partner.id)], |
|||
) |
|||
self.assertEqual(len(changeset), 1, |
|||
"1 changeset expected, got %s" % (changeset,)) |
|||
self.assertEqual(changeset.source, expected_source) |
|||
changes = changeset.change_ids |
|||
missing = [] |
|||
for expected_change in expected_changes: |
|||
for change in changes: |
|||
if (change.field_id, |
|||
change.get_origin_value(), |
|||
change.get_new_value(), |
|||
change.state) == expected_change: |
|||
changes -= change |
|||
break |
|||
else: |
|||
missing.append(expected_change) |
|||
message = u'' |
|||
for field, origin_value, new_value, state in missing: |
|||
message += ("- field: '%s', origin_value: '%s', " |
|||
"new_value: '%s', state: '%s'\n" % |
|||
(field.name, origin_value, new_value, state)) |
|||
for change in changes: |
|||
message += ("+ field: '%s', origin_value: '%s', " |
|||
"new_value: '%s', state: '%s'\n" % |
|||
(change.field_id.name, |
|||
change.get_origin_value(), |
|||
change.get_new_value(), |
|||
change.state)) |
|||
if message: |
|||
raise AssertionError('Changes do not match\n\n:%s' % message) |
|||
|
|||
def _create_changeset(self, partner, changes): |
|||
""" Create a changeset and its associated changes |
|||
|
|||
:param partner: 'res.partner' record |
|||
:param changes: list of changes [(field, new value, state)] |
|||
:returns: 'res.partner.changeset' record |
|||
""" |
|||
ChangesetChange = self.env['res.partner.changeset.change'] |
|||
get_field = ChangesetChange.get_field_for_type |
|||
change_values = [] |
|||
for field, value, state in changes: |
|||
change = { |
|||
'field_id': field.id, |
|||
# write in the field of the appropriate type for the |
|||
# origin field (char, many2one, ...) |
|||
get_field(field, 'new'): value, |
|||
'state': state, |
|||
} |
|||
change_values.append((0, 0, change)) |
|||
values = { |
|||
'partner_id': partner.id, |
|||
'change_ids': change_values, |
|||
} |
|||
return self.env['res.partner.changeset'].create(values) |
@ -0,0 +1,95 @@ |
|||
# -*- coding: utf-8 -*- |
|||
# |
|||
# |
|||
# Authors: Guewen Baconnier |
|||
# Copyright 2015 Camptocamp SA |
|||
# |
|||
# This program is free software: you can redistribute it and/or modify |
|||
# it under the terms of the GNU Affero General Public License as |
|||
# published by the Free Software Foundation, either version 3 of the |
|||
# License, or (at your option) any later version. |
|||
# |
|||
# This program is distributed in the hope that it will be useful, |
|||
# but WITHOUT ANY WARRANTY; without even the implied warranty of |
|||
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the |
|||
# GNU Affero General Public License for more details. |
|||
# |
|||
# You should have received a copy of the GNU Affero General Public License |
|||
# along with this program. If not, see <http://www.gnu.org/licenses/>. |
|||
# |
|||
# |
|||
|
|||
from openerp.tests import common |
|||
|
|||
|
|||
class TestChangesetFieldRule(common.TransactionCase): |
|||
|
|||
def setUp(self): |
|||
super(TestChangesetFieldRule, self).setUp() |
|||
self.company_model_id = self.env.ref('base.model_res_company').id |
|||
self.field_name = self.env.ref('base.field_res_partner_name') |
|||
self.field_street = self.env.ref('base.field_res_partner_street') |
|||
|
|||
def test_get_rules(self): |
|||
ChangesetFieldRule = self.env['changeset.field.rule'] |
|||
ChangesetFieldRule.search([]).unlink() |
|||
rule1 = ChangesetFieldRule.create({ |
|||
'field_id': self.field_name.id, |
|||
'action': 'validate', |
|||
}) |
|||
rule2 = ChangesetFieldRule.create({ |
|||
'field_id': self.field_street.id, |
|||
'action': 'never', |
|||
}) |
|||
get_rules = ChangesetFieldRule.get_rules(None) |
|||
self.assertEqual(get_rules, {'name': rule1, 'street': rule2}) |
|||
|
|||
def test_get_rules_source(self): |
|||
ChangesetFieldRule = self.env['changeset.field.rule'] |
|||
ChangesetFieldRule.search([]).unlink() |
|||
rule1 = ChangesetFieldRule.create({ |
|||
'field_id': self.field_name.id, |
|||
'action': 'validate', |
|||
}) |
|||
rule2 = ChangesetFieldRule.create({ |
|||
'field_id': self.field_street.id, |
|||
'action': 'never', |
|||
}) |
|||
rule3 = ChangesetFieldRule.create({ |
|||
'source_model_id': self.company_model_id, |
|||
'field_id': self.field_street.id, |
|||
'action': 'never', |
|||
}) |
|||
model = ChangesetFieldRule |
|||
rules = model.get_rules(None) |
|||
self.assertEqual(rules, {u'name': rule1, u'street': rule2}) |
|||
rules = model.get_rules('res.company') |
|||
self.assertEqual(rules, {u'name': rule1, u'street': rule3}) |
|||
|
|||
def test_get_rules_cache(self): |
|||
ChangesetFieldRule = self.env['changeset.field.rule'] |
|||
ChangesetFieldRule.search([]).unlink() |
|||
rule = ChangesetFieldRule.create({ |
|||
'field_id': self.field_name.id, |
|||
'action': 'validate', |
|||
}) |
|||
self.assertEqual( |
|||
ChangesetFieldRule.get_rules(None)['name'].action, |
|||
'validate', |
|||
) |
|||
# Write on cursor to bypass the cache invalidation for the |
|||
# matter of the test |
|||
self.env.cr.execute("UPDATE changeset_field_rule " |
|||
"SET action = 'never' " |
|||
"WHERE id = %s", (rule.id,)) |
|||
self.assertEqual( |
|||
ChangesetFieldRule.get_rules(None)['name'].action, |
|||
'validate', |
|||
) |
|||
rule.action = 'auto' |
|||
self.assertEqual( |
|||
ChangesetFieldRule.get_rules(None)['name'].action, |
|||
'auto', |
|||
) |
|||
rule.unlink() |
|||
self.assertFalse(ChangesetFieldRule.get_rules(None)) |
@ -0,0 +1,279 @@ |
|||
# -*- coding: utf-8 -*- |
|||
# © 2015 Camptocamp SA |
|||
# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl.html). |
|||
|
|||
from openerp.tests import common |
|||
from .common import ChangesetMixin |
|||
|
|||
|
|||
class TestChangesetFieldType(ChangesetMixin, common.TransactionCase): |
|||
""" Check that changeset changes are stored expectingly to their types """ |
|||
|
|||
def _setup_rules(self): |
|||
ChangesetFieldRule = self.env['changeset.field.rule'] |
|||
ChangesetFieldRule.search([]).unlink() |
|||
fields = (('char', 'ref'), |
|||
('text', 'comment'), |
|||
('boolean', 'customer'), |
|||
('date', 'date'), |
|||
('integer', 'color'), |
|||
('float', 'credit_limit'), |
|||
('selection', 'type'), |
|||
('many2one', 'country_id'), |
|||
('many2many', 'category_id'), |
|||
('one2many', 'user_ids'), |
|||
('binary', 'image'), |
|||
) |
|||
for field_type, field in fields: |
|||
attr_name = 'field_%s' % field_type |
|||
field_record = self.env['ir.model.fields'].search([ |
|||
('model', '=', 'res.partner'), |
|||
('name', '=', field), |
|||
]) |
|||
# set attribute such as 'self.field_char' is a |
|||
# ir.model.fields record of the field res_partner.ref |
|||
setattr(self, attr_name, field_record) |
|||
ChangesetFieldRule.create({ |
|||
'field_id': field_record.id, |
|||
'action': 'validate', |
|||
}) |
|||
|
|||
def setUp(self): |
|||
super(TestChangesetFieldType, self).setUp() |
|||
self._setup_rules() |
|||
self.partner = self.env['res.partner'].create({ |
|||
'name': 'Original Name', |
|||
'street': 'Original Street', |
|||
}) |
|||
|
|||
def test_new_changeset_char(self): |
|||
""" Add a new changeset on a Char field """ |
|||
self.partner.write({ |
|||
self.field_char.name: 'New value', |
|||
}) |
|||
self.assert_changeset( |
|||
self.partner, |
|||
self.env.user, |
|||
[(self.field_char, self.partner[self.field_char.name], |
|||
'New value', 'draft'), |
|||
] |
|||
) |
|||
|
|||
def test_new_changeset_text(self): |
|||
""" Add a new changeset on a Text field """ |
|||
self.partner.write({ |
|||
self.field_text.name: 'New comment\non 2 lines', |
|||
}) |
|||
self.assert_changeset( |
|||
self.partner, |
|||
self.env.user, |
|||
[(self.field_text, self.partner[self.field_text.name], |
|||
'New comment\non 2 lines', 'draft'), |
|||
] |
|||
) |
|||
|
|||
def test_new_changeset_boolean(self): |
|||
""" Add a new changeset on a Boolean field """ |
|||
# ensure the changeset has to change the value |
|||
self.partner.with_context(__no_changeset=True).write({ |
|||
self.field_boolean.name: False, |
|||
}) |
|||
|
|||
self.partner.write({ |
|||
self.field_boolean.name: True, |
|||
}) |
|||
self.assert_changeset( |
|||
self.partner, |
|||
self.env.user, |
|||
[(self.field_boolean, self.partner[self.field_boolean.name], |
|||
True, 'draft'), |
|||
] |
|||
) |
|||
|
|||
def test_new_changeset_date(self): |
|||
""" Add a new changeset on a Date field """ |
|||
self.partner.write({ |
|||
self.field_date.name: '2015-09-15', |
|||
}) |
|||
self.assert_changeset( |
|||
self.partner, |
|||
self.env.user, |
|||
[(self.field_date, self.partner[self.field_date.name], |
|||
'2015-09-15', 'draft'), |
|||
] |
|||
) |
|||
|
|||
def test_new_changeset_integer(self): |
|||
""" Add a new changeset on a Integer field """ |
|||
self.partner.write({ |
|||
self.field_integer.name: 42, |
|||
}) |
|||
self.assert_changeset( |
|||
self.partner, |
|||
self.env.user, |
|||
[(self.field_integer, self.partner[self.field_integer.name], |
|||
42, 'draft'), |
|||
] |
|||
) |
|||
|
|||
def test_new_changeset_float(self): |
|||
""" Add a new changeset on a Float field """ |
|||
self.partner.write({ |
|||
self.field_float.name: 3.1415, |
|||
}) |
|||
self.assert_changeset( |
|||
self.partner, |
|||
self.env.user, |
|||
[(self.field_float, self.partner[self.field_float.name], |
|||
3.1415, 'draft'), |
|||
] |
|||
) |
|||
|
|||
def test_new_changeset_selection(self): |
|||
""" Add a new changeset on a Selection field """ |
|||
self.partner.write({ |
|||
self.field_selection.name: 'delivery', |
|||
}) |
|||
self.assert_changeset( |
|||
self.partner, |
|||
self.env.user, |
|||
[(self.field_selection, self.partner[self.field_selection.name], |
|||
'delivery', 'draft'), |
|||
] |
|||
) |
|||
|
|||
def test_new_changeset_many2one(self): |
|||
""" Add a new changeset on a Many2one field """ |
|||
self.partner.with_context(__no_changeset=True).write({ |
|||
self.field_many2one.name: self.env.ref('base.fr').id, |
|||
|
|||
}) |
|||
self.partner.write({ |
|||
self.field_many2one.name: self.env.ref('base.ch').id, |
|||
}) |
|||
self.assert_changeset( |
|||
self.partner, |
|||
self.env.user, |
|||
[(self.field_many2one, self.partner[self.field_many2one.name], |
|||
self.env.ref('base.ch'), 'draft'), |
|||
] |
|||
) |
|||
|
|||
def test_new_changeset_many2many(self): |
|||
""" Add a new changeset on a Many2many field is not supported """ |
|||
with self.assertRaises(NotImplementedError): |
|||
self.partner.write({ |
|||
self.field_many2many.name: [self.env.ref('base.ch').id], |
|||
}) |
|||
|
|||
def test_new_changeset_one2many(self): |
|||
""" Add a new changeset on a One2many field is not supported """ |
|||
with self.assertRaises(NotImplementedError): |
|||
self.partner.write({ |
|||
self.field_one2many.name: [self.env.ref('base.user_root').id], |
|||
}) |
|||
|
|||
def test_new_changeset_binary(self): |
|||
""" Add a new changeset on a Binary field is not supported """ |
|||
with self.assertRaises(NotImplementedError): |
|||
self.partner.write({ |
|||
self.field_binary.name: 'xyz', |
|||
}) |
|||
|
|||
def test_apply_char(self): |
|||
""" Apply a change on a Char field """ |
|||
changes = [(self.field_char, 'New Ref', 'draft')] |
|||
changeset = self._create_changeset(self.partner, changes) |
|||
changeset.change_ids.apply() |
|||
self.assertEqual(self.partner[self.field_char.name], 'New Ref') |
|||
|
|||
def test_apply_text(self): |
|||
""" Apply a change on a Text field """ |
|||
changes = [(self.field_text, 'New comment\non 2 lines', 'draft')] |
|||
changeset = self._create_changeset(self.partner, changes) |
|||
changeset.change_ids.apply() |
|||
self.assertEqual(self.partner[self.field_text.name], |
|||
'New comment\non 2 lines') |
|||
|
|||
def test_apply_boolean(self): |
|||
""" Apply a change on a Boolean field """ |
|||
# ensure the changeset has to change the value |
|||
self.partner.write({self.field_boolean.name: False}) |
|||
|
|||
changes = [(self.field_boolean, True, 'draft')] |
|||
changeset = self._create_changeset(self.partner, changes) |
|||
changeset.change_ids.apply() |
|||
self.assertEqual(self.partner[self.field_boolean.name], True) |
|||
|
|||
changes = [(self.field_boolean, False, 'draft')] |
|||
changeset = self._create_changeset(self.partner, changes) |
|||
changeset.change_ids.apply() |
|||
self.assertEqual(self.partner[self.field_boolean.name], False) |
|||
|
|||
def test_apply_date(self): |
|||
""" Apply a change on a Date field """ |
|||
changes = [(self.field_date, '2015-09-15', 'draft')] |
|||
changeset = self._create_changeset(self.partner, changes) |
|||
changeset.change_ids.apply() |
|||
self.assertAlmostEqual(self.partner[self.field_date.name], |
|||
'2015-09-15') |
|||
|
|||
def test_apply_integer(self): |
|||
""" Apply a change on a Integer field """ |
|||
changes = [(self.field_integer, 42, 'draft')] |
|||
changeset = self._create_changeset(self.partner, changes) |
|||
changeset.change_ids.apply() |
|||
self.assertAlmostEqual(self.partner[self.field_integer.name], 42) |
|||
|
|||
def test_apply_float(self): |
|||
""" Apply a change on a Float field """ |
|||
changes = [(self.field_float, 52.47, 'draft')] |
|||
changeset = self._create_changeset(self.partner, changes) |
|||
changeset.change_ids.apply() |
|||
self.assertAlmostEqual(self.partner[self.field_float.name], 52.47) |
|||
|
|||
def test_apply_selection(self): |
|||
""" Apply a change on a Selection field """ |
|||
changes = [(self.field_selection, 'delivery', 'draft')] |
|||
changeset = self._create_changeset(self.partner, changes) |
|||
changeset.change_ids.apply() |
|||
self.assertAlmostEqual(self.partner[self.field_selection.name], |
|||
'delivery') |
|||
|
|||
def test_apply_many2one(self): |
|||
""" Apply a change on a Many2one field """ |
|||
self.partner.with_context(__no_changeset=True).write({ |
|||
self.field_many2one.name: self.env.ref('base.fr').id, |
|||
|
|||
}) |
|||
changes = [(self.field_many2one, |
|||
'res.country,%d' % self.env.ref('base.ch').id, |
|||
'draft')] |
|||
changeset = self._create_changeset(self.partner, changes) |
|||
changeset.change_ids.apply() |
|||
self.assertEqual(self.partner[self.field_many2one.name], |
|||
self.env.ref('base.ch')) |
|||
|
|||
def test_apply_many2many(self): |
|||
""" Apply a change on a Many2many field is not supported """ |
|||
changes = [(self.field_many2many, |
|||
self.env.ref('base.ch').id, |
|||
'draft')] |
|||
with self.assertRaises(NotImplementedError): |
|||
self._create_changeset(self.partner, changes) |
|||
|
|||
def test_apply_one2many(self): |
|||
""" Apply a change on a One2many field is not supported """ |
|||
changes = [(self.field_one2many, |
|||
[self.env.ref('base.user_root').id, |
|||
self.env.ref('base.user_demo').id, |
|||
], |
|||
'draft')] |
|||
with self.assertRaises(NotImplementedError): |
|||
self._create_changeset(self.partner, changes) |
|||
|
|||
def test_apply_binary(self): |
|||
""" Apply a change on a Binary field is not supported """ |
|||
changes = [(self.field_one2many, '', 'draft')] |
|||
with self.assertRaises(NotImplementedError): |
|||
self._create_changeset(self.partner, changes) |
@ -0,0 +1,246 @@ |
|||
# -*- coding: utf-8 -*- |
|||
# © 2015 Camptocamp SA |
|||
# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl.html). |
|||
|
|||
from datetime import datetime, timedelta |
|||
|
|||
from openerp import fields, exceptions |
|||
from openerp.tests import common |
|||
from .common import ChangesetMixin |
|||
|
|||
|
|||
class TestChangesetFlow(ChangesetMixin, common.TransactionCase): |
|||
""" Check how changeset are generated and applied based on the rules. |
|||
|
|||
We do not really care about the types of the fields in this test |
|||
suite, so we only use 'char' fields. We have to ensure that the |
|||
general changeset flows work as expected, that is: |
|||
|
|||
* create a changeset when a manual/system write is made on partner |
|||
* create a changeset according to the changeset rules when a source model |
|||
is specified |
|||
* apply a changeset change writes the value on the partner |
|||
* apply a whole changeset writes all the changes' values on the partner |
|||
* changes in state 'cancel' or 'done' do not write on the partner |
|||
* when all the changes are either 'cancel' or 'done', the changeset |
|||
becomes 'done' |
|||
""" |
|||
|
|||
def _setup_rules(self): |
|||
ChangesetFieldRule = self.env['changeset.field.rule'] |
|||
ChangesetFieldRule.search([]).unlink() |
|||
self.field_name = self.env.ref('base.field_res_partner_name') |
|||
self.field_street = self.env.ref('base.field_res_partner_street') |
|||
self.field_street2 = self.env.ref('base.field_res_partner_street2') |
|||
ChangesetFieldRule.create({ |
|||
'field_id': self.field_name.id, |
|||
'action': 'auto', |
|||
}) |
|||
ChangesetFieldRule.create({ |
|||
'field_id': self.field_street.id, |
|||
'action': 'validate', |
|||
}) |
|||
ChangesetFieldRule.create({ |
|||
'field_id': self.field_street2.id, |
|||
'action': 'never', |
|||
}) |
|||
|
|||
def setUp(self): |
|||
super(TestChangesetFlow, self).setUp() |
|||
self._setup_rules() |
|||
self.partner = self.env['res.partner'].create({ |
|||
'name': 'X', |
|||
'street': 'street X', |
|||
'street2': 'street2 X', |
|||
}) |
|||
|
|||
def test_new_changeset(self): |
|||
""" Add a new changeset on a partner |
|||
|
|||
A new changeset is created when we write on a partner |
|||
""" |
|||
self.partner.write({ |
|||
'name': 'Y', |
|||
'street': 'street Y', |
|||
'street2': 'street2 Y', |
|||
}) |
|||
self.assert_changeset( |
|||
self.partner, |
|||
self.env.user, |
|||
[(self.field_name, 'X', 'Y', 'done'), |
|||
(self.field_street, 'street X', 'street Y', 'draft'), |
|||
(self.field_street2, 'street2 X', 'street2 Y', 'cancel'), |
|||
], |
|||
) |
|||
self.assertEqual(self.partner.name, 'Y') |
|||
self.assertEqual(self.partner.street, 'street X') |
|||
self.assertEqual(self.partner.street2, 'street2 X') |
|||
|
|||
def test_new_changeset_empty_value(self): |
|||
""" Create a changeset change that empty a value """ |
|||
self.partner.write({ |
|||
'street': False, |
|||
}) |
|||
self.assert_changeset( |
|||
self.partner, |
|||
self.env.user, |
|||
[(self.field_street, 'street X', False, 'draft')] |
|||
) |
|||
|
|||
def test_no_changeset_empty_value_both_sides(self): |
|||
""" No changeset created when both sides have an empty value """ |
|||
# we have to ensure that even if we write '' to a False field, we won't |
|||
# write a changeset |
|||
self.partner.with_context(__no_changeset=True).write({ |
|||
'street': False, |
|||
}) |
|||
self.partner.write({ |
|||
'street': '', |
|||
}) |
|||
self.assertFalse(self.partner.changeset_ids) |
|||
|
|||
def test_apply_change(self): |
|||
""" Apply a changeset change on a partner """ |
|||
changes = [ |
|||
(self.field_name, 'Y', 'draft'), |
|||
] |
|||
changeset = self._create_changeset(self.partner, changes) |
|||
changeset.change_ids.apply() |
|||
self.assertEqual(self.partner.name, 'Y') |
|||
self.assertEqual(changeset.change_ids.state, 'done') |
|||
|
|||
def test_apply_done_change(self): |
|||
""" Done changes do not apply (already applied) """ |
|||
changes = [ |
|||
(self.field_name, 'Y', 'done'), |
|||
] |
|||
changeset = self._create_changeset(self.partner, changes) |
|||
changeset.change_ids.apply() |
|||
self.assertEqual(self.partner.name, 'X') |
|||
|
|||
def test_apply_cancel_change(self): |
|||
""" Cancel changes do not apply """ |
|||
changes = [ |
|||
(self.field_name, 'Y', 'cancel'), |
|||
] |
|||
changeset = self._create_changeset(self.partner, changes) |
|||
changeset.change_ids.apply() |
|||
self.assertEqual(self.partner.name, 'X') |
|||
|
|||
def test_apply_empty_value(self): |
|||
""" Apply a change that empty a value """ |
|||
changes = [ |
|||
(self.field_street, False, 'draft'), |
|||
] |
|||
changeset = self._create_changeset(self.partner, changes) |
|||
changeset.change_ids.apply() |
|||
self.assertFalse(self.partner.street) |
|||
|
|||
def test_apply_change_loop(self): |
|||
""" Test @api.multi on the changes """ |
|||
changes = [ |
|||
(self.field_name, 'Y', 'draft'), |
|||
(self.field_street, 'street Y', 'draft'), |
|||
(self.field_street2, 'street2 Y', 'draft'), |
|||
] |
|||
changeset = self._create_changeset(self.partner, changes) |
|||
changeset.change_ids.apply() |
|||
self.assertEqual(self.partner.name, 'Y') |
|||
self.assertEqual(self.partner.street, 'street Y') |
|||
self.assertEqual(self.partner.street2, 'street2 Y') |
|||
|
|||
def test_apply(self): |
|||
""" Apply a full changeset on a partner """ |
|||
changes = [ |
|||
(self.field_name, 'Y', 'draft'), |
|||
(self.field_street, 'street Y', 'draft'), |
|||
(self.field_street2, 'street2 Y', 'draft'), |
|||
] |
|||
changeset = self._create_changeset(self.partner, changes) |
|||
changeset.apply() |
|||
self.assertEqual(self.partner.name, 'Y') |
|||
self.assertEqual(self.partner.street, 'street Y') |
|||
self.assertEqual(self.partner.street2, 'street2 Y') |
|||
|
|||
def test_changeset_state_on_done(self): |
|||
""" Check that changeset state becomes done when changes are done """ |
|||
changes = [(self.field_name, 'Y', 'draft')] |
|||
changeset = self._create_changeset(self.partner, changes) |
|||
self.assertEqual(changeset.state, 'draft') |
|||
changeset.change_ids.apply() |
|||
self.assertEqual(changeset.state, 'done') |
|||
|
|||
def test_changeset_state_on_cancel(self): |
|||
""" Check that rev. state becomes done when changes are canceled """ |
|||
changes = [(self.field_name, 'Y', 'draft')] |
|||
changeset = self._create_changeset(self.partner, changes) |
|||
self.assertEqual(changeset.state, 'draft') |
|||
changeset.change_ids.cancel() |
|||
self.assertEqual(changeset.state, 'done') |
|||
|
|||
def test_changeset_state(self): |
|||
""" Check that changeset state becomes done with multiple changes """ |
|||
changes = [ |
|||
(self.field_name, 'Y', 'draft'), |
|||
(self.field_street, 'street Y', 'draft'), |
|||
(self.field_street2, 'street2 Y', 'draft'), |
|||
] |
|||
changeset = self._create_changeset(self.partner, changes) |
|||
self.assertEqual(changeset.state, 'draft') |
|||
changeset.apply() |
|||
self.assertEqual(changeset.state, 'done') |
|||
|
|||
def test_apply_changeset_with_other_pending(self): |
|||
""" Error when applying when previous pending changesets exist """ |
|||
changes = [(self.field_name, 'Y', 'draft')] |
|||
old_changeset = self._create_changeset(self.partner, changes) |
|||
# if the date is the same, both changeset can be applied |
|||
to_string = fields.Datetime.to_string |
|||
old_changeset.date = to_string(datetime.now() - timedelta(days=1)) |
|||
changes = [(self.field_name, 'Z', 'draft')] |
|||
changeset = self._create_changeset(self.partner, changes) |
|||
with self.assertRaises(exceptions.Warning): |
|||
changeset.change_ids.apply() |
|||
|
|||
def test_apply_different_changesets(self): |
|||
""" Apply different changesets at once """ |
|||
partner2 = self.env['res.partner'].create({'name': 'P2'}) |
|||
changes = [ |
|||
(self.field_name, 'Y', 'draft'), |
|||
(self.field_street, 'street Y', 'draft'), |
|||
(self.field_street2, 'street2 Y', 'draft'), |
|||
] |
|||
changeset = self._create_changeset(self.partner, changes) |
|||
changeset2 = self._create_changeset(partner2, changes) |
|||
self.assertEqual(changeset.state, 'draft') |
|||
self.assertEqual(changeset2.state, 'draft') |
|||
(changeset + changeset2).apply() |
|||
self.assertEqual(self.partner.name, 'Y') |
|||
self.assertEqual(self.partner.street, 'street Y') |
|||
self.assertEqual(self.partner.street2, 'street2 Y') |
|||
self.assertEqual(partner2.name, 'Y') |
|||
self.assertEqual(partner2.street, 'street Y') |
|||
self.assertEqual(partner2.street2, 'street2 Y') |
|||
self.assertEqual(changeset.state, 'done') |
|||
self.assertEqual(changeset2.state, 'done') |
|||
|
|||
def test_new_changeset_source(self): |
|||
""" Source is the user who made the change """ |
|||
self.partner.write({ |
|||
'street': False, |
|||
}) |
|||
changeset = self.partner.changeset_ids |
|||
self.assertEqual(changeset.source, self.env.user) |
|||
|
|||
def test_new_changeset_source_other_model(self): |
|||
""" Define source from another model """ |
|||
company = self.env.ref('base.main_company') |
|||
keys = { |
|||
'__changeset_rules_source_model': 'res.company', |
|||
'__changeset_rules_source_id': company.id, |
|||
} |
|||
self.partner.with_context(**keys).write({ |
|||
'street': False, |
|||
}) |
|||
changeset = self.partner.changeset_ids |
|||
self.assertEqual(changeset.source, company) |
@ -0,0 +1,110 @@ |
|||
# -*- coding: utf-8 -*- |
|||
# © 2015 Camptocamp SA |
|||
# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl.html). |
|||
|
|||
from openerp.tests import common |
|||
from .common import ChangesetMixin |
|||
|
|||
|
|||
class TestChangesetOrigin(ChangesetMixin, common.TransactionCase): |
|||
""" Check that origin - old fields are stored as expected. |
|||
|
|||
'origin' fields dynamically read fields from the partner when the state |
|||
of the change is 'draft'. Once a change becomes 'done' or 'cancel', the |
|||
'old' field copies the value from the partner and then the 'origin' field |
|||
displays the 'old' value. |
|||
""" |
|||
|
|||
def _setup_rules(self): |
|||
ChangesetFieldRule = self.env['changeset.field.rule'] |
|||
ChangesetFieldRule.search([]).unlink() |
|||
self.field_name = self.env.ref('base.field_res_partner_name') |
|||
ChangesetFieldRule.create({ |
|||
'field_id': self.field_name.id, |
|||
'action': 'validate', |
|||
}) |
|||
|
|||
def setUp(self): |
|||
super(TestChangesetOrigin, self).setUp() |
|||
self._setup_rules() |
|||
self.partner = self.env['res.partner'].create({ |
|||
'name': 'X', |
|||
}) |
|||
|
|||
def test_origin_value_of_change_with_apply(self): |
|||
""" Origin field is read from the parter or 'old' - with apply |
|||
|
|||
According to the state of the change. |
|||
""" |
|||
self.partner.write({ |
|||
'name': 'Y', |
|||
}) |
|||
changeset = self.partner.changeset_ids |
|||
change = changeset.change_ids |
|||
self.assertEqual(self.partner.name, 'X') |
|||
self.assertEqual(change.origin_value_char, 'X') |
|||
self.assertEqual(change.origin_value_display, 'X') |
|||
self.partner.with_context(__no_changeset=True).write({'name': 'A'}) |
|||
self.assertEqual(change.origin_value_char, 'A') |
|||
self.assertEqual(change.origin_value_display, 'A') |
|||
change.apply() |
|||
self.assertEqual(change.origin_value_char, 'A') |
|||
self.assertEqual(change.origin_value_display, 'A') |
|||
self.partner.with_context(__no_changeset=True).write({'name': 'B'}) |
|||
self.assertEqual(change.origin_value_char, 'A') |
|||
self.assertEqual(change.origin_value_display, 'A') |
|||
|
|||
def test_origin_value_of_change_with_cancel(self): |
|||
""" Origin field is read from the parter or 'old' - with cancel |
|||
|
|||
According to the state of the change. |
|||
""" |
|||
self.partner.write({ |
|||
'name': 'Y', |
|||
}) |
|||
changeset = self.partner.changeset_ids |
|||
change = changeset.change_ids |
|||
self.assertEqual(self.partner.name, 'X') |
|||
self.assertEqual(change.origin_value_char, 'X') |
|||
self.assertEqual(change.origin_value_display, 'X') |
|||
self.partner.with_context(__no_changeset=True).write({'name': 'A'}) |
|||
self.assertEqual(change.origin_value_char, 'A') |
|||
self.assertEqual(change.origin_value_display, 'A') |
|||
change.cancel() |
|||
self.assertEqual(change.origin_value_char, 'A') |
|||
self.assertEqual(change.origin_value_display, 'A') |
|||
self.partner.with_context(__no_changeset=True).write({'name': 'B'}) |
|||
self.assertEqual(change.origin_value_char, 'A') |
|||
self.assertEqual(change.origin_value_display, 'A') |
|||
|
|||
def test_old_field_of_change_with_apply(self): |
|||
""" Old field is stored when the change is applied """ |
|||
self.partner.write({ |
|||
'name': 'Y', |
|||
}) |
|||
changeset = self.partner.changeset_ids |
|||
change = changeset.change_ids |
|||
self.assertEqual(self.partner.name, 'X') |
|||
self.assertFalse(change.old_value_char) |
|||
self.partner.with_context(__no_changeset=True).write({'name': 'A'}) |
|||
self.assertFalse(change.old_value_char) |
|||
change.apply() |
|||
self.assertEqual(change.old_value_char, 'A') |
|||
self.partner.with_context(__no_changeset=True).write({'name': 'B'}) |
|||
self.assertEqual(change.old_value_char, 'A') |
|||
|
|||
def test_old_field_of_change_with_cancel(self): |
|||
""" Old field is stored when the change is canceled """ |
|||
self.partner.write({ |
|||
'name': 'Y', |
|||
}) |
|||
changeset = self.partner.changeset_ids |
|||
change = changeset.change_ids |
|||
self.assertEqual(self.partner.name, 'X') |
|||
self.assertFalse(change.old_value_char) |
|||
self.partner.with_context(__no_changeset=True).write({'name': 'A'}) |
|||
self.assertFalse(change.old_value_char) |
|||
change.cancel() |
|||
self.assertEqual(change.old_value_char, 'A') |
|||
self.partner.with_context(__no_changeset=True).write({'name': 'B'}) |
|||
self.assertEqual(change.old_value_char, 'A') |
@ -0,0 +1,63 @@ |
|||
<?xml version="1.0" encoding="utf-8"?> |
|||
<openerp> |
|||
<data noupdate="0"> |
|||
<record id="view_changeset_field_rule_tree" model="ir.ui.view"> |
|||
<field name="name">changeset.field.rule.tree</field> |
|||
<field name="model">changeset.field.rule</field> |
|||
<field name="arch" type="xml"> |
|||
<tree string="Changeset Fields Rules"> |
|||
<field name="field_id"/> |
|||
<field name="source_model_id"/> |
|||
<field name="action"/> |
|||
</tree> |
|||
</field> |
|||
</record> |
|||
|
|||
<record id="view_changeset_field_rule_form" model="ir.ui.view"> |
|||
<field name="name">changeset.field.rule.form</field> |
|||
<field name="model">changeset.field.rule</field> |
|||
<field name="arch" type="xml"> |
|||
<form string="Changeset Fields Rules"> |
|||
<sheet string="Changeset Fields Rules"> |
|||
<group> |
|||
<field name="field_id" |
|||
options="{'no_create_edit': True, 'no_open': True}" |
|||
/> |
|||
<field name="action"/> |
|||
<field name="source_model_id" |
|||
widget="selection" /> |
|||
</group> |
|||
</sheet> |
|||
</form> |
|||
</field> |
|||
</record> |
|||
|
|||
<record id="view_changeset_field_rule_search" model="ir.ui.view"> |
|||
<field name="name">changeset.field.rule.search</field> |
|||
<field name="model">changeset.field.rule</field> |
|||
<field name="arch" type="xml"> |
|||
<search string="Changeset Fields Rules"> |
|||
<field name="field_id"/> |
|||
<field name="source_model_id"/> |
|||
<field name="action"/> |
|||
</search> |
|||
</field> |
|||
</record> |
|||
|
|||
<record model="ir.actions.act_window" id="action_changeset_field_rule_view"> |
|||
<field name="name">Changeset Fields Rules</field> |
|||
<field name="type">ir.actions.act_window</field> |
|||
<field name="res_model">changeset.field.rule</field> |
|||
<field name="view_type">form</field> |
|||
<field name="view_mode">tree,form</field> |
|||
<field name="search_view_id" ref="view_changeset_field_rule_search"/> |
|||
</record> |
|||
|
|||
<menuitem id="menu_changeset_field_rule" |
|||
parent="menu_changeset" |
|||
name="Field Rules" |
|||
groups="group_changeset_manager" |
|||
sequence="20" |
|||
action="action_changeset_field_rule_view"/> |
|||
</data> |
|||
</openerp> |
@ -0,0 +1,10 @@ |
|||
<?xml version="1.0" encoding="utf-8"?> |
|||
<openerp> |
|||
<data noupdate="0"> |
|||
<menuitem id="menu_changeset" |
|||
name="Partner Changesets" |
|||
groups="group_changeset_user" |
|||
parent="base.menu_base_config" |
|||
sequence="20"/> |
|||
</data> |
|||
</openerp> |
@ -0,0 +1,159 @@ |
|||
<?xml version="1.0" encoding="utf-8"?> |
|||
<openerp> |
|||
<data noupdate="0"> |
|||
<record id="view_res_partner_changeset_tree" model="ir.ui.view"> |
|||
<field name="name">res.partner.changeset.tree</field> |
|||
<field name="model">res.partner.changeset</field> |
|||
<field name="arch" type="xml"> |
|||
<tree string="Partner Changeset" delete="false" create="false"> |
|||
<field name="partner_id"/> |
|||
<field name="date"/> |
|||
<field name="state"/> |
|||
</tree> |
|||
</field> |
|||
</record> |
|||
|
|||
<record id="view_res_partner_changeset_form" model="ir.ui.view"> |
|||
<field name="name">res.partner.changeset.form</field> |
|||
<field name="model">res.partner.changeset</field> |
|||
<field name="arch" type="xml"> |
|||
<form string="Partner Changeset" delete="false" create="false"> |
|||
<header> |
|||
<button name="apply" |
|||
string="Apply pending changes" type="object" |
|||
class="oe_highlight" |
|||
states="draft"/> |
|||
<button name="cancel" |
|||
string="Reject pending changes" type="object" |
|||
class="oe_highlight" |
|||
states="draft"/> |
|||
<field name="state" widget="statusbar" |
|||
statusbar_visible="draft,done"/> |
|||
</header> |
|||
<sheet string="Partner Changeset"> |
|||
<group> |
|||
<field name="partner_id"/> |
|||
<field name="source"/> |
|||
<field name="date"/> |
|||
</group> |
|||
<group string="Changes"> |
|||
<field name="change_ids" nolabel="1"> |
|||
<tree string="Partner Changeset Change"> |
|||
<field name="field_id" context="{'no_open': true}"/> |
|||
<field name="field_type" invisible="1"/> |
|||
|
|||
<field name="origin_value_display" string="Previous"/> |
|||
<field name="new_value_display"/> |
|||
|
|||
<field name="state"/> |
|||
<button name="apply" |
|||
string="Apply" type="object" |
|||
icon="gtk-apply" |
|||
states="draft"/> |
|||
<button name="cancel" |
|||
string="Reject" type="object" |
|||
icon="gtk-close" |
|||
states="draft"/> |
|||
</tree> |
|||
</field> |
|||
</group> |
|||
<group> |
|||
<field name="note"/> |
|||
</group> |
|||
</sheet> |
|||
</form> |
|||
</field> |
|||
</record> |
|||
|
|||
<record id="view_res_partner_changeset_change_form" model="ir.ui.view"> |
|||
<field name="name">res.partner.changeset.change.form</field> |
|||
<field name="model">res.partner.changeset.change</field> |
|||
<field name="arch" type="xml"> |
|||
<form string="Partner Changeset Change" delete="false" create="false"> |
|||
<header> |
|||
<button name="apply" |
|||
string="Apply" type="object" |
|||
class="oe_highlight" |
|||
states="draft"/> |
|||
<button name="cancel" |
|||
string="Reject" type="object" |
|||
class="oe_highlight" |
|||
states="draft"/> |
|||
<field name="state" widget="statusbar" |
|||
statusbar_visible="draft,done"/> |
|||
</header> |
|||
<sheet> |
|||
<group> |
|||
<field name="field_id" options="{'no_open': true}"/> |
|||
<field name="field_type" invisible="1"/> |
|||
|
|||
<!-- attrs are added in fields_view_get --> |
|||
<field name="origin_value_char"/> |
|||
<field name="new_value_char"/> |
|||
|
|||
<field name="origin_value_date"/> |
|||
<field name="new_value_date"/> |
|||
|
|||
<field name="origin_value_datetime"/> |
|||
<field name="new_value_datetime"/> |
|||
|
|||
<field name="origin_value_float"/> |
|||
<field name="new_value_float"/> |
|||
|
|||
<field name="origin_value_integer"/> |
|||
<field name="new_value_integer"/> |
|||
|
|||
<field name="origin_value_text"/> |
|||
<field name="new_value_text"/> |
|||
|
|||
<field name="origin_value_boolean"/> |
|||
<field name="new_value_boolean"/> |
|||
|
|||
<field name="origin_value_reference"/> |
|||
<field name="new_value_reference"/> |
|||
|
|||
</group> |
|||
</sheet> |
|||
</form> |
|||
</field> |
|||
</record> |
|||
|
|||
<record id="view_res_partner_changeset_search" model="ir.ui.view"> |
|||
<field name="name">res.partner.changeset.search</field> |
|||
<field name="model">res.partner.changeset</field> |
|||
<field name="arch" type="xml"> |
|||
<search string="Partner Changeset"> |
|||
<field name="partner_id"/> |
|||
<filter string="Pending" name="draft" |
|||
domain="[('state','=','draft')]"/> |
|||
<filter string="Done" name="done" |
|||
domain="[('state','=','done')]"/> |
|||
<group expand="0" string="Group By"> |
|||
<filter string="Partner" |
|||
name="groupby_partner_id" |
|||
context="{'group_by': 'partner_id'}"/> |
|||
<filter string="State" |
|||
name="groupby_state" |
|||
context="{'group_by': 'state'}"/> |
|||
</group> |
|||
</search> |
|||
</field> |
|||
</record> |
|||
|
|||
<record model="ir.actions.act_window" id="action_res_partner_changeset_view"> |
|||
<field name="name">Partner Changeset</field> |
|||
<field name="type">ir.actions.act_window</field> |
|||
<field name="res_model">res.partner.changeset</field> |
|||
<field name="view_type">form</field> |
|||
<field name="view_mode">tree,form</field> |
|||
<field name="context">{'search_default_draft': 1}</field> |
|||
<field name="search_view_id" ref="view_res_partner_changeset_search"/> |
|||
</record> |
|||
|
|||
<menuitem id="menu_res_partner_changeset" |
|||
parent="menu_changeset" |
|||
sequence="20" |
|||
name="Changesets" |
|||
action="action_res_partner_changeset_view"/> |
|||
</data> |
|||
</openerp> |
@ -0,0 +1,41 @@ |
|||
<?xml version="1.0" encoding="utf-8"?> |
|||
<openerp> |
|||
<data noupdate="0"> |
|||
|
|||
<record id="res_partner_view_buttons" model="ir.ui.view"> |
|||
<field name="name">res.partner.view.buttons</field> |
|||
<field name="model">res.partner</field> |
|||
<field name="inherit_id" ref="base.view_partner_form" /> |
|||
<field name="priority" eval="18"/> |
|||
<field name="groups_id" eval="[(4, ref('partner_changeset.group_changeset_user'))]"/> |
|||
<field name="arch" type="xml"> |
|||
<xpath expr="//div[@name='buttons']" position="inside"> |
|||
<button class="oe_inline oe_stat_button" |
|||
type="action" |
|||
name="%(partner_changeset.action_res_partner_changeset_view)d" |
|||
context="{'search_default_draft': 1, 'search_default_partner_id': active_id}" |
|||
icon="fa-code-fork"> |
|||
<field string="Changes" |
|||
name="count_pending_changesets" |
|||
widget="statinfo"/> |
|||
</button> |
|||
</xpath> |
|||
</field> |
|||
</record> |
|||
|
|||
<record id="view_res_partner_filter" model="ir.ui.view"> |
|||
<field name="name">res.partner.select</field> |
|||
<field name="model">res.partner</field> |
|||
<field name="inherit_id" ref="base.view_res_partner_filter" /> |
|||
<field name="arch" type="xml"> |
|||
<filter name="customer" position="after"> |
|||
<separator/> |
|||
<filter string="Pending Changesets" |
|||
name="pending_changesets" |
|||
domain="[('count_pending_changesets', '>', 0)]"/> |
|||
</filter> |
|||
</field> |
|||
</record> |
|||
|
|||
</data> |
|||
</openerp> |
Write
Preview
Loading…
Cancel
Save
Reference in new issue