Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

nolintlint: reports false positive warnings about used nolint directive #3228

Open
4 tasks done
fho opened this issue Sep 21, 2022 · 15 comments
Open
4 tasks done

nolintlint: reports false positive warnings about used nolint directive #3228

fho opened this issue Sep 21, 2022 · 15 comments
Labels
area: nolint Related to nolint directive and nolintlint bug Something isn't working

Comments

@fho
Copy link

fho commented Sep 21, 2022

Welcome

  • Yes, I'm using a binary release within 2 latest major releases. Only such installations are supported.
  • Yes, I've searched similar issues on GitHub and didn't find any.
  • Yes, I've included all information below (version, config, etc).
  • Yes, I've tried with the standalone linter if available. (https://golangci-lint.run/usage/linters/)

Description of the problem

We have staticcheck and nonolint enabled.
Sometimes in our CI builds golangci-lint reports a false warnings from nonolint about unused nolint:staticcheck comments.
All warnings that I checked are wrong, the //nolint:staticcheck directives always silence existing deprecation warnings from staticcheck.

I was not able to reproduce this issue.
In CI we reuse the golangci-lint and go cache when running checks on different branches.
It can happen that the same cache was used previously with older go and/or golangci-lint versions.

I believe #1940 (comment) refers to the same issue.

Version of golangci-lint

$ golangci-lint --version
golangci-lint has version 1.49.0 built from cc2d97f3 on 2022-08-24T10:24:37Z

Configuration file

$ cat .golangci.yml
run:
  concurrency: 2
  deadline: 5m

linters:
  disable-all: true
  enable:
    - depguard
    - errcheck
    - exportloopref
    - goimports
    - goprintffuncname
    - ineffassign
    - misspell
    - nolintlint
    - revive
    - rowserrcheck
    - staticcheck
    - typecheck
    - unconvert
    - unused
    - vet

issues:
  exclude-use-default: false

Go environment

$ go version && go env
[/src/go/code/orders]$ go version && go env
go version go1.19.1 linux/amd64
GO111MODULE=""
GOARCH="amd64"
GOBIN="/usr/local/bin"
GOCACHE="/var/cache/go-build"
GOENV="/tmp/.config/go/env"
GOEXE=""
GOEXPERIMENT=""
GOFLAGS="-tags=timetzdata -trimpath"
GOHOSTARCH="amd64"
GOHOSTOS="linux"
GOINSECURE=""
GOMODCACHE="/var/cache/go-modcache"
GONOPROXY=""
GONOSUMDB=""
GOOS="linux"
GOPATH="/tmp/go"
GOPRIVATE=""
GOPROXY="https://proxy.golang.org,direct"
GOROOT="/opt/go"
GOSUMDB="sum.golang.org"
GOTMPDIR=""
GOTOOLDIR="/opt/go/pkg/tool/linux_amd64"
GOVCS=""
GOVERSION="go1.19.1"
GCCGO="gccgo"
GOAMD64="v1"
AR="ar"
CC="gcc"
CXX="g++"
CGO_ENABLED="0"
GOMOD="/src/go/go.mod"
GOWORK=""
CGO_CFLAGS="-g -O2"
CGO_CPPFLAGS=""
CGO_CXXFLAGS="-g -O2"
CGO_FFLAGS="-g -O2"
CGO_LDFLAGS="-g -O2"
PKG_CONFIG="pkg-config"
GOGCCFLAGS="-fPIC -m64 -Wl,--no-gc-sections -fmessage-length=0 -fdebug-prefix-map=/tmp/go-build1328424340=/tmp/go-build -gno-record-gcc-switches"

Verbose output of running

N/A

Code example or link to a public repository

Example case, output from golangci-lint:

[..]/itemdetails.go:36:32: directive `//nolint:staticcheck // Description field is deprecated` is unused for linter "staticcheck" (nolintlint)
		return item.Description, nil //nolint:staticcheck // Description field is deprecated

The referenced Description field has a deprecation comment set:

// Deprecated: Do not use.
Description      string                 `protobuf:"bytes,4,opt,name=description,proto3" json:"description,omitempty"`
@fho fho added the bug Something isn't working label Sep 21, 2022
@ldez
Copy link
Member

ldez commented Sep 21, 2022

Hello,

I change some elements related to facts (they are used by staticcheck) maybe the problem will be fixed with the next release (v1.50.0)

@fho
Copy link
Author

fho commented Nov 1, 2022

The issue happend again with golangci-lint 1.50.0.

@nocive

This comment was marked as off-topic.

justinsb added a commit to justinsb/kustomize that referenced this issue Mar 20, 2023
The lint task was failing at head, due to a nolint:exhaustive error
directive that golangci nolintlint believes is unused.

Issue seems to be golangci/golangci-lint#3228
and seems to be a bug in golang-ci / nolintlint.
justinsb added a commit to justinsb/kustomize that referenced this issue Mar 20, 2023
The lint task was failing at head, due to a nolint:exhaustive error
directive that golangci nolintlint believes is unused.

Issue seems to be golangci/golangci-lint#3228
and seems to be a bug in golang-ci / nolintlint, using the workaround
proposed in golangci/golangci-lint#1940
which is to clear the cache between runs.
haines added a commit to haines/cerbos that referenced this issue Sep 13, 2023
haines added a commit to haines/cerbos that referenced this issue Sep 15, 2023
haines added a commit to haines/cerbos that referenced this issue Sep 19, 2023
haines added a commit to cerbos/cerbos that referenced this issue Sep 19, 2023
…ons (#1778)

* Unify type of `request` in expressions
* Support `P.derivedRoles` in checks
* Add `yaml-language-server` modelines to query planner test policies
* Support `P.derivedRoles` in plans
* Add documentation
* Fix broken link in v0.3.0 release notes
* Disable flaky unused `nolint` detection
  golangci/golangci-lint#3228
* Support `request.auxData` as well as `request.aux_data` in plans
* Rename to `runtime.effectiveDerivedRoles`, and compute lazily

Signed-off-by: Andrew Haines <[email protected]>
@iwata
Copy link

iwata commented Oct 11, 2023

Happened with golangci-lint v1.54.2, too.

EmilienM added a commit to shiftstack/cluster-api-provider-openstack that referenced this issue Oct 23, 2023
EmilienM added a commit to shiftstack/cluster-api-provider-openstack that referenced this issue Oct 23, 2023
Hitting golangci/golangci-lint#3228 when
adding nolint.

So allow to ignore unused nolints.
EmilienM added a commit to shiftstack/cluster-api-provider-openstack that referenced this issue Oct 24, 2023
Hitting golangci/golangci-lint#3228 when
adding nolint.

So allow to ignore unused nolints.
EmilienM added a commit to shiftstack/cluster-api-provider-openstack that referenced this issue Oct 24, 2023
Hitting golangci/golangci-lint#3228 when
adding nolint.

So allow to ignore unused nolints.
EmilienM added a commit to shiftstack/cluster-api-provider-openstack that referenced this issue Oct 24, 2023
Hitting golangci/golangci-lint#3228 when
adding nolint.

So allow to ignore unused nolints.
EmilienM added a commit to shiftstack/cluster-api-provider-openstack that referenced this issue Oct 24, 2023
Hitting golangci/golangci-lint#3228 when
adding nolint.

So allow to ignore unused nolints.
EmilienM added a commit to shiftstack/cluster-api-provider-openstack that referenced this issue Oct 25, 2023
Hitting golangci/golangci-lint#3228 when
adding nolint.

So allow to ignore unused nolints.
EmilienM added a commit to shiftstack/cluster-api-provider-openstack that referenced this issue Oct 25, 2023
Hitting golangci/golangci-lint#3228 when
adding nolint.

So allow to ignore unused nolints.
EmilienM added a commit to shiftstack/cluster-api-provider-openstack that referenced this issue Oct 27, 2023
Hitting golangci/golangci-lint#3228 when
adding nolint.

So allow to ignore unused nolints.

(cherry picked from commit 31b0732)
EmilienM added a commit to shiftstack/cluster-api-provider-openstack that referenced this issue Oct 27, 2023
Hitting golangci/golangci-lint#3228 when
adding nolint.

So allow to ignore unused nolints.
EmilienM added a commit to shiftstack/cluster-api-provider-openstack that referenced this issue Oct 27, 2023
Hitting golangci/golangci-lint#3228 when
adding nolint.

So allow to ignore unused nolints.
EmilienM added a commit to shiftstack/cluster-api-provider-openstack that referenced this issue Oct 27, 2023
Hitting golangci/golangci-lint#3228 when
adding nolint.

So allow to ignore unused nolints.
EmilienM added a commit to shiftstack/cluster-api-provider-openstack that referenced this issue Oct 30, 2023
Hitting golangci/golangci-lint#3228 when
adding nolint.

So allow to ignore unused nolints.
EmilienM added a commit to shiftstack/cluster-api-provider-openstack that referenced this issue Oct 30, 2023
Hitting golangci/golangci-lint#3228 when
adding nolint.

So allow to ignore unused nolints.

(cherry picked from commit 4336f38)
@EvgenySerg
Copy link

Have this issue also when dealing with switch/case:
nolint:exhaustive` is unused for linter "exhaustive" (nolintlint)
v1.55.2

@ldez
Copy link
Member

ldez commented Dec 19, 2023

If you are facing the same problem, the best way to contribute is to provide the following information:

golangci-lint --version
cat .golangci.yml
go version && go env
golangci-lint cache clean
golangci-lint run -v

And a code example or link to a public repository.

mandre pushed a commit to shiftstack/cluster-api-provider-openstack that referenced this issue Feb 7, 2024
Hitting golangci/golangci-lint#3228 when
adding nolint.

So allow to ignore unused nolints.

(cherry picked from commit 4336f38)
@rliebz
Copy link
Contributor

rliebz commented Feb 21, 2024

Here's a reliable repro:

#!/bin/sh -e

## setup module
rm -rf nolintrepro
mkdir nolintrepro && cd $_

## go.mod
cat > go.mod <<EOF
module example.com/nolintrepro

go 1.21

require github.com/jackc/pgx/v5 v5.5.3

require (
	github.com/jackc/pgpassfile v1.0.0 // indirect
	github.com/jackc/pgservicefile v0.0.0-20221227161230-091c0ba34f0a // indirect
	golang.org/x/crypto v0.17.0 // indirect
	golang.org/x/text v0.14.0 // indirect
)
EOF

## main.go
cat > main.go <<EOF
package main

import (
	"fmt"

	"github.com/jackc/pgx/v5"
)

func main() {
	var conn *pgx.Conn
	fmt.Println(conn.PgConn().CheckConn() == nil)
	fmt.Println(conn.PgConn().CheckConn() == nil) //nolint: staticcheck
}
EOF

go mod tidy

## .golangci.yml
cat > .golangci.yml <<EOF
linters:
  disable-all: true
  enable:
    - staticcheck
    - nolintlint
EOF

## Run
golangci-lint run

## Clean
cd ..
rm -rf nolintrepro
version and env
$ go install github.com/golangci/golangci-lint/cmd/golangci-lint@latest
$ golangci-lint --version
golangci-lint has version v1.56.2 built with go1.22.0 from (unknown, mod sum: "h1:dgQzlWHgNbCqJjuxRJhFEnHDVrrjuTGQHJ3RIZMpp/o=") on (unknown)
$ go version && go env
go version go1.22.0 darwin/arm64
GO111MODULE=''
GOARCH='arm64'
GOBIN=''
GOCACHE='/Users/me/Library/Caches/go-build'
GOENV='/Users/me/Library/Application Support/go/env'
GOEXE=''
GOEXPERIMENT=''
GOFLAGS=''
GOHOSTARCH='arm64'
GOHOSTOS='darwin'
GOINSECURE=''
GOMODCACHE='/Users/me/.local/share/go/pkg/mod'
GONOPROXY=''
GONOSUMDB=''
GOOS='darwin'
GOPATH='/Users/me/.local/share/go'
GOPRIVATE=''
GOPROXY='https://proxy.golang.org,direct'
GOROOT='/opt/homebrew/Cellar/go/1.22.0/libexec'
GOSUMDB='sum.golang.org'
GOTMPDIR=''
GOTOOLCHAIN='auto'
GOTOOLDIR='/opt/homebrew/Cellar/go/1.22.0/libexec/pkg/tool/darwin_arm64'
GOVCS=''
GOVERSION='go1.22.0'
GCCGO='gccgo'
AR='ar'
CC='cc'
CXX='c++'
CGO_ENABLED='1'
GOMOD='/Users/me/nolintrepro/go.mod'
GOWORK=''
CGO_CFLAGS='-O2 -g'
CGO_CPPFLAGS=''
CGO_CXXFLAGS='-O2 -g'
CGO_FFLAGS='-O2 -g'
CGO_LDFLAGS='-O2 -g'
PKG_CONFIG='pkg-config'
GOGCCFLAGS='-fPIC -arch arm64 -pthread -fno-caret-diagnostics -Qunused-arguments -fmessage-length=0 -ffile-prefix-map=/var/folders/yd/g6bkd_fd7qs40y6q_8w2fh2w0000gn/T/go-build3765820227=/tmp/go-build -gno-record-gcc-switches -fno-common'
$ golangci-lint cache clean
$ golangci-lint run -v
INFO [config_reader] Config search paths: [./ /Users/me/nolintrepro /Users/me /Users /] 
INFO [config_reader] Used config file .golangci.yml 
INFO [lintersdb] Active 2 linters: [nolintlint staticcheck] 
INFO [loader] Go packages loading at mode 575 (deps|name|types_sizes|compiled_files|exports_file|files|imports) took 523.977875ms 
INFO [runner/filename_unadjuster] Pre-built 0 adjustments in 161µs 
INFO [linters_context/goanalysis] analyzers took 2.663033296s with top 10 stages: buildir: 2.039481668s, fact_deprecated: 150.111285ms, nilness: 126.620753ms, fact_purity: 125.42288ms, SA5012: 115.692744ms, typedness: 103.558214ms, SA4006: 222µs, isgenerated: 130.292µs, SA4027: 99.542µs, SA9004: 88.75µs 
INFO [runner] Issues before processing: 3, after processing: 1 
INFO [runner] Processors filtering stat (out/in): path_shortener: 1/1, cgo: 3/3, filename_unadjuster: 3/3, uniq_by_line: 1/1, nolint: 1/3, severity-rules: 1/1, fixer: 1/1, skip_files: 3/3, autogenerated_exclude: 3/3, exclude: 3/3, max_from_linter: 1/1, skip_dirs: 3/3, identifier_marker: 3/3, diff: 1/1, max_same_issues: 1/1, source_code: 1/1, path_prefixer: 1/1, sort_results: 1/1, path_prettifier: 3/3, exclude-rules: 3/3, max_per_file_from_linter: 1/1 
INFO [runner] processing took 519.497µs with stages: exclude-rules: 189.292µs, nolint: 94.251µs, autogenerated_exclude: 71.416µs, identifier_marker: 69.291µs, source_code: 52.833µs, path_prettifier: 31.583µs, skip_dirs: 4.959µs, uniq_by_line: 1.417µs, cgo: 1.041µs, max_same_issues: 749ns, path_shortener: 458ns, max_from_linter: 417ns, filename_unadjuster: 292ns, sort_results: 292ns, diff: 209ns, fixer: 208ns, max_per_file_from_linter: 208ns, skip_files: 167ns, severity-rules: 166ns, exclude: 166ns, path_prefixer: 82ns 
INFO [runner] linters took 1.05964275s with stages: goanalysis_metalinter: 1.059068917s 
main.go:11:14: SA1019: conn.PgConn().CheckConn is deprecated: CheckConn is deprecated in favor of Ping. CheckConn cannot detect all types of broken connections where the write would still appear to succeed. Prefer Ping unless on a high latency connection. (staticcheck)
	fmt.Println(conn.PgConn().CheckConn() == nil)
	           ^
INFO File cache stats: 1 entries of total size 213B 
INFO Memory: 19 samples, avg is 131.1MB, max is 309.6MB 
INFO Execution took 1.774828083s          

This is the correct behavior—staticcheck is flagging the line without the nolint directive.

But if we modify the file (which I assume affects the cache), we get the incorrect behavior. I added a newline after var conn *pgx.Conn to make a change that should be equivalent from an AST perspective. Then ran golangci-lint run -v again:

INFO [config_reader] Config search paths: [./ /Users/me/nolintrepro /Users/me /Users /] 
INFO [config_reader] Used config file .golangci.yml 
INFO [lintersdb] Active 2 linters: [nolintlint staticcheck] 
INFO [loader] Go packages loading at mode 575 (exports_file|files|imports|types_sizes|compiled_files|deps|name) took 531.612584ms 
INFO [runner/filename_unadjuster] Pre-built 0 adjustments in 198.542µs 
INFO [linters_context/goanalysis] analyzers took 8.812332ms with top 10 stages: buildir: 5.229666ms, SA5012: 532.211µs, fact_purity: 447.049µs, nilness: 416.708µs, fact_deprecated: 405.157µs, typedness: 365.661µs, SA4001: 78.208µs, SA1021: 77.834µs, SA6005: 70.25µs, SA4026: 65.584µs 
INFO [runner] Processors filtering stat (out/in): exclude: 1/1, diff: 1/1, source_code: 1/1, fixer: 1/1, cgo: 1/1, filename_unadjuster: 1/1, skip_dirs: 1/1, autogenerated_exclude: 1/1, uniq_by_line: 1/1, severity-rules: 1/1, skip_files: 1/1, exclude-rules: 1/1, nolint: 1/1, max_same_issues: 1/1, max_from_linter: 1/1, path_shortener: 1/1, path_prettifier: 1/1, identifier_marker: 1/1, max_per_file_from_linter: 1/1, path_prefixer: 1/1, sort_results: 1/1 
INFO [runner] processing took 224.418µs with stages: nolint: 57.251µs, autogenerated_exclude: 51.125µs, source_code: 27.209µs, path_prettifier: 25.876µs, identifier_marker: 25.125µs, exclude-rules: 25.042µs, skip_dirs: 7.542µs, uniq_by_line: 1.042µs, max_same_issues: 1.042µs, max_from_linter: 584ns, cgo: 541ns, path_shortener: 459ns, filename_unadjuster: 292ns, fixer: 208ns, skip_files: 208ns, exclude: 208ns, severity-rules: 207ns, sort_results: 166ns, max_per_file_from_linter: 125ns, diff: 125ns, path_prefixer: 41ns 
INFO [runner] linters took 88.876792ms with stages: goanalysis_metalinter: 88.6125ms 
main.go:13:48: directive `//nolint: staticcheck` is unused for linter "staticcheck" (nolintlint)
	fmt.Println(conn.PgConn().CheckConn() == nil) //nolint: staticcheck
	                                             ^
INFO File cache stats: 1 entries of total size 214B 
INFO Memory: 10 samples, avg is 30.4MB, max is 51.4MB 
INFO Execution took 819.347ms                 

Now it complains about the unused nolint directive, and staticcheck doesn't complain at all.

If you run golangci-lint cache clean, the correct behavior returns until the file is modified again.

@ldez
Copy link
Member

ldez commented Feb 21, 2024

Thank you for the repro.

As a workaround for staticcheck, I recommend using the following example instead of nolint directives:

issues:
  exclude-rules:
    - linters:
        - staticcheck
      text: "SA1019: conn.PgConn().CheckConn is deprecated: CheckConn is deprecated in favor of Ping"

Side note: a directive should not contain space after the ::

  • nolint: xxx not valid
  • nolint:xxx valid.

@stevenh
Copy link

stevenh commented Mar 7, 2024

Seeing this in combination with ireturn too, as mentioned the error goes away for a time if you clean the cache with golangci-lint cache clean.

pkg/mything.go:9:78: directive `//nolint:ireturn // Returns one of multiple types which implement MyThinger.` is unused for linter "ireturn" (nolintlint)
func Testfunc() MyThinger { //nolint:ireturn // Returns one of multiple types which implement MyThinger.

@ldez
Copy link
Member

ldez commented Mar 7, 2024

I think this issue is the same as #4423, I will check my fix #4424 on that. EDIT no it's not the same problem.

May be related to #3476


@stevenh If you are facing the same problem, the best way to contribute is to provide the following information:

golangci-lint --version
cat .golangci.yml
go version && go env
golangci-lint cache clean
golangci-lint run -v

And a code example or link to a public repository.

@ldez ldez added the area: nolint Related to nolint directive and nolintlint label Mar 21, 2024
pierreprinetti pushed a commit to shiftstack/cluster-api-provider-openstack that referenced this issue Apr 22, 2024
Hitting golangci/golangci-lint#3228 when
adding nolint.

So allow to ignore unused nolints.
bartoszmajsak added a commit to opendatahub-io/opendatahub-operator that referenced this issue Jun 26, 2024
Instead of using `//nolint:ireturn` the linter allows a use of `generic`
return types (on top of the listed defaults which now have been
explicitly set).

This approach also eliminates a re-occuring issue where `nolintlint`
reports aforementioned directive as not used by defined linter.

Related issue: golangci/golangci-lint#3228

On top of that we have `--fix` flag enabled for golangci-lint runner.
This results in removal of these comments as `nolintlint` tries to
autofix.

As a consequence the subsequent run of `make lint` yields errors for
previously disabled linters and the cycle continues :)

> [!IMPORTANT]
> This behaviour has also been observed for other linters.

As part of this the declarations of kustomize plugin constructor funcs
has been reworkted to return struct pointers instead.
bartoszmajsak added a commit to opendatahub-io/opendatahub-operator that referenced this issue Jun 26, 2024
Instead of using `//nolint:ireturn` the linter allows a use of `generic`
return types (on top of the listed defaults which now have been
explicitly set).

This approach also eliminates a re-occuring issue where `nolintlint`
reports aforementioned directive as not used by defined linter.

Related issue: golangci/golangci-lint#3228

On top of that we have `--fix` flag enabled for golangci-lint runner.
This results in removal of these comments as `nolintlint` tries to
autofix.

As a consequence the subsequent run of `make lint` yields errors for
previously disabled linters and the cycle continues :)

> [!IMPORTANT]
> This behaviour has also been observed for other linters.

As part of this the declarations of kustomize plugin constructor funcs
has been reworkted to return struct pointers instead.

\#stack-pr
bartoszmajsak added a commit to bartoszmajsak/opendatahub-operator that referenced this issue Jun 26, 2024
Instead of using `//nolint:ireturn` the linter allows a use of `generic`
return types (on top of the listed defaults which now have been
explicitly set).

This approach also eliminates a re-occuring issue where `nolintlint`
reports aforementioned directive as not used by defined linter.

Related issue: golangci/golangci-lint#3228

On top of that we have `--fix` flag enabled for golangci-lint runner.
This results in removal of these comments as `nolintlint` tries to
autofix.

As a consequence the subsequent run of `make lint` yields errors for
previously disabled linters and the cycle continues :)

> [!IMPORTANT]
> This behaviour has also been observed for other linters.

As part of this the declarations of kustomize plugin constructor funcs
has been reworkted to return struct pointers instead. The reason for
this is that in our feature branch we are relying on kustomize plugins
and want to iterate over a slice of resmap.Transfomers but in the
current form these functions cannot be used due to the fact that
structs returned have pointer receivers. From the current functionality
standpoint (on incubation branch) this change is harmless.
openshift-merge-bot bot pushed a commit to opendatahub-io/opendatahub-operator that referenced this issue Jun 26, 2024
Instead of using `//nolint:ireturn` the linter allows a use of `generic`
return types (on top of the listed defaults which now have been
explicitly set).

This approach also eliminates a re-occuring issue where `nolintlint`
reports aforementioned directive as not used by defined linter.

Related issue: golangci/golangci-lint#3228

On top of that we have `--fix` flag enabled for golangci-lint runner.
This results in removal of these comments as `nolintlint` tries to
autofix.

As a consequence the subsequent run of `make lint` yields errors for
previously disabled linters and the cycle continues :)

> [!IMPORTANT]
> This behaviour has also been observed for other linters.

As part of this the declarations of kustomize plugin constructor funcs
has been reworkted to return struct pointers instead. The reason for
this is that in our feature branch we are relying on kustomize plugins
and want to iterate over a slice of resmap.Transfomers but in the
current form these functions cannot be used due to the fact that
structs returned have pointer receivers. From the current functionality
standpoint (on incubation branch) this change is harmless.
VaishnaviHire pushed a commit to VaishnaviHire/opendatahub-operator that referenced this issue Jul 24, 2024
)

Instead of using `//nolint:ireturn` the linter allows a use of `generic`
return types (on top of the listed defaults which now have been
explicitly set).

This approach also eliminates a re-occuring issue where `nolintlint`
reports aforementioned directive as not used by defined linter.

Related issue: golangci/golangci-lint#3228

On top of that we have `--fix` flag enabled for golangci-lint runner.
This results in removal of these comments as `nolintlint` tries to
autofix.

As a consequence the subsequent run of `make lint` yields errors for
previously disabled linters and the cycle continues :)

> [!IMPORTANT]
> This behaviour has also been observed for other linters.

As part of this the declarations of kustomize plugin constructor funcs
has been reworkted to return struct pointers instead. The reason for
this is that in our feature branch we are relying on kustomize plugins
and want to iterate over a slice of resmap.Transfomers but in the
current form these functions cannot be used due to the fact that
structs returned have pointer receivers. From the current functionality
standpoint (on incubation branch) this change is harmless.

(cherry picked from commit 613d552)
VaishnaviHire pushed a commit to VaishnaviHire/opendatahub-operator that referenced this issue Jul 24, 2024
)

Instead of using `//nolint:ireturn` the linter allows a use of `generic`
return types (on top of the listed defaults which now have been
explicitly set).

This approach also eliminates a re-occuring issue where `nolintlint`
reports aforementioned directive as not used by defined linter.

Related issue: golangci/golangci-lint#3228

On top of that we have `--fix` flag enabled for golangci-lint runner.
This results in removal of these comments as `nolintlint` tries to
autofix.

As a consequence the subsequent run of `make lint` yields errors for
previously disabled linters and the cycle continues :)

> [!IMPORTANT]
> This behaviour has also been observed for other linters.

As part of this the declarations of kustomize plugin constructor funcs
has been reworkted to return struct pointers instead. The reason for
this is that in our feature branch we are relying on kustomize plugins
and want to iterate over a slice of resmap.Transfomers but in the
current form these functions cannot be used due to the fact that
structs returned have pointer receivers. From the current functionality
standpoint (on incubation branch) this change is harmless.

(cherry picked from commit 613d552)
@ZackaryWelch
Copy link

I've just started noticing this on 1.63.4

@magnetikonline
Copy link
Contributor

magnetikonline commented Jan 23, 2025

I've just started noticing this on 1.63.4

Likewise @ZackaryWelch - just implementing nolintlint and seeing this.

Was able to solve with golangci-lint cache clean.

@ldez
Copy link
Member

ldez commented Jan 23, 2025

If you are facing the same problem, the best way to contribute is to provide the following information:

golangci-lint --version
cat .golangci.yml
go version && go env
golangci-lint cache clean
golangci-lint run -v

And a code example or link to a public repository.

@bwplotka
Copy link

bwplotka commented Mar 12, 2025

We are hitting this in Prometheus too, with GH action golangci-lint: prometheus/prometheus#16203

It's flaky, 50% chances it brings false positive. Rerunning often helps.

We are disabling nolintlint fo now prometheus/prometheus#16204 as we see contributors trying to remove the nolint directive to resolve false positive only to get a failure again from revive 🙈

Example case

We are hitting this on GH actions for 1.64.6 golangci-lint: https://github.com/prometheus/prometheus/actions/runs/13794350636/job/38582148069

Pasting logs in case of the GH action logs will expire:

Checking for go.mod: go.mod
  Cache hit for: golangci-lint.cache-Linux-2879-2c745dd6972f078a7e744b34011c2d4048562193
  Received 32[15](https://github.com/prometheus/prometheus/actions/runs/13794350636/job/38582148069#step:5:16)192 of 3215192 (100.0%), 9.0 MBs/sec
  Cache Size: ~3 MB (3215192 B)
  /usr/bin/tar -xf /home/runner/work/_temp/de86874b-1a55-4cf6-b9cd-94a93a3d5f95/cache.tzst -P -C /home/runner/work/prometheus/prometheus --use-compress-program unzstd
  Cache restored successfully
  Restored cache for golangci-lint from key 'golangci-lint.cache-Linux-2879-2c745dd6972f078a7e744b34011c2d4048562193' in 1445ms
  Finding needed golangci-lint version...
  Installation mode: binary
  Installing golangci-lint binary v1.64.6...
  Downloading binary https://github.com/golangci/golangci-lint/releases/download/v1.64.6/golangci-lint-1.64.6-linux-amd64.tar.gz ...
  /usr/bin/tar xz --overwrite --warning=no-unknown-keyword --overwrite -C /home/runner -f /home/runner/work/_temp/d2246a6e-8b43-457f-a8f5-cd4b[16](https://github.com/prometheus/prometheus/actions/runs/13794350636/job/38582148069#step:5:17)aa8f71
  Installed golangci-lint into /home/runner/golangci-lint-1.64.6-linux-amd64/golangci-lint in 463ms
  Prepared env in 1911ms
run golangci-lint
  Running [/home/runner/golangci-lint-1.64.6-linux-amd64/golangci-lint config path] in [/home/runner/work/prometheus/prometheus] ...
  Running [/home/runner/golangci-lint-1.64.6-linux-amd64/golangci-lint config verify] in [/home/runner/work/prometheus/prometheus] ...
  Running [/home/runner/golangci-lint-1.64.6-linux-amd64/golangci-lint run  --verbose] in [/home/runner/work/prometheus/prometheus] ...
  Error: tsdb/ooo_head.go:75:1: directive `//nolint:revive` is unused for linter "revive" (nolintlint)
  //nolint:revive
  ^
  
  level=info msg="golangci-lint has version 1.64.6 built with go1.24.0 from d574f356 on 2025-03-02T23:27:45Z"
  level=info msg="[config_reader] Config search paths: [./ /home/runner/work/prometheus/prometheus /home/runner/work/prometheus /home/runner/work /home/runner /home /]"
  level=info msg="[config_reader] Used config file .golangci.yml"
  level=info msg="[goenv] Read go env for 3.628914ms: map[string]string{\"GOCACHE\":\"/home/runner/.cache/go-build\", \"GOROOT\":\"/opt/hostedtoolcache/go/1.24.1/x64\"}"
  level=info msg="[lintersdb] Active 25 linters: [depguard errcheck errorlint exptostd gocritic godot gofumpt goimports gosimple govet ineffassign loggercheck misspell nilnesserr nolintlint perfsprint predeclared revive sloglint staticcheck testifylint unconvert unused usestdlibvars whitespace]"
  level=info msg="[loader] Go packages loading at mode 8767 (deps|files|imports|name|types_sizes|compiled_files|exports_file) took 7.432398889s"
  level=info msg="[runner/filename_unadjuster] Pre-built 0 adjustments in 182.425377ms"
  level=info msg="[linters_context/goanalysis] analyzers took 1m46.207040965s with top 10 stages: buildir: 13.192511297s, goimports: 12.95784788s, gofumpt: 8.426084967s, the_only_name: 8.275105025s, S1038: 5.64723[17](https://github.com/prometheus/prometheus/actions/runs/13794350636/job/38582148069#step:5:18)73s, gocritic: 5.628474353s, unused: 5.034059602s, whitespace: 4.393366523s, unconvert: 3.816960858s, buildssa: 3.486442007s"
  level=info msg="[runner/skip_dirs] Skipped 24 issues from dir storage/remote/otlptranslator/prometheusremotewrite by pattern storage/remote/otlptranslator/prometheusremotewrite"
  level=info msg="[runner/skip_dirs] Skipped 17 issues from dir storage/remote/otlptranslator/prometheus by pattern storage/remote/otlptranslator/prometheus"
  level=info msg="[runner/skip_dirs] Skipped 6 issues from dir documentation/examples/custom-sd/adapter by pattern (^|/)examples($|/)"
  level=info msg="[runner/skip_dirs] Skipped 3 issues from dir documentation/examples/custom-sd/adapter-usage by pattern (^|/)examples($|/)"
  level=info msg="[runner/exclusion_rules] Skipped 20 issues by rules: [Text: \"(?i)stdmethods: method Seek.* should have signature Seek\", Linters: \"govet\"]"
  level=info msg="[runner/exclusion_rules] Skipped 569 issues by rules: [Text: \"(?i)exported (.+) should have comment( \\\\(or a comment on this block\\\\))? or be unexported\", Linters: \"revive\"]"
  level=info msg="[runner/exclusion_rules] Skipped 247 issues by rules: [Path: \"_test.go\", Linters: \"errcheck\"]"
  level=info msg="[runner/exclusion_rules] Skipped 2 issues by rules: [Path: \"tsdb/head_wal.go\", Linters: \"errorlint\"]"
  level=info msg="[runner/exclusion_rules] Skipped 52 issues by rules: [Text: \"(?i)fmt.Sprintf can be replaced with string concatenation\", Linters: \"perfsprint\"]"
  level=info msg="[runner/exclusion_rules] Skipped 356 issues by rules: [Text: \"(?i)Error return value of .((os\\\\.)?std(out|err)\\\\..*|.*Close|.*Flush|os\\\\.Remove(All)?|.*print(f|ln)?|os\\\\.(Un)?Setenv). is not checked\", Linters: \"errcheck\"]"
  level=info msg="[runner/exclusion_rules] Skipped 9 issues by rules: [Text: \"(?i)appendAssign\", Linters: \"gocritic\"]"
  level=info msg="[runner/exclusion_rules] Skipped 72 issues by rules: [Source: \"(?i)^// ===\", Linters: \"godot\"]"
  level=info msg="[runner] Issues before processing: 1977, after processing: 1"
  level=info msg="[runner] Processors filtering stat (in/out): invalid_issue: 1977/1977, exclusion_paths: 1977/1977, skip_files: 1977/1534, generated_file_filter: 1484/1414, diff: 1/1, uniq_by_line: 1/1, max_per_file_from_linter: 1/1, max_same_issues: 1/1, filename_unadjuster: 1977/1977, skip_dirs: 1534/1484, identifier_marker: 1414/1414, nolint_filter: 87/1, fixer: 1/1, max_from_linter: 1/1, source_code: 1/1, severity-rules: 1/1, cgo: 1977/1977, path_relativity: 1977/1977, exclusion_rules: 1414/87, path_shortener: 1/1, sort_results: 1/1, path_absoluter: 1977/1977, path_prettifier: 1/1"
  level=info msg="[runner] processing took 54.838729ms with stages: nolint_filter: 35.67097ms, exclusion_rules: 9.666635ms, generated_file_filter: 5.331944ms, path_relativity: 1.56576ms, skip_files: 1.207978ms, skip_dirs: 804.571µs, invalid_issue: 160.491µs, cgo: 127.4µs, path_absoluter: 113.413µs, identifier_marker: 84.528µs, filename_unadjuster: 67.266µs, source_code: [18](https://github.com/prometheus/prometheus/actions/runs/13794350636/job/38582148069#step:5:19).835µs, path_shortener: 12.815µs, uniq_by_line: 1.714µs, sort_results: 1.353µs, max_same_issues: 951ns, exclusion_paths: 641ns, fixer: 351ns, diff: 340ns, max_per_file_from_linter: 320ns, path_prettifier: 211ns, max_from_linter: 142ns, severity-rules: 100ns"
  level=info msg="[runner] linters took 6.95610147s with stages: goanalysis_metalinter: 6.90113899s"
  level=info msg="File cache stats: 2 entries of total size 73.4KiB"
  level=info msg="Memory: 136 samples, avg is 418.2MB, max is 1467.3MB"
  level=info msg="Execution took 14.574[19](https://github.com/prometheus/prometheus/actions/runs/13794350636/job/38582148069#step:5:20)4377s"
  
  Error: issues found
  Ran golangci-lint in 14696ms

This failed for this code:

// ToEncodedChunks returns chunks with the samples in the OOOChunk.
//
//nolint:revive
func (o *OOOChunk) ToEncodedChunks(mint, maxt int64) (chks []memChunk, err error) {

Removing nolint entry fails with that unexported-return revive error as expected.

It happens in various places, likely when someone is touching the same directory in a PR 🤔

bwplotka added a commit to prometheus/prometheus that referenced this issue Mar 12, 2025
@bernot-dev
Copy link

I also ran into this issue, and thought it might be a cache issue, but clearing the golangci-lint cache does not seem to reliably solve the issue.

Wild speculation, but this feels like a race condition to me. Maybe the order the linters run is important for nolintlint?

machine424 pushed a commit to machine424/prometheus that referenced this issue Mar 26, 2025
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
area: nolint Related to nolint directive and nolintlint bug Something isn't working
Projects
None yet
Development

No branches or pull requests