Skip to content

Latest commit

 

History

History
225 lines (167 loc) · 7.37 KB

README.md

File metadata and controls

225 lines (167 loc) · 7.37 KB

QAway Linter

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).

Rules

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).

Functions: minCommentDensity

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%

Functions: requireHeadlineComment

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

Functions: trivialCommentThreshold

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%

Functions: minLoggingDensity

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%

Interfaces: requireHeadlineComment

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

Interfaces: requireMethodComment

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

Structs: requireHeadlineComment

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

Structs: requireFieldComment

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

Usage

  1. 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
  1. Execute golangci-lint custom in the same directory to build a custom version of golangci-lint with the qaway-linter plugin.

  2. 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
  1. Execute the custom version by running ./custom-gcl run in your project's root directory.

Exclusions

Add // nolint:qawaylinter to the line you want to exclude from the linter. It is not possible to disable individual rules from the configuration.

Internal architecture

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.