前一章介绍了向模型添加一些业务逻辑的能力。现在我们可以将按钮链接到业务代码,但是我们如何防止用户输入错误的数据呢?例如,在我们的房地产模块中,没有什么可以阻止用户设置负预期价格。
Odoo提供了两种方法来设置自动验证的不变量:Python约束和SQL约束。
# -*- coding: utf-8 -*-
# Part of Odoo. See LICENSE file for full copyright and licensing details.
from collections import defaultdict
from odoo import api, fields, models, _
from odoo.osv import expression
from odoo.exceptions import ValidationError
class AccountAnalyticDistribution(models.Model):
_name = 'account.analytic.distribution'
_description = 'Analytic Account Distribution'
_rec_name = 'account_id'
account_id = fields.Many2one('account.analytic.account', string='Analytic Account', required=True)
percentage = fields.Float(string='Percentage', required=True, default=100.0)
name = fields.Char(string='Name', related='account_id.name', readonly=False)
tag_id = fields.Many2one('account.analytic.tag', string="Parent tag", required=True)
_sql_constraints = [
('check_percentage', 'CHECK(percentage >= 0 AND percentage <= 100)',
'The percentage of an analytic distribution should be between 0 and 100.')
]
如果某些产品/服务的价格为零,则无法应用约束。您可以删除 有问题的数据,以便应用新的约束。
*odoo.api.constrains(args)
修饰约束检查器。
每个参数必须是检查中使用的字段名:
@api.constrains('name', 'description')
def _check_description(self):
for record in self:
if record.name == record.description:
raise ValidationError("Fields name and description must be different")
在其中一个命名字段已被修改的记录上调用。
如果验证失败,将引发ValidationError。
@constraints 只支持简单的字段名,不支持带点的名称(关系字段的字段,例如partner_id.customer),并且会被忽略。
只有当修饰方法中声明的字段包含在create或write调用中时,@constraints才会被触发。这意味着视图中不存在的字段将不会在创建记录期间触发调用。create的重写是必要的,以确保约束总是会被触发(例如,测试是值是否为空)。
Python约束被定义为一个用constraints()修饰的方法,并在记录集上调用。装饰符指定约束中涉及哪些字段。当修改这些字段中的任何一个时,将自动评估约束。如果不满足其不变量,该方法预计会引发异常:
from odoo.exceptions import ValidationError
...
@api.constrains('date_end')
def _check_date_end(self):
for record in self:
if record.date_end < fields.Date.today():
raise ValidationError("The end date cannot be set in the past")
# all records passed the test, don't return anything
在这里可以找到一个简单的例子:
@api.constrains('product_id')
def check_product_id(self):
if any(elem.product_id.type != 'product' for elem in self):
raise ValidationError(_('Quants cannot be created for consumables or services.'))
@api.constrains('quantity')
def check_quantity(self):
for quant in self:
if quant.location_id.usage != 'inventory' and quant.lot_id and quant.product_id.tracking == 'serial' \
and float_compare(abs(quant.quantity), 1, precision_rounding=quant.product_uom_id.rounding) > 0:
raise ValidationError(_('The serial number has already been assigned: \n Product: %s, Serial Number: %s') % (quant.product_id.display_name, quant.lot_id.name))
@api.constrains('location_id')
def check_location_id(self):
for quant in self:
if quant.location_id.usage == 'view':
raise ValidationError(_('You cannot take products from or deliver products to a location of type "view" (%s).') % quant.location_id.name)
我们的房地产模块开始看起来不错。我们添加了一些业务逻辑,现在我们确保 数据是一致的。但是,用户界面仍然有点粗糙。让我们看看如何 在下一章中改进它。