django-bom 1.193__tar.gz → 1.260__tar.gz

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.

Potentially problematic release.


This version of django-bom might be problematic. Click here for more details.

Files changed (216) hide show
  1. django_bom-1.260/PKG-INFO +206 -0
  2. {django-bom-1.193 → django_bom-1.260}/README.md +7 -8
  3. {django-bom-1.193 → django_bom-1.260}/bom/admin.py +75 -8
  4. django_bom-1.260/bom/auth_backends.py +47 -0
  5. {django-bom-1.193 → django_bom-1.260}/bom/base_classes.py +1 -1
  6. {django-bom-1.193 → django_bom-1.260}/bom/constants.py +11 -0
  7. {django-bom-1.193 → django_bom-1.260}/bom/csv_headers.py +38 -46
  8. django_bom-1.260/bom/decorators.py +32 -0
  9. django_bom-1.260/bom/forms.py +1328 -0
  10. {django-bom-1.193 → django_bom-1.260}/bom/helpers.py +97 -7
  11. django_bom-1.260/bom/local_settings.py +35 -0
  12. django_bom-1.260/bom/migrations/0040_alter_organization_currency.py +19 -0
  13. django_bom-1.260/bom/migrations/0041_organization_subscription_quantity.py +18 -0
  14. django_bom-1.260/bom/migrations/0042_auto_20210720_2137.py +23 -0
  15. django_bom-1.260/bom/migrations/0043_auto_20211123_0157.py +24 -0
  16. django_bom-1.260/bom/migrations/0044_auto_20220831_1241.py +23 -0
  17. django_bom-1.260/bom/migrations/0045_sellerpart_link.py +18 -0
  18. django_bom-1.260/bom/migrations/0046_alter_sellerpart_unique_together.py +17 -0
  19. django_bom-1.260/bom/migrations/0047_sellerpart_seller_part_number.py +18 -0
  20. django_bom-1.260/bom/migrations/0048_rename_part_organization_number_class_bom_part_organiz_b333d6_idx_and_more.py +1017 -0
  21. django_bom-1.260/bom/migrations/0049_alter_assembly_id_alter_assemblysubparts_id_and_more.py +99 -0
  22. django_bom-1.260/bom/migrations/0050_alter_organization_options.py +17 -0
  23. django_bom-1.260/bom/migrations/0051_alter_manufacturer_organization_and_more.py +41 -0
  24. django_bom-1.260/bom/migrations/0052_remove_partrevision_attribute_and_more.py +584 -0
  25. {django-bom-1.193 → django_bom-1.260}/bom/models.py +386 -182
  26. {django-bom-1.193 → django_bom-1.260}/bom/part_bom.py +21 -10
  27. django_bom-1.260/bom/settings.py +262 -0
  28. django_bom-1.260/bom/static/bom/css/dashboard.css +17 -0
  29. django_bom-1.260/bom/static/bom/css/part-info.css +15 -0
  30. django_bom-1.260/bom/static/bom/css/style.css +418 -0
  31. {django-bom-1.193 → django_bom-1.260}/bom/static/bom/css/treetable-theme.css +10 -10
  32. django_bom-1.260/bom/static/bom/doc/sample_part_classes.csv +38 -0
  33. django_bom-1.260/bom/static/bom/doc/test_bom.csv +6 -0
  34. django_bom-1.260/bom/static/bom/doc/test_bom_5_intelligent.csv +4 -0
  35. django_bom-1.260/bom/static/bom/doc/test_full_bom.csv +37 -0
  36. django_bom-1.260/bom/static/bom/doc/test_new_parts.csv +5 -0
  37. django_bom-1.260/bom/static/bom/doc/test_new_parts_5_intelligent.csv +5 -0
  38. django_bom-1.260/bom/static/bom/js/formset-handler.js +65 -0
  39. django_bom-1.260/bom/templates/bom/account-delete.html +23 -0
  40. {django-bom-1.193 → django_bom-1.260}/bom/templates/bom/add-manufacturer-part.html +2 -5
  41. {django-bom-1.193 → django_bom-1.260}/bom/templates/bom/add-sellerpart.html +9 -5
  42. {django-bom-1.193 → django_bom-1.260}/bom/templates/bom/base-menu.html +10 -1
  43. {django-bom-1.193 → django_bom-1.260}/bom/templates/bom/base.html +3 -2
  44. {django-bom-1.193 → django_bom-1.260}/bom/templates/bom/bom-action-table.html +2 -2
  45. django_bom-1.260/bom/templates/bom/bom-base-menu.html +6 -0
  46. {django-bom-1.193 → django_bom-1.260}/bom/templates/bom/bom-base.html +1 -0
  47. django_bom-1.260/bom/templates/bom/bom-form-modal.html +36 -0
  48. {django-bom-1.193 → django_bom-1.260}/bom/templates/bom/bom-form.html +1 -5
  49. django_bom-1.260/bom/templates/bom/bom-modal-add-users.html +49 -0
  50. {django-bom-1.193 → django_bom-1.260}/bom/templates/bom/bom-signup.html +0 -4
  51. {django-bom-1.193/bom/templates/bom → django_bom-1.260/bom/templates/bom/components}/bom-flat.html +5 -5
  52. {django-bom-1.193/bom/templates/bom → django_bom-1.260/bom/templates/bom/components}/bom-indented.html +40 -7
  53. django_bom-1.260/bom/templates/bom/components/manufacturer-part-list.html +270 -0
  54. django_bom-1.260/bom/templates/bom/components/seller-part-list.html +62 -0
  55. {django-bom-1.193 → django_bom-1.260}/bom/templates/bom/create-part.html +2 -5
  56. django_bom-1.260/bom/templates/bom/dashboard.html +303 -0
  57. {django-bom-1.193 → django_bom-1.260}/bom/templates/bom/edit-manufacturer-part.html +8 -6
  58. django_bom-1.260/bom/templates/bom/edit-part-class.html +120 -0
  59. {django-bom-1.193 → django_bom-1.260}/bom/templates/bom/edit-part.html +2 -6
  60. django_bom-1.260/bom/templates/bom/edit-quantity-of-measure.html +119 -0
  61. django_bom-1.260/bom/templates/bom/edit-user-meta.html +70 -0
  62. django_bom-1.260/bom/templates/bom/help.html +1356 -0
  63. django_bom-1.260/bom/templates/bom/manufacturer-info.html +82 -0
  64. django_bom-1.260/bom/templates/bom/manufacturers.html +97 -0
  65. {django-bom-1.193 → django_bom-1.260}/bom/templates/bom/organization-create.html +8 -8
  66. {django-bom-1.193 → django_bom-1.260}/bom/templates/bom/part-info.html +31 -252
  67. django_bom-1.260/bom/templates/bom/part-revision-display.html +50 -0
  68. {django-bom-1.193 → django_bom-1.260}/bom/templates/bom/part-revision-edit.html +2 -5
  69. {django-bom-1.193 → django_bom-1.260}/bom/templates/bom/part-revision-manage-bom.html +2 -3
  70. {django-bom-1.193 → django_bom-1.260}/bom/templates/bom/part-revision-new.html +2 -5
  71. {django-bom-1.193 → django_bom-1.260}/bom/templates/bom/part-revision-release.html +0 -7
  72. {django-bom-1.193 → django_bom-1.260}/bom/templates/bom/search-help.html +0 -4
  73. django_bom-1.260/bom/templates/bom/seller-info.html +82 -0
  74. django_bom-1.260/bom/templates/bom/sellers.html +97 -0
  75. django_bom-1.260/bom/templates/bom/settings.html +592 -0
  76. {django-bom-1.193 → django_bom-1.260}/bom/templates/bom/signup.html +1 -1
  77. django_bom-1.260/bom/templates/bom/subscription_panel.html +16 -0
  78. django_bom-1.260/bom/templates/bom/table_of_contents.html +47 -0
  79. {django-bom-1.193 → django_bom-1.260}/bom/templates/bom/upload-bom.html +21 -26
  80. {django-bom-1.193 → django_bom-1.260}/bom/templates/bom/upload-parts-help.html +18 -7
  81. {django-bom-1.193 → django_bom-1.260}/bom/templates/bom/upload-parts.html +0 -4
  82. {django-bom-1.193 → django_bom-1.260}/bom/tests.py +365 -75
  83. {django-bom-1.193 → django_bom-1.260}/bom/third_party_apis/base_api.py +1 -1
  84. {django-bom-1.193 → django_bom-1.260}/bom/third_party_apis/google_drive.py +14 -6
  85. {django-bom-1.193 → django_bom-1.260}/bom/third_party_apis/mouser.py +1 -0
  86. {django-bom-1.193 → django_bom-1.260}/bom/urls.py +31 -10
  87. {django-bom-1.193 → django_bom-1.260}/bom/views/json_views.py +3 -3
  88. {django-bom-1.193 → django_bom-1.260}/bom/views/views.py +521 -165
  89. django_bom-1.260/django_bom.egg-info/PKG-INFO +206 -0
  90. {django-bom-1.193 → django_bom-1.260}/django_bom.egg-info/SOURCES.txt +38 -6
  91. django_bom-1.260/django_bom.egg-info/requires.txt +6 -0
  92. django_bom-1.260/pyproject.toml +70 -0
  93. django-bom-1.193/PKG-INFO +0 -196
  94. django-bom-1.193/bom/decorators.py +0 -18
  95. django-bom-1.193/bom/forms.py +0 -1268
  96. django-bom-1.193/bom/local_settings.py +0 -36
  97. django-bom-1.193/bom/settings.py +0 -219
  98. django-bom-1.193/bom/static/.DS_Store +0 -0
  99. django-bom-1.193/bom/static/bom/.DS_Store +0 -0
  100. django-bom-1.193/bom/static/bom/css/style.css +0 -207
  101. django-bom-1.193/bom/static/bom/img/.DS_Store +0 -0
  102. django-bom-1.193/bom/templates/bom/bom-base-menu.html +0 -4
  103. django-bom-1.193/bom/templates/bom/dashboard.html +0 -295
  104. django-bom-1.193/bom/templates/bom/edit-part-class.html +0 -36
  105. django-bom-1.193/bom/templates/bom/edit-user-meta.html +0 -41
  106. django-bom-1.193/bom/templates/bom/help.html +0 -1108
  107. django-bom-1.193/bom/templates/bom/part-revision-display.html +0 -178
  108. django-bom-1.193/bom/templates/bom/settings.html +0 -402
  109. django-bom-1.193/django_bom.egg-info/PKG-INFO +0 -196
  110. django-bom-1.193/django_bom.egg-info/requires.txt +0 -5
  111. django-bom-1.193/setup.py +0 -41
  112. {django-bom-1.193 → django_bom-1.260}/LICENSE +0 -0
  113. {django-bom-1.193 → django_bom-1.260}/MANIFEST.in +0 -0
  114. {django-bom-1.193 → django_bom-1.260}/bom/__init__.py +0 -0
  115. {django-bom-1.193 → django_bom-1.260}/bom/apps.py +0 -0
  116. {django-bom-1.193 → django_bom-1.260}/bom/context_processors.py +0 -0
  117. {django-bom-1.193 → django_bom-1.260}/bom/form_fields.py +0 -0
  118. {django-bom-1.193 → django_bom-1.260}/bom/migrations/0001_initial.py +0 -0
  119. {django-bom-1.193 → django_bom-1.260}/bom/migrations/0002_auto_20180908_2151.py +0 -0
  120. {django-bom-1.193 → django_bom-1.260}/bom/migrations/0003_sellerpart_data_source.py +0 -0
  121. {django-bom-1.193 → django_bom-1.260}/bom/migrations/0004_auto_20180911_0011.py +0 -0
  122. {django-bom-1.193 → django_bom-1.260}/bom/migrations/0005_auto_20181007_1934.py +0 -0
  123. {django-bom-1.193 → django_bom-1.260}/bom/migrations/0006_auto_20181007_1949.py +0 -0
  124. {django-bom-1.193 → django_bom-1.260}/bom/migrations/0007_auto_20181009_0256.py +0 -0
  125. {django-bom-1.193 → django_bom-1.260}/bom/migrations/0008_auto_20181030_0427.py +0 -0
  126. {django-bom-1.193 → django_bom-1.260}/bom/migrations/0009_subpart_reference.py +0 -0
  127. {django-bom-1.193 → django_bom-1.260}/bom/migrations/0010_auto_20181202_0733.py +0 -0
  128. {django-bom-1.193 → django_bom-1.260}/bom/migrations/0011_auto_20181202_2113.py +0 -0
  129. {django-bom-1.193 → django_bom-1.260}/bom/migrations/0012_partchangehistory.py +0 -0
  130. {django-bom-1.193 → django_bom-1.260}/bom/migrations/0013_auto_20190222_1631.py +0 -0
  131. {django-bom-1.193 → django_bom-1.260}/bom/migrations/0014_auto_20190223_2353.py +0 -0
  132. {django-bom-1.193 → django_bom-1.260}/bom/migrations/0015_auto_20190303_1915.py +0 -0
  133. {django-bom-1.193 → django_bom-1.260}/bom/migrations/0016_auto_20190405_2308.py +0 -0
  134. {django-bom-1.193 → django_bom-1.260}/bom/migrations/0017_auto_20190616_1912.py +0 -0
  135. {django-bom-1.193 → django_bom-1.260}/bom/migrations/0018_auto_20190616_2143.py +0 -0
  136. {django-bom-1.193 → django_bom-1.260}/bom/migrations/0019_auto_20190624_1246.py +0 -0
  137. {django-bom-1.193 → django_bom-1.260}/bom/migrations/0020_auto_20190627_0207.py +0 -0
  138. {django-bom-1.193 → django_bom-1.260}/bom/migrations/0021_auto_20190627_0428.py +0 -0
  139. {django-bom-1.193 → django_bom-1.260}/bom/migrations/0022_auto_20190811_2140.py +0 -0
  140. {django-bom-1.193 → django_bom-1.260}/bom/migrations/0023_auto_20191205_2351.py +0 -0
  141. {django-bom-1.193 → django_bom-1.260}/bom/migrations/0024_auto_20191214_1342.py +0 -0
  142. {django-bom-1.193 → django_bom-1.260}/bom/migrations/0025_auto_20191221_1907.py +0 -0
  143. {django-bom-1.193 → django_bom-1.260}/bom/migrations/0026_auto_20191222_2258.py +0 -0
  144. {django-bom-1.193 → django_bom-1.260}/bom/migrations/0027_auto_20191222_2347.py +0 -0
  145. {django-bom-1.193 → django_bom-1.260}/bom/migrations/0028_partrevision_displayable_synopsis.py +0 -0
  146. {django-bom-1.193 → django_bom-1.260}/bom/migrations/0029_auto_20191231_1630.py +0 -0
  147. {django-bom-1.193 → django_bom-1.260}/bom/migrations/0030_auto_20200101_2253.py +0 -0
  148. {django-bom-1.193 → django_bom-1.260}/bom/migrations/0031_auto_20200104_1352.py +0 -0
  149. {django-bom-1.193 → django_bom-1.260}/bom/migrations/0032_auto_20200126_1806.py +0 -0
  150. {django-bom-1.193 → django_bom-1.260}/bom/migrations/0033_auto_20200203_0618.py +0 -0
  151. {django-bom-1.193 → django_bom-1.260}/bom/migrations/0034_auto_20200222_0359.py +0 -0
  152. {django-bom-1.193 → django_bom-1.260}/bom/migrations/0035_auto_20200303_0111.py +0 -0
  153. {django-bom-1.193 → django_bom-1.260}/bom/migrations/0036_auto_20200303_0538.py +0 -0
  154. {django-bom-1.193 → django_bom-1.260}/bom/migrations/0037_auto_20200405_1642.py +0 -0
  155. {django-bom-1.193 → django_bom-1.260}/bom/migrations/0038_auto_20200422_0504.py +0 -0
  156. {django-bom-1.193 → django_bom-1.260}/bom/migrations/0039_auto_20200929_2315.py +0 -0
  157. {django-bom-1.193 → django_bom-1.260}/bom/migrations/__init__.py +0 -0
  158. {django-bom-1.193 → django_bom-1.260}/bom/static/bom/css/jquery.treetable.css +0 -0
  159. {django-bom-1.193 → django_bom-1.260}/bom/static/bom/css/materialize.min.css +0 -0
  160. {django-bom-1.193 → django_bom-1.260}/bom/static/bom/css/tablesorter-theme.materialize.css +0 -0
  161. {django-bom-1.193 → django_bom-1.260}/bom/static/bom/img/_ionicons_svg_md-arrow-dropdown.svg +0 -0
  162. {django-bom-1.193 → django_bom-1.260}/bom/static/bom/img/_ionicons_svg_md-arrow-dropright.svg +0 -0
  163. {django-bom-1.193 → django_bom-1.260}/bom/static/bom/img/favicon.ico +0 -0
  164. {django-bom-1.193 → django_bom-1.260}/bom/static/bom/img/google/web/1x/btn_google_signin_dark_disabled_web.png +0 -0
  165. {django-bom-1.193 → django_bom-1.260}/bom/static/bom/img/google/web/1x/btn_google_signin_dark_focus_web.png +0 -0
  166. {django-bom-1.193 → django_bom-1.260}/bom/static/bom/img/google/web/1x/btn_google_signin_dark_normal_web.png +0 -0
  167. {django-bom-1.193 → django_bom-1.260}/bom/static/bom/img/google/web/1x/btn_google_signin_dark_pressed_web.png +0 -0
  168. {django-bom-1.193 → django_bom-1.260}/bom/static/bom/img/google/web/1x/btn_google_signin_light_disabled_web.png +0 -0
  169. {django-bom-1.193 → django_bom-1.260}/bom/static/bom/img/google/web/1x/btn_google_signin_light_focus_web.png +0 -0
  170. {django-bom-1.193 → django_bom-1.260}/bom/static/bom/img/google/web/1x/btn_google_signin_light_normal_web.png +0 -0
  171. {django-bom-1.193 → django_bom-1.260}/bom/static/bom/img/google/web/1x/btn_google_signin_light_pressed_web.png +0 -0
  172. {django-bom-1.193 → django_bom-1.260}/bom/static/bom/img/google/web/2x/btn_google_signin_dark_disabled_web@2x.png +0 -0
  173. {django-bom-1.193 → django_bom-1.260}/bom/static/bom/img/google/web/2x/btn_google_signin_dark_focus_web@2x.png +0 -0
  174. {django-bom-1.193 → django_bom-1.260}/bom/static/bom/img/google/web/2x/btn_google_signin_dark_normal_web@2x.png +0 -0
  175. {django-bom-1.193 → django_bom-1.260}/bom/static/bom/img/google/web/2x/btn_google_signin_dark_pressed_web@2x.png +0 -0
  176. {django-bom-1.193 → django_bom-1.260}/bom/static/bom/img/google/web/2x/btn_google_signin_light_disabled_web@2x.png +0 -0
  177. {django-bom-1.193 → django_bom-1.260}/bom/static/bom/img/google/web/2x/btn_google_signin_light_focus_web@2x.png +0 -0
  178. {django-bom-1.193 → django_bom-1.260}/bom/static/bom/img/google/web/2x/btn_google_signin_light_normal_web@2x.png +0 -0
  179. {django-bom-1.193 → django_bom-1.260}/bom/static/bom/img/google/web/2x/btn_google_signin_light_pressed_web@2x.png +0 -0
  180. {django-bom-1.193 → django_bom-1.260}/bom/static/bom/img/google/web/vector/btn_google_dark_disabled_ios.eps +0 -0
  181. {django-bom-1.193 → django_bom-1.260}/bom/static/bom/img/google/web/vector/btn_google_dark_disabled_ios.svg +0 -0
  182. {django-bom-1.193 → django_bom-1.260}/bom/static/bom/img/google/web/vector/btn_google_dark_focus_ios.eps +0 -0
  183. {django-bom-1.193 → django_bom-1.260}/bom/static/bom/img/google/web/vector/btn_google_dark_focus_ios.svg +0 -0
  184. {django-bom-1.193 → django_bom-1.260}/bom/static/bom/img/google/web/vector/btn_google_dark_normal_ios.eps +0 -0
  185. {django-bom-1.193 → django_bom-1.260}/bom/static/bom/img/google/web/vector/btn_google_dark_normal_ios.svg +0 -0
  186. {django-bom-1.193 → django_bom-1.260}/bom/static/bom/img/google/web/vector/btn_google_dark_pressed_ios.eps +0 -0
  187. {django-bom-1.193 → django_bom-1.260}/bom/static/bom/img/google/web/vector/btn_google_dark_pressed_ios.svg +0 -0
  188. {django-bom-1.193 → django_bom-1.260}/bom/static/bom/img/google/web/vector/btn_google_light_disabled_ios.eps +0 -0
  189. {django-bom-1.193 → django_bom-1.260}/bom/static/bom/img/google/web/vector/btn_google_light_disabled_ios.svg +0 -0
  190. {django-bom-1.193 → django_bom-1.260}/bom/static/bom/img/google/web/vector/btn_google_light_focus_ios.eps +0 -0
  191. {django-bom-1.193 → django_bom-1.260}/bom/static/bom/img/google/web/vector/btn_google_light_focus_ios.svg +0 -0
  192. {django-bom-1.193 → django_bom-1.260}/bom/static/bom/img/google/web/vector/btn_google_light_normal_ios.eps +0 -0
  193. {django-bom-1.193 → django_bom-1.260}/bom/static/bom/img/google/web/vector/btn_google_light_normal_ios.svg +0 -0
  194. {django-bom-1.193 → django_bom-1.260}/bom/static/bom/img/google/web/vector/btn_google_light_pressed_ios.eps +0 -0
  195. {django-bom-1.193 → django_bom-1.260}/bom/static/bom/img/google/web/vector/btn_google_light_pressed_ios.svg +0 -0
  196. {django-bom-1.193 → django_bom-1.260}/bom/static/bom/img/google_drive_logo.svg +0 -0
  197. {django-bom-1.193 → django_bom-1.260}/bom/static/bom/img/indabom.png +0 -0
  198. {django-bom-1.193 → django_bom-1.260}/bom/static/bom/img/mouser.png +0 -0
  199. {django-bom-1.193 → django_bom-1.260}/bom/static/bom/img/octopart_blue.svg +0 -0
  200. {django-bom-1.193 → django_bom-1.260}/bom/static/bom/js/jquery-3.4.1.min.js +0 -0
  201. {django-bom-1.193 → django_bom-1.260}/bom/static/bom/js/jquery.ba-floatingscrollbar.min.js +0 -0
  202. {django-bom-1.193 → django_bom-1.260}/bom/static/bom/js/jquery.treetable.js +0 -0
  203. {django-bom-1.193 → django_bom-1.260}/bom/static/bom/js/materialize.min.js +0 -0
  204. {django-bom-1.193 → django_bom-1.260}/bom/templates/bom/bom-action-btn.html +0 -0
  205. {django-bom-1.193 → django_bom-1.260}/bom/templates/bom/dashboard-menu.html +0 -0
  206. {django-bom-1.193 → django_bom-1.260}/bom/templates/bom/nothing-to-see.html +0 -0
  207. {django-bom-1.193 → django_bom-1.260}/bom/templates/registration/login.html +0 -0
  208. {django-bom-1.193 → django_bom-1.260}/bom/third_party_apis/__init__.py +0 -0
  209. {django-bom-1.193 → django_bom-1.260}/bom/third_party_apis/test_apis.py +0 -0
  210. {django-bom-1.193 → django_bom-1.260}/bom/utils.py +0 -0
  211. {django-bom-1.193 → django_bom-1.260}/bom/validators.py +0 -0
  212. {django-bom-1.193 → django_bom-1.260}/bom/views/__init__.py +0 -0
  213. {django-bom-1.193 → django_bom-1.260}/bom/wsgi.py +0 -0
  214. {django-bom-1.193 → django_bom-1.260}/django_bom.egg-info/dependency_links.txt +0 -0
  215. {django-bom-1.193 → django_bom-1.260}/django_bom.egg-info/top_level.txt +0 -0
  216. {django-bom-1.193 → django_bom-1.260}/setup.cfg +0 -0
@@ -0,0 +1,206 @@
1
+ Metadata-Version: 2.4
2
+ Name: django-bom
3
+ Version: 1.260
4
+ Summary: A simple Django app to manage a bill of materials.
5
+ Author-email: Mike Kasparian <mpkasp@gmail.com>
6
+ License: GPL-3.0-only
7
+ Project-URL: Homepage, https://www.indabom.com/
8
+ Classifier: Environment :: Web Environment
9
+ Classifier: Framework :: Django
10
+ Classifier: Framework :: Django :: 5
11
+ Classifier: Intended Audience :: Developers
12
+ Classifier: License :: OSI Approved :: GNU General Public License v3 (GPLv3)
13
+ Classifier: Operating System :: OS Independent
14
+ Classifier: Programming Language :: Python
15
+ Classifier: Programming Language :: Python :: 3
16
+ Classifier: Programming Language :: Python :: 3.10
17
+ Classifier: Programming Language :: Python :: 3.11
18
+ Classifier: Programming Language :: Python :: 3.12
19
+ Classifier: Programming Language :: Python :: 3 :: Only
20
+ Classifier: Topic :: Internet :: WWW/HTTP
21
+ Classifier: Topic :: Internet :: WWW/HTTP :: Dynamic Content
22
+ Requires-Python: >=3.10
23
+ Description-Content-Type: text/markdown
24
+ License-File: LICENSE
25
+ Requires-Dist: Django<5.3,>=5.2
26
+ Requires-Dist: social-auth-app-django<6,>=5
27
+ Requires-Dist: social-auth-core<6,>=4
28
+ Requires-Dist: google-api-python-client>=2
29
+ Requires-Dist: django-materializecss-form
30
+ Requires-Dist: django-money<5,>=3
31
+ Dynamic: license-file
32
+
33
+ # BOM
34
+ ![](https://github.com/mpkasp/django-bom/workflows/Django%20CI/badge.svg)
35
+
36
+ BOM is a simple Django app to manage a bill of materials. It supports multiple part numbering schemes, tracking component sourcing information, estimates costs, and contains smart integrations with Mouser to pull in the latest component pricing and Google Drive for part file management. BOM is written in Python 3 and Django 3.
37
+
38
+ [See a live example](https://www.indabom.com).
39
+
40
+ BOM can be added to an existing (or new) Django project, or stand alone on its own, which can be more convenient if you're interested in tweaking the tool.
41
+
42
+ If you already have a django project, you can skip to [Add Django Bom To Your App](#add-django-bom-to-your-app), otherwise [Start From Scratch: Add to new Django project](#start-from-scratch-add-to-a-new-django-project) to add it to a new django project, or [Start From Scratch: Use as standalone Django project](#start-from-scratch-use-as-a-standalone-django-project).
43
+
44
+ ## Table of contents
45
+ * [Start From Scratch: Add to new Django project](#start-from-scratch-add-to-a-new-django-project)
46
+ * [Add Django Bom To Your App](#add-django-bom-to-your-app)
47
+ * [Start From Scratch: Use as standalone Django project](#start-from-scratch-use-as-a-standalone-django-project)
48
+ * [Customize Base Template](#customize-base-template)
49
+ * [Integrations](#integrations)
50
+ * [Contributing](#contributing)
51
+ * [Installation pitfalls](#installation-pitfalls)
52
+
53
+ ## Start From Scratch: Add to a new Django project
54
+ 1. To start from scratch, install your pipenv using Python 3.12
55
+ ```
56
+ pipenv --python 3.12
57
+ pipenv shell
58
+ ```
59
+
60
+ 2. From here install django, and set up your project.
61
+ ```
62
+ pipenv install django
63
+ django-admin startproject mysite
64
+ cd mysite
65
+ python manage.py migrate
66
+ python manage.py createsuperuser # make a user for your development environment
67
+ ```
68
+
69
+ 3. Continue on to [Add Django Bom To Your App](#add-django-bom-to-your-app).
70
+
71
+ ## Add Django Bom To Your App
72
+ django-bom is a [reusable django application](https://docs.djangoproject.com/en/1.11/intro/reusable-apps/). If you don't already have a django project, you can follow some quick steps below to get up and running, or read about creating your first django app [here](https://docs.djangoproject.com/en/1.11/intro/tutorial01/).
73
+
74
+ ```
75
+ pip install django-bom
76
+ ```
77
+
78
+ 1. Add "bom" to your INSTALLED_APPS setting like this::
79
+
80
+ ```
81
+ INSTALLED_APPS = [
82
+ ...
83
+ 'bom',
84
+ 'social_django', # to enable google drive sync in bom
85
+ 'djmoney', # for currency
86
+ 'djmoney.contrib.exchange', # for currency
87
+ 'materializecssform',
88
+ ]
89
+ ```
90
+
91
+ 2. Update your URLconf in your project urls.py like this::
92
+
93
+ ```
94
+ path('bom/', include('bom.urls')),
95
+ ```
96
+
97
+ And don't forget to import include:
98
+
99
+ ```
100
+ from django.conf.urls import include
101
+ ```
102
+
103
+ 3. Update your settings.py to add the bom context processor `'bom.context_processors.bom_config',` to your TEMPLATES variable, and create a new empty dictionary BOM_CONFIG.
104
+
105
+ ```
106
+ TEMPLATES = [
107
+ {
108
+ ...
109
+ 'OPTIONS': {
110
+ 'context_processors': [
111
+ ...
112
+ 'bom.context_processors.bom_config',
113
+ ],
114
+ },
115
+ },
116
+ ]
117
+ ```
118
+
119
+ and
120
+
121
+ ```
122
+ BOM_CONFIG = {'standalone_mode': False}
123
+ ```
124
+
125
+ 4. Run `python manage.py migrate` to create the bom models.
126
+
127
+ 5. Start the development server `python manage.py runserver` and visit http://127.0.0.1:8000/admin/
128
+ to manage the bom (you'll need the Admin app enabled).
129
+
130
+ 6. Visit http://127.0.0.1:8000/bom/ to begin.
131
+
132
+
133
+ ## Start From Scratch: Use as a standalone Django project
134
+ 1. To start from scratch we recommend setting up a virtual environment
135
+ ```
136
+ virtualenv -p python3 mysite
137
+ cd mysite
138
+ source bin/activate
139
+ ```
140
+
141
+ 2. From here install django, and set up your project.
142
+ ```
143
+ git clone https://github.com/mpkasp/django-bom.git
144
+ pipenv install
145
+ python manage.py migrate
146
+ cp bom/local_settings.py.example bom/local_settings.py
147
+ python manage.py runserver
148
+ ```
149
+
150
+ ## Customize Base Template
151
+ The base template can be customized to your pleasing. Just add the following configuration to your settings.py:
152
+
153
+ ```
154
+ BOM_CONFIG = {
155
+ 'base_template': 'base.html',
156
+ }
157
+ ```
158
+
159
+ where `base.html` is your base template.
160
+
161
+ ## Integrations
162
+ ### Mouser Integration
163
+ For part matching, make sure to add your Mouser api key. You can get your key [here](https://www.mouser.com/MyMouser/MouserSearchApplication.aspx).
164
+
165
+ ### Google Drive Integration
166
+ Make sure to add the following to your settings.py:
167
+ ```
168
+ AUTHENTICATION_BACKENDS = (
169
+ 'social_core.backends.google.GoogleOpenId',
170
+ 'social_core.backends.google.GoogleOAuth2',
171
+ 'social_core.backends.google.GoogleOAuth',
172
+ 'django.contrib.auth.backends.ModelBackend',
173
+ )
174
+
175
+ SOCIAL_AUTH_GOOGLE_OAUTH2_SCOPE = ['email', 'profile', 'https://www.googleapis.com/auth/drive', 'https://www.googleapis.com/auth/plus.login']
176
+ SOCIAL_AUTH_GOOGLE_OAUTH2_AUTH_EXTRA_ARGUMENTS = {
177
+ 'access_type': 'offline',
178
+ 'approval_prompt': 'auto'
179
+ }
180
+ ```
181
+ And if you're using https on production add:
182
+ ```
183
+ SOCIAL_AUTH_REDIRECT_IS_HTTPS = not DEBUG
184
+ ```
185
+
186
+ ### FIXER
187
+ Fixer.io is used to handle exchange rate calculations. This is helpful if you may be purchasing parts from another currency (especially via Mouser) and you still need to estimate your part costs.
188
+
189
+ To set this up you just need to add your API key to local_settings.py as shown in the example.
190
+
191
+ To update rates, migrate and run `python manage.py update_rates`. Some day we will need to add a (celerybeat?) task to update rates on a schedule. Explained more [here](https://github.com/django-money/django-money#working-with-exchange-rates).
192
+
193
+ ## Contributing
194
+
195
+ Contributions welcome! Before contributing your work please read the [contributing readme](https://github.com/mpkasp/django-bom/blob/master/CONTRIBUTING.md).
196
+
197
+ Also reach out to mike@indabom.com to discuss features, and join our slack channel.
198
+
199
+ ## Installation Pitfalls
200
+
201
+ ### Windows
202
+ #### Sqlite
203
+ You may get an error during your `pip install -r requirements.txt` related to sqlite. This may be fixed by installing Visual C++ for python...
204
+
205
+ #### Cryptography
206
+ Sometimes you'll have issues installing cryptography, if this is the case you may just need to set up some environment variables. This [stackoverflow](https://stackoverflow.com/questions/46288737/error-while-installing-sqlite-using-pip-on-python-2-7-13) may help.
@@ -19,16 +19,15 @@ If you already have a django project, you can skip to [Add Django Bom To Your Ap
19
19
  * [Installation pitfalls](#installation-pitfalls)
20
20
 
21
21
  ## Start From Scratch: Add to a new Django project
22
- 1. To start from scratch we recommend setting up a virtual environment
22
+ 1. To start from scratch, install your pipenv using Python 3.12
23
23
  ```
24
- virtualenv -p python3 mysite
25
- cd mysite
26
- source bin/activate
24
+ pipenv --python 3.12
25
+ pipenv shell
27
26
  ```
28
27
 
29
28
  2. From here install django, and set up your project.
30
29
  ```
31
- pip install django
30
+ pipenv install django
32
31
  django-admin startproject mysite
33
32
  cd mysite
34
33
  python manage.py migrate
@@ -38,7 +37,7 @@ python manage.py createsuperuser # make a user for your development environment
38
37
  3. Continue on to [Add Django Bom To Your App](#add-django-bom-to-your-app).
39
38
 
40
39
  ## Add Django Bom To Your App
41
- django-bom is a [reusable django application](https://docs.djangoproject.com/en/1.11/intro/reusable-apps/). If you don't already have a django project, you can follow some quick steps below to get up and running, or read about creating your first django app [here](https://docs.djangoproject.com/en/1.11/intro/tutorial01/). Note that django-bom currently supports python 2.7.
40
+ django-bom is a [reusable django application](https://docs.djangoproject.com/en/1.11/intro/reusable-apps/). If you don't already have a django project, you can follow some quick steps below to get up and running, or read about creating your first django app [here](https://docs.djangoproject.com/en/1.11/intro/tutorial01/).
42
41
 
43
42
  ```
44
43
  pip install django-bom
@@ -88,7 +87,7 @@ TEMPLATES = [
88
87
  and
89
88
 
90
89
  ```
91
- BOM_CONFIG = {}
90
+ BOM_CONFIG = {'standalone_mode': False}
92
91
  ```
93
92
 
94
93
  4. Run `python manage.py migrate` to create the bom models.
@@ -110,7 +109,7 @@ source bin/activate
110
109
  2. From here install django, and set up your project.
111
110
  ```
112
111
  git clone https://github.com/mpkasp/django-bom.git
113
- pip install -r requirements.txt
112
+ pipenv install
114
113
  python manage.py migrate
115
114
  cp bom/local_settings.py.example bom/local_settings.py
116
115
  python manage.py runserver
@@ -1,18 +1,37 @@
1
+ from django.conf import settings
1
2
  from django.contrib import admin
3
+ from django.contrib.auth import get_user_model
2
4
  from django.contrib.auth.admin import UserAdmin
3
- from .models import *
4
5
 
6
+ from .models import (
7
+ Assembly,
8
+ Manufacturer,
9
+ ManufacturerPart,
10
+ Part,
11
+ PartClass,
12
+ PartRevision,
13
+ PartRevisionProperty,
14
+ PartRevisionPropertyDefinition,
15
+ Seller,
16
+ SellerPart,
17
+ Subpart,
18
+ UnitDefinition,
19
+ QuantityOfMeasure,
20
+ get_organization_model,
21
+ get_user_meta_model
22
+ )
23
+
24
+ User = get_user_model()
25
+ UserMeta = get_user_meta_model()
26
+ Organization = get_organization_model()
5
27
 
6
28
  class UserMetaInline(admin.TabularInline):
7
29
  model = UserMeta
30
+ verbose_name = 'BOM User Meta'
8
31
  raw_id_fields = ('organization',)
9
32
  can_delete = False
10
33
 
11
34
 
12
- class UserAdmin(UserAdmin):
13
- inlines = (UserMetaInline,)
14
-
15
-
16
35
  class OrganizationAdmin(admin.ModelAdmin):
17
36
  list_display = ('name',)
18
37
 
@@ -36,6 +55,7 @@ class SellerPartAdmin(admin.ModelAdmin):
36
55
  list_display = (
37
56
  'manufacturer_part',
38
57
  'seller',
58
+ 'seller_part_number',
39
59
  'minimum_order_quantity',
40
60
  'minimum_pack_quantity',
41
61
  'unit_cost',
@@ -71,7 +91,10 @@ class PartClassAdmin(admin.ModelAdmin):
71
91
 
72
92
  class PartRevisionAdminInline(admin.TabularInline):
73
93
  model = PartRevision
94
+ extra = 0
74
95
  raw_id_fields = ('assembly',)
96
+ readonly_fields = ('timestamp',)
97
+ show_change_link = True
75
98
 
76
99
 
77
100
  class PartAdmin(admin.ModelAdmin):
@@ -81,8 +104,10 @@ class PartAdmin(admin.ModelAdmin):
81
104
  'organization',
82
105
  'get_full_part_number',
83
106
  )
107
+ list_filter = ('organization', 'number_class',)
84
108
  raw_id_fields = ('number_class', 'primary_manufacturer_part',)
85
109
  inlines = [
110
+ PartRevisionAdminInline,
86
111
  ManufacturerPartAdminInline,
87
112
  ]
88
113
 
@@ -93,10 +118,22 @@ class PartAdmin(admin.ModelAdmin):
93
118
  get_full_part_number.admin_order_field = 'number_class__part_number'
94
119
 
95
120
 
121
+ class QuantityOfMeasureAdmin(admin.ModelAdmin):
122
+ list_display = ('name', 'organization')
123
+ list_filter = ('organization',)
124
+
125
+
126
+ class PartRevisionPropertyInline(admin.TabularInline):
127
+ model = PartRevisionProperty
128
+ extra = 1
129
+ raw_id_fields = ('property_definition', 'part_revision', 'unit_definition')
130
+
131
+
96
132
  class PartRevisionAdmin(admin.ModelAdmin):
97
133
  list_display = ('part', 'revision', 'description', 'get_assembly_size', 'timestamp',)
98
134
  raw_id_fields = ('assembly',)
99
135
  readonly_fields = ('timestamp',)
136
+ inlines = [PartRevisionPropertyInline]
100
137
 
101
138
  def get_assembly_size(self, obj):
102
139
  return None if obj.assembly is None else obj.assembly.subparts.count()
@@ -125,10 +162,37 @@ class AssemblyAdmin(admin.ModelAdmin):
125
162
  ]
126
163
 
127
164
 
128
- admin.site.unregister(User)
165
+ class UnitDefinitionAdmin(admin.ModelAdmin):
166
+ list_display = ('name', 'symbol', 'quantity_of_measure', 'base_multiplier', 'organization')
167
+ list_filter = ('organization',)
168
+
169
+
170
+ class PartRevisionPropertyDefinitionAdmin(admin.ModelAdmin):
171
+ list_display = ('code', 'name', 'type', 'organization', 'quantity_of_measure')
172
+ list_filter = ('organization',)
173
+
174
+
175
+ if settings.BOM_USER_META_MODEL == 'bom.UserMeta':
176
+ current_user_admin = admin.site._registry.get(User)
177
+
178
+ if current_user_admin:
179
+ user_admin_class = current_user_admin.__class__
180
+ existing_inlines = list(user_admin_class.inlines or [])
181
+ if UserMetaInline not in existing_inlines:
182
+ existing_inlines.append(UserMetaInline)
183
+ user_admin_class.inlines = existing_inlines
184
+ else:
185
+ class BomUserAdmin(UserAdmin):
186
+ inlines = [UserMetaInline]
187
+
188
+
189
+ admin.site.register(User, BomUserAdmin)
190
+
191
+ if settings.BOM_ORGANIZATION_MODEL == 'bom.Organization':
192
+ from .models import Organization
193
+
194
+ admin.site.register(Organization, OrganizationAdmin)
129
195
 
130
- admin.site.register(User, UserAdmin)
131
- admin.site.register(Organization, OrganizationAdmin)
132
196
  admin.site.register(Seller, SellerAdmin)
133
197
  admin.site.register(SellerPart, SellerPartAdmin)
134
198
  admin.site.register(ManufacturerPart, ManufacturerPartAdmin)
@@ -138,3 +202,6 @@ admin.site.register(PartRevision, PartRevisionAdmin)
138
202
  admin.site.register(Manufacturer, ManufacturerAdmin)
139
203
  admin.site.register(Assembly, AssemblyAdmin)
140
204
  admin.site.register(Subpart, SubpartAdmin)
205
+ admin.site.register(UnitDefinition, UnitDefinitionAdmin)
206
+ admin.site.register(PartRevisionPropertyDefinition, PartRevisionPropertyDefinitionAdmin)
207
+ admin.site.register(QuantityOfMeasure, QuantityOfMeasureAdmin)
@@ -0,0 +1,47 @@
1
+ from typing import Optional
2
+
3
+ from . import constants
4
+ from .models import get_organization_model
5
+
6
+ Organization = get_organization_model()
7
+
8
+
9
+ class OrganizationPermissionBackend:
10
+ """
11
+ Object-level permission backend for django-bom.
12
+
13
+ - Uses Django's has_perm(user, perm, obj) to evaluate permissions tied to an Organization.
14
+ - Superusers are granted all permissions.
15
+ - For `bom.manage_members`: User must be owner or admin within the organization.
16
+ """
17
+
18
+ def authenticate(self, request, **credentials): # pragma: no cover - not used for auth
19
+ return None
20
+
21
+ def has_perm(self, user_obj, perm: str, obj: Optional[object] = None):
22
+ # Only handle our specific object-level permission. Let other backends process others.
23
+ if not user_obj or not user_obj.is_authenticated:
24
+ return False
25
+
26
+ if user_obj.is_superuser:
27
+ return True
28
+
29
+ if perm != 'bom.manage_members':
30
+ return None
31
+
32
+ if obj is None or not isinstance(obj, Organization):
33
+ return False
34
+
35
+ profile = user_obj.bom_profile()
36
+ if not profile or profile.organization_id != obj.id:
37
+ return False
38
+
39
+ if obj.subscription != constants.SUBSCRIPTION_TYPE_PRO:
40
+ return False
41
+
42
+ is_owner = obj.owner_id == user_obj.id
43
+ is_admin = getattr(profile, 'role', None) == constants.ROLE_TYPE_ADMIN
44
+ if not (is_owner or is_admin):
45
+ return False
46
+
47
+ return True
@@ -11,7 +11,7 @@ class AsDictModel:
11
11
 
12
12
  def __iter__(self):
13
13
  for key in dir(self):
14
- if not key.startswith("_"):
14
+ if not key.startswith("_") and not key == "objects":
15
15
  value = getattr(self, key)
16
16
  if not callable(value):
17
17
  if isinstance(value, (dict, OrderedDict)):
@@ -34,11 +34,19 @@ DATA_SOURCES = (
34
34
  (DATA_SOURCE_MOUSER, 'mouser'),
35
35
  )
36
36
 
37
+ PART_REVISION_PROPERTY_TYPE_STRING = 'S'
38
+ PART_REVISION_PROPERTY_TYPE_DECIMAL = 'D'
39
+ PART_REVISION_PROPERTY_TYPE_BOOLEAN = 'B'
40
+ PART_REVISION_PROPERTY_TYPES = ((PART_REVISION_PROPERTY_TYPE_STRING, 'String'),
41
+ (PART_REVISION_PROPERTY_TYPE_DECIMAL, 'Number'),
42
+ (PART_REVISION_PROPERTY_TYPE_BOOLEAN, 'Boolean'),)
43
+
37
44
  VALUE_UNITS = (
38
45
  NO_CHOICE,
39
46
  ('Ohms', '\u03A9'),
40
47
  ('mOhms', 'm\u03A9'),
41
48
  ('kOhms', 'k\u03A9'),
49
+ ('MOhms', 'M\u03A9'),
42
50
  ('F', 'F'),
43
51
  ('pF', 'pF'),
44
52
  ('nF', 'nF'),
@@ -52,6 +60,7 @@ VALUE_UNITS = (
52
60
  ('C', '\u00B0C'),
53
61
  ('F', '\u00B0F'),
54
62
  ('H', 'H'),
63
+ ('nH', 'nH'),
55
64
  ('mH', 'mH'),
56
65
  ('uH', '\u03BCH'),
57
66
  ('Hz', 'Hz'),
@@ -90,6 +99,8 @@ PACKAGE_TYPES = (
90
99
  ('QFT', 'QFT'),
91
100
  ('PLCC', 'PLCC'),
92
101
  ('VGA', 'VGA'),
102
+ ('BGA', 'BGA'),
103
+ ('CSP', 'CSP'),
93
104
  ('Other', 'Other'),
94
105
  )
95
106
 
@@ -1,4 +1,5 @@
1
1
  from abc import ABC
2
+
2
3
  from .utils import get_from_dict
3
4
 
4
5
 
@@ -34,8 +35,11 @@ class CSVHeader:
34
35
  class CSVHeaders(ABC):
35
36
  all_headers_defns = []
36
37
 
38
+ def __init__(self, *args, **kwargs):
39
+ self.headers_defns = list(self.all_headers_defns)
40
+
37
41
  def get_synoynms(self, hdr_name):
38
- for defn in self.all_headers_defns:
42
+ for defn in self.headers_defns:
39
43
  if hdr_name in defn:
40
44
  return defn.synonyms()
41
45
  else:
@@ -52,7 +56,7 @@ class CSVHeaders(ABC):
52
56
 
53
57
  # Preserves order of definitions as listed in all_header_defns:
54
58
  def get_default_all(self):
55
- return [d.name for d in self.all_headers_defns]
59
+ return [d.name for d in self.headers_defns]
56
60
 
57
61
  # Given a list of header names returns the default name for each. The return list
58
62
  # matches the order of the input list. If a name is not recognized, then returns
@@ -143,6 +147,16 @@ class CSVHeaders(ABC):
143
147
  c = evaluate(headers, operand, operator, c, i + 1 == num_asserts)
144
148
  i += 2
145
149
 
150
+ def validate_header_in(self, headers, header):
151
+ header_synonyms = self.get_synoynms(header)
152
+ for hs in header_synonyms:
153
+ if hs in headers:
154
+ return True
155
+ raise CSVHeaderError(f'Missing column named {header}')
156
+
157
+ def add_header(self, header):
158
+ self.headers_defns.append(header)
159
+
146
160
 
147
161
  #
148
162
  # For each CSV header class, a static data member dictionary uses the key as the default name for the header while the
@@ -164,9 +178,12 @@ class ManufacturerPartCSVHeaders(CSVHeaders):
164
178
  class SellerPartCSVHeaders(CSVHeaders):
165
179
  all_headers_defns = [
166
180
  CSVHeader('seller', name_options=['part_seller', 'part_seller_name', ]),
167
- CSVHeader('part_cost', name_options=['seller_part_unit_cost', 'unit_cost', ]),
168
- CSVHeader('moq', name_options=['minimum_order_quantity', 'moq', 'part_moq', ]),
169
- CSVHeader('nre', name_options=['part_nre', 'part_nre_cost', ]),
181
+ CSVHeader('seller_part_number', name_options=['seller_pn', 'spn', 'pn', 'part_seller_part_number', 'seller_part_number', ]),
182
+ CSVHeader('unit_cost', name_options=['seller_part_unit_cost', 'unit_cost', 'part_cost', ]),
183
+ CSVHeader('minimum_order_quantity', name_options=['minimum_order_quantity', 'moq', 'part_moq', ]),
184
+ CSVHeader('nre_cost', name_options=['part_nre', 'part_nre_cost', 'nre', 'nre_cost', ]),
185
+ CSVHeader('minimum_pack_quantity', name_options=['minimum_pack_quantity', 'mpq', 'part_mpq' ]),
186
+ CSVHeader('lead_time_days', name_options=['lead_time_days', 'lead_time', 'lt']),
170
187
  ]
171
188
 
172
189
 
@@ -179,51 +196,25 @@ class PartClassesCSVHeaders(CSVHeaders):
179
196
  ]
180
197
 
181
198
 
182
- class PartsListCSVHeaders(CSVHeaders):
183
- part_attributes = [
184
- CSVHeader('value', name_options=['val', 'val.', ]),
185
- CSVHeader('value_units', name_options=['value units', 'val. units', 'val units', ]),
186
- CSVHeader('tolerance', name_options=[]),
187
- CSVHeader('attribute', name_options=[]),
188
- CSVHeader('package', name_options=[]),
189
- CSVHeader('pin_count', name_options=[]),
190
- CSVHeader('frequency', name_options=[]),
191
- CSVHeader('frequency_units', name_options=[]),
192
- CSVHeader('wavelength', name_options=[]),
193
- CSVHeader('wavelength_units', name_options=[]),
194
- CSVHeader('memory', name_options=[]),
195
- CSVHeader('memory_units', name_options=[]),
196
- CSVHeader('interface', name_options=[]),
197
- CSVHeader('supply_voltage', name_options=[]),
198
- CSVHeader('supply_voltage_units', name_options=[]),
199
- CSVHeader('temperature_rating', name_options=[]),
200
- CSVHeader('temperature_rating_units', name_options=[]),
201
- CSVHeader('power_rating', name_options=[]),
202
- CSVHeader('power_rating_units', name_options=[]),
203
- CSVHeader('voltage_rating', name_options=[]),
204
- CSVHeader('voltage_rating_units', name_options=[]),
205
- CSVHeader('current_rating', name_options=[]),
206
- CSVHeader('current_rating_units', name_options=[]),
207
- CSVHeader('material', name_options=[]),
208
- CSVHeader('color', name_options=[]),
209
- CSVHeader('finish', name_options=[]),
210
- CSVHeader('length', name_options=[]),
211
- CSVHeader('length_units', name_options=[]),
212
- CSVHeader('width', name_options=[]),
213
- CSVHeader('width_units', name_options=[]),
214
- CSVHeader('height', name_options=[]),
215
- CSVHeader('height_units', name_options=[]),
216
- CSVHeader('weight', name_options=[]),
217
- CSVHeader('weight_units', name_options=[]),
218
- ]
199
+ class PartRevisionPropertyCSVHeaders(CSVHeaders):
200
+ def add_dynamic_headers(self, definitions):
201
+ for defn in definitions:
202
+ self.add_header(
203
+ CSVHeader(defn.form_field_name,
204
+ name_options=[defn.name, f'property_{defn.code}', f'property_{defn.name}', 'p']))
205
+ if defn.quantity_of_measure:
206
+ self.add_header(
207
+ CSVHeader(defn.form_unit_field_name, name_options=[f'{defn.name} Units', f'{defn.code}_units']))
208
+
219
209
 
210
+ class PartsListCSVHeaders(PartRevisionPropertyCSVHeaders):
220
211
  all_headers_defns = [
221
212
  CSVHeader('description', name_options=['desc', 'desc.', ]),
222
213
  CSVHeader('manufacturer_name', name_options=['mfg_name', 'manufacturer_name', 'part_manufacturer', 'mfg', 'manufacturer', 'manufacturer name', ]),
223
214
  CSVHeader('manufacturer_part_number', name_options=['mpn', 'mfg_part_number', 'part_manufacturer_part_number', 'mfg part number', 'manufacturer part number']),
224
215
  CSVHeader('part_number', name_options=['part number', 'part no', ]),
225
216
  CSVHeader('revision', name_options=['rev', 'part_revision', ]),
226
- ] + part_attributes
217
+ ] + SellerPartCSVHeaders.all_headers_defns
227
218
 
228
219
 
229
220
  class PartsListCSVHeadersSemiIntelligent(PartsListCSVHeaders):
@@ -234,17 +225,18 @@ class PartsListCSVHeadersSemiIntelligent(PartsListCSVHeaders):
234
225
  CSVHeader('part_class', name_options=['class', 'part_category']),
235
226
  CSVHeader('part_number', name_options=['part number', 'part no', ]),
236
227
  CSVHeader('revision', name_options=['rev', 'part_revision', ]),
237
- ] + PartsListCSVHeaders.part_attributes
228
+ ] + SellerPartCSVHeaders.all_headers_defns
238
229
 
239
230
 
240
- class BOMFlatCSVHeaders(CSVHeaders):
231
+ class BOMFlatCSVHeaders(PartRevisionPropertyCSVHeaders):
241
232
  all_headers_defns = [
242
233
  CSVHeader('part_number', name_options=['part number', 'part no', ]),
243
- CSVHeader('quantity', name_options=['count', 'qty', ]),
234
+ CSVHeader('quantity', name_options=['count', 'qty', 'quantity', ]),
244
235
  CSVHeader('do_not_load', name_options=['dnl', 'dnp', 'do_not_populate', 'do_not_load', 'do not load', 'do not populate', ]),
245
236
  CSVHeader('part_class', name_options=['class', 'part_category']),
246
237
  CSVHeader('references', name_options=['designator', 'designators', 'reference', ]),
247
238
  CSVHeader('synopsis', name_options=['part_synopsis', ]),
239
+ CSVHeader('description', name_options=['part_description', 'desc', ]),
248
240
  CSVHeader('revision', name_options=['rev', 'part_revision', 'rev.']),
249
241
  ] + ManufacturerPartCSVHeaders.all_headers_defns \
250
242
  + SellerPartCSVHeaders.all_headers_defns + [
@@ -0,0 +1,32 @@
1
+ from django.contrib import messages
2
+ from django.core.exceptions import PermissionDenied
3
+ from django.http import HttpResponseRedirect
4
+ from django.urls import reverse
5
+
6
+ from social_django.models import UserSocialAuth
7
+
8
+
9
+ def google_authenticated(function):
10
+ def wrap(request, *args, **kwargs):
11
+ user = request.user
12
+ try:
13
+ user.social_auth.get(provider='google-oauth2')
14
+ return function(request, *args, **kwargs)
15
+ except UserSocialAuth.DoesNotExist as e:
16
+ messages.error(request, "You must Sign in with Google to access this feature.")
17
+ return HttpResponseRedirect(reverse('bom:settings', kwargs={'tab_anchor': 'organization'}))
18
+
19
+ wrap.__doc__ = function.__doc__
20
+ wrap.__name__ = function.__name__
21
+ return wrap
22
+
23
+ def organization_admin(function):
24
+ def wrap(request, *args, **kwargs):
25
+ if request.user.bom_profile().role != 'A':
26
+ messages.error(request, "You don't have permission to perform this action.")
27
+ return HttpResponseRedirect(request.META.get('HTTP_REFERER'), reverse('bom:home'))
28
+ return function(request, *args, **kwargs)
29
+
30
+ wrap.__doc__ = function.__doc__
31
+ wrap.__name__ = function.__name__
32
+ return wrap