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.
49 50 51 52 |
# File 'lib/kettle/dev/ci_helpers.rb', line 49 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
172 173 174 |
# File 'lib/kettle/dev/ci_helpers.rb', line 172 def default_gitlab_token ENV["GITLAB_TOKEN"] || ENV["GL_TOKEN"] end |
.default_token ⇒ String?
Default GitHub token sourced from environment.
144 145 146 |
# File 'lib/kettle/dev/ci_helpers.rb', line 144 def default_token ENV["GITHUB_TOKEN"] || ENV["GH_TOKEN"] end |
.exclusions ⇒ Array<String>
List of workflow files to exclude from interactive menus and checks.
72 73 74 75 76 77 78 79 80 81 |
# File 'lib/kettle/dev/ci_helpers.rb', line 72 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.
138 139 140 |
# File 'lib/kettle/dev/ci_helpers.rb', line 138 def failed?(run) run && run["status"] == "completed" && run["conclusion"] && run["conclusion"] != "success" end |
.gitlab_failed?(pipeline) ⇒ Boolean
Whether a GitLab pipeline has failed
243 244 245 |
# File 'lib/kettle/dev/ci_helpers.rb', line 243 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
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 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 |
# File 'lib/kettle/dev/ci_helpers.rb', line 183 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) return unless data.is_a?(Array) pipe = data.first return unless pipe.is_a?(Hash) # 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 => e Kettle::Dev.debug_error(e, __method__) # 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 => e Kettle::Dev.debug_error(e, __method__) nil end |
.gitlab_success?(pipeline) ⇒ Boolean
Whether a GitLab pipeline has succeeded
236 237 238 |
# File 'lib/kettle/dev/ci_helpers.rb', line 236 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.
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 121 122 123 124 125 126 |
# File 'lib/kettle/dev/ci_helpers.rb', line 90 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 => e Kettle::Dev.debug_error(e, __method__) nil end |
.origin_url ⇒ String?
Raw origin URL string from git config
152 153 154 155 |
# File 'lib/kettle/dev/ci_helpers.rb', line 152 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 45 |
# 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
159 160 161 162 163 164 165 166 167 168 |
# File 'lib/kettle/dev/ci_helpers.rb', line 159 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.
131 132 133 |
# File 'lib/kettle/dev/ci_helpers.rb', line 131 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.
58 59 60 61 62 63 64 65 66 67 68 |
# File 'lib/kettle/dev/ci_helpers.rb', line 58 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 |