models.py 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462
  1. import datetime
  2. import os
  3. import tempfile
  4. import uuid
  5. from django.core import validators
  6. from django.core.exceptions import ValidationError
  7. from django.core.files.storage import FileSystemStorage
  8. from django.db import models
  9. temp_storage_dir = tempfile.mkdtemp()
  10. temp_storage = FileSystemStorage(temp_storage_dir)
  11. class Person(models.Model):
  12. name = models.CharField(max_length=100)
  13. class Category(models.Model):
  14. name = models.CharField(max_length=20)
  15. slug = models.SlugField(max_length=20)
  16. url = models.CharField('The URL', max_length=40)
  17. def __str__(self):
  18. return self.name
  19. def __repr__(self):
  20. return self.__str__()
  21. class Writer(models.Model):
  22. name = models.CharField(max_length=50, help_text='Use both first and last names.')
  23. class Meta:
  24. ordering = ('name',)
  25. def __str__(self):
  26. return self.name
  27. class Article(models.Model):
  28. ARTICLE_STATUS = (
  29. (1, 'Draft'),
  30. (2, 'Pending'),
  31. (3, 'Live'),
  32. )
  33. headline = models.CharField(max_length=50)
  34. slug = models.SlugField()
  35. pub_date = models.DateField()
  36. created = models.DateField(editable=False)
  37. writer = models.ForeignKey(Writer, models.CASCADE)
  38. article = models.TextField()
  39. categories = models.ManyToManyField(Category, blank=True)
  40. status = models.PositiveIntegerField(choices=ARTICLE_STATUS, blank=True, null=True)
  41. def save(self, *args, **kwargs):
  42. if not self.id:
  43. self.created = datetime.date.today()
  44. return super().save(*args, **kwargs)
  45. def __str__(self):
  46. return self.headline
  47. class ImprovedArticle(models.Model):
  48. article = models.OneToOneField(Article, models.CASCADE)
  49. class ImprovedArticleWithParentLink(models.Model):
  50. article = models.OneToOneField(Article, models.CASCADE, parent_link=True)
  51. class BetterWriter(Writer):
  52. score = models.IntegerField()
  53. class Publication(models.Model):
  54. title = models.CharField(max_length=30)
  55. date_published = models.DateField()
  56. def __str__(self):
  57. return self.title
  58. def default_mode():
  59. return 'di'
  60. def default_category():
  61. return 3
  62. class PublicationDefaults(models.Model):
  63. MODE_CHOICES = (('di', 'direct'), ('de', 'delayed'))
  64. CATEGORY_CHOICES = ((1, 'Games'), (2, 'Comics'), (3, 'Novel'))
  65. title = models.CharField(max_length=30)
  66. date_published = models.DateField(default=datetime.date.today)
  67. datetime_published = models.DateTimeField(default=datetime.datetime(2000, 1, 1))
  68. mode = models.CharField(max_length=2, choices=MODE_CHOICES, default=default_mode)
  69. category = models.IntegerField(choices=CATEGORY_CHOICES, default=default_category)
  70. active = models.BooleanField(default=True)
  71. file = models.FileField(default='default.txt')
  72. class Author(models.Model):
  73. publication = models.OneToOneField(Publication, models.SET_NULL, null=True, blank=True)
  74. full_name = models.CharField(max_length=255)
  75. class Author1(models.Model):
  76. publication = models.OneToOneField(Publication, models.CASCADE, null=False)
  77. full_name = models.CharField(max_length=255)
  78. class WriterProfile(models.Model):
  79. writer = models.OneToOneField(Writer, models.CASCADE, primary_key=True)
  80. age = models.PositiveIntegerField()
  81. def __str__(self):
  82. return "%s is %s" % (self.writer, self.age)
  83. class Document(models.Model):
  84. myfile = models.FileField(upload_to='unused', blank=True)
  85. class TextFile(models.Model):
  86. description = models.CharField(max_length=20)
  87. file = models.FileField(storage=temp_storage, upload_to='tests', max_length=15)
  88. def __str__(self):
  89. return self.description
  90. class CustomFileField(models.FileField):
  91. def save_form_data(self, instance, data):
  92. been_here = getattr(self, 'been_saved', False)
  93. assert not been_here, "save_form_data called more than once"
  94. setattr(self, 'been_saved', True)
  95. class CustomFF(models.Model):
  96. f = CustomFileField(upload_to='unused', blank=True)
  97. class FilePathModel(models.Model):
  98. path = models.FilePathField(path=os.path.dirname(__file__), match='models.py', blank=True)
  99. try:
  100. from PIL import Image # NOQA: detect if Pillow is installed
  101. test_images = True
  102. class ImageFile(models.Model):
  103. def custom_upload_path(self, filename):
  104. path = self.path or 'tests'
  105. return '%s/%s' % (path, filename)
  106. description = models.CharField(max_length=20)
  107. # Deliberately put the image field *after* the width/height fields to
  108. # trigger the bug in #10404 with width/height not getting assigned.
  109. width = models.IntegerField(editable=False)
  110. height = models.IntegerField(editable=False)
  111. image = models.ImageField(storage=temp_storage, upload_to=custom_upload_path,
  112. width_field='width', height_field='height')
  113. path = models.CharField(max_length=16, blank=True, default='')
  114. def __str__(self):
  115. return self.description
  116. class OptionalImageFile(models.Model):
  117. def custom_upload_path(self, filename):
  118. path = self.path or 'tests'
  119. return '%s/%s' % (path, filename)
  120. description = models.CharField(max_length=20)
  121. image = models.ImageField(storage=temp_storage, upload_to=custom_upload_path,
  122. width_field='width', height_field='height',
  123. blank=True, null=True)
  124. width = models.IntegerField(editable=False, null=True)
  125. height = models.IntegerField(editable=False, null=True)
  126. path = models.CharField(max_length=16, blank=True, default='')
  127. def __str__(self):
  128. return self.description
  129. class NoExtensionImageFile(models.Model):
  130. def upload_to(self, filename):
  131. return 'tests/no_extension'
  132. description = models.CharField(max_length=20)
  133. image = models.ImageField(storage=temp_storage, upload_to=upload_to)
  134. def __str__(self):
  135. return self.description
  136. except ImportError:
  137. test_images = False
  138. class Homepage(models.Model):
  139. url = models.URLField()
  140. class Product(models.Model):
  141. slug = models.SlugField(unique=True)
  142. def __str__(self):
  143. return self.slug
  144. class Price(models.Model):
  145. price = models.DecimalField(max_digits=10, decimal_places=2)
  146. quantity = models.PositiveIntegerField()
  147. class Meta:
  148. unique_together = (('price', 'quantity'),)
  149. def __str__(self):
  150. return "%s for %s" % (self.quantity, self.price)
  151. class Triple(models.Model):
  152. left = models.IntegerField()
  153. middle = models.IntegerField()
  154. right = models.IntegerField()
  155. class Meta:
  156. unique_together = (('left', 'middle'), ('middle', 'right'))
  157. class ArticleStatus(models.Model):
  158. ARTICLE_STATUS_CHAR = (
  159. ('d', 'Draft'),
  160. ('p', 'Pending'),
  161. ('l', 'Live'),
  162. )
  163. status = models.CharField(max_length=2, choices=ARTICLE_STATUS_CHAR, blank=True, null=True)
  164. class Inventory(models.Model):
  165. barcode = models.PositiveIntegerField(unique=True)
  166. parent = models.ForeignKey('self', models.SET_NULL, to_field='barcode', blank=True, null=True)
  167. name = models.CharField(blank=False, max_length=20)
  168. class Meta:
  169. ordering = ('name',)
  170. def __str__(self):
  171. return self.name
  172. def __repr__(self):
  173. return self.__str__()
  174. class Book(models.Model):
  175. title = models.CharField(max_length=40)
  176. author = models.ForeignKey(Writer, models.SET_NULL, blank=True, null=True)
  177. special_id = models.IntegerField(blank=True, null=True, unique=True)
  178. class Meta:
  179. unique_together = ('title', 'author')
  180. class BookXtra(models.Model):
  181. isbn = models.CharField(max_length=16, unique=True)
  182. suffix1 = models.IntegerField(blank=True, default=0)
  183. suffix2 = models.IntegerField(blank=True, default=0)
  184. class Meta:
  185. unique_together = (('suffix1', 'suffix2'))
  186. abstract = True
  187. class DerivedBook(Book, BookXtra):
  188. pass
  189. class ExplicitPK(models.Model):
  190. key = models.CharField(max_length=20, primary_key=True)
  191. desc = models.CharField(max_length=20, blank=True, unique=True)
  192. class Meta:
  193. unique_together = ('key', 'desc')
  194. def __str__(self):
  195. return self.key
  196. class Post(models.Model):
  197. title = models.CharField(max_length=50, unique_for_date='posted', blank=True)
  198. slug = models.CharField(max_length=50, unique_for_year='posted', blank=True)
  199. subtitle = models.CharField(max_length=50, unique_for_month='posted', blank=True)
  200. posted = models.DateField()
  201. def __str__(self):
  202. return self.title
  203. class DateTimePost(models.Model):
  204. title = models.CharField(max_length=50, unique_for_date='posted', blank=True)
  205. slug = models.CharField(max_length=50, unique_for_year='posted', blank=True)
  206. subtitle = models.CharField(max_length=50, unique_for_month='posted', blank=True)
  207. posted = models.DateTimeField(editable=False)
  208. def __str__(self):
  209. return self.title
  210. class DerivedPost(Post):
  211. pass
  212. class BigInt(models.Model):
  213. biggie = models.BigIntegerField()
  214. def __str__(self):
  215. return str(self.biggie)
  216. class MarkupField(models.CharField):
  217. def __init__(self, *args, **kwargs):
  218. kwargs["max_length"] = 20
  219. super().__init__(*args, **kwargs)
  220. def formfield(self, **kwargs):
  221. # don't allow this field to be used in form (real use-case might be
  222. # that you know the markup will always be X, but it is among an app
  223. # that allows the user to say it could be something else)
  224. # regressed at r10062
  225. return None
  226. class CustomFieldForExclusionModel(models.Model):
  227. name = models.CharField(max_length=10)
  228. markup = MarkupField()
  229. class FlexibleDatePost(models.Model):
  230. title = models.CharField(max_length=50, unique_for_date='posted', blank=True)
  231. slug = models.CharField(max_length=50, unique_for_year='posted', blank=True)
  232. subtitle = models.CharField(max_length=50, unique_for_month='posted', blank=True)
  233. posted = models.DateField(blank=True, null=True)
  234. class Colour(models.Model):
  235. name = models.CharField(max_length=50)
  236. def __iter__(self):
  237. yield from range(5)
  238. def __str__(self):
  239. return self.name
  240. class ColourfulItem(models.Model):
  241. name = models.CharField(max_length=50)
  242. colours = models.ManyToManyField(Colour)
  243. class CustomErrorMessage(models.Model):
  244. name1 = models.CharField(
  245. max_length=50,
  246. validators=[validators.validate_slug],
  247. error_messages={'invalid': 'Model custom error message.'},
  248. )
  249. name2 = models.CharField(
  250. max_length=50,
  251. validators=[validators.validate_slug],
  252. error_messages={'invalid': 'Model custom error message.'},
  253. )
  254. def clean(self):
  255. if self.name1 == 'FORBIDDEN_VALUE':
  256. raise ValidationError({'name1': [ValidationError('Model.clean() error messages.')]})
  257. elif self.name1 == 'FORBIDDEN_VALUE2':
  258. raise ValidationError({'name1': 'Model.clean() error messages (simpler syntax).'})
  259. elif self.name1 == 'GLOBAL_ERROR':
  260. raise ValidationError("Global error message.")
  261. def today_callable_dict():
  262. return {"last_action__gte": datetime.datetime.today()}
  263. def today_callable_q():
  264. return models.Q(last_action__gte=datetime.datetime.today())
  265. class Character(models.Model):
  266. username = models.CharField(max_length=100)
  267. last_action = models.DateTimeField()
  268. class StumpJoke(models.Model):
  269. most_recently_fooled = models.ForeignKey(
  270. Character,
  271. models.CASCADE,
  272. limit_choices_to=today_callable_dict,
  273. related_name="+",
  274. )
  275. has_fooled_today = models.ManyToManyField(Character, limit_choices_to=today_callable_q, related_name="+")
  276. # Model for #13776
  277. class Student(models.Model):
  278. character = models.ForeignKey(Character, models.CASCADE)
  279. study = models.CharField(max_length=30)
  280. # Model for #639
  281. class Photo(models.Model):
  282. title = models.CharField(max_length=30)
  283. image = models.FileField(storage=temp_storage, upload_to='tests')
  284. # Support code for the tests; this keeps track of how many times save()
  285. # gets called on each instance.
  286. def __init__(self, *args, **kwargs):
  287. super().__init__(*args, **kwargs)
  288. self._savecount = 0
  289. def save(self, force_insert=False, force_update=False):
  290. super().save(force_insert, force_update)
  291. self._savecount += 1
  292. class UUIDPK(models.Model):
  293. uuid = models.UUIDField(primary_key=True, default=uuid.uuid4, editable=False)
  294. name = models.CharField(max_length=30)
  295. # Models for #24706
  296. class StrictAssignmentFieldSpecific(models.Model):
  297. title = models.CharField(max_length=30)
  298. _should_error = False
  299. def __setattr__(self, key, value):
  300. if self._should_error is True:
  301. raise ValidationError(message={key: "Cannot set attribute"}, code='invalid')
  302. super().__setattr__(key, value)
  303. class StrictAssignmentAll(models.Model):
  304. title = models.CharField(max_length=30)
  305. _should_error = False
  306. def __setattr__(self, key, value):
  307. if self._should_error is True:
  308. raise ValidationError(message="Cannot set attribute", code='invalid')
  309. super().__setattr__(key, value)
  310. # A model with ForeignKey(blank=False, null=True)
  311. class Award(models.Model):
  312. name = models.CharField(max_length=30)
  313. character = models.ForeignKey(Character, models.SET_NULL, blank=False, null=True)
  314. class NullableUniqueCharFieldModel(models.Model):
  315. codename = models.CharField(max_length=50, blank=True, null=True, unique=True)