Blacksmith now supports transparent caching as explained in https://www.blacksmith.sh/blog/cache. To this effect it is recommended to switch to the upstream maintained versions of this action as you will continue to leverage our much faster caching without any code changes. This action will no longer receive dependency or security updates.
It's the official GitHub action for golangci-lint from its authors.
The action runs golangci-lint and reports issues from linters.
golangci-lint is a free and open-source project built by volunteers.
If you value it, consider supporting us; we appreciate it! ❤️
We recommend running this action in a job separate from other jobs (go test, etc.)
because different jobs run in parallel.
Add .github/workflows/golangci-lint.yml with the following contents:
Simple Example
name: golangci-lint
on:
push:
branches:
- main
- master
pull_request:
permissions:
contents: read
# Optional: allow read access to pull request. Use with `only-new-issues` option.
# pull-requests: read
jobs:
golangci:
name: lint
runs-on: ubuntu-latest
steps:
- uses: actions/checkout@v4
- uses: actions/setup-go@v5
with:
go-version: stable
- name: golangci-lint
uses: golangci/golangci-lint-action@v8
with:
version: v2.1Multiple OS Example
name: golangci-lint
on:
push:
branches:
- main
- master
pull_request:
permissions:
contents: read
# Optional: allow read access to pull request. Use with `only-new-issues` option.
# pull-requests: read
jobs:
golangci:
strategy:
matrix:
go: [stable]
os: [ubuntu-latest, macos-latest, windows-latest]
name: lint
runs-on: ${{ matrix.os }}
steps:
- uses: actions/checkout@v4
- uses: actions/setup-go@v5
with:
go-version: ${{ matrix.go }}
- name: golangci-lint
uses: golangci/golangci-lint-action@v8
with:
version: v2.1You will also likely need to add the following .gitattributes file to ensure that line endings for Windows builds are properly formatted:
*.go text eol=lf
Go Workspace Example
name: golangci-lint
on:
pull_request:
push:
branches:
- main
- master
env:
GO_VERSION: stable
GOLANGCI_LINT_VERSION: v2.1
jobs:
detect-modules:
runs-on: ubuntu-latest
outputs:
modules: ${{ steps.set-modules.outputs.modules }}
steps:
- uses: actions/checkout@v4
- uses: actions/setup-go@v5
with:
go-version: ${{ env.GO_VERSION }}
- id: set-modules
run: echo "modules=$(go list -m -json | jq -s '.' | jq -c '[.[].Dir]')" >> $GITHUB_OUTPUT
golangci-lint:
needs: detect-modules
runs-on: ubuntu-latest
strategy:
matrix:
modules: ${{ fromJSON(needs.detect-modules.outputs.modules) }}
steps:
- uses: actions/checkout@v4
- uses: actions/setup-go@v5
with:
go-version: ${{ env.GO_VERSION }}
- name: golangci-lint ${{ matrix.modules }}
uses: golangci/golangci-lint-action@v8
with:
version: ${{ env.GOLANGCI_LINT_VERSION }}
working-directory: ${{ matrix.modules }}Go Workspace Example (Multiple OS)
# ./.github/workflows/golangci-lint.yml
name: golangci-lint (multi OS)
on:
pull_request:
push:
branches:
- main
- master
jobs:
golangci-lint:
strategy:
matrix:
go-version: [ stable, oldstable ]
os: [ubuntu-latest, macos-latest, windows-latest]
uses: ./.github/workflows/.golangci-lint-reusable.yml
with:
os: ${{ matrix.os }}
go-version: ${{ matrix.go-version }}
golangci-lint-version: v2.1# ./.github/workflows/.golangci-lint-reusable.yml
name: golangci-lint-reusable
on:
workflow_call:
inputs:
os:
description: 'OS'
required: true
type: string
go-version:
description: 'Go version'
required: true
type: string
default: stable
golangci-lint-version:
description: 'Golangci-lint version'
type: string
default: 'v2.1'
jobs:
detect-modules:
runs-on: ${{ inputs.os }}
outputs:
modules: ${{ steps.set-modules.outputs.modules }}
steps:
- uses: actions/checkout@v4
- uses: actions/setup-go@v5
with:
go-version: ${{ inputs.go-version }}
- id: set-modules
shell: bash # require for Windows to be able to use $GITHUB_OUTPUT https://github.com/actions/runner/issues/2224
run: echo "modules=$(go list -m -json | jq -s '.' | jq -c '[.[].Dir]')" >> $GITHUB_OUTPUT
golangci-lint:
needs: detect-modules
runs-on: ${{ inputs.os }}
strategy:
matrix:
modules: ${{ fromJSON(needs.detect-modules.outputs.modules) }}
steps:
- uses: actions/checkout@v4
- uses: actions/setup-go@v5
with:
go-version: ${{ inputs.go-version }}
- name: golangci-lint ${{ matrix.modules }}
uses: golangci/golangci-lint-action@v8
with:
version: ${{ inputs.golangci-lint-version }}
working-directory: ${{ matrix.modules }}You will also likely need to add the following .gitattributes file to ensure that line endings for Windows builds are properly formatted:
*.go text eol=lf
v8.0.0works withgolangci-lintversion >=v2.1.0v7.0.0supports golangci-lint v2 only.v6.0.0+removesannotationsoption, removes the default output format (github-actions).v5.0.0+removesskip-pkg-cacheandskip-build-cachebecause the cache related to Go itself is already handled byactions/setup-go.v4.0.0+requires an explicitactions/setup-goinstallation step before using this action:uses: actions/setup-go@v5. Theskip-go-installationoption has been removed.v2.0.0+works withgolangci-lintversion >=v1.28.3v1.2.2is deprecated because we forgot to change the minimum version ofgolangci-linttov1.28.3(issue)v1.2.1works withgolangci-lintversion >=v1.14.0(issue)
(optional)
The version of golangci-lint to use.
When install-mode is:
binary(default): the value can be v2.3 or v2.3.4 orlatestto use the latest version.goinstall: the value can be v2.3.4,latest, or the hash of a commit.none: the value is ignored.
Example
uses: golangci/golangci-lint-action@v8
with:
version: v2.1
# ...(optional)
The mode to install golangci-lint: it can be binary, goinstall, or none.
The default value is binary.
Example
uses: golangci/golangci-lint-action@v8
with:
install-mode: "goinstall"
# ...(optional)
When using only-new-issues option, the GitHub API is used, so a token is required.
By default, it uses the github.token from the action.
Example
uses: golangci/golangci-lint-action@v8
with:
github-token: xxx
# ...(optional)
This option is true by default.
If the GitHub Action detects a configuration file, the validation will be performed unless this option is set to false.
If there is no configuration file, the validation is skipped.
The JSON Schema used to validate the configuration depends on the version of golangci-lint you are using.
Example
uses: golangci/golangci-lint-action@v8
with:
verify: false
# ...(optional)
Show only new issues.
The default value is false.
pull_requestandpull_request_target: the action gets the diff of the PR content from the GitHub API and uses it with--new-from-patch.push: the action gets the diff of the push content (difference between commits before and after the push) from the GitHub API and uses it with--new-from-patch.merge_group: the action gets the diff by using--new-from-revoption (relies on git). You should add the optionfetch-depth: 0toactions/checkoutstep.
Example
uses: golangci/golangci-lint-action@v8
with:
only-new-issues: true
# ...(optional)
Working directory, useful for monorepos.
Example
uses: golangci/golangci-lint-action@v8
with:
working-directory: somedir
# ...(optional)
golangci-lint command line arguments.
Note
By default, the .golangci.yml file should be at the root of the repository.
The location of the configuration file can be changed by using --config=.
Important
Adding a = between the flag name and its value is important because the action parses the arguments on spaces.
Example
uses: golangci/golangci-lint-action@v8
with:
# In some rare cases,
# you could have to use `${{ github.workspace }}` as base directory to reference your configuration file.
args: --config=/my/path/.golangci.yml --issues-exit-code=0
# ...(optional)
Force the usage of the embedded problem matchers.
By default, the problem matcher of Go (actions/setup-go) already handles the default golangci-lint output (text).
Works only with text format (the golangci-lint default).
https://golangci-lint.run/usage/configuration/#output-configuration
The default value is false.
Example
uses: golangci/golangci-lint-action@v8
with:
problem-matchers: true
# ...(optional)
If set to true, then all caching functionality will be completely disabled,
takes precedence over all other caching options.
The default value is false.
Example
uses: golangci/golangci-lint-action@v8
with:
skip-cache: true
# ...(optional)
If set to true, caches will not be saved, but they may still be restored, requiring skip-cache: false.
The default value is false.
Example
uses: golangci/golangci-lint-action@v8
with:
skip-save-cache: true
# ...(optional)
Periodically invalidate the cache every cache-invalidation-interval days to ensure that outdated data is removed and fresh data is loaded.
The default value is 7.
If the number is <= 0, the cache will always be invalidated (Not recommended).
Example
uses: golangci/golangci-lint-action@v8
with:
cache-invalidation-interval: 15
# ...Currently, GitHub parses the action's output and creates annotations.
The restrictions of annotations are the following:
- Currently, they don't support Markdown formatting (see the feature request)
- They aren't shown in the list of comments. If you would like to have comments - please, up-vote the issue.
- The number of annotations is limited.
Permissions required:
permissions:
# Required: allow read access to the content for analysis.
contents: read
# Optional: allow read access to pull request. Use with `only-new-issues` option.
pull-requests: readFor annotations to work, use the default format output (text) and either use actions/setup-go in the job or enable the internal problem matchers.
The action was implemented with performance in mind:
- We cache data from golangci-lint analysis between builds by using @actions/cache.
- We don't use Docker because image pulling is slow.
- We do as much as we can in parallel, e.g. we download cache, and golangci-lint binary in parallel.
For example, in a repository of golangci-lint running this action without the cache takes 50s, but with cache takes 14s:
- in parallel:
- 4s to restore 50 MB of cache
- 1s to find and install
golangci-lint
- 1s to run
golangci-lint(it takes 35s without cache)
We use JavaScript-based action. We don't use Docker-based action because:
- Docker pulling is slow currently
- it's easier to use caching from @actions/cache
We support different platforms, such as ubuntu, macos, and windows with x32 and x64 archs.
Inside our action, we perform 3 steps:
- Setup environment running in parallel:
- restore cache of previous analyses
- fetch action config and find the latest
golangci-lintpatch version for needed version (users of this action can specify only minor version ofgolangci-lint). After that install golangci-lint using @actions/tool-cache
- Run
golangci-lintwith specified by userargs - Save cache for later builds
- We save and restore the following directory:
~/.cache/golangci-lint. - The primary caching key looks like
golangci-lint.cache-{runner_os}-{working_directory}-{interval_number}-{go.mod_hash}. Interval number ensures that we periodically invalidate our cache (every 7 days).go.modhash ensures that we invalidate the cache early - as soon as dependencies have changed. - We use restore keys:
golangci-lint.cache-{runner_os}-{working_directory}-{interval_number}-. GitHub matches keys by prefix if we have no exact match for the primary cache.
This scheme is basic and needs improvements. Pull requests and ideas are welcome.

