Skip to content

Commit

Permalink
[ADD] manifest-behind-migrations: check manifest version >= migrations
Browse files Browse the repository at this point in the history
A new check has been added to verify that the migrations and manifest
version for a module are up to date.

Closes #461.
  • Loading branch information
antonag32 committed Oct 27, 2023
1 parent ece62d3 commit 282f4ec
Show file tree
Hide file tree
Showing 7 changed files with 45 additions and 1 deletion.
44 changes: 43 additions & 1 deletion src/pylint_odoo/checkers/odoo_addons.py
Original file line number Diff line number Diff line change
Expand Up @@ -183,6 +183,11 @@
"no-raise-unlink",
"Use @api.ondelete to add any constraints instead",
),
"E8145": (
"Manifest version (%s) is lower than migration scripts (%s)",
"manifest-behind-migrations",
"Update your manifest version, otherwise the migration script won't run",
),
"F8101": ('File "%s": "%s" not found.', "resource-not-exist", CHECK_DESCRIPTION),
"R8101": (
"`odoo.exceptions.Warning` is a deprecated alias to `odoo.exceptions.UserError` "
Expand Down Expand Up @@ -545,8 +550,23 @@ class OdooAddons(OdooBaseChecker, BaseChecker):

def __init__(self, linter: PyLinter):
super().__init__(linter)
self._module_versions = {}
self._module_migrations = defaultdict(set)
self._deprecated_odoo_methods = set()

@staticmethod
def version_greater_equal_than(original, against):
"""Compare two versions and state which one is bigger than the other.
:param str original: Original version
:param str against: Value to compare against
:return: True if a >= b, otherwise False
"""
for original_val, against_val in zip(original.split("."), against.split("."), strict=True):
if int(against_val) > int(original_val):
return False

return True

def close(self):
"""Final process get all cached values and add messages"""
self.linter.config.deprecated_odoo_model_methods = set()
Expand All @@ -568,6 +588,18 @@ def close(self):
"consider-merging-classes-inherited", node=first_node, args=(odoo_class_inherit, ", ".join(path_nodes))
)

if self.linter.is_message_enabled("manifest-behind-migrations"):
for module, migrations in self._module_migrations.items():
module_version, manifest_node = self._module_versions[module]
for migration in migrations:
try:
if not self.version_greater_equal_than(module_version, migration):
self.add_message(
"manifest-behind-migrations", node=manifest_node, args=(module_version, migration)
)
except ValueError:
continue

def visit_module(self, node):
"""Initizalize the cache to save the original library name
of all imported node
Expand All @@ -576,6 +608,13 @@ def visit_module(self, node):
All these methods are these "visit_*" methods are called from pylint API
"""
self._from_imports = {}
basename = os.path.basename(node.file)
if basename not in {"pre-migration.py", "post-migration.py"}:
return

migration_version = os.path.basename(os.path.dirname(node.file))
module_name = os.path.basename(os.path.abspath(os.path.join(node.file, "..", "..", "..")))
self._module_migrations[module_name].add(migration_version)

def leave_module(self, node):
"""Clear variables"""
Expand Down Expand Up @@ -994,6 +1033,7 @@ def visit_call(self, node):
"missing-readme",
"resource-not-exist",
"website-manifest-key-not-valid-uri",
"manifest-behind-migrations",
)
def visit_dict(self, node):
if not os.path.basename(self.linter.current_file) in misc.MANIFEST_FILES or not isinstance(
Expand Down Expand Up @@ -1043,8 +1083,10 @@ def visit_dict(self, node):
if license_str and license_str not in self.linter.config.license_allowed:
self.add_message("license-allowed", node=manifest_keys_nodes.get("license") or node, args=(license_str,))

# Check version format
version_format = manifest_dict.get("version", "")
self._module_versions[os.path.basename(os.path.dirname(self.linter.current_file))] = (version_format, node)

# Check version format
formatrgx = self.formatversion(version_format)
if version_format and not formatrgx:
self.add_message(
Expand Down
Empty file.
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@
# Should raise manifest-behind-migrations but since manifest version is not parseable, it won't
Empty file.
Empty file.
Empty file.
1 change: 1 addition & 0 deletions tests/test_main.py
Original file line number Diff line number Diff line change
Expand Up @@ -67,6 +67,7 @@
"website-manifest-key-not-valid-uri": 1,
"no-raise-unlink": 2,
"deprecated-odoo-model-method": 2,
"manifest-behind-migrations": 3,
}


Expand Down

0 comments on commit 282f4ec

Please sign in to comment.