mirror of
https://github.com/ansible-collections/community.general.git
synced 2025-04-28 13:21:25 -07:00
pylint plugin to catch due/past-due deprecated calls (#44143)
* Start of work on pylint plugin to catch due/past-due deprecated calls * Improve deprecated pylint plugin * Catch call to AnsibleModule.deprecate also * Skip splatted kwargs, we can't infer that info * Add error for invalid version in deprecation * Skip version if it's a reference to a var * Disable ansible-deprecated-no-version for displaying deprecated module info * fix comments * is None * Force specifying a version, this can be disabled on a per case basis * Disable ansible-deprecated-version by default * Remove to look for 2.8 deprecated * Revert "Remove to look for 2.8 deprecated" This reverts commit 4e84034fd104879f429f0262ff0b2317e3d08deb. * Add script and template used for creating issues for deprecated issues * Fix underscore var
This commit is contained in:
parent
e7926cf9f4
commit
49eb53b44d
5 changed files with 239 additions and 1 deletions
97
test/sanity/pylint/plugins/deprecated.py
Normal file
97
test/sanity/pylint/plugins/deprecated.py
Normal file
|
@ -0,0 +1,97 @@
|
|||
# (c) 2018, Matt Martz <matt@sivel.net>
|
||||
# GNU General Public License v3.0+ (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt)
|
||||
# -*- coding: utf-8 -*-
|
||||
from __future__ import (absolute_import, division, print_function)
|
||||
__metaclass__ = type
|
||||
|
||||
from distutils.version import StrictVersion
|
||||
|
||||
import astroid
|
||||
|
||||
from pylint.interfaces import IAstroidChecker
|
||||
from pylint.checkers import BaseChecker
|
||||
from pylint.checkers.utils import check_messages
|
||||
|
||||
from ansible.release import __version__ as ansible_version_raw
|
||||
|
||||
MSGS = {
|
||||
'E9501': ("Deprecated version (%r) found in call to Display.deprecated "
|
||||
"or AnsibleModule.deprecate",
|
||||
"ansible-deprecated-version",
|
||||
"Used when a call to Display.deprecated specifies a version "
|
||||
"less than or equal to the current version of Ansible",
|
||||
{'minversion': (2, 6)}),
|
||||
'E9502': ("Display.deprecated call without a version",
|
||||
"ansible-deprecated-no-version",
|
||||
"Used when a call to Display.deprecated does not specify a "
|
||||
"version",
|
||||
{'minversion': (2, 6)}),
|
||||
'E9503': ("Invalid deprecated version (%r) found in call to "
|
||||
"Display.deprecated or AnsibleModule.deprecate",
|
||||
"ansible-invalid-deprecated-version",
|
||||
"Used when a call to Display.deprecated specifies an invalid "
|
||||
"version number",
|
||||
{'minversion': (2, 6)}),
|
||||
}
|
||||
|
||||
|
||||
ANSIBLE_VERSION = StrictVersion('.'.join(ansible_version_raw.split('.')[:3]))
|
||||
|
||||
|
||||
def _get_expr_name(node):
|
||||
"""Funciton to get either ``attrname`` or ``name`` from ``node.func.expr``
|
||||
|
||||
Created specifically for the case of ``display.deprecated`` or ``self._display.deprecated``
|
||||
"""
|
||||
try:
|
||||
return node.func.expr.attrname
|
||||
except AttributeError:
|
||||
# If this fails too, we'll let it raise, the caller should catch it
|
||||
return node.func.expr.name
|
||||
|
||||
|
||||
class AnsibleDeprecatedChecker(BaseChecker):
|
||||
"""Checks for Display.deprecated calls to ensure that the ``version``
|
||||
has not passed or met the time for removal
|
||||
"""
|
||||
|
||||
__implements__ = (IAstroidChecker,)
|
||||
name = 'deprecated'
|
||||
msgs = MSGS
|
||||
|
||||
@check_messages(*(MSGS.keys()))
|
||||
def visit_call(self, node):
|
||||
version = None
|
||||
try:
|
||||
if (node.func.attrname == 'deprecated' and 'display' in _get_expr_name(node) or
|
||||
node.func.attrname == 'deprecate' and 'module' in _get_expr_name(node)):
|
||||
if node.keywords:
|
||||
for keyword in node.keywords:
|
||||
if len(node.keywords) == 1 and keyword.arg is None:
|
||||
# This is likely a **kwargs splat
|
||||
return
|
||||
elif keyword.arg == 'version':
|
||||
if isinstance(keyword.value.value, astroid.Name):
|
||||
# This is likely a variable
|
||||
return
|
||||
version = keyword.value.value
|
||||
if not version:
|
||||
try:
|
||||
version = node.args[1].value
|
||||
except IndexError:
|
||||
self.add_message('ansible-deprecated-no-version', node=node)
|
||||
return
|
||||
|
||||
try:
|
||||
if ANSIBLE_VERSION >= StrictVersion(str(version)):
|
||||
self.add_message('ansible-deprecated-version', node=node, args=(version,))
|
||||
except ValueError:
|
||||
self.add_message('ansible-invalid-deprecated-version', node=node, args=(version,))
|
||||
except AttributeError:
|
||||
# Not the type of node we are interested in
|
||||
pass
|
||||
|
||||
|
||||
def register(linter):
|
||||
"""required method to auto register this checker """
|
||||
linter.register_checker(AnsibleDeprecatedChecker(linter))
|
Loading…
Add table
Add a link
Reference in a new issue