Skip to content

GispoCoding/flake8-qgis

Repository files navigation

flake8-qgis

PyPI version Downloads CI Code on Github Code style: black Imports: isort pre-commit

A flake8 plugin for QGIS3 python plugins written in Python.

Made with Cookiecutter template cookiecutter-flake8-plugin. Inspired by flake8-simplify.

Installation

Install with pip:

pip install flake8-qgis

Usage

Just call flake8 . in your package or flake your.py.

Rules

Rule Description
QGS101 Avoid using from-imports from qgis protected members
QGS102 Avoid using imports from qgis protected members
QGS103 Avoid using from-imports from PyQt directly
QGS104 Avoid using imports from PyQt directly
QGS105 Avoid passing QgisInterface as an argument
QGS106 Avoid importing gdal directly, import it from osgeo package

Please check the Examples section below for good and bad usage examples for each rule.

While it's important to adhere to these rules, there might be good reasons to ignore some of them. You can do so by using the standard Flake8 configuration. For example, in the setup.cfg file:

[flake8]
ignore = QGS101, QGS102

QGS101

Avoid using from-imports from qgis protected members

An exception is made for importing qgis._3d (since flake-qgis 1.1.0). The underscore in the package name is used to prevent the name from starting with a number, ensuring it is a valid package name.

Why is this bad?

Protected members are potentially unstable across software versions. Future changes in protected members might cause problems.

Example

# Bad
from qgis._core import QgsMapLayer, QgsVectorLayer
from qgis._core import QgsApplication

# Good
from qgis.core import QgsMapLayer, QgsVectorLayer
from qgis.core import QgsApplication

QGS102

Avoid using imports from qgis protected members

An exception is made for importing qgis._3d (since flake-qgis 1.1.0). The underscore in the package name is used to prevent the name from starting with a number, ensuring it is a valid package name.

Why is this bad?

Protected members are potentially unstable across software versions. Future changes in protected members might cause problems.

Example

# Bad
import qgis._core.QgsVectorLayer as QgsVectorLayer

# Good
import qgis.core.QgsVectorLayer as QgsVectorLayer

QGS103

Avoid using from-imports from PyQt directly

Why is this bad?

Importing directly from PyQt might create conflict with QGIS bundled PyQt version

Example

# Bad
from PyQt5.QtCore import pyqtSignal

# Good
from qgis.PyQt.QtCore import pyqtSignal

QGS104

Avoid using imports from PyQt directly

Why is this bad?

Importing directly from PyQt might create conflict with QGIS bundled PyQt version

Example

# Bad
import PyQt5.QtCore.pyqtSignal as pyqtSignal

# Good
import qgis.PyQt.QtCore.pyqtSignal as pyqtSignal

QGS105

Avoid passing QgisInterface as an argument

Why is this bad?

It is much easier to import QgisInterface, and it's easier to mock it as well when writing tests. This approach is not however documented properly, so the API might change at some point to exclude this.

This rule can be excluded safely since this is only a matter of preference. Passing iface as an argument is the documented way of getting QgisInterface in plugins. However, it requires writing more code.

Example

# Bad: iface passed as argument
def some_function(somearg, iface):
    # do something with iface


# Good: iface imported
from qgis.utils import iface

def some_function(somearg):
    # do something with iface
# in classFactory the passing is OK, since QGIS injects it
def classFactory(iface):
    # preferably do not pass the iface to plugin

QGS106

Avoid importing gdal directly, import it from osgeo package

Why is this bad?

Importing directly from gdal might create conflict with different gdal versions

Example

# Bad
import gdal
import ogr

# Good
from osgeo import gdal

About

Flake8 plugin for QGIS python plugins

Resources

License

Stars

Watchers

Forks

Packages

No packages published

Contributors 4

  •  
  •  
  •  
  •  

Languages