debian-mirror-gitlab/app/services/event_create_service.rb

243 lines
7.3 KiB
Ruby
Raw Normal View History

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_resource_event(issue, current_user, :opened)
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_resource_event(issue, current_user, :closed)
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_resource_event(issue, current_user, :reopened)
create_record_event(issue, current_user, :reopened)
2014-09-02 18:07:02 +05:30
end
def open_mr(merge_request, current_user)
2020-06-23 00:09:42 +05:30
create_resource_event(merge_request, current_user, :opened)
create_record_event(merge_request, current_user, :created)
2014-09-02 18:07:02 +05:30
end
def close_mr(merge_request, current_user)
2020-06-23 00:09:42 +05:30
create_resource_event(merge_request, current_user, :closed)
create_record_event(merge_request, current_user, :closed)
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_resource_event(merge_request, current_user, :reopened)
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-06-23 00:09:42 +05:30
create_resource_event(merge_request, current_user, :merged)
create_record_event(merge_request, current_user, :merged)
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-06-23 00:09:42 +05:30
create_record_event(note, current_user, :commented)
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: [])
created = create.group_by(&:project).flat_map do |project, designs|
Feature.enabled?(:design_activity_events, project) ? designs : []
end.to_set
updated = update.group_by(&:project).flat_map do |project, designs|
Feature.enabled?(:design_activity_events, project) ? designs : []
end.to_set
return [] if created.empty? && updated.empty?
records = created.zip([:created].cycle) + updated.zip([:updated].cycle)
create_record_events(records, current_user)
end
def destroy_designs(designs, current_user)
designs = designs.select do |design|
Feature.enabled?(:design_activity_events, design.project)
end
return [] unless designs.present?
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-05-24 23:13:21 +05:30
#
# @return a tuple of event and either :found or :created
def wiki_event(wiki_page_meta, author, action)
2020-04-22 19:07:51 +05:30
return unless Feature.enabled?(:wiki_events)
raise IllegalActionError, action unless Event::WIKI_ACTIONS.include?(action)
2020-05-24 23:13:21 +05:30
if duplicate = existing_wiki_event(wiki_page_meta, action)
return duplicate
end
event = create_record_event(wiki_page_meta, author, action)
# Ensure that the event is linked in time to the metadata, for non-deletes
2020-06-23 00:09:42 +05:30
unless event.destroyed_action?
2020-05-24 23:13:21 +05:30
time_stamp = wiki_page_meta.updated_at
event.update_columns(updated_at: time_stamp, created_at: time_stamp)
end
event
2020-04-22 19:07:51 +05:30
end
2019-12-21 20:55:43 +05:30
private
2020-05-24 23:13:21 +05:30
def existing_wiki_event(wiki_page_meta, action)
2020-06-23 00:09:42 +05:30
if Event.actions.fetch(action) == Event.actions[:destroyed]
2020-05-24 23:13:21 +05:30
most_recent = Event.for_wiki_meta(wiki_page_meta).recent.first
2020-06-23 00:09:42 +05:30
return most_recent if most_recent.present? && Event.actions[most_recent.action] == Event.actions[action]
2020-05-24 23:13:21 +05:30
else
Event.for_wiki_meta(wiki_page_meta).created_at(wiki_page_meta.updated_at).first
end
end
2019-12-21 20:55:43 +05:30
def create_record_event(record, current_user, status)
2020-06-23 00:09:42 +05:30
create_event(record.resource_parent, current_user, status,
target_id: record.id, target_type: record.class.name)
end
# If creating several events, this method will insert them all in a single
# statement
#
# @param [[Eventable, Symbol]] a list of pairs of records and a valid status
# @param [User] the author of the event
def create_record_events(pairs, current_user)
base_attrs = {
created_at: Time.now.utc,
updated_at: Time.now.utc,
author_id: current_user.id
}
attribute_sets = pairs.map do |record, status|
action = Event.actions[status]
raise IllegalActionError, "#{status} is not a valid status" if action.nil?
parent_attrs(record.resource_parent)
.merge(base_attrs)
.merge(action: action, target_id: record.id, target_type: record.class.name)
end
Event.insert_all(attribute_sets, returning: %w[id])
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
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))
Event.create!(attributes)
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
def create_resource_event(issuable, current_user, status)
return unless state_change_tracking_enabled?(issuable)
ResourceEvents::ChangeStateService.new(resource: issuable, user: current_user)
.execute(status)
end
def state_change_tracking_enabled?(issuable)
issuable&.respond_to?(:resource_state_events) &&
::Feature.enabled?(:track_resource_state_change_events, issuable&.project)
2014-09-02 18:07:02 +05:30
end
end
2019-12-04 20:38:33 +05:30
EventCreateService.prepend_if_ee('EE::EventCreateService')