ORM API¶
Models¶
Model fields are defined as attributes on the model itself:
from odoo import models, fields
class AModel(models.Model):
    _name = 'a.model.name'
    field1 = fields.Char()
Waarschuwing
this means you cannot define a field and a method with the same name, the last one will silently overwrite the former ones.
By default, the field’s label (user-visible name) is a capitalized version of
the field name, this can be overridden with the string parameter.
field2 = fields.Integer(string="Field Label")
For the list of field types and parameters, see the fields reference.
Default values are defined as parameters on fields, either as a value:
name = fields.Char(default="a value")
or as a function called to compute the default value, which should return that value:
def _default_name(self):
    return self.get_value()
name = fields.Char(default=lambda self: self._default_name())
API
- class odoo.models.BaseModel[broncode]¶
- Base class for Odoo models. - Odoo models are created by inheriting one of the following: - Modelfor regular database-persisted models
- TransientModelfor temporary data, stored in the database but automatically vacuumed every so often
- AbstractModelfor abstract super classes meant to be shared by multiple inheriting models
 - The system automatically instantiates every model once per database. Those instances represent the available models on each database, and depend on which modules are installed on that database. The actual class of each instance is built from the Python classes that create and inherit from the corresponding model. - Every model instance is a “recordset”, i.e., an ordered collection of records of the model. Recordsets are returned by methods like - browse(),- search(), or field accesses. Records have no explicit representation: a record is represented as a recordset of one record.- To create a class that should not be instantiated, the - _registerattribute may be set to False.- _auto = False¶
- Whether a database table should be created. If set to - False, override- init()to create the database table.- Automatically defaults to - Truefor- Modeland- TransientModel,- Falsefor- AbstractModel.- Tip - To create a model without any table, inherit from - AbstractModel.
 - _log_access¶
- Whether the ORM should automatically generate and update the Access Log fields. - Defaults to whatever value was set for - _auto.
 - _register = False¶
- registry visibility 
 - _abstract = True¶
- Whether the model is abstract. - Zie ook 
 - _transient = False¶
- Whether the model is transient. - Zie ook 
 - _inherits = {}¶
- dictionary {‘parent_model’: ‘m2o_field’} mapping the _name of the parent business objects to the names of the corresponding foreign key fields to use: - _inherits = { 'a.model': 'a_field_id', 'b.model': 'b_field_id' } - implements composition-based inheritance: the new model exposes all the fields of the inherited models but stores none of them: the values themselves remain stored on the linked record. - Waarschuwing - if multiple fields with the same name are defined in the - _inherits-ed models, the inherited field will correspond to the last one (in the inherits list order).
 - _rec_name = None¶
- field to use for labeling records, default: - name
 - _order = 'id'¶
- default order field for searching results 
 - _check_company_auto = False¶
- On write and create, call - _check_companyto ensure companies consistency on the relational fields having- check_company=Trueas attribute.
 - _parent_name = 'parent_id'¶
- the many2one field used as parent field 
 - _parent_store = False¶
- set to True to compute parent_path field. - Alongside a - parent_pathfield, sets up an indexed storage of the tree structure of records, to enable faster hierarchical queries on the records of the current model using the- child_ofand- parent_ofdomain operators.
 - _fold_name = 'fold'¶
- field to determine folded groups in kanban views 
 
AbstractModel¶
- odoo.models.AbstractModel[broncode]¶
- alias of - odoo.orm.models.BaseModel
Model¶
- class odoo.models.Model(env: api.Environment, ids: tuple[IdType, ...], prefetch_ids: Reversible[IdType])[broncode]¶
- Main super-class for regular database-persisted Odoo models. - Odoo models are created by inheriting from this class: - class ResUsers(Model): ... - The system will later instantiate the class once per database (on which the class’ module is installed). - _auto = True¶
- Whether a database table should be created. If set to - False, override- init()to create the database table.- Automatically defaults to - Truefor- Modeland- TransientModel,- Falsefor- AbstractModel.- Tip - To create a model without any table, inherit from - AbstractModel.
 - _abstract = False¶
- Whether the model is abstract. - Zie ook 
 
TransientModel¶
- class odoo.models.TransientModel(env: api.Environment, ids: tuple[IdType, ...], prefetch_ids: Reversible[IdType])[broncode]¶
- Model super-class for transient records, meant to be temporarily persistent, and regularly vacuum-cleaned. - A TransientModel has a simplified access rights management, all users can create new records, and may only access the records they created. The superuser has unrestricted access to all TransientModel records. - _transient_max_count = 0¶
- maximum number of transient records, unlimited if - 0
 - _transient_max_hours = 1.0¶
- maximum idle lifetime (in hours), unlimited if - 0
 - _transient_vacuum()[broncode]¶
- Clean the transient records. - This unlinks old records from the transient model tables whenever the - _transient_max_countor- _transient_max_hoursconditions (if any) are reached.- Actual cleaning will happen only once every 5 minutes. This means this method can be called frequently (e.g. whenever a new record is created). - Example with both max_hours and max_count active: - Suppose max_hours = 0.2 (aka 12 minutes), max_count = 20, there are 55 rows in the table, 10 created/changed in the last 5 minutes, an additional 12 created/changed between 5 and 10 minutes ago, the rest created/changed more than 12 minutes ago. - age based vacuum will leave the 22 rows created/changed in the last 12 minutes 
- count based vacuum will wipe out another 12 rows. Not just 2, otherwise each addition would immediately cause the maximum to be reached again. 
- the 10 rows that have been created/changed the last 5 minutes will NOT be deleted 
 
 
Fields¶
- class odoo.fields.Field[broncode]¶
- The field descriptor contains the field definition, and manages accesses and assignments of the corresponding field on records. The following attributes may be provided when instantiating a field: - Parameters
- string (str) – the label of the field seen by users; if not set, the ORM takes the field name in the class (capitalized). 
- help (str) – the tooltip of the field seen by users 
- readonly (bool) – - whether the field is readonly (default: - False)- This only has an impact on the UI. Any field assignation in code will work (if the field is a stored field or an inversable one). 
- required (bool) – whether the value of the field is required (default: - False)
- index (str) – - whether the field is indexed in database, and the kind of index. Note: this has no effect on non-stored and virtual fields. The possible values are: - "btree"or- True: standard index, good for many2one
- "btree_not_null": BTREE index without NULL values (useful when most
- values are NULL, or when NULL is never searched for) 
 
- "trigram": Generalized Inverted Index (GIN) with trigrams (good for full-text search)
- Noneor- False: no index (default)
 
- default (value or callable) – the default value for the field; this is either a static value, or a function taking a recordset and returning a value; use - default=Noneto discard default values for the field
- groups (str) – comma-separated list of group xml ids (string); this restricts the field access to the users of the given groups only 
- company_dependent (bool) – - whether the field value is dependent of the current company; - The value is stored on the model table as jsonb dict with the company id as the key. - The field’s default values stored in model ir.default are used as fallbacks for unspecified values in the jsonb dict. 
- copy (bool) – whether the field value should be copied when the record is duplicated (default: - Truefor normal fields,- Falsefor- one2manyand computed fields, including property fields and related fields)
- store (bool) – whether the field is stored in database (default: - True,- Falsefor computed fields)
- aggregator (str) – - default aggregate function used by the webclient on this field when using “Group By” feature. - Supported aggregators are: - count: number of rows
- count_distinct: number of distinct rows
- bool_and: true if all values are true, otherwise false
- bool_or: true if at least one value is true, otherwise false
- max: maximum value of all values
- min: minimum value of all values
- avg: the average (arithmetic mean) of all values
- sum: sum of all values
 
- group_expand (str) – - function used to expand results when grouping on the current field for kanban/list/gantt views. For selection fields, - group_expand=Trueautomatically expands groups for all selection keys.- @api.model def _read_group_selection_field(self, values, domain): return ['choice1', 'choice2', ...] # available selection choices. @api.model def _read_group_many2one_field(self, records, domain): return records + self.search([custom_domain]) 
 
 - Computed Fields - Parameters
- compute (str) – - name of a method that computes the field 
- precompute (bool) – - whether the field should be computed before record insertion in database. Should be used to specify manually some fields as precompute=True when the field can be computed before record insertion. (e.g. avoid statistics fields based on search/_read_group), many2one linking to the previous record, … (default: - False)- Waarschuwing - Precomputation only happens when no explicit value and no default value is provided to create(). This means that a default value disables the precomputation, even if the field is specified as precompute=True. - Precomputing a field can be counterproductive if the records of the given model are not created in batch. Consider the situation were many records are created one by one. If the field is not precomputed, it will normally be computed in batch at the flush(), and the prefetching mechanism will help making the computation efficient. On the other hand, if the field is precomputed, the computation will be made one by one, and will therefore not be able to take advantage of the prefetching mechanism. - Following the remark above, precomputed fields can be interesting on the lines of a one2many, which are usually created in batch by the ORM itself, provided that they are created by writing on the record that contains them. 
- compute_sudo (bool) – whether the field should be recomputed as superuser to bypass access rights (by default - Truefor stored fields,- Falsefor non stored fields)
- recursive (bool) – whether the field has recursive dependencies (the field - Xhas a dependency like- parent_id.X); declaring a field recursive must be explicit to guarantee that recomputation is correct
- inverse (str) – name of a method that inverses the field (optional) 
- search (str) – name of a method that implement search on the field (optional) 
- related (str) – sequence of field names 
- default_export_compatible (bool) – - whether the field must be exported by default in an import-compatible export 
 
 
Basic Fields¶
- class odoo.fields.Boolean[broncode]¶
- Encapsulates a - bool.
- class odoo.fields.Char[broncode]¶
- Basic string field, can be length-limited, usually displayed as a single-line string in clients. - Parameters
- size (int) – the maximum size of values stored for that field 
- trim (bool) – states whether the value is trimmed or not (by default, - True). Note that the trim operation is applied only by the web client.
- translate (bool or callable) – enable the translation of the field’s values; use - translate=Trueto translate field values as a whole;- translatemay also be a callable such that- translate(callback, value)translates- valueby using- callback(term)to retrieve the translation of terms.
 
 
- class odoo.fields.Float[broncode]¶
- Encapsulates a - float.- The precision digits are given by the (optional) - digitsattribute.- Parameters
- digits (tuple(int,int) or str) – a pair (total, decimal) or a string referencing a - DecimalPrecisionrecord name.
 - When a float is a quantity associated with an unit of measure, it is important to use the right tool to compare or round values with the correct precision. - The Float class provides some static methods for this purpose: - round()to round a float with the given precision.- is_zero()to check if a float equals zero at the given precision.- compare()to compare two floats at the given precision.- Example - To round a quantity with the precision of the unit of measure: - fields.Float.round(self.product_uom_qty, precision_rounding=self.product_uom_id.rounding) - To check if the quantity is zero with the precision of the unit of measure: - fields.Float.is_zero(self.product_uom_qty, precision_rounding=self.product_uom_id.rounding) - To compare two quantities: - field.Float.compare(self.product_uom_qty, self.qty_done, precision_rounding=self.product_uom_id.rounding) - The compare helper uses the __cmp__ semantics for historic purposes, therefore the proper, idiomatic way to use this helper is like so: - if result == 0, the first and second floats are equal if result < 0, the first float is lower than the second if result > 0, the first float is greater than the second 
- class odoo.fields.Integer[broncode]¶
- Encapsulates an - int.
Advanced Fields¶
- class odoo.fields.Binary[broncode]¶
- Encapsulates a binary content (e.g. a file). - Parameters
- attachment (bool) – whether the field should be stored as - ir_attachmentor in a column of the model’s table (default:- True).
 
- class odoo.fields.Html[broncode]¶
- Encapsulates an html code content. - Parameters
- sanitize (bool) – whether value must be sanitized (default: - True)
- sanitize_overridable (bool) – whether the sanitation can be bypassed by the users part of the - base.group_sanitize_overridegroup (default:- False)
- sanitize_tags (bool) – whether to sanitize tags (only a white list of attributes is accepted, default: - True)
- sanitize_attributes (bool) – whether to sanitize attributes (only a white list of attributes is accepted, default: - True)
- sanitize_style (bool) – whether to sanitize style attributes (default: - False)
- sanitize_conditional_comments (bool) – whether to kill conditional comments. (default: - True)
- sanitize_output_method (bool) – whether to sanitize using html or xhtml (default: - html)
- strip_style (bool) – whether to strip style attributes (removed and therefore not sanitized, default: - False)
- strip_classes (bool) – whether to strip classes attributes (default: - False)
 
 
- class odoo.fields.Image[broncode]¶
- Encapsulates an image, extending - Binary.- If image size is greater than the - max_width/- max_heightlimit of pixels, the image will be resized to the limit by keeping aspect ratio.- Parameters
- max_width (int) – the maximum width of the image (default: - 0, no limit)
- max_height (int) – the maximum height of the image (default: - 0, no limit)
- verify_resolution (bool) – whether the image resolution should be verified to ensure it doesn’t go over the maximum image resolution (default: - True). See- odoo.tools.image.ImageProcessfor maximum image resolution (default:- 50e6).
 
 - Notitie - If no - max_width/- max_heightis specified (or is set to 0) and- verify_resolutionis False, the field content won’t be verified at all and a- Binaryfield should be used.
- class odoo.fields.Monetary[broncode]¶
- Encapsulates a - floatexpressed in a given- res_currency.- The decimal precision and currency symbol are taken from the - currency_fieldattribute.
- class odoo.fields.Selection[broncode]¶
- Encapsulates an exclusive choice between different values. - Parameters
- selection (list(tuple(str,str)) or callable or str) – specifies the possible values for this field. It is given as either a list of pairs - (value, label), or a model method, or a method name.
- selection_add (list(tuple(str,str))) – - provides an extension of the selection in the case of an overridden field. It is a list of pairs - (value, label)or singletons- (value,), where singleton values must appear in the overridden selection. The new values are inserted in an order that is consistent with the overridden selection and this list:- selection = [('a', 'A'), ('b', 'B')] selection_add = [('c', 'C'), ('b',)] > result = [('a', 'A'), ('c', 'C'), ('b', 'B')] 
- ondelete – - provides a fallback mechanism for any overridden field with a selection_add. It is a dict that maps every option from the selection_add to a fallback action. - This fallback action will be applied to all records whose selection_add option maps to it. - The actions can be any of the following:
- ’set null’ – the default, all records with this option will have their selection value set to False. 
- ’cascade’ – all records with this option will be deleted along with the option itself. 
- ’set default’ – all records with this option will be set to the default of the field definition 
- ’set VALUE’ – all records with this option will be set to the given value 
- <callable> – a callable whose first and only argument will be the set of records containing the specified Selection option, for custom processing 
 
 
 
 - The attribute - selectionis mandatory except in the case of- relatedor extended fields.
- class odoo.fields.Text[broncode]¶
- Very similar to - Charbut used for longer contents, does not have a size and usually displayed as a multiline text box.- Parameters
- translate (bool or callable) – enable the translation of the field’s values; use - translate=Trueto translate field values as a whole;- translatemay also be a callable such that- translate(callback, value)translates- valueby using- callback(term)to retrieve the translation of terms.
 
Date(time) Fields¶
Dates and Datetimes
are very important fields in any kind of business application.
Their misuse can create invisible yet painful bugs, this section
aims to provide Odoo developers with the knowledge required
to avoid misusing these fields.
When assigning a value to a Date/Datetime field, the following options are valid:
- A - dateor- datetimeobject.
- A string in the proper server format: 
- Falseor- None.
The Date and Datetime fields class have helper methods to attempt conversion into a compatible type:
- to_date()will convert to a- datetime.date
- to_datetime()will convert to a- datetime.datetime.
Example
To parse date/datetimes coming from external sources:
fields.Date.to_date(self._context.get('date_from'))
Date / Datetime comparison best practices:
- Date fields can only be compared to date objects. 
- Datetime fields can only be compared to datetime objects. 
Waarschuwing
Strings representing dates and datetimes can be compared between each other, however the result may not be the expected result, as a datetime string will always be greater than a date string, therefore this practice is heavily discouraged.
Common operations with dates and datetimes such as addition, subtraction or
fetching the start/end of a period are exposed through both
Date and Datetime.
These helpers are also available by importing odoo.tools.date_utils.
Notitie
Timezones
Datetime fields are stored as timestamp without timezone columns in the database and are stored
in the UTC timezone. This is by design, as it makes the Odoo database independent from the timezone
of the hosting server system. Timezone conversion is managed entirely by the client side.
- class odoo.fields.Date[broncode]¶
- Encapsulates a python - dateobject.- static today(*args)[broncode]¶
- Return the current day in the format expected by the ORM. - Notitie - This function may be used to compute default values. 
 - static context_today(record, timestamp=None)[broncode]¶
- Return the current date as seen in the client’s timezone in a format fit for date fields. - Notitie - This method may be used to compute default values. - Parameters
- record – recordset from which the timezone will be obtained. 
- timestamp (datetime) – optional datetime value to use instead of the current date and time (must be a datetime, regular dates can’t be converted between timezones). 
 
- Return type
- date 
 
 - static to_date(value)[broncode]¶
- Attempt to convert - valueto a- dateobject.- Waarschuwing - If a datetime object is given as value, it will be converted to a date object and all datetime-specific information will be lost (HMS, TZ, …). 
 - static to_string(value)[broncode]¶
- Convert a - dateor- datetimeobject to a string.- Parameters
- value – value to convert. 
- Returns
- a string representing - valuein the server’s date format, if- valueis of type- datetime, the hours, minute, seconds, tzinfo will be truncated.
- Return type
 
 - static start_of(value: odoo.tools.date_utils.D, granularity: Literal['year', 'quarter', 'month', 'week', 'day', 'hour']) odoo.tools.date_utils.D[broncode]¶
- Get start of a time period from a date or a datetime. - Parameters
- value – initial date or datetime. 
- granularity – type of period in string, can be year, quarter, month, week, day or hour. 
 
- Returns
- a date/datetime object corresponding to the start of the specified period. 
 
 - static end_of(value: odoo.tools.date_utils.D, granularity: Literal['year', 'quarter', 'month', 'week', 'day', 'hour']) odoo.tools.date_utils.D[broncode]¶
- Get end of a time period from a date or a datetime. - Parameters
- value – initial date or datetime. 
- granularity – Type of period in string, can be year, quarter, month, week, day or hour. 
 
- Returns
- A date/datetime object corresponding to the start of the specified period. 
 
 - static add(value: odoo.tools.date_utils.D, *args, **kwargs) odoo.tools.date_utils.D[broncode]¶
- Return the sum of - valueand a- relativedelta.- Parameters
- value – initial date or datetime. 
- args – positional args to pass directly to - relativedelta.
- kwargs – keyword args to pass directly to - relativedelta.
 
- Returns
- the resulting date/datetime. 
 
 - static subtract(value: odoo.tools.date_utils.D, *args, **kwargs) odoo.tools.date_utils.D[broncode]¶
- Return the difference between - valueand a- relativedelta.- Parameters
- value – initial date or datetime. 
- args – positional args to pass directly to - relativedelta.
- kwargs – keyword args to pass directly to - relativedelta.
 
- Returns
- the resulting date/datetime. 
 
 
- class odoo.fields.Datetime[broncode]¶
- Encapsulates a python - datetimeobject.- static now(*args)[broncode]¶
- Return the current day and time in the format expected by the ORM. - Notitie - This function may be used to compute default values. 
 - static today(*args)[broncode]¶
- Return the current day, at midnight (00:00:00). 
 - static context_timestamp(record, timestamp)[broncode]¶
- Return the given timestamp converted to the client’s timezone. - Notitie - This method is not meant for use as a default initializer, because datetime fields are automatically converted upon display on client side. For default values, - now()should be used instead.- Parameters
- record – recordset from which the timezone will be obtained. 
- timestamp (datetime) – naive datetime value (expressed in UTC) to be converted to the client timezone. 
 
- Returns
- timestamp converted to timezone-aware datetime in context timezone. 
- Return type
- datetime 
 
 - static to_datetime(value)[broncode]¶
- Convert an ORM - valueinto a- datetimevalue.
 - static to_string(value)[broncode]¶
- Convert a - datetimeor- dateobject to a string.- Parameters
- value (datetime or date) – value to convert. 
- Returns
- a string representing - valuein the server’s datetime format, if- valueis of type- date, the time portion will be midnight (00:00:00).
- Return type
 
 - static start_of(value: odoo.tools.date_utils.D, granularity: Literal['year', 'quarter', 'month', 'week', 'day', 'hour']) odoo.tools.date_utils.D[broncode]¶
- Get start of a time period from a date or a datetime. - Parameters
- value – initial date or datetime. 
- granularity – type of period in string, can be year, quarter, month, week, day or hour. 
 
- Returns
- a date/datetime object corresponding to the start of the specified period. 
 
 - static end_of(value: odoo.tools.date_utils.D, granularity: Literal['year', 'quarter', 'month', 'week', 'day', 'hour']) odoo.tools.date_utils.D[broncode]¶
- Get end of a time period from a date or a datetime. - Parameters
- value – initial date or datetime. 
- granularity – Type of period in string, can be year, quarter, month, week, day or hour. 
 
- Returns
- A date/datetime object corresponding to the start of the specified period. 
 
 - static add(value: odoo.tools.date_utils.D, *args, **kwargs) odoo.tools.date_utils.D[broncode]¶
- Return the sum of - valueand a- relativedelta.- Parameters
- value – initial date or datetime. 
- args – positional args to pass directly to - relativedelta.
- kwargs – keyword args to pass directly to - relativedelta.
 
- Returns
- the resulting date/datetime. 
 
 - static subtract(value: odoo.tools.date_utils.D, *args, **kwargs) odoo.tools.date_utils.D[broncode]¶
- Return the difference between - valueand a- relativedelta.- Parameters
- value – initial date or datetime. 
- args – positional args to pass directly to - relativedelta.
- kwargs – keyword args to pass directly to - relativedelta.
 
- Returns
- the resulting date/datetime. 
 
 
Relational Fields¶
- class odoo.fields.Many2one[broncode]¶
- The value of such a field is a recordset of size 0 (no record) or 1 (a single record). - Parameters
- comodel_name (str) – name of the target model - Mandatoryexcept for related or extended fields.
- domain – an optional domain to set on candidate values on the client side (domain or a python expression that will be evaluated to provide domain) 
- context (dict) – an optional context to use on the client side when handling that field 
- ondelete (str) – what to do when the referred record is deleted; possible values are: - 'set null',- 'restrict',- 'cascade'
- auto_join (bool) – whether JOINs are generated upon search through that field (default: - False)
- delegate (bool) – set it to - Trueto make fields of the target model accessible from the current model (corresponds to- _inherits)
- check_company (bool) – Mark the field to be verified in - _check_company(). Has a different behaviour depending on whether the field is company_dependent or not. Constrains non-company-dependent fields to target records whose company_id(s) are compatible with the record’s company_id(s). Constrains company_dependent fields to target records whose company_id(s) are compatible with the currently active company.
 
 
- class odoo.fields.One2many[broncode]¶
- One2many field; the value of such a field is the recordset of all the records in - comodel_namesuch that the field- inverse_nameis equal to the current record.- Parameters
- comodel_name (str) – name of the target model 
- inverse_name (str) – name of the inverse - Many2onefield in- comodel_name
- domain – an optional domain to set on candidate values on the client side (domain or a python expression that will be evaluated to provide domain) 
- context (dict) – an optional context to use on the client side when handling that field 
- auto_join (bool) – whether JOINs are generated upon search through that field (default: - False)
 
 - The attributes - comodel_nameand- inverse_nameare mandatory except in the case of related fields or field extensions.
- class odoo.fields.Many2many[broncode]¶
- Many2many field; the value of such a field is the recordset. - Parameters
- comodel_name – name of the target model (string) mandatory except in the case of related or extended fields 
- relation (str) – optional name of the table that stores the relation in the database 
- column1 (str) – optional name of the column referring to “these” records in the table - relation
- column2 (str) – optional name of the column referring to “those” records in the table - relation
 
 - The attributes - relation,- column1and- column2are optional. If not given, names are automatically generated from model names, provided- model_nameand- comodel_nameare different!- Note that having several fields with implicit relation parameters on a given model with the same comodel is not accepted by the ORM, since those field would use the same table. The ORM prevents two many2many fields to use the same relation parameters, except if - both fields use the same model, comodel, and relation parameters are explicit; or 
- at least one field belongs to a model with - _auto = False.
 - Parameters
- domain – an optional domain to set on candidate values on the client side (domain or a python expression that will be evaluated to provide domain) 
- context (dict) – an optional context to use on the client side when handling that field 
- check_company (bool) – Mark the field to be verified in - _check_company(). Add a default company domain depending on the field attributes.
 
 
- class odoo.fields.Command[broncode]¶
- One2manyand- Many2manyfields expect a special command to manipulate the relation they implement.- Internally, each command is a 3-elements tuple where the first element is a mandatory integer that identifies the command, the second element is either the related record id to apply the command on (commands update, delete, unlink and link) either 0 (commands create, clear and set), the third element is either the - valuesto write on the record (commands create and update) either the new- idslist of related records (command set), either 0 (commands delete, unlink, link, and clear).- Via Python, we encourage developers craft new commands via the various functions of this namespace. We also encourage developers to use the command identifier constant names when comparing the 1st element of existing commands. - Via RPC, it is impossible nor to use the functions nor the command constant names. It is required to instead write the literal 3-elements tuple where the first element is the integer identifier of the command. - CREATE = 0¶
 - UPDATE = 1¶
 - DELETE = 2¶
 - UNLINK = 3¶
 - LINK = 4¶
 - CLEAR = 5¶
 - SET = 6¶
 - classmethod create(values: dict)[broncode]¶
- Create new records in the comodel using - values, link the created records to- self.- In case of a - Many2manyrelation, one unique new record is created in the comodel such that all records in- selfare linked to the new record.- In case of a - One2manyrelation, one new record is created in the comodel for every record in- selfsuch that every record in- selfis linked to exactly one of the new records.- Return the command triple - (CREATE, 0, values)
 - classmethod update(id: int, values: dict)[broncode]¶
- Write - valueson the related record.- Return the command triple - (UPDATE, id, values)
 - classmethod delete(id: int)[broncode]¶
- Remove the related record from the database and remove its relation with - self.- In case of a - Many2manyrelation, removing the record from the database may be prevented if it is still linked to other records.- Return the command triple - (DELETE, id, 0)
 - classmethod unlink(id: int)[broncode]¶
- Remove the relation between - selfand the related record.- In case of a - One2manyrelation, the given record is deleted from the database if the inverse field is set as- ondelete='cascade'. Otherwise, the value of the inverse field is set to False and the record is kept.- Return the command triple - (UNLINK, id, 0)
 - classmethod link(id: int)[broncode]¶
- Add a relation between - selfand the related record.- Return the command triple - (LINK, id, 0)
 - classmethod clear()[broncode]¶
- Remove all records from the relation with - self. It behaves like executing the- unlinkcommand on every record.- Return the command triple - (CLEAR, 0, 0)
 - classmethod set(ids: list)[broncode]¶
- Replace the current relations of - selfby the given ones. It behaves like executing the- unlinkcommand on every removed relation then executing the- linkcommand on every new relation.- Return the command triple - (SET, 0, ids)
 
Pseudo-relational fields¶
- class odoo.fields.Reference[broncode]¶
- Pseudo-relational field (no FK in database). - The field value is stored as a - stringfollowing the pattern- "res_model,res_id"in database.
- class odoo.fields.Many2oneReference[broncode]¶
- Pseudo-relational field (no FK in database). - The field value is stored as an - integerid in database.- Contrary to - Referencefields, the model has to be specified in a- Charfield, whose name has to be specified in the- model_fieldattribute for the current- Many2oneReferencefield.
Computed Fields¶
Fields can be computed (instead of read straight from the database) using the
compute parameter. It must assign the computed value to the field. If
it uses the values of other fields, it should specify those fields using
depends().
from odoo import api
total = fields.Float(compute='_compute_total')
@api.depends('value', 'tax')
def _compute_total(self):
    for record in self:
        record.total = record.value + record.value * record.tax
- dependencies can be dotted paths when using sub-fields: - @api.depends('line_ids.value') def _compute_total(self): for record in self: record.total = sum(line.value for line in record.line_ids) 
- computed fields are not stored by default, they are computed and returned when requested. Setting - store=Truewill store them in the database and automatically enable searching and grouping. Note that by default,- compute_sudo=Trueis set on the field.
- searching on a computed field can also be enabled by setting the - searchparameter. The value is a method name returning a Search domains.- upper_name = field.Char(compute='_compute_upper', search='_search_upper') def _search_upper(self, operator, value): if operator == 'like': operator = 'ilike' return Domain('name', operator, value) - The search method is invoked when processing domains before doing an actual search on the model. It must return a domain equivalent to the condition: - field operator value.
- computed fields are readonly by default. To allow setting values on a computed field, use the - inverseparameter. It is the name of a function reversing the computation and setting the relevant fields:- document = fields.Char(compute='_get_document', inverse='_set_document') def _get_document(self): for record in self: with open(record.get_document_path) as f: record.document = f.read() def _set_document(self): for record in self: if not record.document: continue with open(record.get_document_path()) as f: f.write(record.document) 
- multiple fields can be computed at the same time by the same method, just use the same method on all fields and set all of them: - discount_value = fields.Float(compute='_apply_discount') total = fields.Float(compute='_apply_discount') @api.depends('value', 'discount') def _apply_discount(self): for record in self: # compute actual discount from discount percentage discount = record.value * record.discount record.discount_value = discount record.total = record.value - discount 
Waarschuwing
While it is possible to use the same compute method for multiple fields, it is not recommended to do the same for the inverse method.
During the computation of the inverse, all fields that use said inverse are protected, meaning that they can’t be computed, even if their value is not in the cache.
If any of those fields is accessed and its value is not in cache,
the ORM will simply return a default value of False for these fields.
This means that the value of the inverse fields (other than the one
triggering the inverse method) may not give their correct value and
this will probably break the expected behavior of the inverse method.
Automatic fields¶
Access Log fields¶
These fields are automatically set and updated if
_log_access is enabled. It can be
disabled to avoid creating or updating those fields on tables for which they are
not useful.
By default, _log_access is set to the same value
as _auto
Waarschuwing
_log_access must be enabled on
TransientModel.
Reserved Field names¶
A few field names are reserved for pre-defined behaviors beyond that of automated fields. They should be defined on a model when the related behavior is desired:
- Model.name¶
- default value for - _rec_name, used to display records in context where a representative “naming” is necessary.
- Model.active¶
- toggles the global visibility of the record, if - activeis set to- Falsethe record is invisible in most searches and listing.- Special methods: - action_archive()[broncode]¶
- Sets - activeto- Falseon a recordset for active records.- Note, you probably want to override - write()method if you want to take action once the active field changes.
 - Model.action_unarchive()[broncode]¶
- Sets - activeto- Trueon a recordset for inactive records.- Note, you probably want to override - write()method if you want to take action once the active field changes.
 
- Model.parent_id¶
- default_value of - _parent_name, used to organize records in a tree structure and enables the- child_ofand- parent_ofoperators in domains.
- Model.parent_path¶
- When - _parent_storeis set to True, used to store a value reflecting the tree structure of- _parent_name, and to optimize the operators- child_ofand- parent_ofin search domains. It must be declared with- index=Truefor proper operation.
Constraints and indexes¶
Similarly to fields, you can declare
Constraint,
Index and UniqueIndex.
The name of the attribute must begin with _ to avoid name clashes with field
names.
You can customize error messages.
They can either be strings and their translation will be provided in the internal
reflected constraint table.
Otherwise, they can be functions that take (env, diag) as parameters
which respectively denote the environment and psycopg diagnostics.
Example
class AModel(models.Model):
    _name = 'a.model'
    _my_check = models.Constraint("CHECK (x > y)", "x > y is not true")
    _name_idx = models.Index("(last_name, first_name)")
Recordsets¶
Interactions with models and records are performed through recordsets, an ordered collection of records of the same model.
Waarschuwing
Contrary to what the name implies, it is currently possible for recordsets to contain duplicates. This may change in the future.
Methods defined on a model are executed on a recordset, and their self is
a recordset:
class AModel(models.Model):
    _name = 'a.model'
    def a_method(self):
        # self can be anything between 0 records and all records in the
        # database
        self.do_operation()
Iterating on a recordset will yield new sets of a single record (“singletons”), much like iterating on a Python string yields strings of a single characters:
def do_operation(self):
    print(self) # => a.model(1, 2, 3, 4, 5)
    for record in self:
        print(record) # => a.model(1), then a.model(2), then a.model(3), ...
Field access¶
Recordsets provide an “Active Record” interface: model fields can be read and written directly from the record as attributes.
Notitie
When accessing non-relational fields on a recordset of potentially multiple
records, use mapped():
total_qty = sum(self.mapped('qty'))
Field values can also be accessed like dict items, which is more elegant and
safer than getattr() for dynamic field names.
Setting a field’s value triggers an update to the database:
>>> record.name
Example Name
>>> record.company_id.name
Company Name
>>> record.name = "Bob"
>>> field = "name"
>>> record[field]
Bob
Waarschuwing
Trying to read a field on multiple records will raise an error for non relational fields.
Accessing a relational field (Many2one,
One2many, Many2many)
always returns a recordset, empty if the field is not set.
Record cache and prefetching¶
Odoo maintains a cache for the fields of the records, so that not every field access issues a database request, which would be terrible for performance. The following example queries the database only for the first statement:
record.name             # first access reads value from database
record.name             # second access gets value from cache
To avoid reading one field on one record at a time, Odoo prefetches records and fields following some heuristics to get good performance. Once a field must be read on a given record, the ORM actually reads that field on a larger recordset, and stores the returned values in cache for later use. The prefetched recordset is usually the recordset from which the record comes by iteration. Moreover, all simple stored fields (boolean, integer, float, char, text, date, datetime, selection, many2one) are fetched altogether; they correspond to the columns of the model’s table, and are fetched efficiently in the same query.
Consider the following example, where partners is a recordset of 1000
records. Without prefetching, the loop would make 2000 queries to the database.
With prefetching, only one query is made:
for partner in partners:
    print partner.name          # first pass prefetches 'name' and 'lang'
                                # (and other fields) on all 'partners'
    print partner.lang
The prefetching also works on secondary records: when relational fields are read, their values (which are records) are subscribed for future prefetching. Accessing one of those secondary records prefetches all secondary records from the same model. This makes the following example generate only two queries, one for partners and one for countries:
countries = set()
for partner in partners:
    country = partner.country_id        # first pass prefetches all partners
    countries.add(country.name)         # first pass prefetches all countries
Zie ook
The methods search_fetch() and
fetch() can be used to populate the cache of
records, typically in cases where the prefetching mechanism does not work
well.
Method decorators¶
- odoo.api.autovacuum(method: C) C[broncode]¶
- Decorate a method so that it is called by the daily vacuum cron job (model - ir.autovacuum). This is typically used for garbage-collection-like tasks that do not deserve a specific cron job.
- odoo.api.constrains(*args) Decorator[broncode]¶
- Decorate a constraint checker. - Each argument must be a field name used in the check: - @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") - Invoked on the records on which one of the named fields has been modified. - Should raise - ValidationErrorif the validation failed.- Waarschuwing - @constrainsonly supports simple field names, dotted names (fields of relational fields e.g.- partner_id.customer) are not supported and will be ignored.- @constrainswill be triggered only if the declared fields in the decorated method are included in the- createor- writecall. It implies that fields not present in a view will not trigger a call during a record creation. A override of- createis necessary to make sure a constraint will always be triggered (e.g. to test the absence of value).- One may also pass a single function as argument. In that case, the field names are given by calling the function with a model instance. 
- odoo.api.depends(*args) Decorator[broncode]¶
- Return a decorator that specifies the field dependencies of a “compute” method (for new-style function fields). Each argument must be a string that consists in a dot-separated sequence of field names: - pname = fields.Char(compute='_compute_pname') @api.depends('partner_id.name', 'partner_id.is_company') def _compute_pname(self): for record in self: if record.partner_id.is_company: record.pname = (record.partner_id.name or "").upper() else: record.pname = record.partner_id.name - One may also pass a single function as argument. In that case, the dependencies are given by calling the function with the field’s model. 
- odoo.api.depends_context(*args: str) Decorator[broncode]¶
- Return a decorator that specifies the context dependencies of a non-stored “compute” method. Each argument is a key in the context’s dictionary: - price = fields.Float(compute='_compute_product_price') @api.depends_context('pricelist') def _compute_product_price(self): for product in self: if product.env.context.get('pricelist'): pricelist = self.env['product.pricelist'].browse(product.env.context['pricelist']) else: pricelist = self.env['product.pricelist'].get_default_pricelist() product.price = pricelist._get_products_price(product).get(product.id, 0.0) - All dependencies must be hashable. The following keys have special support: - company(value in context or current company id),
- uid(current user id and superuser flag),
- active_test(value in env.context or value in field.context).
 
- odoo.api.model(method: C) C[broncode]¶
- Decorate a record-style method where - selfis a recordset, but its contents is not relevant, only the model is. Such a method:- @api.model def method(self, args): ... 
- odoo.api.model_create_multi(method: Callable[[T, list[ValuesType]], T]) Callable[[T, list[ValuesType] | ValuesType], T][broncode]¶
- Decorate a method that takes a list of dictionaries and creates multiple records. The method may be called with either a single dict or a list of dicts: - record = model.create(vals) records = model.create([vals, ...]) 
- odoo.api.onchange(*args: str) Decorator[broncode]¶
- Return a decorator to decorate an onchange method for given fields. - In the form views where the field appears, the method will be called when one of the given fields is modified. The method is invoked on a pseudo-record that contains the values present in the form. Field assignments on that record are automatically sent back to the client. - Each argument must be a field name: - @api.onchange('partner_id') def _onchange_partner(self): self.message = "Dear %s" % (self.partner_id.name or "") - return { 'warning': {'title': "Warning", 'message': "What is this?", 'type': 'notification'}, } - If the type is set to notification, the warning will be displayed in a notification. Otherwise it will be displayed in a dialog as default. - Waarschuwing - @onchangeonly supports simple field names, dotted names (fields of relational fields e.g.- partner_id.tz) are not supported and will be ignored- Gevaar - Since - @onchangereturns a recordset of pseudo-records, calling any one of the CRUD methods (- create(),- read(),- write(),- unlink()) on the aforementioned recordset is undefined behaviour, as they potentially do not exist in the database yet.- Instead, simply set the record’s field like shown in the example above or call the - update()method.- Waarschuwing - It is not possible for a - one2manyor- many2manyfield to modify itself via onchange. This is a webclient limitation - see #2693.
- odoo.api.ondelete(*, at_uninstall: bool) Decorator[broncode]¶
- Mark a method to be executed during - unlink().- The goal of this decorator is to allow client-side errors when unlinking records if, from a business point of view, it does not make sense to delete such records. For instance, a user should not be able to delete a validated sales order. - While this could be implemented by simply overriding the method - unlinkon the model, it has the drawback of not being compatible with module uninstallation. When uninstalling the module, the override could raise user errors, but we shouldn’t care because the module is being uninstalled, and thus all records related to the module should be removed anyway.- This means that by overriding - unlink, there is a big chance that some tables/records may remain as leftover data from the uninstalled module. This leaves the database in an inconsistent state. Moreover, there is a risk of conflicts if the module is ever reinstalled on that database.- Methods decorated with - @ondeleteshould raise an error following some conditions, and by convention, the method should be named either- _unlink_if_<condition>or- _unlink_except_<not_condition>.- @api.ondelete(at_uninstall=False) def _unlink_if_user_inactive(self): if any(user.active for user in self): raise UserError("Can't delete an active user!") # same as above but with _unlink_except_* as method name @api.ondelete(at_uninstall=False) def _unlink_except_active_user(self): if any(user.active for user in self): raise UserError("Can't delete an active user!") - Parameters
- at_uninstall (bool) – Whether the decorated method should be called if the module that implements said method is being uninstalled. Should almost always be - False, so that module uninstallation does not trigger those errors.
 - Gevaar - The parameter - at_uninstallshould only be set to- Trueif the check you are implementing also applies when uninstalling the module.- For instance, it doesn’t matter if when uninstalling - sale, validated sales orders are being deleted because all data pertaining to- saleshould be deleted anyway, in that case- at_uninstallshould be set to- False.- However, it makes sense to prevent the removal of the default language if no other languages are installed, since deleting the default language will break a lot of basic behavior. In this case, - at_uninstallshould be set to- True.
- odoo.api.private(method: C) C[broncode]¶
- Decorate a record-style method to indicate that the method cannot be called using RPC. Example: - @api.private def method(self, args): ... - If you have business methods that should not be called over RPC, you should prefix them with “_”. This decorator may be used in case of existing public methods that become non-RPC callable or for ORM methods. 
Environment¶
- class odoo.api.Environment(cr: odoo.sql_db.BaseCursor, uid: int, context: dict, su: bool = False)[broncode]¶
- The environment stores various contextual data used by the ORM: - cr: the current database cursor (for database queries);
- uid: the current user id (for access rights checks);
- context: the current context dictionary (arbitrary metadata);
- su: whether in superuser mode.
 - It provides access to the registry by implementing a mapping from model names to models. It also holds a cache for records, and a data structure to manage recomputations. 
>>> records.env
<Environment object ...>
>>> records.env.uid
3
>>> records.env.user
res.user(3)
>>> records.env.cr
<Cursor object ...>
When creating a recordset from an other recordset, the environment is inherited. The environment can be used to get an empty recordset in an other model, and query that model:
>>> self.env['res.partner']
res.partner()
>>> self.env['res.partner'].search([('is_company', '=', True), ('customer', '=', True)])
res.partner(7, 18, 12, 14, 17, 19, 8, 31, 26, 16, 13, 20, 30, 22, 29, 15, 23, 28, 74)
Some lazy properties are available to access the environment (contextual) data:
- Environment.user¶
- Return the current user (as an instance). - Returns
- current user - sudoed 
- Return type
- res.users record
 
- Environment.company¶
- Return the current company (as an instance). - If not specified in the context ( - allowed_company_ids), fallback on current user main company.- Veroorzaakt
- AccessError – invalid or unauthorized - allowed_company_idscontext key content.
- Returns
- current company (default=`self.user.company_id`), with the current environment 
- Return type
- res.company record
 - Waarschuwing - No sanity checks applied in sudo mode! When in sudo mode, a user can access any company, even if not in his allowed companies. - This allows to trigger inter-company modifications, even if the current user doesn’t have access to the targeted company. 
- Environment.companies¶
- Return a recordset of the enabled companies by the user. - If not specified in the context( - allowed_company_ids), fallback on current user companies.- Veroorzaakt
- AccessError – invalid or unauthorized - allowed_company_idscontext key content.
- Returns
- current companies (default=`self.user.company_ids`), with the current environment 
- Return type
- res.company recordset
 - Waarschuwing - No sanity checks applied in sudo mode ! When in sudo mode, a user can access any company, even if not in his allowed companies. - This allows to trigger inter-company modifications, even if the current user doesn’t have access to the targeted company. 
Useful environment methods¶
- Environment.ref(xml_id: str, raise_if_not_found: bool = True) BaseModel | None[broncode]¶
- Return the record corresponding to the given - xml_id.- Parameters
- Returns
- Found record or None 
- Veroorzaakt
- ValueError – if record wasn’t found and - raise_if_not_foundis True
 
- Environment.is_superuser() bool[broncode]¶
- Return whether the environment is in superuser mode. 
- Environment.is_admin() bool[broncode]¶
- Return whether the current user has group “Access Rights”, or is in superuser mode. 
- Environment.is_system() bool[broncode]¶
- Return whether the current user has group “Settings”, or is in superuser mode. 
- Environment.execute_query(query: odoo.tools.sql.SQL) list[tuple][broncode]¶
- Execute the given query, fetch its result and it as a list of tuples (or an empty list if no result to fetch). The method automatically flushes all the fields in the metadata of the query. 
Altering the environment¶
- Model.with_context([context][, **overrides]) Model[broncode]¶
- Returns a new version of this recordset attached to an extended context. - The extended context is either the provided - contextin which- overridesare merged or the current context in which- overridesare merged e.g.:- # current context is {'key1': True} r2 = records.with_context({}, key2=True) # -> r2._context is {'key2': True} r2 = records.with_context(key2=True) # -> r2._context is {'key1': True, 'key2': True} 
- Model.with_user(user)[broncode]¶
- Return a new version of this recordset attached to the given user, in non-superuser mode, unless - useris the superuser (by convention, the superuser is always in superuser mode.)
- Model.with_company(company)[broncode]¶
- Return a new version of this recordset with a modified context, such that: - result.env.company = company result.env.companies = self.env.companies | company - Parameters
- company ( - res_companyor int) – main company of the new environment.
 - Waarschuwing - When using an unauthorized company for current user, accessing the company(ies) on the environment may trigger an AccessError if not done in a sudoed environment. 
- Model.with_env(env: api.Environment) Self[broncode]¶
- Return a new version of this recordset attached to the provided environment. - Parameters
- env ( - Environment) –
 - Notitie - The returned recordset has the same prefetch object as - self.
- Model.sudo([flag=True])[broncode]¶
- Returns a new version of this recordset with superuser mode enabled or disabled, depending on - flag. The superuser mode does not change the current user, and simply bypasses access rights checks.- Waarschuwing - Using - sudocould cause data access to cross the boundaries of record rules, possibly mixing records that are meant to be isolated (e.g. records from different companies in multi-company environments).- It may lead to un-intuitive results in methods which select one record among many - for example getting the default company, or selecting a Bill of Materials. - Notitie - The returned recordset has the same prefetch object as - self.
SQL Execution¶
The cr attribute on environments is the
cursor for the current database transaction and allows executing SQL directly,
either for queries which are difficult to express using the ORM (e.g. complex
joins) or for performance reasons:
self.env.cr.execute("some_sql", params)
Waarschuwing
Executing raw SQL bypasses the ORM and, by consequent, Odoo security rules. Please make sure your queries are sanitized when using user input and prefer using ORM utilities if you don’t really need to use SQL queries.
The recommended way to build SQL queries is to use the wrapper object
- class odoo.tools.SQL(code: str | SQL = '', /, *args, to_flush: Field | None = None, **kwargs)[broncode]¶
- An object that wraps SQL code with its parameters, like: - sql = SQL("UPDATE TABLE foo SET a = %s, b = %s", 'hello', 42) cr.execute(sql) - The code is given as a - %-format string, and supports either positional arguments (with- %s) or named arguments (with- %(name)s). The arguments are meant to be merged into the code using the- %formatting operator. Note that the character- %must always be escaped (as- %%), even if the code does not have parameters, like in- SQL("foo LIKE 'a%%'").- The SQL wrapper is designed to be composable: the arguments can be either actual parameters, or SQL objects themselves: - sql = SQL( "UPDATE TABLE %s SET %s", SQL.identifier(tablename), SQL("%s = %s", SQL.identifier(columnname), value), ) - The combined SQL code is given by - sql.code, while the corresponding combined parameters are given by the list- sql.params. This allows to combine any number of SQL terms without having to separately combine their parameters, which can be tedious, bug-prone, and is the main downside of- psycopg2.sql <https://www.psycopg.org/docs/sql.html>.- The second purpose of the wrapper is to discourage SQL injections. Indeed, if - codeis a string literal (not a dynamic string), then the SQL object made with- codeis guaranteed to be safe, provided the SQL objects within its parameters are themselves safe.- The wrapper may also contain some metadata - to_flush. If not- None, its value is a field which the SQL code depends on. The metadata of a wrapper and its parts can be accessed by the iterator- sql.to_flush.- join(args: Iterable) SQL[broncode]¶
- Join SQL objects or parameters with - selfas a separator.
 
One important thing to know about models is that they don’t necessarily perform database updates right away. Indeed, for performance reasons, the framework delays the recomputation of fields after modifying records. And some database updates are delayed, too. Therefore, before querying the database, one has to make sure that it contains the relevant data for the query. This operation is called flushing and performs the expected database updates.
Example
# make sure that 'partner_id' is up-to-date in database
self.env['model'].flush_model(['partner_id'])
self.env.cr.execute(SQL("SELECT id FROM model WHERE partner_id IN %s", ids))
ids = [row[0] for row in self.env.cr.fetchall()]
Before every SQL query, one has to flush the data needed for that query. There are three levels for flushing, each with its own API. One can flush either everything, all the records of a model, or some specific records. Because delaying updates improves performance in general, we recommend to be specific when flushing.
- Environment.flush_all() None[broncode]¶
- Flush all pending computations and updates to the database. 
- Model.flush_model(fnames=None)[broncode]¶
- Process the pending computations and database updates on - self’s model. When the parameter is given, the method guarantees that at least the given fields are flushed to the database. More fields can be flushed, though.- Parameters
- fnames – optional iterable of field names to flush 
 
- Model.flush_recordset(fnames=None)[broncode]¶
- Process the pending computations and database updates on the records - self. When the parameter is given, the method guarantees that at least the given fields on records- selfare flushed to the database. More fields and records can be flushed, though.- Parameters
- fnames – optional iterable of field names to flush 
 
Because models use the same cursor and the Environment
holds various caches, these caches must be invalidated when altering the
database in raw SQL, or further uses of models may become incoherent. It is
necessary to clear caches when using CREATE, UPDATE or DELETE in
SQL, but not SELECT (which simply reads the database).
Example
# make sure 'state' is up-to-date in database
self.env['model'].flush_model(['state'])
self.env.cr.execute("UPDATE model SET state=%s WHERE state=%s", ['new', 'old'])
# invalidate 'state' from the cache
self.env['model'].invalidate_model(['state'])
Just like flushing, one can invalidate either the whole cache, the cache of all the records of a model, or the cache of specific records. One can even invalidate specific fields on some records or all records of a model. As the cache improves performance in general, we recommend to be specific when invalidating.
- Environment.invalidate_all(flush: bool = True) None[broncode]¶
- Invalidate the cache of all records. - Parameters
- flush – whether pending updates should be flushed before invalidation. It is - Trueby default, which ensures cache consistency. Do not use this parameter unless you know what you are doing.
 
- Model.invalidate_model(fnames=None, flush=True)[broncode]¶
- Invalidate the cache of all records of - self’s model, when the cached values no longer correspond to the database values. If the parameter is given, only the given fields are invalidated from cache.- Parameters
- fnames – optional iterable of field names to invalidate 
- flush – whether pending updates should be flushed before invalidation. It is - Trueby default, which ensures cache consistency. Do not use this parameter unless you know what you are doing.
 
 
- Model.invalidate_recordset(fnames=None, flush=True)[broncode]¶
- Invalidate the cache of the records in - self, when the cached values no longer correspond to the database values. If the parameter is given, only the given fields on- selfare invalidated from cache.- Parameters
- fnames – optional iterable of field names to invalidate 
- flush – whether pending updates should be flushed before invalidation. It is - Trueby default, which ensures cache consistency. Do not use this parameter unless you know what you are doing.
 
 
The methods above keep the caches and the database consistent with each other. However, if computed field dependencies have been modified in the database, one has to inform the models for the computed fields to be recomputed. The only thing the framework needs to know is what fields have changed on which records.
Example
# make sure 'state' is up-to-date in database
self.env['model'].flush_model(['state'])
# use the RETURNING clause to retrieve which rows have changed
self.env.cr.execute("UPDATE model SET state=%s WHERE state=%s RETURNING id", ['new', 'old'])
ids = [row[0] for row in self.env.cr.fetchall()]
# invalidate the cache, and notify the update to the framework
records = self.env['model'].browse(ids)
records.invalidate_recordset(['state'])
records.modified(['state'])
One has to figure out which records have been modified. There are many ways to
do this, possibly involving extra SQL queries. In the example above, we take
advantage of the RETURNING clause of PostgreSQL to retrieve the information
without an extra query. After making the cache consistent by invalidation,
invoke the method modified on the modified records with the fields that
have been updated.
- Model.modified(fnames, create=False, before=False)[broncode]¶
- Notify that fields will be or have been modified on - self. This invalidates the cache where necessary, and prepares the recomputation of dependent stored fields.- Parameters
- fnames – iterable of field names modified on records - self
- create – whether called in the context of record creation 
- before – whether called before modifying records - self
 
 
Common ORM methods¶
Create/Update¶
- Model.create(vals_list: list[ValuesType]) Self[broncode]¶
- Creates new records for the model. - The new records are initialized using the values from the list of dicts - vals_list, and if necessary those from- default_get().- Parameters
- vals_list – - values for the model’s fields, as a list of dictionaries: - [{'field_name': field_value, ...}, ...] - For backward compatibility, - vals_listmay be a dictionary. It is treated as a singleton list- [vals], and a single record is returned.- see - write()for details
- Returns
- the created records 
- Veroorzaakt
- AccessError – if the current user is not allowed to create records of the specified model 
- ValidationError – if user tries to enter invalid value for a selection field 
- ValueError – if a field name specified in the create values does not exist. 
- UserError – if a loop would be created in a hierarchy of objects a result of the operation (such as setting an object as its own parent) 
 
 
- Model.copy(default=None)[broncode]¶
- Duplicate record - selfupdating it with default values- Parameters
- default (dict) – dictionary of field values to override in the original values of the copied record, e.g: - {'field_name': overridden_value, ...}
- Returns
- new records 
 
- Model.default_get(fields_list) default_values[broncode]¶
- Return default values for the fields in - fields_list. Default values are determined by the context, user defaults, user fallbacks and the model itself.- Parameters
- fields_list (list) – names of field whose default is requested 
- Returns
- a dictionary mapping field names to their corresponding default values, if they have a default value. 
- Return type
 - Notitie - Unrequested defaults won’t be considered, there is no need to return a value for fields whose names are not in - fields_list.
- Model.name_create(name) record[broncode]¶
- Create a new record by calling - create()with only one value provided: the display name of the new record.- The new record will be initialized with any default values applicable to this model, or provided through the context. The usual behavior of - create()applies.- Parameters
- name – display name of the record to create 
- Return type
- Returns
- the (id, display_name) pair value of the created record 
 
- Model.write(vals)[broncode]¶
- Updates all records in - selfwith the provided values.- Parameters
- vals (dict) – fields to update and the value to set on them 
- Veroorzaakt
- AccessError – if user is not allowed to modify the specified records/fields 
- ValidationError – if invalid values are specified for selection fields 
- UserError – if a loop would be created in a hierarchy of objects a result of the operation (such as setting an object as its own parent) 
 
 - For numeric fields ( - Integer,- Float) the value should be of the corresponding type
- For - Selection, the value should match the selection values (generally- str, sometimes- int)
- For - Many2one, the value should be the database identifier of the record to set
- The expected value of a - One2manyor- Many2manyrelational field is a list of- Commandthat manipulate the relation the implement. There are a total of 7 commands:- create(),- update(),- delete(),- unlink(),- link(),- clear(), and- set().
- For - Dateand- ~odoo.fields.Datetime, the value should be either a date(time), or a string.- Waarschuwing - If a string is provided for Date(time) fields, it must be UTC-only and formatted according to - odoo.tools.misc.DEFAULT_SERVER_DATE_FORMATand- odoo.tools.misc.DEFAULT_SERVER_DATETIME_FORMAT
- Other non-relational fields use a string for value 
 
Search/Read¶
- Model.browse([ids]) records[broncode]¶
- Returns a recordset for the ids provided as parameter in the current environment. - self.browse([7, 18, 12]) res.partner(7, 18, 12) 
- Model.search(domain[, offset=0][, limit=None][, order=None])[broncode]¶
- Search for the records that satisfy the given - domainsearch domain.- Parameters
- domain – A search domain. Use an empty list to match all records. 
- offset (int) – number of results to ignore (default: none) 
- limit (int) – maximum number of records to return (default: all) 
- order (str) – sort string 
 
- Returns
- at most - limitrecords matching the search criteria
- Veroorzaakt
- AccessError – if user is not allowed to access requested information 
 - This is a high-level method, which should not be overridden. Its actual implementation is done by method - _search().
- Model.search_count(domain[, limit=None]) int[broncode]¶
- Returns the number of records in the current model matching the provided domain. - Parameters
- domain – A search domain. Use an empty list to match all records. 
- limit – maximum number of record to count (upperbound) (default: all) 
 
 - This is a high-level method, which should not be overridden. Its actual implementation is done by method - _search().
- Model.search_fetch(domain, field_names[, offset=0][, limit=None][, order=None])[broncode]¶
- Search for the records that satisfy the given - domainsearch domain, and fetch the given fields to the cache. This method is like a combination of methods- search()and- fetch(), but it performs both tasks with a minimal number of SQL queries.- Parameters
- domain – A search domain. Use an empty list to match all records. 
- field_names – a collection of field names to fetch 
- offset (int) – number of results to ignore (default: none) 
- limit (int) – maximum number of records to return (default: all) 
- order (str) – sort string 
 
- Returns
- at most - limitrecords matching the search criteria
- Veroorzaakt
- AccessError – if user is not allowed to access requested information 
 
- Model.name_search(name='', args=None, operator='ilike', limit=100)[broncode]¶
- Search for records that have a display name matching the given - namepattern when compared with the given- operator, while also matching the optional search domain (- args).- This is used for example to provide suggestions based on a partial value for a relational field. Should usually behave as the reverse of - display_name, but that is not guaranteed.- This method is equivalent to calling - search()with a search domain based on- display_nameand mapping id and display_name on the resulting search.- Parameters
- Return type
- Returns
- list of pairs - (id, display_name)for all matching records.
 
- Model.fetch(field_names)[broncode]¶
- Make sure the given fields are in memory for the records in - self, by fetching what is necessary from the database. Non-stored fields are mostly ignored, except for their stored dependencies. This method should be called to optimize code.- Parameters
- field_names – a collection of field names to fetch 
- Veroorzaakt
- AccessError – if user is not allowed to access requested information 
 - This method is implemented thanks to methods - _search()and- _fetch_query(), and should not be overridden.
- Model.read([fields])[broncode]¶
- Read the requested fields for the records in - self, and return their values as a list of dicts.- Parameters
- Returns
- a list of dictionaries mapping field names to their values, with one dictionary per record 
- Return type
- Veroorzaakt
- AccessError – if user is not allowed to access requested information 
- ValueError – if a requested field does not exist 
 
 - This is a high-level method that is not supposed to be overridden. In order to modify how fields are read from database, see methods - _fetch_query()and- _read_format().
- Model._read_group(domain: DomainType, groupby=(), aggregates=(), having=(), offset=0, limit=None, order=None) list[tuple][broncode]¶
- Get fields aggregations specified by - aggregatesgrouped by the given- groupbyfields where record are filtered by the- domain.- Parameters
- domain (list) – A search domain. Use an empty list to match all records. 
- groupby (list) – list of groupby descriptions by which the records will be grouped. A groupby description is either a field (then it will be grouped by that field) or a string - 'field:granularity'. Right now, the only supported granularities are- 'day',- 'week',- 'month',- 'quarter'or- 'year', and they only make sense for date/datetime fields.
- aggregates (list) – list of aggregates specification. Each element is - 'field:agg'(aggregate field with aggregation function- 'agg'). The possible aggregation functions are the ones provided by PostgreSQL,- 'count_distinct'with the expected meaning and- 'recordset'to act like- 'array_agg'converted into a recordset.
- having (list) – A domain where the valid “fields” are the aggregates. 
- offset (int) – optional number of groups to skip 
- limit (int) – optional max number of groups to return 
- order (str) – optional - order byspecification, for overriding the natural sort ordering of the groups, see also- search().
 
- Returns
- list of tuple containing in the order the groups values and aggregates values (flatten): - [(groupby_1_value, ... , aggregate_1_value_aggregate, ...), ...]. If group is related field, the value of it will be a recordset (with a correct prefetch set).
- Return type
- Veroorzaakt
- AccessError – if user is not allowed to access requested information 
 
Fields¶
- Model.fields_get([allfields][, attributes])[broncode]¶
- Return the definition of each field. - The returned value is a dictionary (indexed by field name) of dictionaries. The _inherits’d fields are included. The string, help, and selection (if present) attributes are translated. 
Search domains¶
A Domain is a first-order logical expression used for
filtering and searching recordsets.
A domain can be a simple condition (field_expr, operator, value) where:
- field_expr(- str)
- a field name of the current model, or a relationship traversal through a - Many2oneusing dot-notation e.g.- 'street'or- 'partner_id.country'. If the field is a date(time) field, you can also specify a part of the date using- 'field_name.granularity'. The supported granularities are- 'year_number',- 'quarter_number',- 'month_number',- 'iso_week_number',- 'day_of_week',- 'day_of_month',- 'day_of_year',- 'hour_number',- 'minute_number',- 'second_number'. They all use an integer as value.
 
- operator(- str)
- an operator used to compare the - field_exprwith the- value. Valid operators are:- =
- equals to 
- !=
- not equals to 
- >
- greater than 
- >=
- greater than or equal to 
- <
- less than 
- <=
- less than or equal to 
- =?
- unset or equals to (returns true if - valueis either- Noneor- False, otherwise behaves like- =)
- =like
- matches - field_expragainst the- valuepattern. An underscore- _in the pattern stands for (matches) any single character; a percent sign- %matches any string of zero or more characters.
- like
- matches - field_expragainst the- %value%pattern. Similar to- =likebut wraps- valuewith ‘%’ before matching
- not like
- doesn’t match against the - %value%pattern
- ilike
- case insensitive - like
- not ilike
- case insensitive - not like
- =ilike
- case insensitive - =like
- in
- is equal to any of the items from - value,- valueshould be a collection of items
- not in
- is unequal to all of the items from - value
- child_of
- is a child (descendant) of a - valuerecord (value can be either one item or a list of items).- Takes the semantics of the model into account (i.e following the relationship field named by - _parent_name).
- parent_of
- is a parent (ascendant) of a - valuerecord (value can be either one item or a list of items).- Takes the semantics of the model into account (i.e following the relationship field named by - _parent_name).
- any
- matches if any record in the relationship traversal through - field_expr(- Many2one,- One2many, or- Many2many) satisfies the provided domain- value. The- field_exprshould be a field name.
- not any
- matches if no record in the relationship traversal through - field_expr(- Many2one,- One2many, or- Many2many) satisfies the provided domain- value.
 
 
- value
- variable type, must be comparable (through - operator) to the named field.
 
Domain can be used as a builder for domains.
# parse a domain (from list to Domain)
domain = Domain([('name', '=', 'abc'), ('phone', 'like', '7620')])
# serialize domain as a list (from Domain to list)
domain_list = list(domain)
# simple domains
d1 = Domain('name', '=', 'abc')
d2 = Domain('phone', 'like', '7620')
# combine domains
d3 = d1 & d2  # and
d4 = d1 | d2  # or
d5 = ~d1      # not
# combine and parse multiple domains (any iterable of domains)
Domain.AND([d1, d2, d3, ...])
Domain.OR([d4, d5, ...])
# constants
Domain.TRUE   # true domain
Domain.FALSE  # false domain
- Domain.iter_conditions() Iterable[DomainCondition][broncode]¶
- Yield simple conditions of the domain 
- Domain.map_conditions(function: Callable[[DomainCondition], Domain]) Domain[broncode]¶
- Map a function to each condition and return the combined result 
A domain is serialized as a list of criteria, each criterion being a triple
(either a list or a tuple) representing a simple condition.
Domain criteria can be combined using logical operators in a prefix notation.
You can combine 2 domains using '&' (AND), '|' (OR)
and you can negate 1 using '!' (NOT).
Example
To search for partners named ABC, with a phone or mobile number containing 7620:
[('name', '=', 'ABC'),
 '|', ('phone','ilike','7620'), ('mobile', 'ilike', '7620')]
To search sales orders to invoice that have at least one line with a product that is out of stock:
[('invoice_status', '=', 'to invoice'),
 ('order_line', 'any', [('product_id.qty_available', '<=', 0)])]
To search for all partners born in the month of February:
[('birthday.month_number', '=', 2)]
Unlink¶
- Model.unlink()[broncode]¶
- Deletes the records in - self.- Veroorzaakt
- AccessError – if the user is not allowed to delete all the given records 
- UserError – if the record is default property for other records 
 
 
Record(set) information¶
- Model.ids¶
- Return the list of actual record ids corresponding to - self.
- odoo.models.env¶
- Returns the environment of the given recordset. - Type
 
- Model.exists() records[broncode]¶
- Returns the subset of records in - selfthat exist. It can be used as a test on records:- if record.exists(): ... - By convention, new records are returned as existing. 
- Model.ensure_one() Self[broncode]¶
- Verify that the current recordset holds a single record. - Veroorzaakt
- odoo.exceptions.ValueError – - len(self) != 1
 
- Model.get_metadata()[broncode]¶
- Return some metadata about the given records. - Returns
- list of ownership dictionaries for each requested record 
- Return type
- list of dictionaries with the following keys: - id: object id 
- create_uid: user who created the record 
- create_date: date when the record was created 
- write_uid: last user who changed the record 
- write_date: date of the last change to the record 
- xmlid: XML ID to use to refer to this record (if there is one), in format - module.name
- xmlids: list of dict with xmlid in format - module.name, and noupdate as boolean
- noupdate: A boolean telling if the record will be updated or not 
 
 
Operations¶
Recordsets are immutable, but sets of the same model can be combined using various set operations, returning new recordsets.
- record in setreturns whether- record(which must be a 1-element recordset) is present in- set.- record not in setis the inverse operation
- set1 <= set2and- set1 < set2return whether- set1is a subset of- set2(resp. strict)
- set1 >= set2and- set1 > set2return whether- set1is a superset of- set2(resp. strict)
- set1 | set2returns the union of the two recordsets, a new recordset containing all records present in either source
- set1 & set2returns the intersection of two recordsets, a new recordset containing only records present in both sources
- set1 - set2returns a new recordset containing only records of- set1which are not in- set2
Recordsets are iterable so the usual Python tools are available for
transformation (map(), sorted(),
ifilter(), …) however these return either a
list or an iterator, removing the ability to
call methods on their result, or to use set operations.
Recordsets therefore provide the following operations returning recordsets themselves (when possible):
Filter¶
- Model.filtered(func) Self[broncode]¶
- Return the records in - selfsatisfying- func.- Parameters
- func (callable or str) – a function or a dot-separated sequence of field names 
- Returns
- recordset of records satisfying func, may be empty. 
 - # only keep records whose company is the current user's records.filtered(lambda r: r.company_id == user.company_id) # only keep records whose partner is a company records.filtered("partner_id.is_company") 
- Model.filtered_domain(domain) Self[broncode]¶
- Return the records in - selfsatisfying the domain and keeping the same order.- Parameters
- domain – A search domain. 
 
Map¶
- Model.mapped(func)[broncode]¶
- Apply - funcon all records in- self, and return the result as a list or a recordset (if- funcreturn recordsets). In the latter case, the order of the returned recordset is arbitrary.- Parameters
- func (callable or str) – a function or a dot-separated sequence of field names 
- Returns
- self if func is falsy, result of func applied to all - selfrecords.
- Return type
- list or recordset 
 - # returns a list of summing two fields for each record in the set records.mapped(lambda r: r.field1 + r.field2) - The provided function can be a string to get field values: - # returns a list of names records.mapped('name') # returns a recordset of partners records.mapped('partner_id') # returns the union of all partner banks, with duplicates removed records.mapped('partner_id.bank_ids') 
Notitie
Since V13, multi-relational field access is supported and works like a mapped call:
records.partner_id  # == records.mapped('partner_id')
records.partner_id.bank_ids  # == records.mapped('partner_id.bank_ids')
records.partner_id.mapped('name')  # == records.mapped('partner_id.name')
Sort¶
- Model.sorted(key=None, reverse=False) Self[broncode]¶
- Return the recordset - selfordered by- key.- Parameters
 - # sort records by name records.sorted(key=lambda r: r.name) 
Grouping¶
- Model.grouped(key)[broncode]¶
- Eagerly groups the records of - selfby the- key, returning a dict from the- key’s result to recordsets. All the resulting recordsets are guaranteed to be part of the same prefetch-set.- Provides a convenience method to partition existing recordsets without the overhead of a - _read_group(), but performs no aggregation.- Notitie - unlike - itertools.groupby(), does not care about input ordering, however the tradeoff is that it can not be lazy
Inheritance and extension¶
Odoo provides three different mechanisms to extend models in a modular way:
- creating a new model from an existing one, adding new information to the copy but leaving the original module as-is 
- extending models defined in other modules in-place, replacing the previous version 
- delegating some of the model’s fields to records it contains 
 
Classical inheritance¶
When using the _inherit and
_name attributes together, Odoo creates a new
model using the existing one (provided via
_inherit) as a base. The new model gets all the
fields, methods and meta-information (defaults & al) from its base.
class Inheritance0(models.Model):
    _name = 'inheritance.0'
    _description = 'Inheritance Zero'
    name = fields.Char()
    def call(self):
        return self.check("model 0")
    def check(self, s):
        return "This is {} record {}".format(s, self.name)
class Inheritance1(models.Model):
    _name = 'inheritance.1'
    _inherit = ['inheritance.0']
    _description = 'Inheritance One'
    def call(self):
        return self.check("model 1")
and using them:
a = env['inheritance.0'].create({'name': 'A'})
b = env['inheritance.1'].create({'name': 'B'})
a.call()
b.call()
will yield:
“This is model 0 record A” “This is model 1 record B”
the second model has inherited from the first model’s check method and its
name field, but overridden the call method, as when using standard
Python inheritance.
Extension¶
When using _inherit but leaving out
_name, the new model replaces the existing one,
essentially extending it in-place. This is useful to add new fields or methods
to existing models (created in other modules), or to customize or reconfigure
them (e.g. to change their default sort order):
class Extension0(models.Model):
_name = 'extension.0'
_description = 'Extension zero'
name = fields.Char(default="A")
class Extension0(models.Model):
_inherit = ['extension.0']
description = fields.Char(default="Extended")
record = env['extension.0'].create({})
record.read()[0]
will yield:
{'name': "A", 'description': "Extended"}
Notitie
It will also yield the various automatic fields unless they’ve been disabled
Delegation¶
The third inheritance mechanism provides more flexibility (it can be altered
at runtime) but less power: using the _inherits
a model delegates the lookup of any field not found on the current model
to “children” models. The delegation is performed via
Reference fields automatically set up on the parent
model.
The main difference is in the meaning. When using Delegation, the model has one instead of is one, turning the relationship in a composition instead of inheritance:
class Screen(models.Model):
    _name = 'delegation.screen'
    _description = 'Screen'
    size = fields.Float(string='Screen Size in inches')
class Keyboard(models.Model):
    _name = 'delegation.keyboard'
    _description = 'Keyboard'
    layout = fields.Char(string='Layout')
class Laptop(models.Model):
    _name = 'delegation.laptop'
    _description = 'Laptop'
    _inherits = {
        'delegation.screen': 'screen_id',
        'delegation.keyboard': 'keyboard_id',
    }
    name = fields.Char(string='Name')
    maker = fields.Char(string='Maker')
    # a Laptop has a screen
    screen_id = fields.Many2one('delegation.screen', required=True, ondelete="cascade")
    # a Laptop has a keyboard
    keyboard_id = fields.Many2one('delegation.keyboard', required=True, ondelete="cascade")
record = env['delegation.laptop'].create({
    'screen_id': env['delegation.screen'].create({'size': 13.0}).id,
    'keyboard_id': env['delegation.keyboard'].create({'layout': 'QWERTY'}).id,
})
record.size
record.layout
will result in:
13.0
'QWERTY'
and it’s possible to write directly on the delegated field:
record.write({'size': 14.0})
Waarschuwing
when using delegation inheritance, methods are not inherited, only fields
Waarschuwing
- _inheritsis more or less implemented, avoid it if you can;
- chained - _inheritsis essentially not implemented, we cannot guarantee anything on the final behavior.
Fields Incremental Definition¶
A field is defined as class attribute on a model class. If the model is extended, one can also extend the field definition by redefining a field with the same name and same type on the subclass. In that case, the attributes of the field are taken from the parent class and overridden by the ones given in subclasses.
For instance, the second class below only adds a tooltip on the field
state:
class FirstFoo(models.Model):
    state = fields.Selection([...], required=True)
class FirstFoo(models.Model):
    _inherit = ['first.foo']
    state = fields.Selection(help="Blah blah blah")
class WrongFirstFooClassName(models.Model):
    _name = 'first.foo'  # force the model name
    _inherit = ['first.foo']
    state = fields.Selection(help="Blah blah blah")
Error management¶
The Odoo Exceptions module defines a few core exception types.
Those types are understood by the RPC layer. Any other exception type bubbling until the RPC layer will be treated as a ‘Server error’.
Notitie
If you consider introducing new exceptions,
check out the odoo.addons.test_exceptions module.
- exception odoo.exceptions.UserError(message)[broncode]¶
- Generic error managed by the client. - Typically when the user tries to do something that has no sense given the current state of a record. Semantically comparable to the generic 400 HTTP status codes. 
- exception odoo.exceptions.RedirectWarning(message, action, button_text, additional_context=None)[broncode]¶
- Warning with a possibility to redirect the user instead of simply displaying the warning message. - Parameters
- message (str) – exception message and frontend modal content 
- action_id (int) – id of the action where to perform the redirection 
- button_text (str) – text to put on the button that will trigger the redirection. 
- additional_context (dict) – parameter passed to action_id. Can be used to limit a view to active_ids for example. 
 
 
- exception odoo.exceptions.AccessDenied(message='Access Denied')[broncode]¶
- Login/password error. - Notitie - No traceback. - Example - When you try to log with a wrong password. 
- exception odoo.exceptions.AccessError(message)[broncode]¶
- Access rights error. - Example - When you try to read a record that you are not allowed to. 
- exception odoo.exceptions.CacheMiss(record, field)[broncode]¶
- Missing value(s) in cache. - Example - When you try to read a value in a flushed cache. 
- exception odoo.exceptions.MissingError(message)[broncode]¶
- Missing record(s). - Example - When you try to write on a deleted record. 
- exception odoo.exceptions.ValidationError(message)[broncode]¶
- Violation of python constraints. - Example - When you try to create a new user with a login which already exist in the db.