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.
- django_bom-1.260/PKG-INFO +206 -0
- {django-bom-1.193 → django_bom-1.260}/README.md +7 -8
- {django-bom-1.193 → django_bom-1.260}/bom/admin.py +75 -8
- django_bom-1.260/bom/auth_backends.py +47 -0
- {django-bom-1.193 → django_bom-1.260}/bom/base_classes.py +1 -1
- {django-bom-1.193 → django_bom-1.260}/bom/constants.py +11 -0
- {django-bom-1.193 → django_bom-1.260}/bom/csv_headers.py +38 -46
- django_bom-1.260/bom/decorators.py +32 -0
- django_bom-1.260/bom/forms.py +1328 -0
- {django-bom-1.193 → django_bom-1.260}/bom/helpers.py +97 -7
- django_bom-1.260/bom/local_settings.py +35 -0
- django_bom-1.260/bom/migrations/0040_alter_organization_currency.py +19 -0
- django_bom-1.260/bom/migrations/0041_organization_subscription_quantity.py +18 -0
- django_bom-1.260/bom/migrations/0042_auto_20210720_2137.py +23 -0
- django_bom-1.260/bom/migrations/0043_auto_20211123_0157.py +24 -0
- django_bom-1.260/bom/migrations/0044_auto_20220831_1241.py +23 -0
- django_bom-1.260/bom/migrations/0045_sellerpart_link.py +18 -0
- django_bom-1.260/bom/migrations/0046_alter_sellerpart_unique_together.py +17 -0
- django_bom-1.260/bom/migrations/0047_sellerpart_seller_part_number.py +18 -0
- django_bom-1.260/bom/migrations/0048_rename_part_organization_number_class_bom_part_organiz_b333d6_idx_and_more.py +1017 -0
- django_bom-1.260/bom/migrations/0049_alter_assembly_id_alter_assemblysubparts_id_and_more.py +99 -0
- django_bom-1.260/bom/migrations/0050_alter_organization_options.py +17 -0
- django_bom-1.260/bom/migrations/0051_alter_manufacturer_organization_and_more.py +41 -0
- django_bom-1.260/bom/migrations/0052_remove_partrevision_attribute_and_more.py +584 -0
- {django-bom-1.193 → django_bom-1.260}/bom/models.py +386 -182
- {django-bom-1.193 → django_bom-1.260}/bom/part_bom.py +21 -10
- django_bom-1.260/bom/settings.py +262 -0
- django_bom-1.260/bom/static/bom/css/dashboard.css +17 -0
- django_bom-1.260/bom/static/bom/css/part-info.css +15 -0
- django_bom-1.260/bom/static/bom/css/style.css +418 -0
- {django-bom-1.193 → django_bom-1.260}/bom/static/bom/css/treetable-theme.css +10 -10
- django_bom-1.260/bom/static/bom/doc/sample_part_classes.csv +38 -0
- django_bom-1.260/bom/static/bom/doc/test_bom.csv +6 -0
- django_bom-1.260/bom/static/bom/doc/test_bom_5_intelligent.csv +4 -0
- django_bom-1.260/bom/static/bom/doc/test_full_bom.csv +37 -0
- django_bom-1.260/bom/static/bom/doc/test_new_parts.csv +5 -0
- django_bom-1.260/bom/static/bom/doc/test_new_parts_5_intelligent.csv +5 -0
- django_bom-1.260/bom/static/bom/js/formset-handler.js +65 -0
- django_bom-1.260/bom/templates/bom/account-delete.html +23 -0
- {django-bom-1.193 → django_bom-1.260}/bom/templates/bom/add-manufacturer-part.html +2 -5
- {django-bom-1.193 → django_bom-1.260}/bom/templates/bom/add-sellerpart.html +9 -5
- {django-bom-1.193 → django_bom-1.260}/bom/templates/bom/base-menu.html +10 -1
- {django-bom-1.193 → django_bom-1.260}/bom/templates/bom/base.html +3 -2
- {django-bom-1.193 → django_bom-1.260}/bom/templates/bom/bom-action-table.html +2 -2
- django_bom-1.260/bom/templates/bom/bom-base-menu.html +6 -0
- {django-bom-1.193 → django_bom-1.260}/bom/templates/bom/bom-base.html +1 -0
- django_bom-1.260/bom/templates/bom/bom-form-modal.html +36 -0
- {django-bom-1.193 → django_bom-1.260}/bom/templates/bom/bom-form.html +1 -5
- django_bom-1.260/bom/templates/bom/bom-modal-add-users.html +49 -0
- {django-bom-1.193 → django_bom-1.260}/bom/templates/bom/bom-signup.html +0 -4
- {django-bom-1.193/bom/templates/bom → django_bom-1.260/bom/templates/bom/components}/bom-flat.html +5 -5
- {django-bom-1.193/bom/templates/bom → django_bom-1.260/bom/templates/bom/components}/bom-indented.html +40 -7
- django_bom-1.260/bom/templates/bom/components/manufacturer-part-list.html +270 -0
- django_bom-1.260/bom/templates/bom/components/seller-part-list.html +62 -0
- {django-bom-1.193 → django_bom-1.260}/bom/templates/bom/create-part.html +2 -5
- django_bom-1.260/bom/templates/bom/dashboard.html +303 -0
- {django-bom-1.193 → django_bom-1.260}/bom/templates/bom/edit-manufacturer-part.html +8 -6
- django_bom-1.260/bom/templates/bom/edit-part-class.html +120 -0
- {django-bom-1.193 → django_bom-1.260}/bom/templates/bom/edit-part.html +2 -6
- django_bom-1.260/bom/templates/bom/edit-quantity-of-measure.html +119 -0
- django_bom-1.260/bom/templates/bom/edit-user-meta.html +70 -0
- django_bom-1.260/bom/templates/bom/help.html +1356 -0
- django_bom-1.260/bom/templates/bom/manufacturer-info.html +82 -0
- django_bom-1.260/bom/templates/bom/manufacturers.html +97 -0
- {django-bom-1.193 → django_bom-1.260}/bom/templates/bom/organization-create.html +8 -8
- {django-bom-1.193 → django_bom-1.260}/bom/templates/bom/part-info.html +31 -252
- django_bom-1.260/bom/templates/bom/part-revision-display.html +50 -0
- {django-bom-1.193 → django_bom-1.260}/bom/templates/bom/part-revision-edit.html +2 -5
- {django-bom-1.193 → django_bom-1.260}/bom/templates/bom/part-revision-manage-bom.html +2 -3
- {django-bom-1.193 → django_bom-1.260}/bom/templates/bom/part-revision-new.html +2 -5
- {django-bom-1.193 → django_bom-1.260}/bom/templates/bom/part-revision-release.html +0 -7
- {django-bom-1.193 → django_bom-1.260}/bom/templates/bom/search-help.html +0 -4
- django_bom-1.260/bom/templates/bom/seller-info.html +82 -0
- django_bom-1.260/bom/templates/bom/sellers.html +97 -0
- django_bom-1.260/bom/templates/bom/settings.html +592 -0
- {django-bom-1.193 → django_bom-1.260}/bom/templates/bom/signup.html +1 -1
- django_bom-1.260/bom/templates/bom/subscription_panel.html +16 -0
- django_bom-1.260/bom/templates/bom/table_of_contents.html +47 -0
- {django-bom-1.193 → django_bom-1.260}/bom/templates/bom/upload-bom.html +21 -26
- {django-bom-1.193 → django_bom-1.260}/bom/templates/bom/upload-parts-help.html +18 -7
- {django-bom-1.193 → django_bom-1.260}/bom/templates/bom/upload-parts.html +0 -4
- {django-bom-1.193 → django_bom-1.260}/bom/tests.py +365 -75
- {django-bom-1.193 → django_bom-1.260}/bom/third_party_apis/base_api.py +1 -1
- {django-bom-1.193 → django_bom-1.260}/bom/third_party_apis/google_drive.py +14 -6
- {django-bom-1.193 → django_bom-1.260}/bom/third_party_apis/mouser.py +1 -0
- {django-bom-1.193 → django_bom-1.260}/bom/urls.py +31 -10
- {django-bom-1.193 → django_bom-1.260}/bom/views/json_views.py +3 -3
- {django-bom-1.193 → django_bom-1.260}/bom/views/views.py +521 -165
- django_bom-1.260/django_bom.egg-info/PKG-INFO +206 -0
- {django-bom-1.193 → django_bom-1.260}/django_bom.egg-info/SOURCES.txt +38 -6
- django_bom-1.260/django_bom.egg-info/requires.txt +6 -0
- django_bom-1.260/pyproject.toml +70 -0
- django-bom-1.193/PKG-INFO +0 -196
- django-bom-1.193/bom/decorators.py +0 -18
- django-bom-1.193/bom/forms.py +0 -1268
- django-bom-1.193/bom/local_settings.py +0 -36
- django-bom-1.193/bom/settings.py +0 -219
- django-bom-1.193/bom/static/.DS_Store +0 -0
- django-bom-1.193/bom/static/bom/.DS_Store +0 -0
- django-bom-1.193/bom/static/bom/css/style.css +0 -207
- django-bom-1.193/bom/static/bom/img/.DS_Store +0 -0
- django-bom-1.193/bom/templates/bom/bom-base-menu.html +0 -4
- django-bom-1.193/bom/templates/bom/dashboard.html +0 -295
- django-bom-1.193/bom/templates/bom/edit-part-class.html +0 -36
- django-bom-1.193/bom/templates/bom/edit-user-meta.html +0 -41
- django-bom-1.193/bom/templates/bom/help.html +0 -1108
- django-bom-1.193/bom/templates/bom/part-revision-display.html +0 -178
- django-bom-1.193/bom/templates/bom/settings.html +0 -402
- django-bom-1.193/django_bom.egg-info/PKG-INFO +0 -196
- django-bom-1.193/django_bom.egg-info/requires.txt +0 -5
- django-bom-1.193/setup.py +0 -41
- {django-bom-1.193 → django_bom-1.260}/LICENSE +0 -0
- {django-bom-1.193 → django_bom-1.260}/MANIFEST.in +0 -0
- {django-bom-1.193 → django_bom-1.260}/bom/__init__.py +0 -0
- {django-bom-1.193 → django_bom-1.260}/bom/apps.py +0 -0
- {django-bom-1.193 → django_bom-1.260}/bom/context_processors.py +0 -0
- {django-bom-1.193 → django_bom-1.260}/bom/form_fields.py +0 -0
- {django-bom-1.193 → django_bom-1.260}/bom/migrations/0001_initial.py +0 -0
- {django-bom-1.193 → django_bom-1.260}/bom/migrations/0002_auto_20180908_2151.py +0 -0
- {django-bom-1.193 → django_bom-1.260}/bom/migrations/0003_sellerpart_data_source.py +0 -0
- {django-bom-1.193 → django_bom-1.260}/bom/migrations/0004_auto_20180911_0011.py +0 -0
- {django-bom-1.193 → django_bom-1.260}/bom/migrations/0005_auto_20181007_1934.py +0 -0
- {django-bom-1.193 → django_bom-1.260}/bom/migrations/0006_auto_20181007_1949.py +0 -0
- {django-bom-1.193 → django_bom-1.260}/bom/migrations/0007_auto_20181009_0256.py +0 -0
- {django-bom-1.193 → django_bom-1.260}/bom/migrations/0008_auto_20181030_0427.py +0 -0
- {django-bom-1.193 → django_bom-1.260}/bom/migrations/0009_subpart_reference.py +0 -0
- {django-bom-1.193 → django_bom-1.260}/bom/migrations/0010_auto_20181202_0733.py +0 -0
- {django-bom-1.193 → django_bom-1.260}/bom/migrations/0011_auto_20181202_2113.py +0 -0
- {django-bom-1.193 → django_bom-1.260}/bom/migrations/0012_partchangehistory.py +0 -0
- {django-bom-1.193 → django_bom-1.260}/bom/migrations/0013_auto_20190222_1631.py +0 -0
- {django-bom-1.193 → django_bom-1.260}/bom/migrations/0014_auto_20190223_2353.py +0 -0
- {django-bom-1.193 → django_bom-1.260}/bom/migrations/0015_auto_20190303_1915.py +0 -0
- {django-bom-1.193 → django_bom-1.260}/bom/migrations/0016_auto_20190405_2308.py +0 -0
- {django-bom-1.193 → django_bom-1.260}/bom/migrations/0017_auto_20190616_1912.py +0 -0
- {django-bom-1.193 → django_bom-1.260}/bom/migrations/0018_auto_20190616_2143.py +0 -0
- {django-bom-1.193 → django_bom-1.260}/bom/migrations/0019_auto_20190624_1246.py +0 -0
- {django-bom-1.193 → django_bom-1.260}/bom/migrations/0020_auto_20190627_0207.py +0 -0
- {django-bom-1.193 → django_bom-1.260}/bom/migrations/0021_auto_20190627_0428.py +0 -0
- {django-bom-1.193 → django_bom-1.260}/bom/migrations/0022_auto_20190811_2140.py +0 -0
- {django-bom-1.193 → django_bom-1.260}/bom/migrations/0023_auto_20191205_2351.py +0 -0
- {django-bom-1.193 → django_bom-1.260}/bom/migrations/0024_auto_20191214_1342.py +0 -0
- {django-bom-1.193 → django_bom-1.260}/bom/migrations/0025_auto_20191221_1907.py +0 -0
- {django-bom-1.193 → django_bom-1.260}/bom/migrations/0026_auto_20191222_2258.py +0 -0
- {django-bom-1.193 → django_bom-1.260}/bom/migrations/0027_auto_20191222_2347.py +0 -0
- {django-bom-1.193 → django_bom-1.260}/bom/migrations/0028_partrevision_displayable_synopsis.py +0 -0
- {django-bom-1.193 → django_bom-1.260}/bom/migrations/0029_auto_20191231_1630.py +0 -0
- {django-bom-1.193 → django_bom-1.260}/bom/migrations/0030_auto_20200101_2253.py +0 -0
- {django-bom-1.193 → django_bom-1.260}/bom/migrations/0031_auto_20200104_1352.py +0 -0
- {django-bom-1.193 → django_bom-1.260}/bom/migrations/0032_auto_20200126_1806.py +0 -0
- {django-bom-1.193 → django_bom-1.260}/bom/migrations/0033_auto_20200203_0618.py +0 -0
- {django-bom-1.193 → django_bom-1.260}/bom/migrations/0034_auto_20200222_0359.py +0 -0
- {django-bom-1.193 → django_bom-1.260}/bom/migrations/0035_auto_20200303_0111.py +0 -0
- {django-bom-1.193 → django_bom-1.260}/bom/migrations/0036_auto_20200303_0538.py +0 -0
- {django-bom-1.193 → django_bom-1.260}/bom/migrations/0037_auto_20200405_1642.py +0 -0
- {django-bom-1.193 → django_bom-1.260}/bom/migrations/0038_auto_20200422_0504.py +0 -0
- {django-bom-1.193 → django_bom-1.260}/bom/migrations/0039_auto_20200929_2315.py +0 -0
- {django-bom-1.193 → django_bom-1.260}/bom/migrations/__init__.py +0 -0
- {django-bom-1.193 → django_bom-1.260}/bom/static/bom/css/jquery.treetable.css +0 -0
- {django-bom-1.193 → django_bom-1.260}/bom/static/bom/css/materialize.min.css +0 -0
- {django-bom-1.193 → django_bom-1.260}/bom/static/bom/css/tablesorter-theme.materialize.css +0 -0
- {django-bom-1.193 → django_bom-1.260}/bom/static/bom/img/_ionicons_svg_md-arrow-dropdown.svg +0 -0
- {django-bom-1.193 → django_bom-1.260}/bom/static/bom/img/_ionicons_svg_md-arrow-dropright.svg +0 -0
- {django-bom-1.193 → django_bom-1.260}/bom/static/bom/img/favicon.ico +0 -0
- {django-bom-1.193 → django_bom-1.260}/bom/static/bom/img/google/web/1x/btn_google_signin_dark_disabled_web.png +0 -0
- {django-bom-1.193 → django_bom-1.260}/bom/static/bom/img/google/web/1x/btn_google_signin_dark_focus_web.png +0 -0
- {django-bom-1.193 → django_bom-1.260}/bom/static/bom/img/google/web/1x/btn_google_signin_dark_normal_web.png +0 -0
- {django-bom-1.193 → django_bom-1.260}/bom/static/bom/img/google/web/1x/btn_google_signin_dark_pressed_web.png +0 -0
- {django-bom-1.193 → django_bom-1.260}/bom/static/bom/img/google/web/1x/btn_google_signin_light_disabled_web.png +0 -0
- {django-bom-1.193 → django_bom-1.260}/bom/static/bom/img/google/web/1x/btn_google_signin_light_focus_web.png +0 -0
- {django-bom-1.193 → django_bom-1.260}/bom/static/bom/img/google/web/1x/btn_google_signin_light_normal_web.png +0 -0
- {django-bom-1.193 → django_bom-1.260}/bom/static/bom/img/google/web/1x/btn_google_signin_light_pressed_web.png +0 -0
- {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
- {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
- {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
- {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
- {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
- {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
- {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
- {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
- {django-bom-1.193 → django_bom-1.260}/bom/static/bom/img/google/web/vector/btn_google_dark_disabled_ios.eps +0 -0
- {django-bom-1.193 → django_bom-1.260}/bom/static/bom/img/google/web/vector/btn_google_dark_disabled_ios.svg +0 -0
- {django-bom-1.193 → django_bom-1.260}/bom/static/bom/img/google/web/vector/btn_google_dark_focus_ios.eps +0 -0
- {django-bom-1.193 → django_bom-1.260}/bom/static/bom/img/google/web/vector/btn_google_dark_focus_ios.svg +0 -0
- {django-bom-1.193 → django_bom-1.260}/bom/static/bom/img/google/web/vector/btn_google_dark_normal_ios.eps +0 -0
- {django-bom-1.193 → django_bom-1.260}/bom/static/bom/img/google/web/vector/btn_google_dark_normal_ios.svg +0 -0
- {django-bom-1.193 → django_bom-1.260}/bom/static/bom/img/google/web/vector/btn_google_dark_pressed_ios.eps +0 -0
- {django-bom-1.193 → django_bom-1.260}/bom/static/bom/img/google/web/vector/btn_google_dark_pressed_ios.svg +0 -0
- {django-bom-1.193 → django_bom-1.260}/bom/static/bom/img/google/web/vector/btn_google_light_disabled_ios.eps +0 -0
- {django-bom-1.193 → django_bom-1.260}/bom/static/bom/img/google/web/vector/btn_google_light_disabled_ios.svg +0 -0
- {django-bom-1.193 → django_bom-1.260}/bom/static/bom/img/google/web/vector/btn_google_light_focus_ios.eps +0 -0
- {django-bom-1.193 → django_bom-1.260}/bom/static/bom/img/google/web/vector/btn_google_light_focus_ios.svg +0 -0
- {django-bom-1.193 → django_bom-1.260}/bom/static/bom/img/google/web/vector/btn_google_light_normal_ios.eps +0 -0
- {django-bom-1.193 → django_bom-1.260}/bom/static/bom/img/google/web/vector/btn_google_light_normal_ios.svg +0 -0
- {django-bom-1.193 → django_bom-1.260}/bom/static/bom/img/google/web/vector/btn_google_light_pressed_ios.eps +0 -0
- {django-bom-1.193 → django_bom-1.260}/bom/static/bom/img/google/web/vector/btn_google_light_pressed_ios.svg +0 -0
- {django-bom-1.193 → django_bom-1.260}/bom/static/bom/img/google_drive_logo.svg +0 -0
- {django-bom-1.193 → django_bom-1.260}/bom/static/bom/img/indabom.png +0 -0
- {django-bom-1.193 → django_bom-1.260}/bom/static/bom/img/mouser.png +0 -0
- {django-bom-1.193 → django_bom-1.260}/bom/static/bom/img/octopart_blue.svg +0 -0
- {django-bom-1.193 → django_bom-1.260}/bom/static/bom/js/jquery-3.4.1.min.js +0 -0
- {django-bom-1.193 → django_bom-1.260}/bom/static/bom/js/jquery.ba-floatingscrollbar.min.js +0 -0
- {django-bom-1.193 → django_bom-1.260}/bom/static/bom/js/jquery.treetable.js +0 -0
- {django-bom-1.193 → django_bom-1.260}/bom/static/bom/js/materialize.min.js +0 -0
- {django-bom-1.193 → django_bom-1.260}/bom/templates/bom/bom-action-btn.html +0 -0
- {django-bom-1.193 → django_bom-1.260}/bom/templates/bom/dashboard-menu.html +0 -0
- {django-bom-1.193 → django_bom-1.260}/bom/templates/bom/nothing-to-see.html +0 -0
- {django-bom-1.193 → django_bom-1.260}/bom/templates/registration/login.html +0 -0
- {django-bom-1.193 → django_bom-1.260}/bom/third_party_apis/__init__.py +0 -0
- {django-bom-1.193 → django_bom-1.260}/bom/third_party_apis/test_apis.py +0 -0
- {django-bom-1.193 → django_bom-1.260}/bom/utils.py +0 -0
- {django-bom-1.193 → django_bom-1.260}/bom/validators.py +0 -0
- {django-bom-1.193 → django_bom-1.260}/bom/views/__init__.py +0 -0
- {django-bom-1.193 → django_bom-1.260}/bom/wsgi.py +0 -0
- {django-bom-1.193 → django_bom-1.260}/django_bom.egg-info/dependency_links.txt +0 -0
- {django-bom-1.193 → django_bom-1.260}/django_bom.egg-info/top_level.txt +0 -0
- {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
|
+

|
|
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
|
|
22
|
+
1. To start from scratch, install your pipenv using Python 3.12
|
|
23
23
|
```
|
|
24
|
-
|
|
25
|
-
|
|
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
|
-
|
|
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/).
|
|
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
|
-
|
|
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.
|
|
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
|
|
@@ -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.
|
|
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.
|
|
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('
|
|
168
|
-
CSVHeader('
|
|
169
|
-
CSVHeader('
|
|
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
|
|
183
|
-
|
|
184
|
-
|
|
185
|
-
|
|
186
|
-
|
|
187
|
-
|
|
188
|
-
|
|
189
|
-
|
|
190
|
-
|
|
191
|
-
|
|
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
|
-
|
|
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
|
-
|
|
228
|
+
] + SellerPartCSVHeaders.all_headers_defns
|
|
238
229
|
|
|
239
230
|
|
|
240
|
-
class BOMFlatCSVHeaders(
|
|
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
|