Skip to content

upsun/vinego

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 

History

18 Commits
 
 
 
 
 
 
 
 

Repository files navigation

Vinego

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.

Analyzers

  • 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 the else 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 to time.Duration.

  • capturederr

    Enabled with enable_capturederr: true in .vinego.yaml.

    The staticcheck linter SA4006 check which makes sure we properly consume error variables ignores anything that happens with captured variables. Therefore if you accidentally capture err 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 do err := otherthing(), which would make this error disappear and the SA4006 one appear in its place, as intended.)

Usage

All-in-one development container

  1. 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 with vinego included as a module-type plugin
    • gci
    • 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 the Dockerfile for details (it's a straightforward Go .so build).

  2. 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
    
  3. Run the linters with docker run --rm --volume $PWD:/mnt --workdir /mnt vinego /bin/golangci-lint run --verbose. You should see vinego listed in the output.

Building your own golangci-lint

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.

  1. Install any recent version of golangci-lint

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

  3. Run golangci-lint custom -v

    This will produce a new golangci-lint

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