[FIX] upgrade_analysis: misleading representation of read/write computed fields

Odoo 14 introduced the widescale usage of computed fields with readonly=False.
In that case, the compute method functions as a default that can also be used
to compute a value some time *after* the initial creation of the record.

In the OpenUpgrade analysis files, these fields would be misrepresented as
computed fields rather than fields with a default function. This change fixes
that.
pull/2417/head
Stefan Rijnhart 2022-10-11 21:27:36 +02:00
parent 60d4f51c35
commit 3145683c07
3 changed files with 64 additions and 55 deletions

View File

@ -134,8 +134,6 @@ def report_generic(new, old, attrs, reprs):
if attr == "required":
if old[attr] != new["required"] and new["required"]:
text = "now required"
if new["req_default"]:
text += ", req_default: %s" % new["req_default"]
fieldprint(old, new, "", text, reprs)
elif attr == "stored":
if old[attr] != new[attr]:
@ -284,15 +282,19 @@ def compare_sets(old_records, new_records):
],
)
printkeys = [
# Info that is displayed for deleted fields
printkeys_old = [
"relation",
"required",
"selection_keys",
"req_default",
"_inherits",
"mode",
"attachment",
]
# Info that is displayed for new fields
printkeys_new = printkeys_old + [
"hasdefault",
]
for column in old_records:
if column["field"] == "_order":
continue
@ -304,7 +306,7 @@ def compare_sets(old_records, new_records):
extra_message = ", ".join(
[
k + ": " + str(column[k]) if k != str(column[k]) else k
for k in printkeys
for k in printkeys_old
if column[k]
]
)
@ -312,11 +314,6 @@ def compare_sets(old_records, new_records):
extra_message = " " + extra_message
fieldprint(column, "", "", "DEL" + extra_message, reprs)
printkeys.extend(
[
"hasdefault",
]
)
for column in new_records:
if column["field"] == "_order":
continue
@ -325,13 +322,13 @@ def compare_sets(old_records, new_records):
continue
if column["mode"] == "create":
column["mode"] = ""
printkeys_plus = printkeys.copy()
printkeys = printkeys_new.copy()
if column["isfunction"] or column["isrelated"]:
printkeys_plus.extend(["isfunction", "isrelated", "stored"])
printkeys.extend(["isfunction", "isrelated", "stored"])
extra_message = ", ".join(
[
k + ": " + str(column[k]) if k != str(column[k]) else k
for k in printkeys_plus
for k in printkeys
if column[k]
]
)

View File

@ -104,7 +104,6 @@ class UpgradeRecord(models.Model):
"required",
"stored",
"selection_keys",
"req_default",
"hasdefault",
"table",
"_inherits",

View File

@ -74,33 +74,56 @@ def compare_registries(cr, module, registry, local_registry):
old_field[key] = value
def isfunction(model, k):
def hasdefault(field):
"""Return a representation of the field's default method.
The default method is only meaningful if the field is a regular read/write
field with a `default` method or a `compute` method.
Note that Odoo fields accept a literal value as a `default` attribute
this value is wrapped in a lambda expression in odoo/fields.py:
https://github.com/odoo/odoo/blob/7eeba9d/odoo/fields.py#L484-L487
"""
if (
model._fields[k].compute
and not model._fields[k].related
and not model._fields[k].company_dependent
not field.readonly # It's not a proper computed field
and not field.inverse # It's not a field that delegates their data
and not isrelated(field) # It's not an (unstored) related field.
):
if field.default:
return "default"
if field.compute:
return "compute"
return ""
def isfunction(field):
if (
field.compute
and (field.readonly or field.inverse)
and not field.related
and not field.company_dependent
):
return "function"
return ""
def isproperty(model, k):
if model._fields[k].company_dependent:
def isproperty(field):
if field.company_dependent:
return "property"
return ""
def isrelated(model, k):
if model._fields[k].related:
def isrelated(field):
if field.related:
return "related"
return ""
def _get_relation(v):
if v.type in ("many2many", "many2one", "one2many"):
return v.comodel_name
elif v.type == "many2one_reference":
return v.model_field
def _get_relation(field):
if field.type in ("many2many", "many2one", "one2many"):
return field.comodel_name
elif field.type == "many2one_reference":
return field.model_field
else:
return ""
@ -125,41 +148,31 @@ def log_model(model, local_registry):
if model._inherits:
model_registry["_inherits"] = {"_inherits": str(model._inherits)}
model_registry["_order"] = {"_order": model._order}
for k, v in model._fields.items():
for fieldname, field in model._fields.items():
properties = {
"type": typemap.get(v.type, v.type),
"isfunction": isfunction(model, k),
"isproperty": isproperty(model, k),
"isrelated": isrelated(model, k),
"relation": _get_relation(v),
"table": v.relation if v.type == "many2many" else "",
"required": v.required and "required" or "",
"stored": v.store and "stored" or "",
"type": typemap.get(field.type, field.type),
"isfunction": isfunction(field),
"isproperty": isproperty(field),
"isrelated": isrelated(field),
"relation": _get_relation(field),
"table": field.relation if field.type == "many2many" else "",
"required": field.required and "required" or "",
"stored": field.store and "stored" or "",
"selection_keys": "",
"req_default": "",
"hasdefault": model._fields[k].default and "hasdefault" or "",
"hasdefault": hasdefault(field),
}
if v.type == "selection":
if isinstance(v.selection, (tuple, list)):
properties["selection_keys"] = str(sorted(x[0] for x in v.selection))
if field.type == "selection":
if isinstance(field.selection, (tuple, list)):
properties["selection_keys"] = str(
sorted(x[0] for x in field.selection)
)
else:
properties["selection_keys"] = "function"
elif v.type == "binary":
properties["attachment"] = str(getattr(v, "attachment", False))
default = model._fields[k].default
if v.required and default:
if (
callable(default)
or isinstance(default, str)
and getattr(model._fields[k], default, False)
and callable(getattr(model._fields[k], default))
):
properties["req_default"] = "function"
else:
properties["req_default"] = str(default)
elif field.type == "binary":
properties["attachment"] = str(getattr(field, "attachment", False))
for key, value in properties.items():
if value:
model_registry.setdefault(k, {})[key] = value
model_registry.setdefault(fieldname, {})[key] = value
def log_xml_id(cr, module, xml_id):