Initial import from Docker volume
This commit is contained in:
444
model/fields/one2many.py
Executable file
444
model/fields/one2many.py
Executable file
@@ -0,0 +1,444 @@
|
||||
# This file is part of Tryton. The COPYRIGHT file at the top level of
|
||||
# this repository contains the full copyright notices and license terms.
|
||||
from collections import defaultdict
|
||||
from itertools import chain
|
||||
|
||||
from sql import Literal
|
||||
from sql.conditionals import Coalesce
|
||||
from sql.operators import Exists
|
||||
|
||||
from trytond.config import config
|
||||
from trytond.pool import Pool
|
||||
from trytond.pyson import PYSONEncoder
|
||||
from trytond.tools import cached_property, grouped_slice
|
||||
from trytond.transaction import Transaction
|
||||
|
||||
from .field import (
|
||||
Field, context_validate, domain_method, domain_validate, get_eval_fields,
|
||||
instanciate_values, instantiate_context, search_order_validate,
|
||||
size_validate)
|
||||
|
||||
_subquery_threshold = config.getint('database', 'subquery_threshold')
|
||||
|
||||
|
||||
class One2Many(Field):
|
||||
'''
|
||||
Define one2many field (``list``).
|
||||
'''
|
||||
_type = 'one2many'
|
||||
|
||||
def __init__(self, model_name, field, string='', add_remove=None,
|
||||
order=None, datetime_field=None, size=None, search_order=None,
|
||||
search_context=None, help='', required=False, readonly=False,
|
||||
domain=None, filter=None, states=None, on_change=None,
|
||||
on_change_with=None, depends=None, context=None, loading='lazy'):
|
||||
'''
|
||||
:param model_name: The name of the target model.
|
||||
:param field: The name of the field that handle the reverse many2one or
|
||||
reference.
|
||||
:param add_remove: A list that defines a domain on add/remove.
|
||||
See domain on ModelStorage.search.
|
||||
:param order: a list of tuples that are constructed like this:
|
||||
``('field name', 'DESC|ASC')``
|
||||
allowing to specify the order of result.
|
||||
:param datetime_field: The name of the field that contains the datetime
|
||||
value to read the target records.
|
||||
:param search_order: The order to use when searching for records
|
||||
:param search_context: The context to use when searching for a record
|
||||
:param filter: A domain to filter target records.
|
||||
'''
|
||||
super(One2Many, self).__init__(string=string, help=help,
|
||||
required=required, readonly=readonly, domain=domain, states=states,
|
||||
on_change=on_change, on_change_with=on_change_with,
|
||||
depends=depends, context=context, loading=loading)
|
||||
self.model_name = model_name
|
||||
self.field = field
|
||||
self.__add_remove = None
|
||||
self.add_remove = add_remove
|
||||
self.order = order
|
||||
self.datetime_field = datetime_field
|
||||
self.__size = None
|
||||
self.size = size
|
||||
self.__search_order = None
|
||||
self.search_order = search_order
|
||||
self.__search_context = None
|
||||
self.search_context = search_context or {}
|
||||
self.__filter = None
|
||||
self.filter = filter
|
||||
|
||||
__init__.__doc__ += Field.__init__.__doc__
|
||||
|
||||
def _get_add_remove(self):
|
||||
return self.__add_remove
|
||||
|
||||
def _set_add_remove(self, value):
|
||||
if value is not None:
|
||||
domain_validate(value)
|
||||
self.__add_remove = value
|
||||
|
||||
add_remove = property(_get_add_remove, _set_add_remove)
|
||||
|
||||
def _get_size(self):
|
||||
return self.__size
|
||||
|
||||
def _set_size(self, value):
|
||||
size_validate(value)
|
||||
self.__size = value
|
||||
|
||||
size = property(_get_size, _set_size)
|
||||
|
||||
@property
|
||||
def search_order(self):
|
||||
return self.__search_order
|
||||
|
||||
@search_order.setter
|
||||
def search_order(self, value):
|
||||
search_order_validate(value)
|
||||
self.__search_order = value
|
||||
|
||||
@property
|
||||
def search_context(self):
|
||||
return self.__search_context
|
||||
|
||||
@search_context.setter
|
||||
def search_context(self, value):
|
||||
context_validate(value)
|
||||
self.__search_context = value
|
||||
|
||||
@cached_property
|
||||
def display_depends(self):
|
||||
depends = super().display_depends
|
||||
if self.datetime_field:
|
||||
depends.add(self.datetime_field)
|
||||
return depends
|
||||
|
||||
@cached_property
|
||||
def edition_depends(self):
|
||||
depends = super().edition_depends
|
||||
for attribute in ['add_remove', 'size']:
|
||||
depends |= get_eval_fields(getattr(self, attribute))
|
||||
return depends
|
||||
|
||||
@cached_property
|
||||
def validation_depends(self):
|
||||
return super().validation_depends | get_eval_fields(self.size)
|
||||
|
||||
def sql_type(self):
|
||||
return None
|
||||
|
||||
@property
|
||||
def filter(self):
|
||||
return self.__filter
|
||||
|
||||
@filter.setter
|
||||
def filter(self, value):
|
||||
if value is not None:
|
||||
domain_validate(value)
|
||||
self.__filter = value
|
||||
|
||||
def get(self, ids, model, name, values=None):
|
||||
'''
|
||||
Return target records ordered.
|
||||
'''
|
||||
Target = self.get_target()
|
||||
field = Target._fields[self.field]
|
||||
res = {}
|
||||
for i in ids:
|
||||
res[i] = []
|
||||
|
||||
if field.sortable(Target):
|
||||
if field._type == 'reference':
|
||||
order = [(self.field, None)]
|
||||
else:
|
||||
order = [(self.field + '.id', None)]
|
||||
else:
|
||||
order = []
|
||||
if self.order:
|
||||
order += self.order
|
||||
elif Target._order:
|
||||
order += Target._order
|
||||
targets = []
|
||||
for sub_ids in grouped_slice(ids):
|
||||
if field._type == 'reference':
|
||||
references = ['%s,%s' % (model.__name__, x) for x in sub_ids]
|
||||
clause = [(self.field, 'in', references)]
|
||||
else:
|
||||
clause = [(self.field, 'in', list(sub_ids))]
|
||||
if self.filter:
|
||||
clause.append(self.filter)
|
||||
targets.append(Target.search(clause, order=order))
|
||||
targets = Target.browse(list(chain(*targets)))
|
||||
|
||||
for target in targets:
|
||||
origin_id = getattr(target, self.field).id
|
||||
res[origin_id].append(target.id)
|
||||
return dict((key, tuple(value)) for key, value in res.items())
|
||||
|
||||
def set(self, Model, name, ids, values, *args):
|
||||
'''
|
||||
Set the values.
|
||||
values: A list of tuples:
|
||||
(``create``, ``[{<field name>: value}, ...]``),
|
||||
(``write``, [``<ids>``, ``{<field name>: value}``, ...]),
|
||||
(``delete``, ``<ids>``),
|
||||
(``add``, ``<ids>``),
|
||||
(``remove``, ``<ids>``),
|
||||
(``copy``, ``<ids>``, ``[{<field name>: value}, ...]``)
|
||||
'''
|
||||
Target = self.get_target()
|
||||
field = Target._fields[self.field]
|
||||
to_create = []
|
||||
to_write = []
|
||||
to_delete = []
|
||||
|
||||
def search_clause(ids):
|
||||
if field._type == 'reference':
|
||||
references = ['%s,%s' % (Model.__name__, x) for x in ids]
|
||||
return (self.field, 'in', references)
|
||||
else:
|
||||
return (self.field, 'in', ids)
|
||||
|
||||
def field_value(record_id):
|
||||
if field._type == 'reference':
|
||||
return '%s,%s' % (Model.__name__, record_id)
|
||||
else:
|
||||
return record_id
|
||||
|
||||
def target_value(record):
|
||||
if record is None:
|
||||
return None
|
||||
if field._type == 'reference':
|
||||
return str(record)
|
||||
else:
|
||||
return record.id
|
||||
|
||||
def create(ids, vlist):
|
||||
for record_id in ids:
|
||||
value = field_value(record_id)
|
||||
for values in vlist:
|
||||
values = values.copy()
|
||||
values[self.field] = value
|
||||
to_create.append(values)
|
||||
|
||||
def write(_, *args):
|
||||
actions = iter(args)
|
||||
to_write.extend(sum(((Target.browse(ids), values)
|
||||
for ids, values in zip(actions, actions)), ()))
|
||||
|
||||
def delete(_, target_ids):
|
||||
to_delete.extend(Target.browse(target_ids))
|
||||
|
||||
def add(ids, target_ids):
|
||||
target_ids = list(map(int, target_ids))
|
||||
if not target_ids:
|
||||
return
|
||||
targets = Target.browse(target_ids)
|
||||
for record_id in ids:
|
||||
fvalue = field_value(record_id)
|
||||
to_update = [t for t in targets
|
||||
if target_value(getattr(t, self.field)) != fvalue]
|
||||
if to_update:
|
||||
to_write.extend((to_update, {
|
||||
self.field: fvalue,
|
||||
}))
|
||||
|
||||
def remove(ids, target_ids):
|
||||
target_ids = list(map(int, target_ids))
|
||||
if not target_ids:
|
||||
return
|
||||
for sub_ids in grouped_slice(target_ids):
|
||||
targets = Target.search([
|
||||
search_clause(ids),
|
||||
('id', 'in', list(sub_ids)),
|
||||
])
|
||||
to_write.extend((targets, {
|
||||
self.field: None,
|
||||
}))
|
||||
|
||||
def copy(ids, copy_ids, default=None):
|
||||
copy_ids = list(map(int, copy_ids))
|
||||
|
||||
if default is None:
|
||||
default = {}
|
||||
default = default.copy()
|
||||
copies = Target.browse(copy_ids)
|
||||
for record_id in ids:
|
||||
default[self.field] = field_value(record_id)
|
||||
Target.copy(copies, default=default)
|
||||
|
||||
actions = {
|
||||
'create': create,
|
||||
'write': write,
|
||||
'delete': delete,
|
||||
'add': add,
|
||||
'remove': remove,
|
||||
'copy': copy,
|
||||
}
|
||||
args = iter((ids, values) + args)
|
||||
for ids, values in zip(args, args):
|
||||
if not values:
|
||||
continue
|
||||
for value in values:
|
||||
action = value[0]
|
||||
args = value[1:]
|
||||
actions[action](ids, *args)
|
||||
# Ordered operations to avoid uniqueness/overlapping constraints
|
||||
if to_delete:
|
||||
Target.delete(to_delete)
|
||||
if to_write:
|
||||
Target.write(*to_write)
|
||||
if to_create:
|
||||
Target.create(to_create)
|
||||
|
||||
def get_target(self):
|
||||
'Return the target Model'
|
||||
return Pool().get(self.model_name)
|
||||
|
||||
def __set__(self, inst, value):
|
||||
Target = self.get_target()
|
||||
ctx = instantiate_context(self, inst)
|
||||
extra = {}
|
||||
if self.field:
|
||||
extra[self.field] = inst
|
||||
with Transaction().set_context(ctx):
|
||||
records = instanciate_values(Target, value, **extra)
|
||||
super(One2Many, self).__set__(inst, records)
|
||||
|
||||
def remove(self, inst, records):
|
||||
records = set(records)
|
||||
if inst._removed is None:
|
||||
inst._removed = defaultdict(set)
|
||||
inst._removed[self.name].update(map(int, records))
|
||||
setattr(
|
||||
inst, self.name,
|
||||
[r for r in getattr(inst, self.name) if r not in records])
|
||||
|
||||
@domain_method
|
||||
def convert_domain(self, domain, tables, Model):
|
||||
from ..modelsql import convert_from
|
||||
pool = Pool()
|
||||
Rule = pool.get('ir.rule')
|
||||
Target = self.get_target()
|
||||
transaction = Transaction()
|
||||
table, _ = tables[None]
|
||||
name, operator, value = domain[:3]
|
||||
assert operator not in {'where', 'not where'} or '.' not in name
|
||||
|
||||
if Target._history and transaction.context.get('_datetime'):
|
||||
target = Target.__table_history__()
|
||||
history_where = (
|
||||
Coalesce(target.write_date, target.create_date)
|
||||
<= transaction.context['_datetime'])
|
||||
else:
|
||||
target = Target.__table__()
|
||||
history_where = None
|
||||
origin_field = Target._fields[self.field]
|
||||
origin = getattr(Target, self.field).sql_column(target)
|
||||
origin_where = None
|
||||
if origin_field._type == 'reference':
|
||||
origin_where = origin.like(Model.__name__ + ',%')
|
||||
origin = origin_field.sql_id(origin, Target)
|
||||
|
||||
use_in = Target.estimated_count() < _subquery_threshold
|
||||
if '.' not in name:
|
||||
if value is None:
|
||||
if use_in:
|
||||
where = origin != value
|
||||
if history_where:
|
||||
where &= history_where
|
||||
if origin_where:
|
||||
where &= origin_where
|
||||
if self.filter:
|
||||
query = Target.search(
|
||||
self.filter, order=[], query=True)
|
||||
where &= origin.in_(query)
|
||||
query = target.select(origin, where=where)
|
||||
expression = ~table.id.in_(query)
|
||||
else:
|
||||
where = origin == table.id
|
||||
if history_where:
|
||||
where &= history_where
|
||||
if origin_where:
|
||||
where &= origin_where
|
||||
if self.filter:
|
||||
target_tables = {
|
||||
None: (target, None),
|
||||
}
|
||||
target_tables, clause = Target.search_domain(
|
||||
self.filter, tables=target_tables)
|
||||
where &= clause
|
||||
query_table = convert_from(None, target_tables)
|
||||
query = query_table.select(Literal(1), where=where)
|
||||
else:
|
||||
query = target.select(Literal(1), where=where)
|
||||
expression = ~Exists(query)
|
||||
|
||||
if operator == '!=':
|
||||
expression = ~expression
|
||||
return expression
|
||||
else:
|
||||
if isinstance(value, str):
|
||||
target_name = 'rec_name'
|
||||
else:
|
||||
target_name = 'id'
|
||||
else:
|
||||
_, target_name = name.split('.', 1)
|
||||
if operator not in {'where', 'not where'}:
|
||||
target_domain = [(target_name,) + tuple(domain[1:])]
|
||||
else:
|
||||
target_domain = value
|
||||
if origin_field._type == 'reference':
|
||||
target_domain.append(
|
||||
(self.field, 'like', Model.__name__ + ',%'))
|
||||
rule_domain = Rule.domain_get(Target.__name__, mode='read')
|
||||
if rule_domain:
|
||||
target_domain = [target_domain, rule_domain]
|
||||
if self.filter:
|
||||
target_domain = [target_domain, self.filter]
|
||||
target_tables = {
|
||||
None: (target, None),
|
||||
}
|
||||
tables, expression = Target.search_domain(
|
||||
target_domain, tables=target_tables)
|
||||
query_table = convert_from(None, target_tables)
|
||||
if use_in:
|
||||
query = query_table.select(origin, where=expression)
|
||||
expression = table.id.in_(query)
|
||||
else:
|
||||
query = query_table.select(
|
||||
Literal(1), where=expression & (origin == table.id))
|
||||
expression = Exists(query)
|
||||
|
||||
if operator == 'not where':
|
||||
expression = ~expression
|
||||
elif operator.startswith('!') or operator.startswith('not '):
|
||||
if use_in:
|
||||
expression |= ~table.id.in_(target.select(origin))
|
||||
else:
|
||||
expression |= ~Exists(target.select(
|
||||
Literal(1), where=origin == table.id))
|
||||
return expression
|
||||
|
||||
def definition(self, model, language):
|
||||
encoder = PYSONEncoder()
|
||||
definition = super().definition(model, language)
|
||||
if self.add_remove is not None:
|
||||
definition['add_remove'] = encoder.encode(self.add_remove)
|
||||
definition['datetime_field'] = self.datetime_field
|
||||
if self.filter:
|
||||
definition['domain'] = encoder.encode(
|
||||
['AND', self.domain, self.filter])
|
||||
definition['relation'] = self.model_name
|
||||
if self.field:
|
||||
definition['relation_field'] = self.field
|
||||
definition['search_context'] = encoder.encode(self.search_context)
|
||||
definition['search_order'] = encoder.encode(self.search_order)
|
||||
if self.size is not None:
|
||||
definition['size'] = encoder.encode(self.size)
|
||||
definition['order'] = (
|
||||
getattr(self.get_target(), '_order', None)
|
||||
if self.order is None else self.order)
|
||||
return definition
|
||||
|
||||
def sortable(self, model):
|
||||
return super().sortable(model) and hasattr(model, f'order_{self.name}')
|
||||
Reference in New Issue
Block a user