make user and group management more robust
- remove TaskResultInline and subclasses - add custom perform_delete_selected action to UserAdmin and GroupAdmin - properly clean asynchronous tasks in rabbitmq - wrap user operations in transactions
This commit is contained in:
parent
7bce88e18a
commit
e877087127
2 changed files with 91 additions and 43 deletions
|
@ -25,30 +25,6 @@ class ShadowInline(admin.TabularInline):
|
||||||
can_delete = False
|
can_delete = False
|
||||||
|
|
||||||
|
|
||||||
class TaskResultInline(admin.TabularInline):
|
|
||||||
can_delete = False
|
|
||||||
extra = 0
|
|
||||||
readonly_fields = ['task_uuid', 'task_name', 'is_finished', 'is_success',
|
|
||||||
'state', 'result_body']
|
|
||||||
|
|
||||||
def get_queryset(self, request):
|
|
||||||
qs = super(TaskResultInline, self).get_queryset(request)
|
|
||||||
for entry in qs:
|
|
||||||
entry.update_taskstatus()
|
|
||||||
return qs
|
|
||||||
|
|
||||||
def has_add_permission(self, request, obj=None):
|
|
||||||
return False
|
|
||||||
|
|
||||||
|
|
||||||
class UserTaskResultInline(TaskResultInline):
|
|
||||||
model = UserTaskResult
|
|
||||||
|
|
||||||
|
|
||||||
class GroupTaskResultInline(TaskResultInline):
|
|
||||||
model = GroupTaskResult
|
|
||||||
|
|
||||||
|
|
||||||
class UserCreationForm(forms.ModelForm):
|
class UserCreationForm(forms.ModelForm):
|
||||||
"""
|
"""
|
||||||
A form for creating system users.
|
A form for creating system users.
|
||||||
|
@ -91,9 +67,10 @@ class UserCreationForm(forms.ModelForm):
|
||||||
|
|
||||||
|
|
||||||
class UserAdmin(admin.ModelAdmin):
|
class UserAdmin(admin.ModelAdmin):
|
||||||
inlines = [AdditionalGroupInline, ShadowInline, UserTaskResultInline]
|
actions = ['perform_delete_selected']
|
||||||
readonly_fields = ['uid']
|
readonly_fields = ['uid']
|
||||||
add_form = UserCreationForm
|
add_form = UserCreationForm
|
||||||
|
inlines = [AdditionalGroupInline, ShadowInline]
|
||||||
|
|
||||||
add_fieldsets = (
|
add_fieldsets = (
|
||||||
(None, {
|
(None, {
|
||||||
|
@ -115,37 +92,82 @@ class UserAdmin(admin.ModelAdmin):
|
||||||
defaults.update(kwargs)
|
defaults.update(kwargs)
|
||||||
return super(UserAdmin, self).get_form(request, obj, **defaults)
|
return super(UserAdmin, self).get_form(request, obj, **defaults)
|
||||||
|
|
||||||
def get_inline_instances(self, request, obj=None):
|
def get_readonly_fields(self, request, obj=None):
|
||||||
if obj is None:
|
if obj:
|
||||||
|
return ['uid']
|
||||||
return []
|
return []
|
||||||
return super(UserAdmin, self).get_inline_instances(request, obj)
|
|
||||||
|
def perform_delete_selected(self, request, queryset):
|
||||||
|
for user in queryset.all():
|
||||||
|
user.delete()
|
||||||
|
perform_delete_selected.short_description = _('Delete selected users')
|
||||||
|
|
||||||
|
def get_actions(self, request):
|
||||||
|
actions = super(UserAdmin, self).get_actions(request)
|
||||||
|
if 'delete_selected' in actions:
|
||||||
|
del actions['delete_selected']
|
||||||
|
return actions
|
||||||
|
|
||||||
|
|
||||||
class GroupAdmin(admin.ModelAdmin):
|
class GroupAdmin(admin.ModelAdmin):
|
||||||
inlines = [GroupTaskResultInline]
|
actions = ['perform_delete_selected']
|
||||||
|
|
||||||
def get_inline_instances(self, request, obj=None):
|
def get_inline_instances(self, request, obj=None):
|
||||||
if obj is None:
|
if obj is None:
|
||||||
return []
|
return []
|
||||||
return super(GroupAdmin, self).get_inline_instances(request, obj)
|
return super(GroupAdmin, self).get_inline_instances(request, obj)
|
||||||
|
|
||||||
|
def perform_delete_selected(self, request, queryset):
|
||||||
|
for group in queryset.all():
|
||||||
|
group.delete()
|
||||||
|
perform_delete_selected.short_description = _('Delete selected groups')
|
||||||
|
|
||||||
class DeleteTaskResultAdmin(admin.ModelAdmin):
|
def get_actions(self, request):
|
||||||
readonly_fields = ['task_uuid', 'task_name', 'modeltype', 'modelname',
|
actions = super(GroupAdmin, self).get_actions(request)
|
||||||
'is_finished', 'is_success', 'state', 'result_body']
|
if 'delete_selected' in actions:
|
||||||
list_display = ('task_uuid', 'task_name', 'modeltype', 'modelname',
|
del actions['delete_selected']
|
||||||
'is_finished', 'state')
|
return actions
|
||||||
|
|
||||||
|
|
||||||
|
class TaskResultAdmin(admin.ModelAdmin):
|
||||||
def has_add_permission(self, request, obj=None):
|
def has_add_permission(self, request, obj=None):
|
||||||
return False
|
return False
|
||||||
|
|
||||||
|
def has_delete_permission(self, request, obj=None):
|
||||||
|
return obj is None or obj.is_finished
|
||||||
|
|
||||||
def get_queryset(self, request):
|
def get_queryset(self, request):
|
||||||
qs = super(DeleteTaskResultAdmin, self).get_queryset(request)
|
qs = super(TaskResultAdmin, self).get_queryset(request)
|
||||||
for entry in qs:
|
for entry in qs:
|
||||||
entry.update_taskstatus()
|
entry.update_taskstatus()
|
||||||
return qs
|
return qs
|
||||||
|
|
||||||
|
|
||||||
|
class DeleteTaskResultAdmin(TaskResultAdmin):
|
||||||
|
readonly_fields = ['task_uuid', 'task_name', 'modeltype', 'modelname',
|
||||||
|
'is_finished', 'is_success', 'state', 'result_body']
|
||||||
|
list_display = ('task_uuid', 'task_name', 'modeltype', 'modelname',
|
||||||
|
'is_finished', 'state')
|
||||||
|
|
||||||
|
|
||||||
|
class GroupTaskResultAdmin(TaskResultAdmin):
|
||||||
|
readonly_fields = [
|
||||||
|
'task_uuid', 'task_name', 'group', 'is_finished', 'is_success',
|
||||||
|
'state', 'result_body'
|
||||||
|
]
|
||||||
|
list_display = ('task_uuid', 'task_name', 'group', 'is_finished', 'state')
|
||||||
|
|
||||||
|
|
||||||
|
class UserTaskResultAdmin(TaskResultAdmin):
|
||||||
|
readonly_fields = [
|
||||||
|
'task_uuid', 'task_name', 'user', 'is_finished', 'is_success', 'state',
|
||||||
|
'result_body'
|
||||||
|
]
|
||||||
|
list_display = ('task_uuid', 'task_name', 'user', 'is_finished', 'state')
|
||||||
|
|
||||||
|
|
||||||
admin.site.register(Group, GroupAdmin)
|
admin.site.register(Group, GroupAdmin)
|
||||||
admin.site.register(User, UserAdmin)
|
admin.site.register(User, UserAdmin)
|
||||||
admin.site.register(DeleteTaskResult, DeleteTaskResultAdmin)
|
admin.site.register(DeleteTaskResult, DeleteTaskResultAdmin)
|
||||||
|
admin.site.register(GroupTaskResult, GroupTaskResultAdmin)
|
||||||
|
admin.site.register(UserTaskResult, UserTaskResultAdmin)
|
||||||
|
|
|
@ -1,3 +1,5 @@
|
||||||
|
from __future__ import unicode_literals
|
||||||
|
|
||||||
from datetime import date
|
from datetime import date
|
||||||
import os
|
import os
|
||||||
|
|
||||||
|
@ -42,15 +44,14 @@ class TaskResult(TimeStampedModel, models.Model):
|
||||||
abstract = True
|
abstract = True
|
||||||
|
|
||||||
def _set_result_fields(self, asyncresult):
|
def _set_result_fields(self, asyncresult):
|
||||||
if asyncresult.ready():
|
if not self.is_finished:
|
||||||
|
result = asyncresult.get(no_ack=False)
|
||||||
self.is_finished = True
|
self.is_finished = True
|
||||||
self.is_success = asyncresult.state == 'SUCCESS'
|
self.is_success = asyncresult.state == 'SUCCESS'
|
||||||
self.result_body = str(asyncresult.result)
|
|
||||||
self.state = asyncresult.state
|
self.state = asyncresult.state
|
||||||
asyncresult.get(no_ack=False)
|
self.result_body = str(result)
|
||||||
|
|
||||||
def update_taskstatus(self):
|
def update_taskstatus(self):
|
||||||
if not self.is_finished:
|
|
||||||
asyncresult = AsyncResult(self.task_uuid)
|
asyncresult = AsyncResult(self.task_uuid)
|
||||||
self._set_result_fields(asyncresult)
|
self._set_result_fields(asyncresult)
|
||||||
self.save()
|
self.save()
|
||||||
|
@ -84,6 +85,7 @@ class Group(TimeStampedModel, models.Model):
|
||||||
def __str__(self):
|
def __str__(self):
|
||||||
return '{0} ({1})'.format(self.groupname, self.gid)
|
return '{0} ({1})'.format(self.groupname, self.gid)
|
||||||
|
|
||||||
|
@transaction.atomic
|
||||||
def save(self, *args, **kwargs):
|
def save(self, *args, **kwargs):
|
||||||
super(Group, self).save(*args, **kwargs)
|
super(Group, self).save(*args, **kwargs)
|
||||||
GroupTaskResult.objects.create_grouptaskresult(
|
GroupTaskResult.objects.create_grouptaskresult(
|
||||||
|
@ -93,6 +95,7 @@ class Group(TimeStampedModel, models.Model):
|
||||||
)
|
)
|
||||||
return self
|
return self
|
||||||
|
|
||||||
|
@transaction.atomic
|
||||||
def delete(self, *args, **kwargs):
|
def delete(self, *args, **kwargs):
|
||||||
DeleteTaskResult.objects.create_deletetaskresult(
|
DeleteTaskResult.objects.create_deletetaskresult(
|
||||||
'group', self.groupname,
|
'group', self.groupname,
|
||||||
|
@ -125,6 +128,7 @@ class DeleteTaskResultManager(TaskResultManager):
|
||||||
return taskresult
|
return taskresult
|
||||||
|
|
||||||
|
|
||||||
|
@python_2_unicode_compatible
|
||||||
class DeleteTaskResult(TaskResult):
|
class DeleteTaskResult(TaskResult):
|
||||||
|
|
||||||
modeltype = models.CharField(max_length=20, db_index=True)
|
modeltype = models.CharField(max_length=20, db_index=True)
|
||||||
|
@ -132,6 +136,11 @@ class DeleteTaskResult(TaskResult):
|
||||||
|
|
||||||
objects = DeleteTaskResultManager()
|
objects = DeleteTaskResultManager()
|
||||||
|
|
||||||
|
def __str__(self):
|
||||||
|
return "{task_uuid} {task_name} {modeltype} {modelname}".format(
|
||||||
|
task_uuid=self.task_uuid, task_name=self.task_name,
|
||||||
|
modeltype=self.modeltype, modelname=self.modelname)
|
||||||
|
|
||||||
|
|
||||||
class GroupTaskResultManager(TaskResultManager):
|
class GroupTaskResultManager(TaskResultManager):
|
||||||
|
|
||||||
|
@ -145,12 +154,18 @@ class GroupTaskResultManager(TaskResultManager):
|
||||||
return taskresult
|
return taskresult
|
||||||
|
|
||||||
|
|
||||||
|
@python_2_unicode_compatible
|
||||||
class GroupTaskResult(TaskResult):
|
class GroupTaskResult(TaskResult):
|
||||||
|
|
||||||
group = models.ForeignKey(Group)
|
group = models.ForeignKey(Group)
|
||||||
|
|
||||||
objects = GroupTaskResultManager()
|
objects = GroupTaskResultManager()
|
||||||
|
|
||||||
|
def __str__(self):
|
||||||
|
return "{task_uuid} {task_name} {group}".format(
|
||||||
|
task_uuid=self.task_uuid, task_name=self.task_name,
|
||||||
|
group=self.group)
|
||||||
|
|
||||||
|
|
||||||
class UserManager(models.Manager):
|
class UserManager(models.Manager):
|
||||||
|
|
||||||
|
@ -215,6 +230,7 @@ class User(TimeStampedModel, models.Model):
|
||||||
def __str__(self):
|
def __str__(self):
|
||||||
return '{0} ({1})'.format(self.username, self.uid)
|
return '{0} ({1})'.format(self.username, self.uid)
|
||||||
|
|
||||||
|
@transaction.atomic
|
||||||
def set_password(self, password):
|
def set_password(self, password):
|
||||||
if hasattr(self, 'shadow'):
|
if hasattr(self, 'shadow'):
|
||||||
self.shadow.set_password(password)
|
self.shadow.set_password(password)
|
||||||
|
@ -232,6 +248,7 @@ class User(TimeStampedModel, models.Model):
|
||||||
commit=True
|
commit=True
|
||||||
)
|
)
|
||||||
|
|
||||||
|
@transaction.atomic
|
||||||
def save(self, *args, **kwargs):
|
def save(self, *args, **kwargs):
|
||||||
UserTaskResult.objects.create_usertaskresult(
|
UserTaskResult.objects.create_usertaskresult(
|
||||||
self,
|
self,
|
||||||
|
@ -243,6 +260,7 @@ class User(TimeStampedModel, models.Model):
|
||||||
)
|
)
|
||||||
return super(User, self).save(*args, **kwargs)
|
return super(User, self).save(*args, **kwargs)
|
||||||
|
|
||||||
|
@transaction.atomic
|
||||||
def delete(self, *args, **kwargs):
|
def delete(self, *args, **kwargs):
|
||||||
for group in [
|
for group in [
|
||||||
ag.group for ag in AdditionalGroup.objects.filter(user=self)
|
ag.group for ag in AdditionalGroup.objects.filter(user=self)
|
||||||
|
@ -274,15 +292,21 @@ class UserTaskResultManager(TaskResultManager):
|
||||||
return taskresult
|
return taskresult
|
||||||
|
|
||||||
|
|
||||||
|
@python_2_unicode_compatible
|
||||||
class UserTaskResult(TaskResult):
|
class UserTaskResult(TaskResult):
|
||||||
|
|
||||||
user = models.ForeignKey(User)
|
user = models.ForeignKey(User)
|
||||||
|
|
||||||
objects = UserTaskResultManager()
|
objects = UserTaskResultManager()
|
||||||
|
|
||||||
|
def __str__(self):
|
||||||
|
return "{task_uuid} {task_name} {user}".format(
|
||||||
|
task_uuid=self.task_uuid, task_name=self.task_name, user=self.user)
|
||||||
|
|
||||||
|
|
||||||
class ShadowManager(models.Manager):
|
class ShadowManager(models.Manager):
|
||||||
|
|
||||||
|
@transaction.atomic
|
||||||
def create_shadow(self, user, password):
|
def create_shadow(self, user, password):
|
||||||
changedays = (timezone.now().date() - date(1970, 1, 1)).days
|
changedays = (timezone.now().date() - date(1970, 1, 1)).days
|
||||||
shadow = self.create(
|
shadow = self.create(
|
||||||
|
@ -356,6 +380,7 @@ class AdditionalGroup(TimeStampedModel, models.Model):
|
||||||
if self.user.group == self.group:
|
if self.user.group == self.group:
|
||||||
raise ValidationError(CANNOT_USE_PRIMARY_GROUP_AS_ADDITIONAL)
|
raise ValidationError(CANNOT_USE_PRIMARY_GROUP_AS_ADDITIONAL)
|
||||||
|
|
||||||
|
@transaction.atomic
|
||||||
def save(self, *args, **kwargs):
|
def save(self, *args, **kwargs):
|
||||||
GroupTaskResult.objects.create_grouptaskresult(
|
GroupTaskResult.objects.create_grouptaskresult(
|
||||||
self.group,
|
self.group,
|
||||||
|
@ -365,6 +390,7 @@ class AdditionalGroup(TimeStampedModel, models.Model):
|
||||||
)
|
)
|
||||||
super(AdditionalGroup, self).save(*args, **kwargs)
|
super(AdditionalGroup, self).save(*args, **kwargs)
|
||||||
|
|
||||||
|
@transaction.atomic
|
||||||
def delete(self, *args, **kwargs):
|
def delete(self, *args, **kwargs):
|
||||||
DeleteTaskResult.objects.create_deletetaskresult(
|
DeleteTaskResult.objects.create_deletetaskresult(
|
||||||
'usergroup',
|
'usergroup',
|
||||||
|
|
Loading…
Reference in a new issue