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.
-
.parse_hosted_repo(url, host) ⇒ Array(String, String)?
Parse owner/repo from common hosted Git remote URL shapes.
-
.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.
43 44 45 46 |
# File 'lib/kettle/dev/ci_helpers.rb', line 43 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
162 163 164 |
# File 'lib/kettle/dev/ci_helpers.rb', line 162 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.
66 67 68 69 70 71 72 73 74 75 |
# File 'lib/kettle/dev/ci_helpers.rb', line 66 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
233 234 235 |
# File 'lib/kettle/dev/ci_helpers.rb', line 233 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
173 174 175 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 217 218 219 220 221 |
# File 'lib/kettle/dev/ci_helpers.rb', line 173 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
226 227 228 |
# File 'lib/kettle/dev/ci_helpers.rb', line 226 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.
84 85 86 87 88 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 84 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
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 |
.parse_hosted_repo(url, host) ⇒ Array(String, String)?
Parse owner/repo from common hosted Git remote URL shapes.
241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 |
# File 'lib/kettle/dev/ci_helpers.rb', line 241 def parse_hosted_repo(url, host) return unless url if url =~ %r{\Agit@#{Regexp.escape(host)}:(.+?)/(.+?)(?:\.git)?\z} return [Regexp.last_match(1), Regexp.last_match(2).sub(/\.git\z/, "")] end uri = URI.parse(url) return unless uri.host == host path = uri.path.to_s.sub(%r{\A/}, "") owner, repo = path.split("/", 2) return unless owner && repo && !owner.empty? && !repo.empty? [owner, repo.sub(/\.git\z/, "")] rescue URI::InvalidURIError 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 and HTTPS remote URL forms.
34 35 36 37 38 39 |
# File 'lib/kettle/dev/ci_helpers.rb', line 34 def repo_info out, status = Open3.capture2("git", "config", "--get", "remote.origin.url") return unless status.success? parse_hosted_repo(out.strip, "github.com") end |
.repo_info_gitlab ⇒ Array(String, String)?
Parse GitLab owner/repo from origin if pointing to gitlab.com
153 154 155 156 157 158 |
# File 'lib/kettle/dev/ci_helpers.rb', line 153 def repo_info_gitlab url = origin_url return unless url parse_hosted_repo(url, "gitlab.com") 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.
52 53 54 55 56 57 58 59 60 61 62 |
# File 'lib/kettle/dev/ci_helpers.rb', line 52 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 |