[ADD] upstream patching till 88ff6beb016702cbda710aa31bbf10d51ab68fab

This commit is contained in:
Parthiv 2018-07-23 14:16:49 +05:30
parent efb6c34f02
commit 973d6f5003
20 changed files with 96 additions and 38 deletions

View File

@ -1,3 +1,4 @@
from flectra import api
from flectra.addons.account.tests.account_test_classes import AccountingTestCase from flectra.addons.account.tests.account_test_classes import AccountingTestCase
import time import time
import unittest import unittest

View File

@ -137,6 +137,12 @@ class Partner(models.Model):
for k, v in vals.items(): for k, v in vals.items():
partner[k] = v partner[k] = v
def write(self, vals):
res = super(Partner, self).write(vals)
if 'country_id' in vals:
self._set_street()
return res
class Company(models.Model): class Company(models.Model):
_inherit = 'res.company' _inherit = 'res.company'

View File

@ -51,5 +51,5 @@ class TestStreetFields(TransactionCase):
self.write_and_assert(p1, {'street': 'Chaussee de Namur'}, 'Chaussee de Namur', 'Chaussee de Namur', '', '') self.write_and_assert(p1, {'street': 'Chaussee de Namur'}, 'Chaussee de Namur', 'Chaussee de Namur', '', '')
self.write_and_assert(p1, {'street_name': 'Chee de Namur', 'street_number': '40'}, 'Chee de Namur, 40', 'Chee de Namur', '40', '') self.write_and_assert(p1, {'street_name': 'Chee de Namur', 'street_number': '40'}, 'Chee de Namur, 40', 'Chee de Namur', '40', '')
self.write_and_assert(p1, {'street_number2': '4'}, 'Chee de Namur, 40/4', 'Chee de Namur', '40', '4') self.write_and_assert(p1, {'street_number2': '4'}, 'Chee de Namur, 40/4', 'Chee de Namur', '40', '4')
#we don't recompute the street fields when we change the country #we do recompute the street fields when we change the country
self.write_and_assert(p1, {'country_id': self.env.ref('base.us').id}, 'Chee de Namur, 40/4', 'Chee de Namur', '40', '4') self.write_and_assert(p1, {'country_id': self.env.ref('base.us').id}, '40/4 Chee de Namur', 'Chee de Namur', '40', '4')

View File

@ -1,6 +1,7 @@
# -*- coding: utf-8 -*- # -*- coding: utf-8 -*-
from flectra import models, fields, api from flectra import models, fields, api, _
from flectra.exceptions import UserError
class IrModelFields(models.Model): class IrModelFields(models.Model):

View File

@ -4,7 +4,7 @@ import base64
from email.utils import formataddr from email.utils import formataddr
import re import re
import uuid from uuid import uuid4
from flectra import _, api, fields, models, modules, tools from flectra import _, api, fields, models, modules, tools
from flectra.exceptions import UserError from flectra.exceptions import UserError
@ -56,7 +56,7 @@ class Channel(models.Model):
('channel', 'Channel')], ('channel', 'Channel')],
'Channel Type', default='channel') 'Channel Type', default='channel')
description = fields.Text('Description') description = fields.Text('Description')
uuid = fields.Char('UUID', size=50, index=True, default=lambda self: '%s' % uuid.uuid4(), copy=False) uuid = fields.Char('UUID', size=50, index=True, default=lambda self: str(uuid4()), copy=False)
email_send = fields.Boolean('Send messages by email', default=False) email_send = fields.Boolean('Send messages by email', default=False)
# multi users channel # multi users channel
channel_last_seen_partner_ids = fields.One2many('mail.channel.partner', 'channel_id', string='Last Seen') channel_last_seen_partner_ids = fields.One2many('mail.channel.partner', 'channel_id', string='Last Seen')

View File

@ -7,7 +7,7 @@ from dateutil.relativedelta import relativedelta
from flectra import api, fields, models, SUPERUSER_ID, _ from flectra import api, fields, models, SUPERUSER_ID, _
from flectra.tools import DEFAULT_SERVER_DATETIME_FORMAT from flectra.tools import DEFAULT_SERVER_DATETIME_FORMAT
from flectra.tools.float_utils import float_is_zero, float_compare from flectra.tools.float_utils import float_is_zero, float_compare
from flectra.exceptions import UserError, AccessError, ValidationError from flectra.exceptions import UserError, AccessError
from flectra.tools.misc import formatLang from flectra.tools.misc import formatLang
from flectra.addons.base.res.res_partner import WARNING_MESSAGE, WARNING_HELP from flectra.addons.base.res.res_partner import WARNING_MESSAGE, WARNING_HELP
from flectra.addons import decimal_precision as dp from flectra.addons import decimal_precision as dp
@ -167,8 +167,8 @@ class PurchaseOrder(models.Model):
('invoiced', 'No Bill to Receive'), ('invoiced', 'No Bill to Receive'),
], string='Billing Status', compute='_get_invoiced', store=True, readonly=True, copy=False, default='no') ], string='Billing Status', compute='_get_invoiced', store=True, readonly=True, copy=False, default='no')
picking_count = fields.Integer(compute='_compute_picking', string='Receptions', default=0, store=True) picking_count = fields.Integer(compute='_compute_picking', string='Receptions', default=0, store=True, compute_sudo=True)
picking_ids = fields.Many2many('stock.picking', compute='_compute_picking', string='Receptions', copy=False, store=True) picking_ids = fields.Many2many('stock.picking', compute='_compute_picking', string='Receptions', copy=False, store=True, compute_sudo=True)
# There is no inverse function on purpose since the date may be different on each line # There is no inverse function on purpose since the date may be different on each line
date_planned = fields.Datetime(string='Scheduled Date', compute='_compute_date_planned', store=True, index=True) date_planned = fields.Datetime(string='Scheduled Date', compute='_compute_date_planned', store=True, index=True)
@ -669,7 +669,7 @@ class PurchaseOrderLine(models.Model):
# Replace by invoiced Qty # Replace by invoiced Qty
qty_invoiced = fields.Float(compute='_compute_qty_invoiced', string="Billed Qty", digits=dp.get_precision('Product Unit of Measure'), store=True) qty_invoiced = fields.Float(compute='_compute_qty_invoiced', string="Billed Qty", digits=dp.get_precision('Product Unit of Measure'), store=True)
qty_received = fields.Float(compute='_compute_qty_received', string="Received Qty", digits=dp.get_precision('Product Unit of Measure'), store=True) qty_received = fields.Float(compute='_compute_qty_received', string="Received Qty", digits=dp.get_precision('Product Unit of Measure'), store=True, compute_sudo=True)
partner_id = fields.Many2one('res.partner', related='order_id.partner_id', string='Partner', readonly=True, store=True) partner_id = fields.Many2one('res.partner', related='order_id.partner_id', string='Partner', readonly=True, store=True)
currency_id = fields.Many2one(related='order_id.currency_id', store=True, string='Currency', readonly=True) currency_id = fields.Many2one(related='order_id.currency_id', store=True, string='Currency', readonly=True)

View File

@ -8,7 +8,7 @@ from flectra.tools import float_compare
class PurchaseOrderLine(models.Model): class PurchaseOrderLine(models.Model):
_inherit = 'purchase.order.line' _inherit = 'purchase.order.line'
qty_received = fields.Float(compute='_compute_qty_received', string="Received Qty", store=True) qty_received = fields.Float(compute='_compute_qty_received', string="Received Qty", store=True, compute_sudo=True)
def _compute_qty_received(self): def _compute_qty_received(self):
super(PurchaseOrderLine, self)._compute_qty_received() super(PurchaseOrderLine, self)._compute_qty_received()

View File

@ -1,7 +1,7 @@
# -*- coding: utf-8 -*- # -*- coding: utf-8 -*-
# Part of Odoo, Flectra. See LICENSE file for full copyright and licensing details. # Part of Odoo, Flectra. See LICENSE file for full copyright and licensing details.
from flectra import api, models from flectra import api, models, fields
import logging import logging

View File

@ -107,7 +107,7 @@ class LandedCost(models.Model):
'landed_cost_value': new_landed_cost_value, 'landed_cost_value': new_landed_cost_value,
'value': line.move_id.value + cost_to_add, 'value': line.move_id.value + cost_to_add,
'remaining_value': line.move_id.remaining_value + cost_to_add, 'remaining_value': line.move_id.remaining_value + cost_to_add,
'price_unit': (line.move_id.value + new_landed_cost_value) / line.move_id.product_qty, 'price_unit': (line.move_id.value + cost_to_add) / line.move_id.product_qty,
}) })
# `remaining_qty` is negative if the move is out and delivered proudcts that were not # `remaining_qty` is negative if the move is out and delivered proudcts that were not
# in stock. # in stock.

View File

@ -1758,7 +1758,7 @@ var StatInfo = AbstractField.extend({
*/ */
_render: function () { _render: function () {
var options = { var options = {
value: this.value || 0, value: this._formatValue(this.value || 0),
}; };
if (! this.attrs.nolabel) { if (! this.attrs.nolabel) {
if (this.nodeOptions.label_field && this.recordData[this.nodeOptions.label_field]) { if (this.nodeOptions.label_field && this.recordData[this.nodeOptions.label_field]) {

View File

@ -4000,6 +4000,40 @@ QUnit.module('basic_fields', {
QUnit.module('StatInfo'); QUnit.module('StatInfo');
QUnit.test('statinfo widget formats decimal precision', function (assert) {
// sometimes the round method can return numbers such as 14.000001
// when asked to round a number to 2 decimals, as such is the behaviour of floats.
// we check that even in that eventuality, only two decimals are displayed
assert.expect(2);
this.data.partner.fields.monetary = {string: "Monetary", type: 'monetary'};
this.data.partner.records[0].monetary = 9.999999;
this.data.partner.records[0].currency_id = 1;
var form = createView({
View: FormView,
model: 'partner',
data: this.data,
arch: '<form string="Partners">' +
'<button class="oe_stat_button" name="items" icon="fa-gear">' +
'<field name="qux" widget="statinfo"/>' +
'</button>' +
'<button class="oe_stat_button" name="money" icon="fa-money">' +
'<field name="monetary" widget="statinfo"/>' +
'</button>' +
'</form>',
res_id: 1,
});
// formatFloat renders according to this.field.digits
assert.strictEqual(form.$('.oe_stat_button .o_field_widget.o_stat_info .o_stat_value').eq(0).text(),
'0.4', "Default precision should be [16,1]");
assert.strictEqual(form.$('.oe_stat_button .o_field_widget.o_stat_info .o_stat_value').eq(1).text(),
'10.00', "Currency decimal precision should be 2");
form.destroy();
});
QUnit.test('statinfo widget in form view', function (assert) { QUnit.test('statinfo widget in form view', function (assert) {
assert.expect(9); assert.expect(9);

View File

@ -8,6 +8,7 @@ var Widget = require('web.Widget');
var weContext = require('web_editor.context'); var weContext = require('web_editor.context');
var rte = require('web_editor.rte'); var rte = require('web_editor.rte');
var weWidgets = require('web_editor.widget'); var weWidgets = require('web_editor.widget');
var Dialog = require('web.Dialog');
var _t = core._t; var _t = core._t;
@ -26,6 +27,7 @@ var RTETranslatorWidget = rte.Class.extend({
* @override * @override
*/ */
_saveElement: function ($el, context, withLang) { _saveElement: function ($el, context, withLang) {
var self = this;
if ($el.data('oe-translation-id')) { if ($el.data('oe-translation-id')) {
return this._rpc({ return this._rpc({
model: 'ir.translation', model: 'ir.translation',
@ -35,7 +37,9 @@ var RTETranslatorWidget = rte.Class.extend({
this._getEscapedElement($el).html(), this._getEscapedElement($el).html(),
context || weContext.get() context || weContext.get()
], ],
}); }).fail(function (error) {
Dialog.alert(null, error.data.message);
});
} }
return this._super($el, context, withLang === undefined ? true : withLang); return this._super($el, context, withLang === undefined ? true : withLang);
}, },

View File

@ -546,23 +546,29 @@ class IrModelFields(models.Model):
This method prevents the modification/deletion of many2one fields This method prevents the modification/deletion of many2one fields
that have an inverse one2many, for instance. that have an inverse one2many, for instance.
""" """
failed_dependencies = []
for rec in self:
model = self.env[rec.model]
field = model._fields[rec.name]
for dependant, path in model._field_triggers.get(field, ()):
if dependant.manual:
failed_dependencies.append((field, dependant))
for inverse in model._field_inverses.get(field, ()):
if inverse.manual and inverse.type == 'one2many':
failed_dependencies.append((field, inverse))
if not self._context.get(MODULE_UNINSTALL_FLAG) and failed_dependencies:
msg = _("The field '%s' cannot be removed because the field '%s' depends on it.")
raise UserError(msg % failed_dependencies[0])
elif failed_dependencies:
dependants = {rel[1] for rel in failed_dependencies}
to_unlink = [self._get(field.model_name, field.name) for field in dependants]
self.browse().union(*to_unlink).unlink()
self = self.filtered(lambda record: record.state == 'manual') self = self.filtered(lambda record: record.state == 'manual')
if not self: if not self:
return return
for record in self:
model = self.env[record.model]
field = model._fields[record.name]
if field.type == 'many2one' and model._field_inverses.get(field):
if self._context.get(MODULE_UNINSTALL_FLAG):
# automatically unlink the corresponding one2many field(s)
inverses = self.search([('relation', '=', field.model_name),
('relation_field', '=', field.name)])
inverses.unlink()
continue
msg = _("The field '%s' cannot be removed because the field '%s' depends on it.")
raise UserError(msg % (field, model._field_inverses[field][0]))
# remove fields from registry, and check that views are not broken # remove fields from registry, and check that views are not broken
fields = [self.env[record.model]._pop_field(record.name) for record in self] fields = [self.env[record.model]._pop_field(record.name) for record in self]
domain = expression.OR([('arch_db', 'like', record.name)] for record in self) domain = expression.OR([('arch_db', 'like', record.name)] for record in self)

View File

@ -24,9 +24,10 @@ _logger = logging.getLogger(__name__)
class TestPyLint(TransactionCase): class TestPyLint(TransactionCase):
ENABLED_CODES = [ ENABLED_CODES = [
'E0601', # using variable before assignment 'used-before-assignment',
'W0123', # eval used 'undefined-variable',
'W0101', # unreachable code 'eval-used',
'unreachable',
'mixed-indentation', 'mixed-indentation',

View File

@ -1630,7 +1630,8 @@ class Datetime(Field):
# Received data is returned as buffer (in Python 2) or memoryview (in Python 3). # Received data is returned as buffer (in Python 2) or memoryview (in Python 3).
_BINARY = memoryview _BINARY = memoryview
if pycompat.PY2: if pycompat.PY2:
_BINARY = buffer #pylint: disable=buffer-builtin #pylint: disable=buffer-builtin,undefined-variable
_BINARY = buffer
class Binary(Field): class Binary(Field):
type = 'binary' type = 'binary'

View File

@ -21,7 +21,8 @@ if pycompat.PY2:
fp, fname = tools.file_open(path, pathinfo=True) fp, fname = tools.file_open(path, pathinfo=True)
fp2 = None fp2 = None
if not isinstance(fp, file): # pylint: disable=file-builtin # pylint: disable=file-builtin,undefined-variable
if not isinstance(fp, file):
# imp.load_source need a real file object, so we create # imp.load_source need a real file object, so we create
# one from the file-like object we get from file_open # one from the file-like object we get from file_open
fp2 = os.tmpfile() fp2 = os.tmpfile()

View File

@ -1069,7 +1069,7 @@ def formatLang(env, value, digits=None, grouping=True, monetary=False, dp=False,
digits = decimal_precision_obj.precision_get(dp) digits = decimal_precision_obj.precision_get(dp)
elif currency_obj: elif currency_obj:
digits = currency_obj.decimal_places digits = currency_obj.decimal_places
elif (hasattr(value, '_field') and isinstance(value._field, (float_field, function_field)) and value._field.digits): elif (hasattr(value, '_field') and getattr(value._field, 'digits', None)):
digits = value._field.digits[1] digits = value._field.digits[1]
if not digits and digits is not 0: if not digits and digits is not 0:
digits = DEFAULT_DIGITS digits = DEFAULT_DIGITS

View File

@ -13,7 +13,7 @@ PY2 = sys.version_info[0] == 2
_Writer = collections.namedtuple('_Writer', 'writerow writerows') _Writer = collections.namedtuple('_Writer', 'writerow writerows')
if PY2: if PY2:
# pylint: disable=long-builtin,unichr-builtin,unicode-builtin # pylint: disable=long-builtin,unichr-builtin,unicode-builtin,undefined-variable
unichr = unichr unichr = unichr
text_type = unicode text_type = unicode
string_types = (str, unicode) string_types = (str, unicode)

View File

@ -1,7 +1,7 @@
# -*- coding: utf-8 -*- # -*- coding: utf-8 -*-
from lxml import etree from lxml import etree
from flectra.tools.misc import file_open from flectra.tools.misc import file_open
from flectra.exceptions import UserError
def check_with_xsd(tree_or_str, stream): def check_with_xsd(tree_or_str, stream):
raise UserError("Method 'check_with_xsd' deprecated ") raise UserError("Method 'check_with_xsd' deprecated ")

View File

@ -3,11 +3,14 @@ decorator==4.0.10
docutils==0.12 docutils==0.12
ebaysdk==2.1.5 ebaysdk==2.1.5
feedparser==5.2.1 feedparser==5.2.1
gevent==1.1.2 ; sys_platform != 'win32' gevent==1.1.2 ; sys_platform != 'win32' and python_version < '3.7'
greenlet==0.4.10 gevent==1.3.4 ; sys_platform != 'win32' and python_version >= '3.7'
greenlet==0.4.10 ; python_version < '3.7'
greenlet==0.4.13 ; python_version >= '3.7'
html2text==2016.9.19 html2text==2016.9.19
Jinja2==2.8 Jinja2==2.8
lxml==3.7.1 ; sys_platform != 'win32' lxml==3.7.1 ; sys_platform != 'win32' and python_version < '3.7'
lxml==4.2.3 ; sys_platform != 'win32' and python_version >= '3.7'
lxml ; sys_platform == 'win32' lxml ; sys_platform == 'win32'
Mako==1.0.4 Mako==1.0.4
MarkupSafe==0.23 MarkupSafe==0.23