app/models/concerns/integrations/base/integration.rb (603 lines of code) (raw):

# frozen_string_literal: true module Integrations module Base module Integration extend ActiveSupport::Concern extend Gitlab::Utils::Override UnknownType = Class.new(StandardError) INTEGRATION_NAMES = %w[ asana assembla bamboo bugzilla buildkite campfire clickup confluence custom_issue_tracker datadog diffblue_cover discord drone_ci emails_on_push ewm external_wiki gitlab_slack_application hangouts_chat harbor irker jira matrix mattermost mattermost_slash_commands microsoft_teams packagist phorge pipelines_email pivotaltracker prometheus pumble pushover redmine slack slack_slash_commands squash_tm teamcity telegram unify_circuit webex_teams youtrack zentao ].freeze # Integrations that can only be enabled on the instance-level INSTANCE_LEVEL_ONLY_INTEGRATION_NAMES = %w[ beyond_identity ].freeze # Integrations that can only be enabled on the project-level PROJECT_LEVEL_ONLY_INTEGRATION_NAMES = %w[ apple_app_store google_play jenkins ].freeze # Integrations that cannot be enabled on the instance-level PROJECT_AND_GROUP_LEVEL_ONLY_INTEGRATION_NAMES = %w[ jira_cloud_app ].freeze # Fake integrations to help with local development. DEV_INTEGRATION_NAMES = %w[ mock_ci mock_monitoring ].freeze # Base classes which aren't actual integrations. BASE_CLASSES = %w[].freeze BASE_ATTRIBUTES = %w[id instance project_id group_id created_at updated_at encrypted_properties encrypted_properties_iv properties organization_id].freeze SECTION_TYPE_CONFIGURATION = 'configuration' SECTION_TYPE_CONNECTION = 'connection' SECTION_TYPE_TRIGGER = 'trigger' SNOWPLOW_EVENT_ACTION = 'perform_integrations_action' SNOWPLOW_EVENT_LABEL = 'redis_hll_counters.ecosystem.ecosystem_total_unique_counts_monthly' class_methods do extend Gitlab::Utils::Override def supported_events %w[commit push tag_push issue confidential_issue merge_request wiki_page] end # rubocop:disable Gitlab/NoCodeCoverageComment -- existing code moved as is # :nocov: Tested on subclasses. def field(name, storage: field_storage, **attrs) fields << ::Integrations::Field.new(name: name, integration_class: self, **attrs) case storage when :attribute # noop when :properties prop_accessor(name) when :data_fields data_field(name) else raise ArgumentError, "Unknown field storage: #{storage}" end boolean_accessor(name) if attrs[:type] == :checkbox && storage != :attribute end # :nocov: # rubocop:enable Gitlab/NoCodeCoverageComment def fields @fields ||= [] end # Provide convenient accessor methods for each serialized property. # Also keep track of updated properties in a similar way as ActiveModel::Dirty def prop_accessor(*args) args.each do |arg| class_eval <<~RUBY, __FILE__, __LINE__ + 1 unless method_defined?(arg) def #{arg} properties['#{arg}'] if properties.present? end end def #{arg}=(value) self.properties ||= {} updated_properties['#{arg}'] = #{arg} unless #{arg}_changed? self.properties = self.properties.merge('#{arg}' => value) end def #{arg}_changed? #{arg}_touched? && #{arg} != #{arg}_was end def #{arg}_touched? updated_properties.include?('#{arg}') end def #{arg}_was updated_properties['#{arg}'] end RUBY end end # Provide convenient boolean accessor methods for each serialized property. # Also keep track of updated properties in a similar way as ActiveModel::Dirty def boolean_accessor(*args) args.each do |arg| class_eval <<~RUBY, __FILE__, __LINE__ + 1 # Make the original getter available as a private method. alias_method :#{arg}_before_type_cast, :#{arg} private(:#{arg}_before_type_cast) def #{arg} Gitlab::Utils.to_boolean(#{arg}_before_type_cast) end def #{arg}? # '!!' is used because nil or empty string is converted to nil !!#{arg} end RUBY end end def title raise NotImplementedError end def description raise NotImplementedError end def help # no-op end def to_param raise NotImplementedError end def attribution_notice # no-op end def event_names supported_events.map { |event| IntegrationsHelper.integration_event_field_name(event) } end def default_test_event 'push' end def event_description(event) IntegrationsHelper.integration_event_description(event) end def find_or_initialize_non_project_specific_integration(name, instance: false, group_id: nil) return unless name.in?(available_integration_names( include_project_specific: false, include_group_specific: group_id.present?, include_instance_specific: instance)) integration_name_to_model(name).find_or_initialize_by(instance: instance, group_id: group_id) end def find_or_initialize_all_non_project_specific(scope, include_instance_specific: false) scope + build_nonexistent_integrations_for(scope, include_group_specific: !include_instance_specific, include_instance_specific: include_instance_specific) end def build_nonexistent_integrations_for(...) nonexistent_integration_types_for(...).map do |type| integration_type_to_model(type).new end end # Returns a list of integration types that do not exist in the given scope. # Example: ["AsanaService", ...] def nonexistent_integration_types_for( scope, include_group_specific: false, include_instance_specific: false) # Using #map instead of #pluck to save one query count. This is because # ActiveRecord loaded the object here, so we don't need to query again later. available_integration_types( include_project_specific: false, include_group_specific: include_group_specific, include_instance_specific: include_instance_specific ) - scope.map(&:type) end # Returns a list of available integration names. # Example: ["asana", ...] def available_integration_names( include_project_specific: true, include_group_specific: true, include_instance_specific: true, include_dev: true, include_disabled: false) names = integration_names.dup names.concat(project_specific_integration_names) if include_project_specific names.concat(dev_integration_names) if include_dev names.concat(instance_specific_integration_names) if include_instance_specific if include_project_specific || include_group_specific names.concat(project_and_group_specific_integration_names) end names -= disabled_integration_names unless include_disabled names.sort_by(&:downcase) end def integration_names names = INTEGRATION_NAMES.dup unless Gitlab::CurrentSettings.slack_app_enabled || Gitlab.dev_or_test_env? names.delete('gitlab_slack_application') end names end def instance_specific_integration_names INSTANCE_LEVEL_ONLY_INTEGRATION_NAMES end def instance_specific_integration_types instance_specific_integration_names.map { |name| integration_name_to_type(name) } end def dev_integration_names return [] unless Gitlab.dev_or_test_env? DEV_INTEGRATION_NAMES end def project_specific_integration_names PROJECT_LEVEL_ONLY_INTEGRATION_NAMES.dup end def project_and_group_specific_integration_names PROJECT_AND_GROUP_LEVEL_ONLY_INTEGRATION_NAMES.dup end # Returns a list of available integration types. # Example: ["Integrations::Asana", ...] def available_integration_types(...) available_integration_names(...).map do # rubocop:disable Style/NumberedParameters -- existing code moved as is integration_name_to_type(_1) end end # Returns a list of disabled integration names. # Example: ["gitlab_slack_application", ...] def disabled_integration_names # The GitLab for Slack app integration is only available when enabled through settings. # The Slack Slash Commands integration is only available for customers # who cannot use the GitLab for Slack app. disabled = Gitlab::CurrentSettings.slack_app_enabled ? ['slack_slash_commands'] : ['gitlab_slack_application'] disabled += ['jira_cloud_app'] unless Gitlab::CurrentSettings.jira_connect_application_key.present? disabled end # Returns the model for the given integration name. # Example: :asana => Integrations::Asana def integration_name_to_model(name) type = integration_name_to_type(name) integration_type_to_model(type) end # Returns the STI type for the given integration name. # Example: "asana" => "Integrations::Asana" def integration_name_to_type(name) name = name.to_s if all_integration_names.include?(name) "Integrations::#{name.camelize}" else Gitlab::ErrorTracking.track_and_raise_for_dev_exception(UnknownType.new(name.inspect)) end end # Returns the model for the given STI type. # Example: "Integrations::Asana" => Integrations::Asana def integration_type_to_model(type) type.constantize end def build_from_integration(integration, project_id: nil, group_id: nil) new_integration = integration.dup new_integration.instance = false new_integration.project_id = project_id new_integration.group_id = group_id new_integration.inherit_from_id = integration.id if integration.inheritable? new_integration end def instance_exists_for?(type) exists?(instance: true, type: type) end # Returns the names of all integrations, including: # # - All project, group and instance-level only integrations # - Integrations that are not available on the instance # - Development-only integrations def all_integration_names available_integration_names(include_disabled: true) end def default_integration(type, scope) closest_group_integration(type, scope) || instance_level_integration(type) end def closest_group_integration(type, scope) group_ids = scope.ancestors(hierarchy_order: :asc).reselect(:id) array = group_ids.to_sql.present? ? "array(#{group_ids.to_sql})" : 'ARRAY[]' where(type: type, group_id: group_ids, inherit_from_id: nil) .order(Arel.sql("array_position(#{array}::bigint[], #{table_name}.group_id)")) .first end def instance_level_integration(type) find_by(type: type, instance: true) end def default_integrations(owner, scope) group_ids = sorted_ancestors(owner).select(:id) array = group_ids.to_sql.present? ? "array(#{group_ids.to_sql})" : 'ARRAY[]' order = Arel.sql("type_new ASC, array_position(#{array}::bigint[], #{table_name}.group_id), instance DESC") from_union([scope.where(instance: true), scope.where(group_id: group_ids, inherit_from_id: nil)]) .order(order) .group_by(&:type) .transform_values(&:first) end def create_from_default_integrations(owner, association) active_default_count = create_from_active_default_integrations(owner, association) default_instance_specific_count = create_from_default_instance_specific_integrations(owner, association) active_default_count + default_instance_specific_count end # Returns the number of successfully saved integrations # Duplicate integrations are excluded from this count by their validations. def create_from_active_default_integrations(owner, association) default_integrations( owner, active.where.not(type: instance_specific_integration_types) ).count { |_type, integration| build_from_integration(integration, association => owner.id).save } end def create_from_default_instance_specific_integrations(owner, association) default_integrations( owner, where(type: instance_specific_integration_types) ).count { |_type, integration| build_from_integration(integration, association => owner.id).save } end def descendants_from_self_or_ancestors_from(integration) scope = where(type: integration.type) from_union([ scope.where(group: integration.group.descendants), scope.where(project: Project.in_namespace(integration.group.self_and_descendants)) ]) end def inherited_descendants_from_self_or_ancestors_from(integration) inherit_from_ids = where(type: integration.type, group: integration.group.self_and_ancestors) .or(where(type: integration.type, instance: true)).select(:id) from_union([ where(type: integration.type, inherit_from_id: inherit_from_ids, group: integration.group.descendants), where( type: integration.type, inherit_from_id: inherit_from_ids, project: Project.in_namespace(integration.group.self_and_descendants) ) ]) end def api_arguments fields.filter_map do |field| next if field.if != true { required: field.required?, name: field.name.to_sym, type: field.api_type, desc: field.description } end end def instance_specific? false end def pluck_group_id pluck(:group_id) # rubocop:disable Database/AvoidUsingPluckWithoutLimit -- existing code moved as is end private attr_writer :field_storage def field_storage @field_storage || :properties end def build_help_page_url(url_path, help_text, options = {}, link_text: _("Learn More")) docs_link = ActionController::Base.helpers.link_to( '', Rails.application.routes.url_helpers.help_page_url(url_path, **options), # rubocop:disable Gitlab/DocumentationLinks/Link: -- existing code moved as is target: '_blank', rel: 'noopener noreferrer' ) tag_pair_docs_link = tag_pair(docs_link, :link_start, :link_end) safe_format(help_text + " %{link_start}#{link_text}%{link_end}.", tag_pair_docs_link) end # Ancestors sorted by hierarchy depth in bottom-top order. def sorted_ancestors(scope) if scope.root_ancestor.use_traversal_ids? Namespace.from(scope.ancestors(hierarchy_order: :asc)) else scope.ancestors end end end included do include Sortable include Importable include Integrations::Loggable include Integrations::HasDataFields include Integrations::ResetSecretFields include FromUnion include EachBatch include Gitlab::EncryptedAttribute extend SafeFormatHelper extend ::Gitlab::Utils::Override self.allow_legacy_sti_class = true self.inheritance_column = :type_new # rubocop:disable Database/AvoidInheritanceColumn -- existing code moved as is attr_encrypted :properties, mode: :per_attribute_iv, key: :db_key_base_32, algorithm: 'aes-256-gcm', marshal: true, marshaler: ::Gitlab::Json, encode: false, encode_iv: false alias_attribute :name, :title # Handle assignment of props with symbol keys. # To do this correctly, we need to call the method generated by attr_encrypted. alias_method :attr_encrypted_props=, :properties= private :attr_encrypted_props= def properties=(props) self.attr_encrypted_props = props&.with_indifferent_access&.freeze end alias_attribute :type, :type_new attribute :active, default: false attribute :alert_events, default: true attribute :incident_events, default: false attribute :category, default: 'common' attribute :commit_events, default: true attribute :confidential_issues_events, default: true attribute :confidential_note_events, default: true attribute :deployment_events, default: false attribute :issues_events, default: true attribute :job_events, default: true attribute :merge_requests_events, default: true attribute :note_events, default: true attribute :pipeline_events, default: true attribute :push_events, default: true attribute :tag_push_events, default: true attribute :wiki_page_events, default: true attribute :group_mention_events, default: false attribute :group_confidential_mention_events, default: false after_initialize :initialize_properties after_commit :reset_updated_properties belongs_to :project, inverse_of: :integrations belongs_to :group, inverse_of: :integrations belongs_to :organization, class_name: 'Organizations::Organization', inverse_of: :integrations, optional: true validates :project_id, presence: true, unless: -> { instance_level? || group_level? } validates :group_id, presence: true, unless: -> { instance_level? || project_level? } validates :project_id, :group_id, absence: true, if: -> { instance_level? } validates :type, presence: true, exclusion: BASE_CLASSES validates :type, uniqueness: { scope: :instance }, if: :instance_level? validates :type, uniqueness: { scope: :project_id }, if: :project_level? validates :type, uniqueness: { scope: :group_id }, if: :group_level? validate :validate_belongs_to_project_or_group scope :external_issue_trackers, -> { where(category: 'issue_tracker').active } scope :third_party_wikis, -> { where(category: 'third_party_wiki').active } scope :by_name, ->(name) { by_type(integration_name_to_type(name)) } scope :external_wikis, -> { by_name(:external_wiki).active } scope :active, -> { where(active: true) } scope :by_type, ->(type) { where(type: type) } # INTERNAL USE ONLY: use by_name instead scope :by_active_flag, ->(flag) { where(active: flag) } scope :inherit_from_id, ->(id) { where(inherit_from_id: id) } scope :with_default_settings, -> { where.not(inherit_from_id: nil) } scope :with_custom_settings, -> { where(inherit_from_id: nil) } scope :for_group, ->(group) { types = available_integration_types(include_project_specific: false) where(group_id: group, type: types) } scope :for_instance, -> { types = available_integration_types(include_project_specific: false, include_group_specific: false) where(instance: true, type: types) } scope :push_hooks, -> { where(push_events: true).active } scope :tag_push_hooks, -> { where(tag_push_events: true).active } scope :issue_hooks, -> { where(issues_events: true).active } scope :confidential_issue_hooks, -> { where(confidential_issues_events: true).active } scope :merge_request_hooks, -> { where(merge_requests_events: true).active } scope :note_hooks, -> { where(note_events: true).active } scope :confidential_note_hooks, -> { where(confidential_note_events: true).active } scope :job_hooks, -> { where(job_events: true).active } scope :archive_trace_hooks, -> { where(archive_trace_events: true).active } scope :pipeline_hooks, -> { where(pipeline_events: true).active } scope :wiki_page_hooks, -> { where(wiki_page_events: true).active } scope :deployment_hooks, -> { where(deployment_events: true).active } scope :alert_hooks, -> { where(alert_events: true).active } scope :incident_hooks, -> { where(incident_events: true).active } scope :deployment, -> { where(category: 'deployment') } scope :group_mention_hooks, -> { where(group_mention_events: true).active } scope :group_confidential_mention_hooks, -> { where(group_confidential_mention_events: true).active } scope :exclusions_for_project, ->(project) { where(project: project, active: false) } private_class_method :boolean_accessor private_class_method :build_nonexistent_integrations_for private_class_method :nonexistent_integration_types_for private_class_method :project_and_group_specific_integration_names private_class_method :disabled_integration_names private_class_method :integration_type_to_model private_class_method :default_integrations private_class_method :instance_level_integration private_class_method :closest_group_integration end def fields self.class.fields.dup end # Duplicating an integration also duplicates the data fields. Duped records have different ciphertexts. override :dup def dup new_integration = super new_integration.assign_attributes(reencrypt_properties) if supports_data_fields? fields = data_fields.dup fields.integration = new_integration end new_integration end def inheritable? instance_level? || group_level? end def activated? active end def operating? active && persisted? end def manual_activation? true end def editable? true end def activate_disabled_reason nil end def category read_attribute(:category).to_sym end def initialize_properties self.properties = {} if has_attribute?(:encrypted_properties) && encrypted_properties.nil? end def title self.class.title end def description self.class.description end def help self.class.help end def to_param self.class.to_param end def attribution_notice self.class.attribution_notice end def sections [] end def secret_fields fields.select(&:secret?).pluck(:name) # rubocop:disable Database/AvoidUsingPluckWithoutLimit -- existing code moved as is end # Expose a list of fields in the JSON endpoint. # # This list is used in `Integration#as_json(only: json_fields)`. def json_fields %w[active] end # properties is always nil - ignore it. override :attributes def attributes super.except('properties') end # Returns a hash of attributes (columns => values) used for inserting into the database. def to_database_hash column = self.class.attribute_aliases.fetch('type', 'type') attributes_for_database .except(*BASE_ATTRIBUTES) .merge(column => type) .merge(reencrypt_properties) end def reencrypt_properties unless properties.nil? || properties.empty? attr_encrypted_attributes = self.class.attr_encrypted_encrypted_attributes[:properties] key = dynamic_encryption_key_for_operation(attr_encrypted_attributes[:key]) iv = generate_iv(attr_encrypted_attributes[:algorithm]) ep = self.class.attr_encrypted_encrypt(:properties, properties, { key: key, iv: iv }) end { 'encrypted_properties' => ep, 'encrypted_properties_iv' => iv } end def event_channel_names [] end def event_names self.class.event_names end def api_field_names fields # rubocop:disable Style/NumberedParameters -- existing code moved as is .reject { _1[:type] == :password || _1[:name] == 'webhook' || (_1.key?(:if) && _1[:if] != true) } .pluck(:name) # rubocop:disable Database/AvoidUsingPluckWithoutLimit -- existing code moved as is end def form_fields fields.reject { _1[:api_only] == true || (_1.key?(:if) && _1[:if] != true) } # rubocop:disable Style/NumberedParameters -- existing code moved as is end def configurable_events events = supported_events # No need to disable individual triggers when there is only one if events.count == 1 [] else events end end def supported_events self.class.supported_events end def default_test_event self.class.default_test_event end def execute(data) # implement inside child end def test(data) # default implementation result = execute(data) { success: result.present?, result: result } end # Disable test for instance-level and group-level integrations. # https://gitlab.com/gitlab-org/gitlab/-/issues/213138 def testable? project_level? end def project_level? project_id.present? end def group_level? group_id.present? end def instance_level? instance? end def parent project || group end # Returns a hash of the properties that have been assigned a new value since last save, # indicating their original values (attr => original value). # ActiveRecord does not provide a mechanism to track changes in serialized keys, # so we need a specific implementation for integration 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 end def reset_updated_properties @updated_properties = nil # rubocop:disable Gitlab/ModuleWithInstanceVariables -- legacy use end def async_execute(data) return if ::Gitlab::SilentMode.enabled? return unless active? data = data.with_indifferent_access # Temporarily log when we return within this method to gather data for # https://gitlab.com/gitlab-org/gitlab/-/issues/382999 unless supported_events.include?(data[:object_kind].to_s) log_info( 'async_execute did nothing due to event not being supported', event: data[:object_kind] ) return end Integrations::ExecuteWorker.perform_async(id, data.deep_stringify_keys) end # override if needed def supports_data_fields? false end def chat? category == :chat end def ci? category == :ci end def deactivate! update(active: false) end def activate! update(active: true) end def toggle! active? ? deactivate! : activate! end private def validate_belongs_to_project_or_group return unless project_level? && group_level? errors.add(:project_id, 'The integration cannot belong to both a project and a group') end def validate_recipients? activated? && !importing? end end end end Integration.prepend_mod_with('Integrations::Base::Integration')