2018-09-14 01:42:05 -04:00
|
|
|
# frozen_string_literal: true
|
|
|
|
|
2017-04-27 06:41:26 -04:00
|
|
|
module NotesActions
|
|
|
|
include RendersNotes
|
2017-11-17 07:27:16 -05:00
|
|
|
include Gitlab::Utils::StrongMemoize
|
2017-04-27 06:41:26 -04:00
|
|
|
extend ActiveSupport::Concern
|
|
|
|
|
|
|
|
included do
|
2017-08-02 06:34:56 -04:00
|
|
|
before_action :set_polling_interval_header, only: [:index]
|
2017-11-16 06:23:50 -05:00
|
|
|
before_action :require_noteable!, only: [:index, :create]
|
2017-04-27 06:41:26 -04:00
|
|
|
before_action :authorize_admin_note!, only: [:update, :destroy]
|
2017-07-27 10:36:39 -04:00
|
|
|
before_action :note_project, only: [:create]
|
2017-04-27 06:41:26 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
def index
|
|
|
|
current_fetched_at = Time.now.to_i
|
|
|
|
|
|
|
|
notes_json = { notes: [], last_fetched_at: current_fetched_at }
|
|
|
|
|
2017-08-18 06:47:50 -04:00
|
|
|
notes = notes_finder.execute
|
|
|
|
.inc_relations_for_view
|
|
|
|
|
2018-09-07 09:39:20 -04:00
|
|
|
notes = ResourceEvents::MergeIntoNotesService.new(noteable, current_user, last_fetched_at: current_fetched_at).execute(notes)
|
2017-08-18 06:47:50 -04:00
|
|
|
notes = prepare_notes_for_rendering(notes)
|
2017-10-02 12:47:08 -04:00
|
|
|
notes = notes.reject { |n| n.cross_reference_not_visible_for?(current_user) }
|
2017-04-27 06:41:26 -04:00
|
|
|
|
2017-06-09 17:24:54 -04:00
|
|
|
notes_json[:notes] =
|
2018-02-27 19:10:43 -05:00
|
|
|
if use_note_serializer?
|
2017-08-18 06:47:50 -04:00
|
|
|
note_serializer.represent(notes)
|
2017-06-09 17:24:54 -04:00
|
|
|
else
|
2017-08-18 06:47:50 -04:00
|
|
|
notes.map { |note| note_json(note) }
|
2017-06-09 17:24:54 -04:00
|
|
|
end
|
2017-04-27 06:41:26 -04:00
|
|
|
|
|
|
|
render json: notes_json
|
|
|
|
end
|
|
|
|
|
2017-11-22 02:50:36 -05:00
|
|
|
# rubocop:disable Gitlab/ModuleWithInstanceVariables
|
2017-04-27 06:41:26 -04:00
|
|
|
def create
|
|
|
|
create_params = note_params.merge(
|
|
|
|
merge_request_diff_head_sha: params[:merge_request_diff_head_sha],
|
|
|
|
in_reply_to_discussion_id: params[:in_reply_to_discussion_id]
|
|
|
|
)
|
2017-07-27 10:36:39 -04:00
|
|
|
|
|
|
|
@note = Notes::CreateService.new(note_project, current_user, create_params).execute
|
2017-04-27 06:41:26 -04:00
|
|
|
|
|
|
|
if @note.is_a?(Note)
|
2018-07-16 12:18:52 -04:00
|
|
|
prepare_notes_for_rendering([@note], noteable)
|
2017-04-27 06:41:26 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
respond_to do |format|
|
|
|
|
format.json { render json: note_json(@note) }
|
|
|
|
format.html { redirect_back_or_default }
|
|
|
|
end
|
|
|
|
end
|
2017-11-22 02:50:36 -05:00
|
|
|
# rubocop:enable Gitlab/ModuleWithInstanceVariables
|
2017-04-27 06:41:26 -04:00
|
|
|
|
2017-11-22 02:50:36 -05:00
|
|
|
# rubocop:disable Gitlab/ModuleWithInstanceVariables
|
2017-04-27 06:41:26 -04:00
|
|
|
def update
|
|
|
|
@note = Notes::UpdateService.new(project, current_user, note_params).execute(note)
|
|
|
|
|
|
|
|
if @note.is_a?(Note)
|
2018-07-16 12:18:52 -04:00
|
|
|
prepare_notes_for_rendering([@note])
|
2017-04-27 06:41:26 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
respond_to do |format|
|
|
|
|
format.json { render json: note_json(@note) }
|
|
|
|
format.html { redirect_back_or_default }
|
|
|
|
end
|
|
|
|
end
|
2017-11-22 02:50:36 -05:00
|
|
|
# rubocop:enable Gitlab/ModuleWithInstanceVariables
|
2017-04-27 06:41:26 -04:00
|
|
|
|
|
|
|
def destroy
|
|
|
|
if note.editable?
|
|
|
|
Notes::DestroyService.new(project, current_user).execute(note)
|
|
|
|
end
|
|
|
|
|
|
|
|
respond_to do |format|
|
|
|
|
format.js { head :ok }
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
private
|
|
|
|
|
2017-05-03 04:48:01 -04:00
|
|
|
def note_html(note)
|
|
|
|
render_to_string(
|
|
|
|
"shared/notes/_note",
|
|
|
|
layout: false,
|
|
|
|
formats: [:html],
|
|
|
|
locals: { note: note }
|
|
|
|
)
|
|
|
|
end
|
|
|
|
|
2017-04-27 06:41:26 -04:00
|
|
|
def note_json(note)
|
|
|
|
attrs = {
|
|
|
|
commands_changes: note.commands_changes
|
|
|
|
}
|
|
|
|
|
|
|
|
if note.persisted?
|
2017-06-09 17:24:54 -04:00
|
|
|
attrs[:valid] = true
|
2017-04-27 06:41:26 -04:00
|
|
|
|
2018-02-27 19:10:43 -05:00
|
|
|
if use_note_serializer?
|
2017-06-09 17:24:54 -04:00
|
|
|
attrs.merge!(note_serializer.represent(note))
|
|
|
|
else
|
2017-04-27 06:41:26 -04:00
|
|
|
attrs.merge!(
|
2017-06-09 17:24:54 -04:00
|
|
|
id: note.id,
|
|
|
|
discussion_id: note.discussion_id(noteable),
|
|
|
|
html: note_html(note),
|
2017-10-07 00:25:17 -04:00
|
|
|
note: note.note,
|
|
|
|
on_image: note.try(:on_image?)
|
2017-04-27 06:41:26 -04:00
|
|
|
)
|
2017-06-09 17:24:54 -04:00
|
|
|
|
|
|
|
discussion = note.to_discussion(noteable)
|
|
|
|
unless discussion.individual_note?
|
|
|
|
attrs.merge!(
|
|
|
|
discussion_resolvable: discussion.resolvable?,
|
|
|
|
|
|
|
|
diff_discussion_html: diff_discussion_html(discussion),
|
|
|
|
discussion_html: discussion_html(discussion)
|
|
|
|
)
|
2017-10-21 13:16:35 -04:00
|
|
|
|
2017-11-02 12:29:59 -04:00
|
|
|
attrs[:discussion_line_code] = discussion.line_code if discussion.diff_discussion?
|
2017-06-09 17:24:54 -04:00
|
|
|
end
|
2017-04-27 06:41:26 -04:00
|
|
|
end
|
|
|
|
else
|
|
|
|
attrs.merge!(
|
|
|
|
valid: false,
|
|
|
|
errors: note.errors
|
|
|
|
)
|
|
|
|
end
|
|
|
|
|
|
|
|
attrs
|
|
|
|
end
|
|
|
|
|
2017-05-03 04:48:01 -04:00
|
|
|
def diff_discussion_html(discussion)
|
|
|
|
return unless discussion.diff_discussion?
|
|
|
|
|
2017-10-07 00:25:17 -04:00
|
|
|
on_image = discussion.on_image?
|
|
|
|
|
|
|
|
if params[:view] == 'parallel' && !on_image
|
2017-05-03 04:48:01 -04:00
|
|
|
template = "discussions/_parallel_diff_discussion"
|
|
|
|
locals =
|
|
|
|
if params[:line_type] == 'old'
|
|
|
|
{ discussions_left: [discussion], discussions_right: nil }
|
|
|
|
else
|
|
|
|
{ discussions_left: nil, discussions_right: [discussion] }
|
|
|
|
end
|
|
|
|
else
|
|
|
|
template = "discussions/_diff_discussion"
|
2017-11-22 02:50:36 -05:00
|
|
|
@fresh_discussion = true # rubocop:disable Gitlab/ModuleWithInstanceVariables
|
2017-10-07 00:25:17 -04:00
|
|
|
|
|
|
|
locals = { discussions: [discussion], on_image: on_image }
|
2017-05-03 04:48:01 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
render_to_string(
|
|
|
|
template,
|
|
|
|
layout: false,
|
|
|
|
formats: [:html],
|
|
|
|
locals: locals
|
|
|
|
)
|
|
|
|
end
|
|
|
|
|
|
|
|
def discussion_html(discussion)
|
|
|
|
return if discussion.individual_note?
|
|
|
|
|
|
|
|
render_to_string(
|
|
|
|
"discussions/_discussion",
|
|
|
|
layout: false,
|
|
|
|
formats: [:html],
|
|
|
|
locals: { discussion: discussion }
|
|
|
|
)
|
|
|
|
end
|
|
|
|
|
2017-04-27 06:41:26 -04:00
|
|
|
def authorize_admin_note!
|
|
|
|
return access_denied! unless can?(current_user, :admin_note, note)
|
|
|
|
end
|
|
|
|
|
|
|
|
def note_params
|
|
|
|
params.require(:note).permit(
|
|
|
|
:project_id,
|
|
|
|
:noteable_type,
|
|
|
|
:noteable_id,
|
|
|
|
:commit_id,
|
|
|
|
:noteable,
|
|
|
|
:type,
|
|
|
|
|
|
|
|
:note,
|
|
|
|
:attachment,
|
|
|
|
|
|
|
|
# LegacyDiffNote
|
|
|
|
:line_code,
|
|
|
|
|
|
|
|
# DiffNote
|
|
|
|
:position
|
|
|
|
)
|
|
|
|
end
|
|
|
|
|
2017-08-02 06:34:56 -04:00
|
|
|
def set_polling_interval_header
|
2017-08-03 14:27:33 -04:00
|
|
|
Gitlab::PollingInterval.set_header(response, interval: 6_000)
|
2017-08-02 06:34:56 -04:00
|
|
|
end
|
|
|
|
|
2017-04-27 06:41:26 -04:00
|
|
|
def noteable
|
2017-11-22 02:50:36 -05:00
|
|
|
@noteable ||= notes_finder.target || @note&.noteable # rubocop:disable Gitlab/ModuleWithInstanceVariables
|
2017-11-16 06:23:50 -05:00
|
|
|
end
|
|
|
|
|
|
|
|
def require_noteable!
|
|
|
|
render_404 unless noteable
|
2017-04-27 06:41:26 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
def last_fetched_at
|
|
|
|
request.headers['X-Last-Fetched-At']
|
|
|
|
end
|
|
|
|
|
|
|
|
def notes_finder
|
|
|
|
@notes_finder ||= NotesFinder.new(project, current_user, finder_params)
|
|
|
|
end
|
2017-06-09 17:24:54 -04:00
|
|
|
|
|
|
|
def note_serializer
|
2018-04-03 12:03:00 -04:00
|
|
|
ProjectNoteSerializer.new(project: project, noteable: noteable, current_user: current_user)
|
2017-06-09 17:24:54 -04:00
|
|
|
end
|
2017-07-31 04:23:50 -04:00
|
|
|
|
2017-07-27 10:36:39 -04:00
|
|
|
def note_project
|
2017-11-17 07:27:16 -05:00
|
|
|
strong_memoize(:note_project) do
|
2018-04-18 05:19:40 -04:00
|
|
|
next nil unless project
|
2017-07-27 10:36:39 -04:00
|
|
|
|
2017-11-17 07:27:16 -05:00
|
|
|
note_project_id = params[:note_project_id]
|
2017-07-27 10:36:39 -04:00
|
|
|
|
2017-11-17 07:27:16 -05:00
|
|
|
the_project =
|
|
|
|
if note_project_id.present?
|
|
|
|
Project.find(note_project_id)
|
|
|
|
else
|
|
|
|
project
|
|
|
|
end
|
2017-07-27 10:36:39 -04:00
|
|
|
|
2018-04-18 05:19:40 -04:00
|
|
|
next access_denied! unless can?(current_user, :create_note, the_project)
|
2017-07-27 10:36:39 -04:00
|
|
|
|
2017-11-17 07:27:16 -05:00
|
|
|
the_project
|
|
|
|
end
|
2017-07-27 10:36:39 -04:00
|
|
|
end
|
2018-02-27 19:10:43 -05:00
|
|
|
|
|
|
|
def use_note_serializer?
|
|
|
|
return false if params['html']
|
|
|
|
|
2018-06-21 08:22:40 -04:00
|
|
|
noteable.discussions_rendered_on_frontend?
|
2018-02-27 19:10:43 -05:00
|
|
|
end
|
2017-04-27 06:41:26 -04:00
|
|
|
end
|