123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123 |
- # Copyright (C) 2012-2025 Zammad Foundation, https://zammad-foundation.org/
- class GitHub
- class LinkedIssue
- STATES_MAPPING = {
- 'OPEN' => 'open',
- 'CLOSED' => 'closed'
- }.freeze
- QUERY = <<-GRAPHQL.freeze
- query($repositor_owner: String!, $repository_name: String!, $issue_id: Int!) {
- repository(owner: $repositor_owner, name: $repository_name) {
- issue(number: $issue_id) {
- number
- title
- state
- url
- milestone {
- title
- }
- assignees(last: 100) {
- edges {
- node {
- name
- }
- }
- }
- labels(last: 100) {
- edges {
- node {
- name
- color
- }
- }
- }
- }
- }
- }
- GRAPHQL
- attr_reader :client
- def initialize(client)
- @client = client
- end
- def find_by(url)
- @result = query_by_url(url)
- return if @result.blank?
- to_h
- end
- private
- def to_h
- {
- id: @result['number'].to_s,
- title: @result['title'],
- url: @result['url'],
- icon_state: STATES_MAPPING.fetch(@result['state'], @result['state']),
- milestone: milestone,
- assignees: assignees,
- labels: labels,
- }
- end
- def assignees
- @result['assignees']['edges'].map do |assignee|
- assignee['node']['name']
- end
- end
- def labels
- @result['labels']['edges'].map do |label|
- {
- text_color: text_color(label['node']['color']),
- color: "##{label['node']['color']}",
- title: label['node']['name']
- }
- end
- end
- def text_color(background_color)
- background_color.to_i(16) > 0xFFF / 2 ? '#000000' : '#FFFFFF'
- end
- def milestone
- @result.dig('milestone', 'title')
- end
- def query_by_url(url)
- response = client.perform(
- query: GitHub::LinkedIssue::QUERY,
- variables: variables!(url)
- )
- response.dig('data', 'repository', 'issue')
- end
- def variables!(url)
- if url !~ %r{^https?://([^/]+)/([^/]+)/([^/]+)/issues/(\d+)$}
- raise Exceptions::UnprocessableEntity, __('Invalid GitHub issue link format')
- end
- host = $1
- repositor_owner = $2
- repository_name = $3
- id = $4
- if client.endpoint.exclude?(host)
- raise Exceptions::UnprocessableEntity, "Issue link doesn't match configured GitHub endpoint '#{client.endpoint}'"
- end
- {
- repositor_owner: repositor_owner,
- repository_name: repository_name,
- issue_id: id.to_i,
- }
- end
- end
- end
|