Class: PlanMyStuff::Issue
- Inherits:
-
ApplicationRecord
- Object
- ApplicationRecord
- PlanMyStuff::Issue
- Includes:
- PlanMyStuff::IssueExtractions::Approvals, PlanMyStuff::IssueExtractions::Links, PlanMyStuff::IssueExtractions::Viewers, PlanMyStuff::IssueExtractions::Waiting
- Defined in:
- lib/plan_my_stuff/issue.rb
Overview
Wraps a GitHub issue with parsed PMS metadata and comments. Class methods provide the public API for CRUD operations.
Follows an ActiveRecord-style pattern:
-
‘Issue.new(**attrs)` creates an unpersisted instance
-
‘Issue.create!` / `Issue.find` / `Issue.list` return persisted instances
-
‘issue.save!` / `issue.update!` / `issue.reload` for persistence
Instance Attribute Summary
Attributes inherited from ApplicationRecord
Class Method Summary collapse
-
.check_import!(import_id, repo: nil) ⇒ Hash
Polls a previously-submitted import for its current status.
-
.count(repo: nil, state: :open, labels: []) ⇒ Integer
Counts GitHub issues matching the given filters without paginating full payloads.
-
.create!(title:, body:, repo: nil, labels: [], user: nil, metadata: {}, add_to_project: nil, visibility: 'public', visibility_allowlist: [], issue_type: nil, issue_fields: nil, attachments: []) ⇒ PlanMyStuff::Issue
Creates a GitHub issue with PMS metadata embedded in the body.
-
.find(number, repo: nil) ⇒ PlanMyStuff::Issue
Finds a single GitHub issue by number and parses its PMS metadata.
-
.import!(payloads) ⇒ Array<Hash>
Submits one or more pre-built payloads to GitHub’s “Import Issues” preview endpoint (+POST /repos/:repo/import/issues+).
-
.list(repo: nil, state: :open, labels: [], page: 1, per_page: 25) ⇒ Array<PlanMyStuff::Issue>
Lists GitHub issues with optional filters and pagination.
-
.update!(number:, repo: nil, title: nil, body: nil, metadata: nil, labels: nil, state: nil, assignees: nil, issue_type: ISSUE_TYPE_UNCHANGED, issue_fields: nil) ⇒ Object
Updates an existing GitHub issue.
Instance Method Summary collapse
-
#archive!(now: Time.now.utc) ⇒ self
Tags the issue with the configured
archived_label, removes it from every Projects V2 board it belongs to, locks its conversation on GitHub, and stampsmetadata.archived_at. -
#assignees ⇒ Array<String>
GitHub assignees for this issue, by login.
-
#body ⇒ String?
Returns the issue body content.
-
#body=(value) ⇒ String
Assigning a new body marks the instance dirty so the next
save!rewrites the backing PMS body comment. -
#body_comment ⇒ PlanMyStuff::Comment?
Returns the comment marked as the issue body, if any.
-
#closed_at ⇒ Time?
GitHub’s closed_at timestamp (nil while open).
-
#comments ⇒ Array<PlanMyStuff::Comment>
Lazy-loads and memoizes comments from the GitHub API.
-
#created_at ⇒ Time?
GitHub’s created_at timestamp; settable on unpersisted issues for use with
Issue.import. -
#github_id ⇒ Integer?
GitHub database ID (required for the REST issue-dependency API, which takes integer issue_id rather than issue number).
-
#github_node_id ⇒ String?
GitHub GraphQL node ID (required for native sub-issue mutations).
-
#html_url ⇒ String?
GitHub web URL for this issue, for escape-hatch “View on GitHub” links.
-
#initialize(**attrs) ⇒ Issue
constructor
A new instance of Issue.
-
#issue_fields ⇒ PlanMyStuff::IssueFieldValueSet
Returns a hash-like view of GitHub Issue Field values currently set on this issue.
-
#issue_type ⇒ String?
GitHub issue type name (e.g. “Bug”, “Feature”) or
nilwhen no type is assigned. -
#labels ⇒ Array<String>
Label names.
-
#locked ⇒ Boolean
(also: #locked?)
GitHub’s
lockedflag;truefor archived or manually-locked issues (no new comments). -
#metadata ⇒ PlanMyStuff::IssueMetadata
Parsed metadata (empty when no PMS metadata present).
-
#number ⇒ Integer?
GitHub issue number.
-
#pms_comments ⇒ Array<PlanMyStuff::Comment>
Only comments created via PMS.
- #pms_issue? ⇒ Boolean
-
#raw_body ⇒ String?
Full body as stored on GitHub.
-
#reload ⇒ self
Re-fetches this issue from GitHub and updates all local attributes.
- #repo ⇒ PlanMyStuff::Repo?
- #repo=(value) ⇒ Object
-
#save!(user: nil, skip_notification: false) ⇒ self
Persists the issue.
-
#set_issue_fields!(updates) ⇒ self
Bulk-updates GitHub Issue Field values in a single
setIssueFieldValuemutation. -
#state ⇒ String?
Issue state (“open” or “closed”).
-
#title ⇒ String?
Issue title.
-
#update!(user: nil, skip_notification: false, **attrs) ⇒ self
Applies
attrsto this instance in-memory then callssave!. -
#updated_at ⇒ Time?
GitHub’s updated_at timestamp.
-
#user_link ⇒ String?
Per-issue URL in the consuming app (
config.issues_url_prefix+ “/” +number+ “?repo=Org/Repo”, ornilwhen either prefix or number is missing).
Methods included from PlanMyStuff::IssueExtractions::Waiting
#clear_waiting_on_user!, #enter_waiting_on_user!, #reopen_by_reply!
Methods included from PlanMyStuff::IssueExtractions::Viewers
#add_viewers!, #remove_viewers!, #visible_to?
Methods included from PlanMyStuff::IssueExtractions::Links
#add_blocker!, #add_related!, #add_sub_issue!, #blocked_by, #blocking, #duplicate_of, #mark_duplicate!, #parent, #related, #remove_blocker!, #remove_parent!, #remove_related!, #remove_sub_issue!, #set_parent!, #sub_tickets
Methods included from PlanMyStuff::IssueExtractions::Approvals
#approvals_required?, #approve!, #approvers, #fully_approved?, #pending_approvals, #reject!, #rejected_approvals, #remove_approvers!, #request_approvals!, #revoke_approval!
Methods inherited from ApplicationRecord
#destroyed?, #new_record?, #persisted?, read_field
Constructor Details
#initialize(**attrs) ⇒ Issue
Returns a new instance of Issue.
507 508 509 510 |
# File 'lib/plan_my_stuff/issue.rb', line 507 def initialize(**attrs) @body_dirty = false super end |
Class Method Details
.check_import!(import_id, repo: nil) ⇒ Hash
Polls a previously-submitted import for its current status.
384 385 386 387 388 389 390 391 392 393 394 |
# File 'lib/plan_my_stuff/issue.rb', line 384 def check_import!(import_id, repo: nil) client = PlanMyStuff.import_client resolved_repo = client.resolve_repo!(repo) client.octokit.get( "/repos/#{resolved_repo}/import/issues/#{import_id}", accept: 'application/vnd.github.golden-comet-preview+json', ) rescue Octokit::ClientError, Octokit::ServerError => e raise(PlanMyStuff::APIError.new(e., status: e.respond_to?(:response_status) ? e.response_status : nil)) end |
.count(repo: nil, state: :open, labels: []) ⇒ Integer
Counts GitHub issues matching the given filters without paginating full payloads.
Uses GitHub’s Search API (search/issues), which returns total_count in a single request. The is:issue qualifier excludes PRs server-side.
Caveats:
-
The search index lags writes by up to ~1 minute, so freshly created/closed issues may not be reflected immediately.
-
The Search API has its own rate limit (30 req/min authenticated) separate from the core REST API.
332 333 334 335 336 337 338 339 340 341 342 343 344 |
# File 'lib/plan_my_stuff/issue.rb', line 332 def count(repo: nil, state: :open, labels: []) client = PlanMyStuff.client resolved_repo = client.resolve_repo!(repo) normalized_state = state.to_s qualifiers = ["repo:#{resolved_repo}", 'is:issue'] qualifiers << "is:#{normalized_state}" unless normalized_state == 'all' labels_to_use = Array.wrap(labels).sort qualifiers += labels_to_use.map do |label| "label:\"#{label}\"" end client.rest(:search_issues, qualifiers.join(' '), per_page: 1).total_count end |
.create!(title:, body:, repo: nil, labels: [], user: nil, metadata: {}, add_to_project: nil, visibility: 'public', visibility_allowlist: [], issue_type: nil, issue_fields: nil, attachments: []) ⇒ PlanMyStuff::Issue
Creates a GitHub issue with PMS metadata embedded in the body.
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 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 |
# File 'lib/plan_my_stuff/issue.rb', line 93 def create!( title:, body:, repo: nil, labels: [], user: nil, metadata: {}, add_to_project: nil, visibility: 'public', visibility_allowlist: [], issue_type: nil, issue_fields: nil, attachments: [] ) if issue_fields.present? && !PlanMyStuff.configuration.issue_fields_enabled raise(PlanMyStuff::IssueFieldsNotEnabledError) end if body.blank? raise(PlanMyStuff::ValidationError.new('body must be present', field: :body, expected_type: :string)) end client = PlanMyStuff.client resolved_repo = client.resolve_repo!(repo) = PlanMyStuff::IssueMetadata.build( user: user, visibility: visibility, custom_fields: , ) .visibility_allowlist = Array.wrap(visibility_allowlist) .validate_custom_fields! serialized_body = PlanMyStuff::MetadataParser.serialize!(.to_h, '') resolved_type = resolve_issue_type!(issue_type) = {} [:labels] = labels if labels.present? [:type] = resolved_type if resolved_type.present? result = client.rest(:create_issue, resolved_repo, title, serialized_body, **) number = read_field(result, :number) store_etag_to_cache(client, resolved_repo, number, result, cache_writer: :write_issue) link_body = visible_body_for(number, resolved_repo) if link_body.present? result = client.rest( :update_issue, resolved_repo, number, body: PlanMyStuff::MetadataParser.serialize!(.to_h, link_body), ) store_etag_to_cache(client, resolved_repo, number, result, cache_writer: :write_issue) end issue = find(number, repo: resolved_repo) if add_to_project.present? project_number = resolve_project_number!(add_to_project) PlanMyStuff::ProjectItem.create!(issue, project_number: project_number) end PlanMyStuff::Comment.create!( issue: issue, body: body, user: user, visibility: .visibility.to_sym, skip_responded: true, issue_body: true, attachments: , ) issue.set_issue_fields!(issue_fields) if issue_fields.present? issue.reload PlanMyStuff::Notifications.instrument('issue_created', issue, user: user) issue end |
.find(number, repo: nil) ⇒ PlanMyStuff::Issue
Finds a single GitHub issue by number and parses its PMS metadata.
265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291 |
# File 'lib/plan_my_stuff/issue.rb', line 265 def find(number, repo: nil) client = PlanMyStuff.client resolved_repo = client.resolve_repo!(repo) github_issue = fetch_with_etag_cache( client, resolved_repo, number, rest_method: :issue, cache_reader: :read_issue, cache_writer: :write_issue, ) pull_request = if github_issue.respond_to?(:pull_request) github_issue.pull_request elsif github_issue.is_a?(Hash) github_issue[:pull_request] || github_issue['pull_request'] end if pull_request raise(Octokit::NotFound, { method: 'GET', url: "repos/#{resolved_repo}/issues/#{number}" }) end build(github_issue, repo: resolved_repo) end |
.import!(payloads) ⇒ Array<Hash>
Submits one or more pre-built payloads to GitHub’s “Import Issues” preview endpoint (+POST /repos/:repo/import/issues+). One request per payload: the endpoint only accepts a single {issue:, comments:} payload at a time.
Each payload hash MUST include a :repo key (symbol, string, or PlanMyStuff::Repo) and the GitHub-shaped :issue /+ :comments+ keys; :repo is extracted before the POST. Payloads are passed through to GitHub unchanged otherwise - callers are responsible for shape, encoding, and any PlanMyStuff metadata they want to embed.
The endpoint is async: each response carries an id and url for polling via Issue.check_import.
363 364 365 366 367 368 369 370 371 372 373 |
# File 'lib/plan_my_stuff/issue.rb', line 363 def import!(payloads) client = PlanMyStuff.import_client Array.wrap(payloads).map do |payload| repo = payload[:repo] || payload['repo'] || PlanMyStuff.configuration.default_repo raise(ArgumentError, 'import payload must include :repo') if repo.blank? body = payload.except(:repo, 'repo') submit_import_request!(client, client.resolve_repo!(repo), body) end end |
.list(repo: nil, state: :open, labels: [], page: 1, per_page: 25) ⇒ Array<PlanMyStuff::Issue>
Lists GitHub issues with optional filters and pagination.
303 304 305 306 307 308 309 310 311 312 313 |
# File 'lib/plan_my_stuff/issue.rb', line 303 def list(repo: nil, state: :open, labels: [], page: 1, per_page: 25) client = PlanMyStuff.client resolved_repo = client.resolve_repo!(repo) params = { state: state.to_s, page: page, per_page: per_page } params[:labels] = labels.sort.join(',') if labels.present? github_issues = client.rest(:list_issues, resolved_repo, **params) filtered = github_issues.reject { |gi| gi.respond_to?(:pull_request) && gi.pull_request } filtered.map { |gi| build(gi, repo: resolved_repo) } end |
.update!(number:, repo: nil, title: nil, body: nil, metadata: nil, labels: nil, state: nil, assignees: nil, issue_type: ISSUE_TYPE_UNCHANGED, issue_fields: nil) ⇒ Object
Updates an existing GitHub issue.
metadata: accepts either:
-
a
PlanMyStuff::IssueMetadatainstance - treated as the full authoritative metadata and serialized as-is (used by instancesave!/update!so local @metadata mutations like metadata.commit_sha = … actually persist). -
a
Hash- patch-style merge against the CURRENT remote metadata. Top-level keys are merged in;:custom_fieldsis merged separately so unrelated fields stay intact.
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 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 |
# File 'lib/plan_my_stuff/issue.rb', line 201 def update!( number:, repo: nil, title: nil, body: nil, metadata: nil, labels: nil, state: nil, assignees: nil, issue_type: ISSUE_TYPE_UNCHANGED, issue_fields: nil ) client = PlanMyStuff.client resolved_repo = client.resolve_repo!(repo) = {} [:title] = title unless title.nil? [:labels] = labels unless labels.nil? [:state] = state.to_s unless state.nil? [:assignees] = Array.wrap(assignees) unless assignees.nil? [:type] = resolve_issue_type!(issue_type) unless issue_type.equal?(ISSUE_TYPE_UNCHANGED) case when PlanMyStuff::IssueMetadata .validate_custom_fields! [:body] = PlanMyStuff::MetadataParser.serialize!(.to_h, visible_body_for(number, resolved_repo)) when Hash current = client.rest(:issue, resolved_repo, number) current_body = current.respond_to?(:body) ? current.body : current[:body] parsed = PlanMyStuff::MetadataParser.parse(current_body) = parsed[:metadata] merged_custom_fields = ([:custom_fields] || {}).merge([:custom_fields] || {}) = .merge() [:custom_fields] = merged_custom_fields PlanMyStuff::CustomFields.new( PlanMyStuff.configuration.custom_fields_for(:issue), merged_custom_fields, ).validate! [:body] = PlanMyStuff::MetadataParser.serialize!(, visible_body_for(number, resolved_repo)) end update_body_comment!(number, resolved_repo, body) if body updated_issue = find(number, repo: resolved_repo).set_issue_fields!(issue_fields) if issue_fields.present? return updated_issue if .none? result = client.rest(:update_issue, resolved_repo, number, **) store_etag_to_cache(client, resolved_repo, number, result, cache_writer: :write_issue) result end |
Instance Method Details
#archive!(now: Time.now.utc) ⇒ self
Tags the issue with the configured archived_label, removes it from every Projects V2 board it belongs to, locks its conversation on GitHub, and stamps metadata.archived_at. Emits issue_archived.plan_my_stuff on success.
No-op (no network calls, no event) when the issue is already archived (either metadata.archived_at is set or the archived label is already on the issue).
553 554 555 556 557 558 559 560 561 562 563 564 565 566 567 568 569 570 571 572 573 574 575 576 577 578 579 580 581 |
# File 'lib/plan_my_stuff/issue.rb', line 553 def archive!(now: Time.now.utc) label = PlanMyStuff.configuration.archived_label return self unless state == 'closed' return self if .archived_at.present? return self if labels.include?(label) self.class.update!( number: number, repo: repo, metadata: { archived_at: PlanMyStuff.format_time(now) }, ) PlanMyStuff::Label.ensure!(repo: repo, name: label) PlanMyStuff::Label.add!(issue: self, labels: [label]) remove_from_all_projects! PlanMyStuff.client.rest(:lock_issue, repo.full_name, number) reload PlanMyStuff::Notifications.instrument( 'issue_archived', self, reason: :aged_closed, ) self end |
#assignees ⇒ Array<String>
GitHub assignees for this issue, by login.
657 658 659 |
# File 'lib/plan_my_stuff/issue.rb', line 657 def assignees extract_assignee_logins(github_response) end |
#body ⇒ String?
Returns the issue body content. For PMS issues, this is the body from the body comment (stripped of its header). Falls back to the parsed issue body for non-PMS issues.
41 |
# File 'lib/plan_my_stuff/issue.rb', line 41 attribute :body, :string |
#body=(value) ⇒ String
Assigning a new body marks the instance dirty so the next save! rewrites the backing PMS body comment. Unsaved assignments are reflected by #body until persisted or reloaded.
524 525 526 527 |
# File 'lib/plan_my_stuff/issue.rb', line 524 def body=(value) super @body_dirty = true end |
#body_comment ⇒ PlanMyStuff::Comment?
Returns the comment marked as the issue body, if any.
675 676 677 |
# File 'lib/plan_my_stuff/issue.rb', line 675 def body_comment pms_comments.find { |c| c..issue_body? } end |
#closed_at ⇒ Time?
Returns GitHub’s closed_at timestamp (nil while open).
34 |
# File 'lib/plan_my_stuff/issue.rb', line 34 attribute :closed_at |
#comments ⇒ Array<PlanMyStuff::Comment>
Lazy-loads and memoizes comments from the GitHub API.
641 642 643 |
# File 'lib/plan_my_stuff/issue.rb', line 641 def comments @comments ||= load_comments end |
#created_at ⇒ Time?
Returns GitHub’s created_at timestamp; settable on unpersisted issues for use with Issue.import.
32 |
# File 'lib/plan_my_stuff/issue.rb', line 32 attribute :created_at |
#github_id ⇒ Integer?
GitHub database ID (required for the REST issue-dependency API, which takes integer issue_id rather than issue number).
708 709 710 |
# File 'lib/plan_my_stuff/issue.rb', line 708 def github_id safe_read_field(github_response, :id) end |
#github_node_id ⇒ String?
GitHub GraphQL node ID (required for native sub-issue mutations). Read from the hydrated REST response.
699 700 701 |
# File 'lib/plan_my_stuff/issue.rb', line 699 def github_node_id safe_read_field(github_response, :node_id) end |
#html_url ⇒ String?
GitHub web URL for this issue, for escape-hatch “View on GitHub” links.
649 650 651 |
# File 'lib/plan_my_stuff/issue.rb', line 649 def html_url safe_read_field(github_response, :html_url) end |
#issue_fields ⇒ PlanMyStuff::IssueFieldValueSet
Returns a hash-like view of GitHub Issue Field values currently set on this issue. Reads on first access and memoizes; set_issue_fields! invalidates the cache. Returns an empty set without making a request when config.issue_fields_enabled is false.
718 719 720 |
# File 'lib/plan_my_stuff/issue.rb', line 718 def issue_fields @issue_fields ||= load_issue_fields! end |
#issue_type ⇒ String?
Returns GitHub issue type name (e.g. “Bug”, “Feature”) or nil when no type is assigned. Read from the nested type.name field on the REST response. Settable via the issue_type: kwarg on Issue.create! / Issue.update!.
45 |
# File 'lib/plan_my_stuff/issue.rb', line 45 attribute :issue_type, :string |
#labels ⇒ Array<String>
Returns label names.
28 |
# File 'lib/plan_my_stuff/issue.rb', line 28 attribute :labels, default: -> { [] } |
#locked ⇒ Boolean Also known as: locked?
Returns GitHub’s locked flag; true for archived or manually-locked issues (no new comments).
36 |
# File 'lib/plan_my_stuff/issue.rb', line 36 attribute :locked, :boolean, default: false |
#metadata ⇒ PlanMyStuff::IssueMetadata
Returns parsed metadata (empty when no PMS metadata present).
22 |
# File 'lib/plan_my_stuff/issue.rb', line 22 attribute :metadata, default: -> { PlanMyStuff::IssueMetadata.new } |
#number ⇒ Integer?
Returns GitHub issue number.
18 |
# File 'lib/plan_my_stuff/issue.rb', line 18 attribute :number, :integer |
#pms_comments ⇒ Array<PlanMyStuff::Comment>
Returns only comments created via PMS.
667 668 669 |
# File 'lib/plan_my_stuff/issue.rb', line 667 def pms_comments comments.select(&:pms_comment?) end |
#pms_issue? ⇒ Boolean
662 663 664 |
# File 'lib/plan_my_stuff/issue.rb', line 662 def pms_issue? .schema_version.present? end |
#raw_body ⇒ String?
Returns full body as stored on GitHub.
20 |
# File 'lib/plan_my_stuff/issue.rb', line 20 attribute :raw_body, :string |
#reload ⇒ self
Re-fetches this issue from GitHub and updates all local attributes.
631 632 633 634 635 |
# File 'lib/plan_my_stuff/issue.rb', line 631 def reload fresh = self.class.find(number, repo: repo) hydrate_from_issue(fresh) self end |
#repo=(value) ⇒ Object
513 514 515 |
# File 'lib/plan_my_stuff/issue.rb', line 513 def repo=(value) super(value.present? ? PlanMyStuff::Repo.resolve!(value) : nil) end |
#save!(user: nil, skip_notification: false) ⇒ self
Persists the issue. Creates if new, otherwise performs a full write: serializes @metadata into the GitHub issue body and PATCHes title/state/labels. When #body= has been called since the last load, also rewrites the PMS body comment. Always reloads afterwards.
589 590 591 592 593 594 595 596 597 598 599 600 601 602 603 604 605 606 607 608 609 610 611 612 |
# File 'lib/plan_my_stuff/issue.rb', line 589 def save!(user: nil, skip_notification: false) if new_record? created = self.class.create!( title: title, body: body, repo: repo, labels: labels || [], user: user || .created_by, metadata: .custom_fields.to_h, visibility: .visibility, visibility_allowlist: Array.wrap(.visibility_allowlist), issue_type: issue_type, issue_fields: @pending_issue_fields, ) hydrate_from_issue(created) else captured_changes = changes.dup persist_update! instrument_update(captured_changes, user) unless skip_notification end @pending_issue_fields = nil self end |
#set_issue_fields!(updates) ⇒ self
Bulk-updates GitHub Issue Field values in a single setIssueFieldValue mutation. Each key is the field display name; values are coerced to the right input fragment based on the field’s type. Passing nil as a value clears that field.
733 734 735 736 737 738 739 740 741 742 743 744 745 746 |
# File 'lib/plan_my_stuff/issue.rb', line 733 def set_issue_fields!(updates) raise(PlanMyStuff::IssueFieldsNotEnabledError) unless PlanMyStuff.configuration.issue_fields_enabled fields_by_name = PlanMyStuff::IssueField.list(org: repo.organization).index_by { |field| field.name.downcase } inputs = updates.map { |name, value| build_issue_field_input(fields_by_name, name, value) } PlanMyStuff.client.graphql( PlanMyStuff::GraphQL::Queries::SET_ISSUE_FIELD_VALUES, variables: { issueId: github_node_id, issueFields: inputs }, ) @issue_fields = nil self end |
#state ⇒ String?
Returns issue state (“open” or “closed”).
26 |
# File 'lib/plan_my_stuff/issue.rb', line 26 attribute :state, :string |
#title ⇒ String?
Returns issue title.
24 |
# File 'lib/plan_my_stuff/issue.rb', line 24 attribute :title, :string |
#update!(user: nil, skip_notification: false, **attrs) ⇒ self
Applies attrs to this instance in-memory then calls save!. Supports title:, body:, state:, labels:, assignees:, and metadata:. The metadata: kwarg is a hash whose keys are merged into the existing metadata (top-level attributes assigned directly; :custom_fields merged key-by-key).
622 623 624 625 |
# File 'lib/plan_my_stuff/issue.rb', line 622 def update!(user: nil, skip_notification: false, **attrs) apply_update_attrs(attrs) save!(user: user, skip_notification: skip_notification) end |
#updated_at ⇒ Time?
Returns GitHub’s updated_at timestamp.
30 |
# File 'lib/plan_my_stuff/issue.rb', line 30 attribute :updated_at |
#user_link ⇒ String?
Returns per-issue URL in the consuming app (config.issues_url_prefix + “/” + number + “?repo=Org/Repo”, or nil when either prefix or number is missing). Also rendered as the destination of the markdown link in the GitHub issue body.
532 533 534 535 536 537 538 539 540 |
# File 'lib/plan_my_stuff/issue.rb', line 532 def user_link prefix = PlanMyStuff.configuration.issues_url_prefix return if prefix.blank? || number.blank? base = "#{prefix.to_s.chomp('/')}/#{number}" return base if repo.blank? "#{base}?repo=#{URI.encode_www_form_component(repo.full_name)}" end |