diff --git a/.github/scripts/version_check.py b/.github/scripts/version_check.py index 6b1b707a5b..79567aead2 100644 --- a/.github/scripts/version_check.py +++ b/.github/scripts/version_check.py @@ -175,10 +175,7 @@ if __name__ == '__main__': print(f"Version number '{version}' does not match tag '{version_tag}'") sys.exit - if highest_release: - docker_tags = [version_tag, 'stable'] - else: - docker_tags = [version_tag] + docker_tags = [version_tag, 'stable'] if highest_release else [version_tag] elif GITHUB_REF_TYPE == 'branch': # Otherwise we know we are targeting the 'master' branch diff --git a/docs/docs/hooks.py b/docs/docs/hooks.py index a666931be6..ddd4e38911 100644 --- a/docs/docs/hooks.py +++ b/docs/docs/hooks.py @@ -173,7 +173,7 @@ def on_config(config, *args, **kwargs): # Add *all* readthedocs related keys readthedocs = {} - for key in os.environ.keys(): + for key in os.environ: if key.startswith('READTHEDOCS_'): k = key.replace('READTHEDOCS_', '').lower() readthedocs[k] = os.environ[key] diff --git a/docs/extract_schema.py b/docs/extract_schema.py index 2cfc435740..8f94679fbc 100644 --- a/docs/extract_schema.py +++ b/docs/extract_schema.py @@ -54,9 +54,7 @@ def top_level_path(path: str) -> str: def generate_schema_file(key: str) -> None: """Generate a schema file for the provided key.""" - description = ( - SPECIAL_PATHS[key] if key in SPECIAL_PATHS else 'General API Endpoints' - ) + description = SPECIAL_PATHS.get(key, 'General API Endpoints') output = f""" --- diff --git a/pyproject.toml b/pyproject.toml index fde301ef88..04d9138b2a 100644 --- a/pyproject.toml +++ b/pyproject.toml @@ -20,13 +20,19 @@ src = ["src/backend/InvenTree"] "__init__.py" = ["D104"] [tool.ruff.lint] -select = ["A", "B", "C", "C4", "D", "F", "I", "N", "PIE", "UP", "W"] +select = ["A", "B", "C", "C4", "D", "F", "I", "N", "SIM", "PIE", "UP", "W"] # Things that should be enabled in the future: # - LOG # - DJ # for Django stuff # - S # for security stuff (bandit) ignore = [ + "SIM117", + # - SIM117 - Use a single with statement with multiple contexts instead of nested with statements + "SIM102", + # - SIM102 - Use a single if statement instead of nested if statements + "SIM105", + # - SIM105 - Use contextlib.suppress({exception}) instead of try-except-pass "C901", # - C901 - function is too complex "N999", diff --git a/src/backend/InvenTree/InvenTree/admin.py b/src/backend/InvenTree/InvenTree/admin.py index 4920e0038f..66346e9eef 100644 --- a/src/backend/InvenTree/InvenTree/admin.py +++ b/src/backend/InvenTree/InvenTree/admin.py @@ -104,14 +104,16 @@ class InvenTreeResource(ModelResource): attribute = getattr(field, 'attribute', field_name) # Check if the associated database field is a non-nullable string - if db_field := db_fields.get(attribute): - if ( + if ( + (db_field := db_fields.get(attribute)) + and ( isinstance(db_field, CharField) and db_field.blank and not db_field.null - ): - if column not in self.CONVERT_NULL_FIELDS: - self.CONVERT_NULL_FIELDS.append(column) + ) + and column not in self.CONVERT_NULL_FIELDS + ): + self.CONVERT_NULL_FIELDS.append(column) return super().before_import(dataset, using_transactions, dry_run, **kwargs) diff --git a/src/backend/InvenTree/InvenTree/api.py b/src/backend/InvenTree/InvenTree/api.py index 0822ff1d0e..e8e4b847bb 100644 --- a/src/backend/InvenTree/InvenTree/api.py +++ b/src/backend/InvenTree/InvenTree/api.py @@ -77,7 +77,7 @@ class LicenseView(APIView): # Ensure we do not have any duplicate 'name' values in the list for entry in data: name = None - for key in entry.keys(): + for key in entry: if key.lower() == 'name': name = entry[key] break diff --git a/src/backend/InvenTree/InvenTree/context.py b/src/backend/InvenTree/InvenTree/context.py index 864bed96f8..8d30e6bb8a 100644 --- a/src/backend/InvenTree/InvenTree/context.py +++ b/src/backend/InvenTree/InvenTree/context.py @@ -70,7 +70,7 @@ def user_roles(request): roles = {} - for role in RuleSet.get_ruleset_models().keys(): + for role in RuleSet.get_ruleset_models(): permissions = {} for perm in ['view', 'add', 'change', 'delete']: diff --git a/src/backend/InvenTree/InvenTree/conversion.py b/src/backend/InvenTree/InvenTree/conversion.py index 547d80952d..968c939bc5 100644 --- a/src/backend/InvenTree/InvenTree/conversion.py +++ b/src/backend/InvenTree/InvenTree/conversion.py @@ -245,8 +245,4 @@ def is_dimensionless(value): if value.units == ureg.dimensionless: return True - if value.to_base_units().units == ureg.dimensionless: - return True - - # At this point, the value is not dimensionless - return False + return value.to_base_units().units == ureg.dimensionless diff --git a/src/backend/InvenTree/InvenTree/exceptions.py b/src/backend/InvenTree/InvenTree/exceptions.py index 35b5f4531f..efe941ddfd 100644 --- a/src/backend/InvenTree/InvenTree/exceptions.py +++ b/src/backend/InvenTree/InvenTree/exceptions.py @@ -40,10 +40,7 @@ def log_error(path, error_name=None, error_info=None, error_data=None): if kind in settings.IGNORED_ERRORS: return - if error_name: - kind = error_name - else: - kind = getattr(kind, '__name__', 'Unknown Error') + kind = error_name or getattr(kind, '__name__', 'Unknown Error') if error_info: info = error_info diff --git a/src/backend/InvenTree/InvenTree/exchange.py b/src/backend/InvenTree/InvenTree/exchange.py index ab02883058..d53e6ad49a 100644 --- a/src/backend/InvenTree/InvenTree/exchange.py +++ b/src/backend/InvenTree/InvenTree/exchange.py @@ -31,10 +31,7 @@ class InvenTreeExchange(SimpleExchangeBackend): # Find the selected exchange rate plugin slug = get_global_setting('CURRENCY_UPDATE_PLUGIN', create=False) - if slug: - plugin = registry.get_plugin(slug) - else: - plugin = None + plugin = registry.get_plugin(slug) if slug else None if not plugin: # Find the first active currency exchange plugin diff --git a/src/backend/InvenTree/InvenTree/fields.py b/src/backend/InvenTree/InvenTree/fields.py index 01efe65ae1..2c4bbe174c 100644 --- a/src/backend/InvenTree/InvenTree/fields.py +++ b/src/backend/InvenTree/InvenTree/fields.py @@ -93,9 +93,8 @@ class InvenTreeModelMoneyField(ModelMoneyField): allow_negative = kwargs.pop('allow_negative', False) # If no validators are provided, add some "standard" ones - if len(validators) == 0: - if not allow_negative: - validators.append(MinMoneyValidator(0)) + if len(validators) == 0 and not allow_negative: + validators.append(MinMoneyValidator(0)) kwargs['validators'] = validators diff --git a/src/backend/InvenTree/InvenTree/filters.py b/src/backend/InvenTree/InvenTree/filters.py index f3182d0931..44b4b6143c 100644 --- a/src/backend/InvenTree/InvenTree/filters.py +++ b/src/backend/InvenTree/InvenTree/filters.py @@ -118,10 +118,7 @@ class InvenTreeOrderingFilter(filters.OrderingFilter): field = field[1:] # Are aliases defined for this field? - if field in aliases: - alias = aliases[field] - else: - alias = field + alias = aliases.get(field, field) """ Potentially, a single field could be "aliased" to multiple field, diff --git a/src/backend/InvenTree/InvenTree/format.py b/src/backend/InvenTree/InvenTree/format.py index 6617948bbc..edaff898ef 100644 --- a/src/backend/InvenTree/InvenTree/format.py +++ b/src/backend/InvenTree/InvenTree/format.py @@ -106,10 +106,7 @@ def construct_format_regex(fmt_string: str) -> str: # Add a named capture group for the format entry if name: # Check if integer values are required - if _fmt.endswith('d'): - c = r'\d' - else: - c = '.' + c = '\\d' if _fmt.endswith('d') else '.' # Specify width # TODO: Introspect required width @@ -160,7 +157,7 @@ def extract_named_group(name: str, value: str, fmt_string: str) -> str: """ info = parse_format_string(fmt_string) - if name not in info.keys(): + if name not in info: raise NameError(_(f"Value '{name}' does not appear in pattern format")) # Construct a regular expression for matching against the provided format string @@ -199,7 +196,7 @@ def format_money( Raises: ValueError: format string is incorrectly specified """ - language = (None and translation.get_language()) or settings.LANGUAGE_CODE + language = (None) or settings.LANGUAGE_CODE locale = Locale.parse(translation.to_locale(language)) if format: pattern = parse_pattern(format) diff --git a/src/backend/InvenTree/InvenTree/helpers.py b/src/backend/InvenTree/InvenTree/helpers.py index 2e55ea0fea..ef80d6e886 100644 --- a/src/backend/InvenTree/InvenTree/helpers.py +++ b/src/backend/InvenTree/InvenTree/helpers.py @@ -97,10 +97,7 @@ def generateTestKey(test_name: str) -> str: if char.isidentifier(): return True - if char.isalnum(): - return True - - return False + return bool(char.isalnum()) # Remove any characters that cannot be used to represent a variable key = ''.join([c for c in key if valid_char(c)]) @@ -490,10 +487,7 @@ def extract_serial_numbers(input_string, expected_quantity: int, starting_value= except ValueError: raise ValidationError([_('Invalid quantity provided')]) - if input_string: - input_string = str(input_string).strip() - else: - input_string = '' + input_string = str(input_string).strip() if input_string else '' if len(input_string) == 0: raise ValidationError([_('Empty serial number string')]) @@ -827,7 +821,7 @@ def hash_barcode(barcode_data): def hash_file(filename: Union[str, Path], storage: Union[Storage, None] = None): """Return the MD5 hash of a file.""" content = ( - open(filename, 'rb').read() + open(filename, 'rb').read() # noqa: SIM115 if storage is None else storage.open(str(filename), 'rb').read() ) diff --git a/src/backend/InvenTree/InvenTree/helpers_model.py b/src/backend/InvenTree/InvenTree/helpers_model.py index dc6c947f47..44b6a9f506 100644 --- a/src/backend/InvenTree/InvenTree/helpers_model.py +++ b/src/backend/InvenTree/InvenTree/helpers_model.py @@ -114,10 +114,7 @@ def download_image_from_url(remote_url, timeout=2.5): # Add user specified user-agent to request (if specified) user_agent = get_global_setting('INVENTREE_DOWNLOAD_FROM_URL_USER_AGENT') - if user_agent: - headers = {'User-Agent': user_agent} - else: - headers = None + headers = {'User-Agent': user_agent} if user_agent else None try: response = requests.get( diff --git a/src/backend/InvenTree/InvenTree/management/commands/prerender.py b/src/backend/InvenTree/InvenTree/management/commands/prerender.py index 54ed4601a4..7fa6e5c728 100644 --- a/src/backend/InvenTree/InvenTree/management/commands/prerender.py +++ b/src/backend/InvenTree/InvenTree/management/commands/prerender.py @@ -19,10 +19,9 @@ def render_file(file_name, source, target, locales, ctx): target_file = os.path.join(target, locale + '.' + file_name) - with open(target_file, 'w') as localised_file: - with lang_over(locale): - rendered = render_to_string(os.path.join(source, file_name), ctx) - localised_file.write(rendered) + with open(target_file, 'w') as localised_file, lang_over(locale): + rendered = render_to_string(os.path.join(source, file_name), ctx) + localised_file.write(rendered) class Command(BaseCommand): diff --git a/src/backend/InvenTree/InvenTree/metadata.py b/src/backend/InvenTree/InvenTree/metadata.py index 7805d73c0a..bfafe98f12 100644 --- a/src/backend/InvenTree/InvenTree/metadata.py +++ b/src/backend/InvenTree/InvenTree/metadata.py @@ -204,7 +204,7 @@ class InvenTreeMetadata(SimpleMetadata): # Iterate through simple fields for name, field in model_fields.fields.items(): - if name in serializer_info.keys(): + if name in serializer_info: if name in read_only_fields: serializer_info[name]['read_only'] = True @@ -236,7 +236,7 @@ class InvenTreeMetadata(SimpleMetadata): # Iterate through relations for name, relation in model_fields.relations.items(): - if name not in serializer_info.keys(): + if name not in serializer_info: # Skip relation not defined in serializer continue @@ -307,12 +307,12 @@ class InvenTreeMetadata(SimpleMetadata): instance_filters = instance.api_instance_filters() for field_name, field_filters in instance_filters.items(): - if field_name not in serializer_info.keys(): + if field_name not in serializer_info: # The field might be missing, but is added later on # This function seems to get called multiple times? continue - if 'instance_filters' not in serializer_info[field_name].keys(): + if 'instance_filters' not in serializer_info[field_name]: serializer_info[field_name]['instance_filters'] = {} for key, value in field_filters.items(): diff --git a/src/backend/InvenTree/InvenTree/models.py b/src/backend/InvenTree/InvenTree/models.py index a1d244d2ab..4c7dad2309 100644 --- a/src/backend/InvenTree/InvenTree/models.py +++ b/src/backend/InvenTree/InvenTree/models.py @@ -390,10 +390,7 @@ class ReferenceIndexingMixin(models.Model): except Exception: # If anything goes wrong, return the most recent reference recent = cls.get_most_recent_item() - if recent: - reference = recent.reference - else: - reference = '' + reference = recent.reference if recent else '' return reference @@ -410,14 +407,14 @@ class ReferenceIndexingMixin(models.Model): }) # Check that only 'allowed' keys are provided - for key in info.keys(): - if key not in ctx.keys(): + for key in info: + if key not in ctx: raise ValidationError({ 'value': _('Unknown format key specified') + f": '{key}'" }) # Check that the 'ref' variable is specified - if 'ref' not in info.keys(): + if 'ref' not in info: raise ValidationError({ 'value': _('Missing required format key') + ": 'ref'" }) diff --git a/src/backend/InvenTree/InvenTree/ready.py b/src/backend/InvenTree/InvenTree/ready.py index 7dd69d612c..dc83d3caaf 100644 --- a/src/backend/InvenTree/InvenTree/ready.py +++ b/src/backend/InvenTree/InvenTree/ready.py @@ -58,10 +58,7 @@ def isInServerThread(): if 'runserver' in sys.argv: return True - if 'gunicorn' in sys.argv[0]: - return True - - return False + return 'gunicorn' in sys.argv[0] def isInMainThread(): @@ -122,11 +119,7 @@ def canAppAccessDatabase( if not allow_plugins: excluded_commands.extend(['collectplugins']) - for cmd in excluded_commands: - if cmd in sys.argv: - return False - - return True + return all(cmd not in sys.argv for cmd in excluded_commands) def isPluginRegistryLoaded(): diff --git a/src/backend/InvenTree/InvenTree/serializers.py b/src/backend/InvenTree/InvenTree/serializers.py index 747e02cefe..a86b1228e9 100644 --- a/src/backend/InvenTree/InvenTree/serializers.py +++ b/src/backend/InvenTree/InvenTree/serializers.py @@ -379,7 +379,7 @@ class InvenTreeTaggitSerializer(TaggitSerializer): tag_object = super().update(instance, validated_data) - for key in to_be_tagged.keys(): + for key in to_be_tagged: # re-add the tagmanager new_tagobject = tag_object.__class__.objects.get(id=tag_object.id) setattr(tag_object, key, getattr(new_tagobject, key)) @@ -803,11 +803,10 @@ class DataFileExtractSerializer(serializers.Serializer): required = field.get('required', False) # Check for missing required columns - if required: - if name not in self.columns: - raise serializers.ValidationError( - _(f"Missing required column: '{name}'") - ) + if required and name not in self.columns: + raise serializers.ValidationError( + _(f"Missing required column: '{name}'") + ) for col in self.columns: if not col: diff --git a/src/backend/InvenTree/InvenTree/settings.py b/src/backend/InvenTree/InvenTree/settings.py index ebacebcc28..8b59306e3c 100644 --- a/src/backend/InvenTree/InvenTree/settings.py +++ b/src/backend/InvenTree/InvenTree/settings.py @@ -949,10 +949,7 @@ USE_I18N = True # Do not use native timezone support in "test" mode # It generates a *lot* of cruft in the logs -if not TESTING: - USE_TZ = True # pragma: no cover -else: - USE_TZ = False +USE_TZ = bool(not TESTING) DATE_INPUT_FORMATS = ['%Y-%m-%d'] diff --git a/src/backend/InvenTree/InvenTree/templatetags/inventree_extras.py b/src/backend/InvenTree/InvenTree/templatetags/inventree_extras.py index 9e7119c2f2..315211587d 100644 --- a/src/backend/InvenTree/InvenTree/templatetags/inventree_extras.py +++ b/src/backend/InvenTree/InvenTree/templatetags/inventree_extras.py @@ -410,10 +410,7 @@ def progress_bar(val, max_val, *args, **kwargs): else: style = '' - if max_val != 0: - percent = float(val / max_val) * 100 - else: - percent = 0 + percent = float(val / max_val) * 100 if max_val != 0 else 0 if percent > 100: percent = 100 diff --git a/src/backend/InvenTree/InvenTree/views.py b/src/backend/InvenTree/InvenTree/views.py index 5fd9e5135e..a12705666a 100644 --- a/src/backend/InvenTree/InvenTree/views.py +++ b/src/backend/InvenTree/InvenTree/views.py @@ -285,7 +285,7 @@ class AjaxMixin(InvenTreeRoleMixin): # Custom feedback`data fb = self.get_data() - for key in fb.keys(): + for key in fb: data[key] = fb[key] return JsonResponse(data, safe=False) diff --git a/src/backend/InvenTree/common/api.py b/src/backend/InvenTree/common/api.py index e914941cad..75f0a1c77a 100644 --- a/src/backend/InvenTree/common/api.py +++ b/src/backend/InvenTree/common/api.py @@ -244,10 +244,7 @@ class GlobalSettingsDetail(RetrieveUpdateAPI): """Attempt to find a global setting object with the provided key.""" key = str(self.kwargs['key']).upper() - if ( - key.startswith('_') - or key not in common.models.InvenTreeSetting.SETTINGS.keys() - ): + if key.startswith('_') or key not in common.models.InvenTreeSetting.SETTINGS: raise NotFound() return common.models.InvenTreeSetting.get_setting_object( @@ -318,7 +315,7 @@ class UserSettingsDetail(RetrieveUpdateAPI): if ( key.startswith('_') - or key not in common.models.InvenTreeUserSetting.SETTINGS.keys() + or key not in common.models.InvenTreeUserSetting.SETTINGS ): raise NotFound() diff --git a/src/backend/InvenTree/common/forms.py b/src/backend/InvenTree/common/forms.py index 4800149013..ee20615c4e 100644 --- a/src/backend/InvenTree/common/forms.py +++ b/src/backend/InvenTree/common/forms.py @@ -83,10 +83,7 @@ class MatchItemForm(forms.Form): if 'file_manager' in kwargs: file_manager = kwargs.pop('file_manager') - if 'row_data' in kwargs: - row_data = kwargs.pop('row_data') - else: - row_data = None + row_data = kwargs.pop('row_data') if 'row_data' in kwargs else None super().__init__(*args, **kwargs) diff --git a/src/backend/InvenTree/common/models.py b/src/backend/InvenTree/common/models.py index 69fa3ca39e..99b849fc30 100644 --- a/src/backend/InvenTree/common/models.py +++ b/src/backend/InvenTree/common/models.py @@ -3090,13 +3090,10 @@ class CustomUnit(models.Model): """Ensure that the custom unit is unique.""" super().validate_unique(exclude) - if self.symbol: - if ( - CustomUnit.objects.filter(symbol=self.symbol) - .exclude(pk=self.pk) - .exists() - ): - raise ValidationError({'symbol': _('Unit symbol must be unique')}) + if self.symbol and ( + CustomUnit.objects.filter(symbol=self.symbol).exclude(pk=self.pk).exists() + ): + raise ValidationError({'symbol': _('Unit symbol must be unique')}) def clean(self): """Validate that the provided custom unit is indeed valid.""" diff --git a/src/backend/InvenTree/common/notifications.py b/src/backend/InvenTree/common/notifications.py index cdcb52fba5..f510d0e722 100644 --- a/src/backend/InvenTree/common/notifications.py +++ b/src/backend/InvenTree/common/notifications.py @@ -143,11 +143,7 @@ class NotificationMethod: # Check if method globally enabled plg_instance = registry.get_plugin(plg_cls.NAME.lower()) - if plg_instance and not plg_instance.get_setting(self.GLOBAL_SETTING): - return True - - # Lets go! - return False + return plg_instance and not plg_instance.get_setting(self.GLOBAL_SETTING) def usersetting(self, target): """Returns setting for this method for a given user.""" diff --git a/src/backend/InvenTree/common/serializers.py b/src/backend/InvenTree/common/serializers.py index 4c1f6a30dd..bc82c79ce3 100644 --- a/src/backend/InvenTree/common/serializers.py +++ b/src/backend/InvenTree/common/serializers.py @@ -547,9 +547,8 @@ class AttachmentSerializer(InvenTreeModelSerializer): model_type = self.validated_data.get('model_type', None) - if model_type is None: - if self.instance: - model_type = self.instance.model_type + if model_type is None and self.instance: + model_type = self.instance.model_type # Ensure that the user has permission to attach files to the specified model user = self.context.get('request').user diff --git a/src/backend/InvenTree/common/tests.py b/src/backend/InvenTree/common/tests.py index 6bf08cb234..072cc06081 100644 --- a/src/backend/InvenTree/common/tests.py +++ b/src/backend/InvenTree/common/tests.py @@ -392,7 +392,7 @@ class SettingsTest(InvenTreeTestCase): 'before_save', ] - for k in setting.keys(): + for k in setting: self.assertIn(k, allowed_keys) # Check default value for boolean settings @@ -430,7 +430,7 @@ class SettingsTest(InvenTreeTestCase): @override_settings(SITE_URL=None, PLUGIN_TESTING=True, PLUGIN_TESTING_SETUP=True) def test_defaults(self): """Populate the settings with default values.""" - for key in InvenTreeSetting.SETTINGS.keys(): + for key in InvenTreeSetting.SETTINGS: value = InvenTreeSetting.get_setting_default(key) try: @@ -519,7 +519,7 @@ class GlobalSettingsApiTest(InvenTreeAPITestCase): response = self.get(url, expected_code=200) n_public_settings = len([ - k for k in InvenTreeSetting.SETTINGS.keys() if not k.startswith('_') + k for k in InvenTreeSetting.SETTINGS if not k.startswith('_') ]) # Number of results should match the number of settings diff --git a/src/backend/InvenTree/common/views.py b/src/backend/InvenTree/common/views.py index 1ba221ead0..53079c1a55 100644 --- a/src/backend/InvenTree/common/views.py +++ b/src/backend/InvenTree/common/views.py @@ -362,10 +362,7 @@ class FileManagementFormView(MultiStepFormView): duplicates = [] for col in self.column_names: - if col in self.column_selections: - guess = self.column_selections[col] - else: - guess = None + guess = self.column_selections.get(col, None) if guess: n = list(self.column_selections.values()).count( diff --git a/src/backend/InvenTree/company/models.py b/src/backend/InvenTree/company/models.py index 9e1c878c57..11c333c254 100644 --- a/src/backend/InvenTree/company/models.py +++ b/src/backend/InvenTree/company/models.py @@ -46,10 +46,7 @@ def rename_company_image(instance, filename): """ base = 'company_images' - if filename.count('.') > 0: - ext = filename.split('.')[-1] - else: - ext = '' + ext = filename.split('.')[-1] if filename.count('.') > 0 else '' fn = f'company_{instance.pk}_img' diff --git a/src/backend/InvenTree/generic/states/states.py b/src/backend/InvenTree/generic/states/states.py index c72b201eca..83342aed42 100644 --- a/src/backend/InvenTree/generic/states/states.py +++ b/src/backend/InvenTree/generic/states/states.py @@ -86,9 +86,7 @@ class StatusCode(BaseEnum): return False if callable(value): return False - if not isinstance(value.value, int): - return False - return True + return isinstance(value.value, int) @classmethod def values(cls, key=None): diff --git a/src/backend/InvenTree/importer/models.py b/src/backend/InvenTree/importer/models.py index 0714f99d19..403861bec0 100644 --- a/src/backend/InvenTree/importer/models.py +++ b/src/backend/InvenTree/importer/models.py @@ -209,7 +209,7 @@ class DataImportSession(models.Model): missing_fields = [] - for field in required_fields.keys(): + for field in required_fields: # An override value exists if field in field_overrides: continue diff --git a/src/backend/InvenTree/importer/operations.py b/src/backend/InvenTree/importer/operations.py index 8d7191c1b3..6afa7d08c4 100644 --- a/src/backend/InvenTree/importer/operations.py +++ b/src/backend/InvenTree/importer/operations.py @@ -113,9 +113,8 @@ def get_field_label(field) -> str: Returns: str: Field label """ - if field: - if label := getattr(field, 'label', None): - return label + if field and (label := getattr(field, 'label', None)): + return label # TODO: Check if the field is a model field diff --git a/src/backend/InvenTree/importer/validators.py b/src/backend/InvenTree/importer/validators.py index 166c30acc6..4c72f43615 100644 --- a/src/backend/InvenTree/importer/validators.py +++ b/src/backend/InvenTree/importer/validators.py @@ -36,7 +36,7 @@ def validate_importer_model_type(value): """Validate that the given model type is supported for importing.""" from importer.registry import supported_models - if value not in supported_models().keys(): + if value not in supported_models(): raise ValidationError(f"Unsupported model type '{value}'") diff --git a/src/backend/InvenTree/order/api.py b/src/backend/InvenTree/order/api.py index 5c0ec2e6bb..374941afbc 100644 --- a/src/backend/InvenTree/order/api.py +++ b/src/backend/InvenTree/order/api.py @@ -1356,10 +1356,9 @@ class OrderCalendarExport(ICalFeed): if auth[0].lower() == 'basic': uname, passwd = base64.b64decode(auth[1]).decode('ascii').split(':') user = authenticate(username=uname, password=passwd) - if user is not None: - if user.is_active: - login(request, user) - request.user = user + if user is not None and user.is_active: + login(request, user) + request.user = user # Check again if request.user.is_authenticated: diff --git a/src/backend/InvenTree/order/models.py b/src/backend/InvenTree/order/models.py index 1a8124cf5c..10c17be50a 100644 --- a/src/backend/InvenTree/order/models.py +++ b/src/backend/InvenTree/order/models.py @@ -1047,19 +1047,11 @@ class SalesOrder(TotalPriceMixin, Order): def is_fully_allocated(self): """Return True if all line items are fully allocated.""" - for line in self.lines.all(): - if not line.is_fully_allocated(): - return False - - return True + return all(line.is_fully_allocated() for line in self.lines.all()) def is_overallocated(self): """Return true if any lines in the order are over-allocated.""" - for line in self.lines.all(): - if line.is_overallocated(): - return True - - return False + return any(line.is_overallocated() for line in self.lines.all()) def is_completed(self): """Check if this order is "shipped" (all line items delivered).""" diff --git a/src/backend/InvenTree/order/serializers.py b/src/backend/InvenTree/order/serializers.py index f94ed0c293..e9770f244f 100644 --- a/src/backend/InvenTree/order/serializers.py +++ b/src/backend/InvenTree/order/serializers.py @@ -740,13 +740,12 @@ class PurchaseOrderLineItemReceiveSerializer(serializers.Serializer): base_quantity = line_item.part.base_quantity(quantity) # Does the quantity need to be "integer" (for trackable parts?) - if base_part.trackable: - if Decimal(base_quantity) != int(base_quantity): - raise ValidationError({ - 'quantity': _( - 'An integer quantity must be provided for trackable parts' - ) - }) + if base_part.trackable and Decimal(base_quantity) != int(base_quantity): + raise ValidationError({ + 'quantity': _( + 'An integer quantity must be provided for trackable parts' + ) + }) # If serial numbers are provided if serial_numbers: diff --git a/src/backend/InvenTree/order/test_api.py b/src/backend/InvenTree/order/test_api.py index 4d8a78de8a..3dfd6d4737 100644 --- a/src/backend/InvenTree/order/test_api.py +++ b/src/backend/InvenTree/order/test_api.py @@ -1527,7 +1527,7 @@ class SalesOrderTest(OrderTest): self.download_file( reverse('api-so-list'), {'export': fmt}, - decode=True if fmt == 'csv' else False, + decode=fmt == 'csv', expected_code=200, expected_fn=r'InvenTree_SalesOrder_.+', ) diff --git a/src/backend/InvenTree/part/api.py b/src/backend/InvenTree/part/api.py index 5b0b40ef2f..887f11312f 100644 --- a/src/backend/InvenTree/part/api.py +++ b/src/backend/InvenTree/part/api.py @@ -167,10 +167,9 @@ class CategoryFilter(rest_filters.FilterSet): top_level = str2bool(self.data.get('top_level', None)) # If the parent is *not* provided, update the results based on the "cascade" value - if not parent or top_level: - if not value: - # If "cascade" is False, only return top-level categories - queryset = queryset.filter(parent=None) + if (not parent or top_level) and not value: + # If "cascade" is False, only return top-level categories + queryset = queryset.filter(parent=None) return queryset diff --git a/src/backend/InvenTree/part/bom.py b/src/backend/InvenTree/part/bom.py index d692b66ad9..580347f8a9 100644 --- a/src/backend/InvenTree/part/bom.py +++ b/src/backend/InvenTree/part/bom.py @@ -244,10 +244,7 @@ def ExportBom( manufacturer_name = '' # Extract the "MPN" field from the Manufacturer Part - if mp_part: - manufacturer_mpn = mp_part.MPN - else: - manufacturer_mpn = '' + manufacturer_mpn = mp_part.MPN if mp_part else '' # Generate a column name for this manufacturer k_man = f'{_("Manufacturer")}_{mp_idx}' @@ -270,10 +267,7 @@ def ExportBom( else: supplier_name = '' - if sp_part: - supplier_sku = sp_part.SKU - else: - supplier_sku = '' + supplier_sku = sp_part.SKU if sp_part else '' # Generate column names for this supplier k_sup = ( @@ -307,10 +301,7 @@ def ExportBom( supplier_parts_used.add(sp_part) - if sp_part.supplier: - supplier_name = sp_part.supplier.name - else: - supplier_name = '' + supplier_name = sp_part.supplier.name if sp_part.supplier else '' supplier_sku = sp_part.SKU diff --git a/src/backend/InvenTree/part/models.py b/src/backend/InvenTree/part/models.py index 2a500f1ff2..887e8f176f 100644 --- a/src/backend/InvenTree/part/models.py +++ b/src/backend/InvenTree/part/models.py @@ -230,10 +230,7 @@ class PartCategory(InvenTree.models.InvenTreeTree): """Get all unique parameter names for all parts from this category.""" unique_parameters_names = [] - if prefetch: - parts = prefetch - else: - parts = self.prefetch_parts_parameters(cascade=cascade) + parts = prefetch or self.prefetch_parts_parameters(cascade=cascade) for part in parts: for parameter in part.parameters.all(): @@ -247,10 +244,7 @@ class PartCategory(InvenTree.models.InvenTreeTree): """Get all parameter names and values for all parts from this category.""" category_parameters = [] - if prefetch: - parts = prefetch - else: - parts = self.prefetch_parts_parameters(cascade=cascade) + parts = prefetch or self.prefetch_parts_parameters(cascade=cascade) for part in parts: part_parameters = { @@ -917,24 +911,26 @@ class Part( 'IPN': _('Duplicate IPN not allowed in part settings') }) - if self.revision_of and self.revision: - if ( + if ( + self.revision_of + and self.revision + and ( Part.objects.exclude(pk=self.pk) .filter(revision_of=self.revision_of, revision=self.revision) .exists() - ): - raise ValidationError(_('Duplicate part revision already exists.')) + ) + ): + raise ValidationError(_('Duplicate part revision already exists.')) # Ensure unique across (Name, revision, IPN) (as specified) - if self.revision or self.IPN: - if ( - Part.objects.exclude(pk=self.pk) - .filter(name=self.name, revision=self.revision, IPN=self.IPN) - .exists() - ): - raise ValidationError( - _('Part with this Name, IPN and Revision already exists.') - ) + if (self.revision or self.IPN) and ( + Part.objects.exclude(pk=self.pk) + .filter(name=self.name, revision=self.revision, IPN=self.IPN) + .exists() + ): + raise ValidationError( + _('Part with this Name, IPN and Revision already exists.') + ) def clean(self): """Perform cleaning operations for the Part model. @@ -3872,16 +3868,18 @@ class PartParameter(InvenTree.models.InvenTreeMetadataModel): super().clean() # Validate the parameter data against the template units - if get_global_setting( - 'PART_PARAMETER_ENFORCE_UNITS', True, cache=False, create=False + if ( + get_global_setting( + 'PART_PARAMETER_ENFORCE_UNITS', True, cache=False, create=False + ) + and self.template.units ): - if self.template.units: - try: - InvenTree.conversion.convert_physical_value( - self.data, self.template.units - ) - except ValidationError as e: - raise ValidationError({'data': e.message}) + try: + InvenTree.conversion.convert_physical_value( + self.data, self.template.units + ) + except ValidationError as e: + raise ValidationError({'data': e.message}) # Validate the parameter data against the template choices if choices := self.template.get_choices(): @@ -4189,9 +4187,8 @@ class BomItem( # Check if the part was changed deltas = self.get_field_deltas() - if 'part' in deltas: - if old_part := deltas['part'].get('old', None): - self.check_part_lock(old_part) + if 'part' in deltas and (old_part := deltas['part'].get('old', None)): + self.check_part_lock(old_part) # Update the 'validated' field based on checksum calculation self.validated = self.is_line_valid diff --git a/src/backend/InvenTree/plugin/api.py b/src/backend/InvenTree/plugin/api.py index 061c35bb55..1d3f6763f8 100644 --- a/src/backend/InvenTree/plugin/api.py +++ b/src/backend/InvenTree/plugin/api.py @@ -63,7 +63,7 @@ class PluginFilter(rest_filters.FilterSet): match = True for mixin in mixins: - if mixin not in result.mixins().keys(): + if mixin not in result.mixins(): match = False break diff --git a/src/backend/InvenTree/plugin/base/barcodes/api.py b/src/backend/InvenTree/plugin/base/barcodes/api.py index 89ada2efb2..a497c79a62 100644 --- a/src/backend/InvenTree/plugin/base/barcodes/api.py +++ b/src/backend/InvenTree/plugin/base/barcodes/api.py @@ -335,9 +335,8 @@ class BarcodePOAllocate(BarcodeView): if not part and not supplier_part and not manufacturer_part: raise ValidationError({'error': _('No matching part data found')}) - if part: - if part_id := part.get('pk', None): - supplier_parts = supplier_parts.filter(part__pk=part_id) + if part and (part_id := part.get('pk', None)): + supplier_parts = supplier_parts.filter(part__pk=part_id) if supplier_part: if supplier_part_id := supplier_part.get('pk', None): diff --git a/src/backend/InvenTree/plugin/base/barcodes/serializers.py b/src/backend/InvenTree/plugin/base/barcodes/serializers.py index b31ab1818a..99baebeff0 100644 --- a/src/backend/InvenTree/plugin/base/barcodes/serializers.py +++ b/src/backend/InvenTree/plugin/base/barcodes/serializers.py @@ -41,7 +41,7 @@ class BarcodeGenerateSerializer(serializers.Serializer): plugin.base.barcodes.helper.get_supported_barcode_models_map() ) - if model not in supported_models.keys(): + if model not in supported_models: raise ValidationError(_('Model is not supported')) return model diff --git a/src/backend/InvenTree/plugin/base/event/events.py b/src/backend/InvenTree/plugin/base/event/events.py index 6aa341f3e3..ab8c03017d 100644 --- a/src/backend/InvenTree/plugin/base/event/events.py +++ b/src/backend/InvenTree/plugin/base/event/events.py @@ -147,10 +147,7 @@ def allow_table_event(table_name): 'part_partstocktakereport', ] - if table_name in ignore_tables: - return False - - return True + return not table_name in ignore_tables @receiver(post_save) diff --git a/src/backend/InvenTree/plugin/base/integration/ScheduleMixin.py b/src/backend/InvenTree/plugin/base/integration/ScheduleMixin.py index 8b350a7ed1..8f2b780781 100644 --- a/src/backend/InvenTree/plugin/base/integration/ScheduleMixin.py +++ b/src/backend/InvenTree/plugin/base/integration/ScheduleMixin.py @@ -148,7 +148,7 @@ class ScheduleMixin: def get_task_names(self): """All defined task names.""" # Returns a list of all task names associated with this plugin instance - return [self.get_task_name(key) for key in self.scheduled_tasks.keys()] + return [self.get_task_name(key) for key in self.scheduled_tasks] def register_tasks(self): """Register the tasks with the database.""" diff --git a/src/backend/InvenTree/plugin/installer.py b/src/backend/InvenTree/plugin/installer.py index d69c654d69..c8b0094ae3 100644 --- a/src/backend/InvenTree/plugin/installer.py +++ b/src/backend/InvenTree/plugin/installer.py @@ -216,10 +216,7 @@ def install_plugin(url=None, packagename=None, user=None, version=None): identifier in url for identifier in ['git+https', 'hg+https', 'svn+svn'] ]: # using a VCS provider - if packagename: - full_pkg = f'{packagename}@{url}' - else: - full_pkg = url + full_pkg = f'{packagename}@{url}' if packagename else url elif url: install_name.append('-i') full_pkg = url @@ -244,10 +241,9 @@ def install_plugin(url=None, packagename=None, user=None, version=None): ret['result'] = ret['success'] = _('Installed plugin successfully') ret['output'] = str(result, 'utf-8') - if packagename: - if path := check_package_path(packagename): - # Override result information - ret['result'] = _(f'Installed plugin into {path}') + if packagename and (path := check_package_path(packagename)): + # Override result information + ret['result'] = _(f'Installed plugin into {path}') except subprocess.CalledProcessError as error: handle_pip_error(error, 'plugin_install') diff --git a/src/backend/InvenTree/plugin/staticfiles.py b/src/backend/InvenTree/plugin/staticfiles.py index 0ece4bcc7e..f75f5ba755 100644 --- a/src/backend/InvenTree/plugin/staticfiles.py +++ b/src/backend/InvenTree/plugin/staticfiles.py @@ -39,7 +39,7 @@ def collect_plugins_static_files(): logger.info('Collecting static files for all installed plugins.') - for slug in registry.plugins.keys(): + for slug in registry.plugins: copy_plugin_static_files(slug, check_reload=False) diff --git a/src/backend/InvenTree/plugin/templatetags/plugin_extras.py b/src/backend/InvenTree/plugin/templatetags/plugin_extras.py index c7d15e95a8..9581c5e4fe 100644 --- a/src/backend/InvenTree/plugin/templatetags/plugin_extras.py +++ b/src/backend/InvenTree/plugin/templatetags/plugin_extras.py @@ -112,10 +112,7 @@ def plugin_static(context, file: str, **kwargs): """ plugin = context.get('plugin', None) - if plugin: - plugin = plugin.slug - else: - plugin = kwargs.get('plugin', None) + plugin = plugin.slug if plugin else kwargs.get('plugin', None) if not plugin: return file diff --git a/src/backend/InvenTree/script/translation_stats.py b/src/backend/InvenTree/script/translation_stats.py index a278cf8bdb..04a23ede55 100644 --- a/src/backend/InvenTree/script/translation_stats.py +++ b/src/backend/InvenTree/script/translation_stats.py @@ -59,10 +59,7 @@ if __name__ == '__main__': (total, covered, uncovered) = stats - if total > 0: - percentage = int(covered / total * 100) - else: - percentage = 0 + percentage = int(covered / total * 100) if total > 0 else 0 if verbose: print(f"| {locale.ljust(4, ' ')} : {str(percentage).rjust(4, ' ')}% |") @@ -78,9 +75,6 @@ if __name__ == '__main__': with open(STAT_FILE, 'w') as target: json.dump(locales_perc, target) - if len(percentages) > 0: - avg = int(sum(percentages) / len(percentages)) - else: - avg = 0 + avg = int(sum(percentages) / len(percentages)) if len(percentages) > 0 else 0 print(f'InvenTree translation coverage: {avg}%') diff --git a/src/backend/InvenTree/stock/api.py b/src/backend/InvenTree/stock/api.py index 073a9d1d73..e05064a43f 100644 --- a/src/backend/InvenTree/stock/api.py +++ b/src/backend/InvenTree/stock/api.py @@ -354,10 +354,9 @@ class StockLocationFilter(rest_filters.FilterSet): top_level = str2bool(self.data.get('top_level', None)) # If the parent is *not* provided, update the results based on the "cascade" value - if not parent or top_level: - if not value: - # If "cascade" is False, only return top-level location - queryset = queryset.filter(parent=None) + if (not parent or top_level) and not value: + # If "cascade" is False, only return top-level location + queryset = queryset.filter(parent=None) return queryset @@ -1452,17 +1451,17 @@ class StockTrackingList(DataExportViewMixin, ListAPI): delta_models = self.get_delta_model_map() # Construct a set of related models we need to lookup for later - related_model_lookups = {key: set() for key in delta_models.keys()} + related_model_lookups = {key: set() for key in delta_models} # Run a first pass through the data to determine which related models we need to lookup for item in data: deltas = item['deltas'] or {} - for key in delta_models.keys(): + for key in delta_models: if key in deltas: related_model_lookups[key].add(deltas[key]) - for key in delta_models.keys(): + for key in delta_models: model, serializer = delta_models[key] # Fetch all related models in one go diff --git a/src/backend/InvenTree/stock/models.py b/src/backend/InvenTree/stock/models.py index 462101bfba..4edffbf4c6 100644 --- a/src/backend/InvenTree/stock/models.py +++ b/src/backend/InvenTree/stock/models.py @@ -1183,10 +1183,7 @@ class StockItem( if self.allocations.count() > 0: return True - if self.sales_order_allocations.count() > 0: - return True - - return False + return self.sales_order_allocations.count() > 0 def build_allocation_count(self): """Return the total quantity allocated to builds.""" @@ -1260,10 +1257,7 @@ class StockItem( if self.installed_item_count() > 0: return False - if self.sales_order is not None: - return False - - return True + return not self.sales_order is not None def get_installed_items(self, cascade: bool = False) -> set[StockItem]: """Return all stock items which are *installed* in this one! @@ -1422,10 +1416,7 @@ class StockItem( if self.belongs_to is not None: return False - if self.sales_order is not None: - return False - - return True + return not self.sales_order is not None @property def tracking_info_count(self): @@ -2204,7 +2195,7 @@ class StockItem( for item in installed_items: item_results = item.testResultMap() - for key in item_results.keys(): + for key in item_results: # Results from sub items should not override master ones if key not in result_map: result_map[key] = item_results[key] @@ -2353,7 +2344,7 @@ class StockItemTracking(InvenTree.models.InvenTreeModel): def label(self): """Return label.""" - if self.tracking_type in StockHistoryCode.keys(): + if self.tracking_type in StockHistoryCode: return StockHistoryCode.label(self.tracking_type) return getattr(self, 'title', '') diff --git a/src/backend/InvenTree/stock/test_migrations.py b/src/backend/InvenTree/stock/test_migrations.py index c052325f17..05d320b588 100644 --- a/src/backend/InvenTree/stock/test_migrations.py +++ b/src/backend/InvenTree/stock/test_migrations.py @@ -226,7 +226,7 @@ class TestTestResultMigration(MigratorTestCase): # Two more test templates should have been created self.assertEqual(PartTestTemplate.objects.count(), 3) - for k in self.test_keys.keys(): + for k in self.test_keys: self.assertTrue(PartTestTemplate.objects.filter(key=k).exists()) for result in StockItemTestResult.objects.all(): diff --git a/src/backend/InvenTree/users/admin.py b/src/backend/InvenTree/users/admin.py index 09682e4915..0d06dc664d 100644 --- a/src/backend/InvenTree/users/admin.py +++ b/src/backend/InvenTree/users/admin.py @@ -32,10 +32,7 @@ class ApiTokenAdmin(admin.ModelAdmin): def get_fields(self, request, obj=None): """Return list of fields to display.""" - if obj: - fields = ['token'] - else: - fields = ['key'] + fields = ['token'] if obj else ['key'] fields += [ 'user', diff --git a/src/backend/InvenTree/users/models.py b/src/backend/InvenTree/users/models.py index 601fb93bdd..88f74db69c 100644 --- a/src/backend/InvenTree/users/models.py +++ b/src/backend/InvenTree/users/models.py @@ -489,10 +489,7 @@ def split_model(model): *app, model = model.split('_') # handle models that have - if len(app) > 1: - app = '_'.join(app) - else: - app = app[0] + app = '_'.join(app) if len(app) > 1 else app[0] return model, app @@ -674,7 +671,7 @@ def clear_user_role_cache(user: User): Args: user: The User object to be expunged from the cache """ - for role in RuleSet.get_ruleset_models().keys(): + for role in RuleSet.get_ruleset_models(): for perm in ['add', 'change', 'view', 'delete']: key = f'role_{user.pk}_{role}_{perm}' cache.delete(key) diff --git a/src/backend/InvenTree/users/tests.py b/src/backend/InvenTree/users/tests.py index 1d5d9686c0..669c550aca 100644 --- a/src/backend/InvenTree/users/tests.py +++ b/src/backend/InvenTree/users/tests.py @@ -62,7 +62,7 @@ class RuleSetModelTest(TestCase): assigned_models = set() # Now check that each defined model is a valid table name - for key in RuleSet.get_ruleset_models().keys(): + for key in RuleSet.get_ruleset_models(): models = RuleSet.get_ruleset_models()[key] for m in models: