Нет описания

0001_initial.py 23KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368
  1. # Generated by Django 4.2.24 on 2025-09-24 06:05
  2. from django.conf import settings
  3. from django.db import migrations, models
  4. import django.db.models.deletion
  5. import django.utils.timezone
  6. import markdownfield.models
  7. class Migration(migrations.Migration):
  8. initial = True
  9. dependencies = [
  10. migrations.swappable_dependency(settings.AUTH_USER_MODEL),
  11. ('orgs', '0006_delete_providedservice'),
  12. ('contenttypes', '0002_remove_content_type_name'),
  13. ]
  14. operations = [
  15. migrations.CreateModel(
  16. name='Customer',
  17. fields=[
  18. ('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
  19. ('created_at', models.DateTimeField(auto_now_add=True)),
  20. ('updated_at', models.DateTimeField(auto_now=True)),
  21. ('name', models.CharField(max_length=255)),
  22. ('email', models.EmailField(blank=True, max_length=254)),
  23. ('phone', models.CharField(blank=True, max_length=64)),
  24. ('billing_address', models.TextField(blank=True)),
  25. ('organization', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, related_name='customers', to='orgs.organization')),
  26. ],
  27. options={
  28. 'abstract': False,
  29. },
  30. ),
  31. migrations.CreateModel(
  32. name='CustomerSite',
  33. fields=[
  34. ('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
  35. ('created_at', models.DateTimeField(auto_now_add=True)),
  36. ('updated_at', models.DateTimeField(auto_now=True)),
  37. ('name', models.CharField(blank=True, max_length=255)),
  38. ('address', models.TextField()),
  39. ('contact_name', models.CharField(blank=True, max_length=255)),
  40. ('contact_phone', models.CharField(blank=True, max_length=64)),
  41. ('contact_email', models.EmailField(blank=True, max_length=254)),
  42. ('customer', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, related_name='sites', to='recycle_core.customer')),
  43. ],
  44. options={
  45. 'abstract': False,
  46. },
  47. ),
  48. migrations.CreateModel(
  49. name='Material',
  50. fields=[
  51. ('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
  52. ('created_at', models.DateTimeField(auto_now_add=True)),
  53. ('updated_at', models.DateTimeField(auto_now=True)),
  54. ('category', models.CharField(choices=[('Plastics', 'Plastics'), ('Metals', 'Metals'), ('Paper', 'Paper'), ('Glass', 'Glass'), ('Electronics', 'Electronics'), ('Wood', 'Wood'), ('Rubber', 'Rubber'), ('Textiles', 'Textiles'), ('Organic', 'Organic'), ('Mixed', 'Mixed')], max_length=64)),
  55. ('name', models.CharField(max_length=255)),
  56. ('code', models.CharField(blank=True, max_length=64)),
  57. ('default_unit', models.CharField(choices=[('kg', 'Kilogram'), ('lb', 'Pound'), ('pcs', 'Pieces')], default='kg', max_length=8)),
  58. ('organization', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, related_name='materials', to='orgs.organization')),
  59. ],
  60. ),
  61. migrations.CreateModel(
  62. name='PickupOrder',
  63. fields=[
  64. ('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
  65. ('created_at', models.DateTimeField(auto_now_add=True)),
  66. ('updated_at', models.DateTimeField(auto_now=True)),
  67. ('status', models.CharField(choices=[('requested', 'Requested'), ('scheduled', 'Scheduled'), ('en_route', 'En Route'), ('collecting', 'Collecting'), ('at_facility', 'At Facility'), ('weighed', 'Weighed'), ('invoiced', 'Invoiced'), ('completed', 'Completed'), ('canceled', 'Canceled')], db_index=True, default='requested', max_length=16)),
  68. ('scheduled_at', models.DateTimeField(blank=True, null=True)),
  69. ('completed_at', models.DateTimeField(blank=True, null=True)),
  70. ('notes', models.TextField(blank=True)),
  71. ('assigned_driver', models.ForeignKey(blank=True, null=True, on_delete=django.db.models.deletion.SET_NULL, related_name='assigned_pickups', to=settings.AUTH_USER_MODEL)),
  72. ('created_by', models.ForeignKey(blank=True, null=True, on_delete=django.db.models.deletion.SET_NULL, related_name='created_pickups', to=settings.AUTH_USER_MODEL)),
  73. ('customer', models.ForeignKey(on_delete=django.db.models.deletion.PROTECT, related_name='pickup_orders', to='recycle_core.customer')),
  74. ('organization', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, related_name='pickup_orders', to='orgs.organization')),
  75. ('site', models.ForeignKey(on_delete=django.db.models.deletion.PROTECT, related_name='pickup_orders', to='recycle_core.customersite')),
  76. ],
  77. options={
  78. 'abstract': False,
  79. },
  80. ),
  81. migrations.CreateModel(
  82. name='PriceList',
  83. fields=[
  84. ('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
  85. ('created_at', models.DateTimeField(auto_now_add=True)),
  86. ('updated_at', models.DateTimeField(auto_now=True)),
  87. ('name', models.CharField(max_length=255)),
  88. ('currency_code', models.CharField(default='USD', max_length=8)),
  89. ('effective_from', models.DateField(blank=True, null=True)),
  90. ('effective_to', models.DateField(blank=True, null=True)),
  91. ('organization', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, related_name='price_lists', to='orgs.organization')),
  92. ],
  93. options={
  94. 'abstract': False,
  95. },
  96. ),
  97. migrations.CreateModel(
  98. name='ScrapListing',
  99. fields=[
  100. ('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
  101. ('created_at', models.DateTimeField(auto_now_add=True)),
  102. ('updated_at', models.DateTimeField(auto_now=True)),
  103. ('title', models.CharField(max_length=255)),
  104. ('description', models.TextField(blank=True)),
  105. ('auction_type', models.CharField(choices=[('open', 'Open'), ('sealed', 'Sealed')], default='open', max_length=16)),
  106. ('currency_code', models.CharField(default='USD', max_length=8)),
  107. ('reserve_price', models.DecimalField(blank=True, decimal_places=2, max_digits=14, null=True)),
  108. ('min_increment', models.DecimalField(blank=True, decimal_places=2, max_digits=12, null=True)),
  109. ('starts_at', models.DateTimeField(blank=True, null=True)),
  110. ('ends_at', models.DateTimeField(blank=True, null=True)),
  111. ('status', models.CharField(choices=[('draft', 'Draft'), ('open', 'Open'), ('closed', 'Closed'), ('awarded', 'Awarded'), ('canceled', 'Canceled')], db_index=True, default='draft', max_length=16)),
  112. ('is_public', models.BooleanField(default=True)),
  113. ('created_by', models.ForeignKey(blank=True, null=True, on_delete=django.db.models.deletion.SET_NULL, related_name='created_listings', to=settings.AUTH_USER_MODEL)),
  114. ('customer', models.ForeignKey(on_delete=django.db.models.deletion.PROTECT, related_name='scrap_listings', to='recycle_core.customer')),
  115. ('organization', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, related_name='scrap_listings', to='orgs.organization')),
  116. ('site', models.ForeignKey(on_delete=django.db.models.deletion.PROTECT, related_name='scrap_listings', to='recycle_core.customersite')),
  117. ],
  118. options={
  119. 'abstract': False,
  120. },
  121. ),
  122. migrations.CreateModel(
  123. name='WeighTicket',
  124. fields=[
  125. ('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
  126. ('created_at', models.DateTimeField(auto_now_add=True)),
  127. ('updated_at', models.DateTimeField(auto_now=True)),
  128. ('ticket_number', models.CharField(blank=True, max_length=64)),
  129. ('gross_weight', models.DecimalField(decimal_places=3, max_digits=12)),
  130. ('tare_weight', models.DecimalField(decimal_places=3, max_digits=12)),
  131. ('net_weight', models.DecimalField(decimal_places=3, max_digits=12)),
  132. ('unit', models.CharField(choices=[('kg', 'Kilogram'), ('lb', 'Pound'), ('pcs', 'Pieces')], default='kg', max_length=8)),
  133. ('recorded_at', models.DateTimeField(default=django.utils.timezone.now)),
  134. ('pickup', models.OneToOneField(on_delete=django.db.models.deletion.CASCADE, related_name='weigh_ticket', to='recycle_core.pickuporder')),
  135. ('recorded_by', models.ForeignKey(blank=True, null=True, on_delete=django.db.models.deletion.SET_NULL, related_name='weigh_tickets', to=settings.AUTH_USER_MODEL)),
  136. ],
  137. options={
  138. 'abstract': False,
  139. },
  140. ),
  141. migrations.CreateModel(
  142. name='WeighLine',
  143. fields=[
  144. ('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
  145. ('created_at', models.DateTimeField(auto_now_add=True)),
  146. ('updated_at', models.DateTimeField(auto_now=True)),
  147. ('quantity', models.DecimalField(decimal_places=3, max_digits=12)),
  148. ('unit', models.CharField(choices=[('kg', 'Kilogram'), ('lb', 'Pound'), ('pcs', 'Pieces')], default='kg', max_length=8)),
  149. ('material', models.ForeignKey(on_delete=django.db.models.deletion.PROTECT, to='recycle_core.material')),
  150. ('ticket', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, related_name='lines', to='recycle_core.weighticket')),
  151. ],
  152. options={
  153. 'abstract': False,
  154. },
  155. ),
  156. migrations.CreateModel(
  157. name='ServiceAgreement',
  158. fields=[
  159. ('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
  160. ('created_at', models.DateTimeField(auto_now_add=True)),
  161. ('updated_at', models.DateTimeField(auto_now=True)),
  162. ('valid_from', models.DateField(blank=True, null=True)),
  163. ('valid_to', models.DateField(blank=True, null=True)),
  164. ('customer', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, related_name='agreements', to='recycle_core.customer')),
  165. ('price_list', models.ForeignKey(on_delete=django.db.models.deletion.PROTECT, related_name='agreements', to='recycle_core.pricelist')),
  166. ('site', models.ForeignKey(blank=True, null=True, on_delete=django.db.models.deletion.SET_NULL, related_name='agreements', to='recycle_core.customersite')),
  167. ],
  168. options={
  169. 'abstract': False,
  170. },
  171. ),
  172. migrations.CreateModel(
  173. name='ScrapListingItem',
  174. fields=[
  175. ('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
  176. ('created_at', models.DateTimeField(auto_now_add=True)),
  177. ('updated_at', models.DateTimeField(auto_now=True)),
  178. ('quantity_estimate', models.DecimalField(decimal_places=3, max_digits=12)),
  179. ('unit', models.CharField(choices=[('kg', 'Kilogram'), ('lb', 'Pound'), ('pcs', 'Pieces')], default='kg', max_length=8)),
  180. ('listing', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, related_name='items', to='recycle_core.scraplisting')),
  181. ('material', models.ForeignKey(on_delete=django.db.models.deletion.PROTECT, to='recycle_core.material')),
  182. ],
  183. options={
  184. 'abstract': False,
  185. },
  186. ),
  187. migrations.CreateModel(
  188. name='ScrapBid',
  189. fields=[
  190. ('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
  191. ('created_at', models.DateTimeField(auto_now_add=True)),
  192. ('updated_at', models.DateTimeField(auto_now=True)),
  193. ('price_total', models.DecimalField(decimal_places=2, max_digits=14)),
  194. ('message', models.TextField(blank=True)),
  195. ('status', models.CharField(choices=[('active', 'Active'), ('retracted', 'Retracted'), ('accepted', 'Accepted'), ('rejected', 'Rejected')], default='active', max_length=16)),
  196. ('bidder_org', models.ForeignKey(on_delete=django.db.models.deletion.PROTECT, related_name='bids', to='orgs.organization')),
  197. ('bidder_user', models.ForeignKey(blank=True, null=True, on_delete=django.db.models.deletion.SET_NULL, related_name='bids', to=settings.AUTH_USER_MODEL)),
  198. ('listing', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, related_name='bids', to='recycle_core.scraplisting')),
  199. ],
  200. ),
  201. migrations.CreateModel(
  202. name='ScrapAward',
  203. fields=[
  204. ('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
  205. ('created_at', models.DateTimeField(auto_now_add=True)),
  206. ('updated_at', models.DateTimeField(auto_now=True)),
  207. ('notes', models.TextField(blank=True)),
  208. ('listing', models.OneToOneField(on_delete=django.db.models.deletion.CASCADE, related_name='award', to='recycle_core.scraplisting')),
  209. ('pickup', models.ForeignKey(blank=True, null=True, on_delete=django.db.models.deletion.SET_NULL, related_name='awards', to='recycle_core.pickuporder')),
  210. ('winning_bid', models.ForeignKey(on_delete=django.db.models.deletion.PROTECT, related_name='awards', to='recycle_core.scrapbid')),
  211. ],
  212. options={
  213. 'abstract': False,
  214. },
  215. ),
  216. migrations.CreateModel(
  217. name='ProvidedService',
  218. fields=[
  219. ('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
  220. ('created_at', models.DateTimeField(auto_now_add=True)),
  221. ('updated_at', models.DateTimeField(auto_now=True)),
  222. ('title', models.CharField(choices=[('Pickup & Logistics', 'Pickup & Logistics'), ('Material Sorting', 'Material Sorting'), ('Weighing & Ticketing', 'Weighing & Ticketing'), ('Invoicing & Payouts', 'Invoicing & Payouts'), ('Reporting & Analytics', 'Reporting & Analytics'), ('Marketplace & Bidding', 'Marketplace & Bidding'), ('Compliance & Audits', 'Compliance & Audits'), ('Consulting & Training', 'Consulting & Training')], max_length=100)),
  223. ('description', models.TextField()),
  224. ('body', markdownfield.models.MarkdownField(blank=True, null=True, rendered_field='body_html')),
  225. ('body_html', markdownfield.models.RenderedMarkdownField(null=True)),
  226. ('display_order', models.PositiveIntegerField(default=0)),
  227. ('is_enabled', models.BooleanField(default=True)),
  228. ('organization', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, related_name='services', to='orgs.organization')),
  229. ],
  230. options={
  231. 'ordering': ['display_order'],
  232. },
  233. ),
  234. migrations.CreateModel(
  235. name='PriceListItem',
  236. fields=[
  237. ('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
  238. ('created_at', models.DateTimeField(auto_now_add=True)),
  239. ('updated_at', models.DateTimeField(auto_now=True)),
  240. ('unit', models.CharField(choices=[('kg', 'Kilogram'), ('lb', 'Pound'), ('pcs', 'Pieces')], default='kg', max_length=8)),
  241. ('unit_price', models.DecimalField(decimal_places=2, max_digits=12)),
  242. ('direction', models.CharField(choices=[('buy', 'Buy from customer'), ('sell', 'Sell to customer')], default='sell', max_length=8)),
  243. ('material', models.ForeignKey(on_delete=django.db.models.deletion.PROTECT, related_name='price_items', to='recycle_core.material')),
  244. ('price_list', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, related_name='items', to='recycle_core.pricelist')),
  245. ],
  246. ),
  247. migrations.CreateModel(
  248. name='PickupItem',
  249. fields=[
  250. ('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
  251. ('created_at', models.DateTimeField(auto_now_add=True)),
  252. ('updated_at', models.DateTimeField(auto_now=True)),
  253. ('estimated_qty', models.DecimalField(blank=True, decimal_places=3, max_digits=12, null=True)),
  254. ('unit', models.CharField(choices=[('kg', 'Kilogram'), ('lb', 'Pound'), ('pcs', 'Pieces')], default='kg', max_length=8)),
  255. ('material', models.ForeignKey(on_delete=django.db.models.deletion.PROTECT, to='recycle_core.material')),
  256. ('pickup', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, related_name='items', to='recycle_core.pickuporder')),
  257. ],
  258. options={
  259. 'abstract': False,
  260. },
  261. ),
  262. migrations.CreateModel(
  263. name='MaterialImage',
  264. fields=[
  265. ('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
  266. ('created_at', models.DateTimeField(auto_now_add=True)),
  267. ('updated_at', models.DateTimeField(auto_now=True)),
  268. ('image', models.ImageField(upload_to='materials/%Y/%m/')),
  269. ('caption', models.CharField(blank=True, max_length=255)),
  270. ('display_order', models.PositiveIntegerField(default=0)),
  271. ('material', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, related_name='images', to='recycle_core.material')),
  272. ],
  273. options={
  274. 'ordering': ['display_order', 'id'],
  275. },
  276. ),
  277. migrations.CreateModel(
  278. name='MaterialCategory',
  279. fields=[
  280. ('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
  281. ('created_at', models.DateTimeField(auto_now_add=True)),
  282. ('updated_at', models.DateTimeField(auto_now=True)),
  283. ('name', models.CharField(choices=[('Plastics', 'Plastics'), ('Metals', 'Metals'), ('Paper', 'Paper'), ('Glass', 'Glass'), ('Electronics', 'Electronics'), ('Wood', 'Wood'), ('Rubber', 'Rubber'), ('Textiles', 'Textiles'), ('Organic', 'Organic'), ('Mixed', 'Mixed')], max_length=255)),
  284. ('organization', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, related_name='material_categories', to='orgs.organization')),
  285. ],
  286. ),
  287. migrations.CreateModel(
  288. name='Document',
  289. fields=[
  290. ('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
  291. ('created_at', models.DateTimeField(auto_now_add=True)),
  292. ('updated_at', models.DateTimeField(auto_now=True)),
  293. ('file', models.FileField(upload_to='documents/%Y/%m/%d/')),
  294. ('kind', models.CharField(blank=True, max_length=64)),
  295. ('object_id', models.PositiveIntegerField()),
  296. ('content_type', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, to='contenttypes.contenttype')),
  297. ('organization', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, related_name='documents', to='orgs.organization')),
  298. ('uploaded_by', models.ForeignKey(blank=True, null=True, on_delete=django.db.models.deletion.SET_NULL, to=settings.AUTH_USER_MODEL)),
  299. ],
  300. options={
  301. 'abstract': False,
  302. },
  303. ),
  304. migrations.AddField(
  305. model_name='customer',
  306. name='price_list',
  307. field=models.ForeignKey(blank=True, null=True, on_delete=django.db.models.deletion.SET_NULL, related_name='customers', to='recycle_core.pricelist'),
  308. ),
  309. migrations.CreateModel(
  310. name='AuditLog',
  311. fields=[
  312. ('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
  313. ('created_at', models.DateTimeField(auto_now_add=True)),
  314. ('updated_at', models.DateTimeField(auto_now=True)),
  315. ('action', models.CharField(max_length=64)),
  316. ('object_id', models.PositiveIntegerField()),
  317. ('metadata', models.JSONField(blank=True, default=dict)),
  318. ('content_type', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, to='contenttypes.contenttype')),
  319. ('organization', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, related_name='audit_logs', to='orgs.organization')),
  320. ('user', models.ForeignKey(blank=True, null=True, on_delete=django.db.models.deletion.SET_NULL, to=settings.AUTH_USER_MODEL)),
  321. ],
  322. ),
  323. migrations.CreateModel(
  324. name='ScrapListingInvite',
  325. fields=[
  326. ('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
  327. ('created_at', models.DateTimeField(auto_now_add=True)),
  328. ('updated_at', models.DateTimeField(auto_now=True)),
  329. ('invited_org', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, related_name='listing_invites', to='orgs.organization')),
  330. ('invited_user', models.ForeignKey(blank=True, null=True, on_delete=django.db.models.deletion.SET_NULL, related_name='listing_invites', to=settings.AUTH_USER_MODEL)),
  331. ('listing', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, related_name='invites', to='recycle_core.scraplisting')),
  332. ],
  333. options={
  334. 'unique_together': {('listing', 'invited_org')},
  335. },
  336. ),
  337. migrations.AddIndex(
  338. model_name='scrapbid',
  339. index=models.Index(fields=['listing', 'price_total'], name='recycle_cor_listing_030b96_idx'),
  340. ),
  341. migrations.AlterUniqueTogether(
  342. name='pricelistitem',
  343. unique_together={('price_list', 'material', 'unit', 'direction')},
  344. ),
  345. migrations.AlterUniqueTogether(
  346. name='materialcategory',
  347. unique_together={('organization', 'name')},
  348. ),
  349. migrations.AlterUniqueTogether(
  350. name='material',
  351. unique_together={('organization', 'name')},
  352. ),
  353. migrations.AddIndex(
  354. model_name='auditlog',
  355. index=models.Index(fields=['organization', 'created_at'], name='recycle_cor_organiz_192314_idx'),
  356. ),
  357. migrations.AddIndex(
  358. model_name='auditlog',
  359. index=models.Index(fields=['action', 'created_at'], name='recycle_cor_action_aad429_idx'),
  360. ),
  361. ]