InvenTree/tasks.py

516 lines
14 KiB
Python
Raw Normal View History

import json
2022-05-20 15:24:51 +00:00
import os
2021-11-24 22:07:48 +00:00
import pathlib
import re
2022-05-20 15:24:51 +00:00
import sys
2022-05-15 19:01:55 +00:00
from invoke import task
2021-04-11 04:05:55 +00:00
def apps():
2022-05-28 17:06:07 +00:00
"""Returns a list of installed apps"""
return [
'build',
'common',
'company',
'label',
'order',
'part',
'report',
'stock',
'users',
2022-05-18 23:02:07 +00:00
'plugin',
'InvenTree',
]
def localDir():
2022-05-28 17:06:07 +00:00
"""Returns the directory of *THIS* file.
Used to ensure that the various scripts always run
in the correct directory.
"""
return os.path.dirname(os.path.abspath(__file__))
def managePyDir():
2022-05-28 17:06:07 +00:00
"""Returns the directory of the manage.py file"""
return os.path.join(localDir(), 'InvenTree')
def managePyPath():
2022-05-28 17:06:07 +00:00
"""Return the path of the manage.py file"""
return os.path.join(managePyDir(), 'manage.py')
def manage(c, cmd, pty=False):
2022-05-28 17:06:07 +00:00
"""Runs a given command against django's "manage.py" script.
Args:
c - Command line context
cmd - django command to run
"""
2022-05-20 11:37:12 +00:00
c.run('cd "{path}" && python3 manage.py {cmd}'.format(
path=managePyDir(),
cmd=cmd
), pty=pty)
2022-05-20 11:37:12 +00:00
@task
def plugins(c):
2022-05-28 17:06:07 +00:00
"""Installs all plugins as specified in 'plugins.txt'"""
from InvenTree.InvenTree.config import get_plugin_file
plugin_file = get_plugin_file()
print(f"Installing plugin packages from '{plugin_file}'")
# Install the plugins
Docker improvements (#3042) * Simplified dockerfile - Changed from alpine to python:slim - Removed some database libs (because we *connect* to a db, not host it) * - Add gettext as required package - Only create inventree user as part of production build (leave admin access for dev build) * Tweaks for tasks.py * Fix user permissions (drop to inventree user) * Drop to the 'inventree' user level as part of init.sh - As we have mounted volumes at 'run time' we need to ensure that the inventree user has correct permissions! - Ref: https://stackoverflow.com/questions/39397548/how-to-give-non-root-user-in-docker-container-access-to-a-volume-mounted-on-the * Adjust user setup - Only drop to non-root user as part of "production" build - Mounted external volumes make it tricky when in the dev build - Might want to revisit this later on * More dockerfile changes - reduce required system packages - * Add new docker github workflow * Print some more debug * GITHUB_BASE_REF * Add gnupg to base requirements * Improve debug output during testing * Refactoring updates for label printing API - Update weasyprint version to 55.0 - Generate labels as pdf files - Provide filename to label printing plugin - Additional unit testing - Improve extraction of some hidden debug data during TESTING - Fix a spelling mistake (notifaction -> notification) * Working on github action * More testing * Add requirement for pdf2image * Fix label printing plugin and update unit testing * Add required packages for CI * Move docker files to the top level directory - This allows us to build the production image directly from soure - Don't need to re-download the source code from github - Note: The docker install guide will need to be updated! * Fix for docker ci file * Print GIT SHA * Bake git information into the production image * Add some exta docstrings to dockerfile * Simplify version check script * Extract git commit info * Extract docker tag from check_version.py * Newline * More work on the docker workflow * Dockerfile fixes - Directory / path issues * Dockerfile fixes - Directory / path issues * Ignore certain steps on a pull request * Add poppler-utils to CI * Consolidate version check into existing CI file * Don't run docker workflow on pull request * Pass docker image tag through to the build Also check .j2k files * Add supervisord.conf example file back in * Remove --no-cache-dir option from pip install
2022-05-28 23:40:37 +00:00
c.run(f"pip3 install --disable-pip-version-check -U -r '{plugin_file}'")
2022-05-20 11:37:12 +00:00
@task(post=[plugins])
def install(c):
2022-05-28 17:06:07 +00:00
"""Installs required python packages"""
print("Installing required python packages from 'requirements.txt'")
# Install required Python packages with PIP
Docker improvements (#3042) * Simplified dockerfile - Changed from alpine to python:slim - Removed some database libs (because we *connect* to a db, not host it) * - Add gettext as required package - Only create inventree user as part of production build (leave admin access for dev build) * Tweaks for tasks.py * Fix user permissions (drop to inventree user) * Drop to the 'inventree' user level as part of init.sh - As we have mounted volumes at 'run time' we need to ensure that the inventree user has correct permissions! - Ref: https://stackoverflow.com/questions/39397548/how-to-give-non-root-user-in-docker-container-access-to-a-volume-mounted-on-the * Adjust user setup - Only drop to non-root user as part of "production" build - Mounted external volumes make it tricky when in the dev build - Might want to revisit this later on * More dockerfile changes - reduce required system packages - * Add new docker github workflow * Print some more debug * GITHUB_BASE_REF * Add gnupg to base requirements * Improve debug output during testing * Refactoring updates for label printing API - Update weasyprint version to 55.0 - Generate labels as pdf files - Provide filename to label printing plugin - Additional unit testing - Improve extraction of some hidden debug data during TESTING - Fix a spelling mistake (notifaction -> notification) * Working on github action * More testing * Add requirement for pdf2image * Fix label printing plugin and update unit testing * Add required packages for CI * Move docker files to the top level directory - This allows us to build the production image directly from soure - Don't need to re-download the source code from github - Note: The docker install guide will need to be updated! * Fix for docker ci file * Print GIT SHA * Bake git information into the production image * Add some exta docstrings to dockerfile * Simplify version check script * Extract git commit info * Extract docker tag from check_version.py * Newline * More work on the docker workflow * Dockerfile fixes - Directory / path issues * Dockerfile fixes - Directory / path issues * Ignore certain steps on a pull request * Add poppler-utils to CI * Consolidate version check into existing CI file * Don't run docker workflow on pull request * Pass docker image tag through to the build Also check .j2k files * Add supervisord.conf example file back in * Remove --no-cache-dir option from pip install
2022-05-28 23:40:37 +00:00
c.run('pip3 install --no-cache-dir --disable-pip-version-check -U -r requirements.txt')
2022-05-20 11:37:12 +00:00
2022-05-15 22:19:03 +00:00
@task
2022-05-15 22:15:25 +00:00
def setup_dev(c):
2022-05-28 17:06:07 +00:00
"""Sets up everything needed for the dev enviroment"""
2022-05-15 22:15:25 +00:00
print("Installing required python packages from 'requirements.txt'")
# Install required Python packages with PIP
c.run('pip3 install -U -r requirements.txt')
# Install pre-commit hook
c.run('pre-commit install')
2022-05-15 22:19:37 +00:00
# Update all the hooks
c.run('pre-commit autoupdate')
2022-05-20 11:37:12 +00:00
2020-11-24 10:18:00 +00:00
@task
def shell(c):
2022-05-28 17:06:07 +00:00
"""Open a python shell with access to the InvenTree database models."""
2020-11-24 10:18:00 +00:00
manage(c, 'shell', pty=True)
2020-08-21 11:24:02 +00:00
@task
def superuser(c):
2022-05-28 17:06:07 +00:00
"""Create a superuser/admin account for the database."""
manage(c, 'createsuperuser', pty=True)
2020-08-21 11:24:02 +00:00
2020-09-17 12:44:17 +00:00
@task
def check(c):
2022-05-28 17:06:07 +00:00
"""Check validity of django codebase"""
2020-09-17 12:44:17 +00:00
manage(c, "check")
2021-04-10 12:35:10 +00:00
@task
def wait(c):
2022-05-28 17:06:07 +00:00
"""Wait until the database connection is ready"""
return manage(c, "wait_for_db")
@task(pre=[wait])
def worker(c):
2022-05-28 17:06:07 +00:00
"""Run the InvenTree background worker process"""
manage(c, 'qcluster', pty=True)
2021-04-10 12:35:10 +00:00
@task
def rebuild_models(c):
2022-05-28 17:06:07 +00:00
"""Rebuild database models with MPTT structures"""
manage(c, "rebuild_models", pty=True)
2021-11-19 20:50:41 +00:00
@task
def rebuild_thumbnails(c):
2022-05-28 17:06:07 +00:00
"""Rebuild missing image thumbnails"""
manage(c, "rebuild_thumbnails", pty=True)
2021-11-19 20:50:41 +00:00
2021-07-31 23:06:17 +00:00
@task
def clean_settings(c):
2022-05-28 17:06:07 +00:00
"""Clean the setting tables of old settings"""
2021-07-31 23:06:17 +00:00
manage(c, "clean_settings")
2021-11-19 20:50:41 +00:00
@task(help={'mail': 'mail of the user whos MFA should be disabled'})
def remove_mfa(c, mail=''):
2022-05-28 17:06:07 +00:00
"""Remove MFA for a user"""
if not mail:
print('You must provide a users mail')
manage(c, f"remove_mfa {mail}")
@task(post=[rebuild_models, rebuild_thumbnails])
def migrate(c):
2022-05-28 17:06:07 +00:00
"""Performs database migrations.
This is a critical step if the database schema have been altered!
"""
print("Running InvenTree database migrations...")
print("========================================")
manage(c, "makemigrations")
manage(c, "migrate --noinput")
manage(c, "migrate --run-syncdb")
manage(c, "check")
print("========================================")
print("InvenTree database migrations completed!")
@task
def static(c):
2022-05-28 17:06:07 +00:00
"""Copies required static files to the STATIC_ROOT directory, as per Django requirements."""
2021-04-20 11:37:19 +00:00
manage(c, "prerender")
2021-04-01 13:40:47 +00:00
manage(c, "collectstatic --no-input")
2021-08-19 21:36:54 +00:00
@task
def translate_stats(c):
2022-05-28 17:06:07 +00:00
"""Collect translation stats.
2021-08-19 21:36:54 +00:00
The file generated from this is needed for the UI.
"""
path = os.path.join('InvenTree', 'script', 'translation_stats.py')
c.run(f'python3 {path}')
@task(post=[translate_stats, static])
2020-08-21 11:13:28 +00:00
def translate(c):
2022-05-28 17:06:07 +00:00
"""Rebuild translation source files. (Advanced use only!)
2020-08-21 11:13:28 +00:00
Note: This command should not be used on a local install,
it is performed as part of the InvenTree translation toolchain.
2020-08-21 11:13:28 +00:00
"""
2020-10-24 11:13:40 +00:00
# Translate applicable .py / .html / .js files
manage(c, "makemessages --all -e py,html,js --no-wrap")
2020-08-21 11:13:28 +00:00
manage(c, "compilemessages")
@task(pre=[install, migrate, static, clean_settings])
def update(c):
2022-05-28 17:06:07 +00:00
"""Update InvenTree installation.
This command should be invoked after source code has been updated,
e.g. downloading new code from GitHub.
The following tasks are performed, in order:
- install
- migrate
2021-08-19 21:36:54 +00:00
- translate_stats
2021-08-19 21:37:38 +00:00
- static
- clean_settings
"""
# Recompile the translation files (.mo)
# We do not run 'invoke translate' here, as that will touch the source (.po) files too!
manage(c, 'compilemessages', pty=True)
2020-08-21 11:12:05 +00:00
@task
def style(c):
2022-05-28 17:06:07 +00:00
"""Run PEP style checks against InvenTree sourcecode"""
2020-08-21 11:12:05 +00:00
print("Running PEP style checks...")
c.run('flake8 InvenTree')
@task
2020-09-01 11:01:38 +00:00
def test(c, database=None):
2022-05-28 17:06:07 +00:00
"""Run unit-tests for InvenTree codebase."""
2020-08-21 11:10:14 +00:00
# Run sanity check on the django install
manage(c, 'check')
# Run coverage tests
manage(c, 'test', pty=True)
2020-08-21 11:10:14 +00:00
@task
def coverage(c):
2022-05-28 17:06:07 +00:00
"""Run code-coverage of the InvenTree codebase, using the 'coverage' code-analysis tools.
Generates a code coverage report (available in the htmlcov directory)
"""
# Run sanity check on the django install
manage(c, 'check')
# Run coverage tests
c.run('coverage run {manage} test {apps}'.format(
manage=managePyPath(),
apps=' '.join(apps())
))
# Generate coverage report
c.run('coverage html')
def content_excludes():
2022-05-28 17:06:07 +00:00
"""Returns a list of content types to exclude from import/export"""
excludes = [
"contenttypes",
"auth.permission",
2021-06-20 06:36:39 +00:00
"authtoken.token",
"error_report.error",
"admin.logentry",
"django_q.schedule",
"django_q.task",
"django_q.ormq",
2021-04-27 15:10:46 +00:00
"users.owner",
"exchange.rate",
"exchange.exchangebackend",
"common.notificationentry",
2021-11-28 16:21:54 +00:00
"user_sessions.session",
]
output = ""
for e in excludes:
output += f"--exclude {e} "
return output
@task(help={'filename': "Output filename (default = 'data.json')"})
def export_records(c, filename='data.json'):
2022-05-28 17:06:07 +00:00
"""Export all database records to a file"""
# Get an absolute path to the file
if not os.path.isabs(filename):
filename = os.path.join(localDir(), filename)
2021-11-19 20:50:41 +00:00
filename = os.path.abspath(filename)
print(f"Exporting database records to file '{filename}'")
if os.path.exists(filename):
response = input("Warning: file already exists. Do you want to overwrite? [y/N]: ")
response = str(response).strip().lower()
if response not in ['y', 'yes']:
print("Cancelled export operation")
2020-11-12 05:10:00 +00:00
sys.exit(1)
tmpfile = f"{filename}.tmp"
2021-06-13 18:08:42 +00:00
cmd = f"dumpdata --indent 2 --output '{tmpfile}' {content_excludes()}"
# Dump data to temporary file
2020-11-12 05:10:00 +00:00
manage(c, cmd, pty=True)
2021-04-25 01:41:48 +00:00
print("Running data post-processing step...")
# Post-process the file, to remove any "permissions" specified for a user or group
with open(tmpfile, "r") as f_in:
data = json.loads(f_in.read())
for entry in data:
if "model" in entry:
# Clear out any permissions specified for a group
if entry["model"] == "auth.group":
entry["fields"]["permissions"] = []
# Clear out any permissions specified for a user
if entry["model"] == "auth.user":
entry["fields"]["user_permissions"] = []
# Write the processed data to file
with open(filename, "w") as f_out:
f_out.write(json.dumps(data, indent=2))
print("Data export completed")
@task(help={'filename': 'Input filename', 'clear': 'Clear existing data before import'}, post=[rebuild_models, rebuild_thumbnails])
def import_records(c, filename='data.json', clear=False):
2022-05-28 17:06:07 +00:00
"""Import database records from a file"""
# Get an absolute path to the supplied filename
if not os.path.isabs(filename):
filename = os.path.join(localDir(), filename)
if not os.path.exists(filename):
print(f"Error: File '{filename}' does not exist")
2020-11-12 05:10:00 +00:00
sys.exit(1)
if clear:
delete_data(c, force=True)
print(f"Importing database records from '{filename}'")
# Pre-process the data, to remove any "permissions" specified for a user or group
tmpfile = f"{filename}.tmp.json"
with open(filename, "r") as f_in:
data = json.loads(f_in.read())
for entry in data:
if "model" in entry:
# Clear out any permissions specified for a group
if entry["model"] == "auth.group":
entry["fields"]["permissions"] = []
# Clear out any permissions specified for a user
if entry["model"] == "auth.user":
entry["fields"]["user_permissions"] = []
# Write the processed data to the tmp file
with open(tmpfile, "w") as f_out:
f_out.write(json.dumps(data, indent=2))
2021-06-13 18:08:42 +00:00
cmd = f"loaddata '{tmpfile}' -i {content_excludes()}"
manage(c, cmd, pty=True)
print("Data import completed")
@task
def delete_data(c, force=False):
2022-05-28 17:06:07 +00:00
"""Delete all database records!
Warning: This will REALLY delete all records in the database!!
"""
2022-05-20 11:37:12 +00:00
print("Deleting all data from InvenTree database...")
if force:
2021-06-21 00:38:50 +00:00
manage(c, 'flush --noinput')
else:
manage(c, 'flush')
@task(post=[rebuild_models, rebuild_thumbnails])
def import_fixtures(c):
2022-05-28 17:06:07 +00:00
"""Import fixture data into the database.
This command imports all existing test fixture data into the database.
Warning:
- Intended for testing / development only!
- Running this command may overwrite existing database data!!
- Don't say you were not warned...
"""
fixtures = [
# Build model
'build',
2020-11-12 03:48:57 +00:00
# Common models
'settings',
# Company model
'company',
'price_breaks',
'supplier_part',
# Order model
'order',
# Part model
'bom',
'category',
'params',
'part',
'test_templates',
# Stock model
'location',
'stock_tests',
'stock',
2021-04-25 01:41:48 +00:00
# Users
'users'
]
command = 'loaddata ' + ' '.join(fixtures)
manage(c, command, pty=True)
2021-02-10 15:55:04 +00:00
2020-08-21 11:51:22 +00:00
@task(help={'address': 'Server address:port (default=127.0.0.1:8000)'})
def server(c, address="127.0.0.1:8000"):
2022-05-28 17:06:07 +00:00
"""Launch a (deveopment) server using Django's in-built webserver.
Note: This is *not* sufficient for a production installation.
"""
manage(c, "runserver {address}".format(address=address), pty=True)
2021-08-28 10:59:41 +00:00
2021-11-24 22:07:48 +00:00
@task(post=[translate_stats, static, server])
def test_translations(c):
2022-05-28 17:06:07 +00:00
"""Add a fictional language to test if each component is ready for translations"""
2021-11-24 22:07:48 +00:00
import django
from django.conf import settings
# setup django
base_path = os.getcwd()
new_base_path = pathlib.Path('InvenTree').absolute()
sys.path.append(str(new_base_path))
os.chdir(new_base_path)
os.environ.setdefault('DJANGO_SETTINGS_MODULE', 'InvenTree.settings')
django.setup()
# Add language
print("Add dummy language...")
print("========================================")
manage(c, "makemessages -e py,html,js --no-wrap -l xx")
# change translation
print("Fill in dummy translations...")
print("========================================")
file_path = pathlib.Path(settings.LOCALE_PATHS[0], 'xx', 'LC_MESSAGES', 'django.po')
new_file_path = str(file_path) + '_new'
# complie regex
reg = re.compile(
2022-05-27 17:54:44 +00:00
r"[a-zA-Z0-9]{1}" + # match any single letter and number # noqa: W504
r"(?![^{\(\<]*[}\)\>])" + # that is not inside curly brackets, brackets or a tag # noqa: W504
r"(?<![^\%][^\(][)][a-z])" + # that is not a specially formatted variable with singles # noqa: W504
r"(?![^\\][\n])" # that is not a newline
)
last_string = ''
# loop through input file lines
with open(file_path, "rt") as file_org:
with open(new_file_path, "wt") as file_new:
for line in file_org:
2021-11-24 22:07:48 +00:00
if line.startswith('msgstr "'):
# write output -> replace regex matches with x in the read in (multi)string
file_new.write(f'msgstr "{reg.sub("x", last_string[7:-2])}"\n')
last_string = "" # reset (multi)string
elif line.startswith('msgid "'):
last_string = last_string + line # a new translatable string starts -> start append
file_new.write(line)
2021-11-24 22:07:48 +00:00
else:
if last_string:
last_string = last_string + line # a string is beeing read in -> continue appending
file_new.write(line)
2021-11-24 22:07:48 +00:00
# change out translation files
os.rename(file_path, str(file_path) + '_old')
os.rename(new_file_path, file_path)
# compile languages
print("Compile languages ...")
print("========================================")
manage(c, "compilemessages")
# reset cwd
os.chdir(base_path)
# set env flag
os.environ['TEST_TRANSLATIONS'] = 'True'
2021-08-28 10:59:41 +00:00
@task
def render_js_files(c):
2022-05-28 17:06:07 +00:00
"""Render templated javascript files (used for static testing)."""
2021-08-28 10:59:41 +00:00
manage(c, "test InvenTree.ci_render_js")