Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Add move method to automation rule #5998

Merged
merged 19 commits into from
Sep 4, 2019
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
82 changes: 82 additions & 0 deletions readthedocs/builds/models.py
Original file line number Diff line number Diff line change
Expand Up @@ -8,6 +8,7 @@

from django.conf import settings
from django.db import models
from django.db.models import F
from django.urls import reverse
from django.utils import timezone
from django.utils.translation import ugettext
Expand Down Expand Up @@ -1035,6 +1036,87 @@ def apply_action(self, version, match_result):
raise NotImplementedError
action(version, match_result, self.action_arg)

def move(self, steps):
"""
Change the priority of this Automation Rule.

This is done by moving it ``n`` steps,
relative to the other priority rules.
The priority from the other rules are updated too.

:param steps: Number of steps to be moved
(it can be negative)
:returns: True if the priority was changed
"""
total = self.project.automation_rules.count()
current_priority = self.priority
new_priority = (current_priority + steps) % total

if current_priority == new_priority:
return False

# Move other's priority
if new_priority > current_priority:
# It was moved down
rules = (
self.project.automation_rules
.filter(priority__gt=current_priority, priority__lte=new_priority)
# We sort the queryset in asc order
# to be updated in that order
# to avoid hitting the unique constraint (project, priority).
.order_by('priority')
)
expression = F('priority') - 1
else:
# It was moved up
rules = (
self.project.automation_rules
.filter(priority__lt=current_priority, priority__gte=new_priority)
.exclude(pk=self.pk)
# We sort the queryset in desc order
# to be updated in that order
# to avoid hitting the unique constraint (project, priority).
.order_by('-priority')
)
expression = F('priority') + 1

# Put an imposible priority to avoid
# the unique constraint (project, priority)
# while updating.
self.priority = total + 99
self.save()

# We update each object one by one to
# avoid hitting the unique constraint (project, priority).
for rule in rules:
rule.priority = expression
rule.save()
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

What happens if we hit an exception here? The rule will be stuck with a broken priority. We likely need this to happen in some kind of DB transaction to avoid bad states here.

This feels really complicated and brittle. I don't have a better suggestion currently, but I'm wondering if this concept of priority is something we need to enforce so strongly.

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

We have transactions per request

ATOMIC_REQUESTS = True

So, we need to have a unique priority in all rules of a project, that way they get sorted how the user wants.

We can remove the unique constraint as Manuel suggested, and maybe check for inconsistencies on save?

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Yea, I guess we can keep this for now, but I'm guessing it will cause issues at some point :)


# Put back new priority
self.priority = new_priority
self.save()
return True

def delete(self, *args, **kwargs): # pylint: disable=arguments-differ
"""Override method to update the other priorities after delete."""
current_priority = self.priority
project = self.project
super().delete(*args, **kwargs)

rules = (
project.automation_rules
.filter(priority__gte=current_priority)
# We sort the queryset in asc order
# to be updated in that order
# to avoid hitting the unique constraint (project, priority).
.order_by('priority')
)
# We update each object one by one to
# avoid hitting the unique constraint (project, priority).
for rule in rules:
rule.priority = F('priority') - 1
rule.save()

def get_description(self):
if self.description:
return self.description
Expand Down
256 changes: 256 additions & 0 deletions readthedocs/rtd_tests/tests/test_automation_rules.py
Original file line number Diff line number Diff line change
Expand Up @@ -183,3 +183,259 @@ def test_add_rule_regex(self):
)
assert self.project.automation_rules.count() == 4
assert rule.priority == 10


@pytest.mark.django_db
class TestAutomationRuleMove:

@pytest.fixture(autouse=True)
def setup_method(self):
self.project = get(Project)
self.rule_0 = self._add_rule('Zero')
self.rule_1 = self._add_rule('One')
self.rule_2 = self._add_rule('Two')
self.rule_3 = self._add_rule('Three')
self.rule_4 = self._add_rule('Four')
self.rule_5 = self._add_rule('Five')
assert self.project.automation_rules.count() == 6

def _add_rule(self, description):
rule = RegexAutomationRule.objects.add_rule(
project=self.project,
description=description,
match_arg='.*',
version_type=BRANCH,
action=VersionAutomationRule.ACTIVATE_VERSION_ACTION,
)
return rule

def test_move_rule_one_step(self):
self.rule_0.move(1)
new_order = [
self.rule_1,
self.rule_0,
self.rule_2,
self.rule_3,
self.rule_4,
self.rule_5,
]

for priority, rule in enumerate(self.project.automation_rules.all()):
assert rule == new_order[priority]
assert rule.priority == priority

def test_move_rule_positive_steps(self):
self.rule_1.move(1)
self.rule_1.move(2)

new_order = [
self.rule_0,
self.rule_2,
self.rule_3,
self.rule_4,
self.rule_1,
self.rule_5,
]

for priority, rule in enumerate(self.project.automation_rules.all()):
assert rule == new_order[priority]
assert rule.priority == priority

def test_move_rule_positive_steps_overflow(self):
self.rule_2.move(3)
self.rule_2.move(2)

new_order = [
self.rule_0,
self.rule_2,
self.rule_1,
self.rule_3,
self.rule_4,
self.rule_5,
]

for priority, rule in enumerate(self.project.automation_rules.all()):
assert rule == new_order[priority]
assert rule.priority == priority

def test_move_rules_positive_steps(self):
self.rule_2.move(2)
self.rule_0.refresh_from_db()
self.rule_0.move(7)
self.rule_4.refresh_from_db()
self.rule_4.move(4)
self.rule_1.refresh_from_db()
self.rule_1.move(1)

new_order = [
self.rule_4,
self.rule_1,
self.rule_0,
self.rule_3,
self.rule_2,
self.rule_5,
]

for priority, rule in enumerate(self.project.automation_rules.all()):
assert rule == new_order[priority]
assert rule.priority == priority

def test_move_rule_one_negative_step(self):
self.rule_3.move(-1)
new_order = [
self.rule_0,
self.rule_1,
self.rule_3,
self.rule_2,
self.rule_4,
self.rule_5,
]

for priority, rule in enumerate(self.project.automation_rules.all()):
assert rule == new_order[priority]
assert rule.priority == priority

def test_move_rule_negative_steps(self):
self.rule_4.move(-1)
self.rule_4.move(-2)

new_order = [
self.rule_0,
self.rule_4,
self.rule_1,
self.rule_2,
self.rule_3,
self.rule_5,
]

for priority, rule in enumerate(self.project.automation_rules.all()):
assert rule == new_order[priority]
assert rule.priority == priority

def test_move_rule_negative_steps_overflow(self):
self.rule_2.move(-3)
self.rule_2.move(-2)

new_order = [
self.rule_0,
self.rule_1,
self.rule_3,
self.rule_2,
self.rule_4,
self.rule_5,
]

for priority, rule in enumerate(self.project.automation_rules.all()):
assert rule == new_order[priority]
assert rule.priority == priority

def test_move_rules_negative_steps(self):
self.rule_2.move(-2)
self.rule_5.refresh_from_db()
self.rule_5.move(-7)
self.rule_3.refresh_from_db()
self.rule_3.move(-2)
self.rule_1.refresh_from_db()
self.rule_1.move(-1)

new_order = [
self.rule_2,
self.rule_3,
self.rule_1,
self.rule_0,
self.rule_5,
self.rule_4,
]

for priority, rule in enumerate(self.project.automation_rules.all()):
assert rule == new_order[priority]
assert rule.priority == priority

def test_move_rules_up_and_down(self):
self.rule_2.move(2)
self.rule_5.refresh_from_db()
self.rule_5.move(-3)
self.rule_3.refresh_from_db()
self.rule_3.move(4)
self.rule_1.refresh_from_db()
self.rule_1.move(-1)

new_order = [
self.rule_0,
self.rule_1,
self.rule_3,
self.rule_5,
self.rule_4,
self.rule_2,
]

for priority, rule in enumerate(self.project.automation_rules.all()):
assert rule == new_order[priority]
assert rule.priority == priority

def test_delete_fist_rule(self):
self.rule_0.delete()
assert self.project.automation_rules.all().count() == 5

new_order = [
self.rule_1,
self.rule_2,
self.rule_3,
self.rule_4,
self.rule_5,
]

for priority, rule in enumerate(self.project.automation_rules.all()):
assert rule == new_order[priority]
assert rule.priority == priority

def test_delete_last_rule(self):
self.rule_5.delete()
assert self.project.automation_rules.all().count() == 5

new_order = [
self.rule_0,
self.rule_1,
self.rule_2,
self.rule_3,
self.rule_4,
]

for priority, rule in enumerate(self.project.automation_rules.all()):
assert rule == new_order[priority]
assert rule.priority == priority

def test_delete_some_rule(self):
self.rule_2.delete()
assert self.project.automation_rules.all().count() == 5

new_order = [
self.rule_0,
self.rule_1,
self.rule_3,
self.rule_4,
self.rule_5,
]

for priority, rule in enumerate(self.project.automation_rules.all()):
assert rule == new_order[priority]
assert rule.priority == priority

def test_delete_some_rules(self):
self.rule_2.delete()
self.rule_0.refresh_from_db()
self.rule_0.delete()
self.rule_5.refresh_from_db()
self.rule_5.delete()

assert self.project.automation_rules.all().count() == 3

new_order = [
self.rule_1,
self.rule_3,
self.rule_4,
]

for priority, rule in enumerate(self.project.automation_rules.all()):
assert rule == new_order[priority]
assert rule.priority == priority