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

Grant reward points on evaluation deletion #2358

Open
wants to merge 3 commits into
base: main
Choose a base branch
from

Conversation

richardebeling
Copy link
Member

Fixes #2279

Comment on lines +132 to +138
if not inside_transaction():
# This will always be true in a django TestCase, so our tests can't meaningfully catch calls that are not
# wrapped in a transaction. Requiring a transaction is a precaution so that an (unlikely) failing .delete()
# execution afterwards doesn't leave us in half-deleted state. Chances are, if deletion fails, staff users
# will still want to delete the instance.
# Currently, only staff:evaluation_delete and staff:semester_delete call .delete()
logger.exception("Called while not inside transaction")
Copy link
Member Author

Choose a reason for hiding this comment

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

@niklasmohrin this is all a bit annoying. Ideally, I'd like to

  1. start a transaction
  2. store a list of participants
  3. delete the evaluation, including all m2m-participants
  4. create RewardPointGrantings (with transaction rollback on failure)

While the signal model allows me to run code before and after deletion, I don't see a nice way to pass the list of participants to it without having global state

I think this is fine for now. If we ever run into a "wanted to delete, failed, now need cleanup"-state, we will likely need manual cleanup anyway.

I'll look into whether we can ditch the signals and use a custom manager instead, which is what the Django docs propose to do instead:

Signals can make your code harder to maintain. Consider implementing a helper method on a custom manager, to both update your models and perform additional logic, or else overriding model methods before using model signals.

Copy link
Member

Choose a reason for hiding this comment

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

yeah, it seems a bit fragile; adding a custom manager to the Evaluation model is also a bit annoying because we have rewards logic in the model; maybe we can make our own "register a deletion handler" thing?

@@ -671,6 +674,8 @@ def semester_make_active(request):
@require_POST
@manager_required
def semester_delete(request):
# TODO(rebeling): Do we expect reward point granting here? Do we want to notify? I don't see why it couldn't happen
Copy link
Member Author

Choose a reason for hiding this comment

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

@janno42 I didn't find anything in the code that prohibits deleting semesters with active rewards. I think it's possible that we also get a reward point granting here? Do we want that? (If yes, I guess we want a matching messages.info)

Copy link
Member

Choose a reason for hiding this comment

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

Semesters can only be deleted when they have no evaluations or all evaluations have been archived. So there can't be any evaluations which should be counted towards reward points, right?

# See comment in helper_evaluation_edit
@receiver(RewardPointGranting.granted_by_evaluation_deletion, weak=True)
def notify_reward_points(grantings, **_kwargs):
logger.info(f"Deletion of evaluation {evaluation} has created {len(grantings)} reward point grantings")
Copy link
Member Author

Choose a reason for hiding this comment

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

@janno42 evaluation deletion currently just removes the row, it doesn't reload the page, so a messages.info call will only be shown on the next page (re)load. Do we want this to be shown to users, and if yes, what should it look like?

Copy link
Member

Choose a reason for hiding this comment

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

It's okay to only put it in the log file. We don't have to show it to managers.

Comment on lines +207 to +211
def test_inside_transaction(self):
self.assertEqual(inside_transaction(), False)

with transaction.atomic():
self.assertEqual(inside_transaction(), True)
Copy link
Member

Choose a reason for hiding this comment

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

Any reason to prefer this over the following?

Suggested change
def test_inside_transaction(self):
self.assertEqual(inside_transaction(), False)
with transaction.atomic():
self.assertEqual(inside_transaction(), True)
def test_inside_transaction(self):
self.assertFalse(inside_transaction())
with transaction.atomic():
self.assertTrue(inside_transaction())

# execution afterwards doesn't leave us in half-deleted state. Chances are, if deletion fails, staff users
# will still want to delete the instance.
# Currently, only staff:evaluation_delete and staff:semester_delete call .delete()
logger.exception("Called while not inside transaction")
Copy link
Member

Choose a reason for hiding this comment

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

In the docs they say we should only use logger.exception in an exception handler, so I guess it should be logger.error?

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Development

Successfully merging this pull request may close these issues.

Reward points not granted on evaluation deletion
3 participants