Module: Kettle::Dev::CIHelpers
- Defined in:
- lib/kettle/dev/ci_helpers.rb
Overview
CI-related helper functions used by Rake tasks and release tooling.
This module only exposes module-functions (no instance state) and is
intentionally small so it can be required by both Rake tasks and the
kettle-release executable.
Class Method Summary collapse
-
.current_branch ⇒ String?
Current git branch name, or nil when not in a repository.
-
.default_gitlab_token ⇒ String?
Default GitLab token from environment.
-
.default_token ⇒ String?
Default GitHub token sourced from environment.
-
.exclusions ⇒ Array<String>
List of workflow files to exclude from interactive menus and checks.
-
.failed?(run) ⇒ Boolean
Whether a run has completed with a non-success conclusion.
-
.gitlab_failed?(pipeline) ⇒ Boolean
Whether a GitLab pipeline has failed.
-
.gitlab_latest_pipeline(owner:, repo:, branch: nil, host: "gitlab.com", token: default_gitlab_token) ⇒ Hash{String=>String,Integer}?
Fetch the latest pipeline for a branch on GitLab.
-
.gitlab_success?(pipeline) ⇒ Boolean
Whether a GitLab pipeline has succeeded.
-
.latest_run(owner:, repo:, workflow_file:, branch: nil, token: default_token) ⇒ Hash{String=>String,Integer}?
Fetch latest workflow run info for a given workflow and branch via GitHub API.
-
.origin_url ⇒ String?
Raw origin URL string from git config.
-
.project_root ⇒ String
Determine the project root directory.
-
.repo_info ⇒ Array(String, String)?
Parse the GitHub owner/repo from the configured origin remote.
-
.repo_info_gitlab ⇒ Array(String, String)?
Parse GitLab owner/repo from origin if pointing to gitlab.com.
-
.success?(run) ⇒ Boolean
Whether a run has completed successfully.
-
.workflows_list(root = project_root) ⇒ Array<String>
List workflow YAML basenames under .github/workflows at the given root.
Class Method Details
.current_branch ⇒ String?
Current git branch name, or nil when not in a repository.
48 49 50 51 |
# File 'lib/kettle/dev/ci_helpers.rb', line 48 def current_branch out, status = Open3.capture2("git", "rev-parse", "--abbrev-ref", "HEAD") status.success? ? out.strip : nil end |
.default_gitlab_token ⇒ String?
Default GitLab token from environment
165 166 167 |
# File 'lib/kettle/dev/ci_helpers.rb', line 165 def default_gitlab_token ENV["GITLAB_TOKEN"] || ENV["GL_TOKEN"] end |
.default_token ⇒ String?
Default GitHub token sourced from environment.
138 139 140 |
# File 'lib/kettle/dev/ci_helpers.rb', line 138 def default_token ENV["GITHUB_TOKEN"] || ENV["GH_TOKEN"] end |
.exclusions ⇒ Array<String>
List of workflow files to exclude from interactive menus and checks.
71 72 73 74 75 76 77 78 79 80 |
# File 'lib/kettle/dev/ci_helpers.rb', line 71 def exclusions %w[ auto-assign.yml codeql-analysis.yml danger.yml dependency-review.yml discord-notifier.yml opencollective.yml ] end |
.failed?(run) ⇒ Boolean
Whether a run has completed with a non-success conclusion.
132 133 134 |
# File 'lib/kettle/dev/ci_helpers.rb', line 132 def failed?(run) run && run["status"] == "completed" && run["conclusion"] && run["conclusion"] != "success" end |
.gitlab_failed?(pipeline) ⇒ Boolean
Whether a GitLab pipeline has failed
228 229 230 |
# File 'lib/kettle/dev/ci_helpers.rb', line 228 def gitlab_failed?(pipeline) pipeline && pipeline["status"] == "failed" end |
.gitlab_latest_pipeline(owner:, repo:, branch: nil, host: "gitlab.com", token: default_gitlab_token) ⇒ Hash{String=>String,Integer}?
Fetch the latest pipeline for a branch on GitLab
176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 |
# File 'lib/kettle/dev/ci_helpers.rb', line 176 def gitlab_latest_pipeline(owner:, repo:, branch: nil, host: "gitlab.com", token: default_gitlab_token) return unless owner && repo b = branch || current_branch return unless b project = URI.encode_www_form_component("#{owner}/#{repo}") uri = URI("https://#{host}/api/v4/projects/#{project}/pipelines?ref=#{URI.encode_www_form_component(b)}&per_page=1") req = Net::HTTP::Get.new(uri) req["User-Agent"] = "kettle-dev/ci-helpers" req["PRIVATE-TOKEN"] = token if token && !token.empty? res = Net::HTTP.start(uri.hostname, uri.port, use_ssl: true) { |http| http.request(req) } return unless res.is_a?(Net::HTTPSuccess) data = JSON.parse(res.body) pipe = data&.first return unless pipe # Attempt to enrich with failure_reason by querying the single pipeline endpoint begin if pipe["id"] detail_uri = URI("https://#{host}/api/v4/projects/#{project}/pipelines/#{pipe["id"]}") dreq = Net::HTTP::Get.new(detail_uri) dreq["User-Agent"] = "kettle-dev/ci-helpers" dreq["PRIVATE-TOKEN"] = token if token && !token.empty? dres = Net::HTTP.start(detail_uri.hostname, detail_uri.port, use_ssl: true) { |http| http.request(dreq) } if dres.is_a?(Net::HTTPSuccess) det = JSON.parse(dres.body) pipe["failure_reason"] = det["failure_reason"] if det.is_a?(Hash) pipe["status"] = det["status"] if det["status"] pipe["web_url"] = det["web_url"] if det["web_url"] end end rescue StandardError # ignore enrichment errors; fall back to basic fields end { "status" => pipe["status"], "web_url" => pipe["web_url"], "id" => pipe["id"], "failure_reason" => pipe["failure_reason"], } rescue StandardError nil end |
.gitlab_success?(pipeline) ⇒ Boolean
Whether a GitLab pipeline has succeeded
221 222 223 |
# File 'lib/kettle/dev/ci_helpers.rb', line 221 def gitlab_success?(pipeline) pipeline && pipeline["status"] == "success" end |
.latest_run(owner:, repo:, workflow_file:, branch: nil, token: default_token) ⇒ Hash{String=>String,Integer}?
Fetch latest workflow run info for a given workflow and branch via GitHub API.
89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 |
# File 'lib/kettle/dev/ci_helpers.rb', line 89 def latest_run(owner:, repo:, workflow_file:, branch: nil, token: default_token) return unless owner && repo b = branch || current_branch return unless b # Scope to the exact commit SHA when available to avoid picking up a previous run on the same branch. sha_out, status = Open3.capture2("git", "rev-parse", "HEAD") sha = status.success? ? sha_out.strip : nil base_url = "https://api.github.com/repos/#{owner}/#{repo}/actions/workflows/#{workflow_file}/runs?branch=#{URI.encode_www_form_component(b)}&per_page=5" uri = URI(base_url) req = Net::HTTP::Get.new(uri) req["User-Agent"] = "kettle-dev/ci-helpers" req["Authorization"] = "token #{token}" if token && !token.empty? res = Net::HTTP.start(uri.hostname, uri.port, use_ssl: true) { |http| http.request(req) } return unless res.is_a?(Net::HTTPSuccess) data = JSON.parse(res.body) runs = Array(data["workflow_runs"]) || [] # Try to match by head_sha first; fall back to first run (branch-scoped) if none matches yet. run = if sha runs.find { |r| r["head_sha"] == sha } || runs.first else runs.first end return unless run { "status" => run["status"], "conclusion" => run["conclusion"], "html_url" => run["html_url"], "id" => run["id"], } rescue StandardError nil end |
.origin_url ⇒ String?
Raw origin URL string from git config
146 147 148 149 |
# File 'lib/kettle/dev/ci_helpers.rb', line 146 def origin_url out, status = Open3.capture2("git", "config", "--get", "remote.origin.url") status.success? ? out.strip : nil end |
.project_root ⇒ String
Determine the project root directory.
21 22 23 24 25 26 27 28 29 |
# File 'lib/kettle/dev/ci_helpers.rb', line 21 def project_root # Too difficult to test every possible branch here, so ignoring # :nocov: dir = if defined?(Rake) && Rake&.application&.respond_to?(:original_dir) Rake.application.original_dir end # :nocov: dir || Dir.pwd end |
.repo_info ⇒ Array(String, String)?
Parse the GitHub owner/repo from the configured origin remote.
Supports SSH (git@github.com:owner/repo(.git)) and HTTPS
(https://github.com/owner/repo(.git)) forms.
35 36 37 38 39 40 41 42 43 44 |
# File 'lib/kettle/dev/ci_helpers.rb', line 35 def repo_info out, status = Open3.capture2("git", "config", "--get", "remote.origin.url") return unless status.success? url = out.strip if url =~ %r{git@github.com:(.+?)/(.+?)(\.git)?$} [Regexp.last_match(1), Regexp.last_match(2).sub(/\.git\z/, "")] elsif url =~ %r{https://github.com/(.+?)/(.+?)(\.git)?$} [Regexp.last_match(1), Regexp.last_match(2).sub(/\.git\z/, "")] end end |
.repo_info_gitlab ⇒ Array(String, String)?
Parse GitLab owner/repo from origin if pointing to gitlab.com
153 154 155 156 157 158 159 160 161 |
# File 'lib/kettle/dev/ci_helpers.rb', line 153 def repo_info_gitlab url = origin_url return unless url if url =~ %r{git@gitlab.com:(.+?)/(.+?)(\.git)?$} [Regexp.last_match(1), Regexp.last_match(2).sub(/\.git\z/, "")] elsif url =~ %r{https://gitlab.com/(.+?)/(.+?)(\.git)?$} [Regexp.last_match(1), Regexp.last_match(2).sub(/\.git\z/, "")] end end |
.success?(run) ⇒ Boolean
Whether a run has completed successfully.
125 126 127 |
# File 'lib/kettle/dev/ci_helpers.rb', line 125 def success?(run) run && run["status"] == "completed" && run["conclusion"] == "success" end |
.workflows_list(root = project_root) ⇒ Array<String>
List workflow YAML basenames under .github/workflows at the given root.
Excludes maintenance workflows defined by #exclusions.
57 58 59 60 61 62 63 64 65 66 67 |
# File 'lib/kettle/dev/ci_helpers.rb', line 57 def workflows_list(root = project_root) workflows_dir = File.join(root, ".github", "workflows") files = if Dir.exist?(workflows_dir) Dir[File.join(workflows_dir, "*.yml")] + Dir[File.join(workflows_dir, "*.yaml")] else [] end basenames = files.map { |p| File.basename(p) } basenames = basenames.uniq - exclusions basenames.sort end |