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
0
Merge Requests
0
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
Boxiang Sun
gitlab-ce
Commits
96fe1856
Commit
96fe1856
authored
Mar 09, 2017
by
Valery Sizov
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
Fix relative position calculation
parent
b3eda944
Changes
2
Show whitespace changes
Inline
Side-by-side
Showing
2 changed files
with
39 additions
and
8 deletions
+39
-8
app/models/concerns/relative_positioning.rb
app/models/concerns/relative_positioning.rb
+12
-8
spec/models/concerns/relative_positioning_spec.rb
spec/models/concerns/relative_positioning_spec.rb
+27
-0
No files found.
app/models/concerns/relative_positioning.rb
View file @
96fe1856
...
@@ -3,6 +3,7 @@ module RelativePositioning
...
@@ -3,6 +3,7 @@ module RelativePositioning
MIN_POSITION
=
0
MIN_POSITION
=
0
MAX_POSITION
=
Gitlab
::
Database
::
MAX_INT_VALUE
MAX_POSITION
=
Gitlab
::
Database
::
MAX_INT_VALUE
DISTANCE
=
500
included
do
included
do
after_save
:save_positionable_neighbours
after_save
:save_positionable_neighbours
...
@@ -49,7 +50,9 @@ module RelativePositioning
...
@@ -49,7 +50,9 @@ module RelativePositioning
pos_before
=
before
.
relative_position
pos_before
=
before
.
relative_position
pos_after
=
after
.
relative_position
pos_after
=
after
.
relative_position
if
pos_after
&&
(
pos_before
==
pos_after
)
# We can't insert an issue between two other if the distance is 1 or 0
# so we need to handle this collision properly
if
pos_after
&&
(
pos_after
-
pos_before
).
abs
<=
1
self
.
relative_position
=
pos_before
self
.
relative_position
=
pos_before
before
.
move_before
(
self
)
before
.
move_before
(
self
)
after
.
move_after
(
self
)
after
.
move_after
(
self
)
...
@@ -75,19 +78,20 @@ module RelativePositioning
...
@@ -75,19 +78,20 @@ module RelativePositioning
private
private
# This method takes two integer values (positions) and
# This method takes two integer values (positions) and
# calculates some random position between them. The range is huge as
# calculates the position between them. The range is huge as
# the maximum integer value is 2147483647. Ideally, the calculated value would be
# the maximum integer value is 2147483647. We are incrementing position by 1000 every time
# exactly between those terminating values, but this will introduce possibility of a race condition
# when we have enough space. If distance is less then 500 we are calculating an average number
# so two or more issues can get the same value, we want to avoid that and we also want to avoid
# using a lock here. If we have two issues with distance more than one thousand, we are OK.
# Given the huge range of possible values that integer can fit we shoud never face a problem.
def
position_between
(
pos_before
,
pos_after
)
def
position_between
(
pos_before
,
pos_after
)
pos_before
||=
MIN_POSITION
pos_before
||=
MIN_POSITION
pos_after
||=
MAX_POSITION
pos_after
||=
MAX_POSITION
pos_before
,
pos_after
=
[
pos_before
,
pos_after
].
sort
pos_before
,
pos_after
=
[
pos_before
,
pos_after
].
sort
rand
(
pos_before
.
next
..
pos_after
.
pred
)
if
pos_after
-
pos_before
>
DISTANCE
*
2
pos_before
+
DISTANCE
else
pos_before
+
(
pos_after
-
pos_before
)
/
2
end
end
end
def
save_positionable_neighbours
def
save_positionable_neighbours
...
...
spec/models/concerns/relative_positioning_spec.rb
View file @
96fe1856
...
@@ -100,5 +100,32 @@ describe Issue, 'RelativePositioning' do
...
@@ -100,5 +100,32 @@ describe Issue, 'RelativePositioning' do
expect
(
new_issue
.
relative_position
).
to
be
>
issue
.
relative_position
expect
(
new_issue
.
relative_position
).
to
be
>
issue
.
relative_position
expect
(
issue
.
relative_position
).
to
be
<
issue1
.
relative_position
expect
(
issue
.
relative_position
).
to
be
<
issue1
.
relative_position
end
end
it
'positions issues between other two if distance is 1'
do
issue1
.
update
relative_position:
issue
.
relative_position
+
1
new_issue
.
move_between
(
issue
,
issue1
)
expect
(
new_issue
.
relative_position
).
to
be
>
issue
.
relative_position
expect
(
issue
.
relative_position
).
to
be
<
issue1
.
relative_position
end
it
'positions issue closer to before-issue if distance is big enough'
do
issue
.
update
relative_position:
100
issue1
.
update
relative_position:
6000
new_issue
.
move_between
(
issue
,
issue1
)
expect
(
new_issue
.
relative_position
).
to
eq
(
100
+
RelativePositioning
::
DISTANCE
)
end
it
'positions issue in the middle of other two if distance is not big enough'
do
issue
.
update
relative_position:
100
issue1
.
update
relative_position:
400
new_issue
.
move_between
(
issue
,
issue1
)
expect
(
new_issue
.
relative_position
).
to
eq
(
250
)
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