service.rb 5.9 KB
Newer Older
1 2
# To add new service you should build a class inherited from Service
# and implement a set of methods
3
class Service < ActiveRecord::Base
4
  include Sortable
D
Drew Blessing 已提交
5 6
  serialize :properties, JSON

D
Dmitriy Zaporozhets 已提交
7
  default_value_for :active, false
8 9
  default_value_for :push_events, true
  default_value_for :issues_events, true
10
  default_value_for :confidential_issues_events, true
11 12
  default_value_for :merge_requests_events, true
  default_value_for :tag_push_events, true
13
  default_value_for :note_events, true
14
  default_value_for :build_events, true
15
  default_value_for :wiki_page_events, true
16 17

  after_initialize :initialize_properties
D
Dmitriy Zaporozhets 已提交
18

19
  after_commit :reset_updated_properties
20
  after_commit :cache_project_has_external_issue_tracker
21
  after_commit :cache_project_has_external_wiki
22

23
  belongs_to :project, inverse_of: :services
24 25
  has_one :service_hook

M
Marin Jankovski 已提交
26
  validates :project_id, presence: true, unless: Proc.new { |service| service.template? }
27

K
Kamil Trzcinski 已提交
28
  scope :visible, -> { where.not(type: ['GitlabIssueTrackerService', 'GitlabCiService']) }
29
  scope :issue_trackers, -> { where(category: 'issue_tracker') }
30
  scope :external_wikis, -> { where(type: 'ExternalWikiService').active }
31 32
  scope :active, -> { where(active: true) }
  scope :without_defaults, -> { where(default: false) }
33

34 35 36
  scope :push_hooks, -> { where(push_events: true, active: true) }
  scope :tag_push_hooks, -> { where(tag_push_events: true, active: true) }
  scope :issue_hooks, -> { where(issues_events: true, active: true) }
37
  scope :confidential_issue_hooks, -> { where(confidential_issues_events: true, active: true) }
38
  scope :merge_request_hooks, -> { where(merge_requests_events: true, active: true) }
39
  scope :note_hooks, -> { where(note_events: true, active: true) }
40
  scope :build_hooks, -> { where(build_events: true, active: true) }
41
  scope :pipeline_hooks, -> { where(pipeline_events: true, active: true) }
42
  scope :wiki_page_hooks, -> { where(wiki_page_events: true, active: true) }
43
  scope :external_issue_trackers, -> { issue_trackers.active.without_defaults }
44

45 46
  default_value_for :category, 'common'

47 48 49
  def activated?
    active
  end
50

M
Marin Jankovski 已提交
51 52 53 54
  def template?
    template
  end

55
  def category
56
    read_attribute(:category).to_sym
57 58
  end

59 60 61 62
  def initialize_properties
    self.properties = {} if properties.nil?
  end

63
  def title
64
    # implement inside child
65 66 67
  end

  def description
68
    # implement inside child
69 70
  end

71 72 73 74
  def help
    # implement inside child
  end

75
  def to_param
76
    # implement inside child
77 78 79
  end

  def fields
80
    # implement inside child
81 82
    []
  end
83

84
  def test_data(project, user)
85
    Gitlab::DataBuilder::Push.build_sample(project, user)
86 87
  end

88 89 90 91
  def event_channel_names
    []
  end

92 93 94 95
  def event_names
    supported_events.map { |event| "#{event}_events" }
  end

F
Felipe Artur 已提交
96 97 98 99 100 101 102 103
  def event_field(event)
    nil
  end

  def global_fields
    fields
  end

104
  def supported_events
105
    %w(push tag_push issue confidential_issue merge_request wiki_page)
106 107
  end

108
  def execute(data)
109 110
    # implement inside child
  end
111

112 113 114 115 116 117
  def test(data)
    # default implementation
    result = execute(data)
    { success: result.present?, result: result }
  end

118 119 120
  def can_test?
    !project.empty_repo?
  end
D
Drew Blessing 已提交
121

122 123 124 125 126
  # reason why service cannot be tested
  def disabled_title
    "Please setup a project repository."
  end

D
Drew Blessing 已提交
127 128
  # Provide convenient accessor methods
  # for each serialized property.
129
  # Also keep track of updated properties in a similar way as ActiveModel::Dirty
D
Drew Blessing 已提交
130 131 132 133 134 135 136 137
  def self.prop_accessor(*args)
    args.each do |arg|
      class_eval %{
        def #{arg}
          properties['#{arg}']
        end

        def #{arg}=(value)
138
          updated_properties['#{arg}'] = #{arg} unless #{arg}_changed?
D
Drew Blessing 已提交
139 140
          self.properties['#{arg}'] = value
        end
141 142 143 144 145 146 147 148 149 150 151 152

        def #{arg}_changed?
          #{arg}_touched? && #{arg} != #{arg}_was
        end

        def #{arg}_touched?
          updated_properties.include?('#{arg}')
        end

        def #{arg}_was
          updated_properties['#{arg}']
        end
D
Drew Blessing 已提交
153 154 155
      }
    end
  end
156

157 158 159 160 161 162 163 164
  # Provide convenient boolean accessor methods
  # for each serialized property.
  # Also keep track of updated properties in a similar way as ActiveModel::Dirty
  def self.boolean_accessor(*args)
    self.prop_accessor(*args)

    args.each do |arg|
      class_eval %{
165 166 167 168
        def #{arg}?
          ActiveRecord::ConnectionAdapters::Column::TRUE_VALUES.include?(#{arg})
        end
      }
169 170 171
    end
  end

172 173
  # Returns a hash of the properties that have been assigned a new value since last save,
  # indicating their original values (attr => original value).
174
  # ActiveRecord does not provide a mechanism to track changes in serialized keys,
175 176 177 178 179
  # so we need a specific implementation for service properties.
  # This allows to track changes to properties set with the accessor methods,
  # but not direct manipulation of properties hash.
  def updated_properties
    @updated_properties ||= ActiveSupport::HashWithIndifferentAccess.new
180 181
  end

182 183 184
  def reset_updated_properties
    @updated_properties = nil
  end
185

186
  def async_execute(data)
187
    return unless supported_events.include?(data[:object_kind])
188

189 190
    Sidekiq::Client.enqueue(ProjectServiceWorker, id, data)
  end
191 192 193 194 195

  def issue_tracker?
    self.category == :issue_tracker
  end

196
  def self.available_services_names
197 198
    %w(
      asana
199 200
      assembla
      bamboo
201
      buildkite
202
      builds_email
203
      bugzilla
204 205
      campfire
      custom_issue_tracker
K
Kirilll Zaitsev 已提交
206
      drone_ci
207
      emails_on_push
208 209
      external_wiki
      flowdock
210
      gemnasium
211 212
      hipchat
      irker
213
      jira
214 215
      pivotaltracker
      pushover
216
      redmine
217 218
      slack
      teamcity
219
    )
220 221
  end

M
Marin Jankovski 已提交
222 223 224 225 226
  def self.create_from_template(project_id, template)
    service = template.dup
    service.template = false
    service.project_id = project_id
    service if service.save
227
  end
228 229 230 231 232 233 234 235

  private

  def cache_project_has_external_issue_tracker
    if project && !project.destroyed?
      project.cache_has_external_issue_tracker
    end
  end
236 237 238 239 240 241

  def cache_project_has_external_wiki
    if project && !project.destroyed?
      project.cache_has_external_wiki
    end
  end
242
end