models.py 8.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345
  1. """
  2. Various complex queries that have been problematic in the past.
  3. """
  4. import threading
  5. from django.db import models
  6. class DumbCategory(models.Model):
  7. pass
  8. class NamedCategory(DumbCategory):
  9. name = models.CharField(max_length=10)
  10. class Tag(models.Model):
  11. name = models.CharField(max_length=10)
  12. parent = models.ForeignKey('self', blank=True, null=True,
  13. related_name='children')
  14. category = models.ForeignKey(NamedCategory, null=True, default=None)
  15. class Meta:
  16. ordering = ['name']
  17. def __unicode__(self):
  18. return self.name
  19. class Note(models.Model):
  20. note = models.CharField(max_length=100)
  21. misc = models.CharField(max_length=10)
  22. class Meta:
  23. ordering = ['note']
  24. def __unicode__(self):
  25. return self.note
  26. def __init__(self, *args, **kwargs):
  27. super(Note, self).__init__(*args, **kwargs)
  28. # Regression for #13227 -- having an attribute that
  29. # is unpickleable doesn't stop you from cloning queries
  30. # that use objects of that type as an argument.
  31. self.lock = threading.Lock()
  32. class Annotation(models.Model):
  33. name = models.CharField(max_length=10)
  34. tag = models.ForeignKey(Tag)
  35. notes = models.ManyToManyField(Note)
  36. def __unicode__(self):
  37. return self.name
  38. class ExtraInfo(models.Model):
  39. info = models.CharField(max_length=100)
  40. note = models.ForeignKey(Note)
  41. class Meta:
  42. ordering = ['info']
  43. def __unicode__(self):
  44. return self.info
  45. class Author(models.Model):
  46. name = models.CharField(max_length=10)
  47. num = models.IntegerField(unique=True)
  48. extra = models.ForeignKey(ExtraInfo)
  49. class Meta:
  50. ordering = ['name']
  51. def __unicode__(self):
  52. return self.name
  53. class Item(models.Model):
  54. name = models.CharField(max_length=10)
  55. created = models.DateTimeField()
  56. modified = models.DateTimeField(blank=True, null=True)
  57. tags = models.ManyToManyField(Tag, blank=True, null=True)
  58. creator = models.ForeignKey(Author)
  59. note = models.ForeignKey(Note)
  60. class Meta:
  61. ordering = ['-note', 'name']
  62. def __unicode__(self):
  63. return self.name
  64. class Report(models.Model):
  65. name = models.CharField(max_length=10)
  66. creator = models.ForeignKey(Author, to_field='num', null=True)
  67. def __unicode__(self):
  68. return self.name
  69. class Ranking(models.Model):
  70. rank = models.IntegerField()
  71. author = models.ForeignKey(Author)
  72. class Meta:
  73. # A complex ordering specification. Should stress the system a bit.
  74. ordering = ('author__extra__note', 'author__name', 'rank')
  75. def __unicode__(self):
  76. return '%d: %s' % (self.rank, self.author.name)
  77. class Cover(models.Model):
  78. title = models.CharField(max_length=50)
  79. item = models.ForeignKey(Item)
  80. class Meta:
  81. ordering = ['item']
  82. def __unicode__(self):
  83. return self.title
  84. class Number(models.Model):
  85. num = models.IntegerField()
  86. def __unicode__(self):
  87. return unicode(self.num)
  88. # Symmetrical m2m field with a normal field using the reverse accesor name
  89. # ("valid").
  90. class Valid(models.Model):
  91. valid = models.CharField(max_length=10)
  92. parent = models.ManyToManyField('self')
  93. class Meta:
  94. ordering = ['valid']
  95. # Some funky cross-linked models for testing a couple of infinite recursion
  96. # cases.
  97. class X(models.Model):
  98. y = models.ForeignKey('Y')
  99. class Y(models.Model):
  100. x1 = models.ForeignKey(X, related_name='y1')
  101. # Some models with a cycle in the default ordering. This would be bad if we
  102. # didn't catch the infinite loop.
  103. class LoopX(models.Model):
  104. y = models.ForeignKey('LoopY')
  105. class Meta:
  106. ordering = ['y']
  107. class LoopY(models.Model):
  108. x = models.ForeignKey(LoopX)
  109. class Meta:
  110. ordering = ['x']
  111. class LoopZ(models.Model):
  112. z = models.ForeignKey('self')
  113. class Meta:
  114. ordering = ['z']
  115. # A model and custom default manager combination.
  116. class CustomManager(models.Manager):
  117. def get_query_set(self):
  118. qs = super(CustomManager, self).get_query_set()
  119. return qs.filter(public=True, tag__name='t1')
  120. class ManagedModel(models.Model):
  121. data = models.CharField(max_length=10)
  122. tag = models.ForeignKey(Tag)
  123. public = models.BooleanField(default=True)
  124. objects = CustomManager()
  125. normal_manager = models.Manager()
  126. def __unicode__(self):
  127. return self.data
  128. # An inter-related setup with multiple paths from Child to Detail.
  129. class Detail(models.Model):
  130. data = models.CharField(max_length=10)
  131. class MemberManager(models.Manager):
  132. def get_query_set(self):
  133. return super(MemberManager, self).get_query_set().select_related("details")
  134. class Member(models.Model):
  135. name = models.CharField(max_length=10)
  136. details = models.OneToOneField(Detail, primary_key=True)
  137. objects = MemberManager()
  138. class Child(models.Model):
  139. person = models.OneToOneField(Member, primary_key=True)
  140. parent = models.ForeignKey(Member, related_name="children")
  141. # Custom primary keys interfered with ordering in the past.
  142. class CustomPk(models.Model):
  143. name = models.CharField(max_length=10, primary_key=True)
  144. extra = models.CharField(max_length=10)
  145. class Meta:
  146. ordering = ['name', 'extra']
  147. class Related(models.Model):
  148. custom = models.ForeignKey(CustomPk)
  149. # An inter-related setup with a model subclass that has a nullable
  150. # path to another model, and a return path from that model.
  151. class Celebrity(models.Model):
  152. name = models.CharField("Name", max_length=20)
  153. greatest_fan = models.ForeignKey("Fan", null=True, unique=True)
  154. class TvChef(Celebrity):
  155. pass
  156. class Fan(models.Model):
  157. fan_of = models.ForeignKey(Celebrity)
  158. # Multiple foreign keys
  159. class LeafA(models.Model):
  160. data = models.CharField(max_length=10)
  161. def __unicode__(self):
  162. return self.data
  163. class LeafB(models.Model):
  164. data = models.CharField(max_length=10)
  165. class Join(models.Model):
  166. a = models.ForeignKey(LeafA)
  167. b = models.ForeignKey(LeafB)
  168. class ReservedName(models.Model):
  169. name = models.CharField(max_length=20)
  170. order = models.IntegerField()
  171. def __unicode__(self):
  172. return self.name
  173. # A simpler shared-foreign-key setup that can expose some problems.
  174. class SharedConnection(models.Model):
  175. data = models.CharField(max_length=10)
  176. class PointerA(models.Model):
  177. connection = models.ForeignKey(SharedConnection)
  178. class PointerB(models.Model):
  179. connection = models.ForeignKey(SharedConnection)
  180. # Multi-layer ordering
  181. class SingleObject(models.Model):
  182. name = models.CharField(max_length=10)
  183. class Meta:
  184. ordering = ['name']
  185. def __unicode__(self):
  186. return self.name
  187. class RelatedObject(models.Model):
  188. single = models.ForeignKey(SingleObject)
  189. class Meta:
  190. ordering = ['single']
  191. class Plaything(models.Model):
  192. name = models.CharField(max_length=10)
  193. others = models.ForeignKey(RelatedObject, null=True)
  194. class Meta:
  195. ordering = ['others']
  196. def __unicode__(self):
  197. return self.name
  198. class Article(models.Model):
  199. name = models.CharField(max_length=20)
  200. created = models.DateTimeField()
  201. class Food(models.Model):
  202. name = models.CharField(max_length=20, unique=True)
  203. def __unicode__(self):
  204. return self.name
  205. class Eaten(models.Model):
  206. food = models.ForeignKey(Food, to_field="name")
  207. meal = models.CharField(max_length=20)
  208. def __unicode__(self):
  209. return u"%s at %s" % (self.food, self.meal)
  210. class Node(models.Model):
  211. num = models.IntegerField(unique=True)
  212. parent = models.ForeignKey("self", to_field="num", null=True)
  213. def __unicode__(self):
  214. return u"%s" % self.num
  215. # Bug #12252
  216. class ObjectA(models.Model):
  217. name = models.CharField(max_length=50)
  218. def __unicode__(self):
  219. return self.name
  220. class ObjectB(models.Model):
  221. name = models.CharField(max_length=50)
  222. objecta = models.ForeignKey(ObjectA)
  223. num = models.PositiveSmallIntegerField()
  224. def __unicode__(self):
  225. return self.name
  226. class ObjectC(models.Model):
  227. name = models.CharField(max_length=50)
  228. objecta = models.ForeignKey(ObjectA)
  229. objectb = models.ForeignKey(ObjectB)
  230. def __unicode__(self):
  231. return self.name
  232. class SimpleCategory(models.Model):
  233. name = models.CharField(max_length=15)
  234. def __unicode__(self):
  235. return self.name
  236. class SpecialCategory(SimpleCategory):
  237. special_name = models.CharField(max_length=15)
  238. def __unicode__(self):
  239. return self.name + " " + self.special_name
  240. class CategoryItem(models.Model):
  241. category = models.ForeignKey(SimpleCategory)
  242. def __unicode__(self):
  243. return "category item: " + str(self.category)
  244. class OneToOneCategory(models.Model):
  245. new_name = models.CharField(max_length=15)
  246. category = models.OneToOneField(SimpleCategory)
  247. def __unicode__(self):
  248. return "one2one " + self.new_name