mirror of
https://github.com/inventree/InvenTree.git
synced 2025-05-14 02:53:08 +00:00
* Squashed commit of the following: commit 52d7ff0f650bbcfa2d93ac96562b44269d3812a7 Author: Matthias Mair <code@mjmair.com> Date: Sun Jan 7 23:03:20 2024 +0100 fixed lookup commit 0d076eaea89dce24f08af247479b3b4dff1b4df3 Author: Matthias Mair <code@mjmair.com> Date: Sun Jan 7 23:03:08 2024 +0100 switched to pathlib for lookup commit 473e75eda205793769946e923748356ffd7e5b4b Author: Matthias Mair <code@mjmair.com> Date: Sun Jan 7 22:52:30 2024 +0100 fix wrong url response commit fd74f8d703399c19cb3616ea3b2656a50cd7a6e5 Author: Matthias Mair <code@mjmair.com> Date: Sun Jan 7 21:14:38 2024 +0100 switched to ruff for import sorting commit f83fedbbb8de261ff8c706e179519e58e7a91064 Author: Matthias Mair <code@mjmair.com> Date: Sun Jan 7 21:03:14 2024 +0100 switched to single quotes everywhere commit a92442e60e23be0ff5dcf42d222b0d95823ecb9b Author: Matthias Mair <code@mjmair.com> Date: Sun Jan 7 20:58:23 2024 +0100 added autofixes commit cc66c93136fcae8a701810a4f4f38ef3b570be61 Author: Matthias Mair <code@mjmair.com> Date: Sun Jan 7 20:56:47 2024 +0100 enable autoformat commit 1f343606ec1f2a99acf8a37b9900d78a8fb37282 Author: Matthias Mair <code@mjmair.com> Date: Sun Jan 7 20:42:14 2024 +0100 Squashed commit of the following: commit f5cf7b2e7872fc19633321713965763d1890b495 Author: Matthias Mair <code@mjmair.com> Date: Sun Jan 7 20:36:57 2024 +0100 fixed reqs commit 9d845bee98befa4e53c2ac3c783bd704369e3ad2 Author: Matthias Mair <code@mjmair.com> Date: Sun Jan 7 20:32:35 2024 +0100 disable autofix/format commit aff5f271484c3500df7ddde043767c008ce4af21 Author: Matthias Mair <code@mjmair.com> Date: Sun Jan 7 20:28:50 2024 +0100 adjust checks commit 47271cf1efa848ec8374a0d83b5646d06fffa6e7 Author: Matthias Mair <code@mjmair.com> Date: Sun Jan 7 20:28:22 2024 +0100 reorder order of operations commit e1bf178b40b3f0d2d59ba92209156c43095959d2 Author: Matthias Mair <code@mjmair.com> Date: Sun Jan 7 20:01:09 2024 +0100 adapted ruff settings to better fit code base commit ad7d88a6f4f15c9552522131c4e207256fc2bbf6 Author: Matthias Mair <code@mjmair.com> Date: Sun Jan 7 19:59:45 2024 +0100 auto fixed docstring commit a2e54a760e17932dbbc2de0dec23906107f2cda9 Author: Matthias Mair <code@mjmair.com> Date: Sun Jan 7 19:46:35 2024 +0100 fix getattr useage commit cb80c73bc6c0be7f5d2ed3cc9b2ac03fdefd5c41 Author: Matthias Mair <code@mjmair.com> Date: Sun Jan 7 19:25:09 2024 +0100 fix requirements file commit b7780bbd21a32007f3b0ce495b519bf59bb19bf5 Author: Matthias Mair <code@mjmair.com> Date: Sun Jan 7 18:42:28 2024 +0100 fix removed sections commit 71f1681f55c15f62c16c1d7f30a745adc496db97 Author: Matthias Mair <code@mjmair.com> Date: Sun Jan 7 18:41:21 2024 +0100 fix djlint syntax commit a0bcf1bccef8a8ffd482f38e2063bc9066e1d759 Author: Matthias Mair <code@mjmair.com> Date: Sun Jan 7 18:35:28 2024 +0100 remove flake8 from code base commit 22475b31cc06919785be046e007915e43f356793 Author: Matthias Mair <code@mjmair.com> Date: Sun Jan 7 18:34:56 2024 +0100 remove flake8 from code base commit 0413350f14773ac6161473e0cfb069713c13c691 Author: Matthias Mair <code@mjmair.com> Date: Sun Jan 7 18:24:39 2024 +0100 moved ruff section commit d90c48a0bf98befdfacbbb093ee56cdb28afb40d Author: Matthias Mair <code@mjmair.com> Date: Sun Jan 7 18:24:24 2024 +0100 move djlint config to pyproject commit c5ce55d5119bf2e35e429986f62f875c86178ae1 Author: Matthias Mair <code@mjmair.com> Date: Sun Jan 7 18:20:39 2024 +0100 added isort again commit 42a41d23afc280d4ee6f0e640148abc6f460f05a Author: Matthias Mair <code@mjmair.com> Date: Sun Jan 7 18:19:02 2024 +0100 move config section commit 85692331816348cb1145570340d1f6488a8265cc Author: Matthias Mair <code@mjmair.com> Date: Sun Jan 7 18:17:52 2024 +0100 fix codespell error commit 2897c6704d1311a800ce5aa47878d96d6980b377 Author: Matthias Mair <code@mjmair.com> Date: Sun Jan 7 17:29:21 2024 +0100 replaced flake8 with ruff mostly for speed improvements * enable docstring checks * fix docstrings * fixed D417 Missing argument description * Squashed commit of the following: commit d3b795824b5d6d1c0eda67150b45b5cd672b3f6b Author: Matthias Mair <code@mjmair.com> Date: Sun Jan 7 22:56:17 2024 +0100 fixed source path commit 0bac0c19b88897a19d5c995e4ff50427718b827e Author: Matthias Mair <code@mjmair.com> Date: Sun Jan 7 22:47:53 2024 +0100 fixed req commit 9f61f01d9cc01f1fb7123102f3658c890469b8ce Author: Matthias Mair <code@mjmair.com> Date: Sun Jan 7 22:45:18 2024 +0100 added missing toml req commit 91b71ed24a6761b629768d0ad8829fec2819a966 Author: Matthias Mair <code@mjmair.com> Date: Sun Jan 7 20:49:50 2024 +0100 moved isort config commit 12460b04196b12d0272d40552402476d5492fea5 Author: Matthias Mair <code@mjmair.com> Date: Sun Jan 7 20:43:22 2024 +0100 remove flake8 section from setup.cfg commit f5cf7b2e7872fc19633321713965763d1890b495 Author: Matthias Mair <code@mjmair.com> Date: Sun Jan 7 20:36:57 2024 +0100 fixed reqs commit 9d845bee98befa4e53c2ac3c783bd704369e3ad2 Author: Matthias Mair <code@mjmair.com> Date: Sun Jan 7 20:32:35 2024 +0100 disable autofix/format commit aff5f271484c3500df7ddde043767c008ce4af21 Author: Matthias Mair <code@mjmair.com> Date: Sun Jan 7 20:28:50 2024 +0100 adjust checks commit 47271cf1efa848ec8374a0d83b5646d06fffa6e7 Author: Matthias Mair <code@mjmair.com> Date: Sun Jan 7 20:28:22 2024 +0100 reorder order of operations commit e1bf178b40b3f0d2d59ba92209156c43095959d2 Author: Matthias Mair <code@mjmair.com> Date: Sun Jan 7 20:01:09 2024 +0100 adapted ruff settings to better fit code base commit ad7d88a6f4f15c9552522131c4e207256fc2bbf6 Author: Matthias Mair <code@mjmair.com> Date: Sun Jan 7 19:59:45 2024 +0100 auto fixed docstring commit a2e54a760e17932dbbc2de0dec23906107f2cda9 Author: Matthias Mair <code@mjmair.com> Date: Sun Jan 7 19:46:35 2024 +0100 fix getattr useage commit cb80c73bc6c0be7f5d2ed3cc9b2ac03fdefd5c41 Author: Matthias Mair <code@mjmair.com> Date: Sun Jan 7 19:25:09 2024 +0100 fix requirements file commit b7780bbd21a32007f3b0ce495b519bf59bb19bf5 Author: Matthias Mair <code@mjmair.com> Date: Sun Jan 7 18:42:28 2024 +0100 fix removed sections commit 71f1681f55c15f62c16c1d7f30a745adc496db97 Author: Matthias Mair <code@mjmair.com> Date: Sun Jan 7 18:41:21 2024 +0100 fix djlint syntax commit a0bcf1bccef8a8ffd482f38e2063bc9066e1d759 Author: Matthias Mair <code@mjmair.com> Date: Sun Jan 7 18:35:28 2024 +0100 remove flake8 from code base commit 22475b31cc06919785be046e007915e43f356793 Author: Matthias Mair <code@mjmair.com> Date: Sun Jan 7 18:34:56 2024 +0100 remove flake8 from code base commit 0413350f14773ac6161473e0cfb069713c13c691 Author: Matthias Mair <code@mjmair.com> Date: Sun Jan 7 18:24:39 2024 +0100 moved ruff section commit d90c48a0bf98befdfacbbb093ee56cdb28afb40d Author: Matthias Mair <code@mjmair.com> Date: Sun Jan 7 18:24:24 2024 +0100 move djlint config to pyproject commit c5ce55d5119bf2e35e429986f62f875c86178ae1 Author: Matthias Mair <code@mjmair.com> Date: Sun Jan 7 18:20:39 2024 +0100 added isort again commit 42a41d23afc280d4ee6f0e640148abc6f460f05a Author: Matthias Mair <code@mjmair.com> Date: Sun Jan 7 18:19:02 2024 +0100 move config section commit 85692331816348cb1145570340d1f6488a8265cc Author: Matthias Mair <code@mjmair.com> Date: Sun Jan 7 18:17:52 2024 +0100 fix codespell error commit 2897c6704d1311a800ce5aa47878d96d6980b377 Author: Matthias Mair <code@mjmair.com> Date: Sun Jan 7 17:29:21 2024 +0100 replaced flake8 with ruff mostly for speed improvements * fix pyproject * make docstrings more uniform * auto-format * fix order * revert url change
241 lines
7.6 KiB
Python
241 lines
7.6 KiB
Python
"""Background task definitions for the 'part' app."""
|
|
|
|
import logging
|
|
import random
|
|
import time
|
|
from datetime import datetime, timedelta
|
|
|
|
from django.core.exceptions import ValidationError
|
|
from django.utils.translation import gettext_lazy as _
|
|
|
|
import common.models
|
|
import common.notifications
|
|
import common.settings
|
|
import company.models
|
|
import InvenTree.helpers
|
|
import InvenTree.helpers_model
|
|
import InvenTree.tasks
|
|
import part.models
|
|
import part.stocktake
|
|
from InvenTree.tasks import (
|
|
ScheduledTask,
|
|
check_daily_holdoff,
|
|
record_task_success,
|
|
scheduled_task,
|
|
)
|
|
|
|
logger = logging.getLogger('inventree')
|
|
|
|
|
|
def notify_low_stock(part: part.models.Part):
|
|
"""Notify interested users that a part is 'low stock'.
|
|
|
|
Rules:
|
|
- Triggered when the available stock for a given part falls be low the configured threhsold
|
|
- A notification is delivered to any users who are 'subscribed' to this part
|
|
"""
|
|
name = _('Low stock notification')
|
|
message = _(
|
|
f'The available stock for {part.name} has fallen below the configured minimum level'
|
|
)
|
|
context = {
|
|
'part': part,
|
|
'name': name,
|
|
'message': message,
|
|
'link': InvenTree.helpers_model.construct_absolute_url(part.get_absolute_url()),
|
|
'template': {'html': 'email/low_stock_notification.html', 'subject': name},
|
|
}
|
|
|
|
common.notifications.trigger_notification(
|
|
part, 'part.notify_low_stock', target_fnc=part.get_subscribers, context=context
|
|
)
|
|
|
|
|
|
def notify_low_stock_if_required(part: part.models.Part):
|
|
"""Check if the stock quantity has fallen below the minimum threshold of part.
|
|
|
|
If true, notify the users who have subscribed to the part
|
|
"""
|
|
# Run "up" the tree, to allow notification for "parent" parts
|
|
parts = part.get_ancestors(include_self=True, ascending=True)
|
|
|
|
for p in parts:
|
|
if p.is_part_low_on_stock():
|
|
InvenTree.tasks.offload_task(notify_low_stock, p)
|
|
|
|
|
|
def update_part_pricing(pricing: part.models.PartPricing, counter: int = 0):
|
|
"""Update cached pricing data for the specified PartPricing instance.
|
|
|
|
Arguments:
|
|
pricing: The target PartPricing instance to be updated
|
|
counter: How many times this function has been called in sequence
|
|
"""
|
|
logger.info('Updating part pricing for %s', pricing.part)
|
|
|
|
pricing.update_pricing(counter=counter)
|
|
|
|
|
|
@scheduled_task(ScheduledTask.DAILY)
|
|
def check_missing_pricing(limit=250):
|
|
"""Check for parts with missing or outdated pricing information.
|
|
|
|
Tests for the following conditions:
|
|
- Pricing information does not exist
|
|
- Pricing information is "old"
|
|
- Pricing information is in the wrong currency
|
|
|
|
Arguments:
|
|
limit: Maximum number of parts to process at once
|
|
"""
|
|
# Find parts for which pricing information has never been updated
|
|
results = part.models.PartPricing.objects.filter(updated=None)[:limit]
|
|
|
|
if results.count() > 0:
|
|
logger.info('Found %s parts with empty pricing', results.count())
|
|
|
|
for pp in results:
|
|
pp.schedule_for_update()
|
|
|
|
# Find any parts which have 'old' pricing information
|
|
days = int(common.models.InvenTreeSetting.get_setting('PRICING_UPDATE_DAYS', 30))
|
|
stale_date = datetime.now().date() - timedelta(days=days)
|
|
|
|
results = part.models.PartPricing.objects.filter(updated__lte=stale_date)[:limit]
|
|
|
|
if results.count() > 0:
|
|
logger.info('Found %s stale pricing entries', results.count())
|
|
|
|
for pp in results:
|
|
pp.schedule_for_update()
|
|
|
|
# Find any pricing data which is in the wrong currency
|
|
currency = common.settings.currency_code_default()
|
|
results = part.models.PartPricing.objects.exclude(currency=currency)
|
|
|
|
if results.count() > 0:
|
|
logger.info('Found %s pricing entries in the wrong currency', results.count())
|
|
|
|
for pp in results:
|
|
pp.schedule_for_update()
|
|
|
|
# Find any parts which do not have pricing information
|
|
results = part.models.Part.objects.filter(pricing_data=None)[:limit]
|
|
|
|
if results.count() > 0:
|
|
logger.info('Found %s parts without pricing', results.count())
|
|
|
|
for p in results:
|
|
pricing = p.pricing
|
|
pricing.save()
|
|
pricing.schedule_for_update()
|
|
|
|
|
|
@scheduled_task(ScheduledTask.DAILY)
|
|
def scheduled_stocktake_reports():
|
|
"""Scheduled tasks for creating automated stocktake reports.
|
|
|
|
This task runs daily, and performs the following functions:
|
|
|
|
- Delete 'old' stocktake report files after the specified period
|
|
- Generate new reports at the specified period
|
|
"""
|
|
# Sleep a random number of seconds to prevent worker conflict
|
|
time.sleep(random.randint(1, 5))
|
|
|
|
# First let's delete any old stocktake reports
|
|
delete_n_days = int(
|
|
common.models.InvenTreeSetting.get_setting(
|
|
'STOCKTAKE_DELETE_REPORT_DAYS', 30, cache=False
|
|
)
|
|
)
|
|
threshold = datetime.now() - timedelta(days=delete_n_days)
|
|
old_reports = part.models.PartStocktakeReport.objects.filter(date__lt=threshold)
|
|
|
|
if old_reports.count() > 0:
|
|
logger.info('Deleting %s stale stocktake reports', old_reports.count())
|
|
old_reports.delete()
|
|
|
|
# Next, check if stocktake functionality is enabled
|
|
if not common.models.InvenTreeSetting.get_setting(
|
|
'STOCKTAKE_ENABLE', False, cache=False
|
|
):
|
|
logger.info('Stocktake functionality is not enabled - exiting')
|
|
return
|
|
|
|
report_n_days = int(
|
|
common.models.InvenTreeSetting.get_setting(
|
|
'STOCKTAKE_AUTO_DAYS', 0, cache=False
|
|
)
|
|
)
|
|
|
|
if report_n_days < 1:
|
|
logger.info('Stocktake auto reports are disabled, exiting')
|
|
return
|
|
|
|
if not check_daily_holdoff('STOCKTAKE_RECENT_REPORT', report_n_days):
|
|
logger.info('Stocktake report was recently generated - exiting')
|
|
return
|
|
|
|
# Let's start a new stocktake report for all parts
|
|
part.stocktake.generate_stocktake_report(update_parts=True)
|
|
|
|
# Record the date of this report
|
|
record_task_success('STOCKTAKE_RECENT_REPORT')
|
|
|
|
|
|
def rebuild_parameters(template_id):
|
|
"""Rebuild all parameters for a given template.
|
|
|
|
This function is called when a base template is changed,
|
|
which may cause the base unit to be adjusted.
|
|
"""
|
|
try:
|
|
template = part.models.PartParameterTemplate.objects.get(pk=template_id)
|
|
except part.models.PartParameterTemplate.DoesNotExist:
|
|
return
|
|
|
|
parameters = part.models.PartParameter.objects.filter(template=template)
|
|
|
|
n = 0
|
|
|
|
for parameter in parameters:
|
|
# Update the parameter if the numeric value has changed
|
|
value_old = parameter.data_numeric
|
|
parameter.calculate_numeric_value()
|
|
|
|
if value_old != parameter.data_numeric:
|
|
parameter.full_clean()
|
|
parameter.save()
|
|
n += 1
|
|
|
|
if n > 0:
|
|
logger.info("Rebuilt %s parameters for template '%s'", n, template.name)
|
|
|
|
|
|
def rebuild_supplier_parts(part_id):
|
|
"""Rebuild all SupplierPart objects for a given part.
|
|
|
|
This function is called when a bart part is changed,
|
|
which may cause the native units of any supplier parts to be updated
|
|
"""
|
|
try:
|
|
prt = part.models.Part.objects.get(pk=part_id)
|
|
except part.models.Part.DoesNotExist:
|
|
return
|
|
|
|
supplier_parts = company.models.SupplierPart.objects.filter(part=prt)
|
|
|
|
n = supplier_parts.count()
|
|
|
|
for supplier_part in supplier_parts:
|
|
# Re-save the part, to ensure that the units have updated correctly
|
|
try:
|
|
supplier_part.full_clean()
|
|
supplier_part.save()
|
|
except ValidationError:
|
|
pass
|
|
|
|
if n > 0:
|
|
logger.info("Rebuilt %s supplier parts for part '%s'", n, prt.name)
|