The QAway linter for golangci-lint enforces coding best practices, especially in terms of documentation. It allows you to define rules for your codebase with maximum flexibility as different parts of your code may require different rules.
The linter is not yet integrated directly into go-langci-lint, but can be used as a custom plugin (see Usage).
This chapter demonstrates the rules that can be configured in the qaway-linter
plugin. The rules are defined in the
configuration and can be enabled or disabled for specific packages (see Usage).
A method must have at least minCommentDensity
of comments (headline + inline) compared to its lines of code. Often
used in combination with the filter minLinesOfCode
to exclude small methods from this rule.
Example function for minCommentDensity: 0.1
:
func WithoutComments() string {
string := "Hello, World!"
return string
}
Violation: Method 'WithoutComments' has less than 10% comment density. Actual: 0%
A headline comment is required for every method. A headline comment is the comment on top of the method.
Example function without headline comment:
func WithoutHeadlineComment() string {
// This is an inline comment
string := "Hello, World!"
return string
}
Violation: Method 'WithoutHeadlineComment' is missing required headline comment
Trivial headline comments (similarity to method name) are not allowed.
The threshold indicates the similarity to the method name.
A higher threshold indicates a higher required similarity for throwing a violation, resulting in less warnings. A good
starting point is a value of 0.3
.
Example function with trivial comment:
// DownloadArtifacts downloads the artifacts
func DownloadArtifacts() string {
string := "Hello, World!"
return string
}
Violation: Method 'downloadArtifacts' has a trivial comment. Similarity to method name: 40%
Amount of logging statements compared to lines of code. The rule helps enforcing a minimum amount of logging in certain aspects of your codebase.
Example function with low logging density:
func WithoutLogging() string {
string := "Hello, World!"
return string
}
Violation: Method 'WithoutLogging' has less than 0% logging density. Actual: 0%
A headline comment is required for every interface.
Example interface without headline comment:
type WithoutHeadlineComment interface {
MethodWithoutComment()
}
Violation: Interface 'WithoutHeadlineComment' is missing required headline comment
A comment is required for every method in an interface.
Example interface method without comment:
type WithoutMethodComment interface {
MethodWithoutComment()
}
Violation: Method 'MethodWithoutComment' is missing required comment
A headline comment is required for every struct.
Example struct without headline comment:
type WithoutHeadlineComment struct {
FieldWithoutComment string
}
Violation: Struct 'WithoutHeadlineComment' is missing required headline comment
A comment is required for every field in a struct.
Example struct field without comment:
type WithoutFieldComment struct {
FieldWithoutComment string
}
Violation: Field 'FieldWithoutComment' is missing required comment
- Create a file called
.custom-gcl.yml
in your projects root directory with the following content:
version: v1.62.2 # TODO: update to latest version (see https://github.com/golangci/golangci-lint/releases/tag/v1.62.2)
plugins:
- module: 'github.com/qaware/qaway-linter'
import: 'github.com/qaware/qaway-linter'
version: v0.0.1 # TODO: use latest version from GitHub
-
Execute
golangci-lint custom
in the same directory to build a custom version of golangci-lint with theqaway-linter
plugin. -
Extend the configuration in your
.golangci.yml
. Customize the rules according to your codebase. Note that more concrete packages override the configuration of more general packages .
linters:
enable:
# add to existing list if linters.disable-all is set to true
- qawaylinter
linter-settings:
custom:
qawaylinter:
type: "module"
description: "Checks for appropriate documentation in code"
settings:
rules:
- packages: [ "github.com/myorg/myrepo" ]
# This rule demonstrates all available configuration options
# If a parameter is not set, it is not enforced.
functions:
filters:
# Apply parameters only to functions with at least 10 lines of code
minLinesOfCode: 10
params:
# A method must have at least 10% of comments (headline + inline) compared to its lines of code
minCommentDensity: 0.1
# A headline comment is required for every method
requireHeadlineComment: true
# Trivial comments (similarity to method name) are not allowed.
# The threshold indicates the similarity to the method name.
# A higher threshold indicates a higher similarity, resulting in less warnings.
trivialCommentThreshold: 0.3
# Amount of logging statements compared to lines of code.
minLoggingDensity: 0.0
interfaces:
params:
# A headline comment is required for every interface
requireHeadlineComment: true
# A comment is required for every method in an interface
requireMethodComment: true
structs:
params:
# A headline comment is required for every struct
requireHeadlineComment: true
# A comment is required for every field in a struct
requireFieldComment: false
- packages: [ "github.com/myorg/myrepo/subpkg" ] # rules for subpackage override super packages
functions:
filters:
minLinesOfCode: 20
params:
trivialCommentThreshold: 0.5
minLoggingDensity: 0.1
- Execute the custom version by running
./custom-gcl run
in your project's root directory.
Add // nolint:qawaylinter
to the line you want to exclude from the linter. It is not possible to disable individual
rules from the configuration.
A generic Rule
interface is defined in rule.go. Each rule implements this interface and is responsible for
checking a specific aspect of the code. The Rule
interface defines three methods:
isApplicable
: Checks if the rule is applicable to the given code element (node).Analyse
: Analyzes the code element and returns a list of findings.Apply
: Validates if the results from the analysis violate rules from the configuration and reports errors.
Each rule is called in the analyser for each code element. The analyser is responsible for traversing the code and calling the rules. New rules must be added to the analyser to be executed.