2019-09-02 23:07:03 +00:00
|
|
|
"""
|
|
|
|
Common database model definitions.
|
|
|
|
These models are 'generic' and do not fit a particular business logic object.
|
|
|
|
"""
|
|
|
|
|
|
|
|
# -*- coding: utf-8 -*-
|
|
|
|
from __future__ import unicode_literals
|
|
|
|
|
2020-09-09 19:55:32 +00:00
|
|
|
import os
|
2021-05-05 21:42:52 +00:00
|
|
|
import decimal
|
|
|
|
import math
|
2020-09-09 19:55:32 +00:00
|
|
|
|
2020-11-13 11:22:02 +00:00
|
|
|
from django.db import models, transaction
|
2020-11-13 10:37:39 +00:00
|
|
|
from django.db.utils import IntegrityError, OperationalError
|
2020-09-09 19:55:32 +00:00
|
|
|
from django.conf import settings
|
2020-10-25 10:00:06 +00:00
|
|
|
|
2020-11-10 06:08:08 +00:00
|
|
|
import djmoney.settings
|
2020-11-10 11:25:05 +00:00
|
|
|
from djmoney.models.fields import MoneyField
|
2020-11-10 13:21:06 +00:00
|
|
|
from djmoney.contrib.exchange.models import convert_money
|
|
|
|
from djmoney.contrib.exchange.exceptions import MissingRate
|
2020-11-10 06:08:08 +00:00
|
|
|
|
2021-04-03 02:01:40 +00:00
|
|
|
from django.utils.translation import ugettext_lazy as _
|
2021-02-16 04:31:04 +00:00
|
|
|
from django.core.validators import MinValueValidator, URLValidator
|
2019-09-15 12:46:24 +00:00
|
|
|
from django.core.exceptions import ValidationError
|
|
|
|
|
2020-10-24 22:36:58 +00:00
|
|
|
import InvenTree.helpers
|
2020-09-17 12:44:17 +00:00
|
|
|
import InvenTree.fields
|
|
|
|
|
2019-09-15 12:46:24 +00:00
|
|
|
|
|
|
|
class InvenTreeSetting(models.Model):
|
|
|
|
"""
|
|
|
|
An InvenTreeSetting object is a key:value pair used for storing
|
|
|
|
single values (e.g. one-off settings values).
|
|
|
|
|
|
|
|
The class provides a way of retrieving the value for a particular key,
|
|
|
|
even if that key does not exist.
|
|
|
|
"""
|
|
|
|
|
2020-10-24 20:49:38 +00:00
|
|
|
"""
|
|
|
|
Dict of all global settings values:
|
2020-10-19 21:24:23 +00:00
|
|
|
|
2020-10-24 20:49:38 +00:00
|
|
|
The key of each item is the name of the value as it appears in the database.
|
2020-10-19 21:24:23 +00:00
|
|
|
|
2020-10-24 20:49:38 +00:00
|
|
|
Each global setting has the following parameters:
|
2021-05-06 10:11:38 +00:00
|
|
|
|
2020-10-24 21:04:04 +00:00
|
|
|
- name: Translatable string name of the setting (required)
|
2020-10-24 20:49:38 +00:00
|
|
|
- description: Translatable string description of the setting (required)
|
|
|
|
- default: Default value (optional)
|
|
|
|
- units: Units of the particular setting (optional)
|
|
|
|
- validator: Validation function for the setting (optional)
|
2020-10-19 21:24:23 +00:00
|
|
|
|
2020-10-24 20:49:38 +00:00
|
|
|
The keys must be upper-case
|
|
|
|
"""
|
2020-10-19 21:24:23 +00:00
|
|
|
|
2020-10-24 20:49:38 +00:00
|
|
|
GLOBAL_SETTINGS = {
|
|
|
|
|
|
|
|
'INVENTREE_INSTANCE': {
|
|
|
|
'name': _('InvenTree Instance Name'),
|
|
|
|
'default': 'InvenTree server',
|
|
|
|
'description': _('String descriptor for the server instance'),
|
|
|
|
},
|
|
|
|
|
2021-04-15 12:51:11 +00:00
|
|
|
'INVENTREE_INSTANCE_TITLE': {
|
2021-04-18 10:31:17 +00:00
|
|
|
'name': _('Use instance name'),
|
|
|
|
'description': _('Use the instance name in the title-bar'),
|
2021-04-15 12:51:11 +00:00
|
|
|
'validator': bool,
|
|
|
|
'default': False,
|
|
|
|
},
|
|
|
|
|
2020-10-24 21:17:41 +00:00
|
|
|
'INVENTREE_COMPANY_NAME': {
|
|
|
|
'name': _('Company name'),
|
|
|
|
'description': _('Internal company name'),
|
|
|
|
'default': 'My company name',
|
|
|
|
},
|
|
|
|
|
2021-02-16 04:31:04 +00:00
|
|
|
'INVENTREE_BASE_URL': {
|
|
|
|
'name': _('Base URL'),
|
|
|
|
'description': _('Base URL for server instance'),
|
|
|
|
'validator': URLValidator(),
|
|
|
|
'default': '',
|
|
|
|
},
|
|
|
|
|
2020-11-10 06:08:08 +00:00
|
|
|
'INVENTREE_DEFAULT_CURRENCY': {
|
|
|
|
'name': _('Default Currency'),
|
|
|
|
'description': _('Default currency'),
|
|
|
|
'default': 'USD',
|
|
|
|
'choices': djmoney.settings.CURRENCY_CHOICES,
|
|
|
|
},
|
|
|
|
|
2021-03-16 21:40:30 +00:00
|
|
|
'INVENTREE_DOWNLOAD_FROM_URL': {
|
|
|
|
'name': _('Download from URL'),
|
|
|
|
'description': _('Allow download of remote images and files from external URL'),
|
|
|
|
'validator': bool,
|
|
|
|
'default': False,
|
|
|
|
},
|
|
|
|
|
2021-01-28 09:18:03 +00:00
|
|
|
'BARCODE_ENABLE': {
|
|
|
|
'name': _('Barcode Support'),
|
|
|
|
'description': _('Enable barcode scanner support'),
|
|
|
|
'default': True,
|
|
|
|
'validator': bool,
|
|
|
|
},
|
|
|
|
|
2020-10-24 20:49:38 +00:00
|
|
|
'PART_IPN_REGEX': {
|
|
|
|
'name': _('IPN Regex'),
|
|
|
|
'description': _('Regular expression pattern for matching Part IPN')
|
|
|
|
},
|
|
|
|
|
2020-11-09 22:03:26 +00:00
|
|
|
'PART_ALLOW_DUPLICATE_IPN': {
|
|
|
|
'name': _('Allow Duplicate IPN'),
|
|
|
|
'description': _('Allow multiple parts to share the same IPN'),
|
|
|
|
'default': True,
|
|
|
|
'validator': bool,
|
|
|
|
},
|
|
|
|
|
2021-03-12 15:30:31 +00:00
|
|
|
'PART_ALLOW_EDIT_IPN': {
|
|
|
|
'name': _('Allow Editing IPN'),
|
|
|
|
'description': _('Allow changing the IPN value while editing a part'),
|
|
|
|
'default': True,
|
2020-11-09 22:03:26 +00:00
|
|
|
'validator': bool,
|
|
|
|
},
|
|
|
|
|
2020-10-24 20:49:38 +00:00
|
|
|
'PART_COPY_BOM': {
|
|
|
|
'name': _('Copy Part BOM Data'),
|
|
|
|
'description': _('Copy BOM data by default when duplicating a part'),
|
|
|
|
'default': True,
|
2020-10-24 22:36:58 +00:00
|
|
|
'validator': bool,
|
2020-10-24 20:49:38 +00:00
|
|
|
},
|
|
|
|
|
|
|
|
'PART_COPY_PARAMETERS': {
|
|
|
|
'name': _('Copy Part Parameter Data'),
|
|
|
|
'description': _('Copy parameter data by default when duplicating a part'),
|
|
|
|
'default': True,
|
2020-10-24 22:36:58 +00:00
|
|
|
'validator': bool,
|
2020-10-24 20:49:38 +00:00
|
|
|
},
|
|
|
|
|
|
|
|
'PART_COPY_TESTS': {
|
|
|
|
'name': _('Copy Part Test Data'),
|
|
|
|
'description': _('Copy test data by default when duplicating a part'),
|
|
|
|
'default': True,
|
2020-10-24 22:36:58 +00:00
|
|
|
'validator': bool
|
2020-10-24 20:49:38 +00:00
|
|
|
},
|
|
|
|
|
2020-11-02 18:14:31 +00:00
|
|
|
'PART_CATEGORY_PARAMETERS': {
|
2020-11-04 14:52:26 +00:00
|
|
|
'name': _('Copy Category Parameter Templates'),
|
2020-11-11 11:40:11 +00:00
|
|
|
'description': _('Copy category parameter templates when creating a part'),
|
|
|
|
'default': True,
|
|
|
|
'validator': bool
|
|
|
|
},
|
2021-01-03 11:57:39 +00:00
|
|
|
|
2021-02-23 03:12:16 +00:00
|
|
|
'PART_RECENT_COUNT': {
|
|
|
|
'name': _('Recent Part Count'),
|
|
|
|
'description': _('Number of recent parts to display on index page'),
|
|
|
|
'default': 10,
|
2021-02-23 03:24:09 +00:00
|
|
|
'validator': [int, MinValueValidator(1)]
|
2021-02-23 03:12:16 +00:00
|
|
|
},
|
|
|
|
|
2021-01-03 11:57:39 +00:00
|
|
|
'PART_TEMPLATE': {
|
|
|
|
'name': _('Template'),
|
|
|
|
'description': _('Parts are templates by default'),
|
|
|
|
'default': False,
|
|
|
|
'validator': bool,
|
|
|
|
},
|
|
|
|
|
|
|
|
'PART_ASSEMBLY': {
|
|
|
|
'name': _('Assembly'),
|
|
|
|
'description': _('Parts can be assembled from other components by default'),
|
|
|
|
'default': False,
|
|
|
|
'validator': bool,
|
|
|
|
},
|
|
|
|
|
2020-11-09 12:44:54 +00:00
|
|
|
'PART_COMPONENT': {
|
|
|
|
'name': _('Component'),
|
|
|
|
'description': _('Parts can be used as sub-components by default'),
|
|
|
|
'default': True,
|
|
|
|
'validator': bool,
|
|
|
|
},
|
|
|
|
|
2020-11-09 09:26:19 +00:00
|
|
|
'PART_PURCHASEABLE': {
|
|
|
|
'name': _('Purchaseable'),
|
|
|
|
'description': _('Parts are purchaseable by default'),
|
|
|
|
'default': False,
|
|
|
|
'validator': bool,
|
|
|
|
},
|
|
|
|
|
|
|
|
'PART_SALABLE': {
|
|
|
|
'name': _('Salable'),
|
|
|
|
'description': _('Parts are salable by default'),
|
|
|
|
'default': False,
|
|
|
|
'validator': bool,
|
|
|
|
},
|
|
|
|
|
|
|
|
'PART_TRACKABLE': {
|
|
|
|
'name': _('Trackable'),
|
|
|
|
'description': _('Parts are trackable by default'),
|
|
|
|
'default': False,
|
|
|
|
'validator': bool,
|
|
|
|
},
|
|
|
|
|
2021-01-03 12:13:58 +00:00
|
|
|
'PART_VIRTUAL': {
|
|
|
|
'name': _('Virtual'),
|
|
|
|
'description': _('Parts are virtual by default'),
|
2021-01-04 13:59:10 +00:00
|
|
|
'default': False,
|
|
|
|
'validator': bool,
|
|
|
|
},
|
2021-01-04 21:50:07 +00:00
|
|
|
|
2021-01-14 04:20:42 +00:00
|
|
|
'PART_SHOW_QUANTITY_IN_FORMS': {
|
|
|
|
'name': _('Show Quantity in Forms'),
|
|
|
|
'description': _('Display available part quantity in some forms'),
|
|
|
|
'default': True,
|
|
|
|
'validator': bool,
|
|
|
|
},
|
|
|
|
|
2021-02-06 06:36:22 +00:00
|
|
|
'REPORT_DEBUG_MODE': {
|
|
|
|
'name': _('Debug Mode'),
|
|
|
|
'description': _('Generate reports in debug mode (HTML output)'),
|
|
|
|
'default': False,
|
|
|
|
'validator': bool,
|
|
|
|
},
|
|
|
|
|
2021-02-06 06:11:20 +00:00
|
|
|
'REPORT_DEFAULT_PAGE_SIZE': {
|
|
|
|
'name': _('Page Size'),
|
|
|
|
'description': _('Default page size for PDF reports'),
|
|
|
|
'default': 'A4',
|
|
|
|
'choices': [
|
|
|
|
('A4', 'A4'),
|
|
|
|
('Legal', 'Legal'),
|
|
|
|
('Letter', 'Letter')
|
|
|
|
],
|
|
|
|
},
|
|
|
|
|
2021-02-04 10:15:19 +00:00
|
|
|
'REPORT_ENABLE_TEST_REPORT': {
|
|
|
|
'name': _('Test Reports'),
|
|
|
|
'description': _('Enable generation of test reports'),
|
|
|
|
'default': True,
|
|
|
|
'validator': bool,
|
|
|
|
},
|
|
|
|
|
2021-01-04 21:50:07 +00:00
|
|
|
'STOCK_ENABLE_EXPIRY': {
|
|
|
|
'name': _('Stock Expiry'),
|
|
|
|
'description': _('Enable stock expiry functionality'),
|
|
|
|
'default': False,
|
|
|
|
'validator': bool,
|
|
|
|
},
|
|
|
|
|
2021-01-04 13:54:05 +00:00
|
|
|
'STOCK_ALLOW_EXPIRED_SALE': {
|
|
|
|
'name': _('Sell Expired Stock'),
|
|
|
|
'description': _('Allow sale of expired stock'),
|
|
|
|
'default': False,
|
|
|
|
'validator': bool,
|
|
|
|
},
|
|
|
|
|
2021-01-06 11:20:54 +00:00
|
|
|
'STOCK_STALE_DAYS': {
|
|
|
|
'name': _('Stock Stale Time'),
|
|
|
|
'description': _('Number of days stock items are considered stale before expiring'),
|
|
|
|
'default': 0,
|
|
|
|
'units': _('days'),
|
|
|
|
'validator': [int],
|
|
|
|
},
|
|
|
|
|
2021-01-04 13:54:05 +00:00
|
|
|
'STOCK_ALLOW_EXPIRED_BUILD': {
|
|
|
|
'name': _('Build Expired Stock'),
|
|
|
|
'description': _('Allow building with expired stock'),
|
|
|
|
'default': False,
|
|
|
|
'validator': bool,
|
|
|
|
},
|
|
|
|
|
2021-01-08 19:23:35 +00:00
|
|
|
'STOCK_OWNERSHIP_CONTROL': {
|
2020-12-02 18:38:53 +00:00
|
|
|
'name': _('Stock Ownership Control'),
|
|
|
|
'description': _('Enable ownership control over stock locations and items'),
|
2020-12-01 20:54:05 +00:00
|
|
|
'default': False,
|
|
|
|
'validator': bool,
|
|
|
|
},
|
|
|
|
|
2021-02-19 00:31:38 +00:00
|
|
|
'STOCK_GROUP_BY_PART': {
|
|
|
|
'name': _('Group by Part'),
|
|
|
|
'description': _('Group stock items by part reference in table views'),
|
|
|
|
'default': True,
|
|
|
|
'validator': bool,
|
|
|
|
},
|
|
|
|
|
2021-02-23 03:15:12 +00:00
|
|
|
'STOCK_RECENT_COUNT': {
|
|
|
|
'name': _('Recent Stock Count'),
|
|
|
|
'description': _('Number of recent stock items to display on index page'),
|
|
|
|
'default': 10,
|
2021-02-23 03:24:09 +00:00
|
|
|
'validator': [int, MinValueValidator(1)]
|
2021-02-23 03:15:12 +00:00
|
|
|
},
|
|
|
|
|
2020-10-24 20:49:38 +00:00
|
|
|
'BUILDORDER_REFERENCE_PREFIX': {
|
|
|
|
'name': _('Build Order Reference Prefix'),
|
|
|
|
'description': _('Prefix value for build order reference'),
|
|
|
|
'default': 'BO',
|
|
|
|
},
|
|
|
|
|
|
|
|
'BUILDORDER_REFERENCE_REGEX': {
|
|
|
|
'name': _('Build Order Reference Regex'),
|
|
|
|
'description': _('Regular expression pattern for matching build order reference')
|
|
|
|
},
|
|
|
|
|
|
|
|
'SALESORDER_REFERENCE_PREFIX': {
|
|
|
|
'name': _('Sales Order Reference Prefix'),
|
|
|
|
'description': _('Prefix value for sales order reference'),
|
2020-12-18 05:10:55 +00:00
|
|
|
'default': 'SO',
|
2020-10-24 20:49:38 +00:00
|
|
|
},
|
|
|
|
|
|
|
|
'PURCHASEORDER_REFERENCE_PREFIX': {
|
|
|
|
'name': _('Purchase Order Reference Prefix'),
|
|
|
|
'description': _('Prefix value for purchase order reference'),
|
2020-12-18 05:10:55 +00:00
|
|
|
'default': 'PO',
|
2020-10-24 20:49:38 +00:00
|
|
|
},
|
2020-10-19 21:24:23 +00:00
|
|
|
}
|
|
|
|
|
2020-02-03 09:51:53 +00:00
|
|
|
class Meta:
|
|
|
|
verbose_name = "InvenTree Setting"
|
|
|
|
verbose_name_plural = "InvenTree Settings"
|
|
|
|
|
2020-10-24 20:49:38 +00:00
|
|
|
@classmethod
|
|
|
|
def get_setting_name(cls, key):
|
|
|
|
"""
|
|
|
|
Return the name of a particular setting.
|
|
|
|
|
|
|
|
If it does not exist, return an empty string.
|
|
|
|
"""
|
|
|
|
|
|
|
|
key = str(key).strip().upper()
|
|
|
|
|
|
|
|
if key in cls.GLOBAL_SETTINGS:
|
|
|
|
setting = cls.GLOBAL_SETTINGS[key]
|
|
|
|
return setting.get('name', '')
|
|
|
|
else:
|
|
|
|
return ''
|
|
|
|
|
|
|
|
@classmethod
|
|
|
|
def get_setting_description(cls, key):
|
|
|
|
"""
|
|
|
|
Return the description for a particular setting.
|
|
|
|
|
|
|
|
If it does not exist, return an empty string.
|
|
|
|
"""
|
|
|
|
|
|
|
|
key = str(key).strip().upper()
|
|
|
|
|
|
|
|
if key in cls.GLOBAL_SETTINGS:
|
|
|
|
setting = cls.GLOBAL_SETTINGS[key]
|
|
|
|
return setting.get('description', '')
|
|
|
|
else:
|
|
|
|
return ''
|
|
|
|
|
|
|
|
@classmethod
|
|
|
|
def get_setting_units(cls, key):
|
|
|
|
"""
|
|
|
|
Return the units for a particular setting.
|
|
|
|
|
|
|
|
If it does not exist, return an empty string.
|
|
|
|
"""
|
|
|
|
|
|
|
|
key = str(key).strip().upper()
|
|
|
|
|
|
|
|
if key in cls.GLOBAL_SETTINGS:
|
|
|
|
setting = cls.GLOBAL_SETTINGS[key]
|
|
|
|
return setting.get('units', '')
|
|
|
|
else:
|
|
|
|
return ''
|
|
|
|
|
2020-10-24 22:36:58 +00:00
|
|
|
@classmethod
|
|
|
|
def get_setting_validator(cls, key):
|
|
|
|
"""
|
|
|
|
Return the validator for a particular setting.
|
|
|
|
|
|
|
|
If it does not exist, return None
|
|
|
|
"""
|
|
|
|
|
|
|
|
key = str(key).strip().upper()
|
|
|
|
|
|
|
|
if key in cls.GLOBAL_SETTINGS:
|
|
|
|
setting = cls.GLOBAL_SETTINGS[key]
|
|
|
|
return setting.get('validator', None)
|
|
|
|
else:
|
|
|
|
return None
|
|
|
|
|
2020-10-24 20:49:38 +00:00
|
|
|
@classmethod
|
2020-11-13 02:21:43 +00:00
|
|
|
def get_setting_default(cls, key):
|
2020-10-24 20:49:38 +00:00
|
|
|
"""
|
|
|
|
Return the default value for a particular setting.
|
|
|
|
|
|
|
|
If it does not exist, return an empty string
|
|
|
|
"""
|
|
|
|
|
|
|
|
key = str(key).strip().upper()
|
|
|
|
|
|
|
|
if key in cls.GLOBAL_SETTINGS:
|
|
|
|
setting = cls.GLOBAL_SETTINGS[key]
|
|
|
|
return setting.get('default', '')
|
|
|
|
else:
|
|
|
|
return ''
|
|
|
|
|
2020-11-10 06:08:08 +00:00
|
|
|
@classmethod
|
|
|
|
def get_setting_choices(cls, key):
|
|
|
|
"""
|
|
|
|
Return the validator choices available for a particular setting.
|
|
|
|
"""
|
|
|
|
|
|
|
|
key = str(key).strip().upper()
|
|
|
|
|
|
|
|
if key in cls.GLOBAL_SETTINGS:
|
|
|
|
setting = cls.GLOBAL_SETTINGS[key]
|
|
|
|
choices = setting.get('choices', None)
|
|
|
|
else:
|
|
|
|
choices = None
|
|
|
|
|
|
|
|
"""
|
|
|
|
TODO:
|
|
|
|
if type(choices) is function:
|
|
|
|
# Evaluate the function (we expect it will return a list of tuples...)
|
|
|
|
return choices()
|
|
|
|
"""
|
2021-05-06 10:11:38 +00:00
|
|
|
|
2020-11-10 06:08:08 +00:00
|
|
|
return choices
|
|
|
|
|
2020-10-24 21:02:46 +00:00
|
|
|
@classmethod
|
2020-10-25 10:33:13 +00:00
|
|
|
def get_setting_object(cls, key):
|
2020-10-24 21:02:46 +00:00
|
|
|
"""
|
2020-10-25 10:33:13 +00:00
|
|
|
Return an InvenTreeSetting object matching the given key.
|
2020-10-24 21:02:46 +00:00
|
|
|
|
2020-10-25 10:33:13 +00:00
|
|
|
- Key is case-insensitive
|
|
|
|
- Returns None if no match is made
|
2020-10-24 21:02:46 +00:00
|
|
|
"""
|
|
|
|
|
|
|
|
key = str(key).strip().upper()
|
|
|
|
|
|
|
|
try:
|
|
|
|
setting = InvenTreeSetting.objects.filter(key__iexact=key).first()
|
2020-11-12 07:04:50 +00:00
|
|
|
except (ValueError, InvenTreeSetting.DoesNotExist):
|
2020-11-13 00:50:58 +00:00
|
|
|
setting = None
|
2020-11-13 10:37:39 +00:00
|
|
|
except (IntegrityError, OperationalError):
|
2020-11-13 10:01:30 +00:00
|
|
|
setting = None
|
2020-11-13 00:50:58 +00:00
|
|
|
|
2020-11-13 10:37:39 +00:00
|
|
|
# Setting does not exist! (Try to create it)
|
2020-11-13 00:50:58 +00:00
|
|
|
if not setting:
|
2020-11-13 11:22:02 +00:00
|
|
|
|
|
|
|
setting = InvenTreeSetting(key=key, value=InvenTreeSetting.get_setting_default(key))
|
|
|
|
|
2020-11-13 20:39:51 +00:00
|
|
|
try:
|
|
|
|
# Wrap this statement in "atomic", so it can be rolled back if it fails
|
|
|
|
with transaction.atomic():
|
|
|
|
setting.save()
|
2020-11-13 10:37:39 +00:00
|
|
|
except (IntegrityError, OperationalError):
|
|
|
|
# It might be the case that the database isn't created yet
|
2020-11-13 11:22:02 +00:00
|
|
|
pass
|
2020-10-25 11:07:11 +00:00
|
|
|
|
|
|
|
return setting
|
2020-10-25 10:33:13 +00:00
|
|
|
|
|
|
|
@classmethod
|
|
|
|
def get_setting_pk(cls, key):
|
|
|
|
"""
|
|
|
|
Return the primary-key value for a given setting.
|
|
|
|
|
|
|
|
If the setting does not exist, return None
|
|
|
|
"""
|
|
|
|
|
|
|
|
setting = InvenTreeSetting.get_setting_object(cls)
|
|
|
|
|
|
|
|
if setting:
|
2020-10-24 21:02:46 +00:00
|
|
|
return setting.pk
|
2020-10-25 10:33:13 +00:00
|
|
|
else:
|
2020-10-24 21:02:46 +00:00
|
|
|
return None
|
|
|
|
|
2019-09-15 12:46:24 +00:00
|
|
|
@classmethod
|
|
|
|
def get_setting(cls, key, backup_value=None):
|
|
|
|
"""
|
|
|
|
Get the value of a particular setting.
|
|
|
|
If it does not exist, return the backup value (default = None)
|
|
|
|
"""
|
|
|
|
|
2020-10-19 21:41:08 +00:00
|
|
|
# If no backup value is specified, atttempt to retrieve a "default" value
|
|
|
|
if backup_value is None:
|
2020-11-13 02:21:43 +00:00
|
|
|
backup_value = cls.get_setting_default(key)
|
2020-10-19 21:41:08 +00:00
|
|
|
|
2020-10-25 10:33:13 +00:00
|
|
|
setting = InvenTreeSetting.get_setting_object(key)
|
2020-10-19 21:24:23 +00:00
|
|
|
|
2020-10-25 10:33:13 +00:00
|
|
|
if setting:
|
2020-11-09 12:16:04 +00:00
|
|
|
value = setting.value
|
2020-11-09 12:44:54 +00:00
|
|
|
|
|
|
|
# If the particular setting is defined as a boolean, cast the value to a boolean
|
|
|
|
if setting.is_bool():
|
|
|
|
value = InvenTree.helpers.str2bool(value)
|
|
|
|
|
2021-01-04 21:50:07 +00:00
|
|
|
if setting.is_int():
|
2021-01-06 11:20:54 +00:00
|
|
|
try:
|
|
|
|
value = int(value)
|
|
|
|
except (ValueError, TypeError):
|
|
|
|
value = backup_value
|
2021-01-04 21:50:07 +00:00
|
|
|
|
2020-10-25 10:33:13 +00:00
|
|
|
else:
|
2020-11-09 12:16:04 +00:00
|
|
|
value = backup_value
|
|
|
|
|
|
|
|
return value
|
2019-09-15 12:46:24 +00:00
|
|
|
|
|
|
|
@classmethod
|
|
|
|
def set_setting(cls, key, value, user, create=True):
|
|
|
|
"""
|
|
|
|
Set the value of a particular setting.
|
|
|
|
If it does not exist, option to create it.
|
|
|
|
|
|
|
|
Args:
|
|
|
|
key: settings key
|
|
|
|
value: New value
|
|
|
|
user: User object (must be staff member to update a core setting)
|
2019-09-15 13:11:06 +00:00
|
|
|
create: If True, create a new setting if the specified key does not exist.
|
2019-09-15 12:46:24 +00:00
|
|
|
"""
|
|
|
|
|
2021-03-11 08:56:22 +00:00
|
|
|
if user is not None and not user.is_staff:
|
2019-09-15 12:46:24 +00:00
|
|
|
return
|
|
|
|
|
|
|
|
try:
|
|
|
|
setting = InvenTreeSetting.objects.get(key__iexact=key)
|
|
|
|
except InvenTreeSetting.DoesNotExist:
|
|
|
|
|
|
|
|
if create:
|
|
|
|
setting = InvenTreeSetting(key=key)
|
|
|
|
else:
|
|
|
|
return
|
2020-11-09 22:03:26 +00:00
|
|
|
|
|
|
|
# Enforce standard boolean representation
|
|
|
|
if setting.is_bool():
|
|
|
|
value = InvenTree.helpers.str2bool(value)
|
2021-05-06 10:11:38 +00:00
|
|
|
|
2020-10-19 21:29:06 +00:00
|
|
|
setting.value = str(value)
|
2019-09-15 12:46:24 +00:00
|
|
|
setting.save()
|
|
|
|
|
2019-09-15 13:07:45 +00:00
|
|
|
key = models.CharField(max_length=50, blank=False, unique=True, help_text=_('Settings key (must be unique - case insensitive'))
|
2019-09-15 12:46:24 +00:00
|
|
|
|
|
|
|
value = models.CharField(max_length=200, blank=True, unique=False, help_text=_('Settings value'))
|
|
|
|
|
2020-10-25 10:43:33 +00:00
|
|
|
@property
|
|
|
|
def name(self):
|
|
|
|
return InvenTreeSetting.get_setting_name(self.key)
|
|
|
|
|
2020-11-09 22:03:26 +00:00
|
|
|
@property
|
|
|
|
def default_value(self):
|
2020-11-13 02:21:43 +00:00
|
|
|
return InvenTreeSetting.get_setting_default(self.key)
|
2020-11-09 22:03:26 +00:00
|
|
|
|
2020-10-25 10:43:33 +00:00
|
|
|
@property
|
|
|
|
def description(self):
|
|
|
|
return InvenTreeSetting.get_setting_description(self.key)
|
|
|
|
|
|
|
|
@property
|
|
|
|
def units(self):
|
|
|
|
return InvenTreeSetting.get_setting_units(self.key)
|
|
|
|
|
2020-10-24 22:36:58 +00:00
|
|
|
def clean(self):
|
|
|
|
"""
|
|
|
|
If a validator (or multiple validators) are defined for a particular setting key,
|
|
|
|
run them against the 'value' field.
|
|
|
|
"""
|
|
|
|
|
|
|
|
super().clean()
|
|
|
|
|
|
|
|
validator = InvenTreeSetting.get_setting_validator(self.key)
|
|
|
|
|
2021-02-23 03:12:16 +00:00
|
|
|
if self.is_bool():
|
|
|
|
self.value = InvenTree.helpers.str2bool(self.value)
|
|
|
|
|
|
|
|
if self.is_int():
|
|
|
|
try:
|
|
|
|
self.value = int(self.value)
|
|
|
|
except (ValueError):
|
|
|
|
raise ValidationError(_('Must be an integer value'))
|
|
|
|
|
2020-10-24 22:36:58 +00:00
|
|
|
if validator is not None:
|
|
|
|
self.run_validator(validator)
|
|
|
|
|
|
|
|
def run_validator(self, validator):
|
|
|
|
"""
|
|
|
|
Run a validator against the 'value' field for this InvenTreeSetting object.
|
|
|
|
"""
|
|
|
|
|
|
|
|
if validator is None:
|
|
|
|
return
|
|
|
|
|
2021-02-23 03:12:16 +00:00
|
|
|
value = self.value
|
2021-02-16 04:31:04 +00:00
|
|
|
|
2021-01-06 11:20:54 +00:00
|
|
|
# Boolean validator
|
2021-02-23 03:12:16 +00:00
|
|
|
if self.is_bool():
|
2021-01-06 11:20:54 +00:00
|
|
|
# Value must "look like" a boolean value
|
2021-02-23 03:12:16 +00:00
|
|
|
if InvenTree.helpers.is_bool(value):
|
2021-01-06 11:20:54 +00:00
|
|
|
# Coerce into either "True" or "False"
|
2021-02-23 03:12:16 +00:00
|
|
|
value = InvenTree.helpers.str2bool(value)
|
2021-01-06 11:20:54 +00:00
|
|
|
else:
|
|
|
|
raise ValidationError({
|
|
|
|
'value': _('Value must be a boolean value')
|
|
|
|
})
|
2020-10-25 10:00:06 +00:00
|
|
|
|
2021-02-23 03:24:09 +00:00
|
|
|
# Integer validator
|
2021-02-23 03:12:16 +00:00
|
|
|
if self.is_int():
|
|
|
|
|
2021-01-06 11:20:54 +00:00
|
|
|
try:
|
|
|
|
# Coerce into an integer value
|
2021-02-23 03:12:16 +00:00
|
|
|
value = int(value)
|
2021-01-06 11:20:54 +00:00
|
|
|
except (ValueError, TypeError):
|
|
|
|
raise ValidationError({
|
|
|
|
'value': _('Value must be an integer value'),
|
|
|
|
})
|
2020-10-24 22:36:58 +00:00
|
|
|
|
2021-02-23 03:12:16 +00:00
|
|
|
# If a list of validators is supplied, iterate through each one
|
|
|
|
if type(validator) in [list, tuple]:
|
|
|
|
for v in validator:
|
|
|
|
self.run_validator(v)
|
|
|
|
|
|
|
|
if callable(validator):
|
|
|
|
# We can accept function validators with a single argument
|
|
|
|
validator(self.value)
|
|
|
|
|
2019-09-15 12:46:24 +00:00
|
|
|
def validate_unique(self, exclude=None):
|
|
|
|
""" Ensure that the key:value pair is unique.
|
|
|
|
In addition to the base validators, this ensures that the 'key'
|
|
|
|
is unique, using a case-insensitive comparison.
|
|
|
|
"""
|
|
|
|
|
|
|
|
super().validate_unique(exclude)
|
|
|
|
|
|
|
|
try:
|
|
|
|
setting = InvenTreeSetting.objects.exclude(id=self.id).filter(key__iexact=self.key)
|
|
|
|
if setting.exists():
|
|
|
|
raise ValidationError({'key': _('Key string must be unique')})
|
|
|
|
except InvenTreeSetting.DoesNotExist:
|
|
|
|
pass
|
2019-09-02 23:07:03 +00:00
|
|
|
|
2020-11-10 06:08:08 +00:00
|
|
|
def choices(self):
|
|
|
|
"""
|
|
|
|
Return the available choices for this setting (or None if no choices are defined)
|
|
|
|
"""
|
|
|
|
|
|
|
|
return InvenTreeSetting.get_setting_choices(self.key)
|
|
|
|
|
2020-10-25 10:00:06 +00:00
|
|
|
def is_bool(self):
|
|
|
|
"""
|
|
|
|
Check if this setting is required to be a boolean value
|
|
|
|
"""
|
|
|
|
|
|
|
|
validator = InvenTreeSetting.get_setting_validator(self.key)
|
|
|
|
|
2021-02-23 03:12:16 +00:00
|
|
|
if validator == bool:
|
|
|
|
return True
|
|
|
|
|
|
|
|
if type(validator) in [list, tuple]:
|
|
|
|
for v in validator:
|
|
|
|
if v == bool:
|
|
|
|
return True
|
2020-10-25 10:00:06 +00:00
|
|
|
|
2020-10-25 10:33:13 +00:00
|
|
|
def as_bool(self):
|
|
|
|
"""
|
|
|
|
Return the value of this setting converted to a boolean value.
|
|
|
|
|
|
|
|
Warning: Only use on values where is_bool evaluates to true!
|
|
|
|
"""
|
|
|
|
|
|
|
|
return InvenTree.helpers.str2bool(self.value)
|
|
|
|
|
2021-01-04 13:54:05 +00:00
|
|
|
def is_int(self):
|
|
|
|
"""
|
|
|
|
Check if the setting is required to be an integer value:
|
|
|
|
"""
|
|
|
|
|
2021-01-04 21:50:07 +00:00
|
|
|
validator = InvenTreeSetting.get_setting_validator(self.key)
|
2021-01-04 13:54:05 +00:00
|
|
|
|
2021-01-06 11:20:54 +00:00
|
|
|
if validator == int:
|
|
|
|
return True
|
2021-05-06 10:11:38 +00:00
|
|
|
|
2021-01-06 11:20:54 +00:00
|
|
|
if type(validator) in [list, tuple]:
|
|
|
|
for v in validator:
|
|
|
|
if v == int:
|
|
|
|
return True
|
|
|
|
|
2021-02-23 03:12:16 +00:00
|
|
|
return False
|
|
|
|
|
2021-01-06 11:20:54 +00:00
|
|
|
def as_int(self):
|
|
|
|
"""
|
|
|
|
Return the value of this setting converted to a boolean value.
|
2021-05-06 10:11:38 +00:00
|
|
|
|
2021-01-06 11:20:54 +00:00
|
|
|
If an error occurs, return the default value
|
|
|
|
"""
|
|
|
|
|
|
|
|
try:
|
2021-01-06 12:09:26 +00:00
|
|
|
value = int(self.value)
|
2021-01-06 11:20:54 +00:00
|
|
|
except (ValueError, TypeError):
|
2021-01-06 12:09:26 +00:00
|
|
|
value = self.default_value()
|
2021-01-04 13:54:05 +00:00
|
|
|
|
2021-01-06 12:09:26 +00:00
|
|
|
return value
|
2021-05-06 10:11:38 +00:00
|
|
|
|
2019-09-02 23:07:03 +00:00
|
|
|
|
2020-09-17 12:44:17 +00:00
|
|
|
class PriceBreak(models.Model):
|
2020-09-17 12:47:31 +00:00
|
|
|
"""
|
|
|
|
Represents a PriceBreak model
|
|
|
|
"""
|
2020-09-17 12:44:17 +00:00
|
|
|
|
|
|
|
class Meta:
|
|
|
|
abstract = True
|
|
|
|
|
2020-11-12 00:14:50 +00:00
|
|
|
quantity = InvenTree.fields.RoundingDecimalField(
|
|
|
|
max_digits=15,
|
|
|
|
decimal_places=5,
|
|
|
|
default=1,
|
|
|
|
validators=[MinValueValidator(1)],
|
|
|
|
verbose_name=_('Quantity'),
|
|
|
|
help_text=_('Price break quantity'),
|
|
|
|
)
|
2020-09-17 12:44:17 +00:00
|
|
|
|
2020-11-10 11:25:05 +00:00
|
|
|
price = MoneyField(
|
|
|
|
max_digits=19,
|
|
|
|
decimal_places=4,
|
|
|
|
default_currency='USD',
|
|
|
|
null=True,
|
|
|
|
verbose_name=_('Price'),
|
|
|
|
help_text=_('Unit price at specified quantity'),
|
|
|
|
)
|
|
|
|
|
2020-11-10 13:21:06 +00:00
|
|
|
def convert_to(self, currency_code):
|
2020-09-17 12:47:31 +00:00
|
|
|
"""
|
2020-11-10 13:21:06 +00:00
|
|
|
Convert the unit-price at this price break to the specified currency code.
|
2020-09-17 12:47:31 +00:00
|
|
|
|
2020-11-10 13:21:06 +00:00
|
|
|
Args:
|
|
|
|
currency_code - The currency code to convert to (e.g "USD" or "AUD")
|
|
|
|
"""
|
2020-09-17 12:47:31 +00:00
|
|
|
|
2020-11-10 13:21:06 +00:00
|
|
|
try:
|
|
|
|
converted = convert_money(self.price, currency_code)
|
|
|
|
except MissingRate:
|
|
|
|
print(f"WARNING: No currency conversion rate available for {self.price_currency} -> {currency_code}")
|
|
|
|
return self.price.amount
|
2020-09-17 12:47:31 +00:00
|
|
|
|
2020-11-10 13:21:06 +00:00
|
|
|
return converted.amount
|
2020-09-17 12:44:17 +00:00
|
|
|
|
|
|
|
|
2021-05-05 21:42:52 +00:00
|
|
|
def get_price(instance, quantity, moq=True, multiples=True, currency=None):
|
|
|
|
""" Calculate the price based on quantity price breaks.
|
|
|
|
|
|
|
|
- Don't forget to add in flat-fee cost (base_cost field)
|
|
|
|
- If MOQ (minimum order quantity) is required, bump quantity
|
|
|
|
- If order multiples are to be observed, then we need to calculate based on that, too
|
|
|
|
"""
|
|
|
|
|
|
|
|
price_breaks = instance.price_breaks.all()
|
|
|
|
|
|
|
|
# No price break information available?
|
|
|
|
if len(price_breaks) == 0:
|
|
|
|
return None
|
|
|
|
|
|
|
|
# Check if quantity is fraction and disable multiples
|
|
|
|
multiples = (quantity % 1 == 0)
|
|
|
|
|
|
|
|
# Order multiples
|
|
|
|
if multiples:
|
|
|
|
quantity = int(math.ceil(quantity / instance.multiple) * instance.multiple)
|
|
|
|
|
|
|
|
pb_found = False
|
|
|
|
pb_quantity = -1
|
|
|
|
pb_cost = 0.0
|
|
|
|
|
|
|
|
if currency is None:
|
|
|
|
# Default currency selection
|
|
|
|
currency = InvenTreeSetting.get_setting('INVENTREE_DEFAULT_CURRENCY')
|
|
|
|
|
|
|
|
pb_min = None
|
|
|
|
for pb in instance.price_breaks.all():
|
|
|
|
# Store smallest price break
|
|
|
|
if not pb_min:
|
|
|
|
pb_min = pb
|
|
|
|
|
|
|
|
# Ignore this pricebreak (quantity is too high)
|
|
|
|
if pb.quantity > quantity:
|
|
|
|
continue
|
|
|
|
|
|
|
|
pb_found = True
|
|
|
|
|
|
|
|
# If this price-break quantity is the largest so far, use it!
|
|
|
|
if pb.quantity > pb_quantity:
|
|
|
|
pb_quantity = pb.quantity
|
|
|
|
|
|
|
|
# Convert everything to the selected currency
|
|
|
|
pb_cost = pb.convert_to(currency)
|
|
|
|
|
|
|
|
# Use smallest price break
|
|
|
|
if not pb_found and pb_min:
|
|
|
|
# Update price break information
|
|
|
|
pb_quantity = pb_min.quantity
|
|
|
|
pb_cost = pb_min.convert_to(currency)
|
|
|
|
# Trigger cost calculation using smallest price break
|
|
|
|
pb_found = True
|
|
|
|
|
|
|
|
# Convert quantity to decimal.Decimal format
|
|
|
|
quantity = decimal.Decimal(f'{quantity}')
|
|
|
|
|
|
|
|
if pb_found:
|
|
|
|
cost = pb_cost * quantity
|
|
|
|
return InvenTree.helpers.normalize(cost + instance.base_cost)
|
|
|
|
else:
|
|
|
|
return None
|
|
|
|
|
|
|
|
|
2020-09-07 21:27:23 +00:00
|
|
|
class ColorTheme(models.Model):
|
2020-09-08 21:24:22 +00:00
|
|
|
""" Color Theme Setting """
|
2020-09-07 16:29:24 +00:00
|
|
|
|
2020-09-09 19:55:32 +00:00
|
|
|
default_color_theme = ('', _('Default'))
|
2020-09-07 16:29:24 +00:00
|
|
|
|
2020-09-07 21:27:23 +00:00
|
|
|
name = models.CharField(max_length=20,
|
2020-09-09 19:55:32 +00:00
|
|
|
default='',
|
2020-09-07 21:27:23 +00:00
|
|
|
blank=True)
|
2020-09-07 16:29:24 +00:00
|
|
|
|
2020-09-09 19:55:32 +00:00
|
|
|
user = models.CharField(max_length=150,
|
|
|
|
unique=True)
|
|
|
|
|
|
|
|
@classmethod
|
|
|
|
def get_color_themes_choices(cls):
|
|
|
|
""" Get all color themes from static folder """
|
|
|
|
|
|
|
|
# Get files list from css/color-themes/ folder
|
|
|
|
files_list = []
|
|
|
|
for file in os.listdir(settings.STATIC_COLOR_THEMES_DIR):
|
|
|
|
files_list.append(os.path.splitext(file))
|
|
|
|
|
|
|
|
# Get color themes choices (CSS sheets)
|
|
|
|
choices = [(file_name.lower(), _(file_name.replace('-', ' ').title()))
|
|
|
|
for file_name, file_ext in files_list
|
|
|
|
if file_ext == '.css' and file_name.lower() != 'default']
|
|
|
|
|
|
|
|
# Add default option as empty option
|
|
|
|
choices.insert(0, cls.default_color_theme)
|
|
|
|
|
|
|
|
return choices
|
|
|
|
|
|
|
|
@classmethod
|
|
|
|
def is_valid_choice(cls, user_color_theme):
|
|
|
|
""" Check if color theme is valid choice """
|
|
|
|
try:
|
|
|
|
user_color_theme_name = user_color_theme.name
|
|
|
|
except AttributeError:
|
|
|
|
return False
|
|
|
|
|
|
|
|
for color_theme in cls.get_color_themes_choices():
|
|
|
|
if user_color_theme_name == color_theme[0]:
|
|
|
|
return True
|
|
|
|
|
|
|
|
return False
|