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
3f9b1ecd
Commit
3f9b1ecd
authored
Jul 24, 2019
by
Heinrich Lee Yu
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
Use SQL to find the gap instead of iterating
Also removes unnecessary methods causing extra queries
parent
afbe0b61
Changes
4
Hide whitespace changes
Inline
Side-by-side
Showing
4 changed files
with
92 additions
and
147 deletions
+92
-147
app/models/concerns/relative_positioning.rb
app/models/concerns/relative_positioning.rb
+66
-91
db/migrate/20190802012622_reorder_issues_project_id_relative_position_index.rb
...2622_reorder_issues_project_id_relative_position_index.rb
+24
-0
db/schema.rb
db/schema.rb
+2
-2
spec/support/shared_examples/relative_positioning_shared_examples.rb
...t/shared_examples/relative_positioning_shared_examples.rb
+0
-54
No files found.
app/models/concerns/relative_positioning.rb
View file @
3f9b1ecd
...
@@ -28,13 +28,6 @@ module RelativePositioning
...
@@ -28,13 +28,6 @@ module RelativePositioning
START_POSITION
=
Gitlab
::
Database
::
MAX_INT_VALUE
/
2
START_POSITION
=
Gitlab
::
Database
::
MAX_INT_VALUE
/
2
MAX_POSITION
=
Gitlab
::
Database
::
MAX_INT_VALUE
MAX_POSITION
=
Gitlab
::
Database
::
MAX_INT_VALUE
IDEAL_DISTANCE
=
500
IDEAL_DISTANCE
=
500
MAX_SEQUENCE_LIMIT
=
1000
class
GapNotFound
<
StandardError
def
message
'Could not find a gap in the sequence of relative positions.'
end
end
class_methods
do
class_methods
do
def
move_nulls_to_end
(
objects
)
def
move_nulls_to_end
(
objects
)
...
@@ -117,8 +110,8 @@ module RelativePositioning
...
@@ -117,8 +110,8 @@ module RelativePositioning
return
move_after
(
before
)
unless
after
return
move_after
(
before
)
unless
after
return
move_before
(
after
)
unless
before
return
move_before
(
after
)
unless
before
# If there is no place to insert an item we need to create one by moving the
before item closer
# If there is no place to insert an item we need to create one by moving the
item
#
to its predecessor. This process will recursively move all the predecessors until we have a place
#
before this and all preceding items until there is a gap
before
,
after
=
after
,
before
if
after
.
relative_position
<
before
.
relative_position
before
,
after
=
after
,
before
if
after
.
relative_position
<
before
.
relative_position
if
(
after
.
relative_position
-
before
.
relative_position
)
<
2
if
(
after
.
relative_position
-
before
.
relative_position
)
<
2
after
.
move_sequence_before
after
.
move_sequence_before
...
@@ -132,7 +125,7 @@ module RelativePositioning
...
@@ -132,7 +125,7 @@ module RelativePositioning
pos_before
=
before
.
relative_position
pos_before
=
before
.
relative_position
pos_after
=
before
.
next_relative_position
pos_after
=
before
.
next_relative_position
if
before
.
shift_after?
if
pos_after
&&
(
pos_after
-
pos_before
)
<
2
before
.
move_sequence_after
before
.
move_sequence_after
end
end
...
@@ -143,7 +136,7 @@ module RelativePositioning
...
@@ -143,7 +136,7 @@ module RelativePositioning
pos_after
=
after
.
relative_position
pos_after
=
after
.
relative_position
pos_before
=
after
.
prev_relative_position
pos_before
=
after
.
prev_relative_position
if
after
.
shift_before?
if
pos_before
&&
(
pos_after
-
pos_before
)
<
2
after
.
move_sequence_before
after
.
move_sequence_before
end
end
...
@@ -158,29 +151,76 @@ module RelativePositioning
...
@@ -158,29 +151,76 @@ module RelativePositioning
self
.
relative_position
=
self
.
class
.
position_between
(
min_relative_position
||
START_POSITION
,
MIN_POSITION
)
self
.
relative_position
=
self
.
class
.
position_between
(
min_relative_position
||
START_POSITION
,
MIN_POSITION
)
end
end
# Indicates if there is an item that should be shifted to free the place
# Moves the sequence before the current item to the middle of the next gap
def
shift_after?
# For example, we have 5 11 12 13 14 15 and the current item is 15
next_pos
=
next_relative_position
# This moves the sequence 11 12 13 14 to 8 9 10 11
next_pos
&&
(
next_pos
-
relative_position
)
==
1
def
move_sequence_before
next_gap
=
find_next_gap_before
delta
=
optimum_delta_for_gap
(
next_gap
)
move_sequence
(
next_gap
[
:start
],
relative_position
,
-
delta
)
end
end
# Indicates if there is an item that should be shifted to free the place
# Moves the sequence after the current item to the middle of the next gap
def
shift_before?
# For example, we have 11 12 13 14 15 21 and the current item is 11
prev_pos
=
prev_relative_position
# This moves the sequence 12 13 14 15 to 15 16 17 18
prev_pos
&&
(
relative_position
-
prev_pos
)
==
1
def
move_sequence_after
next_gap
=
find_next_gap_after
delta
=
optimum_delta_for_gap
(
next_gap
)
move_sequence
(
relative_position
,
next_gap
[
:start
],
delta
)
end
end
def
move_sequence_before
private
items_to_move
=
scoped_items_batch
.
where
(
'relative_position <= ?'
,
relative_position
).
order
(
'relative_position DESC'
)
move_nearest_sequence
(
items_to_move
,
MIN_POSITION
)
# Supposing that we have a sequence of items: 1 5 11 12 13 and the current item is 13
# This would return: `{ start: 11, end: 5 }`
def
find_next_gap_before
items_with_next_pos
=
scoped_items
.
select
(
'relative_position AS pos, LEAD(relative_position) OVER (ORDER BY relative_position DESC) AS next_pos'
)
.
where
(
'relative_position <= ?'
,
relative_position
)
.
order
(
relative_position: :desc
)
find_next_gap
(
items_with_next_pos
).
tap
do
|
gap
|
gap
[
:end
]
||=
MIN_POSITION
end
end
end
def
move_sequence_after
# Supposing that we have a sequence of items: 13 14 15 20 24 and the current item is 13
items_to_move
=
scoped_items_batch
.
where
(
'relative_position >= ?'
,
relative_position
).
order
(
:relative_position
)
# This would return: `{ start: 15, end: 20 }`
move_nearest_sequence
(
items_to_move
,
MAX_POSITION
)
def
find_next_gap_after
items_with_next_pos
=
scoped_items
.
select
(
'relative_position AS pos, LEAD(relative_position) OVER (ORDER BY relative_position ASC) AS next_pos'
)
.
where
(
'relative_position >= ?'
,
relative_position
)
.
order
(
:relative_position
)
find_next_gap
(
items_with_next_pos
).
tap
do
|
gap
|
gap
[
:end
]
||=
MAX_POSITION
end
end
end
private
def
find_next_gap
(
items_with_next_pos
)
gap
=
self
.
class
.
from
(
items_with_next_pos
,
:items_with_next_pos
)
.
where
(
'ABS(pos - next_pos) > 1 OR next_pos IS NULL'
)
.
limit
(
1
)
.
pluck
(
:pos
,
:next_pos
)
.
first
{
start:
gap
[
0
],
end:
gap
[
1
]
}
end
def
optimum_delta_for_gap
(
gap
)
delta
=
((
gap
[
:start
]
-
gap
[
:end
])
/
2.0
).
abs
.
ceil
[
delta
,
IDEAL_DISTANCE
].
min
end
def
move_sequence
(
start_pos
,
end_pos
,
delta
)
scoped_items
.
where
.
not
(
id:
self
.
id
)
.
where
(
'relative_position BETWEEN ? AND ?'
,
start_pos
,
end_pos
)
.
update_all
(
"relative_position = relative_position +
#{
delta
}
"
)
end
def
calculate_relative_position
(
calculation
)
def
calculate_relative_position
(
calculation
)
# When calculating across projects, this is much more efficient than
# When calculating across projects, this is much more efficient than
...
@@ -202,69 +242,4 @@ module RelativePositioning
...
@@ -202,69 +242,4 @@ module RelativePositioning
def
scoped_items
def
scoped_items
self
.
class
.
relative_positioning_query_base
(
self
)
self
.
class
.
relative_positioning_query_base
(
self
)
end
end
def
scoped_items_batch
scoped_items
.
limit
(
MAX_SEQUENCE_LIMIT
).
select
(
:id
,
:relative_position
).
where
.
not
(
id:
self
.
id
)
end
# Supposing that we have a sequence of positions: 5 11 12 13 14 15,
# and we want to move another item between 14 and 15, then
# we shift previous positions at least by one item, but ideally to the middle
# of the nearest gap. In this case gap is between 5 and 11 so
# this would move 11 12 13 14 to 8 9 10 11.
def
move_nearest_sequence
(
items
,
end_position
)
gap_idx
,
gap_size
=
find_gap_in_sequence
(
items
)
# If we didn't find a gap in the sequence, it's still possible that there
# are some free positions before the first item
if
gap_idx
.
nil?
&&
!
items
.
empty?
&&
items
.
size
<
MAX_SEQUENCE_LIMIT
&&
items
.
last
.
relative_position
!=
end_position
gap_idx
=
items
.
size
gap_size
=
end_position
-
items
.
last
.
relative_position
end
# The chance that there is a sequence of 1000 positions w/o gap is really
# low, but it would be good to rebalance all positions in the scope instead
# of raising an exception:
# https://gitlab.com/gitlab-org/gitlab-ce/issues/64514#note_192657097
raise
GapNotFound
if
gap_idx
.
nil?
# No shift is needed if gap is next to the item being moved
return
true
if
gap_idx
==
0
delta
=
max_delta_for_sequence
(
gap_size
)
sequence_ids
=
items
.
first
(
gap_idx
).
map
(
&
:id
)
move_ids_by_delta
(
sequence_ids
,
delta
)
end
def
max_delta_for_sequence
(
gap_size
)
delta
=
gap_size
/
2
if
delta
.
abs
>
IDEAL_DISTANCE
delta
=
delta
<
0
?
-
IDEAL_DISTANCE
:
IDEAL_DISTANCE
end
delta
end
def
move_ids_by_delta
(
ids
,
delta
)
self
.
class
.
where
(
id:
ids
).
update_all
(
"relative_position=relative_position+
#{
delta
}
"
)
end
def
find_gap_in_sequence
(
items
)
prev
=
relative_position
gap
=
nil
items
.
each_with_index
do
|
rec
,
idx
|
size
=
rec
.
relative_position
-
prev
if
size
.
abs
>
1
gap
=
[
idx
,
size
]
break
end
prev
=
rec
.
relative_position
end
gap
end
end
end
db/migrate/20190802012622_reorder_issues_project_id_relative_position_index.rb
0 → 100644
View file @
3f9b1ecd
# frozen_string_literal: true
class
ReorderIssuesProjectIdRelativePositionIndex
<
ActiveRecord
::
Migration
[
5.2
]
include
Gitlab
::
Database
::
MigrationHelpers
DOWNTIME
=
false
disable_ddl_transaction!
OLD_INDEX_NAME
=
'index_issues_on_project_id_and_state_and_rel_position_and_id'
NEW_INDEX_NAME
=
'index_issues_on_project_id_and_rel_position_and_state_and_id'
def
up
add_concurrent_index
:issues
,
[
:project_id
,
:relative_position
,
:state
,
:id
],
order:
{
id: :desc
},
name:
NEW_INDEX_NAME
remove_concurrent_index_by_name
:issues
,
OLD_INDEX_NAME
end
def
down
add_concurrent_index
:issues
,
[
:project_id
,
:state
,
:relative_position
,
:id
],
order:
{
id: :desc
},
name:
OLD_INDEX_NAME
remove_concurrent_index_by_name
:issues
,
NEW_INDEX_NAME
end
end
db/schema.rb
View file @
3f9b1ecd
...
@@ -10,7 +10,7 @@
...
@@ -10,7 +10,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:
2019_08_0
1_114109
)
do
ActiveRecord
::
Schema
.
define
(
version:
2019_08_0
2_012622
)
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
"pg_trgm"
enable_extension
"pg_trgm"
...
@@ -1716,7 +1716,7 @@ ActiveRecord::Schema.define(version: 2019_08_01_114109) do
...
@@ -1716,7 +1716,7 @@ ActiveRecord::Schema.define(version: 2019_08_01_114109) do
t
.
index
[
"project_id"
,
"created_at"
,
"id"
,
"state"
],
name:
"index_issues_on_project_id_and_created_at_and_id_and_state"
t
.
index
[
"project_id"
,
"created_at"
,
"id"
,
"state"
],
name:
"index_issues_on_project_id_and_created_at_and_id_and_state"
t
.
index
[
"project_id"
,
"due_date"
,
"id"
,
"state"
],
name:
"idx_issues_on_project_id_and_due_date_and_id_and_state_partial"
,
where:
"(due_date IS NOT NULL)"
t
.
index
[
"project_id"
,
"due_date"
,
"id"
,
"state"
],
name:
"idx_issues_on_project_id_and_due_date_and_id_and_state_partial"
,
where:
"(due_date IS NOT NULL)"
t
.
index
[
"project_id"
,
"iid"
],
name:
"index_issues_on_project_id_and_iid"
,
unique:
true
t
.
index
[
"project_id"
,
"iid"
],
name:
"index_issues_on_project_id_and_iid"
,
unique:
true
t
.
index
[
"project_id"
,
"
state"
,
"relative_position"
,
"id"
],
name:
"index_issues_on_project_id_and_state_and_rel_position
_and_id"
,
order:
{
id: :desc
}
t
.
index
[
"project_id"
,
"
relative_position"
,
"state"
,
"id"
],
name:
"index_issues_on_project_id_and_rel_position_and_state
_and_id"
,
order:
{
id: :desc
}
t
.
index
[
"project_id"
,
"updated_at"
,
"id"
,
"state"
],
name:
"index_issues_on_project_id_and_updated_at_and_id_and_state"
t
.
index
[
"project_id"
,
"updated_at"
,
"id"
,
"state"
],
name:
"index_issues_on_project_id_and_updated_at_and_id_and_state"
t
.
index
[
"relative_position"
],
name:
"index_issues_on_relative_position"
t
.
index
[
"relative_position"
],
name:
"index_issues_on_relative_position"
t
.
index
[
"state"
],
name:
"index_issues_on_state"
t
.
index
[
"state"
],
name:
"index_issues_on_state"
...
...
spec/support/shared_examples/relative_positioning_shared_examples.rb
View file @
3f9b1ecd
...
@@ -110,46 +110,6 @@ RSpec.shared_examples 'a class that supports relative positioning' do
...
@@ -110,46 +110,6 @@ RSpec.shared_examples 'a class that supports relative positioning' do
end
end
end
end
describe
'#shift_after?'
do
before
do
[
item1
,
item2
].
each
do
|
item1
|
item1
.
move_to_end
&&
item1
.
save
end
end
it
'returns true'
do
item1
.
update
(
relative_position:
item2
.
relative_position
-
1
)
expect
(
item1
.
shift_after?
).
to
be_truthy
end
it
'returns false'
do
item1
.
update
(
relative_position:
item2
.
relative_position
-
2
)
expect
(
item1
.
shift_after?
).
to
be_falsey
end
end
describe
'#shift_before?'
do
before
do
[
item1
,
item2
].
each
do
|
item1
|
item1
.
move_to_end
&&
item1
.
save
end
end
it
'returns true'
do
item1
.
update
(
relative_position:
item2
.
relative_position
+
1
)
expect
(
item1
.
shift_before?
).
to
be_truthy
end
it
'returns false'
do
item1
.
update
(
relative_position:
item2
.
relative_position
+
2
)
expect
(
item1
.
shift_before?
).
to
be_falsey
end
end
describe
'#move_between'
do
describe
'#move_between'
do
before
do
before
do
[
item1
,
item2
].
each
do
|
item1
|
[
item1
,
item2
].
each
do
|
item1
|
...
@@ -297,13 +257,6 @@ RSpec.shared_examples 'a class that supports relative positioning' do
...
@@ -297,13 +257,6 @@ RSpec.shared_examples 'a class that supports relative positioning' do
positions
=
items
.
map
{
|
item
|
item
.
reload
.
relative_position
}
positions
=
items
.
map
{
|
item
|
item
.
reload
.
relative_position
}
expect
(
positions
).
to
eq
([
50
,
51
,
102
])
expect
(
positions
).
to
eq
([
50
,
51
,
102
])
end
end
it
'if raises an exception if gap is not found'
do
stub_const
(
'RelativePositioning::MAX_SEQUENCE_LIMIT'
,
2
)
items
=
create_items_with_positions
([
100
,
101
,
102
])
expect
{
items
.
last
.
move_sequence_before
}.
to
raise_error
(
RelativePositioning
::
GapNotFound
)
end
end
end
describe
'#move_sequence_after'
do
describe
'#move_sequence_after'
do
...
@@ -326,12 +279,5 @@ RSpec.shared_examples 'a class that supports relative positioning' do
...
@@ -326,12 +279,5 @@ RSpec.shared_examples 'a class that supports relative positioning' do
positions
=
items
.
map
{
|
item
|
item
.
reload
.
relative_position
}
positions
=
items
.
map
{
|
item
|
item
.
reload
.
relative_position
}
expect
(
positions
).
to
eq
([
100
,
601
,
602
])
expect
(
positions
).
to
eq
([
100
,
601
,
602
])
end
end
it
'if raises an exception if gap is not found'
do
stub_const
(
'RelativePositioning::MAX_SEQUENCE_LIMIT'
,
2
)
items
=
create_items_with_positions
([
100
,
101
,
102
])
expect
{
items
.
first
.
move_sequence_after
}.
to
raise_error
(
RelativePositioning
::
GapNotFound
)
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