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
5c42ad75
Commit
5c42ad75
authored
Sep 07, 2018
by
Jan Provaznik
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
[EE] Use ResourceLabelEvent for tracking label changes
parent
d8a44cef
Changes
64
Hide whitespace changes
Inline
Side-by-side
Showing
64 changed files
with
1339 additions
and
189 deletions
+1339
-189
app/assets/javascripts/notes/components/note_actions.vue
app/assets/javascripts/notes/components/note_actions.vue
+5
-4
app/assets/javascripts/notes/components/note_awards_list.vue
app/assets/javascripts/notes/components/note_awards_list.vue
+1
-1
app/assets/javascripts/notes/components/note_form.vue
app/assets/javascripts/notes/components/note_form.vue
+6
-3
app/assets/javascripts/notes/components/note_header.vue
app/assets/javascripts/notes/components/note_header.vue
+10
-3
app/controllers/concerns/issuable_actions.rb
app/controllers/concerns/issuable_actions.rb
+1
-0
app/controllers/concerns/notes_actions.rb
app/controllers/concerns/notes_actions.rb
+1
-0
app/helpers/notes_helper.rb
app/helpers/notes_helper.rb
+1
-1
app/models/concerns/issuable.rb
app/models/concerns/issuable.rb
+0
-4
app/models/concerns/noteable.rb
app/models/concerns/noteable.rb
+21
-0
app/models/label_note.rb
app/models/label_note.rb
+97
-0
app/models/note.rb
app/models/note.rb
+1
-12
app/models/resource_label_event.rb
app/models/resource_label_event.rb
+94
-6
app/serializers/note_entity.rb
app/serializers/note_entity.rb
+8
-2
app/serializers/project_note_entity.rb
app/serializers/project_note_entity.rb
+2
-2
app/services/issuable/common_system_notes_service.rb
app/services/issuable/common_system_notes_service.rb
+3
-1
app/services/issues/move_service.rb
app/services/issues/move_service.rb
+13
-0
app/services/labels/promote_service.rb
app/services/labels/promote_service.rb
+7
-0
app/services/resource_events/change_labels_service.rb
app/services/resource_events/change_labels_service.rb
+1
-2
app/services/resource_events/merge_into_notes_service.rb
app/services/resource_events/merge_into_notes_service.rb
+55
-0
app/services/system_note_service.rb
app/services/system_note_service.rb
+0
-41
changelogs/unreleased/label-event.yml
changelogs/unreleased/label-event.yml
+6
-0
db/migrate/20180901200537_add_resource_label_event_reference_fields.rb
...180901200537_add_resource_label_event_reference_fields.rb
+11
-0
db/schema.rb
db/schema.rb
+4
-1
doc/api/README.md
doc/api/README.md
+1
-0
doc/api/resource_label_events.md
doc/api/resource_label_events.md
+261
-0
ee/app/models/concerns/ee/noteable.rb
ee/app/models/concerns/ee/noteable.rb
+17
-0
ee/app/models/ee/epic.rb
ee/app/models/ee/epic.rb
+5
-1
ee/app/models/ee/note.rb
ee/app/models/ee/note.rb
+0
-9
ee/app/models/ee/resource_label_event.rb
ee/app/models/ee/resource_label_event.rb
+2
-2
ee/app/serializers/epic_note_entity.rb
ee/app/serializers/epic_note_entity.rb
+1
-1
ee/spec/controllers/projects/issues_controller_spec.rb
ee/spec/controllers/projects/issues_controller_spec.rb
+2
-2
ee/spec/models/label_note_spec.rb
ee/spec/models/label_note_spec.rb
+18
-0
lib/api/api.rb
lib/api/api.rb
+1
-0
lib/api/entities.rb
lib/api/entities.rb
+14
-0
lib/api/resource_label_events.rb
lib/api/resource_label_events.rb
+52
-0
lib/banzai/pipeline/label_pipeline.rb
lib/banzai/pipeline/label_pipeline.rb
+14
-0
lib/gitlab/import_export/import_export.yml
lib/gitlab/import_export/import_export.yml
+10
-0
locale/gitlab.pot
locale/gitlab.pot
+3
-0
spec/controllers/projects/notes_controller_spec.rb
spec/controllers/projects/notes_controller_spec.rb
+1
-1
spec/factories/resource_label_events.rb
spec/factories/resource_label_events.rb
+5
-2
spec/features/issues/resource_label_events_spec.rb
spec/features/issues/resource_label_events_spec.rb
+60
-0
spec/javascripts/diffs/mock_data/diff_discussions.js
spec/javascripts/diffs/mock_data/diff_discussions.js
+5
-5
spec/javascripts/notes/components/note_actions_spec.js
spec/javascripts/notes/components/note_actions_spec.js
+16
-2
spec/javascripts/notes/components/note_awards_list_spec.js
spec/javascripts/notes/components/note_awards_list_spec.js
+2
-2
spec/javascripts/notes/components/note_form_spec.js
spec/javascripts/notes/components/note_form_spec.js
+17
-1
spec/javascripts/notes/components/note_header_spec.js
spec/javascripts/notes/components/note_header_spec.js
+12
-2
spec/javascripts/notes/mock_data.js
spec/javascripts/notes/mock_data.js
+23
-23
spec/javascripts/vue_shared/components/notes/system_note_spec.js
...vascripts/vue_shared/components/notes/system_note_spec.js
+1
-1
spec/lib/gitlab/import_export/all_models.yml
spec/lib/gitlab/import_export/all_models.yml
+6
-0
spec/lib/gitlab/import_export/project.json
spec/lib/gitlab/import_export/project.json
+33
-0
spec/lib/gitlab/import_export/project_tree_restorer_spec.rb
spec/lib/gitlab/import_export/project_tree_restorer_spec.rb
+8
-0
spec/lib/gitlab/import_export/project_tree_saver_spec.rb
spec/lib/gitlab/import_export/project_tree_saver_spec.rb
+11
-0
spec/lib/gitlab/import_export/safe_model_attributes.yml
spec/lib/gitlab/import_export/safe_model_attributes.yml
+8
-0
spec/models/label_note_spec.rb
spec/models/label_note_spec.rb
+23
-0
spec/models/resource_label_event_spec.rb
spec/models/resource_label_event_spec.rb
+49
-3
spec/requests/api/resource_label_events_spec.rb
spec/requests/api/resource_label_events_spec.rb
+91
-0
spec/services/issuable/common_system_notes_service_spec.rb
spec/services/issuable/common_system_notes_service_spec.rb
+11
-2
spec/services/issues/move_service_spec.rb
spec/services/issues/move_service_spec.rb
+11
-0
spec/services/issues/update_service_spec.rb
spec/services/issues/update_service_spec.rb
+5
-4
spec/services/merge_requests/update_service_spec.rb
spec/services/merge_requests/update_service_spec.rb
+5
-4
spec/services/resource_events/change_labels_service_spec.rb
spec/services/resource_events/change_labels_service_spec.rb
+8
-0
spec/services/resource_events/merge_into_notes_service_spec.rb
...services/resource_events/merge_into_notes_service_spec.rb
+70
-0
spec/services/system_note_service_spec.rb
spec/services/system_note_service_spec.rb
+0
-39
spec/support/shared_examples/models/label_note_shared_examples.rb
...port/shared_examples/models/label_note_shared_examples.rb
+109
-0
No files found.
app/assets/javascripts/notes/components/note_actions.vue
View file @
5c42ad75
...
@@ -24,12 +24,13 @@ export default {
...
@@ -24,12 +24,13 @@ export default {
required
:
true
,
required
:
true
,
},
},
noteId
:
{
noteId
:
{
type
:
Number
,
type
:
String
,
required
:
true
,
required
:
true
,
},
},
noteUrl
:
{
noteUrl
:
{
type
:
String
,
type
:
String
,
required
:
true
,
required
:
false
,
default
:
''
,
},
},
accessLevel
:
{
accessLevel
:
{
type
:
String
,
type
:
String
,
...
@@ -225,11 +226,11 @@ export default {
...
@@ -225,11 +226,11 @@ export default {
Report as abuse
Report as abuse
</a>
</a>
</li>
</li>
<li>
<li
v-if=
"noteUrl"
>
<button
<button
:data-clipboard-text=
"noteUrl"
:data-clipboard-text=
"noteUrl"
type=
"button"
type=
"button"
c
ss-class=
"btn-default btn-transparent
"
c
lass=
"btn-default btn-transparent js-btn-copy-note-link
"
>
>
Copy link
Copy link
</button>
</button>
...
...
app/assets/javascripts/notes/components/note_awards_list.vue
View file @
5c42ad75
...
@@ -25,7 +25,7 @@ export default {
...
@@ -25,7 +25,7 @@ export default {
required
:
true
,
required
:
true
,
},
},
noteId
:
{
noteId
:
{
type
:
Number
,
type
:
String
,
required
:
true
,
required
:
true
,
},
},
canAwardEmoji
:
{
canAwardEmoji
:
{
...
...
app/assets/javascripts/notes/components/note_form.vue
View file @
5c42ad75
...
@@ -20,9 +20,9 @@ export default {
...
@@ -20,9 +20,9 @@ export default {
default
:
''
,
default
:
''
,
},
},
noteId
:
{
noteId
:
{
type
:
Number
,
type
:
String
,
required
:
false
,
required
:
false
,
default
:
0
,
default
:
''
,
},
},
markdownVersion
:
{
markdownVersion
:
{
type
:
Number
,
type
:
Number
,
...
@@ -67,7 +67,10 @@ export default {
...
@@ -67,7 +67,10 @@ export default {
'
getUserDataByProp
'
,
'
getUserDataByProp
'
,
]),
]),
noteHash
()
{
noteHash
()
{
return
`#note_
${
this
.
noteId
}
`
;
if
(
this
.
noteId
)
{
return
`#note_
${
this
.
noteId
}
`
;
}
return
'
#
'
;
},
},
markdownPreviewPath
()
{
markdownPreviewPath
()
{
return
this
.
getNoteableDataByProp
(
'
preview_note_path
'
);
return
this
.
getNoteableDataByProp
(
'
preview_note_path
'
);
...
...
app/assets/javascripts/notes/components/note_header.vue
View file @
5c42ad75
...
@@ -9,7 +9,8 @@ export default {
...
@@ -9,7 +9,8 @@ export default {
props
:
{
props
:
{
author
:
{
author
:
{
type
:
Object
,
type
:
Object
,
required
:
true
,
required
:
false
,
default
:
()
=>
({}),
},
},
createdAt
:
{
createdAt
:
{
type
:
String
,
type
:
String
,
...
@@ -21,7 +22,7 @@ export default {
...
@@ -21,7 +22,7 @@ export default {
default
:
''
,
default
:
''
,
},
},
noteId
:
{
noteId
:
{
type
:
Number
,
type
:
String
,
required
:
true
,
required
:
true
,
},
},
includeToggle
:
{
includeToggle
:
{
...
@@ -72,7 +73,10 @@ export default {
...
@@ -72,7 +73,10 @@ export default {
{{
__
(
'
Toggle discussion
'
)
}}
{{
__
(
'
Toggle discussion
'
)
}}
</button>
</button>
</div>
</div>
<a
:href=
"author.path"
>
<a
v-if=
"Object.keys(author).length"
:href=
"author.path"
>
<span
class=
"note-header-author-name"
>
{{
author
.
name
}}
</span>
<span
class=
"note-header-author-name"
>
{{
author
.
name
}}
</span>
<span
<span
v-if=
"author.status_tooltip_html"
v-if=
"author.status_tooltip_html"
...
@@ -81,6 +85,9 @@ export default {
...
@@ -81,6 +85,9 @@ export default {
@
{{
author
.
username
}}
@
{{
author
.
username
}}
</span>
</span>
</a>
</a>
<span
v-else
>
{{
__
(
'
A deleted user
'
)
}}
</span>
<span
class=
"note-headline-light"
>
<span
class=
"note-headline-light"
>
<span
class=
"note-headline-meta"
>
<span
class=
"note-headline-meta"
>
<template
v-if=
"actionText"
>
<template
v-if=
"actionText"
>
...
...
app/controllers/concerns/issuable_actions.rb
View file @
5c42ad75
...
@@ -96,6 +96,7 @@ module IssuableActions
...
@@ -96,6 +96,7 @@ module IssuableActions
.
includes
(
:noteable
)
.
includes
(
:noteable
)
.
fresh
.
fresh
notes
=
ResourceEvents
::
MergeIntoNotesService
.
new
(
issuable
,
current_user
).
execute
(
notes
)
notes
=
prepare_notes_for_rendering
(
notes
)
notes
=
prepare_notes_for_rendering
(
notes
)
notes
=
notes
.
reject
{
|
n
|
n
.
cross_reference_not_visible_for?
(
current_user
)
}
notes
=
notes
.
reject
{
|
n
|
n
.
cross_reference_not_visible_for?
(
current_user
)
}
...
...
app/controllers/concerns/notes_actions.rb
View file @
5c42ad75
...
@@ -18,6 +18,7 @@ module NotesActions
...
@@ -18,6 +18,7 @@ module NotesActions
notes
=
notes_finder
.
execute
notes
=
notes_finder
.
execute
.
inc_relations_for_view
.
inc_relations_for_view
notes
=
ResourceEvents
::
MergeIntoNotesService
.
new
(
noteable
,
current_user
,
last_fetched_at:
current_fetched_at
).
execute
(
notes
)
notes
=
prepare_notes_for_rendering
(
notes
)
notes
=
prepare_notes_for_rendering
(
notes
)
notes
=
notes
.
reject
{
|
n
|
n
.
cross_reference_not_visible_for?
(
current_user
)
}
notes
=
notes
.
reject
{
|
n
|
n
.
cross_reference_not_visible_for?
(
current_user
)
}
...
...
app/helpers/notes_helper.rb
View file @
5c42ad75
...
@@ -110,7 +110,7 @@ module NotesHelper
...
@@ -110,7 +110,7 @@ module NotesHelper
end
end
def
noteable_note_url
(
note
)
def
noteable_note_url
(
note
)
Gitlab
::
UrlBuilder
.
build
(
note
)
Gitlab
::
UrlBuilder
.
build
(
note
)
if
note
.
id
end
end
def
form_resources
def
form_resources
...
...
app/models/concerns/issuable.rb
View file @
5c42ad75
...
@@ -111,10 +111,6 @@ module Issuable
...
@@ -111,10 +111,6 @@ module Issuable
def
allows_multiple_assignees?
def
allows_multiple_assignees?
false
false
end
end
def
etag_caching_enabled?
false
end
end
end
class_methods
do
class_methods
do
...
...
app/models/concerns/noteable.rb
View file @
5c42ad75
# frozen_string_literal: true
# frozen_string_literal: true
module
Noteable
module
Noteable
prepend
EE
::
Noteable
# Names of all implementers of `Noteable` that support resolvable notes.
# Names of all implementers of `Noteable` that support resolvable notes.
RESOLVABLE_TYPES
=
%w(MergeRequest)
.
freeze
RESOLVABLE_TYPES
=
%w(MergeRequest)
.
freeze
...
@@ -82,4 +84,23 @@ module Noteable
...
@@ -82,4 +84,23 @@ module Noteable
def
lockable?
def
lockable?
[
MergeRequest
,
Issue
].
include?
(
self
.
class
)
[
MergeRequest
,
Issue
].
include?
(
self
.
class
)
end
end
def
etag_caching_enabled?
false
end
def
expire_note_etag_cache
return
unless
discussions_rendered_on_frontend?
return
unless
etag_caching_enabled?
Gitlab
::
EtagCaching
::
Store
.
new
.
touch
(
note_etag_key
)
end
def
note_etag_key
Gitlab
::
Routing
.
url_helpers
.
project_noteable_notes_path
(
project
,
target_type:
self
.
class
.
name
.
underscore
,
target_id:
id
)
end
end
end
app/models/label_note.rb
0 → 100644
View file @
5c42ad75
# frozen_string_literal: true
class
LabelNote
<
Note
attr_accessor
:resource_parent
attr_reader
:events
def
self
.
from_events
(
events
,
resource:
nil
,
resource_parent:
nil
)
resource
||=
events
.
first
.
issuable
attrs
=
{
system:
true
,
author:
events
.
first
.
user
,
created_at:
events
.
first
.
created_at
,
discussion_id:
events
.
first
.
discussion_id
,
noteable:
resource
,
system_note_metadata:
SystemNoteMetadata
.
new
(
action:
'label'
),
events:
events
,
resource_parent:
resource_parent
}
if
resource_parent
.
is_a?
(
Project
)
attrs
[
:project_id
]
=
resource_parent
.
id
end
LabelNote
.
new
(
attrs
)
end
def
events
=
(
events
)
@events
=
events
update_outdated_markdown
end
def
cached_html_up_to_date?
(
markdown_field
)
true
end
def
note
@note
||=
note_text
end
def
note_html
@note_html
||=
"<p dir=
\"
auto
\"
>
#{
note_text
(
html:
true
)
}
</p>"
end
def
project
resource_parent
if
resource_parent
.
is_a?
(
Project
)
end
def
group
resource_parent
if
resource_parent
.
is_a?
(
Group
)
end
private
def
update_outdated_markdown
events
.
each
do
|
event
|
if
event
.
outdated_markdown?
event
.
refresh_invalid_reference
end
end
end
def
note_text
(
html:
false
)
added
=
labels_str
(
'added'
,
label_refs_by_action
(
'add'
,
html
))
removed
=
labels_str
(
'removed'
,
label_refs_by_action
(
'remove'
,
html
))
[
added
,
removed
].
compact
.
join
(
' and '
)
end
# returns string containing added/removed labels including
# count of deleted labels:
#
# added ~1 ~2 + 1 deleted label
# added 3 deleted labels
# added ~1 ~2 labels
def
labels_str
(
prefix
,
label_refs
)
existing_refs
=
label_refs
.
select
{
|
ref
|
ref
.
present?
}.
sort
refs_str
=
existing_refs
.
empty?
?
nil
:
existing_refs
.
join
(
' '
)
deleted
=
label_refs
.
count
-
existing_refs
.
count
deleted_str
=
deleted
==
0
?
nil
:
"
#{
deleted
}
deleted"
return
nil
unless
refs_str
||
deleted_str
label_list_str
=
[
refs_str
,
deleted_str
].
compact
.
join
(
' + '
)
suffix
=
'label'
.
pluralize
(
deleted
>
0
?
deleted
:
existing_refs
.
count
)
"
#{
prefix
}
#{
label_list_str
}
#{
suffix
}
"
end
def
label_refs_by_action
(
action
,
html
)
field
=
html
?
:reference_html
:
:reference
events
.
select
{
|
e
|
e
.
action
==
action
}.
map
(
&
field
)
end
end
app/models/note.rb
View file @
5c42ad75
...
@@ -397,18 +397,7 @@ class Note < ActiveRecord::Base
...
@@ -397,18 +397,7 @@ class Note < ActiveRecord::Base
end
end
def
expire_etag_cache
def
expire_etag_cache
return
unless
noteable
&
.
discussions_rendered_on_frontend?
noteable
&
.
expire_note_etag_cache
return
unless
noteable
&
.
etag_caching_enabled?
Gitlab
::
EtagCaching
::
Store
.
new
.
touch
(
etag_key
)
end
def
etag_key
Gitlab
::
Routing
.
url_helpers
.
project_noteable_notes_path
(
project
,
target_type:
noteable_type
.
underscore
,
target_id:
noteable_id
)
end
end
def
touch
(
*
args
)
def
touch
(
*
args
)
...
...
app/models/resource_label_event.rb
View file @
5c42ad75
...
@@ -4,34 +4,122 @@
...
@@ -4,34 +4,122 @@
# https://gitlab.com/gitlab-org/gitlab-ce/issues/48483
# https://gitlab.com/gitlab-org/gitlab-ce/issues/48483
class
ResourceLabelEvent
<
ActiveRecord
::
Base
class
ResourceLabelEvent
<
ActiveRecord
::
Base
prepend
EE
::
ResourceLabelEvent
prepend
EE
::
ResourceLabelEvent
include
Importable
include
Gitlab
::
Utils
::
StrongMemoize
include
CacheMarkdownField
cache_markdown_field
:reference
belongs_to
:user
belongs_to
:user
belongs_to
:issue
belongs_to
:issue
belongs_to
:merge_request
belongs_to
:merge_request
belongs_to
:label
belongs_to
:label
validates
:user
,
presence:
true
,
on: :create
scope
:created_after
,
->
(
time
)
{
where
(
'created_at > ?'
,
time
)
}
validates
:label
,
presence:
true
,
on: :create
validates
:user
,
presence:
{
unless: :importing?
},
on: :create
validates
:label
,
presence:
{
unless: :importing?
},
on: :create
validate
:exactly_one_issuable
validate
:exactly_one_issuable
after_save
:expire_etag_cache
after_destroy
:expire_etag_cache
enum
action:
{
enum
action:
{
add:
1
,
add:
1
,
remove:
2
remove:
2
}
}
def
self
.
issuable_
column
s
def
self
.
issuable_
attr
s
%i(issue
_id merge_request_id
)
.
freeze
%i(issue
merge_request
)
.
freeze
end
end
def
issuable
def
issuable
issue
||
merge_request
issue
||
merge_request
end
end
# create same discussion id for all actions with the same user and time
def
discussion_id
(
resource
=
nil
)
strong_memoize
(
:discussion_id
)
do
Digest
::
SHA1
.
hexdigest
([
self
.
class
.
name
,
created_at
,
user_id
].
join
(
"-"
))
end
end
def
project
issuable
.
project
end
def
group
issuable
.
group
if
issuable
.
respond_to?
(
:group
)
end
def
outdated_markdown?
return
true
if
label_id
.
nil?
&&
reference
.
present?
reference
.
nil?
||
latest_cached_markdown_version
!=
cached_markdown_version
end
def
banzai_render_context
(
field
)
super
.
merge
(
pipeline:
'label'
,
only_path:
true
)
end
def
refresh_invalid_reference
# label_id could be nullified on label delete
self
.
reference
=
''
if
label_id
.
nil?
# reference is not set for events which were not rendered yet
self
.
reference
||=
label_reference
if
changed?
save
elsif
invalidated_markdown_cache?
refresh_markdown_cache!
end
end
private
private
def
label_reference
if
local_label?
label
.
to_reference
(
format: :id
)
elsif
label
.
is_a?
(
GroupLabel
)
label
.
to_reference
(
label
.
group
,
target_project:
resource_parent
,
format: :id
)
else
label
.
to_reference
(
resource_parent
,
format: :id
)
end
end
def
exactly_one_issuable
def
exactly_one_issuable
if
self
.
class
.
issuable_columns
.
count
{
|
attr
|
self
[
attr
]
}
!=
1
issuable_count
=
self
.
class
.
issuable_attrs
.
count
{
|
attr
|
self
[
"
#{
attr
}
_id"
]
}
errors
.
add
(
:base
,
"Exactly one of
#{
self
.
class
.
issuable_columns
.
join
(
', '
)
}
is required"
)
return
true
if
issuable_count
==
1
# if none of issuable IDs is set, check explicitly if nested issuable
# object is set, this is used during project import
if
issuable_count
==
0
&&
importing?
issuable_count
=
self
.
class
.
issuable_attrs
.
count
{
|
attr
|
self
.
public_send
(
attr
)
}
# rubocop:disable GitlabSecurity/PublicSend
return
true
if
issuable_count
==
1
end
end
errors
.
add
(
:base
,
"Exactly one of
#{
self
.
class
.
issuable_attrs
.
join
(
', '
)
}
is required"
)
end
def
expire_etag_cache
issuable
.
expire_note_etag_cache
end
def
local_label?
params
=
{
include_ancestor_groups:
true
}
if
resource_parent
.
is_a?
(
Project
)
params
[
:project_id
]
=
resource_parent
.
id
else
params
[
:group_id
]
=
resource_parent
.
id
end
LabelsFinder
.
new
(
nil
,
params
).
execute
(
skip_authorization:
true
).
where
(
id:
label
.
id
).
any?
end
def
resource_parent
issuable
.
project
||
issuable
.
group
end
end
end
end
app/serializers/note_entity.rb
View file @
5c42ad75
...
@@ -4,6 +4,12 @@ class NoteEntity < API::Entities::Note
...
@@ -4,6 +4,12 @@ class NoteEntity < API::Entities::Note
include
RequestAwareEntity
include
RequestAwareEntity
include
NotesHelper
include
NotesHelper
expose
:id
do
|
note
|
# resource events are represented as notes too, but don't
# have ID, discussion ID is used for them instead
note
.
id
?
note
.
id
.
to_s
:
note
.
discussion_id
end
expose
:type
expose
:type
expose
:author
,
using:
NoteUserEntity
expose
:author
,
using:
NoteUserEntity
...
@@ -46,8 +52,8 @@ class NoteEntity < API::Entities::Note
...
@@ -46,8 +52,8 @@ class NoteEntity < API::Entities::Note
expose
:emoji_awardable?
,
as: :emoji_awardable
expose
:emoji_awardable?
,
as: :emoji_awardable
expose
:award_emoji
,
if:
->
(
note
,
_
)
{
note
.
emoji_awardable?
},
using:
AwardEmojiEntity
expose
:award_emoji
,
if:
->
(
note
,
_
)
{
note
.
emoji_awardable?
},
using:
AwardEmojiEntity
expose
:report_abuse_path
do
|
note
|
expose
:report_abuse_path
,
if:
->
(
note
,
_
)
{
note
.
author_id
}
do
|
note
|
new_abuse_report_path
(
user_id:
note
.
author
.
id
,
ref_url:
Gitlab
::
UrlBuilder
.
build
(
note
))
new_abuse_report_path
(
user_id:
note
.
author
_
id
,
ref_url:
Gitlab
::
UrlBuilder
.
build
(
note
))
end
end
expose
:noteable_note_url
do
|
note
|
expose
:noteable_note_url
do
|
note
|
...
...
app/serializers/project_note_entity.rb
View file @
5c42ad75
# frozen_string_literal: true
# frozen_string_literal: true
class
ProjectNoteEntity
<
NoteEntity
class
ProjectNoteEntity
<
NoteEntity
expose
:human_access
do
|
note
|
expose
:human_access
,
if:
->
(
note
,
_
)
{
note
.
project
.
present?
}
do
|
note
|
note
.
project
.
team
.
human_max_access
(
note
.
author_id
)
note
.
project
.
team
.
human_max_access
(
note
.
author_id
)
end
end
...
@@ -9,7 +9,7 @@ class ProjectNoteEntity < NoteEntity
...
@@ -9,7 +9,7 @@ class ProjectNoteEntity < NoteEntity
toggle_award_emoji_project_note_path
(
note
.
project
,
note
.
id
)
toggle_award_emoji_project_note_path
(
note
.
project
,
note
.
id
)
end
end
expose
:path
do
|
note
|
expose
:path
,
if:
->
(
note
,
_
)
{
note
.
id
}
do
|
note
|
project_note_path
(
note
.
project
,
note
)
project_note_path
(
note
.
project
,
note
)
end
end
...
...
app/services/issuable/common_system_notes_service.rb
View file @
5c42ad75
...
@@ -56,7 +56,9 @@ module Issuable
...
@@ -56,7 +56,9 @@ module Issuable
added_labels
=
issuable
.
labels
-
old_labels
added_labels
=
issuable
.
labels
-
old_labels
removed_labels
=
old_labels
-
issuable
.
labels
removed_labels
=
old_labels
-
issuable
.
labels
SystemNoteService
.
change_label
(
issuable
,
issuable
.
project
,
current_user
,
added_labels
,
removed_labels
)
ResourceEvents
::
ChangeLabelsService
.
new
(
issuable
,
current_user
)
.
execute
(
added_labels:
added_labels
,
removed_labels:
removed_labels
)
end
end
def
create_title_change_note
(
old_title
)
def
create_title_change_note
(
old_title
)
...
...
app/services/issues/move_service.rb
View file @
5c42ad75
...
@@ -38,6 +38,7 @@ module Issues
...
@@ -38,6 +38,7 @@ module Issues
def
update_new_issue
def
update_new_issue
rewrite_notes
rewrite_notes
copy_resource_label_events
rewrite_issue_award_emoji
rewrite_issue_award_emoji
add_note_moved_from
add_note_moved_from
end
end
...
@@ -98,6 +99,18 @@ module Issues
...
@@ -98,6 +99,18 @@ module Issues
end
end
end
end
def
copy_resource_label_events
@old_issue
.
resource_label_events
.
find_in_batches
do
|
batch
|
events
=
batch
.
map
do
|
event
|
event
.
attributes
.
except
(
'id'
,
'reference'
,
'reference_html'
)
.
merge
(
'issue_id'
=>
@new_issue
.
id
,
'created_at'
=>
event
.
created_at
)
end
Gitlab
::
Database
.
bulk_insert
(
ResourceLabelEvent
.
table_name
,
events
)
end
end
def
rewrite_issue_award_emoji
def
rewrite_issue_award_emoji
rewrite_award_emoji
(
@old_issue
,
@new_issue
)
rewrite_award_emoji
(
@old_issue
,
@new_issue
)
end
end
...
...
app/services/labels/promote_service.rb
View file @
5c42ad75
...
@@ -13,6 +13,7 @@ module Labels
...
@@ -13,6 +13,7 @@ module Labels
label_ids_for_merge
(
new_label
).
find_in_batches
(
batch_size:
BATCH_SIZE
)
do
|
batched_ids
|
label_ids_for_merge
(
new_label
).
find_in_batches
(
batch_size:
BATCH_SIZE
)
do
|
batched_ids
|
update_issuables
(
new_label
,
batched_ids
)
update_issuables
(
new_label
,
batched_ids
)
update_resource_label_events
(
new_label
,
batched_ids
)
update_issue_board_lists
(
new_label
,
batched_ids
)
update_issue_board_lists
(
new_label
,
batched_ids
)
update_priorities
(
new_label
,
batched_ids
)
update_priorities
(
new_label
,
batched_ids
)
subscribe_users
(
new_label
,
batched_ids
)
subscribe_users
(
new_label
,
batched_ids
)
...
@@ -52,6 +53,12 @@ module Labels
...
@@ -52,6 +53,12 @@ module Labels
.
update_all
(
label_id:
new_label
)
.
update_all
(
label_id:
new_label
)
end
end
def
update_resource_label_events
(
new_label
,
label_ids
)
ResourceLabelEvent
.
where
(
label:
label_ids
)
.
update_all
(
label_id:
new_label
)
end
def
update_issue_board_lists
(
new_label
,
label_ids
)
def
update_issue_board_lists
(
new_label
,
label_ids
)
List
List
.
where
(
label:
label_ids
)
.
where
(
label:
label_ids
)
...
...
app/services/resource_events/change_labels_service.rb
View file @
5c42ad75
# frozen_string_literal: true
# frozen_string_literal: true
# This service is not used yet, it will be used for:
# https://gitlab.com/gitlab-org/gitlab-ce/issues/48483
module
ResourceEvents
module
ResourceEvents
class
ChangeLabelsService
class
ChangeLabelsService
prepend
EE
::
ResourceEvents
::
ChangeLabelsService
prepend
EE
::
ResourceEvents
::
ChangeLabelsService
...
@@ -27,6 +25,7 @@ module ResourceEvents
...
@@ -27,6 +25,7 @@ module ResourceEvents
end
end
Gitlab
::
Database
.
bulk_insert
(
ResourceLabelEvent
.
table_name
,
labels
)
Gitlab
::
Database
.
bulk_insert
(
ResourceLabelEvent
.
table_name
,
labels
)
resource
.
expire_note_etag_cache
end
end
private
private
...
...
app/services/resource_events/merge_into_notes_service.rb
0 → 100644
View file @
5c42ad75
# frozen_string_literal: true
# We store events about issuable label changes in a separate table (not as
# other system notes), but we still want to display notes about label changes
# as classic system notes in UI. This service generates "synthetic" notes for
# label event changes and merges them with classic notes and sorts them by
# creation time.
module
ResourceEvents
class
MergeIntoNotesService
include
Gitlab
::
Utils
::
StrongMemoize
attr_reader
:resource
,
:current_user
,
:params
def
initialize
(
resource
,
current_user
,
params
=
{})
@resource
=
resource
@current_user
=
current_user
@params
=
params
end
def
execute
(
notes
=
[])
(
notes
+
label_notes
).
sort_by
{
|
n
|
n
.
created_at
}
end
private
def
label_notes
label_events_by_discussion_id
.
map
do
|
discussion_id
,
events
|
LabelNote
.
from_events
(
events
,
resource:
resource
,
resource_parent:
resource_parent
)
end
end
def
label_events_by_discussion_id
return
[]
unless
resource
.
respond_to?
(
:resource_label_events
)
events
=
resource
.
resource_label_events
.
includes
(
:label
,
:user
)
events
=
since_fetch_at
(
events
)
events
.
group_by
{
|
event
|
event
.
discussion_id
}
end
def
since_fetch_at
(
events
)
return
events
unless
params
[
:last_fetched_at
].
present?
last_fetched_at
=
Time
.
at
(
params
.
fetch
(
:last_fetched_at
).
to_i
)
events
.
created_after
(
last_fetched_at
-
NotesFinder
::
FETCH_OVERLAP
)
end
def
resource_parent
strong_memoize
(
:resource_parent
)
do
resource
.
project
||
resource
.
group
end
end
end
end
app/services/system_note_service.rb
View file @
5c42ad75
...
@@ -99,47 +99,6 @@ module SystemNoteService
...
@@ -99,47 +99,6 @@ module SystemNoteService
create_note
(
NoteSummary
.
new
(
issue
,
project
,
author
,
body
,
action:
'assignee'
))
create_note
(
NoteSummary
.
new
(
issue
,
project
,
author
,
body
,
action:
'assignee'
))
end
end
# Called when one or more labels on a Noteable are added and/or removed
#
# noteable - Noteable object
# project - Project owning noteable
# author - User performing the change
# added_labels - Array of Labels added
# removed_labels - Array of Labels removed
#
# Example Note text:
#
# "added ~1 and removed ~2 ~3 labels"
#
# "added ~4 label"
#
# "removed ~5 label"
#
# Returns the created Note object
def
change_label
(
noteable
,
project
,
author
,
added_labels
,
removed_labels
)
labels_count
=
added_labels
.
count
+
removed_labels
.
count
references
=
->
(
label
)
{
label
.
to_reference
(
format: :id
)
}
added_labels
=
added_labels
.
map
(
&
references
).
join
(
' '
)
removed_labels
=
removed_labels
.
map
(
&
references
).
join
(
' '
)
text_parts
=
[]
if
added_labels
.
present?
text_parts
<<
"added
#{
added_labels
}
"
text_parts
<<
'and'
if
removed_labels
.
present?
end
if
removed_labels
.
present?
text_parts
<<
"removed
#{
removed_labels
}
"
end
text_parts
<<
'label'
.
pluralize
(
labels_count
)
body
=
text_parts
.
join
(
' '
)
create_note
(
NoteSummary
.
new
(
noteable
,
project
,
author
,
body
,
action:
'label'
))
end
# Called when the milestone of a Noteable is changed
# Called when the milestone of a Noteable is changed
#
#
# noteable - Noteable object
# noteable - Noteable object
...
...
changelogs/unreleased/label-event.yml
0 → 100644
View file @
5c42ad75
---
title
:
Use separate model for tracking resource label changes and render label system
notes based on data from this model.
merge_request
:
author
:
type
:
added
db/migrate/20180901200537_add_resource_label_event_reference_fields.rb
0 → 100644
View file @
5c42ad75
# frozen_string_literal: true
class
AddResourceLabelEventReferenceFields
<
ActiveRecord
::
Migration
DOWNTIME
=
false
def
change
add_column
:resource_label_events
,
:cached_markdown_version
,
:integer
add_column
:resource_label_events
,
:reference
,
:text
add_column
:resource_label_events
,
:reference_html
,
:text
end
end
db/schema.rb
View file @
5c42ad75
...
@@ -11,7 +11,7 @@
...
@@ -11,7 +11,7 @@
#
#
# It's strongly recommended that you check this file into your version control system.
# It's strongly recommended that you check this file into your version control system.
ActiveRecord
::
Schema
.
define
(
version:
20180901
171833
)
do
ActiveRecord
::
Schema
.
define
(
version:
20180901
200537
)
do
# These are extensions that must be enabled in order to support this database
# These are extensions that must be enabled in order to support this database
enable_extension
"plpgsql"
enable_extension
"plpgsql"
...
@@ -2481,6 +2481,9 @@ ActiveRecord::Schema.define(version: 20180901171833) do
...
@@ -2481,6 +2481,9 @@ ActiveRecord::Schema.define(version: 20180901171833) do
t
.
integer
"label_id"
t
.
integer
"label_id"
t
.
integer
"user_id"
t
.
integer
"user_id"
t
.
datetime_with_timezone
"created_at"
,
null:
false
t
.
datetime_with_timezone
"created_at"
,
null:
false
t
.
integer
"cached_markdown_version"
t
.
text
"reference"
t
.
text
"reference_html"
end
end
add_index
"resource_label_events"
,
[
"epic_id"
],
name:
"index_resource_label_events_on_epic_id"
,
using: :btree
add_index
"resource_label_events"
,
[
"epic_id"
],
name:
"index_resource_label_events_on_epic_id"
,
using: :btree
...
...
doc/api/README.md
View file @
5c42ad75
...
@@ -47,6 +47,7 @@ following locations:
...
@@ -47,6 +47,7 @@ following locations:
-
[
Namespaces
](
namespaces.md
)
-
[
Namespaces
](
namespaces.md
)
-
[
Notes
](
notes.md
)
(
comments
)
-
[
Notes
](
notes.md
)
(
comments
)
-
[
Discussions
](
discussions.md
)
(
threaded
comments)
-
[
Discussions
](
discussions.md
)
(
threaded
comments)
-
[
Resource Label Events
](
resource_label_events.md
)
-
[
Notification settings
](
notification_settings.md
)
-
[
Notification settings
](
notification_settings.md
)
-
[
Open source license templates
](
templates/licenses.md
)
-
[
Open source license templates
](
templates/licenses.md
)
-
[
Pages Domains
](
pages_domains.md
)
-
[
Pages Domains
](
pages_domains.md
)
...
...
doc/api/resource_label_events.md
0 → 100644
View file @
5c42ad75
# Resource label events API
Resource label events keep track about who, when, and which label was added or removed to an issuable.
## Issues
### List project issue label events
Gets a list of all label events for a single issue.
```
GET /projects/:id/issues/:issue_iid/resource_label_events
```
| Attribute | Type | Required | Description |
| ------------------- | ---------------- | ---------- | ------------ |
|
`id`
| integer/string | yes | The ID or
[
URL-encoded path of the project
](
README.md#namespaced-path-encoding
)
|
|
`issue_iid`
| integer | yes | The IID of an issue |
```
json
[
{
"id"
:
142
,
"user"
:
{
"id"
:
1
,
"name"
:
"Administrator"
,
"username"
:
"root"
,
"state"
:
"active"
,
"avatar_url"
:
"https://www.gravatar.com/avatar/e64c7d89f26bd1972efa854d13d7dd61?s=80&d=identicon"
,
"web_url"
:
"http://gitlab.example.com/root"
},
"created_at"
:
"2018-08-20T13:38:20.077Z"
,
"resource_type"
:
"Issue"
,
"resource_id"
:
253
,
"label"
:
{
"id"
:
73
,
"name"
:
"a1"
,
"color"
:
"#34495E"
,
"description"
:
""
},
"action"
:
"add"
},
{
"id"
:
143
,
"user"
:
{
"id"
:
1
,
"name"
:
"Administrator"
,
"username"
:
"root"
,
"state"
:
"active"
,
"avatar_url"
:
"https://www.gravatar.com/avatar/e64c7d89f26bd1972efa854d13d7dd61?s=80&d=identicon"
,
"web_url"
:
"http://gitlab.example.com/root"
},
"created_at"
:
"2018-08-20T13:38:20.077Z"
,
"resource_type"
:
"Issue"
,
"resource_id"
:
253
,
"label"
:
{
"id"
:
74
,
"name"
:
"p1"
,
"color"
:
"#0033CC"
,
"description"
:
""
},
"action"
:
"remove"
}
]
```
```
bash
curl
--request
GET
--header
"PRIVATE-TOKEN: 9koXpg98eAheJpvBs5tK"
https://gitlab.example.com/api/v4/projects/5/issues/11/resource_label_events
```
### Get single issue label event
Returns a single label event for a specific project issue
```
GET /projects/:id/issues/:issue_iid/resource_label_events/:resource_label_event_id
```
Parameters:
| Attribute | Type | Required | Description |
| --------------- | -------------- | -------- | ----------- |
|
`id`
| integer/string | yes | The ID or
[
URL-encoded path of the project
](
README.md#namespaced-path-encoding
)
|
|
`issue_iid`
| integer | yes | The IID of an issue |
|
`resource_label_event_id`
| integer | yes | The ID of a label event |
```
bash
curl
--request
GET
--header
"PRIVATE-TOKEN: 9koXpg98eAheJpvBs5tK"
https://gitlab.example.com/api/v4/projects/5/issues/11/resource_label_events/1
```
## Epics
### List group epic label events
Gets a list of all label events for a single epic.
```
GET /groups/:id/epics/:epic_id/resource_label_events
```
| Attribute | Type | Required | Description |
| ------------------- | ---------------- | ---------- | ------------ |
|
`id`
| integer/string | yes | The ID or
[
URL-encoded path of the group
](
README.md#namespaced-path-encoding
)
|
|
`epic_id`
| integer | yes | The ID of an epic |
```
json
[
{
"id"
:
106
,
"user"
:
{
"id"
:
1
,
"name"
:
"Administrator"
,
"username"
:
"root"
,
"state"
:
"active"
,
"avatar_url"
:
"https://www.gravatar.com/avatar/e64c7d89f26bd1972efa854d13d7dd61?s=80&d=identicon"
,
"web_url"
:
"http://gitlab.example.com/root"
},
"created_at"
:
"2018-08-19T11:43:01.746Z"
,
"resource_type"
:
"Epic"
,
"resource_id"
:
33
,
"label"
:
{
"id"
:
73
,
"name"
:
"a1"
,
"color"
:
"#34495E"
,
"description"
:
""
},
"action"
:
"add"
},
{
"id"
:
107
,
"user"
:
{
"id"
:
1
,
"name"
:
"Administrator"
,
"username"
:
"root"
,
"state"
:
"active"
,
"avatar_url"
:
"https://www.gravatar.com/avatar/e64c7d89f26bd1972efa854d13d7dd61?s=80&d=identicon"
,
"web_url"
:
"http://gitlab.example.com/root"
},
"created_at"
:
"2018-08-19T11:43:01.746Z"
,
"resource_type"
:
"Epic"
,
"resource_id"
:
33
,
"label"
:
{
"id"
:
37
,
"name"
:
"glabel2"
,
"color"
:
"#A8D695"
,
"description"
:
""
},
"action"
:
"add"
}
]
```
```
bash
curl
--request
GET
--header
"PRIVATE-TOKEN: 9koXpg98eAheJpvBs5tK"
https://gitlab.example.com/api/v4/groups/1/epics/11/resource_label_events
```
### Get single epic label event
Returns a single label event for a specific group epic
```
GET /groups/:id/epics/:epic_id/resource_label_events/:resource_label_event_id
```
Parameters:
| Attribute | Type | Required | Description |
| --------------- | -------------- | -------- | ----------- |
|
`id`
| integer/string | yes | The ID or
[
URL-encoded path of the group
](
README.md#namespaced-path-encoding
)
|
|
`epic_id`
| integer | yes | The ID of an epic |
|
`resource_label_event_id`
| integer | yes | The ID of a label event |
```
bash
curl
--request
POST
--header
"PRIVATE-TOKEN: 9koXpg98eAheJpvBs5tK"
https://gitlab.example.com/api/v4/groups/1/epics/11/resource_label_events/107
```
## Merge requests
### List project merge request label events
Gets a list of all label events for a single merge request.
```
GET /projects/:id/merge_requests/:merge_request_iid/resource_label_events
```
| Attribute | Type | Required | Description |
| ------------------- | ---------------- | ---------- | ------------ |
|
`id`
| integer/string | yes | The ID or
[
URL-encoded path of the project
](
README.md#namespaced-path-encoding
)
|
|
`merge_request_iid`
| integer | yes | The IID of a merge request |
```
json
[
{
"id"
:
119
,
"user"
:
{
"id"
:
1
,
"name"
:
"Administrator"
,
"username"
:
"root"
,
"state"
:
"active"
,
"avatar_url"
:
"https://www.gravatar.com/avatar/e64c7d89f26bd1972efa854d13d7dd61?s=80&d=identicon"
,
"web_url"
:
"http://gitlab.example.com/root"
},
"created_at"
:
"2018-08-20T06:17:28.394Z"
,
"resource_type"
:
"MergeRequest"
,
"resource_id"
:
28
,
"label"
:
{
"id"
:
74
,
"name"
:
"p1"
,
"color"
:
"#0033CC"
,
"description"
:
""
},
"action"
:
"add"
},
{
"id"
:
120
,
"user"
:
{
"id"
:
1
,
"name"
:
"Administrator"
,
"username"
:
"root"
,
"state"
:
"active"
,
"avatar_url"
:
"https://www.gravatar.com/avatar/e64c7d89f26bd1972efa854d13d7dd61?s=80&d=identicon"
,
"web_url"
:
"http://gitlab.example.com/root"
},
"created_at"
:
"2018-08-20T06:17:28.394Z"
,
"resource_type"
:
"MergeRequest"
,
"resource_id"
:
28
,
"label"
:
{
"id"
:
41
,
"name"
:
"project"
,
"color"
:
"#D1D100"
,
"description"
:
""
},
"action"
:
"add"
}
]
```
```
bash
curl
--request
GET
--header
"PRIVATE-TOKEN: 9koXpg98eAheJpvBs5tK"
https://gitlab.example.com/api/v4/projects/5/merge_requests/11/resource_label_events
```
### Get single merge request label event
Returns a single label event for a specific project merge request
```
GET /projects/:id/merge_requests/:merge_request_iid/resource_label_events/:resource_label_event_id
```
Parameters:
| Attribute | Type | Required | Description |
| ------------------- | -------------- | -------- | ----------- |
|
`id`
| integer/string | yes | The ID or
[
URL-encoded path of the project
](
README.md#namespaced-path-encoding
)
|
|
`merge_request_iid`
| integer | yes | The IID of a merge request |
|
`resource_label_event_id`
| integer | yes | The ID of a label event |
```
bash
curl
--request
GET
--header
"PRIVATE-TOKEN: 9koXpg98eAheJpvBs5tK"
https://gitlab.example.com/api/v4/projects/5/merge_requests/11/resource_label_events/120
```
ee/app/models/concerns/ee/noteable.rb
0 → 100644
View file @
5c42ad75
# frozen_string_literal: true
module
EE
module
Noteable
extend
ActiveSupport
::
Concern
extend
::
Gitlab
::
Utils
::
Override
override
:note_etag_key
def
note_etag_key
if
self
.
is_a?
(
Epic
)
::
Gitlab
::
Routing
.
url_helpers
.
group_epic_notes_path
(
group
,
self
)
else
super
end
end
end
end
ee/app/models/ee/epic.rb
View file @
5c42ad75
...
@@ -6,7 +6,7 @@ module EE
...
@@ -6,7 +6,7 @@ module EE
include
AtomicInternalId
include
AtomicInternalId
include
IidRoutes
include
IidRoutes
include
::
Issuable
include
::
Issuable
include
Noteable
include
::
Noteable
include
Referable
include
Referable
include
Awardable
include
Awardable
include
LabelEventable
include
LabelEventable
...
@@ -37,6 +37,10 @@ module EE
...
@@ -37,6 +37,10 @@ module EE
scope
:order_end_date_asc
,
->
do
scope
:order_end_date_asc
,
->
do
reorder
(
::
Gitlab
::
Database
.
nulls_last_order
(
'end_date'
),
'id DESC'
)
reorder
(
::
Gitlab
::
Database
.
nulls_last_order
(
'end_date'
),
'id DESC'
)
end
end
def
etag_caching_enabled?
true
end
end
end
class_methods
do
class_methods
do
...
...
ee/app/models/ee/note.rb
View file @
5c42ad75
...
@@ -16,15 +16,6 @@ module EE
...
@@ -16,15 +16,6 @@ module EE
!
for_epic?
&&
super
!
for_epic?
&&
super
end
end
override
:etag_key
def
etag_key
if
for_epic?
return
::
Gitlab
::
Routing
.
url_helpers
.
group_epic_notes_path
(
noteable
.
group
,
noteable
)
end
super
end
override
:banzai_render_context
override
:banzai_render_context
def
banzai_render_context
(
field
)
def
banzai_render_context
(
field
)
return
super
unless
for_epic?
return
super
unless
for_epic?
...
...
ee/app/models/ee/resource_label_event.rb
View file @
5c42ad75
...
@@ -10,8 +10,8 @@ module EE
...
@@ -10,8 +10,8 @@ module EE
end
end
class_methods
do
class_methods
do
def
issuable_
column
s
def
issuable_
attr
s
%i(epic
_id
)
.
freeze
+
super
%i(epic)
.
freeze
+
super
end
end
end
end
...
...
ee/app/serializers/epic_note_entity.rb
View file @
5c42ad75
...
@@ -3,7 +3,7 @@ class EpicNoteEntity < NoteEntity
...
@@ -3,7 +3,7 @@ class EpicNoteEntity < NoteEntity
toggle_award_emoji_group_epic_note_path
(
note
.
noteable
.
group
,
note
.
noteable
,
note
)
toggle_award_emoji_group_epic_note_path
(
note
.
noteable
.
group
,
note
.
noteable
,
note
)
end
end
expose
:path
do
|
note
|
expose
:path
,
if:
->
(
note
,
_
)
{
note
.
id
}
do
|
note
|
group_epic_note_path
(
note
.
noteable
.
group
,
note
.
noteable
,
note
)
group_epic_note_path
(
note
.
noteable
.
group
,
note
.
noteable
,
note
)
end
end
end
end
ee/spec/controllers/projects/issues_controller_spec.rb
View file @
5c42ad75
...
@@ -270,7 +270,7 @@ describe Projects::IssuesController do
...
@@ -270,7 +270,7 @@ describe Projects::IssuesController do
notes
=
discussions
.
flat_map
{
|
d
|
d
[
'notes'
]}
notes
=
discussions
.
flat_map
{
|
d
|
d
[
'notes'
]}
expect
(
discussions
.
count
).
to
equal
(
2
)
expect
(
discussions
.
count
).
to
equal
(
2
)
expect
(
notes
).
to
include
(
a_hash_including
(
'id'
=>
system_note
.
id
))
expect
(
notes
).
to
include
(
a_hash_including
(
'id'
=>
system_note
.
id
.
to_s
))
end
end
end
end
end
end
...
@@ -288,7 +288,7 @@ describe Projects::IssuesController do
...
@@ -288,7 +288,7 @@ describe Projects::IssuesController do
notes
=
discussions
.
flat_map
{
|
d
|
d
[
'notes'
]}
notes
=
discussions
.
flat_map
{
|
d
|
d
[
'notes'
]}
expect
(
discussions
.
count
).
to
equal
(
1
)
expect
(
discussions
.
count
).
to
equal
(
1
)
expect
(
notes
).
not_to
include
(
a_hash_including
(
'id'
=>
system_note
.
id
))
expect
(
notes
).
not_to
include
(
a_hash_including
(
'id'
=>
system_note
.
id
.
to_s
))
end
end
end
end
end
end
...
...
ee/spec/models/label_note_spec.rb
0 → 100644
View file @
5c42ad75
# frozen_string_literal: true
require
'spec_helper'
describe
LabelNote
do
set
(
:group
)
{
create
(
:group
)
}
set
(
:user
)
{
create
(
:user
)
}
set
(
:label
)
{
create
(
:group_label
,
group:
group
)
}
set
(
:label2
)
{
create
(
:group_label
,
group:
group
)
}
let
(
:resource_parent
)
{
group
}
context
'when resource is epic'
do
set
(
:resource
)
{
create
(
:epic
,
group:
group
)
}
let
(
:project
)
{
nil
}
it_behaves_like
'label note created from events'
end
end
lib/api/api.rb
View file @
5c42ad75
...
@@ -127,6 +127,7 @@ module API
...
@@ -127,6 +127,7 @@ module API
mount
::
API
::
Namespaces
mount
::
API
::
Namespaces
mount
::
API
::
Notes
mount
::
API
::
Notes
mount
::
API
::
Discussions
mount
::
API
::
Discussions
mount
::
API
::
ResourceLabelEvents
mount
::
API
::
NotificationSettings
mount
::
API
::
NotificationSettings
mount
::
API
::
PagesDomains
mount
::
API
::
PagesDomains
mount
::
API
::
Pipelines
mount
::
API
::
Pipelines
...
...
lib/api/entities.rb
View file @
5c42ad75
...
@@ -1486,6 +1486,20 @@ module API
...
@@ -1486,6 +1486,20 @@ module API
class
ManagedLicense
<
Grape
::
Entity
class
ManagedLicense
<
Grape
::
Entity
expose
:id
,
:name
,
:approval_status
expose
:id
,
:name
,
:approval_status
end
end
class
ResourceLabelEvent
<
Grape
::
Entity
expose
:id
expose
:user
,
using:
Entities
::
UserBasic
expose
:created_at
expose
:resource_type
do
|
event
,
options
|
event
.
issuable
.
class
.
name
end
expose
:resource_id
do
|
event
,
options
|
event
.
issuable
.
id
end
expose
:label
,
using:
Entities
::
LabelBasic
expose
:action
end
end
end
end
end
...
...
lib/api/resource_label_events.rb
0 → 100644
View file @
5c42ad75
# frozen_string_literal: true
module
API
class
ResourceLabelEvents
<
Grape
::
API
include
PaginationParams
helpers
::
API
::
Helpers
::
NotesHelpers
before
{
authenticate!
}
EVENTABLE_TYPES
=
[
Issue
,
Epic
,
MergeRequest
].
freeze
EVENTABLE_TYPES
.
each
do
|
eventable_type
|
parent_type
=
eventable_type
.
parent_class
.
to_s
.
underscore
eventables_str
=
eventable_type
.
to_s
.
underscore
.
pluralize
params
do
requires
:id
,
type:
String
,
desc:
"The ID of a
#{
parent_type
}
"
end
resource
parent_type
.
pluralize
.
to_sym
,
requirements:
API
::
PROJECT_ENDPOINT_REQUIREMENTS
do
desc
"Get a list of
#{
eventable_type
.
to_s
.
downcase
}
resource label events"
do
success
Entities
::
ResourceLabelEvent
detail
'This feature was introduced in 11.3'
end
params
do
requires
:eventable_id
,
types:
[
Integer
,
String
],
desc:
'The ID of the eventable'
use
:pagination
end
get
":id/
#{
eventables_str
}
/:eventable_id/resource_label_events"
do
eventable
=
find_noteable
(
parent_type
,
eventables_str
,
params
[
:eventable_id
])
events
=
eventable
.
resource_label_events
.
includes
(
:label
,
:user
)
present
paginate
(
events
),
with:
Entities
::
ResourceLabelEvent
end
desc
"Get a single
#{
eventable_type
.
to_s
.
downcase
}
resource label event"
do
success
Entities
::
ResourceLabelEvent
detail
'This feature was introduced in 11.3'
end
params
do
requires
:event_id
,
type:
String
,
desc:
'The ID of a resource label event'
requires
:eventable_id
,
types:
[
Integer
,
String
],
desc:
'The ID of the eventable'
end
get
":id/
#{
eventables_str
}
/:eventable_id/resource_label_events/:event_id"
do
eventable
=
find_noteable
(
parent_type
,
eventables_str
,
params
[
:eventable_id
])
event
=
eventable
.
resource_label_events
.
find
(
params
[
:event_id
])
present
event
,
with:
Entities
::
ResourceLabelEvent
end
end
end
end
end
lib/banzai/pipeline/label_pipeline.rb
0 → 100644
View file @
5c42ad75
# frozen_string_literal: true
module
Banzai
module
Pipeline
class
LabelPipeline
<
BasePipeline
def
self
.
filters
@filters
||=
FilterArray
[
Filter
::
SanitizationFilter
,
Filter
::
LabelReferenceFilter
]
end
end
end
end
lib/gitlab/import_export/import_export.yml
View file @
5c42ad75
...
@@ -19,6 +19,9 @@ project_tree:
...
@@ -19,6 +19,9 @@ project_tree:
-
milestone
:
-
milestone
:
-
events
:
-
events
:
-
:push_event_payload
-
:push_event_payload
-
resource_label_events
:
-
label
:
:priorities
-
:issue_assignees
-
:issue_assignees
-
snippets
:
-
snippets
:
-
:award_emoji
-
:award_emoji
...
@@ -45,6 +48,9 @@ project_tree:
...
@@ -45,6 +48,9 @@ project_tree:
-
milestone
:
-
milestone
:
-
events
:
-
events
:
-
:push_event_payload
-
:push_event_payload
-
resource_label_events
:
-
label
:
:priorities
-
pipelines
:
-
pipelines
:
-
notes
:
-
notes
:
-
:author
-
:author
...
@@ -148,6 +154,10 @@ excluded_attributes:
...
@@ -148,6 +154,10 @@ excluded_attributes:
-
:event_id
-
:event_id
project_badges
:
project_badges
:
-
:group_id
-
:group_id
resource_label_events
:
-
:reference
-
:reference_html
-
:epic_id
methods
:
methods
:
labels
:
labels
:
...
...
locale/gitlab.pot
View file @
5c42ad75
...
@@ -324,6 +324,9 @@ msgstr ""
...
@@ -324,6 +324,9 @@ msgstr ""
msgid "A default branch cannot be chosen for an empty project."
msgid "A default branch cannot be chosen for an empty project."
msgstr ""
msgstr ""
msgid "A deleted user"
msgstr ""
msgid "A new branch will be created in your fork and a new merge request will be started."
msgid "A new branch will be created in your fork and a new merge request will be started."
msgstr ""
msgstr ""
...
...
spec/controllers/projects/notes_controller_spec.rb
View file @
5c42ad75
...
@@ -154,7 +154,7 @@ describe Projects::NotesController do
...
@@ -154,7 +154,7 @@ describe Projects::NotesController do
get
:index
,
request_params
get
:index
,
request_params
expect
(
parsed_response
[
:notes
].
count
).
to
eq
(
1
)
expect
(
parsed_response
[
:notes
].
count
).
to
eq
(
1
)
expect
(
note_json
[
:id
]).
to
eq
(
note
.
id
)
expect
(
note_json
[
:id
]).
to
eq
(
note
.
id
.
to_s
)
end
end
it
'does not result in N+1 queries'
do
it
'does not result in N+1 queries'
do
...
...
spec/factories/resource_label_events.rb
View file @
5c42ad75
...
@@ -2,9 +2,12 @@
...
@@ -2,9 +2,12 @@
FactoryBot
.
define
do
FactoryBot
.
define
do
factory
:resource_label_event
do
factory
:resource_label_event
do
user
{
issue
.
project
.
creator
}
action
:add
action
:add
label
label
issue
user
{
issuable
&
.
author
||
create
(
:user
)
}
after
(
:build
)
do
|
event
,
evaluator
|
event
.
issue
=
create
(
:issue
)
unless
event
.
issuable
end
end
end
end
end
spec/features/issues/resource_label_events_spec.rb
0 → 100644
View file @
5c42ad75
# frozen_string_literal: true
require
'rails_helper'
describe
'List issue resource label events'
,
:js
do
let
(
:user
)
{
create
(
:user
)
}
let
(
:project
)
{
create
(
:project
,
:public
)
}
let
(
:issue
)
{
create
(
:issue
,
project:
project
,
author:
user
)
}
let!
(
:label
)
{
create
(
:label
,
project:
project
,
title:
'foo'
)
}
context
'when user displays the issue'
do
let!
(
:note
)
{
create
(
:note_on_issue
,
author:
user
,
project:
project
,
noteable:
issue
,
note:
'some note'
)
}
let!
(
:event
)
{
create
(
:resource_label_event
,
user:
user
,
issue:
issue
,
label:
label
)
}
before
do
visit
project_issue_path
(
project
,
issue
)
wait_for_requests
end
it
'shows both notes and resource label events'
do
page
.
within
(
'#notes'
)
do
expect
(
find
(
"#note_
#{
note
.
id
}
"
)).
to
have_content
'some note'
expect
(
find
(
"#note_
#{
event
.
discussion_id
}
"
)).
to
have_content
'added foo label'
end
end
end
context
'when user adds label to the issue'
do
def
toggle_labels
(
labels
)
page
.
within
'.labels'
do
click_link
'Edit'
wait_for_requests
labels
.
each
{
|
label
|
click_link
label
}
click_link
'Edit'
wait_for_requests
end
end
before
do
create
(
:label
,
project:
project
,
title:
'bar'
)
project
.
add_developer
(
user
)
sign_in
(
user
)
visit
project_issue_path
(
project
,
issue
)
wait_for_requests
end
it
'shows add note for newly added labels'
do
toggle_labels
(
%w(foo bar)
)
visit
project_issue_path
(
project
,
issue
)
wait_for_requests
page
.
within
(
'#notes'
)
do
expect
(
page
).
to
have_content
'added bar foo labels'
end
end
end
end
spec/javascripts/diffs/mock_data/diff_discussions.js
View file @
5c42ad75
...
@@ -16,7 +16,7 @@ export default {
...
@@ -16,7 +16,7 @@ export default {
expanded
:
true
,
expanded
:
true
,
notes
:
[
notes
:
[
{
{
id
:
1749
,
id
:
'
1749
'
,
type
:
'
DiffNote
'
,
type
:
'
DiffNote
'
,
attachment
:
null
,
attachment
:
null
,
author
:
{
author
:
{
...
@@ -68,7 +68,7 @@ export default {
...
@@ -68,7 +68,7 @@ export default {
'
/gitlab-org/gitlab-test/issues/new?discussion_to_resolve=6b232e05bea388c6b043ccc243ba505faac04ea8&merge_request_to_resolve_discussions_of=20
'
,
'
/gitlab-org/gitlab-test/issues/new?discussion_to_resolve=6b232e05bea388c6b043ccc243ba505faac04ea8&merge_request_to_resolve_discussions_of=20
'
,
},
},
{
{
id
:
1753
,
id
:
'
1753
'
,
type
:
'
DiffNote
'
,
type
:
'
DiffNote
'
,
attachment
:
null
,
attachment
:
null
,
author
:
{
author
:
{
...
@@ -120,7 +120,7 @@ export default {
...
@@ -120,7 +120,7 @@ export default {
'
/gitlab-org/gitlab-test/issues/new?discussion_to_resolve=6b232e05bea388c6b043ccc243ba505faac04ea8&merge_request_to_resolve_discussions_of=20
'
,
'
/gitlab-org/gitlab-test/issues/new?discussion_to_resolve=6b232e05bea388c6b043ccc243ba505faac04ea8&merge_request_to_resolve_discussions_of=20
'
,
},
},
{
{
id
:
1754
,
id
:
'
1754
'
,
type
:
'
DiffNote
'
,
type
:
'
DiffNote
'
,
attachment
:
null
,
attachment
:
null
,
author
:
{
author
:
{
...
@@ -162,7 +162,7 @@ export default {
...
@@ -162,7 +162,7 @@ export default {
'
/gitlab-org/gitlab-test/issues/new?discussion_to_resolve=6b232e05bea388c6b043ccc243ba505faac04ea8&merge_request_to_resolve_discussions_of=20
'
,
'
/gitlab-org/gitlab-test/issues/new?discussion_to_resolve=6b232e05bea388c6b043ccc243ba505faac04ea8&merge_request_to_resolve_discussions_of=20
'
,
},
},
{
{
id
:
1755
,
id
:
'
1755
'
,
type
:
'
DiffNote
'
,
type
:
'
DiffNote
'
,
attachment
:
null
,
attachment
:
null
,
author
:
{
author
:
{
...
@@ -204,7 +204,7 @@ export default {
...
@@ -204,7 +204,7 @@ export default {
'
/gitlab-org/gitlab-test/issues/new?discussion_to_resolve=6b232e05bea388c6b043ccc243ba505faac04ea8&merge_request_to_resolve_discussions_of=20
'
,
'
/gitlab-org/gitlab-test/issues/new?discussion_to_resolve=6b232e05bea388c6b043ccc243ba505faac04ea8&merge_request_to_resolve_discussions_of=20
'
,
},
},
{
{
id
:
1756
,
id
:
'
1756
'
,
type
:
'
DiffNote
'
,
type
:
'
DiffNote
'
,
attachment
:
null
,
attachment
:
null
,
author
:
{
author
:
{
...
...
spec/javascripts/notes/components/note_actions_spec.js
View file @
5c42ad75
...
@@ -28,7 +28,7 @@ describe('issue_note_actions component', () => {
...
@@ -28,7 +28,7 @@ describe('issue_note_actions component', () => {
canEdit
:
true
,
canEdit
:
true
,
canAwardEmoji
:
true
,
canAwardEmoji
:
true
,
canReportAsAbuse
:
true
,
canReportAsAbuse
:
true
,
noteId
:
539
,
noteId
:
'
539
'
,
noteUrl
:
'
https://localhost:3000/group/project/merge_requests/1#note_1
'
,
noteUrl
:
'
https://localhost:3000/group/project/merge_requests/1#note_1
'
,
reportAbusePath
:
reportAbusePath
:
'
/abuse_reports/new?ref_url=http%3A%2F%2Flocalhost%3A3000%2Fgitlab-org%2Fgitlab-ce%2Fissues%2F7%23note_539&user_id=26
'
,
'
/abuse_reports/new?ref_url=http%3A%2F%2Flocalhost%3A3000%2Fgitlab-org%2Fgitlab-ce%2Fissues%2F7%23note_539&user_id=26
'
,
...
@@ -59,6 +59,20 @@ describe('issue_note_actions component', () => {
...
@@ -59,6 +59,20 @@ describe('issue_note_actions component', () => {
expect
(
vm
.
$el
.
querySelector
(
`a[href="
${
props
.
reportAbusePath
}
"]`
)).
toBeDefined
();
expect
(
vm
.
$el
.
querySelector
(
`a[href="
${
props
.
reportAbusePath
}
"]`
)).
toBeDefined
();
});
});
it
(
'
should be possible to copy link to a note
'
,
()
=>
{
expect
(
vm
.
$el
.
querySelector
(
'
.js-btn-copy-note-link
'
)).
not
.
toBeNull
();
});
it
(
'
should not show copy link action when `noteUrl` prop is empty
'
,
done
=>
{
vm
.
noteUrl
=
''
;
Vue
.
nextTick
()
.
then
(()
=>
{
expect
(
vm
.
$el
.
querySelector
(
'
.js-btn-copy-note-link
'
)).
toBeNull
();
})
.
then
(
done
)
.
catch
(
done
.
fail
);
});
it
(
'
should be possible to delete comment
'
,
()
=>
{
it
(
'
should be possible to delete comment
'
,
()
=>
{
expect
(
vm
.
$el
.
querySelector
(
'
.js-note-delete
'
)).
toBeDefined
();
expect
(
vm
.
$el
.
querySelector
(
'
.js-note-delete
'
)).
toBeDefined
();
});
});
...
@@ -77,7 +91,7 @@ describe('issue_note_actions component', () => {
...
@@ -77,7 +91,7 @@ describe('issue_note_actions component', () => {
canEdit
:
false
,
canEdit
:
false
,
canAwardEmoji
:
false
,
canAwardEmoji
:
false
,
canReportAsAbuse
:
false
,
canReportAsAbuse
:
false
,
noteId
:
539
,
noteId
:
'
539
'
,
noteUrl
:
'
https://localhost:3000/group/project/merge_requests/1#note_1
'
,
noteUrl
:
'
https://localhost:3000/group/project/merge_requests/1#note_1
'
,
reportAbusePath
:
reportAbusePath
:
'
/abuse_reports/new?ref_url=http%3A%2F%2Flocalhost%3A3000%2Fgitlab-org%2Fgitlab-ce%2Fissues%2F7%23note_539&user_id=26
'
,
'
/abuse_reports/new?ref_url=http%3A%2F%2Flocalhost%3A3000%2Fgitlab-org%2Fgitlab-ce%2Fissues%2F7%23note_539&user_id=26
'
,
...
...
spec/javascripts/notes/components/note_awards_list_spec.js
View file @
5c42ad75
...
@@ -30,7 +30,7 @@ describe('note_awards_list component', () => {
...
@@ -30,7 +30,7 @@ describe('note_awards_list component', () => {
propsData
:
{
propsData
:
{
awards
:
awardsMock
,
awards
:
awardsMock
,
noteAuthorId
:
2
,
noteAuthorId
:
2
,
noteId
:
545
,
noteId
:
'
545
'
,
canAwardEmoji
:
true
,
canAwardEmoji
:
true
,
toggleAwardPath
:
'
/gitlab-org/gitlab-ce/notes/545/toggle_award_emoji
'
,
toggleAwardPath
:
'
/gitlab-org/gitlab-ce/notes/545/toggle_award_emoji
'
,
},
},
...
@@ -70,7 +70,7 @@ describe('note_awards_list component', () => {
...
@@ -70,7 +70,7 @@ describe('note_awards_list component', () => {
propsData
:
{
propsData
:
{
awards
:
awardsMock
,
awards
:
awardsMock
,
noteAuthorId
:
2
,
noteAuthorId
:
2
,
noteId
:
545
,
noteId
:
'
545
'
,
canAwardEmoji
:
false
,
canAwardEmoji
:
false
,
toggleAwardPath
:
'
/gitlab-org/gitlab-ce/notes/545/toggle_award_emoji
'
,
toggleAwardPath
:
'
/gitlab-org/gitlab-ce/notes/545/toggle_award_emoji
'
,
},
},
...
...
spec/javascripts/notes/components/note_form_spec.js
View file @
5c42ad75
...
@@ -19,7 +19,7 @@ describe('issue_note_form component', () => {
...
@@ -19,7 +19,7 @@ describe('issue_note_form component', () => {
props
=
{
props
=
{
isEditing
:
false
,
isEditing
:
false
,
noteBody
:
'
Magni suscipit eius consectetur enim et ex et commodi.
'
,
noteBody
:
'
Magni suscipit eius consectetur enim et ex et commodi.
'
,
noteId
:
545
,
noteId
:
'
545
'
,
};
};
vm
=
new
Component
({
vm
=
new
Component
({
...
@@ -32,6 +32,22 @@ describe('issue_note_form component', () => {
...
@@ -32,6 +32,22 @@ describe('issue_note_form component', () => {
vm
.
$destroy
();
vm
.
$destroy
();
});
});
describe
(
'
noteHash
'
,
()
=>
{
it
(
'
returns note hash string based on `noteId`
'
,
()
=>
{
expect
(
vm
.
noteHash
).
toBe
(
`#note_
${
props
.
noteId
}
`
);
});
it
(
'
return note hash as `#` when `noteId` is empty
'
,
done
=>
{
vm
.
noteId
=
''
;
Vue
.
nextTick
()
.
then
(()
=>
{
expect
(
vm
.
noteHash
).
toBe
(
'
#
'
);
})
.
then
(
done
)
.
catch
(
done
.
fail
);
});
});
describe
(
'
conflicts editing
'
,
()
=>
{
describe
(
'
conflicts editing
'
,
()
=>
{
it
(
'
should show conflict message if note changes outside the component
'
,
done
=>
{
it
(
'
should show conflict message if note changes outside the component
'
,
done
=>
{
vm
.
isEditing
=
true
;
vm
.
isEditing
=
true
;
...
...
spec/javascripts/notes/components/note_header_spec.js
View file @
5c42ad75
...
@@ -33,7 +33,7 @@ describe('note_header component', () => {
...
@@ -33,7 +33,7 @@ describe('note_header component', () => {
},
},
createdAt
:
'
2017-08-02T10:51:58.559Z
'
,
createdAt
:
'
2017-08-02T10:51:58.559Z
'
,
includeToggle
:
false
,
includeToggle
:
false
,
noteId
:
1394
,
noteId
:
'
1394
'
,
expanded
:
true
,
expanded
:
true
,
},
},
}).
$mount
();
}).
$mount
();
...
@@ -47,6 +47,16 @@ describe('note_header component', () => {
...
@@ -47,6 +47,16 @@ describe('note_header component', () => {
it
(
'
should render timestamp link
'
,
()
=>
{
it
(
'
should render timestamp link
'
,
()
=>
{
expect
(
vm
.
$el
.
querySelector
(
'
a[href="#note_1394"]
'
)).
toBeDefined
();
expect
(
vm
.
$el
.
querySelector
(
'
a[href="#note_1394"]
'
)).
toBeDefined
();
});
});
it
(
'
should not render user information when prop `author` is empty object
'
,
done
=>
{
vm
.
author
=
{};
Vue
.
nextTick
()
.
then
(()
=>
{
expect
(
vm
.
$el
.
querySelector
(
'
.note-header-author-name
'
)).
toBeNull
();
})
.
then
(
done
)
.
catch
(
done
.
fail
);
});
});
});
describe
(
'
discussion
'
,
()
=>
{
describe
(
'
discussion
'
,
()
=>
{
...
@@ -66,7 +76,7 @@ describe('note_header component', () => {
...
@@ -66,7 +76,7 @@ describe('note_header component', () => {
},
},
createdAt
:
'
2017-08-02T10:51:58.559Z
'
,
createdAt
:
'
2017-08-02T10:51:58.559Z
'
,
includeToggle
:
true
,
includeToggle
:
true
,
noteId
:
1395
,
noteId
:
'
1395
'
,
expanded
:
true
,
expanded
:
true
,
},
},
}).
$mount
();
}).
$mount
();
...
...
spec/javascripts/notes/mock_data.js
View file @
5c42ad75
...
@@ -66,7 +66,7 @@ export const individualNote = {
...
@@ -66,7 +66,7 @@ export const individualNote = {
individual_note
:
true
,
individual_note
:
true
,
notes
:
[
notes
:
[
{
{
id
:
1390
,
id
:
'
1390
'
,
attachment
:
{
attachment
:
{
url
:
null
,
url
:
null
,
filename
:
null
,
filename
:
null
,
...
@@ -111,7 +111,7 @@ export const individualNote = {
...
@@ -111,7 +111,7 @@ export const individualNote = {
};
};
export
const
note
=
{
export
const
note
=
{
id
:
546
,
id
:
'
546
'
,
attachment
:
{
attachment
:
{
url
:
null
,
url
:
null
,
filename
:
null
,
filename
:
null
,
...
@@ -174,7 +174,7 @@ export const discussionMock = {
...
@@ -174,7 +174,7 @@ export const discussionMock = {
expanded
:
true
,
expanded
:
true
,
notes
:
[
notes
:
[
{
{
id
:
1395
,
id
:
'
1395
'
,
attachment
:
{
attachment
:
{
url
:
null
,
url
:
null
,
filename
:
null
,
filename
:
null
,
...
@@ -211,7 +211,7 @@ export const discussionMock = {
...
@@ -211,7 +211,7 @@ export const discussionMock = {
path
:
'
/gitlab-org/gitlab-ce/notes/1395
'
,
path
:
'
/gitlab-org/gitlab-ce/notes/1395
'
,
},
},
{
{
id
:
1396
,
id
:
'
1396
'
,
attachment
:
{
attachment
:
{
url
:
null
,
url
:
null
,
filename
:
null
,
filename
:
null
,
...
@@ -257,7 +257,7 @@ export const discussionMock = {
...
@@ -257,7 +257,7 @@ export const discussionMock = {
path
:
'
/gitlab-org/gitlab-ce/notes/1396
'
,
path
:
'
/gitlab-org/gitlab-ce/notes/1396
'
,
},
},
{
{
id
:
1437
,
id
:
'
1437
'
,
attachment
:
{
attachment
:
{
url
:
null
,
url
:
null
,
filename
:
null
,
filename
:
null
,
...
@@ -308,7 +308,7 @@ export const discussionMock = {
...
@@ -308,7 +308,7 @@ export const discussionMock = {
};
};
export
const
loggedOutnoteableData
=
{
export
const
loggedOutnoteableData
=
{
id
:
98
,
id
:
'
98
'
,
iid
:
26
,
iid
:
26
,
author_id
:
1
,
author_id
:
1
,
description
:
''
,
description
:
''
,
...
@@ -358,7 +358,7 @@ export const collapseNotesMock = [
...
@@ -358,7 +358,7 @@ export const collapseNotesMock = [
individual_note
:
true
,
individual_note
:
true
,
notes
:
[
notes
:
[
{
{
id
:
1390
,
id
:
'
1390
'
,
attachment
:
null
,
attachment
:
null
,
author
:
{
author
:
{
id
:
1
,
id
:
1
,
...
@@ -393,7 +393,7 @@ export const collapseNotesMock = [
...
@@ -393,7 +393,7 @@ export const collapseNotesMock = [
individual_note
:
true
,
individual_note
:
true
,
notes
:
[
notes
:
[
{
{
id
:
1391
,
id
:
'
1391
'
,
attachment
:
null
,
attachment
:
null
,
author
:
{
author
:
{
id
:
1
,
id
:
1
,
...
@@ -433,7 +433,7 @@ export const INDIVIDUAL_NOTE_RESPONSE_MAP = {
...
@@ -433,7 +433,7 @@ export const INDIVIDUAL_NOTE_RESPONSE_MAP = {
expanded
:
true
,
expanded
:
true
,
notes
:
[
notes
:
[
{
{
id
:
1390
,
id
:
'
1390
'
,
attachment
:
{
attachment
:
{
url
:
null
,
url
:
null
,
filename
:
null
,
filename
:
null
,
...
@@ -495,7 +495,7 @@ export const INDIVIDUAL_NOTE_RESPONSE_MAP = {
...
@@ -495,7 +495,7 @@ export const INDIVIDUAL_NOTE_RESPONSE_MAP = {
expanded
:
true
,
expanded
:
true
,
notes
:
[
notes
:
[
{
{
id
:
1391
,
id
:
'
1391
'
,
attachment
:
{
attachment
:
{
url
:
null
,
url
:
null
,
filename
:
null
,
filename
:
null
,
...
@@ -544,7 +544,7 @@ export const INDIVIDUAL_NOTE_RESPONSE_MAP = {
...
@@ -544,7 +544,7 @@ export const INDIVIDUAL_NOTE_RESPONSE_MAP = {
'
/gitlab-org/gitlab-ce/notes/1471
'
:
{
'
/gitlab-org/gitlab-ce/notes/1471
'
:
{
commands_changes
:
null
,
commands_changes
:
null
,
valid
:
true
,
valid
:
true
,
id
:
1471
,
id
:
'
1471
'
,
attachment
:
null
,
attachment
:
null
,
author
:
{
author
:
{
id
:
1
,
id
:
1
,
...
@@ -600,7 +600,7 @@ export const DISCUSSION_NOTE_RESPONSE_MAP = {
...
@@ -600,7 +600,7 @@ export const DISCUSSION_NOTE_RESPONSE_MAP = {
expanded
:
true
,
expanded
:
true
,
notes
:
[
notes
:
[
{
{
id
:
1471
,
id
:
'
1471
'
,
attachment
:
{
attachment
:
{
url
:
null
,
url
:
null
,
filename
:
null
,
filename
:
null
,
...
@@ -671,7 +671,7 @@ export const notesWithDescriptionChanges = [
...
@@ -671,7 +671,7 @@ export const notesWithDescriptionChanges = [
expanded
:
true
,
expanded
:
true
,
notes
:
[
notes
:
[
{
{
id
:
901
,
id
:
'
901
'
,
type
:
null
,
type
:
null
,
attachment
:
null
,
attachment
:
null
,
author
:
{
author
:
{
...
@@ -718,7 +718,7 @@ export const notesWithDescriptionChanges = [
...
@@ -718,7 +718,7 @@ export const notesWithDescriptionChanges = [
expanded
:
true
,
expanded
:
true
,
notes
:
[
notes
:
[
{
{
id
:
902
,
id
:
'
902
'
,
type
:
null
,
type
:
null
,
attachment
:
null
,
attachment
:
null
,
author
:
{
author
:
{
...
@@ -765,7 +765,7 @@ export const notesWithDescriptionChanges = [
...
@@ -765,7 +765,7 @@ export const notesWithDescriptionChanges = [
expanded
:
true
,
expanded
:
true
,
notes
:
[
notes
:
[
{
{
id
:
903
,
id
:
'
903
'
,
type
:
null
,
type
:
null
,
attachment
:
null
,
attachment
:
null
,
author
:
{
author
:
{
...
@@ -809,7 +809,7 @@ export const notesWithDescriptionChanges = [
...
@@ -809,7 +809,7 @@ export const notesWithDescriptionChanges = [
expanded
:
true
,
expanded
:
true
,
notes
:
[
notes
:
[
{
{
id
:
904
,
id
:
'
904
'
,
type
:
null
,
type
:
null
,
attachment
:
null
,
attachment
:
null
,
author
:
{
author
:
{
...
@@ -854,7 +854,7 @@ export const notesWithDescriptionChanges = [
...
@@ -854,7 +854,7 @@ export const notesWithDescriptionChanges = [
expanded
:
true
,
expanded
:
true
,
notes
:
[
notes
:
[
{
{
id
:
905
,
id
:
'
905
'
,
type
:
null
,
type
:
null
,
attachment
:
null
,
attachment
:
null
,
author
:
{
author
:
{
...
@@ -898,7 +898,7 @@ export const notesWithDescriptionChanges = [
...
@@ -898,7 +898,7 @@ export const notesWithDescriptionChanges = [
expanded
:
true
,
expanded
:
true
,
notes
:
[
notes
:
[
{
{
id
:
906
,
id
:
'
906
'
,
type
:
null
,
type
:
null
,
attachment
:
null
,
attachment
:
null
,
author
:
{
author
:
{
...
@@ -945,7 +945,7 @@ export const collapsedSystemNotes = [
...
@@ -945,7 +945,7 @@ export const collapsedSystemNotes = [
expanded
:
true
,
expanded
:
true
,
notes
:
[
notes
:
[
{
{
id
:
901
,
id
:
'
901
'
,
type
:
null
,
type
:
null
,
attachment
:
null
,
attachment
:
null
,
author
:
{
author
:
{
...
@@ -992,7 +992,7 @@ export const collapsedSystemNotes = [
...
@@ -992,7 +992,7 @@ export const collapsedSystemNotes = [
expanded
:
true
,
expanded
:
true
,
notes
:
[
notes
:
[
{
{
id
:
902
,
id
:
'
902
'
,
type
:
null
,
type
:
null
,
attachment
:
null
,
attachment
:
null
,
author
:
{
author
:
{
...
@@ -1039,7 +1039,7 @@ export const collapsedSystemNotes = [
...
@@ -1039,7 +1039,7 @@ export const collapsedSystemNotes = [
expanded
:
true
,
expanded
:
true
,
notes
:
[
notes
:
[
{
{
id
:
904
,
id
:
'
904
'
,
type
:
null
,
type
:
null
,
attachment
:
null
,
attachment
:
null
,
author
:
{
author
:
{
...
@@ -1084,7 +1084,7 @@ export const collapsedSystemNotes = [
...
@@ -1084,7 +1084,7 @@ export const collapsedSystemNotes = [
expanded
:
true
,
expanded
:
true
,
notes
:
[
notes
:
[
{
{
id
:
905
,
id
:
'
905
'
,
type
:
null
,
type
:
null
,
attachment
:
null
,
attachment
:
null
,
author
:
{
author
:
{
...
@@ -1129,7 +1129,7 @@ export const collapsedSystemNotes = [
...
@@ -1129,7 +1129,7 @@ export const collapsedSystemNotes = [
expanded
:
true
,
expanded
:
true
,
notes
:
[
notes
:
[
{
{
id
:
906
,
id
:
'
906
'
,
type
:
null
,
type
:
null
,
attachment
:
null
,
attachment
:
null
,
author
:
{
author
:
{
...
...
spec/javascripts/vue_shared/components/notes/system_note_spec.js
View file @
5c42ad75
...
@@ -9,7 +9,7 @@ describe('system note component', () => {
...
@@ -9,7 +9,7 @@ describe('system note component', () => {
beforeEach
(()
=>
{
beforeEach
(()
=>
{
props
=
{
props
=
{
note
:
{
note
:
{
id
:
1424
,
id
:
'
1424
'
,
author
:
{
author
:
{
id
:
1
,
id
:
1
,
name
:
'
Root
'
,
name
:
'
Root
'
,
...
...
spec/lib/gitlab/import_export/all_models.yml
View file @
5c42ad75
...
@@ -379,3 +379,9 @@ metrics:
...
@@ -379,3 +379,9 @@ metrics:
-
latest_closed_by
-
latest_closed_by
-
merged_by
-
merged_by
-
pipeline
-
pipeline
resource_label_events
:
-
user
-
issue
-
merge_request
-
epic
-
label
spec/lib/gitlab/import_export/project.json
View file @
5c42ad75
...
@@ -331,6 +331,28 @@
...
@@ -331,6 +331,28 @@
},
},
"events"
:
[]
"events"
:
[]
}
}
],
"resource_label_events"
:
[
{
"id"
:
244
,
"action"
:
"remove"
,
"issue_id"
:
40
,
"merge_request_id"
:
null
,
"label_id"
:
2
,
"user_id"
:
1
,
"created_at"
:
"2018-08-28T08:24:00.494Z"
,
"label"
:
{
"id"
:
2
,
"title"
:
"test2"
,
"color"
:
"#428bca"
,
"project_id"
:
8
,
"created_at"
:
"2016-07-22T08:55:44.161Z"
,
"updated_at"
:
"2016-07-22T08:55:44.161Z"
,
"template"
:
false
,
"description"
:
""
,
"type"
:
"ProjectLabel"
}
}
]
]
},
},
{
{
...
@@ -2515,6 +2537,17 @@
...
@@ -2515,6 +2537,17 @@
"events"
:
[]
"events"
:
[]
}
}
],
],
"resource_label_events"
:
[
{
"id"
:
243
,
"action"
:
"add"
,
"issue_id"
:
null
,
"merge_request_id"
:
27
,
"label_id"
:
null
,
"user_id"
:
1
,
"created_at"
:
"2018-08-28T08:24:00.494Z"
}
],
"merge_request_diff"
:
{
"merge_request_diff"
:
{
"id"
:
27
,
"id"
:
27
,
"state"
:
"collected"
,
"state"
:
"collected"
,
...
...
spec/lib/gitlab/import_export/project_tree_restorer_spec.rb
View file @
5c42ad75
...
@@ -89,6 +89,14 @@ describe Gitlab::ImportExport::ProjectTreeRestorer do
...
@@ -89,6 +89,14 @@ describe Gitlab::ImportExport::ProjectTreeRestorer do
expect
(
ProtectedTag
.
first
.
create_access_levels
).
not_to
be_empty
expect
(
ProtectedTag
.
first
.
create_access_levels
).
not_to
be_empty
end
end
it
'restores issue resource label events'
do
expect
(
Issue
.
find_by
(
title:
'Voluptatem'
).
resource_label_events
).
not_to
be_empty
end
it
'restores merge requests resource label events'
do
expect
(
MergeRequest
.
find_by
(
title:
'MR1'
).
resource_label_events
).
not_to
be_empty
end
context
'event at forth level of the tree'
do
context
'event at forth level of the tree'
do
let
(
:event
)
{
Event
.
where
(
action:
6
).
first
}
let
(
:event
)
{
Event
.
where
(
action:
6
).
first
}
...
...
spec/lib/gitlab/import_export/project_tree_saver_spec.rb
View file @
5c42ad75
...
@@ -173,6 +173,14 @@ describe Gitlab::ImportExport::ProjectTreeSaver do
...
@@ -173,6 +173,14 @@ describe Gitlab::ImportExport::ProjectTreeSaver do
expect
(
priorities
.
flatten
).
not_to
be_empty
expect
(
priorities
.
flatten
).
not_to
be_empty
end
end
it
'has issue resource label events'
do
expect
(
saved_project_json
[
'issues'
].
first
[
'resource_label_events'
]).
not_to
be_empty
end
it
'has merge request resource label events'
do
expect
(
saved_project_json
[
'merge_requests'
].
first
[
'resource_label_events'
]).
not_to
be_empty
end
it
'saves the correct service type'
do
it
'saves the correct service type'
do
expect
(
saved_project_json
[
'services'
].
first
[
'type'
]).
to
eq
(
'CustomIssueTrackerService'
)
expect
(
saved_project_json
[
'services'
].
first
[
'type'
]).
to
eq
(
'CustomIssueTrackerService'
)
end
end
...
@@ -296,6 +304,9 @@ describe Gitlab::ImportExport::ProjectTreeSaver do
...
@@ -296,6 +304,9 @@ describe Gitlab::ImportExport::ProjectTreeSaver do
project:
project
,
project:
project
,
commit_id:
ci_build
.
pipeline
.
sha
)
commit_id:
ci_build
.
pipeline
.
sha
)
create
(
:resource_label_event
,
label:
project_label
,
issue:
issue
)
create
(
:resource_label_event
,
label:
group_label
,
merge_request:
merge_request
)
create
(
:event
,
:created
,
target:
milestone
,
project:
project
,
author:
user
)
create
(
:event
,
:created
,
target:
milestone
,
project:
project
,
author:
user
)
create
(
:service
,
project:
project
,
type:
'CustomIssueTrackerService'
,
category:
'issue_tracker'
,
properties:
{
one:
'value'
})
create
(
:service
,
project:
project
,
type:
'CustomIssueTrackerService'
,
category:
'issue_tracker'
,
properties:
{
one:
'value'
})
...
...
spec/lib/gitlab/import_export/safe_model_attributes.yml
View file @
5c42ad75
...
@@ -627,3 +627,11 @@ ProtectedEnvironment::DeployAccessLevel:
...
@@ -627,3 +627,11 @@ ProtectedEnvironment::DeployAccessLevel:
-
updated_at
-
updated_at
-
user_id
-
user_id
-
group_id
-
group_id
ResourceLabelEvent
:
-
id
-
action
-
issue_id
-
merge_request_id
-
label_id
-
user_id
-
created_at
spec/models/label_note_spec.rb
0 → 100644
View file @
5c42ad75
# frozen_string_literal: true
require
'spec_helper'
describe
LabelNote
do
set
(
:project
)
{
create
(
:project
,
:repository
)
}
set
(
:user
)
{
create
(
:user
)
}
set
(
:label
)
{
create
(
:label
,
project:
project
)
}
set
(
:label2
)
{
create
(
:label
,
project:
project
)
}
let
(
:resource_parent
)
{
project
}
context
'when resource is issue'
do
set
(
:resource
)
{
create
(
:issue
,
project:
project
)
}
it_behaves_like
'label note created from events'
end
context
'when resource is merge request'
do
set
(
:resource
)
{
create
(
:merge_request
,
source_project:
project
,
target_project:
project
)
}
it_behaves_like
'label note created from events'
end
end
spec/models/resource_label_event_spec.rb
View file @
5c42ad75
...
@@ -3,7 +3,7 @@
...
@@ -3,7 +3,7 @@
require
'rails_helper'
require
'rails_helper'
RSpec
.
describe
ResourceLabelEvent
,
type: :model
do
RSpec
.
describe
ResourceLabelEvent
,
type: :model
do
subject
{
build
(
:resource_label_event
)
}
subject
{
build
(
:resource_label_event
,
issue:
issue
)
}
let
(
:issue
)
{
create
(
:issue
)
}
let
(
:issue
)
{
create
(
:issue
)
}
let
(
:merge_request
)
{
create
(
:merge_request
)
}
let
(
:merge_request
)
{
create
(
:merge_request
)
}
...
@@ -16,8 +16,6 @@ RSpec.describe ResourceLabelEvent, type: :model do
...
@@ -16,8 +16,6 @@ RSpec.describe ResourceLabelEvent, type: :model do
describe
'validations'
do
describe
'validations'
do
it
{
is_expected
.
to
be_valid
}
it
{
is_expected
.
to
be_valid
}
it
{
is_expected
.
to
validate_presence_of
(
:label
)
}
it
{
is_expected
.
to
validate_presence_of
(
:user
)
}
describe
'Issuable validation'
do
describe
'Issuable validation'
do
it
'is invalid if issue_id and merge_request_id are missing'
do
it
'is invalid if issue_id and merge_request_id are missing'
do
...
@@ -45,4 +43,52 @@ RSpec.describe ResourceLabelEvent, type: :model do
...
@@ -45,4 +43,52 @@ RSpec.describe ResourceLabelEvent, type: :model do
end
end
end
end
end
end
describe
'#expire_etag_cache'
do
def
expect_expiration
(
issue
)
expect_any_instance_of
(
Gitlab
::
EtagCaching
::
Store
)
.
to
receive
(
:touch
)
.
with
(
"/
#{
issue
.
project
.
namespace
.
to_param
}
/
#{
issue
.
project
.
to_param
}
/noteable/issue/
#{
issue
.
id
}
/notes"
)
end
it
'expires resource note etag cache on event save'
do
expect_expiration
(
subject
.
issuable
)
subject
.
save!
end
it
'expires resource note etag cache on event destroy'
do
subject
.
save!
expect_expiration
(
subject
.
issuable
)
subject
.
destroy!
end
end
describe
'#outdated_markdown?'
do
it
'returns true if label is missing and reference is not empty'
do
subject
.
attributes
=
{
reference:
'ref'
,
label_id:
nil
}
expect
(
subject
.
outdated_markdown?
).
to
be
true
end
it
'returns true if reference is not set yet'
do
subject
.
attributes
=
{
reference:
nil
}
expect
(
subject
.
outdated_markdown?
).
to
be
true
end
it
'returns true markdown is outdated'
do
subject
.
attributes
=
{
cached_markdown_version:
0
}
expect
(
subject
.
outdated_markdown?
).
to
be
true
end
it
'returns false if label and reference are set'
do
subject
.
attributes
=
{
reference:
'whatever'
,
cached_markdown_version:
CacheMarkdownField
::
CACHE_COMMONMARK_VERSION
}
expect
(
subject
.
outdated_markdown?
).
to
be
false
end
end
end
end
spec/requests/api/resource_label_events_spec.rb
0 → 100644
View file @
5c42ad75
# frozen_string_literal: true
require
'spec_helper'
describe
API
::
ResourceLabelEvents
do
set
(
:user
)
{
create
(
:user
)
}
set
(
:project
)
{
create
(
:project
,
:public
,
:repository
,
namespace:
user
.
namespace
)
}
set
(
:private_user
)
{
create
(
:user
)
}
before
do
project
.
add_developer
(
user
)
end
shared_examples
'resource_label_events API'
do
|
parent_type
,
eventable_type
,
id_name
|
describe
"GET /
#{
parent_type
}
/:id/
#{
eventable_type
}
/:noteable_id/resource_label_events"
do
it
"returns an array of resource label events"
do
get
api
(
"/
#{
parent_type
}
/
#{
parent
.
id
}
/
#{
eventable_type
}
/
#{
eventable
[
id_name
]
}
/resource_label_events"
,
user
)
expect
(
response
).
to
have_gitlab_http_status
(
200
)
expect
(
response
).
to
include_pagination_headers
expect
(
json_response
).
to
be_an
Array
expect
(
json_response
.
first
[
'id'
]).
to
eq
(
event
.
id
)
end
it
"returns a 404 error when eventable id not found"
do
get
api
(
"/
#{
parent_type
}
/
#{
parent
.
id
}
/
#{
eventable_type
}
/12345/resource_label_events"
,
user
)
expect
(
response
).
to
have_gitlab_http_status
(
404
)
end
it
"returns 404 when not authorized"
do
parent
.
update!
(
visibility_level:
Gitlab
::
VisibilityLevel
::
PRIVATE
)
get
api
(
"/
#{
parent_type
}
/
#{
parent
.
id
}
/
#{
eventable_type
}
/
#{
eventable
[
id_name
]
}
/resource_label_events"
,
private_user
)
expect
(
response
).
to
have_gitlab_http_status
(
404
)
end
end
describe
"GET /
#{
parent_type
}
/:id/
#{
eventable_type
}
/:noteable_id/resource_label_events/:event_id"
do
it
"returns a resource label event by id"
do
get
api
(
"/
#{
parent_type
}
/
#{
parent
.
id
}
/
#{
eventable_type
}
/
#{
eventable
[
id_name
]
}
/resource_label_events/
#{
event
.
id
}
"
,
user
)
expect
(
response
).
to
have_gitlab_http_status
(
200
)
expect
(
json_response
[
'id'
]).
to
eq
(
event
.
id
)
end
it
"returns a 404 error if resource label event not found"
do
get
api
(
"/
#{
parent_type
}
/
#{
parent
.
id
}
/
#{
eventable_type
}
/
#{
eventable
[
id_name
]
}
/resource_label_events/12345"
,
user
)
expect
(
response
).
to
have_gitlab_http_status
(
404
)
end
end
end
context
'when eventable is an Issue'
do
let
(
:issue
)
{
create
(
:issue
,
project:
project
,
author:
user
)
}
it_behaves_like
'resource_label_events API'
,
'projects'
,
'issues'
,
'iid'
do
let
(
:parent
)
{
project
}
let
(
:eventable
)
{
issue
}
let!
(
:event
)
{
create
(
:resource_label_event
,
issue:
issue
)
}
end
end
context
'when eventable is an Epic'
do
let
(
:group
)
{
create
(
:group
,
:public
)
}
let
(
:epic
)
{
create
(
:epic
,
group:
group
,
author:
user
)
}
before
do
group
.
add_owner
(
user
)
stub_licensed_features
(
epics:
true
)
end
it_behaves_like
'resource_label_events API'
,
'groups'
,
'epics'
,
'id'
do
let
(
:parent
)
{
group
}
let
(
:eventable
)
{
epic
}
let!
(
:event
)
{
create
(
:resource_label_event
,
epic:
epic
)
}
end
end
context
'when eventable is a Merge Request'
do
let
(
:merge_request
)
{
create
(
:merge_request
,
source_project:
project
,
target_project:
project
,
author:
user
)
}
it_behaves_like
'resource_label_events API'
,
'projects'
,
'merge_requests'
,
'iid'
do
let
(
:parent
)
{
project
}
let
(
:eventable
)
{
merge_request
}
let!
(
:event
)
{
create
(
:resource_label_event
,
merge_request:
merge_request
)
}
end
end
end
spec/services/issuable/common_system_notes_service_spec.rb
View file @
5c42ad75
...
@@ -12,12 +12,21 @@ describe Issuable::CommonSystemNotesService do
...
@@ -12,12 +12,21 @@ describe Issuable::CommonSystemNotesService do
it_behaves_like
'system note creation'
,
{
time_estimate:
5
},
'changed time estimate'
it_behaves_like
'system note creation'
,
{
time_estimate:
5
},
'changed time estimate'
context
'when new label is added'
do
context
'when new label is added'
do
let
(
:label
)
{
create
(
:label
,
project:
project
)
}
before
do
before
do
label
=
create
(
:label
,
project:
project
)
issuable
.
labels
<<
label
issuable
.
labels
<<
label
issuable
.
save
end
end
it_behaves_like
'system note creation'
,
{},
/added ~\w+ label/
it
'creates a resource label event'
do
described_class
.
new
(
project
,
user
).
execute
(
issuable
,
[])
event
=
issuable
.
reload
.
resource_label_events
.
last
expect
(
event
).
not_to
be_nil
expect
(
event
.
label_id
).
to
eq
label
.
id
expect
(
event
.
user_id
).
to
eq
user
.
id
end
end
end
context
'when new milestone is assigned'
do
context
'when new milestone is assigned'
do
...
...
spec/services/issues/move_service_spec.rb
View file @
5c42ad75
...
@@ -122,6 +122,17 @@ describe Issues::MoveService do
...
@@ -122,6 +122,17 @@ describe Issues::MoveService do
end
end
end
end
context
'issue with resource label events'
do
it
'assigns resource label events to new issue'
do
old_issue
.
resource_label_events
=
create_list
(
:resource_label_event
,
2
,
issue:
old_issue
)
new_issue
=
move_service
.
execute
(
old_issue
,
new_project
)
expected
=
old_issue
.
resource_label_events
.
map
(
&
:label_id
)
expect
(
new_issue
.
resource_label_events
.
map
(
&
:label_id
)).
to
match_array
(
expected
)
end
end
context
'generic issue'
do
context
'generic issue'
do
include_context
'issue move executed'
include_context
'issue move executed'
...
...
spec/services/issues/update_service_spec.rb
View file @
5c42ad75
...
@@ -189,11 +189,12 @@ describe Issues::UpdateService, :mailer do
...
@@ -189,11 +189,12 @@ describe Issues::UpdateService, :mailer do
expect
(
note
.
note
).
to
include
"assigned to
#{
user2
.
to_reference
}
"
expect
(
note
.
note
).
to
include
"assigned to
#{
user2
.
to_reference
}
"
end
end
it
'creates
system note about issue label edi
t'
do
it
'creates
a resource label even
t'
do
note
=
find_note
(
'added ~'
)
event
=
issue
.
resource_label_events
.
last
expect
(
note
).
not_to
be_nil
expect
(
event
).
not_to
be_nil
expect
(
note
.
note
).
to
include
"added
#{
label
.
to_reference
}
label"
expect
(
event
.
label_id
).
to
eq
label
.
id
expect
(
event
.
user_id
).
to
eq
user
.
id
end
end
it
'creates system note about title change'
do
it
'creates system note about title change'
do
...
...
spec/services/merge_requests/update_service_spec.rb
View file @
5c42ad75
...
@@ -109,11 +109,12 @@ describe MergeRequests::UpdateService, :mailer do
...
@@ -109,11 +109,12 @@ describe MergeRequests::UpdateService, :mailer do
expect
(
note
.
note
).
to
include
"assigned to
#{
user2
.
to_reference
}
"
expect
(
note
.
note
).
to
include
"assigned to
#{
user2
.
to_reference
}
"
end
end
it
'creates
system note about merge_request label edi
t'
do
it
'creates
a resource label even
t'
do
note
=
find_note
(
'added ~'
)
event
=
merge_request
.
resource_label_events
.
last
expect
(
note
).
not_to
be_nil
expect
(
event
).
not_to
be_nil
expect
(
note
.
note
).
to
include
"added
#{
label
.
to_reference
}
label"
expect
(
event
.
label_id
).
to
eq
label
.
id
expect
(
event
.
user_id
).
to
eq
user
.
id
end
end
it
'creates system note about title change'
do
it
'creates system note about title change'
do
...
...
spec/services/resource_events/change_labels_service_spec.rb
View file @
5c42ad75
...
@@ -18,6 +18,14 @@ describe ResourceEvents::ChangeLabelsService do
...
@@ -18,6 +18,14 @@ describe ResourceEvents::ChangeLabelsService do
expect
(
event
.
action
).
to
eq
(
action
)
expect
(
event
.
action
).
to
eq
(
action
)
end
end
it
'expires resource note etag cache'
do
expect_any_instance_of
(
Gitlab
::
EtagCaching
::
Store
)
.
to
receive
(
:touch
)
.
with
(
"/
#{
resource
.
project
.
namespace
.
to_param
}
/
#{
resource
.
project
.
to_param
}
/noteable/issue/
#{
resource
.
id
}
/notes"
)
described_class
.
new
(
resource
,
author
).
execute
(
added_labels:
[
labels
[
0
]])
end
context
'when adding a label'
do
context
'when adding a label'
do
let
(
:added
)
{
[
labels
[
0
]]
}
let
(
:added
)
{
[
labels
[
0
]]
}
let
(
:removed
)
{
[]
}
let
(
:removed
)
{
[]
}
...
...
spec/services/resource_events/merge_into_notes_service_spec.rb
0 → 100644
View file @
5c42ad75
# frozen_string_literal: true
require
'spec_helper'
describe
ResourceEvents
::
MergeIntoNotesService
do
def
create_event
(
params
)
event_params
=
{
action: :add
,
label:
label
,
issue:
resource
,
user:
user
}
create
(
:resource_label_event
,
event_params
.
merge
(
params
))
end
def
create_note
(
params
)
opts
=
{
noteable:
resource
,
project:
project
}
create
(
:note_on_issue
,
opts
.
merge
(
params
))
end
set
(
:project
)
{
create
(
:project
)
}
set
(
:user
)
{
create
(
:user
)
}
set
(
:resource
)
{
create
(
:issue
,
project:
project
)
}
set
(
:label
)
{
create
(
:label
,
project:
project
)
}
set
(
:label2
)
{
create
(
:label
,
project:
project
)
}
let
(
:time
)
{
Time
.
now
}
describe
'#execute'
do
it
'merges label events into notes in order of created_at'
do
note1
=
create_note
(
created_at:
4
.
days
.
ago
)
note2
=
create_note
(
created_at:
2
.
days
.
ago
)
event1
=
create_event
(
created_at:
3
.
days
.
ago
)
event2
=
create_event
(
created_at:
1
.
day
.
ago
)
notes
=
described_class
.
new
(
resource
,
user
).
execute
([
note1
,
note2
])
expected
=
[
note1
,
event1
,
note2
,
event2
].
map
(
&
:discussion_id
)
expect
(
notes
.
map
(
&
:discussion_id
)).
to
eq
expected
end
it
'squashes events with same time and author into single note'
do
user2
=
create
(
:user
)
create_event
(
created_at:
time
)
create_event
(
created_at:
time
,
label:
label2
,
action: :remove
)
create_event
(
created_at:
time
,
user:
user2
)
create_event
(
created_at:
1
.
day
.
ago
,
label:
label2
)
notes
=
described_class
.
new
(
resource
,
user
).
execute
()
expected
=
[
"added
#{
label
.
to_reference
}
label and removed
#{
label2
.
to_reference
}
label"
,
"added
#{
label
.
to_reference
}
label"
,
"added
#{
label2
.
to_reference
}
label"
]
expect
(
notes
.
count
).
to
eq
3
expect
(
notes
.
map
(
&
:note
)).
to
match_array
expected
end
it
'fetches only notes created after last_fetched_at'
do
create_event
(
created_at:
4
.
days
.
ago
)
event
=
create_event
(
created_at:
1
.
day
.
ago
)
notes
=
described_class
.
new
(
resource
,
user
,
last_fetched_at:
2
.
days
.
ago
.
to_i
).
execute
()
expect
(
notes
.
count
).
to
eq
1
expect
(
notes
.
first
.
discussion_id
).
to
eq
event
.
discussion_id
end
end
end
spec/services/system_note_service_spec.rb
View file @
5c42ad75
...
@@ -199,45 +199,6 @@ describe SystemNoteService do
...
@@ -199,45 +199,6 @@ describe SystemNoteService do
end
end
end
end
describe
'.change_label'
do
subject
{
described_class
.
change_label
(
noteable
,
project
,
author
,
added
,
removed
)
}
let
(
:labels
)
{
create_list
(
:label
,
2
,
project:
project
)
}
let
(
:added
)
{
[]
}
let
(
:removed
)
{
[]
}
it_behaves_like
'a system note'
do
let
(
:action
)
{
'label'
}
end
context
'with added labels'
do
let
(
:added
)
{
labels
}
let
(
:removed
)
{
[]
}
it
'sets the note text'
do
expect
(
subject
.
note
).
to
eq
"added ~
#{
labels
[
0
].
id
}
~
#{
labels
[
1
].
id
}
labels"
end
end
context
'with removed labels'
do
let
(
:added
)
{
[]
}
let
(
:removed
)
{
labels
}
it
'sets the note text'
do
expect
(
subject
.
note
).
to
eq
"removed ~
#{
labels
[
0
].
id
}
~
#{
labels
[
1
].
id
}
labels"
end
end
context
'with added and removed labels'
do
let
(
:added
)
{
[
labels
[
0
]]
}
let
(
:removed
)
{
[
labels
[
1
]]
}
it
'sets the note text'
do
expect
(
subject
.
note
).
to
eq
"added ~
#{
labels
[
0
].
id
}
and removed ~
#{
labels
[
1
].
id
}
labels"
end
end
end
describe
'.change_milestone'
do
describe
'.change_milestone'
do
context
'for a project milestone'
do
context
'for a project milestone'
do
subject
{
described_class
.
change_milestone
(
noteable
,
project
,
author
,
milestone
)
}
subject
{
described_class
.
change_milestone
(
noteable
,
project
,
author
,
milestone
)
}
...
...
spec/support/shared_examples/models/label_note_shared_examples.rb
0 → 100644
View file @
5c42ad75
# frozen_string_literal: true
shared_examples
'label note created from events'
do
def
create_event
(
params
=
{})
event_params
=
{
action: :add
,
label:
label
,
user:
user
}
resource_key
=
resource
.
class
.
name
.
underscore
.
to_s
event_params
[
resource_key
]
=
resource
build
(
:resource_label_event
,
event_params
.
merge
(
params
))
end
def
label_refs
(
events
)
sorted_labels
=
events
.
map
(
&
:label
).
compact
.
sort_by
(
&
:title
)
sorted_labels
.
map
{
|
l
|
l
.
to_reference
}.
join
(
' '
)
end
let
(
:time
)
{
Time
.
now
}
let
(
:local_label_ids
)
{
[
label
.
id
,
label2
.
id
]
}
describe
'.from_events'
do
it
'returns system note with expected attributes'
do
event
=
create_event
note
=
described_class
.
from_events
([
event
,
create_event
])
expect
(
note
.
system
).
to
be
true
expect
(
note
.
author_id
).
to
eq
event
.
user_id
expect
(
note
.
discussion_id
).
to
eq
event
.
discussion_id
expect
(
note
.
noteable
).
to
eq
event
.
issuable
expect
(
note
.
note
).
to
be_present
expect
(
note
.
note_html
).
to
be_present
end
it
'updates markdown cache if reference is not set yet'
do
event
=
create_event
(
reference:
nil
)
described_class
.
from_events
([
event
])
expect
(
event
.
reference
).
not_to
be_nil
end
it
'updates markdown cache if label was deleted'
do
event
=
create_event
(
reference:
'some_ref'
,
label:
nil
)
described_class
.
from_events
([
event
])
expect
(
event
.
reference
).
to
eq
''
end
it
'returns html note'
do
events
=
[
create_event
(
created_at:
time
)]
note
=
described_class
.
from_events
(
events
)
expect
(
note
.
note_html
).
to
include
label
.
title
end
it
'returns text note for added labels'
do
events
=
[
create_event
(
created_at:
time
),
create_event
(
created_at:
time
,
label:
label2
),
create_event
(
created_at:
time
,
label:
nil
)]
note
=
described_class
.
from_events
(
events
)
expect
(
note
.
note
).
to
eq
"added
#{
label_refs
(
events
)
}
+ 1 deleted label"
end
it
'returns text note for removed labels'
do
events
=
[
create_event
(
action: :remove
,
created_at:
time
),
create_event
(
action: :remove
,
created_at:
time
,
label:
label2
),
create_event
(
action: :remove
,
created_at:
time
,
label:
nil
)]
note
=
described_class
.
from_events
(
events
)
expect
(
note
.
note
).
to
eq
"removed
#{
label_refs
(
events
)
}
+ 1 deleted label"
end
it
'returns text note for added and removed labels'
do
add_events
=
[
create_event
(
created_at:
time
),
create_event
(
created_at:
time
,
label:
nil
)]
remove_events
=
[
create_event
(
action: :remove
,
created_at:
time
),
create_event
(
action: :remove
,
created_at:
time
,
label:
nil
)]
note
=
described_class
.
from_events
(
add_events
+
remove_events
)
expect
(
note
.
note
).
to
eq
"added
#{
label_refs
(
add_events
)
}
+ 1 deleted label and removed
#{
label_refs
(
remove_events
)
}
+ 1 deleted label"
end
it
'returns text note for cross-project label'
do
other_label
=
create
(
:label
)
event
=
create_event
(
label:
other_label
)
note
=
described_class
.
from_events
([
event
])
expect
(
note
.
note
).
to
eq
"added
#{
other_label
.
to_reference
(
resource_parent
)
}
label"
end
it
'returns text note for cross-group label'
do
other_label
=
create
(
:group_label
)
event
=
create_event
(
label:
other_label
)
note
=
described_class
.
from_events
([
event
])
expect
(
note
.
note
).
to
eq
"added
#{
other_label
.
to_reference
(
other_label
.
group
,
target_project:
project
,
full:
true
)
}
label"
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