From 55cd9f78a10b8092c3a7941a88d9fc31051469b4 Mon Sep 17 00:00:00 2001
From: Isaev Denis <idenx@yandex.com>
Date: Sat, 9 May 2020 15:30:54 +0300
Subject: [PATCH] dev: generate assets/github-action-config.json (#1073)

It will be used by GitHub action `golangci-lint-action`.

Relates: golangci/golangci-lint-action#11
---
 .github/workflows/pr.yml                 |   3 +
 Makefile                                 |  11 +-
 README.md                                |   4 +-
 README.tmpl.md                           |   4 +-
 {docs => assets}/demo.svg                |   0
 assets/github-action-config.json         |  89 +++++++++
 {docs => assets}/go.png                  | Bin
 go.sum                                   |   1 +
 scripts/gen_github_action_config/go.mod  |  10 +
 scripts/gen_github_action_config/go.sum  |  22 +++
 scripts/gen_github_action_config/main.go | 231 +++++++++++++++++++++++
 11 files changed, 367 insertions(+), 8 deletions(-)
 rename {docs => assets}/demo.svg (100%)
 create mode 100644 assets/github-action-config.json
 rename {docs => assets}/go.png (100%)
 create mode 100644 scripts/gen_github_action_config/go.mod
 create mode 100644 scripts/gen_github_action_config/go.sum
 create mode 100644 scripts/gen_github_action_config/main.go

diff --git a/.github/workflows/pr.yml b/.github/workflows/pr.yml
index 3ea50de2..b1304c5e 100644
--- a/.github/workflows/pr.yml
+++ b/.github/workflows/pr.yml
@@ -64,6 +64,9 @@ jobs:
   check_generated:
     needs: golangci-lint # run after golangci-lint action to not produce duplicated errors
     runs-on: ubuntu-latest
+    env:
+      # needed for github-action-config.json generation
+      GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
     steps:
       - uses: actions/checkout@v2
       - name: Unshallow
diff --git a/Makefile b/Makefile
index f628c89c..41221377 100644
--- a/Makefile
+++ b/Makefile
@@ -42,14 +42,14 @@ test_linters:
 
 # Maintenance
 
-generate: README.md docs/demo.svg install.sh
+generate: README.md assets/demo.svg install.sh assets/github-action-config.json
 .PHONY: generate
 
 fast_generate: README.md
 .PHONY: fast_generate
 
 maintainer-clean: clean
-	rm -rf docs/demo.svg README.md install.sh
+	rm -rf assets/demo.svg README.md install.sh
 .PHONY: maintainer-clean
 
 check_generated:
@@ -92,8 +92,8 @@ tools/svg-term: tools/package.json tools/package-lock.json
 tools/Dracula.itermcolors:
 	curl -fL -o $@ https://raw.githubusercontent.com/dracula/iterm/master/Dracula.itermcolors
 
-docs/demo.svg: tools/svg-term tools/Dracula.itermcolors
-	./tools/svg-term --cast=183662 --out docs/demo.svg --window --width 110 --height 30 --from 2000 --to 20000 --profile ./tools/Dracula.itermcolors --term iterm2
+assets/demo.svg: tools/svg-term tools/Dracula.itermcolors
+	./tools/svg-term --cast=183662 --out assets/demo.svg --window --width 110 --height 30 --from 2000 --to 20000 --profile ./tools/Dracula.itermcolors --term iterm2
 
 install.sh: .goreleaser.yml tools/godownloader
 	./tools/godownloader .goreleaser.yml | sed '/DO NOT EDIT/s/ on [0-9TZ:-]*//' > $@
@@ -101,6 +101,9 @@ install.sh: .goreleaser.yml tools/godownloader
 README.md: FORCE golangci-lint
 	go run ./scripts/gen_readme/main.go
 
+assets/github-action-config.json: FORCE golangci-lint
+	go run ./scripts/gen_github_action_config/main.go $@
+
 go.mod: FORCE
 	go mod tidy
 	go mod verify
diff --git a/README.md b/README.md
index bcb862af..d8762364 100644
--- a/README.md
+++ b/README.md
@@ -14,7 +14,7 @@ Follow the news and releases on our [twitter](https://twitter.com/golangci) and
 
 Sponsored by [GolangCI.com](https://golangci.com): SaaS service for running linters on GitHub pull requests. Free for Open Source.
 
-<a href="https://golangci.com/"><img src="docs/go.png" width="250px"></a>
+<a href="https://golangci.com/"><img src="assets/go.png" width="250px"></a>
 
 - [GolangCI-Lint](#golangci-lint)
   - [Demo](#demo)
@@ -60,7 +60,7 @@ Sponsored by [GolangCI.com](https://golangci.com): SaaS service for running lint
 ## Demo
 
 <p align="center">
-  <img src="./docs/demo.svg" width="100%">
+  <img src="./assets/demo.svg" width="100%">
 </p>
 
 Short 1.5 min video demo of analyzing [beego](https://github.com/astaxie/beego).
diff --git a/README.tmpl.md b/README.tmpl.md
index 0efccd6f..b6f602be 100644
--- a/README.tmpl.md
+++ b/README.tmpl.md
@@ -14,7 +14,7 @@ Follow the news and releases on our [twitter](https://twitter.com/golangci) and
 
 Sponsored by [GolangCI.com](https://golangci.com): SaaS service for running linters on GitHub pull requests. Free for Open Source.
 
-<a href="https://golangci.com/"><img src="docs/go.png" width="250px"></a>
+<a href="https://golangci.com/"><img src="assets/go.png" width="250px"></a>
 
 - [GolangCI-Lint](#golangci-lint)
   - [Demo](#demo)
@@ -60,7 +60,7 @@ Sponsored by [GolangCI.com](https://golangci.com): SaaS service for running lint
 ## Demo
 
 <p align="center">
-  <img src="./docs/demo.svg" width="100%">
+  <img src="./assets/demo.svg" width="100%">
 </p>
 
 Short 1.5 min video demo of analyzing [beego](https://github.com/astaxie/beego).
diff --git a/docs/demo.svg b/assets/demo.svg
similarity index 100%
rename from docs/demo.svg
rename to assets/demo.svg
diff --git a/assets/github-action-config.json b/assets/github-action-config.json
new file mode 100644
index 00000000..9b1d08c0
--- /dev/null
+++ b/assets/github-action-config.json
@@ -0,0 +1,89 @@
+{
+  "MinorVersionToConfig": {
+    "v1.10": {
+      "Error": "golangci-lint version 'v1.10' isn't supported: we support only v1.14.0 and later versions"
+    },
+    "v1.11": {
+      "Error": "golangci-lint version 'v1.11' isn't supported: we support only v1.14.0 and later versions"
+    },
+    "v1.12": {
+      "Error": "golangci-lint version 'v1.12' isn't supported: we support only v1.14.0 and later versions"
+    },
+    "v1.13": {
+      "Error": "golangci-lint version 'v1.13' isn't supported: we support only v1.14.0 and later versions"
+    },
+    "v1.14": {
+      "TargetVersion": "v1.14.0",
+      "AssetURL": "https://github.com/golangci/golangci-lint/releases/download/v1.14.0/golangci-lint-1.14.0-linux-amd64.tar.gz"
+    },
+    "v1.15": {
+      "TargetVersion": "v1.15.0",
+      "AssetURL": "https://github.com/golangci/golangci-lint/releases/download/v1.15.0/golangci-lint-1.15.0-linux-amd64.tar.gz"
+    },
+    "v1.16": {
+      "TargetVersion": "v1.16.0",
+      "AssetURL": "https://github.com/golangci/golangci-lint/releases/download/v1.16.0/golangci-lint-1.16.0-linux-amd64.tar.gz"
+    },
+    "v1.17": {
+      "TargetVersion": "v1.17.1",
+      "AssetURL": "https://github.com/golangci/golangci-lint/releases/download/v1.17.1/golangci-lint-1.17.1-linux-amd64.tar.gz"
+    },
+    "v1.18": {
+      "TargetVersion": "v1.18.0",
+      "AssetURL": "https://github.com/golangci/golangci-lint/releases/download/v1.18.0/golangci-lint-1.18.0-linux-amd64.tar.gz"
+    },
+    "v1.19": {
+      "TargetVersion": "v1.19.1",
+      "AssetURL": "https://github.com/golangci/golangci-lint/releases/download/v1.19.1/golangci-lint-1.19.1-linux-amd64.tar.gz"
+    },
+    "v1.20": {
+      "TargetVersion": "v1.20.1",
+      "AssetURL": "https://github.com/golangci/golangci-lint/releases/download/v1.20.1/golangci-lint-1.20.1-linux-amd64.tar.gz"
+    },
+    "v1.21": {
+      "TargetVersion": "v1.21.0",
+      "AssetURL": "https://github.com/golangci/golangci-lint/releases/download/v1.21.0/golangci-lint-1.21.0-linux-amd64.tar.gz"
+    },
+    "v1.22": {
+      "TargetVersion": "v1.22.2",
+      "AssetURL": "https://github.com/golangci/golangci-lint/releases/download/v1.22.2/golangci-lint-1.22.2-linux-amd64.tar.gz"
+    },
+    "v1.23": {
+      "TargetVersion": "v1.23.8",
+      "AssetURL": "https://github.com/golangci/golangci-lint/releases/download/v1.23.8/golangci-lint-1.23.8-linux-amd64.tar.gz"
+    },
+    "v1.24": {
+      "TargetVersion": "v1.24.0",
+      "AssetURL": "https://github.com/golangci/golangci-lint/releases/download/v1.24.0/golangci-lint-1.24.0-linux-amd64.tar.gz"
+    },
+    "v1.25": {
+      "TargetVersion": "v1.25.1",
+      "AssetURL": "https://github.com/golangci/golangci-lint/releases/download/v1.25.1/golangci-lint-1.25.1-linux-amd64.tar.gz"
+    },
+    "v1.26": {
+      "TargetVersion": "v1.26.0",
+      "AssetURL": "https://github.com/golangci/golangci-lint/releases/download/v1.26.0/golangci-lint-1.26.0-linux-amd64.tar.gz"
+    },
+    "v1.3": {
+      "Error": "golangci-lint version 'v1.3' isn't supported: we support only v1.14.0 and later versions"
+    },
+    "v1.4": {
+      "Error": "golangci-lint version 'v1.4' isn't supported: we support only v1.14.0 and later versions"
+    },
+    "v1.5": {
+      "Error": "golangci-lint version 'v1.5' isn't supported: we support only v1.14.0 and later versions"
+    },
+    "v1.6": {
+      "Error": "golangci-lint version 'v1.6' isn't supported: we support only v1.14.0 and later versions"
+    },
+    "v1.7": {
+      "Error": "golangci-lint version 'v1.7' isn't supported: we support only v1.14.0 and later versions"
+    },
+    "v1.8": {
+      "Error": "golangci-lint version 'v1.8' isn't supported: we support only v1.14.0 and later versions"
+    },
+    "v1.9": {
+      "Error": "golangci-lint version 'v1.9' isn't supported: we support only v1.14.0 and later versions"
+    }
+  }
+}
diff --git a/docs/go.png b/assets/go.png
similarity index 100%
rename from docs/go.png
rename to assets/go.png
diff --git a/go.sum b/go.sum
index 12ca6161..268765a5 100644
--- a/go.sum
+++ b/go.sum
@@ -325,6 +325,7 @@ golang.org/x/net v0.0.0-20190522155817-f3200d17e092/go.mod h1:HSz+uSET+XFnRR8LxR
 golang.org/x/net v0.0.0-20190620200207-3b0461eec859/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s=
 golang.org/x/net v0.0.0-20200226121028-0de0cce0169b h1:0mm1VjtFUOIlE1SbDlwjYaDxZVDP2S5ou6y0gSgXHu8=
 golang.org/x/net v0.0.0-20200226121028-0de0cce0169b/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s=
+golang.org/x/oauth2 v0.0.0-20180821212333-d2e6202438be h1:vEDujvNQGv4jgYKudGeI/+DAX4Jffq6hpD55MmoEvKs=
 golang.org/x/oauth2 v0.0.0-20180821212333-d2e6202438be/go.mod h1:N/0e6XlmueqKjAGxoOufVs8QHGRruUQn6yWY3a++T0U=
 golang.org/x/sync v0.0.0-20180314180146-1d60e4601c6f/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
 golang.org/x/sync v0.0.0-20181108010431-42b317875d0f/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
diff --git a/scripts/gen_github_action_config/go.mod b/scripts/gen_github_action_config/go.mod
new file mode 100644
index 00000000..dd8448b2
--- /dev/null
+++ b/scripts/gen_github_action_config/go.mod
@@ -0,0 +1,10 @@
+module github.com/golangci/golangci-lint/scripts/gen_github_action_config
+
+go 1.13
+
+require (
+	github.com/shurcooL/githubv4 v0.0.0-20200414012201-bbc966b061dd
+	github.com/shurcooL/graphql v0.0.0-20181231061246-d48a9a75455f // indirect
+	golang.org/x/net v0.0.0-20200506145744-7e3656a0809f // indirect
+	golang.org/x/oauth2 v0.0.0-20200107190931-bf48bf16ab8d
+)
diff --git a/scripts/gen_github_action_config/go.sum b/scripts/gen_github_action_config/go.sum
new file mode 100644
index 00000000..145c6167
--- /dev/null
+++ b/scripts/gen_github_action_config/go.sum
@@ -0,0 +1,22 @@
+cloud.google.com/go v0.34.0/go.mod h1:aQUYkXzVsufM+DwF1aE+0xfcU+56JwCaLick0ClmMTw=
+github.com/golang/protobuf v1.2.0 h1:P3YflyNX/ehuJFLhxviNdFxQPkGK5cDcApsge1SqnvM=
+github.com/golang/protobuf v1.2.0/go.mod h1:6lQm79b+lXiMfvg/cZm0SGofjICqVBUtrP5yJMmIC1U=
+github.com/shurcooL/githubv4 v0.0.0-20200414012201-bbc966b061dd h1:EwtC+kDj8s9OKiaStPZtTv3neldOyr98AXIxvmn3Gss=
+github.com/shurcooL/githubv4 v0.0.0-20200414012201-bbc966b061dd/go.mod h1:hAF0iLZy4td2EX+/8Tw+4nodhlMrwN3HupfaXj3zkGo=
+github.com/shurcooL/graphql v0.0.0-20181231061246-d48a9a75455f h1:tygelZueB1EtXkPI6mQ4o9DQ0+FKW41hTbunoXZCTqk=
+github.com/shurcooL/graphql v0.0.0-20181231061246-d48a9a75455f/go.mod h1:AuYgA5Kyo4c7HfUmvRGs/6rGlMMV/6B1bVnB9JxJEEg=
+golang.org/x/crypto v0.0.0-20190308221718-c2843e01d9a2/go.mod h1:djNgcEr1/C05ACkg1iLfiJU5Ep61QUkGW8qpdssI0+w=
+golang.org/x/net v0.0.0-20180724234803-3673e40ba225/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4=
+golang.org/x/net v0.0.0-20190108225652-1e06a53dbb7e h1:bRhVy7zSSasaqNksaRZiA5EEI+Ei4I1nO5Jh72wfHlg=
+golang.org/x/net v0.0.0-20190108225652-1e06a53dbb7e/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4=
+golang.org/x/net v0.0.0-20200506145744-7e3656a0809f h1:QBjCr1Fz5kw158VqdE9JfI9cJnl/ymnJWAdMuinqL7Y=
+golang.org/x/net v0.0.0-20200506145744-7e3656a0809f/go.mod h1:qpuaurCH72eLCgpAm/N6yyVIVM9cpaDIP3A8BGJEC5A=
+golang.org/x/oauth2 v0.0.0-20200107190931-bf48bf16ab8d h1:TzXSXBo42m9gQenoE3b9BGiEpg5IG2JkU5FkPIawgtw=
+golang.org/x/oauth2 v0.0.0-20200107190931-bf48bf16ab8d/go.mod h1:gOpvHmFTYa4IltrdGE7lF6nIHvwfUNPOp7c8zoXwtLw=
+golang.org/x/sync v0.0.0-20181221193216-37e7f081c4d4 h1:YUO/7uOKsKeq9UokNS62b8FYywz3ker1l1vDZRCRefw=
+golang.org/x/sync v0.0.0-20181221193216-37e7f081c4d4/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
+golang.org/x/sys v0.0.0-20190215142949-d0b11bdaac8a/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY=
+golang.org/x/sys v0.0.0-20200323222414-85ca7c5b95cd/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
+golang.org/x/text v0.3.0/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ=
+google.golang.org/appengine v1.4.0 h1:/wp5JvzpHIxhs/dumFmF7BXTf3Z+dd4uXta4kVyO508=
+google.golang.org/appengine v1.4.0/go.mod h1:xpcJRLb0r/rnEns0DIKYYv+WjYCduHsrkT7/EB5XEv4=
diff --git a/scripts/gen_github_action_config/main.go b/scripts/gen_github_action_config/main.go
new file mode 100644
index 00000000..e34537d6
--- /dev/null
+++ b/scripts/gen_github_action_config/main.go
@@ -0,0 +1,231 @@
+package main
+
+import (
+	"context"
+	"encoding/json"
+	"errors"
+	"fmt"
+	"log"
+	"os"
+	"strconv"
+	"strings"
+
+	"github.com/shurcooL/githubv4"
+	"golang.org/x/oauth2"
+)
+
+func main() {
+	if err := generate(context.Background()); err != nil {
+		log.Fatal(err)
+	}
+}
+
+func generate(ctx context.Context) error {
+	allReleases, err := fetchAllReleases(ctx)
+	if err != nil {
+		return fmt.Errorf("failed to fetch all releases: %w", err)
+	}
+
+	cfg, err := buildConfig(allReleases)
+	if err != nil {
+		return fmt.Errorf("failed to build config: %w", err)
+	}
+
+	if len(os.Args) != 2 { //nolint:gomnd
+		return fmt.Errorf("usage: go run .../main.go out-path.json")
+	}
+	outFile, err := os.Create(os.Args[1])
+	if err != nil {
+		return fmt.Errorf("failed to create output config file: %w", err)
+	}
+	defer outFile.Close()
+	enc := json.NewEncoder(outFile)
+	enc.SetIndent("", "  ")
+	if err = enc.Encode(cfg); err != nil {
+		return fmt.Errorf("failed to json encode config: %w", err)
+	}
+
+	return nil
+}
+
+type logInfo struct {
+	Warning string `json:",omitempty"`
+	Info    string `json:",omitempty"`
+}
+
+type versionConfig struct {
+	Error string `json:",omitempty"`
+
+	Log *logInfo `json:",omitempty"`
+
+	TargetVersion string `json:",omitempty"`
+	AssetURL      string `json:",omitempty"`
+}
+
+type actionConfig struct {
+	MinorVersionToConfig map[string]versionConfig
+}
+
+type version struct {
+	major, minor, patch int
+}
+
+func (v version) String() string {
+	ret := fmt.Sprintf("v%d.%d", v.major, v.minor)
+	if v.patch != noPatch {
+		ret += fmt.Sprintf(".%d", v.patch)
+	}
+	return ret
+}
+
+func (v *version) isAfterOrEq(vv *version) bool {
+	if v.major != vv.major {
+		return v.major >= vv.major
+	}
+	if v.minor != vv.minor {
+		return v.minor >= vv.minor
+	}
+
+	return v.patch >= vv.patch
+}
+
+const noPatch = -1
+
+func parseVersion(s string) (*version, error) {
+	const vPrefix = "v"
+	if !strings.HasPrefix(s, vPrefix) {
+		return nil, fmt.Errorf("version should start with %q", vPrefix)
+	}
+	s = strings.TrimPrefix(s, vPrefix)
+
+	parts := strings.Split(s, ".")
+
+	var nums []int
+	for _, part := range parts {
+		num, err := strconv.Atoi(part)
+		if err != nil {
+			return nil, fmt.Errorf("failed to parse version part: %w", err)
+		}
+		nums = append(nums, num)
+	}
+
+	if len(nums) == 2 { //nolint:gomnd
+		return &version{major: nums[0], minor: nums[1], patch: noPatch}, nil
+	}
+	if len(nums) == 3 { //nolint:gomnd
+		return &version{major: nums[0], minor: nums[1], patch: nums[2]}, nil
+	}
+
+	return nil, errors.New("invalid version format")
+}
+
+func findLinuxAssetURL(ver *version, releaseAssets []releaseAsset) (string, error) {
+	pattern := fmt.Sprintf("golangci-lint-%d.%d.%d-linux-amd64.tar.gz", ver.major, ver.minor, ver.patch)
+	for _, relAsset := range releaseAssets {
+		if strings.HasSuffix(relAsset.DownloadURL, pattern) {
+			return relAsset.DownloadURL, nil
+		}
+	}
+	return "", fmt.Errorf("no matched asset url for pattern %q", pattern)
+}
+
+func buildConfig(releases []release) (*actionConfig, error) {
+	versionToRelease := map[version]release{}
+	for _, rel := range releases {
+		ver, err := parseVersion(rel.TagName)
+		if err != nil {
+			return nil, fmt.Errorf("failed to parse release %s version: %w", rel.TagName, err)
+		}
+		if _, ok := versionToRelease[*ver]; ok {
+			return nil, fmt.Errorf("duplicate release %s", rel.TagName)
+		}
+		versionToRelease[*ver] = rel
+	}
+
+	maxPatchReleases := map[string]version{}
+	for ver := range versionToRelease {
+		key := fmt.Sprintf("v%d.%d", ver.major, ver.minor)
+		if mapVer, ok := maxPatchReleases[key]; !ok || ver.isAfterOrEq(&mapVer) {
+			maxPatchReleases[key] = ver
+		}
+	}
+
+	minorVersionToConfig := map[string]versionConfig{}
+	minAllowedVersion := version{major: 1, minor: 14, patch: 0}
+
+	for minorVersionedStr, maxPatchVersion := range maxPatchReleases {
+		if !maxPatchVersion.isAfterOrEq(&minAllowedVersion) {
+			minorVersionToConfig[minorVersionedStr] = versionConfig{
+				Error: fmt.Sprintf("golangci-lint version '%s' isn't supported: we support only %s and later versions",
+					minorVersionedStr, minAllowedVersion),
+			}
+			continue
+		}
+		maxPatchVersion := maxPatchVersion
+		assetURL, err := findLinuxAssetURL(&maxPatchVersion, versionToRelease[maxPatchVersion].ReleaseAssets.Nodes)
+		if err != nil {
+			return nil, fmt.Errorf("failed to find linux asset url for release %s: %w", maxPatchVersion, err)
+		}
+		minorVersionToConfig[minorVersionedStr] = versionConfig{
+			TargetVersion: maxPatchVersion.String(),
+			AssetURL:      assetURL,
+		}
+	}
+
+	return &actionConfig{MinorVersionToConfig: minorVersionToConfig}, nil
+}
+
+type release struct {
+	TagName       string
+	ReleaseAssets struct {
+		Nodes []releaseAsset
+	} `graphql:"releaseAssets(first: 50)"`
+}
+
+type releaseAsset struct {
+	DownloadURL string
+}
+
+func fetchAllReleases(ctx context.Context) ([]release, error) {
+	githubToken := os.Getenv("GITHUB_TOKEN")
+	if githubToken == "" {
+		return nil, errors.New("no GITHUB_TOKEN environment variable")
+	}
+	src := oauth2.StaticTokenSource(&oauth2.Token{AccessToken: githubToken})
+	httpClient := oauth2.NewClient(context.Background(), src)
+	client := githubv4.NewClient(httpClient)
+
+	var q struct {
+		Repository struct {
+			Releases struct {
+				Nodes    []release
+				PageInfo struct {
+					EndCursor   githubv4.String
+					HasNextPage bool
+				}
+			} `graphql:"releases(first: 100, after: $releasesCursor)"`
+		} `graphql:"repository(owner: $owner, name: $name)"`
+	}
+
+	vars := map[string]interface{}{
+		"owner":          githubv4.String("golangci"),
+		"name":           githubv4.String("golangci-lint"),
+		"releasesCursor": (*githubv4.String)(nil),
+	}
+
+	var allReleases []release
+	for {
+		err := client.Query(ctx, &q, vars)
+		if err != nil {
+			return nil, fmt.Errorf("failed to fetch releases page from github: %w", err)
+		}
+		releases := q.Repository.Releases
+		allReleases = append(allReleases, releases.Nodes...)
+		if !releases.PageInfo.HasNextPage {
+			break
+		}
+		vars["releasesCursor"] = githubv4.NewString(releases.PageInfo.EndCursor)
+	}
+
+	return allReleases, nil
+}