PageRenderTime 24ms CodeModel.GetById 27ms RepoModel.GetById 0ms app.codeStats 0ms

/src/reversion/models.py

https://github.com/etienned/django-reversion
Python | 299 lines | 167 code | 49 blank | 83 comment | 29 complexity | 84930c2fe66c2376ce688fb5294b2208 MD5 | raw file
Possible License(s): BSD-3-Clause
  1. """Database models used by django-reversion."""
  2. import warnings
  3. from django.contrib.auth.models import User
  4. from django.contrib.contenttypes.models import ContentType
  5. from django.contrib.contenttypes import generic
  6. from django.core import serializers
  7. from django.core.exceptions import ObjectDoesNotExist
  8. from django.conf import settings
  9. from django.db import models, IntegrityError
  10. def deprecated(original, replacement):
  11. """Decorator that defines a deprecated method."""
  12. def decorator(func):
  13. if not settings.DEBUG:
  14. return func
  15. def do_pending_deprication(*args, **kwargs):
  16. warnings.warn(
  17. "%s is deprecated, and will be removed in django-reversion 1.7. Use %s instead" % (original, replacement),
  18. PendingDeprecationWarning,
  19. )
  20. return func(*args, **kwargs)
  21. return do_pending_deprication
  22. return decorator
  23. def safe_revert(versions):
  24. """
  25. Attempts to revert the given models contained in the give versions.
  26. This method will attempt to resolve dependencies between the versions to revert
  27. them in the correct order to avoid database integrity errors.
  28. """
  29. unreverted_versions = []
  30. for version in versions:
  31. try:
  32. version.revert()
  33. except (IntegrityError, ObjectDoesNotExist):
  34. unreverted_versions.append(version)
  35. if len(unreverted_versions) == len(versions):
  36. raise RevertError("Could not revert revision, due to database integrity errors.")
  37. if unreverted_versions:
  38. safe_revert(unreverted_versions)
  39. class RevertError(Exception):
  40. """Exception thrown when something goes wrong with reverting a model."""
  41. class Revision(models.Model):
  42. """A group of related object versions."""
  43. manager_slug = models.CharField(
  44. max_length = 200,
  45. db_index = True,
  46. default = "default",
  47. )
  48. date_created = models.DateTimeField(auto_now_add=True,
  49. help_text="The date and time this revision was created.")
  50. user = models.ForeignKey(User,
  51. blank=True,
  52. null=True,
  53. help_text="The user who created this revision.")
  54. comment = models.TextField(blank=True,
  55. help_text="A text comment on this revision.")
  56. def revert(self, delete=False):
  57. """Reverts all objects in this revision."""
  58. version_set = self.version_set.all()
  59. # Optionally delete objects no longer in the current revision.
  60. if delete:
  61. # Get a dict of all objects in this revision.
  62. old_revision = {}
  63. for version in version_set:
  64. try:
  65. obj = version.object
  66. except ContentType.objects.get_for_id(version.content_type_id).model_class().DoesNotExist:
  67. pass
  68. else:
  69. old_revision[obj] = version
  70. # Calculate the set of all objects that are in the revision now.
  71. from reversion.revisions import RevisionManager
  72. current_revision = RevisionManager.get_manager(self.manager_slug)._follow_relationships(old_revision.keys())
  73. # Delete objects that are no longer in the current revision.
  74. for item in current_revision:
  75. if item in old_revision:
  76. if old_revision[item].type == VERSION_DELETE:
  77. item.delete()
  78. else:
  79. item.delete()
  80. # Attempt to revert all revisions.
  81. safe_revert([version for version in version_set if version.type != VERSION_DELETE])
  82. def __unicode__(self):
  83. """Returns a unicode representation."""
  84. return u", ".join(unicode(version) for version in self.version_set.all())
  85. # Version types.
  86. VERSION_ADD = 0
  87. VERSION_CHANGE = 1
  88. VERSION_DELETE = 2
  89. VERSION_TYPE_CHOICES = (
  90. (VERSION_ADD, "Addition"),
  91. (VERSION_CHANGE, "Change"),
  92. (VERSION_DELETE, "Deletion"),
  93. )
  94. def has_int_pk(model):
  95. """Tests whether the given model has an integer primary key."""
  96. pk = model._meta.pk
  97. return (
  98. (
  99. isinstance(pk, (models.IntegerField, models.AutoField)) and
  100. not isinstance(pk, models.BigIntegerField)
  101. ) or (
  102. isinstance(pk, models.ForeignKey) and has_int_pk(pk.rel.to)
  103. )
  104. )
  105. class VersionManager(models.Manager):
  106. """Manager for Version models."""
  107. @deprecated("Version.objects.get_for_object_reference()", "reversion.get_for_object_reference()")
  108. def get_for_object_reference(self, model, object_id):
  109. """
  110. Returns all versions for the given object reference.
  111. This method was deprecated in django-reversion 1.5, and will be removed in django-reversion 1.7.
  112. New applications should use reversion.get_for_object_reference(). The new version of this method
  113. returns results ordered with the most recent versions first. This legacy version of the method
  114. continues to return the results ordered with the oldest versions first.
  115. """
  116. from reversion.revisions import default_revision_manager
  117. return default_revision_manager.get_for_object_reference(model, object_id).order_by("pk")
  118. @deprecated("Version.objects.get_for_object()", "reversion.get_for_object()")
  119. def get_for_object(self, object):
  120. """
  121. Returns all the versions of the given object, ordered by date created.
  122. This method was deprecated in django-reversion 1.5, and will be removed in django-reversion 1.7.
  123. New applications should use reversion.get_for_object(). The new version of this method
  124. returns results ordered with the most recent versions first. This legacy version of the method
  125. continues to return the results ordered with the oldest versions first.
  126. """
  127. from reversion.revisions import default_revision_manager
  128. return default_revision_manager.get_for_object(object).order_by("pk")
  129. @deprecated("Version.objects.get_unique_for_object()", "reversion.get_unique_for_object()")
  130. def get_unique_for_object(self, obj):
  131. """
  132. Returns unique versions associated with the object.
  133. This method was deprecated in django-reversion 1.5, and will be removed in django-reversion 1.7.
  134. New applications should use reversion.get_unique_for_object(). The new version of this method
  135. returns results ordered with the most recent versions first. This legacy version of the method
  136. continues to return the results ordered with the oldest versions first.
  137. """
  138. from reversion.revisions import default_revision_manager
  139. versions = default_revision_manager.get_unique_for_object(obj)
  140. versions.reverse()
  141. return versions
  142. @deprecated("Version.objects.get_for_date()", "reversion.get_for_date()")
  143. def get_for_date(self, object, date):
  144. """
  145. Returns the latest version of an object for the given date.
  146. This method was deprecated in django-reversion 1.5, and will be removed in django-reversion 1.7.
  147. New applications should use reversion.get_for_date().
  148. """
  149. from reversion.revisions import default_revision_manager
  150. return default_revision_manager.get_for_date(object, date)
  151. @deprecated("Version.objects.get_deleted_object()", "reversion.get_for_object_reference()[0]")
  152. def get_deleted_object(self, model_class, object_id, select_related=None):
  153. """
  154. Returns the version corresponding to the deletion of the object with
  155. the given id.
  156. This method was deprecated in django-reversion 1.5, and will be removed in django-reversion 1.7.
  157. New applications should use reversion.get_for_date()[0].
  158. """
  159. from reversion.revisions import default_revision_manager
  160. return default_revision_manager.get_for_object_reference(model_class, object_id)[0]
  161. @deprecated("Version.objects.get_deleted()", "reversion.get_deleted()")
  162. def get_deleted(self, model_class, select_related=None):
  163. """
  164. Returns all the deleted versions for the given model class.
  165. This method was deprecated in django-reversion 1.5, and will be removed in django-reversion 1.7.
  166. New applications should use reversion.get_deleted(). The new version of this method
  167. returns results ordered with the most recent versions first. This legacy version of the method
  168. continues to return the results ordered with the oldest versions first.
  169. """
  170. from reversion.revisions import default_revision_manager
  171. return list(default_revision_manager.get_deleted(model_class).order_by("pk"))
  172. class Version(models.Model):
  173. """A saved version of a database model."""
  174. objects = VersionManager()
  175. revision = models.ForeignKey(Revision,
  176. help_text="The revision that contains this version.")
  177. object_id = models.TextField(help_text="Primary key of the model under version control.")
  178. object_id_int = models.IntegerField(
  179. blank = True,
  180. null = True,
  181. db_index = True,
  182. help_text = "An indexed, integer version of the stored model's primary key, used for faster lookups.",
  183. )
  184. content_type = models.ForeignKey(ContentType,
  185. help_text="Content type of the model under version control.")
  186. # A link to the current instance, not the version stored in this Version!
  187. object = generic.GenericForeignKey()
  188. format = models.CharField(max_length=255,
  189. help_text="The serialization format used by this model.")
  190. serialized_data = models.TextField(help_text="The serialized form of this version of the model.")
  191. object_repr = models.TextField(help_text="A string representation of the object.")
  192. @property
  193. def object_version(self):
  194. """The stored version of the model."""
  195. data = self.serialized_data
  196. if isinstance(data, unicode):
  197. data = data.encode("utf8")
  198. return list(serializers.deserialize(self.format, data))[0]
  199. type = models.PositiveSmallIntegerField(choices=VERSION_TYPE_CHOICES, db_index=True)
  200. @property
  201. def field_dict(self):
  202. """
  203. A dictionary mapping field names to field values in this version
  204. of the model.
  205. This method will follow parent links, if present.
  206. """
  207. if not hasattr(self, "_field_dict_cache"):
  208. object_version = self.object_version
  209. obj = object_version.object
  210. result = {}
  211. for field in obj._meta.fields:
  212. result[field.name] = field.value_from_object(obj)
  213. result.update(object_version.m2m_data)
  214. # Add parent data.
  215. for parent_class, field in obj._meta.parents.items():
  216. content_type = ContentType.objects.get_for_model(parent_class)
  217. if field:
  218. parent_id = unicode(getattr(obj, field.attname))
  219. else:
  220. parent_id = obj.pk
  221. try:
  222. parent_version = Version.objects.get(revision__id=self.revision_id,
  223. content_type=content_type,
  224. object_id=parent_id)
  225. except Version.DoesNotExist:
  226. pass
  227. else:
  228. result.update(parent_version.field_dict)
  229. setattr(self, "_field_dict_cache", result)
  230. return getattr(self, "_field_dict_cache")
  231. def revert(self):
  232. """Recovers the model in this version."""
  233. self.object_version.save()
  234. def __unicode__(self):
  235. """Returns a unicode representation."""
  236. return self.object_repr