""" .. deprecated:: 1.0 Use the other converter instead, kept for ease of porting. """ import warnings from typing import Any, Dict, Optional from odoo import api, models from .base import Converter, Skip, logger from .field import Field from .model import Model from .models.ir_model_data import _XREF_IMD_MODULE warnings.warn("legacy converters are deprecated", DeprecationWarning) class Many2oneConverter(Field): """Converts many2one xref.""" def odoo_to_message( self, instance: models.Model, ctx: Optional[Dict] = None ) -> Any: field_value = super().odoo_to_message(instance, ctx) if field_value is not Skip: return field_value.env["ir.model.data"].object_to_xmlid( field_value ) if self.required_blank_value is not None: return self.required_blank_value return Skip def message_to_odoo( self, odoo_env: api.Environment, phase: str, message_value: Any, instance: models.Model, value_present: bool = True, ) -> Dict: if not value_present: return {} # Empty string is considered as NULL rel_record_id = ( odoo_env.ref(_XREF_IMD_MODULE, message_value).id if message_value else None ) if instance: field_value = getattr(instance, self.field_name) if ( rel_record_id and field_value.id == rel_record_id ) or not field_value: return {} return {self.field_name: rel_record_id} class Many2manyConverter(Field): """Converts many2many xref.""" def __init__(self, field_name: str, model_name: str): super().__init__(field_name) self.model_name = model_name def odoo_to_message( self, instance: models.Model, ctx: Optional[Dict] = None ) -> Any: values = super().odoo_to_message(instance, ctx) return [ value.env["ir.model.data"].object_to_xmlid(value) for value in values ] def message_to_odoo( self, odoo_env: api.Environment, phase: str, message_value: Any, instance: models.Model, value_present: bool = True, ) -> Dict: # if not present or value is None, do not update the values. if not value_present or message_value is None: return {} field_instances = odoo_env[self.model_name] for value in message_value: field_instances |= odoo_env["ir.model.data"].ref( _XREF_IMD_MODULE, value ) if instance and getattr(instance, self.field_name) == field_instances: return {} return {self.field_name: [(6, 0, field_instances.ids)]} class Many2oneFieldValueConverter(Field): """Converter for many2one field where a field of the many2one instance is used. The value must be unique by record. """ def __init__( self, field_name: str, model_name: str, many2one_field_name: str = "name", ): super().__init__(field_name) self.model_name = model_name self.many2one_field_name = many2one_field_name def odoo_to_message( self, instance: models.Model, ctx: Optional[Dict] = None ) -> Any: logger.debug(self.field_name) record = super().odoo_to_message(instance, ctx) if record is Skip: return Skip return getattr(record, self.many2one_field_name) def message_to_odoo( self, odoo_env: api.Environment, phase: str, message_value: Any, instance: models.Model, value_present: bool = True, ) -> Dict: if not value_present: return {} field_instance = odoo_env[self.model_name].search( [(self.many2one_field_name, "=", message_value)] ) if len(field_instance) > 1: logger.warning( "More than one match for %s: %s", self.many2one_field_name, field_instance, ) if instance and getattr(instance, self.field_name) == field_instance: return {} return {self.field_name: field_instance.id} class Many2manyFieldValueConverter(Field): """Converter for many2many field where a field of the many2one instance is used. The value must be unique by record. """ def __init__( self, field_name: str, model_name: str, many2many_field_name: str = "name", ): super().__init__(field_name) self.model_name = model_name self.many2many_field_name = many2many_field_name def odoo_to_message( self, instance: models.Model, ctx: Optional[Dict] = None ) -> Any: values = super().odoo_to_message(instance, ctx) if values is Skip: return Skip return [getattr(value, self.many2many_field_name) for value in values] def message_to_odoo( self, odoo_env: api.Environment, phase: str, message_value: Any, instance: models.Model, value_present: bool = True, ) -> Dict: if not value_present: return {} field_instances = odoo_env[self.model_name].search( [(self.many2many_field_name, "in", message_value)] ) if instance and getattr(instance, self.field_name) == field_instances: return {} return {self.field_name: [(6, 0, field_instances.ids)]} class One2manyConverter(Model): def __init__( self, field_name: str, __type__: str, converters: Dict[str, Converter], model_name: str, ): super().__init__(__type__, converters) self.field_name = field_name self.model_name = model_name def odoo_to_message( self, instance: models.Model, ctx: Optional[Dict] = None ) -> Any: return [ msg for msg in ( super().odoo_to_message(element, ctx) for element in getattr(instance, self.field_name) ) if msg is not Skip ] def message_to_odoo( self, odoo_env: api.Environment, phase: str, message_value: Any, instance: models.Model, value_present: bool = True, ) -> Dict: return {} def post_hook(self, instance: models.Model, message_data): if self.field_name in message_data: ids = [] for element in message_data: o2m_instance = super().get_instance(instance.env, element) vals_by_context = super().message_to_odoo( instance.env, "update", element, o2m_instance ) if not o2m_instance: context = dict(no_sync=True) context_, vals = vals_by_context.popitem(last=False) context.update(context_) o2m_instance = ( instance.env[self.model_name] .with_context(**context) .create(vals) ) o2m_instance.context_write(vals_by_context) ids.append(o2m_instance.id) super().post_hook(o2m_instance, element) instance.with_context(no_sync=True).write( {self.field_name: [(6, 0, ids)]} )