'\" t .\" Man page generated from reStructuredText. . . .nr rst2man-indent-level 0 . .de1 rstReportMargin \\$1 \\n[an-margin] level \\n[rst2man-indent-level] level margin: \\n[rst2man-indent\\n[rst2man-indent-level]] - \\n[rst2man-indent0] \\n[rst2man-indent1] \\n[rst2man-indent2] .. .de1 INDENT .\" .rstReportMargin pre: . RS \\$1 . nr rst2man-indent\\n[rst2man-indent-level] \\n[an-margin] . nr rst2man-indent-level +1 .\" .rstReportMargin post: .. .de UNINDENT . RE .\" indent \\n[an-margin] .\" old: \\n[rst2man-indent\\n[rst2man-indent-level]] .nr rst2man-indent-level -1 .\" new: \\n[rst2man-indent\\n[rst2man-indent-level]] .in \\n[rst2man-indent\\n[rst2man-indent-level]]u .. .TH "PYLINT" "1" "Mar 09, 2025" "3.3" "Pylint" .SH NAME pylint \- Pylint Documentation \X'tty: link https://github.com/pylint-dev/pylint/actions'\fI\%\fP\X'tty: link'\X'tty: link https://codecov.io/gh/pylint-dev/pylint'\fI\%\fP\X'tty: link'\X'tty: link https://pypi.python.org/pypi/pylint'\fI\%PyPI Package version\fP\X'tty: link'\X'tty: link https://pylint.readthedocs.io/en/latest/?badge=latest'\fI\%Documentation Status\fP\X'tty: link'\X'tty: link https://github.com/ambv/black'\fI\%\fP\X'tty: link'\X'tty: link https://github.com/pylint-dev/pylint'\fI\%\fP\X'tty: link'\X'tty: link https://results.pre-commit.ci/latest/github/pylint-dev/pylint/main'\fI\%pre\-commit.ci status\fP\X'tty: link'\X'tty: link https://bestpractices.coreinfrastructure.org/projects/6328'\fI\%CII Best Practices\fP\X'tty: link'\X'tty: link https://api.securityscorecards.dev/projects/github.com/PyCQA/pylint'\fI\%OpenSSF Scorecard\fP\X'tty: link'\X'tty: link https://discord.gg/qYxpadCgkx'\fI\%Discord\fP\X'tty: link'.SH WHAT IS PYLINT? .sp Pylint is a \X'tty: link https://en.wikipedia.org/wiki/Static_code_analysis'\fI\%static code analyser\fP\X'tty: link' for Python 2 or 3. The latest version supports Python 3.9.0 and above. .sp Pylint analyses your code without actually running it. It checks for errors, enforces a coding standard, looks for \X'tty: link https://martinfowler.com/bliki/CodeSmell.html'\fI\%code smells\fP\X'tty: link', and can make suggestions about how the code could be refactored. .SH INSTALL .sp For command line use, pylint is installed with: .INDENT 0.0 .INDENT 3.5 .sp .EX pip install pylint .EE .UNINDENT .UNINDENT .sp Or if you want to also check spelling with \fBenchant\fP (you might need to \X'tty: link https://pyenchant.github.io/pyenchant/install.html#installing-the-enchant-c-library'\fI\%install the enchant C library\fP\X'tty: link'): .INDENT 0.0 .INDENT 3.5 .sp .EX pip install pylint[spelling] .EE .UNINDENT .UNINDENT .sp It can also be integrated in most editors or IDEs. More information can be found \X'tty: link https://pylint.readthedocs.io/en/latest/user_guide/installation/index.html'\fI\%in the documentation\fP\X'tty: link'\&. .SH WHAT DIFFERENTIATES PYLINT? .sp Pylint is not trusting your typing and is inferring the actual values of nodes (for a start because there was no typing when pylint started off) using its internal code representation (astroid). If your code is \fBimport logging as argparse\fP, Pylint can check and know that \fBargparse.error(...)\fP is in fact a logging call and not an argparse call. This makes pylint slower, but it also lets pylint find more issues if your code is not fully typed. .INDENT 0.0 .INDENT 3.5 [inference] is the killer feature that keeps us using [pylint] in our project despite how painfully slow it is. \- \X'tty: link https://github.com/charliermarsh/ruff/issues/970#issuecomment-1381067064'\fI\%Realist pylint user\fP\X'tty: link', 2022 .UNINDENT .UNINDENT .sp pylint, not afraid of being a little slower than it already is, is also a lot more thorough than other linters. There are more checks, including some opinionated ones that are deactivated by default but can be enabled using configuration. .SH HOW TO USE PYLINT .sp Pylint isn\(aqt smarter than you: it may warn you about things that you have conscientiously done or check for some things that you don\(aqt care about. During adoption, especially in a legacy project where pylint was never enforced, it\(aqs best to start with the \fB\-\-errors\-only\fP flag, then disable convention and refactor messages with \fB\-\-disable=C,R\fP and progressively re\-evaluate and re\-enable messages as your priorities evolve. .sp Pylint is highly configurable and permits to write plugins in order to add your own checks (for example, for internal libraries or an internal rule). Pylint also has an ecosystem of existing plugins for popular frameworks and third\-party libraries. .sp \fBNOTE:\fP .INDENT 0.0 .INDENT 3.5 Pylint supports the Python standard library out of the box. Third\-party libraries are not always supported, so a plugin might be needed. A good place to start is \fBPyPI\fP which often returns a plugin by searching for \fBpylint \fP\&. \X'tty: link https://pypi.org/project/pylint-pydantic'\fI\%pylint\-pydantic\fP\X'tty: link', \X'tty: link https://github.com/pylint-dev/pylint-django'\fI\%pylint\-django\fP\X'tty: link' and \X'tty: link https://github.com/cnescatlab/pylint-sonarjson-catlab'\fI\%pylint\-sonarjson\fP\X'tty: link' are examples of such plugins. More information about plugins and how to load them can be found at \X'tty: link https://pylint.readthedocs.io/en/latest/development_guide/how_tos/plugins.html#plugins'\fI\%plugins\fP\X'tty: link'\&. .UNINDENT .UNINDENT .SH ADVISED LINTERS ALONGSIDE PYLINT .sp Projects that you might want to use alongside pylint include \X'tty: link https://github.com/astral-sh/ruff'\fI\%ruff\fP\X'tty: link' (\fBreally\fP fast, with builtin auto\-fix and a large number of checks taken from popular linters, but implemented in \fBrust\fP) or \X'tty: link https://github.com/PyCQA/flake8'\fI\%flake8\fP\X'tty: link' (a framework to implement your own checks in python using \fBast\fP directly), \X'tty: link https://github.com/python/mypy'\fI\%mypy\fP\X'tty: link', \X'tty: link https://github.com/microsoft/pyright'\fI\%pyright\fP\X'tty: link' / pylance or \X'tty: link https://github.com/facebook/pyre-check'\fI\%pyre\fP\X'tty: link' (typing checks), \X'tty: link https://github.com/PyCQA/bandit'\fI\%bandit\fP\X'tty: link' (security oriented checks), \X'tty: link https://github.com/psf/black'\fI\%black\fP\X'tty: link' and \X'tty: link https://pycqa.github.io/isort/'\fI\%isort\fP\X'tty: link' (auto\-formatting), \X'tty: link https://github.com/myint/autoflake'\fI\%autoflake\fP\X'tty: link' (automated removal of unused imports or variables), \X'tty: link https://github.com/asottile/pyupgrade'\fI\%pyupgrade\fP\X'tty: link' (automated upgrade to newer python syntax) and \X'tty: link https://github.com/DanielNoord/pydocstringformatter'\fI\%pydocstringformatter\fP\X'tty: link' (automated pep257). .SH ADDITIONAL TOOLS INCLUDED IN PYLINT .sp Pylint ships with two additional tools: .INDENT 0.0 .IP \(bu 2 \X'tty: link https://pylint.readthedocs.io/en/latest/pyreverse.html'\fI\%pyreverse\fP\X'tty: link' (standalone tool that generates package and class diagrams.) .IP \(bu 2 \X'tty: link https://pylint.readthedocs.io/en/latest/symilar.html'\fI\%symilar\fP\X'tty: link' (duplicate code finder that is also integrated in pylint) .UNINDENT .SH CONTRIBUTING .sp We welcome all forms of contributions such as updates for documentation, new code, checking issues for duplicates or telling us that we can close them, confirming that issues still exist, \X'tty: link https://pylint.readthedocs.io/en/latest/contact.html#bug-reports-feedback'\fI\%creating issues because you found a bug or want a feature\fP\X'tty: link', etc. Everything is much appreciated! .sp Please follow the \X'tty: link https://github.com/pylint-dev/pylint/blob/main/CODE_OF_CONDUCT.md'\fI\%code of conduct\fP\X'tty: link' and check \X'tty: link https://pylint.readthedocs.io/en/latest/development_guide/contribute.html'\fI\%the Contributor Guides\fP\X'tty: link' if you want to make a code contribution. .SH SHOW YOUR USAGE .sp You can place this badge in your README to let others know your project uses pylint. .INDENT 0.0 .INDENT 3.5 \X'tty: link https://github.com/pylint-dev/pylint'\fI\%\fP\X'tty: link'.UNINDENT .UNINDENT .sp Learn how to add a badge to your documentation in \X'tty: link https://pylint.readthedocs.io/en/latest/user_guide/installation/badge.html'\fI\%the badge documentation\fP\X'tty: link'\&. .SH LICENSE .sp pylint is, with a few exceptions listed below, \X'tty: link https://github.com/pylint-dev/pylint/blob/main/LICENSE'\fI\%GPLv2\fP\X'tty: link'\&. .sp The icon files are licensed under the \X'tty: link https://creativecommons.org/licenses/by-sa/4.0/'\fI\%CC BY\-SA 4.0\fP\X'tty: link' license: .INDENT 0.0 .IP \(bu 2 \X'tty: link https://raw.githubusercontent.com/pylint-dev/pylint/main/doc/logo.png'\fI\%doc/logo.png\fP\X'tty: link' .IP \(bu 2 \X'tty: link https://raw.githubusercontent.com/pylint-dev/pylint/main/doc/logo.svg'\fI\%doc/logo.svg\fP\X'tty: link' .UNINDENT .SH SUPPORT .sp Please check \X'tty: link https://pylint.readthedocs.io/en/latest/contact.html'\fI\%the contact information\fP\X'tty: link'\&. .TS box center; l|l. T{ [image: Tidelift] [image] T} T{ Professional support for pylint is available as part of the \X'tty: link https://tidelift.com/subscription/pkg/pypi-pylint?utm_source=pypi-pylint&utm_medium=referral&utm_campaign=readme'\fI\%Tidelift Subscription\fP\X'tty: link'\&. Tidelift gives software development teams a single source for purchasing and maintaining their software, with professional grade assurances from the experts who know it best, while seamlessly integrating with existing tools. T} .TE .SS Tutorial .sp This tutorial is all about approaching coding standards with little or no knowledge of in\-depth programming or the code standards themselves. It\(aqs the equivalent of skipping the manual and jumping right in. .sp The command line prompt for these examples is: .INDENT 0.0 .INDENT 3.5 .sp .EX tutor Desktop$ .EE .UNINDENT .UNINDENT .SS Getting Started .sp Running Pylint with the \fB\-\-help\fP arguments will give you an idea of the arguments available. Do that now, i.e.: .INDENT 0.0 .INDENT 3.5 .sp .EX pylint \-\-help .EE .UNINDENT .UNINDENT .sp A couple of the options that we\(aqll focus on here are: .INDENT 0.0 .INDENT 3.5 .sp .EX Commands: \-\-help\-msg= \-\-generate\-toml\-config Messages control: \-\-disable= Reports: \-\-reports= \-\-output\-format= .EE .UNINDENT .UNINDENT .sp If you need more detail, you can also ask for an even longer help message: .INDENT 0.0 .INDENT 3.5 .sp .EX pylint \-\-long\-help .EE .UNINDENT .UNINDENT .sp Pay attention to the last bit of this longer help output. This gives you a hint of what Pylint is going to \fBpick on\fP: .INDENT 0.0 .INDENT 3.5 .sp .EX Output: Using the default text output, the message format is : MESSAGE_TYPE: LINE_NUM:[OBJECT:] MESSAGE There are 5 kind of message types : * (C) convention, for programming standard violation * (R) refactor, for bad code smell * (W) warning, for python specific problems * (E) error, for probable bugs in the code * (F) fatal, if an error occurred which prevented pylint from doing further processing. .EE .UNINDENT .UNINDENT .sp When Pylint is first run on a fresh piece of code, a common complaint is that it is too \fBnoisy\fP\&. The default configuration enforce a lot of warnings. We\(aqll use some of the options we noted above to make it suit your preferences a bit better. .SS Your First Pylint\(aqing .sp We\(aqll use a basic Python script with \fBblack\fP already applied on it, as fodder for our tutorial. The starting code we will use is called \fBsimplecaesar.py\fP and is here in its entirety: .INDENT 0.0 .INDENT 3.5 .sp .EX #!/usr/bin/env python3 import string shift = 3 choice = input(\(dqwould you like to encode or decode?\(dq) word = input(\(dqPlease enter text\(dq) letters = string.ascii_letters + string.punctuation + string.digits encoded = \(dq\(dq if choice == \(dqencode\(dq: for letter in word: if letter == \(dq \(dq: encoded = encoded + \(dq \(dq else: x = letters.index(letter) + shift encoded = encoded + letters[x] if choice == \(dqdecode\(dq: for letter in word: if letter == \(dq \(dq: encoded = encoded + \(dq \(dq else: x = letters.index(letter) \- shift encoded = encoded + letters[x] print(encoded) .EE .UNINDENT .UNINDENT .sp Let\(aqs get started. If we run this: .INDENT 0.0 .INDENT 3.5 .sp .EX tutor Desktop$ pylint simplecaesar.py ************* Module simplecaesar simplecaesar.py:1:0: C0114: Missing module docstring (missing\-module\-docstring) simplecaesar.py:5:0: C0103: Constant name \(dqshift\(dq doesn\(aqt conform to UPPER_CASE naming style (invalid\-name) simplecaesar.py:8:0: C0103: Constant name \(dqletters\(dq doesn\(aqt conform to UPPER_CASE naming style (invalid\-name) simplecaesar.py:9:0: C0103: Constant name \(dqencoded\(dq doesn\(aqt conform to UPPER_CASE naming style (invalid\-name) simplecaesar.py:13:12: C0103: Constant name \(dqencoded\(dq doesn\(aqt conform to UPPER_CASE naming style (invalid\-name) simplecaesar.py:15:12: C0103: Constant name \(dqx\(dq doesn\(aqt conform to UPPER_CASE naming style (invalid\-name) simplecaesar.py:16:12: C0103: Constant name \(dqencoded\(dq doesn\(aqt conform to UPPER_CASE naming style (invalid\-name) simplecaesar.py:20:12: C0103: Constant name \(dqencoded\(dq doesn\(aqt conform to UPPER_CASE naming style (invalid\-name) simplecaesar.py:22:12: C0103: Constant name \(dqx\(dq doesn\(aqt conform to UPPER_CASE naming style (invalid\-name) simplecaesar.py:23:12: C0103: Constant name \(dqencoded\(dq doesn\(aqt conform to UPPER_CASE naming style (invalid\-name) \-\-\-\-\-\-\-\-\-\-\-\-\-\-\-\-\-\-\-\-\-\-\-\-\-\-\-\-\-\-\-\-\-\-\- Your code has been rated at 4.74/10 .EE .UNINDENT .UNINDENT .sp We can see the second line is: .INDENT 0.0 .INDENT 3.5 .sp .EX \(dqsimplecaesar.py:1:0: C0114: Missing module docstring (missing\-module\-docstring)\(dq .EE .UNINDENT .UNINDENT .sp This basically means that line 1 at column 0 violates the convention \fBC0114\fP\&. Another piece of information is the message symbol between parens, \fBmissing\-module\-docstring\fP\&. .sp If we want to read up a bit more about that, we can go back to the command line and try this: .INDENT 0.0 .INDENT 3.5 .sp .EX tutor Desktop$ pylint \-\-help\-msg=missing\-module\-docstring :missing\-module\-docstring (C0114): *Missing module docstring* Used when a module has no docstring.Empty modules do not require a docstring. This message belongs to the basic checker. .EE .UNINDENT .UNINDENT .sp That one was a bit of a no\-brainer, but we can also run into error messages where we are unfamiliar with the underlying code theory. .SS The Next Step .sp Now that we got some configuration stuff out of the way, let\(aqs see what we can do with the remaining warnings. If we add a docstring to describe what the code is meant to do that will help. There are \fBinvalid\-name\fP messages that we will get to later. Here is the updated code: .INDENT 0.0 .INDENT 3.5 .sp .EX #!/usr/bin/env python3 \(dq\(dq\(dqThis script prompts a user to enter a message to encode or decode using a classic Caesar shift substitution (3 letter shift)\(dq\(dq\(dq import string shift = 3 choice = input(\(dqwould you like to encode or decode?\(dq) word = input(\(dqPlease enter text\(dq) letters = string.ascii_letters + string.punctuation + string.digits encoded = \(dq\(dq if choice == \(dqencode\(dq: for letter in word: if letter == \(dq \(dq: encoded = encoded + \(dq \(dq else: x = letters.index(letter) + shift encoded = encoded + letters[x] if choice == \(dqdecode\(dq: for letter in word: if letter == \(dq \(dq: encoded = encoded + \(dq \(dq else: x = letters.index(letter) \- shift encoded = encoded + letters[x] print(encoded) .EE .UNINDENT .UNINDENT .sp Here is what happens when we run it: .INDENT 0.0 .INDENT 3.5 .sp .EX tutor Desktop$ pylint simplecaesar.py ************* Module simplecaesar simplecaesar.py:8:0: C0103: Constant name \(dqshift\(dq doesn\(aqt conform to UPPER_CASE naming style (invalid\-name) simplecaesar.py:11:0: C0103: Constant name \(dqletters\(dq doesn\(aqt conform to UPPER_CASE naming style (invalid\-name) simplecaesar.py:12:0: C0103: Constant name \(dqencoded\(dq doesn\(aqt conform to UPPER_CASE naming style (invalid\-name) simplecaesar.py:16:12: C0103: Constant name \(dqencoded\(dq doesn\(aqt conform to UPPER_CASE naming style (invalid\-name) simplecaesar.py:18:12: C0103: Constant name \(dqx\(dq doesn\(aqt conform to UPPER_CASE naming style (invalid\-name) simplecaesar.py:19:12: C0103: Constant name \(dqencoded\(dq doesn\(aqt conform to UPPER_CASE naming style (invalid\-name) simplecaesar.py:23:12: C0103: Constant name \(dqencoded\(dq doesn\(aqt conform to UPPER_CASE naming style (invalid\-name) simplecaesar.py:25:12: C0103: Constant name \(dqx\(dq doesn\(aqt conform to UPPER_CASE naming style (invalid\-name) simplecaesar.py:26:12: C0103: Constant name \(dqencoded\(dq doesn\(aqt conform to UPPER_CASE naming style (invalid\-name) \-\-\-\-\-\-\-\-\-\-\-\-\-\-\-\-\-\-\-\-\-\-\-\-\-\-\-\-\-\-\-\-\-\-\-\-\-\-\-\-\-\-\-\-\-\-\-\-\-\-\-\-\-\-\-\-\-\-\-\-\-\-\-\-\-\- Your code has been rated at 5.26/10 (previous run: 4.74/10, +0.53) .EE .UNINDENT .UNINDENT .sp Nice! Pylint told us how much our code rating has improved since our last run, and we\(aqre down to just the \fBinvalid\-name\fP messages. .sp There are fairly well defined conventions around naming things like instance variables, functions, classes, etc. The conventions focus on the use of UPPERCASE and lowercase as well as the characters that separate multiple words in the name. This lends itself well to checking via a regular expression, thus the \fBshould match (([A\-Z_][A\-Z1\-9_]*)|(__.*__))$\fP\&. .sp In this case Pylint is telling us that those variables appear to be constants and should be all UPPERCASE. This is an in\-house convention that has lived with Pylint since its inception. You too can create your own in\-house naming conventions but for the purpose of this tutorial, we want to stick to the \X'tty: link https://peps.python.org/pep-0008/'\fI\%PEP 8\fP\X'tty: link' standard. In this case, the variables we declared should follow the convention of all lowercase. The appropriate rule would be something like: \(dqshould match [a\-z_][a\-z0\-9_]{2,30}$\(dq. Notice the lowercase letters in the regular expression (a\-z versus A\-Z). .sp If we run that rule using a \fB\-\-const\-rgx=\(aq[a\-z\e_][a\-z0\-9\e_]{2,30}$\(aq\fP option, it will now be quite quiet: .INDENT 0.0 .INDENT 3.5 .sp .EX tutor Desktop$ pylint simplecaesar.py \-\-const\-rgx=\(aq[a\-z\e_][a\-z0\-9\e_]{2,30}$\(aq ************* Module simplecaesar simplecaesar.py:18:12: C0103: Constant name \(dqx\(dq doesn\(aqt conform to \(aq[a\-z\e\e_][a\-z0\-9\e\e_]{2,30}$\(aq pattern (invalid\-name) simplecaesar.py:25:12: C0103: Constant name \(dqx\(dq doesn\(aqt conform to \(aq[a\-z\e\e_][a\-z0\-9\e\e_]{2,30}$\(aq pattern (invalid\-name) \-\-\-\-\-\-\-\-\-\-\-\-\-\-\-\-\-\-\-\-\-\-\-\-\-\-\-\-\-\-\-\-\-\-\-\-\-\-\-\-\-\-\-\-\-\-\-\-\-\-\-\-\-\-\-\-\-\-\-\-\-\-\-\-\-\- Your code has been rated at 8.95/10 (previous run: 5.26/10, +3.68) .EE .UNINDENT .UNINDENT .sp You can \X'tty: link https://docs.python.org/library/re.html'\fI\%read up\fP\X'tty: link' on regular expressions or use \X'tty: link https://regex101.com/'\fI\%a website to help you\fP\X'tty: link'\&. .sp \fBTIP:\fP .INDENT 0.0 .INDENT 3.5 It would really be a pain to specify that regex on the command line all the time, particularly if we\(aqre using many other options. That\(aqs what a configuration file is for. We can configure our Pylint to store our options for us so we don\(aqt have to declare them on the command line. Using a configuration file is a nice way of formalizing your rules and quickly sharing them with others. Invoking \fBpylint \-\-generate\-toml\-config\fP will create a sample \fB\&.toml\fP section with all the options set and explained in comments. This can then be added to your \fBpyproject.toml\fP file or any other \fB\&.toml\fP file pointed to with the \fB\-\-rcfile\fP option. .UNINDENT .UNINDENT .SS Installation .sp Pylint can be installed: .INDENT 0.0 .IP \(bu 2 \fI\%As a command line tool\fP .IP \(bu 2 \fI\%Integrated in your editor/ide\fP .IP \(bu 2 \fI\%As a pre\-commit hook\fP .IP \(bu 2 \fI\%For multiple python interpreters in your continuous integration\fP .UNINDENT .SS Command line installation .sp Pylint is installable using a package manager. Your package manager will find a version that works with your interpreter. We recommend \fBpip\fP: .INDENT 0.0 .INDENT 3.5 .sp .EX pip install pylint .EE .UNINDENT .UNINDENT .sp Or if you want to also check spelling with \fBenchant\fP (you might need to \X'tty: link https://pyenchant.github.io/pyenchant/install.html#installing-the-enchant-c-library'\fI\%install the enchant C library\fP\X'tty: link'): .INDENT 0.0 .INDENT 3.5 .sp .EX pip install pylint[spelling] .EE .UNINDENT .UNINDENT .sp The newest pylint supports all Python interpreters that are not past end of life. .sp We recommend to use the latest interpreter because we rely on the \fBast\fP builtin module that gets better with each new Python interpreter. For example a Python 3.6 interpreter can\(aqt analyse 3.8 syntax (amongst others, because of the new walrus operator) while a 3.8 interpreter can also deal with Python 3.6. See \fI\%using pylint with multiple interpreters\fP for more details. .sp \fBNOTE:\fP .INDENT 0.0 .INDENT 3.5 You can also use \fBconda\fP or your system package manager on debian based OS. These package managers lag a little behind as they are maintained by a separate entity on a slower release cycle. .INDENT 0.0 .INDENT 3.5 .sp .EX conda install pylint .EE .UNINDENT .UNINDENT .INDENT 0.0 .INDENT 3.5 .sp .EX sudo apt\-get install pylint .EE .UNINDENT .UNINDENT .UNINDENT .UNINDENT .SS Editor and IDE integration .sp Pylint can be integrated in various editors and IDE\(aqs. Below you can find tutorials for some of the most common ones. .INDENT 0.0 .IP \(bu 2 \X'tty: link https://www.pydev.org/manual_adv_pylint.html'\fI\%Eclipse\fP\X'tty: link' .IP \(bu 2 \X'tty: link https://www.emacswiki.org/emacs/PythonProgrammingInEmacs'\fI\%Emacs\fP\X'tty: link' .IP \(bu 2 \X'tty: link https://eric-ide.python-projects.org/'\fI\%Eric IDE\fP\X'tty: link' in the \fIProject > Check\fP menu, .IP \(bu 2 \X'tty: link https://launchpad.net/gedit-pylint-2'\fI\%gedit\fP\X'tty: link' (\X'tty: link https://wiki.gnome.org/Apps/Gedit/PylintPlugin'\fI\%another option for gedit\fP\X'tty: link') .IP \(bu 2 \fI\%Flymake\fP .IP \(bu 2 \X'tty: link https://github.com/nbQA-dev/nbQA'\fI\%Jupyter Notebook\fP\X'tty: link' .IP \(bu 2 \X'tty: link https://mateusz.loskot.net/post/2006/01/15/running-pylint-from-komodo/'\fI\%Komodo\fP\X'tty: link' .IP \(bu 2 \X'tty: link https://stackoverflow.com/a/46409649/2519059'\fI\%Pycharm\fP\X'tty: link' .IP \(bu 2 \X'tty: link https://www.pydev.org/manual_adv_pylint.html'\fI\%PyDev\fP\X'tty: link' .IP \(bu 2 \X'tty: link https://github.com/pyscripter/pyscripter'\fI\%pyscripter\fP\X'tty: link' in the \fITool \-> Tools\fP menu. .IP \(bu 2 \X'tty: link https://docs.spyder-ide.org/current/panes/pylint.html'\fI\%Spyder\fP\X'tty: link' in the \fIView \-> Panes \-> Static code analysis\fP .IP \(bu 2 \X'tty: link https://packagecontrol.io/packages/SublimeLinter-pylint'\fI\%Sublime Text\fP\X'tty: link' .IP \(bu 2 \fI\%TextMate\fP .IP \(bu 2 \X'tty: link https://www.vim.org/scripts/script.php?script_id=891'\fI\%Vim\fP\X'tty: link' .IP \(bu 2 \X'tty: link https://code.visualstudio.com/docs/python/linting'\fI\%Visual Studio Code\fP\X'tty: link' in the \fIPreferences \-> Settings\fP menu .IP \(bu 2 \X'tty: link https://marketplace.visualstudio.com/items?itemName=ms-python.pylint'\fI\%Visual Studio Code Pylint Extension\fP\X'tty: link' .IP \(bu 2 \X'tty: link https://docs.microsoft.com/visualstudio/python/code-pylint'\fI\%Visual Studio\fP\X'tty: link' in the \fIPython > Run PyLint\fP command on a project\(aqs context menu. .IP \(bu 2 \X'tty: link https://wingware.com/doc/warnings/external-checkers'\fI\%WingIDE\fP\X'tty: link' .UNINDENT .SS Using Pylint through Flymake in Emacs .sp \fBWARNING:\fP .INDENT 0.0 .INDENT 3.5 epylint was deprecated in 2.16.0 and targeted for deletion in 3.0.0. All emacs and flymake related files were removed and their support will now happen in an external repository: \X'tty: link https://github.com/emacsorphanage/pylint'\fI\%https://github.com/emacsorphanage/pylint\fP\X'tty: link'\&. .UNINDENT .UNINDENT .SS Integrate Pylint with TextMate .sp Install Pylint in the usual way: .INDENT 0.0 .INDENT 3.5 .sp .EX pip install pylint .EE .UNINDENT .UNINDENT .sp Install the \X'tty: link https://github.com/textmate/python.tmbundle'\fI\%Python bundle for TextMate\fP\X'tty: link': .INDENT 0.0 .IP 1. 3 select \fITextMate\fP > \fIPreferences\fP .IP 2. 3 select the \fIBundles\fP tab .IP 3. 3 find and tick the \fIPython\fP bundle in the list .UNINDENT .sp You should now see it in \fIBundles\fP > \fIPython\fP\&. .sp In \fIPreferences\fP, select the \fIVariables\fP tab. If a \fBTM_PYCHECKER\fP variable is not already listed, add it, with the value \fBpylint\fP\&. .sp The default keyboard shortcut to run the syntax checker is \fIControl\-Shift\-V\fP \- open a \fB\&.py\fP file in Textmate, and try it. .sp You should see the output in a new window: .INDENT 0.0 .INDENT 3.5 PyCheckMate 1.2 – Pylint 1.4.4 .sp No config file found, using default configuration .UNINDENT .UNINDENT .sp Then all is well, and most likely Pylint will have expressed some opinions about your Python code (or will exit with \fB0\fP if your code already conforms to its expectations). .sp If you receive a message: .INDENT 0.0 .INDENT 3.5 Please install PyChecker, PyFlakes, Pylint, PEP 8 or flake8 for more extensive code checking. .UNINDENT .UNINDENT .sp That means that Pylint wasn\(aqt found, which is likely an issue with command paths \- TextMate needs be looking for Pylint on the right paths. .sp Check where Pylint has been installed, using \fBwhich\fP: .INDENT 0.0 .INDENT 3.5 .sp .EX $ which pylint /usr/local/bin/pylint .EE .UNINDENT .UNINDENT .sp The output will tell you where Pylint can be found; in this case, in \fB/usr/local/bin\fP\&. .INDENT 0.0 .IP 1. 3 select \fITextMate\fP > \fIPreferences\fP .IP 2. 3 select the \fIVariables\fP tab .IP 3. 3 find and check that a \fBPATH\fP variable exists, and that it contains the appropriate path (if the path to Pylint were \fB/usr/local/bin/pylint\fP as above, then the variable would need to contain \fB/usr/local/bin\fP). An actual example in this case might be \fB$PATH:/opt/local/bin:/usr/local/bin:/usr/texbin\fP, which includes other paths. .UNINDENT .sp \&... and try running Pylint again. .SS Pre\-commit integration .sp \fBpylint\fP can be used as a \X'tty: link https://pre-commit.com'\fI\%pre\-commit\fP\X'tty: link' hook. We however discourage it as pylint \-\- due to its speed \-\- is more suited to a continuous integration job or a git \fBpre\-push\fP hook, especially if your repository is large. .sp Since \fBpylint\fP needs to import modules and dependencies to work correctly, the hook only works with a local installation of \fBpylint\fP (in your environment). It means it can\(aqt be used with \fBpre\-commit.ci\fP, and you will need to add the following to your \fB\&.pre\-commit\-config.yaml\fP .INDENT 0.0 .INDENT 3.5 .sp .EX \&.. sourcecode:: yaml .EE .UNINDENT .UNINDENT .INDENT 0.0 .INDENT 3.5 .INDENT 0.0 .TP .B ci: skip: [pylint] .UNINDENT .UNINDENT .UNINDENT .sp Another limitation is that pylint should analyse all your code at once in order to best infer the actual values that result from calls. If only some of the files are given, pylint might miss a particular value\(aqs type and produce inferior inference for the subset. Since pre\-commit slices the files given to it in order to parallelize the processing, the result can be degraded. It can also be unexpectedly different when the file set changes because the new slicing can change the inference. Thus the \fBrequire_serial\fP option should be set to \fBtrue\fP if correctness and determinism are more important than parallelization to you. .sp If you installed \fBpylint\fP locally it can be added to \fB\&.pre\-commit\-config.yaml\fP as follows: .INDENT 0.0 .INDENT 3.5 .sp .EX \- repo: local hooks: \- id: pylint name: pylint entry: pylint language: system types: [python] require_serial: true args: [ \(dq\-rn\(dq, # Only display messages \(dq\-sn\(dq, # Don\(aqt display the score ] .EE .UNINDENT .UNINDENT .sp You can use \fBargs\fP to pass command line arguments as described in the \fI\%Tutorial\fP\&. A hook with more arguments could look something like this: .INDENT 0.0 .INDENT 3.5 .sp .EX \- repo: local hooks: \- id: pylint name: pylint entry: pylint language: system types: [python] args: [ \(dq\-rn\(dq, # Only display messages \(dq\-sn\(dq, # Don\(aqt display the score \(dq\-\-rcfile=pylintrc\(dq, # Link to your config file \(dq\-\-load\-plugins=pylint.extensions.docparams\(dq, # Load an extension ] .EE .UNINDENT .UNINDENT .SS Installation with multiple interpreters .sp It\(aqs possible to analyse code written for older or multiple interpreters by using the \fBpy\-version\fP option and setting it to the oldest supported interpreter of your code. For example you can check that there are no \fBf\-strings\fP in Python 3.5 code using Python 3.8 with an up\-to\-date pylint even if Python 3.5 is past end of life (EOL) and the version of pylint you use is not compatible with it. .sp We do not guarantee that \fBpy\-version\fP will work for all EOL Python interpreters indefinitely, (for anything before Python 3.5, it probably won\(aqt). If a newer version does not work for you, the best available pylint might be an old version that works with your old interpreter but without the bug fixes and features of later versions. .SS Show your usage .sp You can place this badge in your README to let others know your project uses pylint. .INDENT 0.0 .INDENT 3.5 \X'tty: link https://github.com/pylint-dev/pylint'\fI\%\fP\X'tty: link'.UNINDENT .UNINDENT .sp Use the badge in your project\(aqs README.md (or any other Markdown file): .INDENT 0.0 .INDENT 3.5 .sp .EX [![linting: pylint](https://img.shields.io/badge/linting\-pylint\-yellowgreen)](https://github.com/pylint\-dev/pylint) .EE .UNINDENT .UNINDENT .sp Use the badge in your project\(aqs README.rst (or any other rst file): .INDENT 0.0 .INDENT 3.5 .sp .EX \&.. image:: https://img.shields.io/badge/linting\-pylint\-yellowgreen :target: https://github.com/pylint\-dev/pylint .EE .UNINDENT .UNINDENT .sp If you use GitHub Actions, and one of your CI workflows begins with \(dqname: pylint\(dq, you can use GitHub\(aqs \X'tty: link https://docs.github.com/en/actions/monitoring-and-troubleshooting-workflows/adding-a-workflow-status-badge#using-the-workflow-file-name'\fI\%workflow status badges\fP\X'tty: link' to show an up\-to\-date indication of whether pushes to your default branch pass pylint. For more detailed information, check the documentation. .SS Upgrading pylint .sp You should probably set the version of pylint in your development environment in order to choose when you actually upgrade pylint\(aqs warnings. pylint is following semver versioning. But we can\(aqt guarantee that the output between version will stays the same. What this means is that: .INDENT 0.0 .IP \(bu 2 In \fBpatch release\fP (\fB1.2.3\fP), we only fix false positives and crashes. .UNINDENT .sp You can expect less messages if you set the minor and upgrade to a new patch version. But still, if you enable \fBuseless\-suppression\fP it still means you can get a new \fBuseless\-suppression\fP when a false positive that you disabled is now fixed. Also, if a library you\(aqre using was upgraded and is understood better or worse than the previous one, you could get new messages too. .INDENT 0.0 .IP \(bu 2 In \fBminor releases\fP (\fB1.2.0\fP), we add checks, remove checks, drop python interpreters past end of life, upgrade astroid minor/major versions and fix false negatives. .UNINDENT .sp You can expect a lot more change in output, the main one being new checks. .INDENT 0.0 .IP \(bu 2 In \fBmajor releases\fP (\fB1.0.0\fP) we change everything else (pylint options, json output, dev API...) while still trying to minimize disruption. .UNINDENT .sp You could have to change the command you\(aqre launching or the plugin and editor integration you\(aqre using. .SS Usage .SS Running Pylint .SS On module packages or directories .sp Pylint is meant to be called from the command line. The usage is .INDENT 0.0 .INDENT 3.5 .sp .EX pylint [options] modules_or_packages .EE .UNINDENT .UNINDENT .sp By default the \fBpylint\fP command only accepts a list of python modules and packages. On versions below 2.15, specifying a directory that is not an explicit package (with \fB__init__.py\fP) results in an error: .INDENT 0.0 .INDENT 3.5 .sp .EX pylint mydir ************* Module mydir mydir/__init__.py:1:0: F0010: error while code parsing: Unable to load file mydir/__init__.py: [Errno 2] No such file or directory: \(aqmydir/__init__.py\(aq (parse\-error) .EE .UNINDENT .UNINDENT .sp Thus, on versions before 2.15 using the \fB\-\-recursive=y\fP option allows for linting a namespace package: .INDENT 0.0 .INDENT 3.5 .sp .EX pylint \-\-recursive=y mydir mymodule mypackage .EE .UNINDENT .UNINDENT .sp This option makes \fBpylint\fP attempt to discover all modules (files ending with \fB\&.py\fP extension) and all explicit packages (all directories containing a \fB__init__.py\fP file). .sp Pylint \fBwill not import\fP this package or module, but it does use Python internals to locate them and as such is subject to the same rules and configuration. You should pay attention to your \fBPYTHONPATH\fP, since it is a common error to analyze an installed version of a module instead of the development version. .SS On files .sp It is also possible to analyze Python files, with a few restrictions. As a convenience, you can give it a file name if it\(aqs possible to guess a module name from the file\(aqs path using the python path. Some examples: .sp \fBpylint mymodule.py\fP should always work since the current working directory is automatically added on top of the python path .sp \fBpylint directory/mymodule.py\fP will work if: \fBdirectory\fP is a python package (i.e. has an \fB__init__.py\fP file), an implicit namespace package or if \fBdirectory\fP is in the python path. .SS With implicit namespace packages .sp If the analyzed sources use implicit namespace packages (PEP 420), the source root(s) should be specified using the \fB\-\-source\-roots\fP option. Otherwise, the package names are detected incorrectly, since implicit namespace packages don\(aqt contain an \fB__init__.py\fP\&. .SS Globbing support .sp It is also possible to specify both directories and files using globbing patterns: .INDENT 0.0 .INDENT 3.5 .sp .EX pylint [options] packages/*/src .EE .UNINDENT .UNINDENT .SS Command line options .sp First of all, we have two basic (but useful) options. .INDENT 0.0 .TP .B \-\-version show program\(aqs version number and exit .TP .B \-h\fP,\fB \-\-help show help about the command line options .UNINDENT .sp Pylint is architected around several checkers. You can disable a specific checker or some of its messages or message categories by specifying \fB\-\-disable=\fP\&. If you want to enable only some checkers or some message symbols, first use \fB\-\-disable=all\fP then \fB\-\-enable=\fP with \fB\fP being a comma\-separated list of checker names and message symbols. See the list of available features for a description of provided checkers with their functionalities. The \fB\-\-disable\fP and \fB\-\-enable\fP options can be used with comma\-separated lists mixing checkers, message ids and categories like \fB\-d C,W,no\-error,design\fP .sp It is possible to disable all messages with \fB\-\-disable=all\fP\&. This is useful to enable only a few checkers or a few messages by first disabling everything, and then re\-enabling only what you need. .sp Each checker has some specific options, which can take either a yes/no value, an integer, a python regular expression, or a comma\-separated list of values (which are generally used to override a regular expression in special cases). For a full list of options, use \fB\-\-help\fP .sp Specifying all the options suitable for your setup and coding standards can be tedious, so it is possible to use a configuration file to specify the default values. You can specify a configuration file on the command line using the \fB\-\-rcfile\fP option. Otherwise, Pylint searches for a configuration file in the following order and uses the first one it finds: .INDENT 0.0 .IP 1. 4 \fBpylintrc\fP in the current working directory .IP 2. 4 \fBpylintrc.toml\fP in the current working directory, providing it has at least one \fBtool.pylint.\fP section. .IP 3. 4 \fB\&.pylintrc\fP in the current working directory .IP 4. 4 \fB\&.pylintrc.toml\fP in the current working directory, providing it has at least one \fBtool.pylint.\fP section. .IP 5. 4 \fBpyproject.toml\fP in the current working directory, providing it has at least one \fBtool.pylint.\fP section. The \fBpyproject.toml\fP must prepend section names with \fBtool.pylint.\fP, for example \fB[tool.pylint.\(aqMESSAGES CONTROL\(aq]\fP\&. They can also be passed in on the command line. .IP 6. 4 \fBsetup.cfg\fP in the current working directory, providing it has at least one \fBpylint.\fP section .IP 7. 4 \fBtox.ini\fP in the current working directory, providing it has at least one \fBpylint.\fP section .IP 8. 4 Pylint will search for the \fBpyproject.toml\fP file up the directories hierarchy unless it\(aqs found, or a \fB\&.git\fP/\fB\&.hg\fP directory is found, or the file system root is approached. .IP 9. 4 If the current working directory is in a Python package, Pylint searches up the hierarchy of Python packages until it finds a \fBpylintrc\fP file. This allows you to specify coding standards on a module\-by\-module basis. Of course, a directory is judged to be a Python package if it contains an \fB__init__.py\fP file. .IP 10. 4 The file named by environment variable \fBPYLINTRC\fP .IP 11. 4 if you have a home directory which isn\(aqt \fB/root\fP: .INDENT 4.0 .IP 1. 3 \fB\&.pylintrc\fP in your home directory .IP 2. 3 \fB\&.config/pylintrc\fP in your home directory .UNINDENT .IP 12. 4 \fB/etc/pylintrc\fP .UNINDENT .sp The \fB\-\-generate\-toml\-config\fP option will generate a commented configuration file on standard output according to the current configuration and exit. This includes: .INDENT 0.0 .IP \(bu 2 Any configuration file found as explained above .IP \(bu 2 Options appearing before \fB\-\-generate\-toml\-config\fP on the Pylint command line .UNINDENT .sp Of course you can also start with the default values and hand\-tune the configuration. .sp Other useful global options include: .INDENT 0.0 .TP .BI \-\-ignore\fB= Files or directories to be skipped. They should be base names, not paths. .TP .BI \-\-output\-format\fB= Select output format (text, json, custom). .TP .BI \-\-msg\-template\fB=