mirror of
https://github.com/inventree/InvenTree.git
synced 2025-04-28 03:26:45 +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
210 lines
6.4 KiB
Python
210 lines
6.4 KiB
Python
"""label app specification."""
|
|
|
|
import hashlib
|
|
import logging
|
|
import os
|
|
import shutil
|
|
import warnings
|
|
from pathlib import Path
|
|
|
|
from django.apps import AppConfig
|
|
from django.conf import settings
|
|
from django.core.exceptions import AppRegistryNotReady
|
|
from django.db.utils import IntegrityError, OperationalError, ProgrammingError
|
|
|
|
import InvenTree.ready
|
|
|
|
logger = logging.getLogger('inventree')
|
|
|
|
|
|
def hashFile(filename):
|
|
"""Calculate the MD5 hash of a file."""
|
|
md5 = hashlib.md5()
|
|
|
|
with open(filename, 'rb') as f:
|
|
data = f.read()
|
|
md5.update(data)
|
|
|
|
return md5.hexdigest()
|
|
|
|
|
|
class LabelConfig(AppConfig):
|
|
"""App configuration class for the 'label' app."""
|
|
|
|
name = 'label'
|
|
|
|
def ready(self):
|
|
"""This function is called whenever the label app is loaded."""
|
|
# skip loading if plugin registry is not loaded or we run in a background thread
|
|
if (
|
|
not InvenTree.ready.isPluginRegistryLoaded()
|
|
or not InvenTree.ready.isInMainThread()
|
|
):
|
|
return
|
|
|
|
if InvenTree.ready.isRunningMigrations():
|
|
return
|
|
|
|
if (
|
|
InvenTree.ready.canAppAccessDatabase(allow_test=False)
|
|
and not InvenTree.ready.isImportingData()
|
|
):
|
|
try:
|
|
self.create_labels() # pragma: no cover
|
|
except (
|
|
AppRegistryNotReady,
|
|
IntegrityError,
|
|
OperationalError,
|
|
ProgrammingError,
|
|
):
|
|
# Database might not yet be ready
|
|
warnings.warn(
|
|
'Database was not ready for creating labels', stacklevel=2
|
|
)
|
|
|
|
def create_labels(self):
|
|
"""Create all default templates."""
|
|
# Test if models are ready
|
|
import label.models
|
|
|
|
assert bool(label.models.StockLocationLabel is not None)
|
|
|
|
# Create the categories
|
|
self.create_labels_category(
|
|
label.models.StockItemLabel,
|
|
'stockitem',
|
|
[
|
|
{
|
|
'file': 'qr.html',
|
|
'name': 'QR Code',
|
|
'description': 'Simple QR code label',
|
|
'width': 24,
|
|
'height': 24,
|
|
}
|
|
],
|
|
)
|
|
|
|
self.create_labels_category(
|
|
label.models.StockLocationLabel,
|
|
'stocklocation',
|
|
[
|
|
{
|
|
'file': 'qr.html',
|
|
'name': 'QR Code',
|
|
'description': 'Simple QR code label',
|
|
'width': 24,
|
|
'height': 24,
|
|
},
|
|
{
|
|
'file': 'qr_and_text.html',
|
|
'name': 'QR and text',
|
|
'description': 'Label with QR code and name of location',
|
|
'width': 50,
|
|
'height': 24,
|
|
},
|
|
],
|
|
)
|
|
|
|
self.create_labels_category(
|
|
label.models.PartLabel,
|
|
'part',
|
|
[
|
|
{
|
|
'file': 'part_label.html',
|
|
'name': 'Part Label',
|
|
'description': 'Simple part label',
|
|
'width': 70,
|
|
'height': 24,
|
|
},
|
|
{
|
|
'file': 'part_label_code128.html',
|
|
'name': 'Barcode Part Label',
|
|
'description': 'Simple part label with Code128 barcode',
|
|
'width': 70,
|
|
'height': 24,
|
|
},
|
|
],
|
|
)
|
|
|
|
self.create_labels_category(
|
|
label.models.BuildLineLabel,
|
|
'buildline',
|
|
[
|
|
{
|
|
'file': 'buildline_label.html',
|
|
'name': 'Build Line Label',
|
|
'description': 'Example build line label',
|
|
'width': 125,
|
|
'height': 48,
|
|
}
|
|
],
|
|
)
|
|
|
|
def create_labels_category(self, model, ref_name, labels):
|
|
"""Create folder and database entries for the default templates, if they do not already exist."""
|
|
# Create root dir for templates
|
|
src_dir = Path(__file__).parent.joinpath('templates', 'label', ref_name)
|
|
|
|
dst_dir = settings.MEDIA_ROOT.joinpath('label', 'inventree', ref_name)
|
|
|
|
if not dst_dir.exists():
|
|
logger.info("Creating required directory: '%s'", dst_dir)
|
|
dst_dir.mkdir(parents=True, exist_ok=True)
|
|
|
|
# Create labels
|
|
for label in labels:
|
|
self.create_template_label(model, src_dir, ref_name, label)
|
|
|
|
def create_template_label(self, model, src_dir, ref_name, label):
|
|
"""Ensure a label template is in place."""
|
|
filename = os.path.join('label', 'inventree', ref_name, label['file'])
|
|
|
|
src_file = src_dir.joinpath(label['file'])
|
|
dst_file = settings.MEDIA_ROOT.joinpath(filename)
|
|
|
|
to_copy = False
|
|
|
|
if dst_file.exists():
|
|
# File already exists - let's see if it is the "same"
|
|
|
|
if hashFile(dst_file) != hashFile(src_file): # pragma: no cover
|
|
logger.info("Hash differs for '%s'", filename)
|
|
to_copy = True
|
|
|
|
else:
|
|
logger.info("Label template '%s' is not present", filename)
|
|
to_copy = True
|
|
|
|
if to_copy:
|
|
logger.info("Copying label template '%s'", dst_file)
|
|
# Ensure destination dir exists
|
|
dst_file.parent.mkdir(parents=True, exist_ok=True)
|
|
|
|
# Copy file
|
|
shutil.copyfile(src_file, dst_file)
|
|
|
|
# Check if a label matching the template already exists
|
|
try:
|
|
if model.objects.filter(label=filename).exists():
|
|
return # pragma: no cover
|
|
except Exception:
|
|
logger.exception(
|
|
"Failed to query label for '%s' - you should run 'invoke update' first!",
|
|
filename,
|
|
)
|
|
|
|
logger.info("Creating entry for %s '%s'", model, label['name'])
|
|
|
|
try:
|
|
model.objects.create(
|
|
name=label['name'],
|
|
description=label['description'],
|
|
label=filename,
|
|
filters='',
|
|
enabled=True,
|
|
width=label['width'],
|
|
height=label['height'],
|
|
)
|
|
except Exception:
|
|
logger.warning("Failed to create label '%s'", label['name'])
|