Vinego is a new set of linters built on the Go analysis.Analyzer
framework. Many of the linters focus on increasing strictness around variable initialization and type safety. They fall somewhere in-between "drop these into your codebase with no changes" and "rewrite all your code to conform to weird new language-extra conventions" in terms of invasiveness.
Each linter is an analysis.Analyzer
and they're aggregated as a golangci-lint
plugin. Individual non-opt-in linters can be enabled via the golangci-lint
settings. The analyzers should work with any analysis.Analyzer
framework though.
-
allfields
Confirms that all required fields are explicitly initialized in struct literals.
Add a comment before a struct type like:
// check:allfields type MyStruct { X int Y string Z bool `optional:""` }
Then, if you do (for instance):
m := MyStruct{X: 4}
you'll get an error that
Y
is missing. -
varinit
Enabled with
enable_varinit: true
in.vinego.yaml
.Checks that all variables have been explicitly initialized (with a value) before usage.
For example:
var i int if something { i = 4 } else { doOtherThings() } myFunc(i)
would produce an error saying that
i
hasn't been initialized in theelse
branch.Taking the address of a variable is considered initialization (ex: in doing
json.Unmarshal(bytes, &config)
config would be marked as initialized). -
explicitcast
Enabled with
enable_explicitcast: true
in.vinego.yaml
.Checks that primitive literals are never implicitly casted (during assignments, function calls, and returns).
For example:
var x time.Duration x = 24
would produce an error saying that
24
is being implicitly cast totime.Duration
. -
capturederr
Enabled with
enable_capturederr: true
in.vinego.yaml
.The
staticcheck
linterSA4006
check which makes sure we properly consume error variables ignores anything that happens with captured variables. Therefore if you accidentally captureerr
from an outer function, assign it a value, then never check it,SA4006
won't help you. Go's behavior using variable reuse/reinitialization with=
and:=
makes it easy to transplant code and accidentally reuse an existing variable, which makes it easy to accidentally capture external variables in closures.capturederror
will give you an error when an error variable is captured by a closure (specifically error variables). As with the other linters here, capturing an error variable isn't necessarily incorrect, but it's generally unintended and when done unintentionally can lead to hard to track incorrect failure behavior.Example usage:
err := something() if err != nil { return err } wrapper(func() error { err = otherthing() }) ...
would produce an error saying that we're assgning to the captured variable
err
. (The workaround is to doerr := otherthing()
, which would make this error disappear and theSA4006
one appear in its place, as intended.)
-
We provide a pre-made Docker batteries-included image for CI and development environments:
ghcr.io/upsun/vinego:latest
It includes
- Go
golangci-lint
built withvinego
included as amodule
-type plugingci
goimports
dlv
staticcheck
You can build the Docker container yourself with
docker build --tag vinego src
at the root of this repo.Alternatively, you can build just the plugin
.so
- see theDockerfile
for details (it's a straightforward Go.so
build). -
Add custom linter plugin to your project's
.golangci.yaml
file:linters: enable: - vinego linters-settings: custom: vinego: type: "module" settings: enable_varinit: true enable_explicitcast: true enable_capturederr: true
-
Run the linters with
docker run --rm --volume $PWD:/mnt --workdir /mnt vinego /bin/golangci-lint run --verbose
. You should seevinego
listed in the output.
If you need an image with different tools or want to use the linter in some other situation, we recommend using golangci-lint's module system to bootstrap a new golangci-lint with the vinego linters included.
-
Install any recent version of golangci-lint
-
Create
.custom-gcl.yml
with:version: v1.64.6 plugins: - module: 'github.com/upsun/vinego/src' import: 'github.com/upsun/vinego/src' version: latest
The top-level version is the version of golangci-lint that the process will bootstrap - it doesn't need to be the same version as the golangci-lint you installed in (1.).
-
Run
golangci-lint custom -v
This will produce a new
golangci-lint
-
Use the new
golangci-lint
with this.golangci.yml
:linters-settings: custom: vinego: type: "module" settings: enable_varinit: true enable_explicitcast: true enable_capturederr: true linters: enable: - vinego