2018-11-18 11:00:15 +05:30
|
|
|
# frozen_string_literal: true
|
|
|
|
|
2014-09-02 18:07:02 +05:30
|
|
|
# EventCreateService class
|
|
|
|
#
|
|
|
|
# Used for creating events feed on dashboard after certain user action
|
|
|
|
#
|
|
|
|
# Ex.
|
|
|
|
# EventCreateService.new.new_issue(issue, current_user)
|
|
|
|
#
|
|
|
|
class EventCreateService
|
2020-04-22 19:07:51 +05:30
|
|
|
IllegalActionError = Class.new(StandardError)
|
|
|
|
|
2014-09-02 18:07:02 +05:30
|
|
|
def open_issue(issue, current_user)
|
2020-06-23 00:09:42 +05:30
|
|
|
create_record_event(issue, current_user, :created)
|
2014-09-02 18:07:02 +05:30
|
|
|
end
|
|
|
|
|
|
|
|
def close_issue(issue, current_user)
|
2020-06-23 00:09:42 +05:30
|
|
|
create_record_event(issue, current_user, :closed)
|
2014-09-02 18:07:02 +05:30
|
|
|
end
|
|
|
|
|
|
|
|
def reopen_issue(issue, current_user)
|
2020-06-23 00:09:42 +05:30
|
|
|
create_record_event(issue, current_user, :reopened)
|
2014-09-02 18:07:02 +05:30
|
|
|
end
|
|
|
|
|
|
|
|
def open_mr(merge_request, current_user)
|
2020-11-24 15:15:51 +05:30
|
|
|
create_record_event(merge_request, current_user, :created).tap do
|
|
|
|
track_event(event_action: :created, event_target: MergeRequest, author_id: current_user.id)
|
2023-01-13 00:05:48 +05:30
|
|
|
track_snowplow_event(
|
|
|
|
:created,
|
|
|
|
merge_request,
|
|
|
|
current_user
|
|
|
|
)
|
2020-11-24 15:15:51 +05:30
|
|
|
end
|
2014-09-02 18:07:02 +05:30
|
|
|
end
|
|
|
|
|
|
|
|
def close_mr(merge_request, current_user)
|
2020-11-24 15:15:51 +05:30
|
|
|
create_record_event(merge_request, current_user, :closed).tap do
|
|
|
|
track_event(event_action: :closed, event_target: MergeRequest, author_id: current_user.id)
|
2023-01-13 00:05:48 +05:30
|
|
|
track_snowplow_event(
|
|
|
|
:closed,
|
|
|
|
merge_request,
|
|
|
|
current_user
|
|
|
|
)
|
2020-11-24 15:15:51 +05:30
|
|
|
end
|
2014-09-02 18:07:02 +05:30
|
|
|
end
|
|
|
|
|
|
|
|
def reopen_mr(merge_request, current_user)
|
2020-06-23 00:09:42 +05:30
|
|
|
create_record_event(merge_request, current_user, :reopened)
|
2014-09-02 18:07:02 +05:30
|
|
|
end
|
|
|
|
|
|
|
|
def merge_mr(merge_request, current_user)
|
2020-11-24 15:15:51 +05:30
|
|
|
create_record_event(merge_request, current_user, :merged).tap do
|
|
|
|
track_event(event_action: :merged, event_target: MergeRequest, author_id: current_user.id)
|
2023-01-13 00:05:48 +05:30
|
|
|
track_snowplow_event(
|
|
|
|
:merged,
|
|
|
|
merge_request,
|
|
|
|
current_user
|
|
|
|
)
|
2020-11-24 15:15:51 +05:30
|
|
|
end
|
2014-09-02 18:07:02 +05:30
|
|
|
end
|
|
|
|
|
|
|
|
def open_milestone(milestone, current_user)
|
2020-06-23 00:09:42 +05:30
|
|
|
create_record_event(milestone, current_user, :created)
|
2014-09-02 18:07:02 +05:30
|
|
|
end
|
|
|
|
|
|
|
|
def close_milestone(milestone, current_user)
|
2020-06-23 00:09:42 +05:30
|
|
|
create_record_event(milestone, current_user, :closed)
|
2014-09-02 18:07:02 +05:30
|
|
|
end
|
|
|
|
|
|
|
|
def reopen_milestone(milestone, current_user)
|
2020-06-23 00:09:42 +05:30
|
|
|
create_record_event(milestone, current_user, :reopened)
|
2014-09-02 18:07:02 +05:30
|
|
|
end
|
|
|
|
|
2015-09-25 12:07:36 +05:30
|
|
|
def destroy_milestone(milestone, current_user)
|
2020-06-23 00:09:42 +05:30
|
|
|
create_record_event(milestone, current_user, :destroyed)
|
2015-09-25 12:07:36 +05:30
|
|
|
end
|
|
|
|
|
2014-09-02 18:07:02 +05:30
|
|
|
def leave_note(note, current_user)
|
2020-11-24 15:15:51 +05:30
|
|
|
create_record_event(note, current_user, :commented).tap do
|
|
|
|
if note.is_a?(DiffNote) && note.for_merge_request?
|
|
|
|
track_event(event_action: :commented, event_target: MergeRequest, author_id: current_user.id)
|
2023-01-13 00:05:48 +05:30
|
|
|
track_snowplow_event(
|
|
|
|
:commented,
|
|
|
|
note,
|
|
|
|
current_user
|
|
|
|
)
|
|
|
|
|
2020-11-24 15:15:51 +05:30
|
|
|
end
|
|
|
|
end
|
2015-04-26 12:48:37 +05:30
|
|
|
end
|
|
|
|
|
|
|
|
def join_project(project, current_user)
|
2020-06-23 00:09:42 +05:30
|
|
|
create_event(project, current_user, :joined)
|
2015-04-26 12:48:37 +05:30
|
|
|
end
|
|
|
|
|
|
|
|
def leave_project(project, current_user)
|
2020-06-23 00:09:42 +05:30
|
|
|
create_event(project, current_user, :left)
|
2015-04-26 12:48:37 +05:30
|
|
|
end
|
|
|
|
|
2017-08-17 22:00:37 +05:30
|
|
|
def expired_leave_project(project, current_user)
|
2020-06-23 00:09:42 +05:30
|
|
|
create_event(project, current_user, :expired)
|
2017-08-17 22:00:37 +05:30
|
|
|
end
|
|
|
|
|
2015-04-26 12:48:37 +05:30
|
|
|
def create_project(project, current_user)
|
2020-06-23 00:09:42 +05:30
|
|
|
create_event(project, current_user, :created)
|
2015-04-26 12:48:37 +05:30
|
|
|
end
|
|
|
|
|
|
|
|
def push(project, current_user, push_data)
|
2019-12-21 20:55:43 +05:30
|
|
|
create_push_event(PushEventPayloadService, project, current_user, push_data)
|
|
|
|
end
|
|
|
|
|
|
|
|
def bulk_push(project, current_user, push_data)
|
|
|
|
create_push_event(BulkPushEventPayloadService, project, current_user, push_data)
|
|
|
|
end
|
|
|
|
|
2020-06-23 00:09:42 +05:30
|
|
|
def save_designs(current_user, create: [], update: [])
|
2020-07-28 23:09:34 +05:30
|
|
|
records = create.zip([:created].cycle) + update.zip([:updated].cycle)
|
|
|
|
return [] if records.empty?
|
2020-06-23 00:09:42 +05:30
|
|
|
|
2022-08-13 15:12:31 +05:30
|
|
|
if create.any?
|
2023-01-13 00:05:48 +05:30
|
|
|
old_track_snowplow_event(create.first, current_user,
|
2022-08-13 15:12:31 +05:30
|
|
|
Gitlab::UsageDataCounters::TrackUniqueEvents::DESIGN_ACTION,
|
|
|
|
:create, 'design_users')
|
|
|
|
end
|
|
|
|
|
|
|
|
if update.any?
|
2023-01-13 00:05:48 +05:30
|
|
|
old_track_snowplow_event(update.first, current_user,
|
2022-08-13 15:12:31 +05:30
|
|
|
Gitlab::UsageDataCounters::TrackUniqueEvents::DESIGN_ACTION,
|
|
|
|
:update, 'design_users')
|
|
|
|
end
|
|
|
|
|
2020-06-23 00:09:42 +05:30
|
|
|
create_record_events(records, current_user)
|
|
|
|
end
|
|
|
|
|
|
|
|
def destroy_designs(designs, current_user)
|
|
|
|
return [] unless designs.present?
|
|
|
|
|
2023-01-13 00:05:48 +05:30
|
|
|
old_track_snowplow_event(designs.first, current_user,
|
2022-08-13 15:12:31 +05:30
|
|
|
Gitlab::UsageDataCounters::TrackUniqueEvents::DESIGN_ACTION,
|
|
|
|
:destroy, 'design_users')
|
2020-06-23 00:09:42 +05:30
|
|
|
create_record_events(designs.zip([:destroyed].cycle), current_user)
|
|
|
|
end
|
|
|
|
|
2020-04-22 19:07:51 +05:30
|
|
|
# Create a new wiki page event
|
|
|
|
#
|
|
|
|
# @param [WikiPage::Meta] wiki_page_meta The event target
|
2020-05-24 23:13:21 +05:30
|
|
|
# @param [User] author The event author
|
2020-06-23 00:09:42 +05:30
|
|
|
# @param [Symbol] action One of the Event::WIKI_ACTIONS
|
2020-10-24 23:57:45 +05:30
|
|
|
# @param [String] fingerprint The de-duplication fingerprint
|
2020-05-24 23:13:21 +05:30
|
|
|
#
|
2020-10-24 23:57:45 +05:30
|
|
|
# The fingerprint, if provided, should be sufficient to find duplicate events.
|
|
|
|
# Suitable values would be, for example, the current page SHA.
|
|
|
|
#
|
|
|
|
# @return [Event] the event
|
|
|
|
def wiki_event(wiki_page_meta, author, action, fingerprint)
|
2020-04-22 19:07:51 +05:30
|
|
|
raise IllegalActionError, action unless Event::WIKI_ACTIONS.include?(action)
|
|
|
|
|
2020-11-24 15:15:51 +05:30
|
|
|
track_event(event_action: action, event_target: wiki_page_meta.class, author_id: author.id)
|
2020-05-24 23:13:21 +05:30
|
|
|
|
2020-10-24 23:57:45 +05:30
|
|
|
duplicate = Event.for_wiki_meta(wiki_page_meta).for_fingerprint(fingerprint).first
|
|
|
|
return duplicate if duplicate.present?
|
2020-05-24 23:13:21 +05:30
|
|
|
|
2020-10-24 23:57:45 +05:30
|
|
|
create_record_event(wiki_page_meta, author, action, fingerprint.presence)
|
2020-04-22 19:07:51 +05:30
|
|
|
end
|
|
|
|
|
2020-07-28 23:09:34 +05:30
|
|
|
def approve_mr(merge_request, current_user)
|
|
|
|
create_record_event(merge_request, current_user, :approved)
|
|
|
|
end
|
|
|
|
|
2019-12-21 20:55:43 +05:30
|
|
|
private
|
|
|
|
|
2020-10-24 23:57:45 +05:30
|
|
|
def create_record_event(record, current_user, status, fingerprint = nil)
|
2020-06-23 00:09:42 +05:30
|
|
|
create_event(record.resource_parent, current_user, status,
|
2020-10-24 23:57:45 +05:30
|
|
|
fingerprint: fingerprint,
|
|
|
|
target_id: record.id,
|
|
|
|
target_type: record.class.name)
|
2020-06-23 00:09:42 +05:30
|
|
|
end
|
|
|
|
|
|
|
|
# If creating several events, this method will insert them all in a single
|
|
|
|
# statement
|
|
|
|
#
|
2020-10-24 23:57:45 +05:30
|
|
|
# @param [[Eventable, Symbol, String]] a list of tuples of records, a valid status, and fingerprint
|
2020-06-23 00:09:42 +05:30
|
|
|
# @param [User] the author of the event
|
2020-10-24 23:57:45 +05:30
|
|
|
def create_record_events(tuples, current_user)
|
2020-06-23 00:09:42 +05:30
|
|
|
base_attrs = {
|
|
|
|
created_at: Time.now.utc,
|
|
|
|
updated_at: Time.now.utc,
|
|
|
|
author_id: current_user.id
|
|
|
|
}
|
|
|
|
|
2020-10-24 23:57:45 +05:30
|
|
|
attribute_sets = tuples.map do |record, status, fingerprint|
|
2020-06-23 00:09:42 +05:30
|
|
|
action = Event.actions[status]
|
|
|
|
raise IllegalActionError, "#{status} is not a valid status" if action.nil?
|
|
|
|
|
|
|
|
parent_attrs(record.resource_parent)
|
|
|
|
.merge(base_attrs)
|
2020-10-24 23:57:45 +05:30
|
|
|
.merge(action: action, fingerprint: fingerprint, target_id: record.id, target_type: record.class.name)
|
2020-06-23 00:09:42 +05:30
|
|
|
end
|
|
|
|
|
2020-07-28 23:09:34 +05:30
|
|
|
result = Event.insert_all(attribute_sets, returning: %w[id])
|
|
|
|
|
2020-10-24 23:57:45 +05:30
|
|
|
tuples.each do |record, status, _|
|
2020-11-24 15:15:51 +05:30
|
|
|
track_event(event_action: status, event_target: record.class, author_id: current_user.id)
|
2020-07-28 23:09:34 +05:30
|
|
|
end
|
|
|
|
|
|
|
|
result
|
2019-12-21 20:55:43 +05:30
|
|
|
end
|
|
|
|
|
|
|
|
def create_push_event(service_class, project, current_user, push_data)
|
2017-09-10 17:25:29 +05:30
|
|
|
# We're using an explicit transaction here so that any errors that may occur
|
|
|
|
# when creating push payload data will result in the event creation being
|
|
|
|
# rolled back as well.
|
2018-03-17 18:26:18 +05:30
|
|
|
event = Event.transaction do
|
2020-06-23 00:09:42 +05:30
|
|
|
new_event = create_event(project, current_user, :pushed)
|
2017-09-10 17:25:29 +05:30
|
|
|
|
2019-12-21 20:55:43 +05:30
|
|
|
service_class.new(new_event, push_data).execute
|
2018-03-17 18:26:18 +05:30
|
|
|
|
|
|
|
new_event
|
2017-09-10 17:25:29 +05:30
|
|
|
end
|
2017-08-17 22:00:37 +05:30
|
|
|
|
2020-11-24 15:15:51 +05:30
|
|
|
track_event(event_action: :pushed, event_target: Project, author_id: current_user.id)
|
2020-07-28 23:09:34 +05:30
|
|
|
|
2022-06-21 17:19:12 +05:30
|
|
|
namespace = project.namespace
|
2022-07-16 23:28:13 +05:30
|
|
|
if Feature.enabled?(:route_hll_to_snowplow, namespace)
|
2023-01-13 00:05:48 +05:30
|
|
|
Gitlab::Tracking.event(
|
|
|
|
self.class.to_s,
|
|
|
|
:push,
|
|
|
|
label: 'usage_activity_by_stage_monthly.create.action_monthly_active_users_project_repo',
|
|
|
|
namespace: namespace,
|
|
|
|
user: current_user,
|
|
|
|
project: project,
|
|
|
|
context: [Gitlab::Tracking::ServicePingContext.new(data_source: :redis_hll, event: 'action_active_users_project_repo').to_context]
|
|
|
|
)
|
2022-06-21 17:19:12 +05:30
|
|
|
end
|
|
|
|
|
2018-03-17 18:26:18 +05:30
|
|
|
Users::LastPushEventService.new(current_user)
|
|
|
|
.cache_last_push_event(event)
|
|
|
|
|
2020-03-13 15:44:24 +05:30
|
|
|
Users::ActivityService.new(current_user).execute
|
2014-09-02 18:07:02 +05:30
|
|
|
end
|
|
|
|
|
2019-12-04 20:38:33 +05:30
|
|
|
def create_event(resource_parent, current_user, status, attributes = {})
|
2015-04-26 12:48:37 +05:30
|
|
|
attributes.reverse_merge!(
|
2014-09-02 18:07:02 +05:30
|
|
|
action: status,
|
|
|
|
author_id: current_user.id
|
|
|
|
)
|
2020-06-23 00:09:42 +05:30
|
|
|
attributes.merge!(parent_attrs(resource_parent))
|
|
|
|
|
2020-10-24 23:57:45 +05:30
|
|
|
if attributes[:fingerprint].present?
|
|
|
|
Event.safe_find_or_create_by!(attributes)
|
|
|
|
else
|
|
|
|
Event.create!(attributes)
|
|
|
|
end
|
2020-06-23 00:09:42 +05:30
|
|
|
end
|
2015-04-26 12:48:37 +05:30
|
|
|
|
2020-06-23 00:09:42 +05:30
|
|
|
def parent_attrs(resource_parent)
|
2019-12-04 20:38:33 +05:30
|
|
|
resource_parent_attr = case resource_parent
|
|
|
|
when Project
|
2020-06-23 00:09:42 +05:30
|
|
|
:project_id
|
2019-12-04 20:38:33 +05:30
|
|
|
when Group
|
2020-06-23 00:09:42 +05:30
|
|
|
:group_id
|
2019-12-04 20:38:33 +05:30
|
|
|
end
|
|
|
|
|
2020-06-23 00:09:42 +05:30
|
|
|
return {} unless resource_parent_attr
|
|
|
|
|
|
|
|
{ resource_parent_attr => resource_parent.id }
|
|
|
|
end
|
2020-11-24 15:15:51 +05:30
|
|
|
|
|
|
|
def track_event(**params)
|
|
|
|
Gitlab::UsageDataCounters::TrackUniqueEvents.track_event(**params)
|
|
|
|
end
|
2022-07-23 23:45:48 +05:30
|
|
|
|
2023-01-13 00:05:48 +05:30
|
|
|
# This will be deleted as a part of
|
|
|
|
# https://gitlab.com/groups/gitlab-org/-/epics/8641
|
|
|
|
# once all the events are fixed
|
|
|
|
def old_track_snowplow_event(record, current_user, category, action, label)
|
2022-07-23 23:45:48 +05:30
|
|
|
return unless Feature.enabled?(:route_hll_to_snowplow_phase2)
|
|
|
|
|
|
|
|
project = record.project
|
|
|
|
Gitlab::Tracking.event(
|
2022-08-13 15:12:31 +05:30
|
|
|
category.to_s,
|
2022-07-23 23:45:48 +05:30
|
|
|
action.to_s,
|
2022-08-13 15:12:31 +05:30
|
|
|
label: label,
|
2022-07-23 23:45:48 +05:30
|
|
|
project: project,
|
|
|
|
namespace: project.namespace,
|
|
|
|
user: current_user
|
|
|
|
)
|
|
|
|
end
|
2023-01-13 00:05:48 +05:30
|
|
|
|
|
|
|
def track_snowplow_event(action, record, user)
|
|
|
|
project = record.project
|
|
|
|
Gitlab::Tracking.event(
|
|
|
|
self.class.to_s,
|
|
|
|
action.to_s,
|
|
|
|
label: 'usage_activity_by_stage_monthly.create.merge_requests_users',
|
|
|
|
namespace: project.namespace,
|
|
|
|
user: user,
|
|
|
|
project: project,
|
|
|
|
context: [Gitlab::Tracking::ServicePingContext.new(data_source: :redis_hll, event: 'merge_requests_users').to_context]
|
|
|
|
)
|
|
|
|
end
|
2014-09-02 18:07:02 +05:30
|
|
|
end
|
2019-12-04 20:38:33 +05:30
|
|
|
|
2021-06-08 01:23:25 +05:30
|
|
|
EventCreateService.prepend_mod_with('EventCreateService')
|