Преглед изворни кода

Add unique_together constraints to ManyToMany through models

Oscar uses explicit through-models for some ManyToMany relationships.
The expectation is that for each combinatination of the two foreign
keys, only one model can exist. But this hasn't been enforced until now.

Fixes #1309.
master
Maik Hoepfel пре 11 година
родитељ
комит
0cb1bad391

+ 14
- 0
docs/source/releases/v0.8.rst Прегледај датотеку

@@ -150,10 +150,24 @@ are no nulls in your ``basket_line`` table:
150 150
 Migrations
151 151
 ~~~~~~~~~~
152 152
 
153
+* Catalogue:
154
+
155
+    - ``0021`` - Add ``unique_together`` to ``ProductAttributeValue``,
156
+      ``ProductRecommendation`` and ``ProductCategory``
157
+
158
+* Order:
159
+
160
+    - ``0029`` - Add ``unique_together`` to ``PaymentEventQuantity`` and ``ShippingEventQuantity``
161
+
162
+* Promotions:
163
+
164
+    - ``0006`` - Add ``unique_together`` to ``OrderedProduct``
165
+
153 166
 * Shipping:
154 167
 
155 168
     - ``0007`` - Change ``WeightBand.upper_limit`` from ``FloatField`` to ``DecimalField``
156 169
 
170
+
157 171
 .. _deprecated_features:
158 172
 
159 173
 Removal of deprecated features

+ 3
- 0
oscar/apps/catalogue/abstract_models.py Прегледај датотеку

@@ -183,6 +183,7 @@ class AbstractProductCategory(models.Model):
183 183
         ordering = ['product', 'category']
184 184
         verbose_name = _('Product Category')
185 185
         verbose_name_plural = _('Product Categories')
186
+        unique_together = ('product', 'category')
186 187
 
187 188
     def __unicode__(self):
188 189
         return u"<productcategory for product '%s'>" % self.product
@@ -525,6 +526,7 @@ class ProductRecommendation(models.Model):
525 526
         verbose_name = _('Product Recommendation')
526 527
         verbose_name_plural = _('Product Recomendations')
527 528
         ordering = ['primary', '-ranking']
529
+        unique_together = ('primary', 'recommendation')
528 530
 
529 531
 
530 532
 class ProductAttributesContainer(object):
@@ -817,6 +819,7 @@ class AbstractProductAttributeValue(models.Model):
817 819
         abstract = True
818 820
         verbose_name = _('Product Attribute Value')
819 821
         verbose_name_plural = _('Product Attribute Values')
822
+        unique_together = ('attribute', 'product')
820 823
 
821 824
     def __unicode__(self):
822 825
         return self.summary()

+ 156
- 0
oscar/apps/catalogue/migrations/0021_auto__add_unique_productattributevalue_attribute_product__add_unique_p.py Прегледај датотеку

@@ -0,0 +1,156 @@
1
+# -*- coding: utf-8 -*-
2
+from south.utils import datetime_utils as datetime
3
+from south.db import db
4
+from south.v2 import SchemaMigration
5
+from django.db import models
6
+
7
+
8
+class Migration(SchemaMigration):
9
+
10
+    def forwards(self, orm):
11
+        # Adding unique constraint on 'ProductAttributeValue', fields ['attribute', 'product']
12
+        db.create_unique(u'catalogue_productattributevalue', ['attribute_id', 'product_id'])
13
+
14
+        # Adding unique constraint on 'ProductRecommendation', fields ['primary', 'recommendation']
15
+        db.create_unique(u'catalogue_productrecommendation', ['primary_id', 'recommendation_id'])
16
+
17
+        # Adding unique constraint on 'ProductCategory', fields ['product', 'category']
18
+        db.create_unique(u'catalogue_productcategory', ['product_id', 'category_id'])
19
+
20
+
21
+    def backwards(self, orm):
22
+        # Removing unique constraint on 'ProductCategory', fields ['product', 'category']
23
+        db.delete_unique(u'catalogue_productcategory', ['product_id', 'category_id'])
24
+
25
+        # Removing unique constraint on 'ProductRecommendation', fields ['primary', 'recommendation']
26
+        db.delete_unique(u'catalogue_productrecommendation', ['primary_id', 'recommendation_id'])
27
+
28
+        # Removing unique constraint on 'ProductAttributeValue', fields ['attribute', 'product']
29
+        db.delete_unique(u'catalogue_productattributevalue', ['attribute_id', 'product_id'])
30
+
31
+
32
+    models = {
33
+        u'catalogue.attributeentity': {
34
+            'Meta': {'object_name': 'AttributeEntity'},
35
+            u'id': ('django.db.models.fields.AutoField', [], {'primary_key': 'True'}),
36
+            'name': ('django.db.models.fields.CharField', [], {'max_length': '255'}),
37
+            'slug': ('django.db.models.fields.SlugField', [], {'max_length': '255', 'blank': 'True'}),
38
+            'type': ('django.db.models.fields.related.ForeignKey', [], {'related_name': "'entities'", 'to': u"orm['catalogue.AttributeEntityType']"})
39
+        },
40
+        u'catalogue.attributeentitytype': {
41
+            'Meta': {'object_name': 'AttributeEntityType'},
42
+            u'id': ('django.db.models.fields.AutoField', [], {'primary_key': 'True'}),
43
+            'name': ('django.db.models.fields.CharField', [], {'max_length': '255'}),
44
+            'slug': ('django.db.models.fields.SlugField', [], {'max_length': '255', 'blank': 'True'})
45
+        },
46
+        u'catalogue.attributeoption': {
47
+            'Meta': {'object_name': 'AttributeOption'},
48
+            'group': ('django.db.models.fields.related.ForeignKey', [], {'related_name': "'options'", 'to': u"orm['catalogue.AttributeOptionGroup']"}),
49
+            u'id': ('django.db.models.fields.AutoField', [], {'primary_key': 'True'}),
50
+            'option': ('django.db.models.fields.CharField', [], {'max_length': '255'})
51
+        },
52
+        u'catalogue.attributeoptiongroup': {
53
+            'Meta': {'object_name': 'AttributeOptionGroup'},
54
+            u'id': ('django.db.models.fields.AutoField', [], {'primary_key': 'True'}),
55
+            'name': ('django.db.models.fields.CharField', [], {'max_length': '128'})
56
+        },
57
+        u'catalogue.category': {
58
+            'Meta': {'ordering': "['full_name']", 'object_name': 'Category'},
59
+            'depth': ('django.db.models.fields.PositiveIntegerField', [], {}),
60
+            'description': ('django.db.models.fields.TextField', [], {'blank': 'True'}),
61
+            'full_name': ('django.db.models.fields.CharField', [], {'max_length': '255', 'db_index': 'True'}),
62
+            u'id': ('django.db.models.fields.AutoField', [], {'primary_key': 'True'}),
63
+            'image': ('django.db.models.fields.files.ImageField', [], {'max_length': '255', 'null': 'True', 'blank': 'True'}),
64
+            'name': ('django.db.models.fields.CharField', [], {'max_length': '255', 'db_index': 'True'}),
65
+            'numchild': ('django.db.models.fields.PositiveIntegerField', [], {'default': '0'}),
66
+            'path': ('django.db.models.fields.CharField', [], {'unique': 'True', 'max_length': '255'}),
67
+            'slug': ('django.db.models.fields.SlugField', [], {'max_length': '255'})
68
+        },
69
+        u'catalogue.option': {
70
+            'Meta': {'object_name': 'Option'},
71
+            'code': ('oscar.models.fields.autoslugfield.AutoSlugField', [], {'allow_duplicates': 'False', 'max_length': '128', 'separator': "u'-'", 'blank': 'True', 'unique': 'True', 'populate_from': "'name'", 'overwrite': 'False'}),
72
+            u'id': ('django.db.models.fields.AutoField', [], {'primary_key': 'True'}),
73
+            'name': ('django.db.models.fields.CharField', [], {'max_length': '128'}),
74
+            'type': ('django.db.models.fields.CharField', [], {'default': "'Required'", 'max_length': '128'})
75
+        },
76
+        u'catalogue.product': {
77
+            'Meta': {'ordering': "['-date_created']", 'object_name': 'Product'},
78
+            'attributes': ('django.db.models.fields.related.ManyToManyField', [], {'to': u"orm['catalogue.ProductAttribute']", 'through': u"orm['catalogue.ProductAttributeValue']", 'symmetrical': 'False'}),
79
+            'categories': ('django.db.models.fields.related.ManyToManyField', [], {'to': u"orm['catalogue.Category']", 'through': u"orm['catalogue.ProductCategory']", 'symmetrical': 'False'}),
80
+            'date_created': ('django.db.models.fields.DateTimeField', [], {'auto_now_add': 'True', 'blank': 'True'}),
81
+            'date_updated': ('django.db.models.fields.DateTimeField', [], {'auto_now': 'True', 'db_index': 'True', 'blank': 'True'}),
82
+            'description': ('django.db.models.fields.TextField', [], {'blank': 'True'}),
83
+            u'id': ('django.db.models.fields.AutoField', [], {'primary_key': 'True'}),
84
+            'is_discountable': ('django.db.models.fields.BooleanField', [], {'default': 'True'}),
85
+            'parent': ('django.db.models.fields.related.ForeignKey', [], {'blank': 'True', 'related_name': "'variants'", 'null': 'True', 'to': u"orm['catalogue.Product']"}),
86
+            'product_class': ('django.db.models.fields.related.ForeignKey', [], {'related_name': "'products'", 'null': 'True', 'on_delete': 'models.PROTECT', 'to': u"orm['catalogue.ProductClass']"}),
87
+            'product_options': ('django.db.models.fields.related.ManyToManyField', [], {'to': u"orm['catalogue.Option']", 'symmetrical': 'False', 'blank': 'True'}),
88
+            'rating': ('django.db.models.fields.FloatField', [], {'null': 'True'}),
89
+            'recommended_products': ('django.db.models.fields.related.ManyToManyField', [], {'to': u"orm['catalogue.Product']", 'symmetrical': 'False', 'through': u"orm['catalogue.ProductRecommendation']", 'blank': 'True'}),
90
+            'related_products': ('django.db.models.fields.related.ManyToManyField', [], {'symmetrical': 'False', 'related_name': "'relations'", 'blank': 'True', 'to': u"orm['catalogue.Product']"}),
91
+            'score': ('django.db.models.fields.FloatField', [], {'default': '0.0', 'db_index': 'True'}),
92
+            'slug': ('django.db.models.fields.SlugField', [], {'max_length': '255'}),
93
+            'title': ('django.db.models.fields.CharField', [], {'max_length': '255', 'blank': 'True'}),
94
+            'upc': ('oscar.models.fields.NullCharField', [], {'max_length': '64', 'unique': 'True', 'null': 'True', 'blank': 'True'})
95
+        },
96
+        u'catalogue.productattribute': {
97
+            'Meta': {'ordering': "['code']", 'object_name': 'ProductAttribute'},
98
+            'code': ('django.db.models.fields.SlugField', [], {'max_length': '128'}),
99
+            'entity_type': ('django.db.models.fields.related.ForeignKey', [], {'to': u"orm['catalogue.AttributeEntityType']", 'null': 'True', 'blank': 'True'}),
100
+            u'id': ('django.db.models.fields.AutoField', [], {'primary_key': 'True'}),
101
+            'name': ('django.db.models.fields.CharField', [], {'max_length': '128'}),
102
+            'option_group': ('django.db.models.fields.related.ForeignKey', [], {'to': u"orm['catalogue.AttributeOptionGroup']", 'null': 'True', 'blank': 'True'}),
103
+            'product_class': ('django.db.models.fields.related.ForeignKey', [], {'blank': 'True', 'related_name': "'attributes'", 'null': 'True', 'to': u"orm['catalogue.ProductClass']"}),
104
+            'required': ('django.db.models.fields.BooleanField', [], {'default': 'False'}),
105
+            'type': ('django.db.models.fields.CharField', [], {'default': "'text'", 'max_length': '20'})
106
+        },
107
+        u'catalogue.productattributevalue': {
108
+            'Meta': {'unique_together': "(('attribute', 'product'),)", 'object_name': 'ProductAttributeValue'},
109
+            'attribute': ('django.db.models.fields.related.ForeignKey', [], {'to': u"orm['catalogue.ProductAttribute']"}),
110
+            u'id': ('django.db.models.fields.AutoField', [], {'primary_key': 'True'}),
111
+            'product': ('django.db.models.fields.related.ForeignKey', [], {'related_name': "'attribute_values'", 'to': u"orm['catalogue.Product']"}),
112
+            'value_boolean': ('django.db.models.fields.NullBooleanField', [], {'null': 'True', 'blank': 'True'}),
113
+            'value_date': ('django.db.models.fields.DateField', [], {'null': 'True', 'blank': 'True'}),
114
+            'value_entity': ('django.db.models.fields.related.ForeignKey', [], {'to': u"orm['catalogue.AttributeEntity']", 'null': 'True', 'blank': 'True'}),
115
+            'value_file': ('django.db.models.fields.files.FileField', [], {'max_length': '255', 'null': 'True', 'blank': 'True'}),
116
+            'value_float': ('django.db.models.fields.FloatField', [], {'null': 'True', 'blank': 'True'}),
117
+            'value_image': ('django.db.models.fields.files.ImageField', [], {'max_length': '255', 'null': 'True', 'blank': 'True'}),
118
+            'value_integer': ('django.db.models.fields.IntegerField', [], {'null': 'True', 'blank': 'True'}),
119
+            'value_option': ('django.db.models.fields.related.ForeignKey', [], {'to': u"orm['catalogue.AttributeOption']", 'null': 'True', 'blank': 'True'}),
120
+            'value_richtext': ('django.db.models.fields.TextField', [], {'null': 'True', 'blank': 'True'}),
121
+            'value_text': ('django.db.models.fields.CharField', [], {'max_length': '255', 'null': 'True', 'blank': 'True'})
122
+        },
123
+        u'catalogue.productcategory': {
124
+            'Meta': {'ordering': "['product', 'category']", 'unique_together': "(('product', 'category'),)", 'object_name': 'ProductCategory'},
125
+            'category': ('django.db.models.fields.related.ForeignKey', [], {'to': u"orm['catalogue.Category']"}),
126
+            u'id': ('django.db.models.fields.AutoField', [], {'primary_key': 'True'}),
127
+            'product': ('django.db.models.fields.related.ForeignKey', [], {'to': u"orm['catalogue.Product']"})
128
+        },
129
+        u'catalogue.productclass': {
130
+            'Meta': {'ordering': "['name']", 'object_name': 'ProductClass'},
131
+            u'id': ('django.db.models.fields.AutoField', [], {'primary_key': 'True'}),
132
+            'name': ('django.db.models.fields.CharField', [], {'max_length': '128'}),
133
+            'options': ('django.db.models.fields.related.ManyToManyField', [], {'to': u"orm['catalogue.Option']", 'symmetrical': 'False', 'blank': 'True'}),
134
+            'requires_shipping': ('django.db.models.fields.BooleanField', [], {'default': 'True'}),
135
+            'slug': ('oscar.models.fields.autoslugfield.AutoSlugField', [], {'allow_duplicates': 'False', 'max_length': '128', 'separator': "u'-'", 'blank': 'True', 'unique': 'True', 'populate_from': "'name'", 'overwrite': 'False'}),
136
+            'track_stock': ('django.db.models.fields.BooleanField', [], {'default': 'True'})
137
+        },
138
+        u'catalogue.productimage': {
139
+            'Meta': {'ordering': "['display_order']", 'unique_together': "(('product', 'display_order'),)", 'object_name': 'ProductImage'},
140
+            'caption': ('django.db.models.fields.CharField', [], {'max_length': '200', 'blank': 'True'}),
141
+            'date_created': ('django.db.models.fields.DateTimeField', [], {'auto_now_add': 'True', 'blank': 'True'}),
142
+            'display_order': ('django.db.models.fields.PositiveIntegerField', [], {'default': '0'}),
143
+            u'id': ('django.db.models.fields.AutoField', [], {'primary_key': 'True'}),
144
+            'original': ('django.db.models.fields.files.ImageField', [], {'max_length': '255'}),
145
+            'product': ('django.db.models.fields.related.ForeignKey', [], {'related_name': "'images'", 'to': u"orm['catalogue.Product']"})
146
+        },
147
+        u'catalogue.productrecommendation': {
148
+            'Meta': {'unique_together': "(('primary', 'recommendation'),)", 'object_name': 'ProductRecommendation'},
149
+            u'id': ('django.db.models.fields.AutoField', [], {'primary_key': 'True'}),
150
+            'primary': ('django.db.models.fields.related.ForeignKey', [], {'related_name': "'primary_recommendations'", 'to': u"orm['catalogue.Product']"}),
151
+            'ranking': ('django.db.models.fields.PositiveSmallIntegerField', [], {'default': '0'}),
152
+            'recommendation': ('django.db.models.fields.related.ForeignKey', [], {'to': u"orm['catalogue.Product']"})
153
+        }
154
+    }
155
+
156
+    complete_apps = ['catalogue']

+ 1
- 1
oscar/apps/offer/models.py Прегледај датотеку

@@ -902,7 +902,7 @@ class RangeProduct(models.Model):
902 902
     display_order = models.IntegerField(default=0)
903 903
 
904 904
     class Meta:
905
-        unique_together = (('range', 'product'),)
905
+        unique_together = ('range', 'product')
906 906
 
907 907
 # ==========
908 908
 # Conditions

+ 2
- 0
oscar/apps/order/abstract_models.py Прегледај датотеку

@@ -818,6 +818,7 @@ class PaymentEventQuantity(models.Model):
818 818
     class Meta:
819 819
         verbose_name = _("Payment Event Quantity")
820 820
         verbose_name_plural = _("Payment Event Quantities")
821
+        unique_together = ('event', 'line')
821 822
 
822 823
 
823 824
 # SHIPPING EVENTS
@@ -874,6 +875,7 @@ class ShippingEventQuantity(models.Model):
874 875
     class Meta:
875 876
         verbose_name = _("Shipping Event Quantity")
876 877
         verbose_name_plural = _("Shipping Event Quantities")
878
+        unique_together = ('event', 'line')
877 879
 
878 880
     def save(self, *args, **kwargs):
879 881
         # Default quantity to full quantity of line

+ 489
- 0
oscar/apps/order/migrations/0029_auto__add_unique_paymenteventquantity_event_line__add_unique_shippinge.py Прегледај датотеку

@@ -0,0 +1,489 @@
1
+# -*- coding: utf-8 -*-
2
+from south.utils import datetime_utils as datetime
3
+from south.db import db
4
+from south.v2 import SchemaMigration
5
+from django.db import models
6
+
7
+from oscar.core.compat import AUTH_USER_MODEL, AUTH_USER_MODEL_NAME
8
+
9
+
10
+class Migration(SchemaMigration):
11
+
12
+    def forwards(self, orm):
13
+        # Adding unique constraint on 'PaymentEventQuantity', fields ['event', 'line']
14
+        db.create_unique(u'order_paymenteventquantity', ['event_id', 'line_id'])
15
+
16
+        # Adding unique constraint on 'ShippingEventQuantity', fields ['event', 'line']
17
+        db.create_unique(u'order_shippingeventquantity', ['event_id', 'line_id'])
18
+
19
+
20
+    def backwards(self, orm):
21
+        # Removing unique constraint on 'ShippingEventQuantity', fields ['event', 'line']
22
+        db.delete_unique(u'order_shippingeventquantity', ['event_id', 'line_id'])
23
+
24
+        # Removing unique constraint on 'PaymentEventQuantity', fields ['event', 'line']
25
+        db.delete_unique(u'order_paymenteventquantity', ['event_id', 'line_id'])
26
+
27
+
28
+    models = {
29
+        u'address.country': {
30
+            'Meta': {'ordering': "('-display_order', 'name')", 'object_name': 'Country'},
31
+            'display_order': ('django.db.models.fields.PositiveSmallIntegerField', [], {'default': '0', 'db_index': 'True'}),
32
+            'is_shipping_country': ('django.db.models.fields.BooleanField', [], {'default': 'False', 'db_index': 'True'}),
33
+            'iso_3166_1_a2': ('django.db.models.fields.CharField', [], {'max_length': '2', 'primary_key': 'True'}),
34
+            'iso_3166_1_a3': ('django.db.models.fields.CharField', [], {'db_index': 'True', 'max_length': '3', 'blank': 'True'}),
35
+            'iso_3166_1_numeric': ('django.db.models.fields.PositiveSmallIntegerField', [], {'null': 'True', 'db_index': 'True'}),
36
+            'name': ('django.db.models.fields.CharField', [], {'max_length': '128'}),
37
+            'printable_name': ('django.db.models.fields.CharField', [], {'max_length': '128'})
38
+        },
39
+        u'auth.group': {
40
+            'Meta': {'object_name': 'Group'},
41
+            u'id': ('django.db.models.fields.AutoField', [], {'primary_key': 'True'}),
42
+            'name': ('django.db.models.fields.CharField', [], {'unique': 'True', 'max_length': '80'}),
43
+            'permissions': ('django.db.models.fields.related.ManyToManyField', [], {'to': u"orm['auth.Permission']", 'symmetrical': 'False', 'blank': 'True'})
44
+        },
45
+        u'auth.permission': {
46
+            'Meta': {'ordering': "(u'content_type__app_label', u'content_type__model', u'codename')", 'unique_together': "((u'content_type', u'codename'),)", 'object_name': 'Permission'},
47
+            'codename': ('django.db.models.fields.CharField', [], {'max_length': '100'}),
48
+            'content_type': ('django.db.models.fields.related.ForeignKey', [], {'to': u"orm['contenttypes.ContentType']"}),
49
+            u'id': ('django.db.models.fields.AutoField', [], {'primary_key': 'True'}),
50
+            'name': ('django.db.models.fields.CharField', [], {'max_length': '50'})
51
+        },
52
+        AUTH_USER_MODEL: {
53
+            'Meta': {'object_name': AUTH_USER_MODEL_NAME},
54
+            'date_joined': ('django.db.models.fields.DateTimeField', [], {'default': 'datetime.datetime.now'}),
55
+            'email': ('django.db.models.fields.EmailField', [], {'max_length': '75', 'blank': 'True'}),
56
+            'first_name': ('django.db.models.fields.CharField', [], {'max_length': '30', 'blank': 'True'}),
57
+            'groups': ('django.db.models.fields.related.ManyToManyField', [], {'symmetrical': 'False', 'related_name': "u'user_set'", 'blank': 'True', 'to': u"orm['auth.Group']"}),
58
+            u'id': ('django.db.models.fields.AutoField', [], {'primary_key': 'True'}),
59
+            'is_active': ('django.db.models.fields.BooleanField', [], {'default': 'True'}),
60
+            'is_staff': ('django.db.models.fields.BooleanField', [], {'default': 'False'}),
61
+            'is_superuser': ('django.db.models.fields.BooleanField', [], {'default': 'False'}),
62
+            'last_login': ('django.db.models.fields.DateTimeField', [], {'default': 'datetime.datetime.now'}),
63
+            'last_name': ('django.db.models.fields.CharField', [], {'max_length': '30', 'blank': 'True'}),
64
+            'password': ('django.db.models.fields.CharField', [], {'max_length': '128'}),
65
+            'user_permissions': ('django.db.models.fields.related.ManyToManyField', [], {'symmetrical': 'False', 'related_name': "u'user_set'", 'blank': 'True', 'to': u"orm['auth.Permission']"}),
66
+            'username': ('django.db.models.fields.CharField', [], {'unique': 'True', 'max_length': '30'})
67
+        },
68
+        u'basket.basket': {
69
+            'Meta': {'object_name': 'Basket'},
70
+            'date_created': ('django.db.models.fields.DateTimeField', [], {'auto_now_add': 'True', 'blank': 'True'}),
71
+            'date_merged': ('django.db.models.fields.DateTimeField', [], {'null': 'True', 'blank': 'True'}),
72
+            'date_submitted': ('django.db.models.fields.DateTimeField', [], {'null': 'True', 'blank': 'True'}),
73
+            u'id': ('django.db.models.fields.AutoField', [], {'primary_key': 'True'}),
74
+            'owner': ('django.db.models.fields.related.ForeignKey', [], {'related_name': "'baskets'", 'null': 'True', 'to': u"orm['{0}']".format(AUTH_USER_MODEL)}),
75
+            'status': ('django.db.models.fields.CharField', [], {'default': "'Open'", 'max_length': '128'}),
76
+            'vouchers': ('django.db.models.fields.related.ManyToManyField', [], {'symmetrical': 'False', 'to': u"orm['voucher.Voucher']", 'null': 'True', 'blank': 'True'})
77
+        },
78
+        u'catalogue.attributeentity': {
79
+            'Meta': {'object_name': 'AttributeEntity'},
80
+            u'id': ('django.db.models.fields.AutoField', [], {'primary_key': 'True'}),
81
+            'name': ('django.db.models.fields.CharField', [], {'max_length': '255'}),
82
+            'slug': ('django.db.models.fields.SlugField', [], {'max_length': '255', 'blank': 'True'}),
83
+            'type': ('django.db.models.fields.related.ForeignKey', [], {'related_name': "'entities'", 'to': u"orm['catalogue.AttributeEntityType']"})
84
+        },
85
+        u'catalogue.attributeentitytype': {
86
+            'Meta': {'object_name': 'AttributeEntityType'},
87
+            u'id': ('django.db.models.fields.AutoField', [], {'primary_key': 'True'}),
88
+            'name': ('django.db.models.fields.CharField', [], {'max_length': '255'}),
89
+            'slug': ('django.db.models.fields.SlugField', [], {'max_length': '255', 'blank': 'True'})
90
+        },
91
+        u'catalogue.attributeoption': {
92
+            'Meta': {'object_name': 'AttributeOption'},
93
+            'group': ('django.db.models.fields.related.ForeignKey', [], {'related_name': "'options'", 'to': u"orm['catalogue.AttributeOptionGroup']"}),
94
+            u'id': ('django.db.models.fields.AutoField', [], {'primary_key': 'True'}),
95
+            'option': ('django.db.models.fields.CharField', [], {'max_length': '255'})
96
+        },
97
+        u'catalogue.attributeoptiongroup': {
98
+            'Meta': {'object_name': 'AttributeOptionGroup'},
99
+            u'id': ('django.db.models.fields.AutoField', [], {'primary_key': 'True'}),
100
+            'name': ('django.db.models.fields.CharField', [], {'max_length': '128'})
101
+        },
102
+        u'catalogue.category': {
103
+            'Meta': {'ordering': "['full_name']", 'object_name': 'Category'},
104
+            'depth': ('django.db.models.fields.PositiveIntegerField', [], {}),
105
+            'description': ('django.db.models.fields.TextField', [], {'blank': 'True'}),
106
+            'full_name': ('django.db.models.fields.CharField', [], {'max_length': '255', 'db_index': 'True'}),
107
+            u'id': ('django.db.models.fields.AutoField', [], {'primary_key': 'True'}),
108
+            'image': ('django.db.models.fields.files.ImageField', [], {'max_length': '255', 'null': 'True', 'blank': 'True'}),
109
+            'name': ('django.db.models.fields.CharField', [], {'max_length': '255', 'db_index': 'True'}),
110
+            'numchild': ('django.db.models.fields.PositiveIntegerField', [], {'default': '0'}),
111
+            'path': ('django.db.models.fields.CharField', [], {'unique': 'True', 'max_length': '255'}),
112
+            'slug': ('django.db.models.fields.SlugField', [], {'max_length': '255'})
113
+        },
114
+        u'catalogue.option': {
115
+            'Meta': {'object_name': 'Option'},
116
+            'code': ('oscar.models.fields.autoslugfield.AutoSlugField', [], {'allow_duplicates': 'False', 'max_length': '128', 'separator': "u'-'", 'blank': 'True', 'unique': 'True', 'populate_from': "'name'", 'overwrite': 'False'}),
117
+            u'id': ('django.db.models.fields.AutoField', [], {'primary_key': 'True'}),
118
+            'name': ('django.db.models.fields.CharField', [], {'max_length': '128'}),
119
+            'type': ('django.db.models.fields.CharField', [], {'default': "'Required'", 'max_length': '128'})
120
+        },
121
+        u'catalogue.product': {
122
+            'Meta': {'ordering': "['-date_created']", 'object_name': 'Product'},
123
+            'attributes': ('django.db.models.fields.related.ManyToManyField', [], {'to': u"orm['catalogue.ProductAttribute']", 'through': u"orm['catalogue.ProductAttributeValue']", 'symmetrical': 'False'}),
124
+            'categories': ('django.db.models.fields.related.ManyToManyField', [], {'to': u"orm['catalogue.Category']", 'through': u"orm['catalogue.ProductCategory']", 'symmetrical': 'False'}),
125
+            'date_created': ('django.db.models.fields.DateTimeField', [], {'auto_now_add': 'True', 'blank': 'True'}),
126
+            'date_updated': ('django.db.models.fields.DateTimeField', [], {'auto_now': 'True', 'db_index': 'True', 'blank': 'True'}),
127
+            'description': ('django.db.models.fields.TextField', [], {'blank': 'True'}),
128
+            u'id': ('django.db.models.fields.AutoField', [], {'primary_key': 'True'}),
129
+            'is_discountable': ('django.db.models.fields.BooleanField', [], {'default': 'True'}),
130
+            'parent': ('django.db.models.fields.related.ForeignKey', [], {'blank': 'True', 'related_name': "'variants'", 'null': 'True', 'to': u"orm['catalogue.Product']"}),
131
+            'product_class': ('django.db.models.fields.related.ForeignKey', [], {'related_name': "'products'", 'null': 'True', 'on_delete': 'models.PROTECT', 'to': u"orm['catalogue.ProductClass']"}),
132
+            'product_options': ('django.db.models.fields.related.ManyToManyField', [], {'to': u"orm['catalogue.Option']", 'symmetrical': 'False', 'blank': 'True'}),
133
+            'rating': ('django.db.models.fields.FloatField', [], {'null': 'True'}),
134
+            'recommended_products': ('django.db.models.fields.related.ManyToManyField', [], {'to': u"orm['catalogue.Product']", 'symmetrical': 'False', 'through': u"orm['catalogue.ProductRecommendation']", 'blank': 'True'}),
135
+            'related_products': ('django.db.models.fields.related.ManyToManyField', [], {'symmetrical': 'False', 'related_name': "'relations'", 'blank': 'True', 'to': u"orm['catalogue.Product']"}),
136
+            'score': ('django.db.models.fields.FloatField', [], {'default': '0.0', 'db_index': 'True'}),
137
+            'slug': ('django.db.models.fields.SlugField', [], {'max_length': '255'}),
138
+            'title': ('django.db.models.fields.CharField', [], {'max_length': '255', 'blank': 'True'}),
139
+            'upc': ('oscar.models.fields.NullCharField', [], {'max_length': '64', 'unique': 'True', 'null': 'True', 'blank': 'True'})
140
+        },
141
+        u'catalogue.productattribute': {
142
+            'Meta': {'ordering': "['code']", 'object_name': 'ProductAttribute'},
143
+            'code': ('django.db.models.fields.SlugField', [], {'max_length': '128'}),
144
+            'entity_type': ('django.db.models.fields.related.ForeignKey', [], {'to': u"orm['catalogue.AttributeEntityType']", 'null': 'True', 'blank': 'True'}),
145
+            u'id': ('django.db.models.fields.AutoField', [], {'primary_key': 'True'}),
146
+            'name': ('django.db.models.fields.CharField', [], {'max_length': '128'}),
147
+            'option_group': ('django.db.models.fields.related.ForeignKey', [], {'to': u"orm['catalogue.AttributeOptionGroup']", 'null': 'True', 'blank': 'True'}),
148
+            'product_class': ('django.db.models.fields.related.ForeignKey', [], {'blank': 'True', 'related_name': "'attributes'", 'null': 'True', 'to': u"orm['catalogue.ProductClass']"}),
149
+            'required': ('django.db.models.fields.BooleanField', [], {'default': 'False'}),
150
+            'type': ('django.db.models.fields.CharField', [], {'default': "'text'", 'max_length': '20'})
151
+        },
152
+        u'catalogue.productattributevalue': {
153
+            'Meta': {'unique_together': "(('attribute', 'product'),)", 'object_name': 'ProductAttributeValue'},
154
+            'attribute': ('django.db.models.fields.related.ForeignKey', [], {'to': u"orm['catalogue.ProductAttribute']"}),
155
+            u'id': ('django.db.models.fields.AutoField', [], {'primary_key': 'True'}),
156
+            'product': ('django.db.models.fields.related.ForeignKey', [], {'related_name': "'attribute_values'", 'to': u"orm['catalogue.Product']"}),
157
+            'value_boolean': ('django.db.models.fields.NullBooleanField', [], {'null': 'True', 'blank': 'True'}),
158
+            'value_date': ('django.db.models.fields.DateField', [], {'null': 'True', 'blank': 'True'}),
159
+            'value_entity': ('django.db.models.fields.related.ForeignKey', [], {'to': u"orm['catalogue.AttributeEntity']", 'null': 'True', 'blank': 'True'}),
160
+            'value_file': ('django.db.models.fields.files.FileField', [], {'max_length': '255', 'null': 'True', 'blank': 'True'}),
161
+            'value_float': ('django.db.models.fields.FloatField', [], {'null': 'True', 'blank': 'True'}),
162
+            'value_image': ('django.db.models.fields.files.ImageField', [], {'max_length': '255', 'null': 'True', 'blank': 'True'}),
163
+            'value_integer': ('django.db.models.fields.IntegerField', [], {'null': 'True', 'blank': 'True'}),
164
+            'value_option': ('django.db.models.fields.related.ForeignKey', [], {'to': u"orm['catalogue.AttributeOption']", 'null': 'True', 'blank': 'True'}),
165
+            'value_richtext': ('django.db.models.fields.TextField', [], {'null': 'True', 'blank': 'True'}),
166
+            'value_text': ('django.db.models.fields.CharField', [], {'max_length': '255', 'null': 'True', 'blank': 'True'})
167
+        },
168
+        u'catalogue.productcategory': {
169
+            'Meta': {'ordering': "['product', 'category']", 'unique_together': "(('product', 'category'),)", 'object_name': 'ProductCategory'},
170
+            'category': ('django.db.models.fields.related.ForeignKey', [], {'to': u"orm['catalogue.Category']"}),
171
+            u'id': ('django.db.models.fields.AutoField', [], {'primary_key': 'True'}),
172
+            'product': ('django.db.models.fields.related.ForeignKey', [], {'to': u"orm['catalogue.Product']"})
173
+        },
174
+        u'catalogue.productclass': {
175
+            'Meta': {'ordering': "['name']", 'object_name': 'ProductClass'},
176
+            u'id': ('django.db.models.fields.AutoField', [], {'primary_key': 'True'}),
177
+            'name': ('django.db.models.fields.CharField', [], {'max_length': '128'}),
178
+            'options': ('django.db.models.fields.related.ManyToManyField', [], {'to': u"orm['catalogue.Option']", 'symmetrical': 'False', 'blank': 'True'}),
179
+            'requires_shipping': ('django.db.models.fields.BooleanField', [], {'default': 'True'}),
180
+            'slug': ('oscar.models.fields.autoslugfield.AutoSlugField', [], {'allow_duplicates': 'False', 'max_length': '128', 'separator': "u'-'", 'blank': 'True', 'unique': 'True', 'populate_from': "'name'", 'overwrite': 'False'}),
181
+            'track_stock': ('django.db.models.fields.BooleanField', [], {'default': 'True'})
182
+        },
183
+        u'catalogue.productrecommendation': {
184
+            'Meta': {'unique_together': "(('primary', 'recommendation'),)", 'object_name': 'ProductRecommendation'},
185
+            u'id': ('django.db.models.fields.AutoField', [], {'primary_key': 'True'}),
186
+            'primary': ('django.db.models.fields.related.ForeignKey', [], {'related_name': "'primary_recommendations'", 'to': u"orm['catalogue.Product']"}),
187
+            'ranking': ('django.db.models.fields.PositiveSmallIntegerField', [], {'default': '0'}),
188
+            'recommendation': ('django.db.models.fields.related.ForeignKey', [], {'to': u"orm['catalogue.Product']"})
189
+        },
190
+        u'contenttypes.contenttype': {
191
+            'Meta': {'ordering': "('name',)", 'unique_together': "(('app_label', 'model'),)", 'object_name': 'ContentType', 'db_table': "'django_content_type'"},
192
+            'app_label': ('django.db.models.fields.CharField', [], {'max_length': '100'}),
193
+            u'id': ('django.db.models.fields.AutoField', [], {'primary_key': 'True'}),
194
+            'model': ('django.db.models.fields.CharField', [], {'max_length': '100'}),
195
+            'name': ('django.db.models.fields.CharField', [], {'max_length': '100'})
196
+        },
197
+        u'customer.communicationeventtype': {
198
+            'Meta': {'object_name': 'CommunicationEventType'},
199
+            'category': ('django.db.models.fields.CharField', [], {'default': "u'Order related'", 'max_length': '255'}),
200
+            'code': ('oscar.models.fields.autoslugfield.AutoSlugField', [], {'allow_duplicates': 'False', 'max_length': '128', 'separator': "u'_'", 'blank': 'True', 'unique': 'True', 'populate_from': "'name'", 'overwrite': 'False'}),
201
+            'date_created': ('django.db.models.fields.DateTimeField', [], {'auto_now_add': 'True', 'blank': 'True'}),
202
+            'date_updated': ('django.db.models.fields.DateTimeField', [], {'auto_now': 'True', 'blank': 'True'}),
203
+            'email_body_html_template': ('django.db.models.fields.TextField', [], {'null': 'True', 'blank': 'True'}),
204
+            'email_body_template': ('django.db.models.fields.TextField', [], {'null': 'True', 'blank': 'True'}),
205
+            'email_subject_template': ('django.db.models.fields.CharField', [], {'max_length': '255', 'null': 'True', 'blank': 'True'}),
206
+            u'id': ('django.db.models.fields.AutoField', [], {'primary_key': 'True'}),
207
+            'name': ('django.db.models.fields.CharField', [], {'max_length': '255'}),
208
+            'sms_template': ('django.db.models.fields.CharField', [], {'max_length': '170', 'null': 'True', 'blank': 'True'})
209
+        },
210
+        u'offer.benefit': {
211
+            'Meta': {'object_name': 'Benefit'},
212
+            u'id': ('django.db.models.fields.AutoField', [], {'primary_key': 'True'}),
213
+            'max_affected_items': ('django.db.models.fields.PositiveIntegerField', [], {'null': 'True', 'blank': 'True'}),
214
+            'proxy_class': ('oscar.models.fields.NullCharField', [], {'default': 'None', 'max_length': '255', 'unique': 'True', 'null': 'True', 'blank': 'True'}),
215
+            'range': ('django.db.models.fields.related.ForeignKey', [], {'to': u"orm['offer.Range']", 'null': 'True', 'blank': 'True'}),
216
+            'type': ('django.db.models.fields.CharField', [], {'max_length': '128', 'blank': 'True'}),
217
+            'value': ('oscar.models.fields.PositiveDecimalField', [], {'null': 'True', 'max_digits': '12', 'decimal_places': '2', 'blank': 'True'})
218
+        },
219
+        u'offer.condition': {
220
+            'Meta': {'object_name': 'Condition'},
221
+            u'id': ('django.db.models.fields.AutoField', [], {'primary_key': 'True'}),
222
+            'proxy_class': ('oscar.models.fields.NullCharField', [], {'default': 'None', 'max_length': '255', 'unique': 'True', 'null': 'True', 'blank': 'True'}),
223
+            'range': ('django.db.models.fields.related.ForeignKey', [], {'to': u"orm['offer.Range']", 'null': 'True', 'blank': 'True'}),
224
+            'type': ('django.db.models.fields.CharField', [], {'max_length': '128', 'blank': 'True'}),
225
+            'value': ('oscar.models.fields.PositiveDecimalField', [], {'null': 'True', 'max_digits': '12', 'decimal_places': '2', 'blank': 'True'})
226
+        },
227
+        u'offer.conditionaloffer': {
228
+            'Meta': {'ordering': "['-priority']", 'object_name': 'ConditionalOffer'},
229
+            'benefit': ('django.db.models.fields.related.ForeignKey', [], {'to': u"orm['offer.Benefit']"}),
230
+            'condition': ('django.db.models.fields.related.ForeignKey', [], {'to': u"orm['offer.Condition']"}),
231
+            'date_created': ('django.db.models.fields.DateTimeField', [], {'auto_now_add': 'True', 'blank': 'True'}),
232
+            'description': ('django.db.models.fields.TextField', [], {'blank': 'True'}),
233
+            'end_datetime': ('django.db.models.fields.DateTimeField', [], {'null': 'True', 'blank': 'True'}),
234
+            u'id': ('django.db.models.fields.AutoField', [], {'primary_key': 'True'}),
235
+            'max_basket_applications': ('django.db.models.fields.PositiveIntegerField', [], {'null': 'True', 'blank': 'True'}),
236
+            'max_discount': ('django.db.models.fields.DecimalField', [], {'null': 'True', 'max_digits': '12', 'decimal_places': '2', 'blank': 'True'}),
237
+            'max_global_applications': ('django.db.models.fields.PositiveIntegerField', [], {'null': 'True', 'blank': 'True'}),
238
+            'max_user_applications': ('django.db.models.fields.PositiveIntegerField', [], {'null': 'True', 'blank': 'True'}),
239
+            'name': ('django.db.models.fields.CharField', [], {'unique': 'True', 'max_length': '128'}),
240
+            'num_applications': ('django.db.models.fields.PositiveIntegerField', [], {'default': '0'}),
241
+            'num_orders': ('django.db.models.fields.PositiveIntegerField', [], {'default': '0'}),
242
+            'offer_type': ('django.db.models.fields.CharField', [], {'default': "'Site'", 'max_length': '128'}),
243
+            'priority': ('django.db.models.fields.IntegerField', [], {'default': '0'}),
244
+            'redirect_url': ('oscar.models.fields.ExtendedURLField', [], {'max_length': '200', 'blank': 'True'}),
245
+            'slug': ('oscar.models.fields.autoslugfield.AutoSlugField', [], {'allow_duplicates': 'False', 'max_length': '128', 'separator': "u'-'", 'blank': 'True', 'unique': 'True', 'populate_from': "'name'", 'overwrite': 'False'}),
246
+            'start_datetime': ('django.db.models.fields.DateTimeField', [], {'null': 'True', 'blank': 'True'}),
247
+            'status': ('django.db.models.fields.CharField', [], {'default': "'Open'", 'max_length': '64'}),
248
+            'total_discount': ('django.db.models.fields.DecimalField', [], {'default': "'0.00'", 'max_digits': '12', 'decimal_places': '2'})
249
+        },
250
+        u'offer.range': {
251
+            'Meta': {'object_name': 'Range'},
252
+            'classes': ('django.db.models.fields.related.ManyToManyField', [], {'symmetrical': 'False', 'related_name': "'classes'", 'blank': 'True', 'to': u"orm['catalogue.ProductClass']"}),
253
+            'date_created': ('django.db.models.fields.DateTimeField', [], {'auto_now_add': 'True', 'blank': 'True'}),
254
+            'description': ('django.db.models.fields.TextField', [], {'blank': 'True'}),
255
+            'excluded_products': ('django.db.models.fields.related.ManyToManyField', [], {'symmetrical': 'False', 'related_name': "'excludes'", 'blank': 'True', 'to': u"orm['catalogue.Product']"}),
256
+            u'id': ('django.db.models.fields.AutoField', [], {'primary_key': 'True'}),
257
+            'included_categories': ('django.db.models.fields.related.ManyToManyField', [], {'symmetrical': 'False', 'related_name': "'includes'", 'blank': 'True', 'to': u"orm['catalogue.Category']"}),
258
+            'included_products': ('django.db.models.fields.related.ManyToManyField', [], {'symmetrical': 'False', 'related_name': "'includes'", 'blank': 'True', 'through': u"orm['offer.RangeProduct']", 'to': u"orm['catalogue.Product']"}),
259
+            'includes_all_products': ('django.db.models.fields.BooleanField', [], {'default': 'False'}),
260
+            'is_public': ('django.db.models.fields.BooleanField', [], {'default': 'False'}),
261
+            'name': ('django.db.models.fields.CharField', [], {'unique': 'True', 'max_length': '128'}),
262
+            'proxy_class': ('oscar.models.fields.NullCharField', [], {'default': 'None', 'max_length': '255', 'unique': 'True', 'null': 'True', 'blank': 'True'}),
263
+            'slug': ('django.db.models.fields.SlugField', [], {'max_length': '128', 'unique': 'True', 'null': 'True'})
264
+        },
265
+        u'offer.rangeproduct': {
266
+            'Meta': {'unique_together': "(('range', 'product'),)", 'object_name': 'RangeProduct'},
267
+            'display_order': ('django.db.models.fields.IntegerField', [], {'default': '0'}),
268
+            u'id': ('django.db.models.fields.AutoField', [], {'primary_key': 'True'}),
269
+            'product': ('django.db.models.fields.related.ForeignKey', [], {'to': u"orm['catalogue.Product']"}),
270
+            'range': ('django.db.models.fields.related.ForeignKey', [], {'to': u"orm['offer.Range']"})
271
+        },
272
+        u'order.billingaddress': {
273
+            'Meta': {'object_name': 'BillingAddress'},
274
+            'country': ('django.db.models.fields.related.ForeignKey', [], {'to': u"orm['address.Country']"}),
275
+            'first_name': ('django.db.models.fields.CharField', [], {'max_length': '255', 'blank': 'True'}),
276
+            u'id': ('django.db.models.fields.AutoField', [], {'primary_key': 'True'}),
277
+            'last_name': ('django.db.models.fields.CharField', [], {'max_length': '255', 'blank': 'True'}),
278
+            'line1': ('django.db.models.fields.CharField', [], {'max_length': '255'}),
279
+            'line2': ('django.db.models.fields.CharField', [], {'max_length': '255', 'blank': 'True'}),
280
+            'line3': ('django.db.models.fields.CharField', [], {'max_length': '255', 'blank': 'True'}),
281
+            'line4': ('django.db.models.fields.CharField', [], {'max_length': '255', 'blank': 'True'}),
282
+            'postcode': ('oscar.models.fields.UppercaseCharField', [], {'max_length': '64', 'blank': 'True'}),
283
+            'search_text': ('django.db.models.fields.CharField', [], {'max_length': '1000'}),
284
+            'state': ('django.db.models.fields.CharField', [], {'max_length': '255', 'blank': 'True'}),
285
+            'title': ('django.db.models.fields.CharField', [], {'max_length': '64', 'blank': 'True'})
286
+        },
287
+        u'order.communicationevent': {
288
+            'Meta': {'ordering': "['-date_created']", 'object_name': 'CommunicationEvent'},
289
+            'date_created': ('django.db.models.fields.DateTimeField', [], {'auto_now_add': 'True', 'blank': 'True'}),
290
+            'event_type': ('django.db.models.fields.related.ForeignKey', [], {'to': u"orm['customer.CommunicationEventType']"}),
291
+            u'id': ('django.db.models.fields.AutoField', [], {'primary_key': 'True'}),
292
+            'order': ('django.db.models.fields.related.ForeignKey', [], {'related_name': "'communication_events'", 'to': u"orm['order.Order']"})
293
+        },
294
+        u'order.line': {
295
+            'Meta': {'object_name': 'Line'},
296
+            'est_dispatch_date': ('django.db.models.fields.DateField', [], {'null': 'True', 'blank': 'True'}),
297
+            u'id': ('django.db.models.fields.AutoField', [], {'primary_key': 'True'}),
298
+            'line_price_before_discounts_excl_tax': ('django.db.models.fields.DecimalField', [], {'max_digits': '12', 'decimal_places': '2'}),
299
+            'line_price_before_discounts_incl_tax': ('django.db.models.fields.DecimalField', [], {'max_digits': '12', 'decimal_places': '2'}),
300
+            'line_price_excl_tax': ('django.db.models.fields.DecimalField', [], {'max_digits': '12', 'decimal_places': '2'}),
301
+            'line_price_incl_tax': ('django.db.models.fields.DecimalField', [], {'max_digits': '12', 'decimal_places': '2'}),
302
+            'order': ('django.db.models.fields.related.ForeignKey', [], {'related_name': "'lines'", 'to': u"orm['order.Order']"}),
303
+            'partner': ('django.db.models.fields.related.ForeignKey', [], {'blank': 'True', 'related_name': "'order_lines'", 'null': 'True', 'on_delete': 'models.SET_NULL', 'to': u"orm['partner.Partner']"}),
304
+            'partner_line_notes': ('django.db.models.fields.TextField', [], {'blank': 'True'}),
305
+            'partner_line_reference': ('django.db.models.fields.CharField', [], {'max_length': '128', 'blank': 'True'}),
306
+            'partner_name': ('django.db.models.fields.CharField', [], {'max_length': '128', 'blank': 'True'}),
307
+            'partner_sku': ('django.db.models.fields.CharField', [], {'max_length': '128'}),
308
+            'product': ('django.db.models.fields.related.ForeignKey', [], {'to': u"orm['catalogue.Product']", 'null': 'True', 'on_delete': 'models.SET_NULL', 'blank': 'True'}),
309
+            'quantity': ('django.db.models.fields.PositiveIntegerField', [], {'default': '1'}),
310
+            'status': ('django.db.models.fields.CharField', [], {'max_length': '255', 'blank': 'True'}),
311
+            'stockrecord': ('django.db.models.fields.related.ForeignKey', [], {'to': u"orm['partner.StockRecord']", 'null': 'True', 'on_delete': 'models.SET_NULL', 'blank': 'True'}),
312
+            'title': ('django.db.models.fields.CharField', [], {'max_length': '255'}),
313
+            'unit_cost_price': ('django.db.models.fields.DecimalField', [], {'null': 'True', 'max_digits': '12', 'decimal_places': '2', 'blank': 'True'}),
314
+            'unit_price_excl_tax': ('django.db.models.fields.DecimalField', [], {'null': 'True', 'max_digits': '12', 'decimal_places': '2', 'blank': 'True'}),
315
+            'unit_price_incl_tax': ('django.db.models.fields.DecimalField', [], {'null': 'True', 'max_digits': '12', 'decimal_places': '2', 'blank': 'True'}),
316
+            'unit_retail_price': ('django.db.models.fields.DecimalField', [], {'null': 'True', 'max_digits': '12', 'decimal_places': '2', 'blank': 'True'}),
317
+            'upc': ('django.db.models.fields.CharField', [], {'max_length': '128', 'null': 'True', 'blank': 'True'})
318
+        },
319
+        u'order.lineattribute': {
320
+            'Meta': {'object_name': 'LineAttribute'},
321
+            u'id': ('django.db.models.fields.AutoField', [], {'primary_key': 'True'}),
322
+            'line': ('django.db.models.fields.related.ForeignKey', [], {'related_name': "'attributes'", 'to': u"orm['order.Line']"}),
323
+            'option': ('django.db.models.fields.related.ForeignKey', [], {'related_name': "'line_attributes'", 'null': 'True', 'on_delete': 'models.SET_NULL', 'to': u"orm['catalogue.Option']"}),
324
+            'type': ('django.db.models.fields.CharField', [], {'max_length': '128'}),
325
+            'value': ('django.db.models.fields.CharField', [], {'max_length': '255'})
326
+        },
327
+        u'order.lineprice': {
328
+            'Meta': {'ordering': "('id',)", 'object_name': 'LinePrice'},
329
+            u'id': ('django.db.models.fields.AutoField', [], {'primary_key': 'True'}),
330
+            'line': ('django.db.models.fields.related.ForeignKey', [], {'related_name': "'prices'", 'to': u"orm['order.Line']"}),
331
+            'order': ('django.db.models.fields.related.ForeignKey', [], {'related_name': "'line_prices'", 'to': u"orm['order.Order']"}),
332
+            'price_excl_tax': ('django.db.models.fields.DecimalField', [], {'max_digits': '12', 'decimal_places': '2'}),
333
+            'price_incl_tax': ('django.db.models.fields.DecimalField', [], {'max_digits': '12', 'decimal_places': '2'}),
334
+            'quantity': ('django.db.models.fields.PositiveIntegerField', [], {'default': '1'}),
335
+            'shipping_excl_tax': ('django.db.models.fields.DecimalField', [], {'default': '0', 'max_digits': '12', 'decimal_places': '2'}),
336
+            'shipping_incl_tax': ('django.db.models.fields.DecimalField', [], {'default': '0', 'max_digits': '12', 'decimal_places': '2'})
337
+        },
338
+        u'order.order': {
339
+            'Meta': {'ordering': "['-date_placed']", 'object_name': 'Order'},
340
+            'basket': ('django.db.models.fields.related.ForeignKey', [], {'to': u"orm['basket.Basket']", 'null': 'True', 'on_delete': 'models.SET_NULL', 'blank': 'True'}),
341
+            'billing_address': ('django.db.models.fields.related.ForeignKey', [], {'to': u"orm['order.BillingAddress']", 'null': 'True', 'on_delete': 'models.SET_NULL', 'blank': 'True'}),
342
+            'currency': ('django.db.models.fields.CharField', [], {'default': "'GBP'", 'max_length': '12'}),
343
+            'date_placed': ('django.db.models.fields.DateTimeField', [], {'auto_now_add': 'True', 'db_index': 'True', 'blank': 'True'}),
344
+            'guest_email': ('django.db.models.fields.EmailField', [], {'max_length': '75', 'blank': 'True'}),
345
+            u'id': ('django.db.models.fields.AutoField', [], {'primary_key': 'True'}),
346
+            'number': ('django.db.models.fields.CharField', [], {'max_length': '128', 'db_index': 'True'}),
347
+            'shipping_address': ('django.db.models.fields.related.ForeignKey', [], {'to': u"orm['order.ShippingAddress']", 'null': 'True', 'on_delete': 'models.SET_NULL', 'blank': 'True'}),
348
+            'shipping_code': ('django.db.models.fields.CharField', [], {'default': "''", 'max_length': '128', 'blank': 'True'}),
349
+            'shipping_excl_tax': ('django.db.models.fields.DecimalField', [], {'default': '0', 'max_digits': '12', 'decimal_places': '2'}),
350
+            'shipping_incl_tax': ('django.db.models.fields.DecimalField', [], {'default': '0', 'max_digits': '12', 'decimal_places': '2'}),
351
+            'shipping_method': ('django.db.models.fields.CharField', [], {'max_length': '128', 'blank': 'True'}),
352
+            'site': ('django.db.models.fields.related.ForeignKey', [], {'to': u"orm['sites.Site']", 'null': 'True', 'on_delete': 'models.SET_NULL'}),
353
+            'status': ('django.db.models.fields.CharField', [], {'max_length': '100', 'blank': 'True'}),
354
+            'total_excl_tax': ('django.db.models.fields.DecimalField', [], {'max_digits': '12', 'decimal_places': '2'}),
355
+            'total_incl_tax': ('django.db.models.fields.DecimalField', [], {'max_digits': '12', 'decimal_places': '2'}),
356
+            'user': ('django.db.models.fields.related.ForeignKey', [], {'blank': 'True', 'related_name': "'orders'", 'null': 'True', 'on_delete': 'models.SET_NULL', 'to': u"orm['{0}']".format(AUTH_USER_MODEL)})
357
+        },
358
+        u'order.orderdiscount': {
359
+            'Meta': {'object_name': 'OrderDiscount'},
360
+            'amount': ('django.db.models.fields.DecimalField', [], {'default': '0', 'max_digits': '12', 'decimal_places': '2'}),
361
+            'category': ('django.db.models.fields.CharField', [], {'default': "'Basket'", 'max_length': '64'}),
362
+            'frequency': ('django.db.models.fields.PositiveIntegerField', [], {'null': 'True'}),
363
+            u'id': ('django.db.models.fields.AutoField', [], {'primary_key': 'True'}),
364
+            'message': ('django.db.models.fields.TextField', [], {'blank': 'True'}),
365
+            'offer_id': ('django.db.models.fields.PositiveIntegerField', [], {'null': 'True', 'blank': 'True'}),
366
+            'offer_name': ('django.db.models.fields.CharField', [], {'db_index': 'True', 'max_length': '128', 'blank': 'True'}),
367
+            'order': ('django.db.models.fields.related.ForeignKey', [], {'related_name': "'discounts'", 'to': u"orm['order.Order']"}),
368
+            'voucher_code': ('django.db.models.fields.CharField', [], {'db_index': 'True', 'max_length': '128', 'blank': 'True'}),
369
+            'voucher_id': ('django.db.models.fields.PositiveIntegerField', [], {'null': 'True', 'blank': 'True'})
370
+        },
371
+        u'order.ordernote': {
372
+            'Meta': {'object_name': 'OrderNote'},
373
+            'date_created': ('django.db.models.fields.DateTimeField', [], {'auto_now_add': 'True', 'blank': 'True'}),
374
+            'date_updated': ('django.db.models.fields.DateTimeField', [], {'auto_now': 'True', 'blank': 'True'}),
375
+            u'id': ('django.db.models.fields.AutoField', [], {'primary_key': 'True'}),
376
+            'message': ('django.db.models.fields.TextField', [], {}),
377
+            'note_type': ('django.db.models.fields.CharField', [], {'max_length': '128', 'blank': 'True'}),
378
+            'order': ('django.db.models.fields.related.ForeignKey', [], {'related_name': "'notes'", 'to': u"orm['order.Order']"}),
379
+            'user': ('django.db.models.fields.related.ForeignKey', [], {'to': u"orm['{0}']".format(AUTH_USER_MODEL), 'null': 'True'})
380
+        },
381
+        u'order.paymentevent': {
382
+            'Meta': {'ordering': "['-date_created']", 'object_name': 'PaymentEvent'},
383
+            'amount': ('django.db.models.fields.DecimalField', [], {'max_digits': '12', 'decimal_places': '2'}),
384
+            'date_created': ('django.db.models.fields.DateTimeField', [], {'auto_now_add': 'True', 'blank': 'True'}),
385
+            'event_type': ('django.db.models.fields.related.ForeignKey', [], {'to': u"orm['order.PaymentEventType']"}),
386
+            u'id': ('django.db.models.fields.AutoField', [], {'primary_key': 'True'}),
387
+            'lines': ('django.db.models.fields.related.ManyToManyField', [], {'to': u"orm['order.Line']", 'through': u"orm['order.PaymentEventQuantity']", 'symmetrical': 'False'}),
388
+            'order': ('django.db.models.fields.related.ForeignKey', [], {'related_name': "'payment_events'", 'to': u"orm['order.Order']"}),
389
+            'reference': ('django.db.models.fields.CharField', [], {'max_length': '128', 'blank': 'True'}),
390
+            'shipping_event': ('django.db.models.fields.related.ForeignKey', [], {'related_name': "'payment_events'", 'null': 'True', 'to': u"orm['order.ShippingEvent']"})
391
+        },
392
+        u'order.paymenteventquantity': {
393
+            'Meta': {'unique_together': "(('event', 'line'),)", 'object_name': 'PaymentEventQuantity'},
394
+            'event': ('django.db.models.fields.related.ForeignKey', [], {'related_name': "'line_quantities'", 'to': u"orm['order.PaymentEvent']"}),
395
+            u'id': ('django.db.models.fields.AutoField', [], {'primary_key': 'True'}),
396
+            'line': ('django.db.models.fields.related.ForeignKey', [], {'related_name': "'payment_event_quantities'", 'to': u"orm['order.Line']"}),
397
+            'quantity': ('django.db.models.fields.PositiveIntegerField', [], {})
398
+        },
399
+        u'order.paymenteventtype': {
400
+            'Meta': {'ordering': "('name',)", 'object_name': 'PaymentEventType'},
401
+            'code': ('oscar.models.fields.autoslugfield.AutoSlugField', [], {'allow_duplicates': 'False', 'max_length': '128', 'separator': "u'-'", 'blank': 'True', 'unique': 'True', 'populate_from': "'name'", 'overwrite': 'False'}),
402
+            u'id': ('django.db.models.fields.AutoField', [], {'primary_key': 'True'}),
403
+            'name': ('django.db.models.fields.CharField', [], {'unique': 'True', 'max_length': '128'})
404
+        },
405
+        u'order.shippingaddress': {
406
+            'Meta': {'object_name': 'ShippingAddress'},
407
+            'country': ('django.db.models.fields.related.ForeignKey', [], {'to': u"orm['address.Country']"}),
408
+            'first_name': ('django.db.models.fields.CharField', [], {'max_length': '255', 'blank': 'True'}),
409
+            u'id': ('django.db.models.fields.AutoField', [], {'primary_key': 'True'}),
410
+            'last_name': ('django.db.models.fields.CharField', [], {'max_length': '255', 'blank': 'True'}),
411
+            'line1': ('django.db.models.fields.CharField', [], {'max_length': '255'}),
412
+            'line2': ('django.db.models.fields.CharField', [], {'max_length': '255', 'blank': 'True'}),
413
+            'line3': ('django.db.models.fields.CharField', [], {'max_length': '255', 'blank': 'True'}),
414
+            'line4': ('django.db.models.fields.CharField', [], {'max_length': '255', 'blank': 'True'}),
415
+            'notes': ('django.db.models.fields.TextField', [], {'blank': 'True'}),
416
+            'phone_number': ('oscar.models.fields.PhoneNumberField', [], {'max_length': '128', 'blank': 'True'}),
417
+            'postcode': ('oscar.models.fields.UppercaseCharField', [], {'max_length': '64', 'blank': 'True'}),
418
+            'search_text': ('django.db.models.fields.CharField', [], {'max_length': '1000'}),
419
+            'state': ('django.db.models.fields.CharField', [], {'max_length': '255', 'blank': 'True'}),
420
+            'title': ('django.db.models.fields.CharField', [], {'max_length': '64', 'blank': 'True'})
421
+        },
422
+        u'order.shippingevent': {
423
+            'Meta': {'ordering': "['-date_created']", 'object_name': 'ShippingEvent'},
424
+            'date_created': ('django.db.models.fields.DateTimeField', [], {'auto_now_add': 'True', 'blank': 'True'}),
425
+            'event_type': ('django.db.models.fields.related.ForeignKey', [], {'to': u"orm['order.ShippingEventType']"}),
426
+            u'id': ('django.db.models.fields.AutoField', [], {'primary_key': 'True'}),
427
+            'lines': ('django.db.models.fields.related.ManyToManyField', [], {'related_name': "'shipping_events'", 'symmetrical': 'False', 'through': u"orm['order.ShippingEventQuantity']", 'to': u"orm['order.Line']"}),
428
+            'notes': ('django.db.models.fields.TextField', [], {'blank': 'True'}),
429
+            'order': ('django.db.models.fields.related.ForeignKey', [], {'related_name': "'shipping_events'", 'to': u"orm['order.Order']"})
430
+        },
431
+        u'order.shippingeventquantity': {
432
+            'Meta': {'unique_together': "(('event', 'line'),)", 'object_name': 'ShippingEventQuantity'},
433
+            'event': ('django.db.models.fields.related.ForeignKey', [], {'related_name': "'line_quantities'", 'to': u"orm['order.ShippingEvent']"}),
434
+            u'id': ('django.db.models.fields.AutoField', [], {'primary_key': 'True'}),
435
+            'line': ('django.db.models.fields.related.ForeignKey', [], {'related_name': "'shipping_event_quantities'", 'to': u"orm['order.Line']"}),
436
+            'quantity': ('django.db.models.fields.PositiveIntegerField', [], {})
437
+        },
438
+        u'order.shippingeventtype': {
439
+            'Meta': {'ordering': "('name',)", 'object_name': 'ShippingEventType'},
440
+            'code': ('oscar.models.fields.autoslugfield.AutoSlugField', [], {'allow_duplicates': 'False', 'max_length': '128', 'separator': "u'-'", 'blank': 'True', 'unique': 'True', 'populate_from': "'name'", 'overwrite': 'False'}),
441
+            u'id': ('django.db.models.fields.AutoField', [], {'primary_key': 'True'}),
442
+            'name': ('django.db.models.fields.CharField', [], {'unique': 'True', 'max_length': '255'})
443
+        },
444
+        u'partner.partner': {
445
+            'Meta': {'object_name': 'Partner'},
446
+            'code': ('oscar.models.fields.autoslugfield.AutoSlugField', [], {'allow_duplicates': 'False', 'max_length': '128', 'separator': "u'-'", 'blank': 'True', 'unique': 'True', 'populate_from': "'name'", 'overwrite': 'False'}),
447
+            u'id': ('django.db.models.fields.AutoField', [], {'primary_key': 'True'}),
448
+            'name': ('django.db.models.fields.CharField', [], {'max_length': '128', 'blank': 'True'}),
449
+            'users': ('django.db.models.fields.related.ManyToManyField', [], {'blank': 'True', 'related_name': "'partners'", 'null': 'True', 'symmetrical': 'False', 'to': u"orm['{0}']".format(AUTH_USER_MODEL)})
450
+        },
451
+        u'partner.stockrecord': {
452
+            'Meta': {'unique_together': "(('partner', 'partner_sku'),)", 'object_name': 'StockRecord'},
453
+            'cost_price': ('django.db.models.fields.DecimalField', [], {'null': 'True', 'max_digits': '12', 'decimal_places': '2', 'blank': 'True'}),
454
+            'date_created': ('django.db.models.fields.DateTimeField', [], {'auto_now_add': 'True', 'blank': 'True'}),
455
+            'date_updated': ('django.db.models.fields.DateTimeField', [], {'auto_now': 'True', 'db_index': 'True', 'blank': 'True'}),
456
+            u'id': ('django.db.models.fields.AutoField', [], {'primary_key': 'True'}),
457
+            'low_stock_threshold': ('django.db.models.fields.PositiveIntegerField', [], {'null': 'True', 'blank': 'True'}),
458
+            'num_allocated': ('django.db.models.fields.IntegerField', [], {'null': 'True', 'blank': 'True'}),
459
+            'num_in_stock': ('django.db.models.fields.PositiveIntegerField', [], {'null': 'True', 'blank': 'True'}),
460
+            'partner': ('django.db.models.fields.related.ForeignKey', [], {'related_name': "'stockrecords'", 'to': u"orm['partner.Partner']"}),
461
+            'partner_sku': ('django.db.models.fields.CharField', [], {'max_length': '128'}),
462
+            'price_currency': ('django.db.models.fields.CharField', [], {'default': "'GBP'", 'max_length': '12'}),
463
+            'price_excl_tax': ('django.db.models.fields.DecimalField', [], {'null': 'True', 'max_digits': '12', 'decimal_places': '2', 'blank': 'True'}),
464
+            'price_retail': ('django.db.models.fields.DecimalField', [], {'null': 'True', 'max_digits': '12', 'decimal_places': '2', 'blank': 'True'}),
465
+            'product': ('django.db.models.fields.related.ForeignKey', [], {'related_name': "'stockrecords'", 'to': u"orm['catalogue.Product']"})
466
+        },
467
+        u'sites.site': {
468
+            'Meta': {'ordering': "(u'domain',)", 'object_name': 'Site', 'db_table': "u'django_site'"},
469
+            'domain': ('django.db.models.fields.CharField', [], {'max_length': '100'}),
470
+            u'id': ('django.db.models.fields.AutoField', [], {'primary_key': 'True'}),
471
+            'name': ('django.db.models.fields.CharField', [], {'max_length': '50'})
472
+        },
473
+        u'voucher.voucher': {
474
+            'Meta': {'object_name': 'Voucher'},
475
+            'code': ('django.db.models.fields.CharField', [], {'unique': 'True', 'max_length': '128', 'db_index': 'True'}),
476
+            'date_created': ('django.db.models.fields.DateField', [], {'auto_now_add': 'True', 'blank': 'True'}),
477
+            'end_datetime': ('django.db.models.fields.DateTimeField', [], {}),
478
+            u'id': ('django.db.models.fields.AutoField', [], {'primary_key': 'True'}),
479
+            'name': ('django.db.models.fields.CharField', [], {'max_length': '128'}),
480
+            'num_basket_additions': ('django.db.models.fields.PositiveIntegerField', [], {'default': '0'}),
481
+            'num_orders': ('django.db.models.fields.PositiveIntegerField', [], {'default': '0'}),
482
+            'offers': ('django.db.models.fields.related.ManyToManyField', [], {'related_name': "'vouchers'", 'symmetrical': 'False', 'to': u"orm['offer.ConditionalOffer']"}),
483
+            'start_datetime': ('django.db.models.fields.DateTimeField', [], {}),
484
+            'total_discount': ('django.db.models.fields.DecimalField', [], {'default': "'0.00'", 'max_digits': '12', 'decimal_places': '2'}),
485
+            'usage': ('django.db.models.fields.CharField', [], {'default': "'Multi-use'", 'max_length': '128'})
486
+        }
487
+    }
488
+
489
+    complete_apps = ['order']

+ 236
- 0
oscar/apps/promotions/migrations/0006_auto__add_unique_orderedproduct_list_product.py Прегледај датотеку

@@ -0,0 +1,236 @@
1
+# -*- coding: utf-8 -*-
2
+from south.utils import datetime_utils as datetime
3
+from south.db import db
4
+from south.v2 import SchemaMigration
5
+from django.db import models
6
+
7
+
8
+class Migration(SchemaMigration):
9
+
10
+    def forwards(self, orm):
11
+        # Adding unique constraint on 'OrderedProduct', fields ['list', 'product']
12
+        db.create_unique(u'promotions_orderedproduct', ['list_id', 'product_id'])
13
+
14
+
15
+    def backwards(self, orm):
16
+        # Removing unique constraint on 'OrderedProduct', fields ['list', 'product']
17
+        db.delete_unique(u'promotions_orderedproduct', ['list_id', 'product_id'])
18
+
19
+
20
+    models = {
21
+        u'catalogue.attributeentity': {
22
+            'Meta': {'object_name': 'AttributeEntity'},
23
+            u'id': ('django.db.models.fields.AutoField', [], {'primary_key': 'True'}),
24
+            'name': ('django.db.models.fields.CharField', [], {'max_length': '255'}),
25
+            'slug': ('django.db.models.fields.SlugField', [], {'max_length': '255', 'blank': 'True'}),
26
+            'type': ('django.db.models.fields.related.ForeignKey', [], {'related_name': "'entities'", 'to': u"orm['catalogue.AttributeEntityType']"})
27
+        },
28
+        u'catalogue.attributeentitytype': {
29
+            'Meta': {'object_name': 'AttributeEntityType'},
30
+            u'id': ('django.db.models.fields.AutoField', [], {'primary_key': 'True'}),
31
+            'name': ('django.db.models.fields.CharField', [], {'max_length': '255'}),
32
+            'slug': ('django.db.models.fields.SlugField', [], {'max_length': '255', 'blank': 'True'})
33
+        },
34
+        u'catalogue.attributeoption': {
35
+            'Meta': {'object_name': 'AttributeOption'},
36
+            'group': ('django.db.models.fields.related.ForeignKey', [], {'related_name': "'options'", 'to': u"orm['catalogue.AttributeOptionGroup']"}),
37
+            u'id': ('django.db.models.fields.AutoField', [], {'primary_key': 'True'}),
38
+            'option': ('django.db.models.fields.CharField', [], {'max_length': '255'})
39
+        },
40
+        u'catalogue.attributeoptiongroup': {
41
+            'Meta': {'object_name': 'AttributeOptionGroup'},
42
+            u'id': ('django.db.models.fields.AutoField', [], {'primary_key': 'True'}),
43
+            'name': ('django.db.models.fields.CharField', [], {'max_length': '128'})
44
+        },
45
+        u'catalogue.category': {
46
+            'Meta': {'ordering': "['full_name']", 'object_name': 'Category'},
47
+            'depth': ('django.db.models.fields.PositiveIntegerField', [], {}),
48
+            'description': ('django.db.models.fields.TextField', [], {'blank': 'True'}),
49
+            'full_name': ('django.db.models.fields.CharField', [], {'max_length': '255', 'db_index': 'True'}),
50
+            u'id': ('django.db.models.fields.AutoField', [], {'primary_key': 'True'}),
51
+            'image': ('django.db.models.fields.files.ImageField', [], {'max_length': '255', 'null': 'True', 'blank': 'True'}),
52
+            'name': ('django.db.models.fields.CharField', [], {'max_length': '255', 'db_index': 'True'}),
53
+            'numchild': ('django.db.models.fields.PositiveIntegerField', [], {'default': '0'}),
54
+            'path': ('django.db.models.fields.CharField', [], {'unique': 'True', 'max_length': '255'}),
55
+            'slug': ('django.db.models.fields.SlugField', [], {'max_length': '255'})
56
+        },
57
+        u'catalogue.option': {
58
+            'Meta': {'object_name': 'Option'},
59
+            'code': ('oscar.models.fields.autoslugfield.AutoSlugField', [], {'allow_duplicates': 'False', 'max_length': '128', 'separator': "u'-'", 'blank': 'True', 'unique': 'True', 'populate_from': "'name'", 'overwrite': 'False'}),
60
+            u'id': ('django.db.models.fields.AutoField', [], {'primary_key': 'True'}),
61
+            'name': ('django.db.models.fields.CharField', [], {'max_length': '128'}),
62
+            'type': ('django.db.models.fields.CharField', [], {'default': "'Required'", 'max_length': '128'})
63
+        },
64
+        u'catalogue.product': {
65
+            'Meta': {'ordering': "['-date_created']", 'object_name': 'Product'},
66
+            'attributes': ('django.db.models.fields.related.ManyToManyField', [], {'to': u"orm['catalogue.ProductAttribute']", 'through': u"orm['catalogue.ProductAttributeValue']", 'symmetrical': 'False'}),
67
+            'categories': ('django.db.models.fields.related.ManyToManyField', [], {'to': u"orm['catalogue.Category']", 'through': u"orm['catalogue.ProductCategory']", 'symmetrical': 'False'}),
68
+            'date_created': ('django.db.models.fields.DateTimeField', [], {'auto_now_add': 'True', 'blank': 'True'}),
69
+            'date_updated': ('django.db.models.fields.DateTimeField', [], {'auto_now': 'True', 'db_index': 'True', 'blank': 'True'}),
70
+            'description': ('django.db.models.fields.TextField', [], {'blank': 'True'}),
71
+            u'id': ('django.db.models.fields.AutoField', [], {'primary_key': 'True'}),
72
+            'is_discountable': ('django.db.models.fields.BooleanField', [], {'default': 'True'}),
73
+            'parent': ('django.db.models.fields.related.ForeignKey', [], {'blank': 'True', 'related_name': "'variants'", 'null': 'True', 'to': u"orm['catalogue.Product']"}),
74
+            'product_class': ('django.db.models.fields.related.ForeignKey', [], {'related_name': "'products'", 'null': 'True', 'on_delete': 'models.PROTECT', 'to': u"orm['catalogue.ProductClass']"}),
75
+            'product_options': ('django.db.models.fields.related.ManyToManyField', [], {'to': u"orm['catalogue.Option']", 'symmetrical': 'False', 'blank': 'True'}),
76
+            'rating': ('django.db.models.fields.FloatField', [], {'null': 'True'}),
77
+            'recommended_products': ('django.db.models.fields.related.ManyToManyField', [], {'to': u"orm['catalogue.Product']", 'symmetrical': 'False', 'through': u"orm['catalogue.ProductRecommendation']", 'blank': 'True'}),
78
+            'related_products': ('django.db.models.fields.related.ManyToManyField', [], {'symmetrical': 'False', 'related_name': "'relations'", 'blank': 'True', 'to': u"orm['catalogue.Product']"}),
79
+            'score': ('django.db.models.fields.FloatField', [], {'default': '0.0', 'db_index': 'True'}),
80
+            'slug': ('django.db.models.fields.SlugField', [], {'max_length': '255'}),
81
+            'title': ('django.db.models.fields.CharField', [], {'max_length': '255', 'blank': 'True'}),
82
+            'upc': ('oscar.models.fields.NullCharField', [], {'max_length': '64', 'unique': 'True', 'null': 'True', 'blank': 'True'})
83
+        },
84
+        u'catalogue.productattribute': {
85
+            'Meta': {'ordering': "['code']", 'object_name': 'ProductAttribute'},
86
+            'code': ('django.db.models.fields.SlugField', [], {'max_length': '128'}),
87
+            'entity_type': ('django.db.models.fields.related.ForeignKey', [], {'to': u"orm['catalogue.AttributeEntityType']", 'null': 'True', 'blank': 'True'}),
88
+            u'id': ('django.db.models.fields.AutoField', [], {'primary_key': 'True'}),
89
+            'name': ('django.db.models.fields.CharField', [], {'max_length': '128'}),
90
+            'option_group': ('django.db.models.fields.related.ForeignKey', [], {'to': u"orm['catalogue.AttributeOptionGroup']", 'null': 'True', 'blank': 'True'}),
91
+            'product_class': ('django.db.models.fields.related.ForeignKey', [], {'blank': 'True', 'related_name': "'attributes'", 'null': 'True', 'to': u"orm['catalogue.ProductClass']"}),
92
+            'required': ('django.db.models.fields.BooleanField', [], {'default': 'False'}),
93
+            'type': ('django.db.models.fields.CharField', [], {'default': "'text'", 'max_length': '20'})
94
+        },
95
+        u'catalogue.productattributevalue': {
96
+            'Meta': {'unique_together': "(('attribute', 'product'),)", 'object_name': 'ProductAttributeValue'},
97
+            'attribute': ('django.db.models.fields.related.ForeignKey', [], {'to': u"orm['catalogue.ProductAttribute']"}),
98
+            u'id': ('django.db.models.fields.AutoField', [], {'primary_key': 'True'}),
99
+            'product': ('django.db.models.fields.related.ForeignKey', [], {'related_name': "'attribute_values'", 'to': u"orm['catalogue.Product']"}),
100
+            'value_boolean': ('django.db.models.fields.NullBooleanField', [], {'null': 'True', 'blank': 'True'}),
101
+            'value_date': ('django.db.models.fields.DateField', [], {'null': 'True', 'blank': 'True'}),
102
+            'value_entity': ('django.db.models.fields.related.ForeignKey', [], {'to': u"orm['catalogue.AttributeEntity']", 'null': 'True', 'blank': 'True'}),
103
+            'value_file': ('django.db.models.fields.files.FileField', [], {'max_length': '255', 'null': 'True', 'blank': 'True'}),
104
+            'value_float': ('django.db.models.fields.FloatField', [], {'null': 'True', 'blank': 'True'}),
105
+            'value_image': ('django.db.models.fields.files.ImageField', [], {'max_length': '255', 'null': 'True', 'blank': 'True'}),
106
+            'value_integer': ('django.db.models.fields.IntegerField', [], {'null': 'True', 'blank': 'True'}),
107
+            'value_option': ('django.db.models.fields.related.ForeignKey', [], {'to': u"orm['catalogue.AttributeOption']", 'null': 'True', 'blank': 'True'}),
108
+            'value_richtext': ('django.db.models.fields.TextField', [], {'null': 'True', 'blank': 'True'}),
109
+            'value_text': ('django.db.models.fields.CharField', [], {'max_length': '255', 'null': 'True', 'blank': 'True'})
110
+        },
111
+        u'catalogue.productcategory': {
112
+            'Meta': {'ordering': "['product', 'category']", 'unique_together': "(('product', 'category'),)", 'object_name': 'ProductCategory'},
113
+            'category': ('django.db.models.fields.related.ForeignKey', [], {'to': u"orm['catalogue.Category']"}),
114
+            u'id': ('django.db.models.fields.AutoField', [], {'primary_key': 'True'}),
115
+            'product': ('django.db.models.fields.related.ForeignKey', [], {'to': u"orm['catalogue.Product']"})
116
+        },
117
+        u'catalogue.productclass': {
118
+            'Meta': {'ordering': "['name']", 'object_name': 'ProductClass'},
119
+            u'id': ('django.db.models.fields.AutoField', [], {'primary_key': 'True'}),
120
+            'name': ('django.db.models.fields.CharField', [], {'max_length': '128'}),
121
+            'options': ('django.db.models.fields.related.ManyToManyField', [], {'to': u"orm['catalogue.Option']", 'symmetrical': 'False', 'blank': 'True'}),
122
+            'requires_shipping': ('django.db.models.fields.BooleanField', [], {'default': 'True'}),
123
+            'slug': ('oscar.models.fields.autoslugfield.AutoSlugField', [], {'allow_duplicates': 'False', 'max_length': '128', 'separator': "u'-'", 'blank': 'True', 'unique': 'True', 'populate_from': "'name'", 'overwrite': 'False'}),
124
+            'track_stock': ('django.db.models.fields.BooleanField', [], {'default': 'True'})
125
+        },
126
+        u'catalogue.productrecommendation': {
127
+            'Meta': {'unique_together': "(('primary', 'recommendation'),)", 'object_name': 'ProductRecommendation'},
128
+            u'id': ('django.db.models.fields.AutoField', [], {'primary_key': 'True'}),
129
+            'primary': ('django.db.models.fields.related.ForeignKey', [], {'related_name': "'primary_recommendations'", 'to': u"orm['catalogue.Product']"}),
130
+            'ranking': ('django.db.models.fields.PositiveSmallIntegerField', [], {'default': '0'}),
131
+            'recommendation': ('django.db.models.fields.related.ForeignKey', [], {'to': u"orm['catalogue.Product']"})
132
+        },
133
+        u'contenttypes.contenttype': {
134
+            'Meta': {'ordering': "('name',)", 'unique_together': "(('app_label', 'model'),)", 'object_name': 'ContentType', 'db_table': "'django_content_type'"},
135
+            'app_label': ('django.db.models.fields.CharField', [], {'max_length': '100'}),
136
+            u'id': ('django.db.models.fields.AutoField', [], {'primary_key': 'True'}),
137
+            'model': ('django.db.models.fields.CharField', [], {'max_length': '100'}),
138
+            'name': ('django.db.models.fields.CharField', [], {'max_length': '100'})
139
+        },
140
+        u'promotions.automaticproductlist': {
141
+            'Meta': {'object_name': 'AutomaticProductList'},
142
+            'date_created': ('django.db.models.fields.DateTimeField', [], {'auto_now_add': 'True', 'blank': 'True'}),
143
+            'description': ('django.db.models.fields.TextField', [], {'blank': 'True'}),
144
+            u'id': ('django.db.models.fields.AutoField', [], {'primary_key': 'True'}),
145
+            'link_text': ('django.db.models.fields.CharField', [], {'max_length': '255', 'blank': 'True'}),
146
+            'link_url': ('oscar.models.fields.ExtendedURLField', [], {'max_length': '200', 'blank': 'True'}),
147
+            'method': ('django.db.models.fields.CharField', [], {'max_length': '128'}),
148
+            'name': ('django.db.models.fields.CharField', [], {'max_length': '255'}),
149
+            'num_products': ('django.db.models.fields.PositiveSmallIntegerField', [], {'default': '4'})
150
+        },
151
+        u'promotions.handpickedproductlist': {
152
+            'Meta': {'object_name': 'HandPickedProductList'},
153
+            'date_created': ('django.db.models.fields.DateTimeField', [], {'auto_now_add': 'True', 'blank': 'True'}),
154
+            'description': ('django.db.models.fields.TextField', [], {'blank': 'True'}),
155
+            u'id': ('django.db.models.fields.AutoField', [], {'primary_key': 'True'}),
156
+            'link_text': ('django.db.models.fields.CharField', [], {'max_length': '255', 'blank': 'True'}),
157
+            'link_url': ('oscar.models.fields.ExtendedURLField', [], {'max_length': '200', 'blank': 'True'}),
158
+            'name': ('django.db.models.fields.CharField', [], {'max_length': '255'}),
159
+            'products': ('django.db.models.fields.related.ManyToManyField', [], {'symmetrical': 'False', 'to': u"orm['catalogue.Product']", 'null': 'True', 'through': u"orm['promotions.OrderedProduct']", 'blank': 'True'})
160
+        },
161
+        u'promotions.image': {
162
+            'Meta': {'object_name': 'Image'},
163
+            'date_created': ('django.db.models.fields.DateTimeField', [], {'auto_now_add': 'True', 'blank': 'True'}),
164
+            u'id': ('django.db.models.fields.AutoField', [], {'primary_key': 'True'}),
165
+            'image': ('django.db.models.fields.files.ImageField', [], {'max_length': '255'}),
166
+            'link_url': ('oscar.models.fields.ExtendedURLField', [], {'max_length': '200', 'blank': 'True'}),
167
+            'name': ('django.db.models.fields.CharField', [], {'max_length': '128'})
168
+        },
169
+        u'promotions.keywordpromotion': {
170
+            'Meta': {'object_name': 'KeywordPromotion'},
171
+            'clicks': ('django.db.models.fields.PositiveIntegerField', [], {'default': '0'}),
172
+            'content_type': ('django.db.models.fields.related.ForeignKey', [], {'to': u"orm['contenttypes.ContentType']"}),
173
+            'date_created': ('django.db.models.fields.DateTimeField', [], {'auto_now_add': 'True', 'blank': 'True'}),
174
+            'display_order': ('django.db.models.fields.PositiveIntegerField', [], {'default': '0'}),
175
+            'filter': ('django.db.models.fields.CharField', [], {'max_length': '200', 'blank': 'True'}),
176
+            u'id': ('django.db.models.fields.AutoField', [], {'primary_key': 'True'}),
177
+            'keyword': ('django.db.models.fields.CharField', [], {'max_length': '200'}),
178
+            'object_id': ('django.db.models.fields.PositiveIntegerField', [], {}),
179
+            'position': ('django.db.models.fields.CharField', [], {'max_length': '100'})
180
+        },
181
+        u'promotions.multiimage': {
182
+            'Meta': {'object_name': 'MultiImage'},
183
+            'date_created': ('django.db.models.fields.DateTimeField', [], {'auto_now_add': 'True', 'blank': 'True'}),
184
+            u'id': ('django.db.models.fields.AutoField', [], {'primary_key': 'True'}),
185
+            'images': ('django.db.models.fields.related.ManyToManyField', [], {'symmetrical': 'False', 'to': u"orm['promotions.Image']", 'null': 'True', 'blank': 'True'}),
186
+            'name': ('django.db.models.fields.CharField', [], {'max_length': '128'})
187
+        },
188
+        u'promotions.orderedproduct': {
189
+            'Meta': {'ordering': "('display_order',)", 'unique_together': "(('list', 'product'),)", 'object_name': 'OrderedProduct'},
190
+            'display_order': ('django.db.models.fields.PositiveIntegerField', [], {'default': '0'}),
191
+            u'id': ('django.db.models.fields.AutoField', [], {'primary_key': 'True'}),
192
+            'list': ('django.db.models.fields.related.ForeignKey', [], {'to': u"orm['promotions.HandPickedProductList']"}),
193
+            'product': ('django.db.models.fields.related.ForeignKey', [], {'to': u"orm['catalogue.Product']"})
194
+        },
195
+        u'promotions.orderedproductlist': {
196
+            'Meta': {'ordering': "('display_order',)", 'object_name': 'OrderedProductList', '_ormbases': [u'promotions.HandPickedProductList']},
197
+            'display_order': ('django.db.models.fields.PositiveIntegerField', [], {'default': '0'}),
198
+            u'handpickedproductlist_ptr': ('django.db.models.fields.related.OneToOneField', [], {'to': u"orm['promotions.HandPickedProductList']", 'unique': 'True', 'primary_key': 'True'}),
199
+            'tabbed_block': ('django.db.models.fields.related.ForeignKey', [], {'related_name': "'tabs'", 'to': u"orm['promotions.TabbedBlock']"})
200
+        },
201
+        u'promotions.pagepromotion': {
202
+            'Meta': {'object_name': 'PagePromotion'},
203
+            'clicks': ('django.db.models.fields.PositiveIntegerField', [], {'default': '0'}),
204
+            'content_type': ('django.db.models.fields.related.ForeignKey', [], {'to': u"orm['contenttypes.ContentType']"}),
205
+            'date_created': ('django.db.models.fields.DateTimeField', [], {'auto_now_add': 'True', 'blank': 'True'}),
206
+            'display_order': ('django.db.models.fields.PositiveIntegerField', [], {'default': '0'}),
207
+            u'id': ('django.db.models.fields.AutoField', [], {'primary_key': 'True'}),
208
+            'object_id': ('django.db.models.fields.PositiveIntegerField', [], {}),
209
+            'page_url': ('oscar.models.fields.ExtendedURLField', [], {'max_length': '128', 'db_index': 'True'}),
210
+            'position': ('django.db.models.fields.CharField', [], {'max_length': '100'})
211
+        },
212
+        u'promotions.rawhtml': {
213
+            'Meta': {'object_name': 'RawHTML'},
214
+            'body': ('django.db.models.fields.TextField', [], {}),
215
+            'date_created': ('django.db.models.fields.DateTimeField', [], {'auto_now_add': 'True', 'blank': 'True'}),
216
+            'display_type': ('django.db.models.fields.CharField', [], {'max_length': '128', 'blank': 'True'}),
217
+            u'id': ('django.db.models.fields.AutoField', [], {'primary_key': 'True'}),
218
+            'name': ('django.db.models.fields.CharField', [], {'max_length': '128'})
219
+        },
220
+        u'promotions.singleproduct': {
221
+            'Meta': {'object_name': 'SingleProduct'},
222
+            'date_created': ('django.db.models.fields.DateTimeField', [], {'auto_now_add': 'True', 'blank': 'True'}),
223
+            'description': ('django.db.models.fields.TextField', [], {'blank': 'True'}),
224
+            u'id': ('django.db.models.fields.AutoField', [], {'primary_key': 'True'}),
225
+            'name': ('django.db.models.fields.CharField', [], {'max_length': '128'}),
226
+            'product': ('django.db.models.fields.related.ForeignKey', [], {'to': u"orm['catalogue.Product']"})
227
+        },
228
+        u'promotions.tabbedblock': {
229
+            'Meta': {'object_name': 'TabbedBlock'},
230
+            'date_created': ('django.db.models.fields.DateTimeField', [], {'auto_now_add': 'True', 'blank': 'True'}),
231
+            u'id': ('django.db.models.fields.AutoField', [], {'primary_key': 'True'}),
232
+            'name': ('django.db.models.fields.CharField', [], {'max_length': '255'})
233
+        }
234
+    }
235
+
236
+    complete_apps = ['promotions']

+ 1
- 0
oscar/apps/promotions/models.py Прегледај датотеку

@@ -279,6 +279,7 @@ class OrderedProduct(models.Model):
279 279
         ordering = ('display_order',)
280 280
         verbose_name = _("Ordered Product")
281 281
         verbose_name_plural = _("Ordered Product")
282
+        unique_together = ('list', 'product')
282 283
 
283 284
 
284 285
 class AutomaticProductList(AbstractProductList):

Loading…
Откажи
Сачувај