Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in / Register
Toggle navigation
G
gitlab-ce
Project overview
Project overview
Details
Activity
Releases
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Issues
0
Issues
0
List
Boards
Labels
Milestones
Merge Requests
1
Merge Requests
1
Analytics
Analytics
Repository
Value Stream
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Create a new issue
Commits
Issue Boards
Open sidebar
nexedi
gitlab-ce
Commits
e4eb6465
Commit
e4eb6465
authored
Feb 06, 2019
by
Constance Okoghenun
Committed by
Phil Hughes
Feb 06, 2019
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
EE Port - Show reply button in comments
parent
cdd8027a
Changes
29
Hide whitespace changes
Inline
Side-by-side
Showing
29 changed files
with
578 additions
and
75 deletions
+578
-75
app/assets/javascripts/notes/components/note_actions.vue
app/assets/javascripts/notes/components/note_actions.vue
+21
-1
app/assets/javascripts/notes/components/note_actions/reply_button.vue
...avascripts/notes/components/note_actions/reply_button.vue
+40
-0
app/assets/javascripts/notes/components/noteable_discussion.vue
...sets/javascripts/notes/components/noteable_discussion.vue
+1
-0
app/assets/javascripts/notes/components/noteable_note.vue
app/assets/javascripts/notes/components/noteable_note.vue
+22
-2
app/assets/javascripts/notes/components/notes_app.vue
app/assets/javascripts/notes/components/notes_app.vue
+6
-1
app/assets/javascripts/notes/stores/actions.js
app/assets/javascripts/notes/stores/actions.js
+3
-0
app/assets/javascripts/notes/stores/mutation_types.js
app/assets/javascripts/notes/stores/mutation_types.js
+1
-0
app/assets/javascripts/notes/stores/mutations.js
app/assets/javascripts/notes/stores/mutations.js
+5
-0
app/controllers/concerns/issuable_actions.rb
app/controllers/concerns/issuable_actions.rb
+3
-0
app/models/concerns/noteable.rb
app/models/concerns/noteable.rb
+15
-1
app/models/discussion.rb
app/models/discussion.rb
+6
-0
app/models/individual_note_discussion.rb
app/models/individual_note_discussion.rb
+8
-0
app/models/sent_notification.rb
app/models/sent_notification.rb
+4
-21
app/services/notes/build_service.rb
app/services/notes/build_service.rb
+2
-0
app/services/notes/create_service.rb
app/services/notes/create_service.rb
+4
-0
ee/app/assets/javascripts/batch_comments/mixins/resolved_status.js
...sets/javascripts/batch_comments/mixins/resolved_status.js
+5
-0
ee/app/models/concerns/ee/noteable.rb
ee/app/models/concerns/ee/noteable.rb
+8
-0
ee/spec/javascripts/notes/components/note_actions_spec.js
ee/spec/javascripts/notes/components/note_actions_spec.js
+54
-0
ee/spec/models/concerns/ee/noteable_spec.rb
ee/spec/models/concerns/ee/noteable_spec.rb
+15
-0
locale/gitlab.pot
locale/gitlab.pot
+3
-0
spec/features/merge_request/user_posts_notes_spec.rb
spec/features/merge_request/user_posts_notes_spec.rb
+32
-0
spec/javascripts/notes/components/note_actions/reply_button_spec.js
...cripts/notes/components/note_actions/reply_button_spec.js
+46
-0
spec/javascripts/notes/components/note_actions_spec.js
spec/javascripts/notes/components/note_actions_spec.js
+111
-38
spec/javascripts/notes/stores/actions_spec.js
spec/javascripts/notes/stores/actions_spec.js
+14
-0
spec/javascripts/notes/stores/mutation_spec.js
spec/javascripts/notes/stores/mutation_spec.js
+23
-0
spec/lib/gitlab/email/handler/create_note_handler_spec.rb
spec/lib/gitlab/email/handler/create_note_handler_spec.rb
+21
-5
spec/models/sent_notification_spec.rb
spec/models/sent_notification_spec.rb
+28
-6
spec/services/notes/build_service_spec.rb
spec/services/notes/build_service_spec.rb
+40
-0
spec/services/notes/create_service_spec.rb
spec/services/notes/create_service_spec.rb
+37
-0
No files found.
app/assets/javascripts/notes/components/note_actions.vue
View file @
e4eb6465
...
@@ -3,11 +3,13 @@ import { mapGetters } from 'vuex';
...
@@ -3,11 +3,13 @@ import { mapGetters } from 'vuex';
import
Icon
from
'
~/vue_shared/components/icon.vue
'
;
import
Icon
from
'
~/vue_shared/components/icon.vue
'
;
import
{
GlLoadingIcon
,
GlTooltipDirective
}
from
'
@gitlab/ui
'
;
import
{
GlLoadingIcon
,
GlTooltipDirective
}
from
'
@gitlab/ui
'
;
import
resolvedStatusMixin
from
'
ee/batch_comments/mixins/resolved_status
'
;
import
resolvedStatusMixin
from
'
ee/batch_comments/mixins/resolved_status
'
;
import
ReplyButton
from
'
./note_actions/reply_button.vue
'
;
export
default
{
export
default
{
name
:
'
NoteActions
'
,
name
:
'
NoteActions
'
,
components
:
{
components
:
{
Icon
,
Icon
,
ReplyButton
,
GlLoadingIcon
,
GlLoadingIcon
,
},
},
directives
:
{
directives
:
{
...
@@ -23,6 +25,11 @@ export default {
...
@@ -23,6 +25,11 @@ export default {
type
:
[
String
,
Number
],
type
:
[
String
,
Number
],
required
:
true
,
required
:
true
,
},
},
discussionId
:
{
type
:
String
,
required
:
false
,
default
:
''
,
},
noteUrl
:
{
noteUrl
:
{
type
:
String
,
type
:
String
,
required
:
false
,
required
:
false
,
...
@@ -38,6 +45,10 @@ export default {
...
@@ -38,6 +45,10 @@ export default {
required
:
false
,
required
:
false
,
default
:
null
,
default
:
null
,
},
},
showReply
:
{
type
:
Boolean
,
required
:
true
,
},
canEdit
:
{
canEdit
:
{
type
:
Boolean
,
type
:
Boolean
,
required
:
true
,
required
:
true
,
...
@@ -82,6 +93,9 @@ export default {
...
@@ -82,6 +93,9 @@ export default {
},
},
computed
:
{
computed
:
{
...
mapGetters
([
'
getUserDataByProp
'
]),
...
mapGetters
([
'
getUserDataByProp
'
]),
showReplyButton
()
{
return
gon
.
features
&&
gon
.
features
.
replyToIndividualNotes
&&
this
.
showReply
;
},
shouldShowActionsDropdown
()
{
shouldShowActionsDropdown
()
{
return
this
.
currentUserId
&&
(
this
.
canEdit
||
this
.
canReportAsAbuse
);
return
this
.
currentUserId
&&
(
this
.
canEdit
||
this
.
canReportAsAbuse
);
},
},
...
@@ -95,7 +109,7 @@ export default {
...
@@ -95,7 +109,7 @@ export default {
return
this
.
getUserDataByProp
(
'
id
'
);
return
this
.
getUserDataByProp
(
'
id
'
);
},
},
resolveButtonTitle
()
{
resolveButtonTitle
()
{
if
(
this
.
discussionId
)
return
this
.
resolvedStatusMessage
;
if
(
this
.
isDraft
||
this
.
discussionId
)
return
this
.
resolvedStatusMessage
;
let
title
=
'
Mark as resolved
'
;
let
title
=
'
Mark as resolved
'
;
...
@@ -158,6 +172,12 @@ export default {
...
@@ -158,6 +172,12 @@ export default {
<icon
css-classes=
"link-highlight award-control-icon-super-positive"
name=
"emoji_smiley"
/>
<icon
css-classes=
"link-highlight award-control-icon-super-positive"
name=
"emoji_smiley"
/>
</a>
</a>
</div>
</div>
<reply-button
v-if=
"showReplyButton"
ref=
"replyButton"
class=
"js-reply-button"
:note-id=
"discussionId"
/>
<div
v-if=
"canEdit"
class=
"note-actions-item"
>
<div
v-if=
"canEdit"
class=
"note-actions-item"
>
<button
<button
v-gl-tooltip
.
bottom
v-gl-tooltip
.
bottom
...
...
app/assets/javascripts/notes/components/note_actions/reply_button.vue
0 → 100644
View file @
e4eb6465
<
script
>
import
{
mapActions
}
from
'
vuex
'
;
import
{
GlTooltipDirective
,
GlButton
}
from
'
@gitlab/ui
'
;
import
Icon
from
'
~/vue_shared/components/icon.vue
'
;
export
default
{
name
:
'
ReplyButton
'
,
components
:
{
Icon
,
GlButton
,
},
directives
:
{
GlTooltip
:
GlTooltipDirective
,
},
props
:
{
noteId
:
{
type
:
String
,
required
:
true
,
},
},
methods
:
{
...
mapActions
([
'
convertToDiscussion
'
]),
},
};
</
script
>
<
template
>
<div
class=
"note-actions-item"
>
<gl-button
ref=
"button"
v-gl-tooltip
.
bottom
class=
"note-action-button"
variant=
"transparent"
:title=
"__('Reply to comment')"
@
click=
"convertToDiscussion(noteId)"
>
<icon
name=
"comment"
css-classes=
"link-highlight"
/>
</gl-button>
</div>
</
template
>
app/assets/javascripts/notes/components/noteable_discussion.vue
View file @
e4eb6465
...
@@ -408,6 +408,7 @@ Please check your network connection and try again.`;
...
@@ -408,6 +408,7 @@ Please check your network connection and try again.`;
:line=
"line"
:line=
"line"
:commit=
"commit"
:commit=
"commit"
:help-page-path=
"helpPagePath"
:help-page-path=
"helpPagePath"
:show-reply-button=
"canReply"
@
handleDeleteNote=
"deleteNoteHandler"
@
handleDeleteNote=
"deleteNoteHandler"
>
>
<note-edited-text
<note-edited-text
...
...
app/assets/javascripts/notes/components/noteable_note.vue
View file @
e4eb6465
...
@@ -29,6 +29,11 @@ export default {
...
@@ -29,6 +29,11 @@ export default {
type
:
Object
,
type
:
Object
,
required
:
true
,
required
:
true
,
},
},
discussion
:
{
type
:
Object
,
required
:
false
,
default
:
null
,
},
line
:
{
line
:
{
type
:
Object
,
type
:
Object
,
required
:
false
,
required
:
false
,
...
@@ -54,7 +59,7 @@ export default {
...
@@ -54,7 +59,7 @@ export default {
};
};
},
},
computed
:
{
computed
:
{
...
mapGetters
([
'
targetNoteHash
'
,
'
getNoteableData
'
,
'
getUserData
'
]),
...
mapGetters
([
'
targetNoteHash
'
,
'
getNoteableData
'
,
'
getUserData
'
,
'
commentsDisabled
'
]),
author
()
{
author
()
{
return
this
.
note
.
author
;
return
this
.
note
.
author
;
},
},
...
@@ -80,6 +85,19 @@ export default {
...
@@ -80,6 +85,19 @@ export default {
isTarget
()
{
isTarget
()
{
return
this
.
targetNoteHash
===
this
.
noteAnchorId
;
return
this
.
targetNoteHash
===
this
.
noteAnchorId
;
},
},
discussionId
()
{
if
(
this
.
discussion
)
{
return
this
.
discussion
.
id
;
}
return
''
;
},
showReplyButton
()
{
if
(
!
this
.
discussion
||
!
this
.
getNoteableData
.
current_user
.
can_create_note
)
{
return
false
;
}
return
this
.
discussion
.
individual_note
&&
!
this
.
commentsDisabled
;
},
actionText
()
{
actionText
()
{
if
(
!
this
.
commit
)
{
if
(
!
this
.
commit
)
{
return
''
;
return
''
;
...
@@ -236,6 +254,7 @@ export default {
...
@@ -236,6 +254,7 @@ export default {
:note-id=
"note.id"
:note-id=
"note.id"
:note-url=
"note.noteable_note_url"
:note-url=
"note.noteable_note_url"
:access-level=
"note.human_access"
:access-level=
"note.human_access"
:show-reply=
"showReplyButton"
:can-edit=
"note.current_user.can_edit"
:can-edit=
"note.current_user.can_edit"
:can-award-emoji=
"note.current_user.can_award_emoji"
:can-award-emoji=
"note.current_user.can_award_emoji"
:can-delete=
"note.current_user.can_edit"
:can-delete=
"note.current_user.can_edit"
...
@@ -248,8 +267,9 @@ export default {
...
@@ -248,8 +267,9 @@ export default {
:is-resolved=
"note.resolved || note.resolve_discussion"
:is-resolved=
"note.resolved || note.resolve_discussion"
:is-resolving=
"isResolving"
:is-resolving=
"isResolving"
:resolved-by=
"note.resolved_by"
:resolved-by=
"note.resolved_by"
:
discussion-id=
"note.isDraft && note.discussion_id
"
:
is-draft=
"note.isDraft
"
:resolve-discussion=
"note.isDraft && note.resolve_discussion"
:resolve-discussion=
"note.isDraft && note.resolve_discussion"
:discussion-id=
"discussionId"
@
handleEdit=
"editHandler"
@
handleEdit=
"editHandler"
@
handleDelete=
"deleteHandler"
@
handleDelete=
"deleteHandler"
@
handleResolve=
"resolveHandler"
@
handleResolve=
"resolveHandler"
...
...
app/assets/javascripts/notes/components/notes_app.vue
View file @
e4eb6465
...
@@ -199,7 +199,12 @@ export default {
...
@@ -199,7 +199,12 @@ export default {
:key=
"discussion.id"
:key=
"discussion.id"
:note=
"discussion.notes[0]"
:note=
"discussion.notes[0]"
/>
/>
<noteable-note
v-else
:key=
"discussion.id"
:note=
"discussion.notes[0]"
/>
<noteable-note
v-else
:key=
"discussion.id"
:note=
"discussion.notes[0]"
:discussion=
"discussion"
/>
</
template
>
</
template
>
<noteable-discussion
<noteable-discussion
v-else
v-else
...
...
app/assets/javascripts/notes/stores/actions.js
View file @
e4eb6465
...
@@ -426,5 +426,8 @@ export const submitSuggestion = (
...
@@ -426,5 +426,8 @@ export const submitSuggestion = (
});
});
};
};
export
const
convertToDiscussion
=
({
commit
},
noteId
)
=>
commit
(
types
.
CONVERT_TO_DISCUSSION
,
noteId
);
// prevent babel-plugin-rewire from generating an invalid default during karma tests
// prevent babel-plugin-rewire from generating an invalid default during karma tests
export
default
()
=>
{};
export
default
()
=>
{};
app/assets/javascripts/notes/stores/mutation_types.js
View file @
e4eb6465
...
@@ -17,6 +17,7 @@ export const SET_NOTES_FETCHED_STATE = 'SET_NOTES_FETCHED_STATE';
...
@@ -17,6 +17,7 @@ export const SET_NOTES_FETCHED_STATE = 'SET_NOTES_FETCHED_STATE';
export
const
SET_NOTES_LOADING_STATE
=
'
SET_NOTES_LOADING_STATE
'
;
export
const
SET_NOTES_LOADING_STATE
=
'
SET_NOTES_LOADING_STATE
'
;
export
const
DISABLE_COMMENTS
=
'
DISABLE_COMMENTS
'
;
export
const
DISABLE_COMMENTS
=
'
DISABLE_COMMENTS
'
;
export
const
APPLY_SUGGESTION
=
'
APPLY_SUGGESTION
'
;
export
const
APPLY_SUGGESTION
=
'
APPLY_SUGGESTION
'
;
export
const
CONVERT_TO_DISCUSSION
=
'
CONVERT_TO_DISCUSSION
'
;
// DISCUSSION
// DISCUSSION
export
const
COLLAPSE_DISCUSSION
=
'
COLLAPSE_DISCUSSION
'
;
export
const
COLLAPSE_DISCUSSION
=
'
COLLAPSE_DISCUSSION
'
;
...
...
app/assets/javascripts/notes/stores/mutations.js
View file @
e4eb6465
...
@@ -264,4 +264,9 @@ export default {
...
@@ -264,4 +264,9 @@ export default {
).
length
;
).
length
;
state
.
hasUnresolvedDiscussions
=
state
.
unresolvedDiscussionsCount
>
1
;
state
.
hasUnresolvedDiscussions
=
state
.
unresolvedDiscussionsCount
>
1
;
},
},
[
types
.
CONVERT_TO_DISCUSSION
](
state
,
discussionId
)
{
const
discussion
=
utils
.
findNoteObjectById
(
state
.
discussions
,
discussionId
);
Object
.
assign
(
discussion
,
{
individual_note
:
false
});
},
};
};
app/controllers/concerns/issuable_actions.rb
View file @
e4eb6465
...
@@ -7,6 +7,9 @@ module IssuableActions
...
@@ -7,6 +7,9 @@ module IssuableActions
included
do
included
do
before_action
:authorize_destroy_issuable!
,
only: :destroy
before_action
:authorize_destroy_issuable!
,
only: :destroy
before_action
:authorize_admin_issuable!
,
only: :bulk_update
before_action
:authorize_admin_issuable!
,
only: :bulk_update
before_action
only: :show
do
push_frontend_feature_flag
(
:reply_to_individual_notes
)
end
end
end
def
permitted_keys
def
permitted_keys
...
...
app/models/concerns/noteable.rb
View file @
e4eb6465
# frozen_string_literal: true
# frozen_string_literal: true
module
Noteable
module
Noteable
# Names of all implementers of `Noteable` that support resolvable notes.
extend
ActiveSupport
::
Concern
# `Noteable` class names that support resolvable notes.
RESOLVABLE_TYPES
=
%w(MergeRequest)
.
freeze
RESOLVABLE_TYPES
=
%w(MergeRequest)
.
freeze
class_methods
do
# `Noteable` class names that support replying to individual notes.
def
replyable_types
%w(Issue MergeRequest)
end
end
def
base_class_name
def
base_class_name
self
.
class
.
base_class
.
name
self
.
class
.
base_class
.
name
end
end
...
@@ -26,6 +35,10 @@ module Noteable
...
@@ -26,6 +35,10 @@ module Noteable
DiscussionNote
.
noteable_types
.
include?
(
base_class_name
)
DiscussionNote
.
noteable_types
.
include?
(
base_class_name
)
end
end
def
supports_replying_to_individual_notes?
supports_discussions?
&&
self
.
class
.
replyable_types
.
include?
(
base_class_name
)
end
def
supports_suggestion?
def
supports_suggestion?
false
false
end
end
...
@@ -111,4 +124,5 @@ module Noteable
...
@@ -111,4 +124,5 @@ module Noteable
end
end
end
end
Noteable
::
ClassMethods
.
prepend
(
EE
::
Noteable
::
ClassMethods
)
# rubocop: disable Cop/InjectEnterpriseEditionModule
Noteable
.
prepend
(
EE
::
Noteable
)
Noteable
.
prepend
(
EE
::
Noteable
)
app/models/discussion.rb
View file @
e4eb6465
...
@@ -17,6 +17,8 @@ class Discussion
...
@@ -17,6 +17,8 @@ class Discussion
:for_commit?
,
:for_commit?
,
:for_merge_request?
,
:for_merge_request?
,
:save
,
to: :first_note
to: :first_note
def
project_id
def
project_id
...
@@ -116,6 +118,10 @@ class Discussion
...
@@ -116,6 +118,10 @@ class Discussion
false
false
end
end
def
can_convert_to_discussion?
false
end
def
new_discussion?
def
new_discussion?
notes
.
length
==
1
notes
.
length
==
1
end
end
...
...
app/models/individual_note_discussion.rb
View file @
e4eb6465
...
@@ -13,6 +13,14 @@ class IndividualNoteDiscussion < Discussion
...
@@ -13,6 +13,14 @@ class IndividualNoteDiscussion < Discussion
true
true
end
end
def
can_convert_to_discussion?
noteable
.
supports_replying_to_individual_notes?
&&
Feature
.
enabled?
(
:reply_to_individual_notes
)
end
def
convert_to_discussion!
first_note
.
becomes!
(
Discussion
.
note_class
).
to_discussion
end
def
reply_attributes
def
reply_attributes
super
.
tap
{
|
attrs
|
attrs
.
delete
(
:discussion_id
)
}
super
.
tap
{
|
attrs
|
attrs
.
delete
(
:discussion_id
)
}
end
end
...
...
app/models/sent_notification.rb
View file @
e4eb6465
...
@@ -48,7 +48,7 @@ class SentNotification < ActiveRecord::Base
...
@@ -48,7 +48,7 @@ class SentNotification < ActiveRecord::Base
end
end
def
record_note
(
note
,
recipient_id
,
reply_key
=
self
.
reply_key
,
attrs
=
{})
def
record_note
(
note
,
recipient_id
,
reply_key
=
self
.
reply_key
,
attrs
=
{})
attrs
[
:in_reply_to_discussion_id
]
=
note
.
discussion_id
attrs
[
:in_reply_to_discussion_id
]
=
note
.
discussion_id
if
note
.
part_of_discussion?
record
(
note
.
noteable
,
recipient_id
,
reply_key
,
attrs
)
record
(
note
.
noteable
,
recipient_id
,
reply_key
,
attrs
)
end
end
...
@@ -99,29 +99,12 @@ class SentNotification < ActiveRecord::Base
...
@@ -99,29 +99,12 @@ class SentNotification < ActiveRecord::Base
private
private
def
reply_params
def
reply_params
attrs
=
{
{
noteable_type:
self
.
noteable_type
,
noteable_type:
self
.
noteable_type
,
noteable_id:
self
.
noteable_id
,
noteable_id:
self
.
noteable_id
,
commit_id:
self
.
commit_id
commit_id:
self
.
commit_id
,
in_reply_to_discussion_id:
self
.
in_reply_to_discussion_id
}
}
if
self
.
in_reply_to_discussion_id
.
present?
attrs
[
:in_reply_to_discussion_id
]
=
self
.
in_reply_to_discussion_id
else
# Remove in GitLab 10.0, when we will not support replying to SentNotifications
# that don't have `in_reply_to_discussion_id` anymore.
attrs
.
merge!
(
type:
self
.
note_type
,
# LegacyDiffNote
line_code:
self
.
line_code
,
# DiffNote
position:
self
.
position
.
to_json
)
end
attrs
end
end
def
note_valid
def
note_valid
...
...
app/services/notes/build_service.rb
View file @
e4eb6465
...
@@ -15,6 +15,8 @@ module Notes
...
@@ -15,6 +15,8 @@ module Notes
return
note
return
note
end
end
discussion
=
discussion
.
convert_to_discussion!
if
discussion
.
can_convert_to_discussion?
params
.
merge!
(
discussion
.
reply_attributes
)
params
.
merge!
(
discussion
.
reply_attributes
)
should_resolve
=
discussion
.
resolved?
should_resolve
=
discussion
.
resolved?
end
end
...
...
app/services/notes/create_service.rb
View file @
e4eb6465
...
@@ -34,6 +34,10 @@ module Notes
...
@@ -34,6 +34,10 @@ module Notes
end
end
if
!
only_commands
&&
note
.
save
if
!
only_commands
&&
note
.
save
if
note
.
part_of_discussion?
&&
note
.
discussion
.
can_convert_to_discussion?
note
.
discussion
.
convert_to_discussion!
.
save
(
touch:
false
)
end
todo_service
.
new_note
(
note
,
current_user
)
todo_service
.
new_note
(
note
,
current_user
)
clear_noteable_diffs_cache
(
note
)
clear_noteable_diffs_cache
(
note
)
Suggestions
::
CreateService
.
new
(
note
).
execute
Suggestions
::
CreateService
.
new
(
note
).
execute
...
...
ee/app/assets/javascripts/batch_comments/mixins/resolved_status.js
View file @
e4eb6465
...
@@ -13,6 +13,11 @@ export default {
...
@@ -13,6 +13,11 @@ export default {
required
:
false
,
required
:
false
,
default
:
false
,
default
:
false
,
},
},
isDraft
:
{
type
:
Boolean
,
required
:
false
,
default
:
false
,
},
},
},
computed
:
{
computed
:
{
...
mapGetters
([
'
isDiscussionResolved
'
]),
...
mapGetters
([
'
isDiscussionResolved
'
]),
...
...
ee/app/models/concerns/ee/noteable.rb
View file @
e4eb6465
...
@@ -5,6 +5,14 @@ module EE
...
@@ -5,6 +5,14 @@ module EE
extend
ActiveSupport
::
Concern
extend
ActiveSupport
::
Concern
extend
::
Gitlab
::
Utils
::
Override
extend
::
Gitlab
::
Utils
::
Override
class_methods
do
# We can't specify `override` here:
# https://gitlab.com/gitlab-org/gitlab-ce/issues/50911
def
replyable_types
super
+
%w(Epic)
end
end
override
:note_etag_key
override
:note_etag_key
def
note_etag_key
def
note_etag_key
if
self
.
is_a?
(
Epic
)
if
self
.
is_a?
(
Epic
)
...
...
ee/spec/javascripts/notes/components/note_actions_spec.js
0 → 100644
View file @
e4eb6465
import
{
shallowMount
,
createLocalVue
}
from
'
@vue/test-utils
'
;
import
createStore
from
'
~/notes/stores
'
;
import
noteActions
from
'
~/notes/components/note_actions.vue
'
;
import
{
TEST_HOST
}
from
'
spec/test_constants
'
;
import
{
userDataMock
}
from
'
spec/notes/mock_data
'
;
describe
(
'
noteActions
'
,
()
=>
{
let
wrapper
;
let
store
;
let
props
;
const
createWrapper
=
propsData
=>
{
const
localVue
=
createLocalVue
();
return
shallowMount
(
noteActions
,
{
store
,
propsData
,
localVue
,
sync
:
false
,
});
};
beforeEach
(()
=>
{
store
=
createStore
();
props
=
{
accessLevel
:
'
Maintainer
'
,
authorId
:
26
,
canDelete
:
true
,
canEdit
:
true
,
canAwardEmoji
:
true
,
canReportAsAbuse
:
true
,
noteId
:
'
539
'
,
noteUrl
:
`
${
TEST_HOST
}
/group/project/merge_requests/1#note_1`
,
reportAbusePath
:
`
${
TEST_HOST
}
/abuse_reports/new?ref_url=http%3A%2F%2Flocalhost%3A3000%2Fgitlab-org%2Fgitlab-ce%2Fissues%2F7%23note_539&user_id=26`
,
showReply
:
false
,
isDraft
:
true
,
};
});
afterEach
(()
=>
{
wrapper
.
destroy
();
});
describe
(
'
Draft notes
'
,
()
=>
{
beforeEach
(()
=>
{
store
.
dispatch
(
'
setUserData
'
,
userDataMock
);
wrapper
=
createWrapper
(
props
);
});
it
(
'
should render the right resolve button title
'
,
()
=>
{
expect
(
wrapper
.
vm
.
resolveButtonTitle
).
toEqual
(
'
Discussion stays unresolved
'
);
});
});
});
ee/spec/models/concerns/ee/noteable_spec.rb
0 → 100644
View file @
e4eb6465
# frozen_string_literal: true
require
'rails_helper'
describe
EE
::
Noteable
do
describe
'.replyable_types'
do
it
'adds Epic to replyable_types after being included'
do
class
SomeClass
include
Noteable
end
expect
(
SomeClass
.
replyable_types
).
to
include
(
"Epic"
)
end
end
end
locale/gitlab.pot
View file @
e4eb6465
...
@@ -7723,6 +7723,9 @@ msgstr ""
...
@@ -7723,6 +7723,9 @@ msgstr ""
msgid "Repair authentication"
msgid "Repair authentication"
msgstr ""
msgstr ""
msgid "Reply to comment"
msgstr ""
msgid "Reply to this email directly or %{view_it_on_gitlab}."
msgid "Reply to this email directly or %{view_it_on_gitlab}."
msgstr ""
msgstr ""
...
...
spec/features/merge_request/user_posts_notes_spec.rb
View file @
e4eb6465
...
@@ -66,6 +66,38 @@ describe 'Merge request > User posts notes', :js do
...
@@ -66,6 +66,38 @@ describe 'Merge request > User posts notes', :js do
is_expected
.
to
have_css
(
'.js-note-text'
,
visible:
true
)
is_expected
.
to
have_css
(
'.js-note-text'
,
visible:
true
)
end
end
end
end
describe
'when reply_to_individual_notes feature flag is not set'
do
before
do
stub_feature_flags
(
reply_to_individual_notes:
false
)
visit
project_merge_request_path
(
project
,
merge_request
)
end
it
'does not show a reply button'
do
expect
(
page
).
to
have_no_selector
(
'.js-reply-button'
)
end
end
describe
'when reply_to_individual_notes feature flag is set'
do
before
do
stub_feature_flags
(
reply_to_individual_notes:
true
)
visit
project_merge_request_path
(
project
,
merge_request
)
end
it
'shows a reply button'
do
reply_button
=
find
(
'.js-reply-button'
,
match: :first
)
expect
(
reply_button
).
to
have_selector
(
'.ic-comment'
)
end
it
'shows reply placeholder when clicking reply button'
do
reply_button
=
find
(
'.js-reply-button'
,
match: :first
)
reply_button
.
click
expect
(
page
).
to
have_selector
(
'.discussion-reply-holder'
)
end
end
end
end
describe
'when previewing a note'
do
describe
'when previewing a note'
do
...
...
spec/javascripts/notes/components/note_actions/reply_button_spec.js
0 → 100644
View file @
e4eb6465
import
Vuex
from
'
vuex
'
;
import
{
createLocalVue
,
mount
}
from
'
@vue/test-utils
'
;
import
ReplyButton
from
'
~/notes/components/note_actions/reply_button.vue
'
;
describe
(
'
ReplyButton
'
,
()
=>
{
const
noteId
=
'
dummy-note-id
'
;
let
wrapper
;
let
convertToDiscussion
;
beforeEach
(()
=>
{
const
localVue
=
createLocalVue
();
convertToDiscussion
=
jasmine
.
createSpy
(
'
convertToDiscussion
'
);
localVue
.
use
(
Vuex
);
const
store
=
new
Vuex
.
Store
({
actions
:
{
convertToDiscussion
,
},
});
wrapper
=
mount
(
ReplyButton
,
{
propsData
:
{
noteId
,
},
store
,
sync
:
false
,
localVue
,
});
});
afterEach
(()
=>
{
wrapper
.
destroy
();
});
it
(
'
dispatches convertToDiscussion with note ID on click
'
,
()
=>
{
const
button
=
wrapper
.
find
({
ref
:
'
button
'
});
button
.
trigger
(
'
click
'
);
expect
(
convertToDiscussion
).
toHaveBeenCalledTimes
(
1
);
const
[,
payload
]
=
convertToDiscussion
.
calls
.
argsFor
(
0
);
expect
(
payload
).
toBe
(
noteId
);
});
});
spec/javascripts/notes/components/note_actions_spec.js
View file @
e4eb6465
...
@@ -2,14 +2,38 @@ import Vue from 'vue';
...
@@ -2,14 +2,38 @@ import Vue from 'vue';
import
{
shallowMount
,
createLocalVue
}
from
'
@vue/test-utils
'
;
import
{
shallowMount
,
createLocalVue
}
from
'
@vue/test-utils
'
;
import
createStore
from
'
~/notes/stores
'
;
import
createStore
from
'
~/notes/stores
'
;
import
noteActions
from
'
~/notes/components/note_actions.vue
'
;
import
noteActions
from
'
~/notes/components/note_actions.vue
'
;
import
{
TEST_HOST
}
from
'
spec/test_constants
'
;
import
{
userDataMock
}
from
'
../mock_data
'
;
import
{
userDataMock
}
from
'
../mock_data
'
;
describe
(
'
noteActions
'
,
()
=>
{
describe
(
'
noteActions
'
,
()
=>
{
let
wrapper
;
let
wrapper
;
let
store
;
let
store
;
let
props
;
const
createWrapper
=
propsData
=>
{
const
localVue
=
createLocalVue
();
return
shallowMount
(
noteActions
,
{
store
,
propsData
,
localVue
,
sync
:
false
,
});
};
beforeEach
(()
=>
{
beforeEach
(()
=>
{
store
=
createStore
();
store
=
createStore
();
props
=
{
accessLevel
:
'
Maintainer
'
,
authorId
:
26
,
canDelete
:
true
,
canEdit
:
true
,
canAwardEmoji
:
true
,
canReportAsAbuse
:
true
,
noteId
:
'
539
'
,
noteUrl
:
`
${
TEST_HOST
}
/group/project/merge_requests/1#note_1`
,
reportAbusePath
:
`
${
TEST_HOST
}
/abuse_reports/new?ref_url=http%3A%2F%2Flocalhost%3A3000%2Fgitlab-org%2Fgitlab-ce%2Fissues%2F7%23note_539&user_id=26`
,
showReply
:
false
,
};
});
});
afterEach
(()
=>
{
afterEach
(()
=>
{
...
@@ -17,31 +41,10 @@ describe('noteActions', () => {
...
@@ -17,31 +41,10 @@ describe('noteActions', () => {
});
});
describe
(
'
user is logged in
'
,
()
=>
{
describe
(
'
user is logged in
'
,
()
=>
{
let
props
;
beforeEach
(()
=>
{
beforeEach
(()
=>
{
props
=
{
accessLevel
:
'
Maintainer
'
,
authorId
:
26
,
canDelete
:
true
,
canEdit
:
true
,
canAwardEmoji
:
true
,
canReportAsAbuse
:
true
,
noteId
:
'
539
'
,
noteUrl
:
'
https://localhost:3000/group/project/merge_requests/1#note_1
'
,
reportAbusePath
:
'
/abuse_reports/new?ref_url=http%3A%2F%2Flocalhost%3A3000%2Fgitlab-org%2Fgitlab-ce%2Fissues%2F7%23note_539&user_id=26
'
,
};
store
.
dispatch
(
'
setUserData
'
,
userDataMock
);
store
.
dispatch
(
'
setUserData
'
,
userDataMock
);
const
localVue
=
createLocalVue
();
wrapper
=
createWrapper
(
props
);
wrapper
=
shallowMount
(
noteActions
,
{
store
,
propsData
:
props
,
localVue
,
sync
:
false
,
});
});
});
it
(
'
should render access level badge
'
,
()
=>
{
it
(
'
should render access level badge
'
,
()
=>
{
...
@@ -91,28 +94,14 @@ describe('noteActions', () => {
...
@@ -91,28 +94,14 @@ describe('noteActions', () => {
});
});
describe
(
'
user is not logged in
'
,
()
=>
{
describe
(
'
user is not logged in
'
,
()
=>
{
let
props
;
beforeEach
(()
=>
{
beforeEach
(()
=>
{
store
.
dispatch
(
'
setUserData
'
,
{});
store
.
dispatch
(
'
setUserData
'
,
{});
props
=
{
wrapper
=
createWrapper
({
accessLevel
:
'
Maintainer
'
,
...
props
,
authorId
:
26
,
canDelete
:
false
,
canDelete
:
false
,
canEdit
:
false
,
canEdit
:
false
,
canAwardEmoji
:
false
,
canAwardEmoji
:
false
,
canReportAsAbuse
:
false
,
canReportAsAbuse
:
false
,
noteId
:
'
539
'
,
noteUrl
:
'
https://localhost:3000/group/project/merge_requests/1#note_1
'
,
reportAbusePath
:
'
/abuse_reports/new?ref_url=http%3A%2F%2Flocalhost%3A3000%2Fgitlab-org%2Fgitlab-ce%2Fissues%2F7%23note_539&user_id=26
'
,
};
const
localVue
=
createLocalVue
();
wrapper
=
shallowMount
(
noteActions
,
{
store
,
propsData
:
props
,
localVue
,
sync
:
false
,
});
});
});
});
...
@@ -124,4 +113,88 @@ describe('noteActions', () => {
...
@@ -124,4 +113,88 @@ describe('noteActions', () => {
expect
(
wrapper
.
find
(
'
.more-actions
'
).
exists
()).
toBe
(
false
);
expect
(
wrapper
.
find
(
'
.more-actions
'
).
exists
()).
toBe
(
false
);
});
});
});
});
describe
(
'
with feature flag replyToIndividualNotes enabled
'
,
()
=>
{
beforeEach
(()
=>
{
gon
.
features
=
{
replyToIndividualNotes
:
true
,
};
});
afterEach
(()
=>
{
gon
.
features
=
{};
});
describe
(
'
for showReply = true
'
,
()
=>
{
beforeEach
(()
=>
{
wrapper
=
createWrapper
({
...
props
,
showReply
:
true
,
});
});
it
(
'
shows a reply button
'
,
()
=>
{
const
replyButton
=
wrapper
.
find
({
ref
:
'
replyButton
'
});
expect
(
replyButton
.
exists
()).
toBe
(
true
);
});
});
describe
(
'
for showReply = false
'
,
()
=>
{
beforeEach
(()
=>
{
wrapper
=
createWrapper
({
...
props
,
showReply
:
false
,
});
});
it
(
'
does not show a reply button
'
,
()
=>
{
const
replyButton
=
wrapper
.
find
({
ref
:
'
replyButton
'
});
expect
(
replyButton
.
exists
()).
toBe
(
false
);
});
});
});
describe
(
'
with feature flag replyToIndividualNotes disabled
'
,
()
=>
{
beforeEach
(()
=>
{
gon
.
features
=
{
replyToIndividualNotes
:
false
,
};
});
afterEach
(()
=>
{
gon
.
features
=
{};
});
describe
(
'
for showReply = true
'
,
()
=>
{
beforeEach
(()
=>
{
wrapper
=
createWrapper
({
...
props
,
showReply
:
true
,
});
});
it
(
'
does not show a reply button
'
,
()
=>
{
const
replyButton
=
wrapper
.
find
({
ref
:
'
replyButton
'
});
expect
(
replyButton
.
exists
()).
toBe
(
false
);
});
});
describe
(
'
for showReply = false
'
,
()
=>
{
beforeEach
(()
=>
{
wrapper
=
createWrapper
({
...
props
,
showReply
:
false
,
});
});
it
(
'
does not show a reply button
'
,
()
=>
{
const
replyButton
=
wrapper
.
find
({
ref
:
'
replyButton
'
});
expect
(
replyButton
.
exists
()).
toBe
(
false
);
});
});
});
});
});
spec/javascripts/notes/stores/actions_spec.js
View file @
e4eb6465
...
@@ -585,4 +585,18 @@ describe('Actions Notes Store', () => {
...
@@ -585,4 +585,18 @@ describe('Actions Notes Store', () => {
);
);
});
});
});
});
describe
(
'
convertToDiscussion
'
,
()
=>
{
it
(
'
commits CONVERT_TO_DISCUSSION with noteId
'
,
done
=>
{
const
noteId
=
'
dummy-note-id
'
;
testAction
(
actions
.
convertToDiscussion
,
noteId
,
{},
[{
type
:
'
CONVERT_TO_DISCUSSION
'
,
payload
:
noteId
}],
[],
done
,
);
});
});
});
});
spec/javascripts/notes/stores/mutation_spec.js
View file @
e4eb6465
...
@@ -517,4 +517,27 @@ describe('Notes Store mutations', () => {
...
@@ -517,4 +517,27 @@ describe('Notes Store mutations', () => {
);
);
});
});
});
});
describe
(
'
CONVERT_TO_DISCUSSION
'
,
()
=>
{
let
discussion
;
let
state
;
beforeEach
(()
=>
{
discussion
=
{
id
:
42
,
individual_note
:
true
,
};
state
=
{
discussions
:
[
discussion
]
};
});
it
(
'
toggles individual_note
'
,
()
=>
{
mutations
.
CONVERT_TO_DISCUSSION
(
state
,
discussion
.
id
);
expect
(
discussion
.
individual_note
).
toBe
(
false
);
});
it
(
'
throws if discussion was not found
'
,
()
=>
{
expect
(()
=>
mutations
.
CONVERT_TO_DISCUSSION
(
state
,
99
)).
toThrow
();
});
});
});
});
spec/lib/gitlab/email/handler/create_note_handler_spec.rb
View file @
e4eb6465
...
@@ -155,11 +155,7 @@ describe Gitlab::Email::Handler::CreateNoteHandler do
...
@@ -155,11 +155,7 @@ describe Gitlab::Email::Handler::CreateNoteHandler do
it_behaves_like
"checks permissions on noteable"
it_behaves_like
"checks permissions on noteable"
end
end
context
"when everything is fine"
do
shared_examples
'a reply to existing comment'
do
before
do
setup_attachment
end
it
"creates a comment"
do
it
"creates a comment"
do
expect
{
receiver
.
execute
}.
to
change
{
noteable
.
notes
.
count
}.
by
(
1
)
expect
{
receiver
.
execute
}.
to
change
{
noteable
.
notes
.
count
}.
by
(
1
)
new_note
=
noteable
.
notes
.
last
new_note
=
noteable
.
notes
.
last
...
@@ -168,7 +164,21 @@ describe Gitlab::Email::Handler::CreateNoteHandler do
...
@@ -168,7 +164,21 @@ describe Gitlab::Email::Handler::CreateNoteHandler do
expect
(
new_note
.
position
).
to
eq
(
note
.
position
)
expect
(
new_note
.
position
).
to
eq
(
note
.
position
)
expect
(
new_note
.
note
).
to
include
(
"I could not disagree more."
)
expect
(
new_note
.
note
).
to
include
(
"I could not disagree more."
)
expect
(
new_note
.
in_reply_to?
(
note
)).
to
be_truthy
expect
(
new_note
.
in_reply_to?
(
note
)).
to
be_truthy
if
note
.
part_of_discussion?
expect
(
new_note
.
discussion_id
).
to
eq
(
note
.
discussion_id
)
else
expect
(
new_note
.
discussion_id
).
not_to
eq
(
note
.
discussion_id
)
end
end
end
end
context
"when everything is fine"
do
before
do
setup_attachment
end
it_behaves_like
'a reply to existing comment'
it
"adds all attachments"
do
it
"adds all attachments"
do
receiver
.
execute
receiver
.
execute
...
@@ -207,4 +217,10 @@ describe Gitlab::Email::Handler::CreateNoteHandler do
...
@@ -207,4 +217,10 @@ describe Gitlab::Email::Handler::CreateNoteHandler do
end
end
end
end
end
end
context
"when note is not a discussion"
do
let
(
:note
)
{
create
(
:note_on_merge_request
,
project:
project
)
}
it_behaves_like
'a reply to existing comment'
end
end
end
spec/models/sent_notification_spec.rb
View file @
e4eb6465
...
@@ -36,19 +36,41 @@ describe SentNotification do
...
@@ -36,19 +36,41 @@ describe SentNotification do
end
end
end
end
shared_examples
'a successful sent notification'
do
it
'creates a new SentNotification'
do
expect
{
subject
}.
to
change
{
described_class
.
count
}.
by
(
1
)
end
end
describe
'.record'
do
describe
'.record'
do
let
(
:issue
)
{
create
(
:issue
)
}
let
(
:issue
)
{
create
(
:issue
)
}
it
'creates a new SentNotification'
do
subject
{
described_class
.
record
(
issue
,
user
.
id
)
}
expect
{
described_class
.
record
(
issue
,
user
.
id
)
}.
to
change
{
described_class
.
count
}.
by
(
1
)
end
it_behaves_like
'a successful sent notification'
end
end
describe
'.record_note'
do
describe
'.record_note'
do
let
(
:note
)
{
create
(
:diff_note_on_merge_request
)
}
subject
{
described_class
.
record_note
(
note
,
note
.
author
.
id
)
}
it
'creates a new SentNotification'
do
context
'for a discussion note'
do
expect
{
described_class
.
record_note
(
note
,
note
.
author
.
id
)
}.
to
change
{
described_class
.
count
}.
by
(
1
)
let
(
:note
)
{
create
(
:diff_note_on_merge_request
)
}
it_behaves_like
'a successful sent notification'
it
'sets in_reply_to_discussion_id'
do
expect
(
subject
.
in_reply_to_discussion_id
).
to
eq
(
note
.
discussion_id
)
end
end
context
'for an individual note'
do
let
(
:note
)
{
create
(
:note_on_merge_request
)
}
it_behaves_like
'a successful sent notification'
it
'does not set in_reply_to_discussion_id'
do
expect
(
subject
.
in_reply_to_discussion_id
).
to
be_nil
end
end
end
end
end
...
...
spec/services/notes/build_service_spec.rb
View file @
e4eb6465
...
@@ -123,6 +123,46 @@ describe Notes::BuildService do
...
@@ -123,6 +123,46 @@ describe Notes::BuildService do
end
end
end
end
context
'when replying to individual note'
do
let
(
:note
)
{
create
(
:note_on_issue
)
}
subject
{
described_class
.
new
(
project
,
author
,
note:
'Test'
,
in_reply_to_discussion_id:
note
.
discussion_id
).
execute
}
shared_examples
'an individual note reply'
do
it
'builds another individual note'
do
expect
(
subject
).
to
be_valid
expect
(
subject
).
to
be_a
(
Note
)
expect
(
subject
.
discussion_id
).
not_to
eq
(
note
.
discussion_id
)
end
end
context
'when reply_to_individual_notes is disabled'
do
before
do
stub_feature_flags
(
reply_to_individual_notes:
false
)
end
it_behaves_like
'an individual note reply'
end
context
'when reply_to_individual_notes is enabled'
do
before
do
stub_feature_flags
(
reply_to_individual_notes:
true
)
end
it
'sets the note up to be in reply to that note'
do
expect
(
subject
).
to
be_valid
expect
(
subject
).
to
be_a
(
DiscussionNote
)
expect
(
subject
.
discussion_id
).
to
eq
(
note
.
discussion_id
)
end
context
'when noteable does not support replies'
do
let
(
:note
)
{
create
(
:note_on_commit
)
}
it_behaves_like
'an individual note reply'
end
end
end
it
'builds a note without saving it'
do
it
'builds a note without saving it'
do
new_note
=
described_class
.
new
(
project
,
new_note
=
described_class
.
new
(
project
,
author
,
author
,
...
...
spec/services/notes/create_service_spec.rb
View file @
e4eb6465
...
@@ -278,5 +278,42 @@ describe Notes::CreateService do
...
@@ -278,5 +278,42 @@ describe Notes::CreateService do
expect
(
note
.
note
).
to
eq
(
':smile:'
)
expect
(
note
.
note
).
to
eq
(
':smile:'
)
end
end
end
end
context
'reply to individual note'
do
let
(
:existing_note
)
{
create
(
:note_on_issue
,
noteable:
issue
,
project:
project
)
}
let
(
:reply_opts
)
{
opts
.
merge
(
in_reply_to_discussion_id:
existing_note
.
discussion_id
)
}
subject
{
described_class
.
new
(
project
,
user
,
reply_opts
).
execute
}
context
'when reply_to_individual_notes is disabled'
do
before
do
stub_feature_flags
(
reply_to_individual_notes:
false
)
end
it
'creates an individual note'
do
expect
(
subject
.
type
).
to
eq
(
nil
)
expect
(
subject
.
discussion_id
).
not_to
eq
(
existing_note
.
discussion_id
)
end
it
'does not convert existing note'
do
expect
{
subject
}.
not_to
change
{
existing_note
.
reload
.
type
}
end
end
context
'when reply_to_individual_notes is enabled'
do
before
do
stub_feature_flags
(
reply_to_individual_notes:
true
)
end
it
'creates a DiscussionNote in reply to existing note'
do
expect
(
subject
).
to
be_a
(
DiscussionNote
)
expect
(
subject
.
discussion_id
).
to
eq
(
existing_note
.
discussion_id
)
end
it
'converts existing note to DiscussionNote'
do
expect
{
subject
}.
to
change
{
existing_note
.
reload
.
type
}.
from
(
nil
).
to
(
'DiscussionNote'
)
end
end
end
end
end
end
end
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment