diff --git a/InvenTree/InvenTree/api.py b/InvenTree/InvenTree/api.py index e468e8d1cf..9d901516d5 100644 --- a/InvenTree/InvenTree/api.py +++ b/InvenTree/InvenTree/api.py @@ -13,15 +13,11 @@ from django_filters.rest_framework import DjangoFilterBackend from rest_framework import filters from rest_framework import permissions -from rest_framework.response import Response -from rest_framework.views import APIView from .views import AjaxView from .version import inventreeVersion, inventreeApiVersion, inventreeInstanceName from .status import is_worker_running -from plugin import registry - class InfoView(AjaxView): """ Simple JSON endpoint for InvenTree information. @@ -119,40 +115,3 @@ class AttachmentMixin: attachment = serializer.save() attachment.user = self.request.user attachment.save() - - -class ActionPluginView(APIView): - """ - Endpoint for running custom action plugins. - """ - - permission_classes = [ - permissions.IsAuthenticated, - ] - - def post(self, request, *args, **kwargs): - - action = request.data.get('action', None) - - data = request.data.get('data', None) - - if action is None: - return Response({ - 'error': _("No action specified") - }) - - action_plugins = registry.with_mixin('action') - for plugin in action_plugins: - if plugin.action_name() == action: - # TODO @matmair use easier syntax once InvenTree 0.7.0 is released - plugin.init(request.user, data=data) - - plugin.perform_action() - - return Response(plugin.get_response()) - - # If we got to here, no matching action was found - return Response({ - 'error': _("No matching action found"), - "action": action, - }) diff --git a/InvenTree/InvenTree/apps.py b/InvenTree/InvenTree/apps.py index 3531cc11d7..3a3290f78b 100644 --- a/InvenTree/InvenTree/apps.py +++ b/InvenTree/InvenTree/apps.py @@ -197,8 +197,6 @@ class InvenTreeConfig(AppConfig): logger.info(f'User {str(new_user)} was created!') except IntegrityError as _e: logger.warning(f'The user "{add_user}" could not be created due to the following error:\n{str(_e)}') - if settings.TESTING_ENV: - raise _e # do not try again settings.USER_ADDED = True diff --git a/InvenTree/InvenTree/test_api.py b/InvenTree/InvenTree/test_api.py index 0e813d48de..f55dfbcda2 100644 --- a/InvenTree/InvenTree/test_api.py +++ b/InvenTree/InvenTree/test_api.py @@ -96,6 +96,12 @@ class HTMLAPITests(TestCase): response = self.client.get(url, HTTP_ACCEPT='text/html') self.assertEqual(response.status_code, 200) + def test_not_found(self): + """Test that the NotFoundView is working""" + + response = self.client.get('/api/anc') + self.assertEqual(response.status_code, 404) + class APITests(InvenTreeAPITestCase): """ Tests for the InvenTree API """ diff --git a/InvenTree/InvenTree/tests.py b/InvenTree/InvenTree/tests.py index dc3aff85e6..8b2f4c133a 100644 --- a/InvenTree/InvenTree/tests.py +++ b/InvenTree/InvenTree/tests.py @@ -451,6 +451,11 @@ class TestSettings(TestCase): self.user_mdl = get_user_model() self.env = EnvironmentVarGuard() + # Create a user for auth + user = get_user_model() + self.user = user.objects.create_superuser('testuser1', 'test1@testing.com', 'password1') + self.client.login(username='testuser1', password='password1') + def run_reload(self): from plugin import registry @@ -467,23 +472,49 @@ class TestSettings(TestCase): # nothing set self.run_reload() - self.assertEqual(user_count(), 0) + self.assertEqual(user_count(), 1) # not enough set self.env.set('INVENTREE_ADMIN_USER', 'admin') # set username self.run_reload() - self.assertEqual(user_count(), 0) + self.assertEqual(user_count(), 1) # enough set self.env.set('INVENTREE_ADMIN_USER', 'admin') # set username self.env.set('INVENTREE_ADMIN_EMAIL', 'info@example.com') # set email self.env.set('INVENTREE_ADMIN_PASSWORD', 'password123') # set password self.run_reload() - self.assertEqual(user_count(), 1) + self.assertEqual(user_count(), 2) + + # create user manually + self.user_mdl.objects.create_user('testuser', 'test@testing.com', 'password') + self.assertEqual(user_count(), 3) + # check it will not be created again + self.env.set('INVENTREE_ADMIN_USER', 'testuser') + self.env.set('INVENTREE_ADMIN_EMAIL', 'test@testing.com') + self.env.set('INVENTREE_ADMIN_PASSWORD', 'password') + self.run_reload() + self.assertEqual(user_count(), 3) # make sure to clean up settings.TESTING_ENV = False + def test_initial_install(self): + """Test if install of plugins on startup works""" + from plugin import registry + + # Check an install run + response = registry.install_plugin_file() + self.assertEqual(response, 'first_run') + + # Set dynamic setting to True and rerun to launch install + InvenTreeSetting.set_setting('PLUGIN_ON_STARTUP', True, self.user) + registry.reload_plugins() + + # Check that there was anotehr run + response = registry.install_plugin_file() + self.assertEqual(response, True) + def test_helpers_cfg_file(self): # normal run - not configured self.assertIn('InvenTree/InvenTree/config.yaml', config.get_config_file()) diff --git a/InvenTree/InvenTree/urls.py b/InvenTree/InvenTree/urls.py index d408bd7346..ab761c611d 100644 --- a/InvenTree/InvenTree/urls.py +++ b/InvenTree/InvenTree/urls.py @@ -27,7 +27,7 @@ from order.api import order_api_urls from label.api import label_api_urls from report.api import report_api_urls from plugin.api import plugin_api_urls -from plugin.barcode import barcode_api_urls +from users.api import user_urls from django.conf import settings from django.conf.urls.static import static @@ -45,20 +45,11 @@ from .views import DynamicJsView from .views import NotificationsView from .api import InfoView, NotFoundView -from .api import ActionPluginView - -from users.api import user_urls admin.site.site_header = "InvenTree Admin" -apipatterns = [] -if settings.PLUGINS_ENABLED: - apipatterns.append( - re_path(r'^plugin/', include(plugin_api_urls)) - ) - -apipatterns += [ +apipatterns = [ re_path(r'^settings/', include(settings_api_urls)), re_path(r'^part/', include(part_api_urls)), re_path(r'^bom/', include(bom_api_urls)), @@ -68,13 +59,10 @@ apipatterns += [ re_path(r'^order/', include(order_api_urls)), re_path(r'^label/', include(label_api_urls)), re_path(r'^report/', include(report_api_urls)), - - # User URLs re_path(r'^user/', include(user_urls)), # Plugin endpoints - re_path(r'^action/', ActionPluginView.as_view(), name='api-action-plugin'), - re_path(r'^barcode/', include(barcode_api_urls)), + path('', include(plugin_api_urls)), # Webhook enpoint path('', include(common_api_urls)), diff --git a/InvenTree/common/models.py b/InvenTree/common/models.py index 2ec83c962c..8b50d05413 100644 --- a/InvenTree/common/models.py +++ b/InvenTree/common/models.py @@ -271,9 +271,9 @@ class BaseInvenTreeSetting(models.Model): plugin = kwargs.get('plugin', None) if plugin is not None: - from plugin import InvenTreePluginBase + from plugin import InvenTreePlugin - if issubclass(plugin.__class__, InvenTreePluginBase): + if issubclass(plugin.__class__, InvenTreePlugin): plugin = plugin.plugin_config() filters['plugin'] = plugin @@ -375,9 +375,9 @@ class BaseInvenTreeSetting(models.Model): filters['user'] = user if plugin is not None: - from plugin import InvenTreePluginBase + from plugin import InvenTreePlugin - if issubclass(plugin.__class__, InvenTreePluginBase): + if issubclass(plugin.__class__, InvenTreePlugin): filters['plugin'] = plugin.plugin_config() else: filters['plugin'] = plugin diff --git a/InvenTree/common/notifications.py b/InvenTree/common/notifications.py index ef4de4fc61..7f6aece282 100644 --- a/InvenTree/common/notifications.py +++ b/InvenTree/common/notifications.py @@ -108,7 +108,7 @@ class NotificationMethod: return False # Check if method globally enabled - plg_instance = registry.plugins.get(plg_cls.PLUGIN_NAME.lower()) + plg_instance = registry.plugins.get(plg_cls.NAME.lower()) if plg_instance and not plg_instance.get_setting(self.GLOBAL_SETTING): return True diff --git a/InvenTree/common/tests.py b/InvenTree/common/tests.py index 8fa0f3b28e..eba05639a6 100644 --- a/InvenTree/common/tests.py +++ b/InvenTree/common/tests.py @@ -10,7 +10,8 @@ from django.urls import reverse from InvenTree.api_tester import InvenTreeAPITestCase from InvenTree.helpers import str2bool -from plugin.models import NotificationUserSetting +from plugin.models import NotificationUserSetting, PluginConfig +from plugin import registry from .models import InvenTreeSetting, InvenTreeUserSetting, WebhookEndpoint, WebhookMessage, NotificationEntry from .api import WebhookView @@ -477,15 +478,36 @@ class PluginSettingsApiTest(InvenTreeAPITestCase): self.get(url, expected_code=200) - def test_invalid_plugin_slug(self): - """Test that an invalid plugin slug returns a 404""" + def test_valid_plugin_slug(self): + """Test that an valid plugin slug runs through""" + # load plugin configs + fixtures = PluginConfig.objects.all() + if not fixtures: + registry.reload_plugins() + fixtures = PluginConfig.objects.all() + # get data + url = reverse('api-plugin-setting-detail', kwargs={'plugin': 'sample', 'key': 'API_KEY'}) + response = self.get(url, expected_code=200) + + # check the right setting came through + self.assertTrue(response.data['key'], 'API_KEY') + self.assertTrue(response.data['plugin'], 'sample') + self.assertTrue(response.data['type'], 'string') + self.assertTrue(response.data['description'], 'Key required for accessing external API') + + # Failure mode tests + + # Non - exsistant plugin url = reverse('api-plugin-setting-detail', kwargs={'plugin': 'doesnotexist', 'key': 'doesnotmatter'}) - response = self.get(url, expected_code=404) - self.assertIn("Plugin 'doesnotexist' not installed", str(response.data)) + # Wrong key + url = reverse('api-plugin-setting-detail', kwargs={'plugin': 'sample', 'key': 'doesnotexsist'}) + response = self.get(url, expected_code=404) + self.assertIn("Plugin 'sample' has no setting matching 'doesnotexsist'", str(response.data)) + def test_invalid_setting_key(self): """Test that an invalid setting key returns a 404""" ... diff --git a/InvenTree/label/api.py b/InvenTree/label/api.py index c7a824d8c8..cb4b939157 100644 --- a/InvenTree/label/api.py +++ b/InvenTree/label/api.py @@ -156,7 +156,7 @@ class LabelPrintMixin: # Offload a background task to print the provided label offload_task( - 'plugin.events.print_label', + 'plugin.base.label.label.print_label', plugin.plugin_slug(), image, label_instance=label_instance, diff --git a/InvenTree/plugin/__init__.py b/InvenTree/plugin/__init__.py index b8e40e4271..ae38ae6e5b 100644 --- a/InvenTree/plugin/__init__.py +++ b/InvenTree/plugin/__init__.py @@ -3,17 +3,14 @@ Utility file to enable simper imports """ from .registry import registry -from .plugin import InvenTreePluginBase -from .integration import IntegrationPluginBase -from .action import ActionPlugin - +from .plugin import InvenTreePlugin, IntegrationPluginBase from .helpers import MixinNotImplementedError, MixinImplementationError __all__ = [ - 'ActionPlugin', - 'IntegrationPluginBase', - 'InvenTreePluginBase', 'registry', + + 'InvenTreePlugin', + IntegrationPluginBase, 'MixinNotImplementedError', 'MixinImplementationError', ] diff --git a/InvenTree/plugin/action.py b/InvenTree/plugin/action.py deleted file mode 100644 index 9586355aea..0000000000 --- a/InvenTree/plugin/action.py +++ /dev/null @@ -1,23 +0,0 @@ -# -*- coding: utf-8 -*- -"""Class for ActionPlugin""" - -import logging -import warnings - -from plugin.builtin.action.mixins import ActionMixin -import plugin.integration - - -logger = logging.getLogger("inventree") - - -class ActionPlugin(ActionMixin, plugin.integration.IntegrationPluginBase): - """ - Legacy action definition - will be replaced - Please use the new Integration Plugin API and the Action mixin - """ - # TODO @matmair remove this with InvenTree 0.7.0 - def __init__(self, user=None, data=None): - warnings.warn("using the ActionPlugin is depreceated", DeprecationWarning) - super().__init__() - self.init(user, data) diff --git a/InvenTree/plugin/api.py b/InvenTree/plugin/api.py index b9fd6e643d..f3710e4835 100644 --- a/InvenTree/plugin/api.py +++ b/InvenTree/plugin/api.py @@ -5,6 +5,7 @@ JSON API for the plugin app # -*- coding: utf-8 -*- from __future__ import unicode_literals +from django.conf import settings from django.urls import include, re_path from rest_framework import generics @@ -16,6 +17,8 @@ from rest_framework.response import Response from django_filters.rest_framework import DjangoFilterBackend from common.api import GlobalSettingsPermissions +from plugin.base.barcodes.api import barcode_api_urls +from plugin.base.action.api import ActionPluginView from plugin.models import PluginConfig, PluginSetting import plugin.serializers as PluginSerializers from plugin.registry import registry @@ -141,7 +144,8 @@ class PluginSettingDetail(generics.RetrieveUpdateAPIView): plugin = registry.get_plugin(plugin_slug) if plugin is None: - raise NotFound(detail=f"Plugin '{plugin_slug}' not found") + # This only occurs if the plugin mechanism broke + raise NotFound(detail=f"Plugin '{plugin_slug}' not found") # pragma: no cover settings = getattr(plugin, 'SETTINGS', {}) @@ -157,6 +161,11 @@ class PluginSettingDetail(generics.RetrieveUpdateAPIView): plugin_api_urls = [ + re_path(r'^action/', ActionPluginView.as_view(), name='api-action-plugin'), + re_path(r'^barcode/', include(barcode_api_urls)), +] + +general_plugin_api_urls = [ # Plugin settings URLs re_path(r'^settings/', include([ @@ -174,3 +183,8 @@ plugin_api_urls = [ # Anything else re_path(r'^.*$', PluginList.as_view(), name='api-plugin-list'), ] + +if settings.PLUGINS_ENABLED: + plugin_api_urls.append( + re_path(r'^plugin/', include(general_plugin_api_urls)) + ) diff --git a/InvenTree/plugin/base/__init__.py b/InvenTree/plugin/base/__init__.py new file mode 100644 index 0000000000..e69de29bb2 diff --git a/InvenTree/plugin/base/action/__init__.py b/InvenTree/plugin/base/action/__init__.py new file mode 100644 index 0000000000..e69de29bb2 diff --git a/InvenTree/plugin/base/action/api.py b/InvenTree/plugin/base/action/api.py new file mode 100644 index 0000000000..998e410dce --- /dev/null +++ b/InvenTree/plugin/base/action/api.py @@ -0,0 +1,45 @@ +"""APIs for action plugins""" +from django.utils.translation import gettext_lazy as _ + +from rest_framework import permissions +from rest_framework.response import Response +from rest_framework.views import APIView + +from plugin import registry + + +class ActionPluginView(APIView): + """ + Endpoint for running custom action plugins. + """ + + permission_classes = [ + permissions.IsAuthenticated, + ] + + def post(self, request, *args, **kwargs): + + action = request.data.get('action', None) + + data = request.data.get('data', None) + + if action is None: + return Response({ + 'error': _("No action specified") + }) + + action_plugins = registry.with_mixin('action') + for plugin in action_plugins: + if plugin.action_name() == action: + # TODO @matmair use easier syntax once InvenTree 0.7.0 is released + plugin.init(request.user, data=data) + + plugin.perform_action() + + return Response(plugin.get_response()) + + # If we got to here, no matching action was found + return Response({ + 'error': _("No matching action found"), + "action": action, + }) diff --git a/InvenTree/plugin/builtin/action/mixins.py b/InvenTree/plugin/base/action/mixins.py similarity index 93% rename from InvenTree/plugin/builtin/action/mixins.py rename to InvenTree/plugin/base/action/mixins.py index 18a1876659..70fea86a7e 100644 --- a/InvenTree/plugin/builtin/action/mixins.py +++ b/InvenTree/plugin/base/action/mixins.py @@ -15,16 +15,17 @@ class ActionMixin: """ MIXIN_NAME = 'Actions' - def __init__(self): + def __init__(self, user=None, data=None): super().__init__() self.add_mixin('action', True, __class__) + self.init(user, data) def action_name(self): """ Action name for this plugin. If the ACTION_NAME parameter is empty, - uses the PLUGIN_NAME instead. + uses the NAME instead. """ if self.ACTION_NAME: return self.ACTION_NAME diff --git a/InvenTree/plugin/test_action.py b/InvenTree/plugin/base/action/test_action.py similarity index 50% rename from InvenTree/plugin/test_action.py rename to InvenTree/plugin/base/action/test_action.py index 2a9e4a9a37..9de672cd6f 100644 --- a/InvenTree/plugin/test_action.py +++ b/InvenTree/plugin/base/action/test_action.py @@ -1,34 +1,38 @@ """ Unit tests for action plugins """ from django.test import TestCase +from django.contrib.auth import get_user_model -from plugin.action import ActionPlugin +from plugin import InvenTreePlugin +from plugin.mixins import ActionMixin -class ActionPluginTests(TestCase): - """ Tests for ActionPlugin """ +class ActionMixinTests(TestCase): + """ Tests for ActionMixin """ ACTION_RETURN = 'a action was performed' def setUp(self): - self.plugin = ActionPlugin('user') + class SimplePlugin(ActionMixin, InvenTreePlugin): + pass + self.plugin = SimplePlugin('user') - class TestActionPlugin(ActionPlugin): + class TestActionPlugin(ActionMixin, InvenTreePlugin): """a action plugin""" ACTION_NAME = 'abc123' def perform_action(self): - return ActionPluginTests.ACTION_RETURN + 'action' + return ActionMixinTests.ACTION_RETURN + 'action' def get_result(self): - return ActionPluginTests.ACTION_RETURN + 'result' + return ActionMixinTests.ACTION_RETURN + 'result' def get_info(self): - return ActionPluginTests.ACTION_RETURN + 'info' + return ActionMixinTests.ACTION_RETURN + 'info' self.action_plugin = TestActionPlugin('user') - class NameActionPlugin(ActionPlugin): - PLUGIN_NAME = 'Aplugin' + class NameActionPlugin(ActionMixin, InvenTreePlugin): + NAME = 'Aplugin' self.action_name = NameActionPlugin('user') @@ -59,3 +63,32 @@ class ActionPluginTests(TestCase): "result": self.ACTION_RETURN + 'result', "info": self.ACTION_RETURN + 'info', }) + + +class APITests(TestCase): + """ Tests for action api """ + + def setUp(self): + # Create a user for auth + user = get_user_model() + self.test_user = user.objects.create_user('testuser', 'test@testing.com', 'password') + self.client.login(username='testuser', password='password') + + def test_post_errors(self): + """Check the possible errors with post""" + + # Test empty request + response = self.client.post('/api/action/') + self.assertEqual(response.status_code, 200) + self.assertEqual( + response.data, + {'error': 'No action specified'} + ) + + # Test non-exsisting action + response = self.client.post('/api/action/', data={'action': "nonexsisting"}) + self.assertEqual(response.status_code, 200) + self.assertEqual( + response.data, + {'error': 'No matching action found', 'action': 'nonexsisting'} + ) diff --git a/InvenTree/plugin/base/barcodes/__init__.py b/InvenTree/plugin/base/barcodes/__init__.py new file mode 100644 index 0000000000..e69de29bb2 diff --git a/InvenTree/plugin/barcode.py b/InvenTree/plugin/base/barcodes/api.py similarity index 98% rename from InvenTree/plugin/barcode.py rename to InvenTree/plugin/base/barcodes/api.py index 98b111f073..296986b2d1 100644 --- a/InvenTree/plugin/barcode.py +++ b/InvenTree/plugin/base/barcodes/api.py @@ -11,7 +11,7 @@ from stock.models import StockItem from stock.serializers import StockItemSerializer from plugin.builtin.barcodes.inventree_barcode import InvenTreeBarcodePlugin -from plugin.builtin.barcodes.mixins import hash_barcode +from plugin.base.barcodes.mixins import hash_barcode from plugin import registry @@ -237,7 +237,7 @@ class BarcodeAssign(APIView): barcode_api_urls = [ - + # Link a barcode to a part path('link/', BarcodeAssign.as_view(), name='api-barcode-link'), # Catch-all performs barcode 'scan' diff --git a/InvenTree/plugin/builtin/barcodes/mixins.py b/InvenTree/plugin/base/barcodes/mixins.py similarity index 100% rename from InvenTree/plugin/builtin/barcodes/mixins.py rename to InvenTree/plugin/base/barcodes/mixins.py diff --git a/InvenTree/plugin/test_barcode.py b/InvenTree/plugin/base/barcodes/test_barcode.py similarity index 100% rename from InvenTree/plugin/test_barcode.py rename to InvenTree/plugin/base/barcodes/test_barcode.py diff --git a/InvenTree/plugin/base/event/__init__.py b/InvenTree/plugin/base/event/__init__.py new file mode 100644 index 0000000000..e69de29bb2 diff --git a/InvenTree/plugin/base/event/events.py b/InvenTree/plugin/base/event/events.py new file mode 100644 index 0000000000..59dfdce86a --- /dev/null +++ b/InvenTree/plugin/base/event/events.py @@ -0,0 +1,192 @@ +""" +Functions for triggering and responding to server side events +""" + +# -*- coding: utf-8 -*- +from __future__ import unicode_literals + +import logging + +from django.conf import settings +from django.db import transaction +from django.db.models.signals import post_save, post_delete +from django.dispatch.dispatcher import receiver + +from InvenTree.ready import canAppAccessDatabase, isImportingData +from InvenTree.tasks import offload_task + +from plugin.registry import registry + + +logger = logging.getLogger('inventree') + + +def trigger_event(event, *args, **kwargs): + """ + Trigger an event with optional arguments. + + This event will be stored in the database, + and the worker will respond to it later on. + """ + + if not settings.PLUGINS_ENABLED: + # Do nothing if plugins are not enabled + return + + if not canAppAccessDatabase(): + logger.debug(f"Ignoring triggered event '{event}' - database not ready") + return + + logger.debug(f"Event triggered: '{event}'") + + offload_task( + 'plugin.events.register_event', + event, + *args, + **kwargs + ) + + +def register_event(event, *args, **kwargs): + """ + Register the event with any interested plugins. + + Note: This function is processed by the background worker, + as it performs multiple database access operations. + """ + from common.models import InvenTreeSetting + + logger.debug(f"Registering triggered event: '{event}'") + + # Determine if there are any plugins which are interested in responding + if settings.PLUGIN_TESTING or InvenTreeSetting.get_setting('ENABLE_PLUGINS_EVENTS'): + + with transaction.atomic(): + + for slug, plugin in registry.plugins.items(): + + if plugin.mixin_enabled('events'): + + config = plugin.plugin_config() + + if config and config.active: + + logger.debug(f"Registering callback for plugin '{slug}'") + + # Offload a separate task for each plugin + offload_task( + 'plugin.events.process_event', + slug, + event, + *args, + **kwargs + ) + + +def process_event(plugin_slug, event, *args, **kwargs): + """ + Respond to a triggered event. + + This function is run by the background worker process. + + This function may queue multiple functions to be handled by the background worker. + """ + + logger.info(f"Plugin '{plugin_slug}' is processing triggered event '{event}'") + + plugin = registry.plugins.get(plugin_slug, None) + + if plugin is None: + logger.error(f"Could not find matching plugin for '{plugin_slug}'") + return + + plugin.process_event(event, *args, **kwargs) + + +def allow_table_event(table_name): + """ + Determine if an automatic event should be fired for a given table. + We *do not* want events to be fired for some tables! + """ + + if isImportingData(): + # Prevent table events during the data import process + return False + + table_name = table_name.lower().strip() + + # Ignore any tables which start with these prefixes + ignore_prefixes = [ + 'account_', + 'auth_', + 'authtoken_', + 'django_', + 'error_', + 'exchange_', + 'otp_', + 'plugin_', + 'socialaccount_', + 'user_', + 'users_', + ] + + if any([table_name.startswith(prefix) for prefix in ignore_prefixes]): + return False + + ignore_tables = [ + 'common_notificationentry', + 'common_webhookendpoint', + 'common_webhookmessage', + ] + + if table_name in ignore_tables: + return False + + return True + + +@receiver(post_save) +def after_save(sender, instance, created, **kwargs): + """ + Trigger an event whenever a database entry is saved + """ + + table = sender.objects.model._meta.db_table + + instance_id = getattr(instance, 'id', None) + + if instance_id is None: + return + + if not allow_table_event(table): + return + + if created: + trigger_event( + f'{table}.created', + id=instance.id, + model=sender.__name__, + ) + else: + trigger_event( + f'{table}.saved', + id=instance.id, + model=sender.__name__, + ) + + +@receiver(post_delete) +def after_delete(sender, instance, **kwargs): + """ + Trigger an event whenever a database entry is deleted + """ + + table = sender.objects.model._meta.db_table + + if not allow_table_event(table): + return + + trigger_event( + f'{table}.deleted', + model=sender.__name__, + ) diff --git a/InvenTree/plugin/base/event/mixins.py b/InvenTree/plugin/base/event/mixins.py new file mode 100644 index 0000000000..8df9bfd164 --- /dev/null +++ b/InvenTree/plugin/base/event/mixins.py @@ -0,0 +1,29 @@ +"""Plugin mixin class for events""" + +from plugin.helpers import MixinNotImplementedError + + +class EventMixin: + """ + Mixin that provides support for responding to triggered events. + + Implementing classes must provide a "process_event" function: + """ + + def process_event(self, event, *args, **kwargs): + """ + Function to handle events + Must be overridden by plugin + """ + # Default implementation does not do anything + raise MixinNotImplementedError + + class MixinMeta: + """ + Meta options for this mixin + """ + MIXIN_NAME = 'Events' + + def __init__(self): + super().__init__() + self.add_mixin('events', True, __class__) diff --git a/InvenTree/plugin/base/integration/__init__.py b/InvenTree/plugin/base/integration/__init__.py new file mode 100644 index 0000000000..e69de29bb2 diff --git a/InvenTree/plugin/builtin/integration/mixins.py b/InvenTree/plugin/base/integration/mixins.py similarity index 90% rename from InvenTree/plugin/builtin/integration/mixins.py rename to InvenTree/plugin/base/integration/mixins.py index b22efc9415..86e3092e4f 100644 --- a/InvenTree/plugin/builtin/integration/mixins.py +++ b/InvenTree/plugin/base/integration/mixins.py @@ -11,9 +11,8 @@ from django.db.utils import OperationalError, ProgrammingError import InvenTree.helpers -from plugin.helpers import MixinImplementationError, MixinNotImplementedError +from plugin.helpers import MixinImplementationError, MixinNotImplementedError, render_template from plugin.models import PluginConfig, PluginSetting -from plugin.template import render_template from plugin.urls import PLUGIN_BASE @@ -238,32 +237,6 @@ class ScheduleMixin: logger.warning("unregister_tasks failed, database not ready") -class EventMixin: - """ - Mixin that provides support for responding to triggered events. - - Implementing classes must provide a "process_event" function: - """ - - def process_event(self, event, *args, **kwargs): - """ - Function to handle events - Must be overridden by plugin - """ - # Default implementation does not do anything - raise MixinNotImplementedError - - class MixinMeta: - """ - Meta options for this mixin - """ - MIXIN_NAME = 'Events' - - def __init__(self): - super().__init__() - self.add_mixin('events', True, __class__) - - class UrlsMixin: """ Mixin that enables custom URLs for the plugin @@ -396,42 +369,6 @@ class AppMixin: return True -class LabelPrintingMixin: - """ - Mixin which enables direct printing of stock labels. - - Each plugin must provide a PLUGIN_NAME attribute, which is used to uniquely identify the printer. - - The plugin must also implement the print_label() function - """ - - class MixinMeta: - """ - Meta options for this mixin - """ - MIXIN_NAME = 'Label printing' - - def __init__(self): # pragma: no cover - super().__init__() - self.add_mixin('labels', True, __class__) - - def print_label(self, label, **kwargs): - """ - Callback to print a single label - - Arguments: - label: A black-and-white pillow Image object - - kwargs: - length: The length of the label (in mm) - width: The width of the label (in mm) - - """ - - # Unimplemented (to be implemented by the particular plugin class) - ... # pragma: no cover - - class APICallMixin: """ Mixin that enables easier API calls for a plugin @@ -447,15 +384,15 @@ class APICallMixin: Example: ``` - from plugin import IntegrationPluginBase + from plugin import InvenTreePlugin from plugin.mixins import APICallMixin, SettingsMixin - class SampleApiCallerPlugin(APICallMixin, SettingsMixin, IntegrationPluginBase): + class SampleApiCallerPlugin(APICallMixin, SettingsMixin, InvenTreePlugin): ''' A small api call sample ''' - PLUGIN_NAME = "Sample API Caller" + NAME = "Sample API Caller" SETTINGS = { 'API_TOKEN': { @@ -496,9 +433,9 @@ class APICallMixin: def has_api_call(self): """Is the mixin ready to call external APIs?""" if not bool(self.API_URL_SETTING): - raise ValueError("API_URL_SETTING must be defined") + raise MixinNotImplementedError("API_URL_SETTING must be defined") if not bool(self.API_TOKEN_SETTING): - raise ValueError("API_TOKEN_SETTING must be defined") + raise MixinNotImplementedError("API_TOKEN_SETTING must be defined") return True @property diff --git a/InvenTree/plugin/test_integration.py b/InvenTree/plugin/base/integration/test_mixins.py similarity index 61% rename from InvenTree/plugin/test_integration.py rename to InvenTree/plugin/base/integration/test_mixins.py index 5b211e2d96..ef3f7062e3 100644 --- a/InvenTree/plugin/test_integration.py +++ b/InvenTree/plugin/base/integration/test_mixins.py @@ -1,17 +1,14 @@ -""" Unit tests for integration plugins """ +""" Unit tests for base mixins for plugins """ from django.test import TestCase from django.conf import settings from django.urls import include, re_path from django.contrib.auth import get_user_model -from datetime import datetime - -from plugin import IntegrationPluginBase +from plugin import InvenTreePlugin from plugin.mixins import AppMixin, SettingsMixin, UrlsMixin, NavigationMixin, APICallMixin from plugin.urls import PLUGIN_BASE - -from plugin.samples.integration.sample import SampleIntegrationPlugin +from plugin.helpers import MixinNotImplementedError class BaseMixinDefinition: @@ -30,11 +27,11 @@ class SettingsMixinTest(BaseMixinDefinition, TestCase): TEST_SETTINGS = {'SETTING1': {'default': '123', }} def setUp(self): - class SettingsCls(SettingsMixin, IntegrationPluginBase): + class SettingsCls(SettingsMixin, InvenTreePlugin): SETTINGS = self.TEST_SETTINGS self.mixin = SettingsCls() - class NoSettingsCls(SettingsMixin, IntegrationPluginBase): + class NoSettingsCls(SettingsMixin, InvenTreePlugin): pass self.mixin_nothing = NoSettingsCls() @@ -65,13 +62,13 @@ class UrlsMixinTest(BaseMixinDefinition, TestCase): MIXIN_ENABLE_CHECK = 'has_urls' def setUp(self): - class UrlsCls(UrlsMixin, IntegrationPluginBase): + class UrlsCls(UrlsMixin, InvenTreePlugin): def test(): return 'ccc' URLS = [re_path('testpath', test, name='test'), ] self.mixin = UrlsCls() - class NoUrlsCls(UrlsMixin, IntegrationPluginBase): + class NoUrlsCls(UrlsMixin, InvenTreePlugin): pass self.mixin_nothing = NoUrlsCls() @@ -104,7 +101,7 @@ class AppMixinTest(BaseMixinDefinition, TestCase): MIXIN_ENABLE_CHECK = 'has_app' def setUp(self): - class TestCls(AppMixin, IntegrationPluginBase): + class TestCls(AppMixin, InvenTreePlugin): pass self.mixin = TestCls() @@ -119,30 +116,32 @@ class NavigationMixinTest(BaseMixinDefinition, TestCase): MIXIN_ENABLE_CHECK = 'has_naviation' def setUp(self): - class NavigationCls(NavigationMixin, IntegrationPluginBase): + class NavigationCls(NavigationMixin, InvenTreePlugin): NAVIGATION = [ {'name': 'aa', 'link': 'plugin:test:test_view'}, ] NAVIGATION_TAB_NAME = 'abcd1' self.mixin = NavigationCls() - class NothingNavigationCls(NavigationMixin, IntegrationPluginBase): + class NothingNavigationCls(NavigationMixin, InvenTreePlugin): pass self.nothing_mixin = NothingNavigationCls() def test_function(self): # check right configuration self.assertEqual(self.mixin.navigation, [{'name': 'aa', 'link': 'plugin:test:test_view'}, ]) - # check wrong links fails - with self.assertRaises(NotImplementedError): - class NavigationCls(NavigationMixin, IntegrationPluginBase): - NAVIGATION = ['aa', 'aa'] - NavigationCls() # navigation name self.assertEqual(self.mixin.navigation_name, 'abcd1') self.assertEqual(self.nothing_mixin.navigation_name, '') + def test_fail(self): + # check wrong links fails + with self.assertRaises(NotImplementedError): + class NavigationCls(NavigationMixin, InvenTreePlugin): + NAVIGATION = ['aa', 'aa'] + NavigationCls() + class APICallMixinTest(BaseMixinDefinition, TestCase): MIXIN_HUMAN_NAME = 'API calls' @@ -150,8 +149,8 @@ class APICallMixinTest(BaseMixinDefinition, TestCase): MIXIN_ENABLE_CHECK = 'has_api_call' def setUp(self): - class MixinCls(APICallMixin, SettingsMixin, IntegrationPluginBase): - PLUGIN_NAME = "Sample API Caller" + class MixinCls(APICallMixin, SettingsMixin, InvenTreePlugin): + NAME = "Sample API Caller" SETTINGS = { 'API_TOKEN': { @@ -167,22 +166,23 @@ class APICallMixinTest(BaseMixinDefinition, TestCase): API_URL_SETTING = 'API_URL' API_TOKEN_SETTING = 'API_TOKEN' - def get_external_url(self): + def get_external_url(self, simple: bool = True): ''' returns data from the sample endpoint ''' - return self.api_call('api/users/2') + return self.api_call('api/users/2', simple_response=simple) self.mixin = MixinCls() - class WrongCLS(APICallMixin, IntegrationPluginBase): + class WrongCLS(APICallMixin, InvenTreePlugin): pass self.mixin_wrong = WrongCLS() - class WrongCLS2(APICallMixin, IntegrationPluginBase): + class WrongCLS2(APICallMixin, InvenTreePlugin): API_URL_SETTING = 'test' self.mixin_wrong2 = WrongCLS2() - def test_function(self): + def test_base_setup(self): + """Test that the base settings work""" # check init self.assertTrue(self.mixin.has_api_call) # api_url @@ -192,6 +192,8 @@ class APICallMixinTest(BaseMixinDefinition, TestCase): headers = self.mixin.api_headers self.assertEqual(headers, {'Bearer': '', 'Content-Type': 'application/json'}) + def test_args(self): + """Test that building up args work""" # api_build_url_args # 1 arg result = self.mixin.api_build_url_args({'a': 'b'}) @@ -203,88 +205,42 @@ class APICallMixinTest(BaseMixinDefinition, TestCase): result = self.mixin.api_build_url_args({'a': 'b', 'c': ['d', 'e', 'f', ]}) self.assertEqual(result, '?a=b&c=d,e,f') + def test_api_call(self): + """Test that api calls work""" # api_call result = self.mixin.get_external_url() self.assertTrue(result) self.assertIn('data', result,) + # api_call without json conversion + result = self.mixin.get_external_url(False) + self.assertTrue(result) + self.assertEqual(result.reason, 'OK') + + # api_call with full url + result = self.mixin.api_call('https://reqres.in/api/users/2', endpoint_is_url=True) + self.assertTrue(result) + + # api_call with post and data + result = self.mixin.api_call( + 'api/users/', + data={"name": "morpheus", "job": "leader"}, + method='POST' + ) + self.assertTrue(result) + self.assertEqual(result['name'], 'morpheus') + + # api_call with filter + result = self.mixin.api_call('api/users', url_args={'page': '2'}) + self.assertTrue(result) + self.assertEqual(result['page'], 2) + + def test_function_errors(self): + """Test function errors""" # wrongly defined plugins should not load - with self.assertRaises(ValueError): + with self.assertRaises(MixinNotImplementedError): self.mixin_wrong.has_api_call() # cover wrong token setting - with self.assertRaises(ValueError): - self.mixin_wrong.has_api_call() - - -class IntegrationPluginBaseTests(TestCase): - """ Tests for IntegrationPluginBase """ - - def setUp(self): - self.plugin = IntegrationPluginBase() - - class SimpeIntegrationPluginBase(IntegrationPluginBase): - PLUGIN_NAME = 'SimplePlugin' - - self.plugin_simple = SimpeIntegrationPluginBase() - - class NameIntegrationPluginBase(IntegrationPluginBase): - PLUGIN_NAME = 'Aplugin' - PLUGIN_SLUG = 'a' - PLUGIN_TITLE = 'a titel' - PUBLISH_DATE = "1111-11-11" - AUTHOR = 'AA BB' - DESCRIPTION = 'A description' - VERSION = '1.2.3a' - WEBSITE = 'http://aa.bb/cc' - LICENSE = 'MIT' - - self.plugin_name = NameIntegrationPluginBase() - self.plugin_sample = SampleIntegrationPlugin() - - def test_action_name(self): - """check the name definition possibilities""" - # plugin_name - self.assertEqual(self.plugin.plugin_name(), '') - self.assertEqual(self.plugin_simple.plugin_name(), 'SimplePlugin') - self.assertEqual(self.plugin_name.plugin_name(), 'Aplugin') - - # is_sampe - self.assertEqual(self.plugin.is_sample, False) - self.assertEqual(self.plugin_sample.is_sample, True) - - # slug - self.assertEqual(self.plugin.slug, '') - self.assertEqual(self.plugin_simple.slug, 'simpleplugin') - self.assertEqual(self.plugin_name.slug, 'a') - - # human_name - self.assertEqual(self.plugin.human_name, '') - self.assertEqual(self.plugin_simple.human_name, 'SimplePlugin') - self.assertEqual(self.plugin_name.human_name, 'a titel') - - # description - self.assertEqual(self.plugin.description, '') - self.assertEqual(self.plugin_simple.description, 'SimplePlugin') - self.assertEqual(self.plugin_name.description, 'A description') - - # author - self.assertEqual(self.plugin_name.author, 'AA BB') - - # pub_date - self.assertEqual(self.plugin_name.pub_date, datetime(1111, 11, 11, 0, 0)) - - # version - self.assertEqual(self.plugin.version, None) - self.assertEqual(self.plugin_simple.version, None) - self.assertEqual(self.plugin_name.version, '1.2.3a') - - # website - self.assertEqual(self.plugin.website, None) - self.assertEqual(self.plugin_simple.website, None) - self.assertEqual(self.plugin_name.website, 'http://aa.bb/cc') - - # license - self.assertEqual(self.plugin.license, None) - self.assertEqual(self.plugin_simple.license, None) - self.assertEqual(self.plugin_name.license, 'MIT') + with self.assertRaises(MixinNotImplementedError): + self.mixin_wrong2.has_api_call() diff --git a/InvenTree/plugin/base/label/__init__.py b/InvenTree/plugin/base/label/__init__.py new file mode 100644 index 0000000000..e69de29bb2 diff --git a/InvenTree/plugin/base/label/label.py b/InvenTree/plugin/base/label/label.py new file mode 100644 index 0000000000..7f29ba70f4 --- /dev/null +++ b/InvenTree/plugin/base/label/label.py @@ -0,0 +1,53 @@ +"""Functions to print a label to a mixin printer""" +import logging + +from django.utils.translation import gettext_lazy as _ + +from plugin.registry import registry +import common.notifications + + +logger = logging.getLogger('inventree') + + +def print_label(plugin_slug, label_image, label_instance=None, user=None): + """ + Print label with the provided plugin. + + This task is nominally handled by the background worker. + + If the printing fails (throws an exception) then the user is notified. + + Arguments: + plugin_slug: The unique slug (key) of the plugin + label_image: A PIL.Image image object to be printed + """ + + logger.info(f"Plugin '{plugin_slug}' is printing a label") + + plugin = registry.plugins.get(plugin_slug, None) + + if plugin is None: + logger.error(f"Could not find matching plugin for '{plugin_slug}'") + return + + try: + plugin.print_label(label_image, width=label_instance.width, height=label_instance.height) + except Exception as e: + # Plugin threw an error - notify the user who attempted to print + + ctx = { + 'name': _('Label printing failed'), + 'message': str(e), + } + + logger.error(f"Label printing failed: Sending notification to user '{user}'") + + # Throw an error against the plugin instance + common.notifications.trigger_notifaction( + plugin.plugin_config(), + 'label.printing_failed', + targets=[user], + context=ctx, + delivery_methods=[common.notifications.UIMessageNotification] + ) diff --git a/InvenTree/plugin/base/label/mixins.py b/InvenTree/plugin/base/label/mixins.py new file mode 100644 index 0000000000..b6c4d53f46 --- /dev/null +++ b/InvenTree/plugin/base/label/mixins.py @@ -0,0 +1,39 @@ +"""Plugin mixin classes for label plugins""" + +from plugin.helpers import MixinNotImplementedError + + +class LabelPrintingMixin: + """ + Mixin which enables direct printing of stock labels. + + Each plugin must provide a NAME attribute, which is used to uniquely identify the printer. + + The plugin must also implement the print_label() function + """ + + class MixinMeta: + """ + Meta options for this mixin + """ + MIXIN_NAME = 'Label printing' + + def __init__(self): # pragma: no cover + super().__init__() + self.add_mixin('labels', True, __class__) + + def print_label(self, label, **kwargs): + """ + Callback to print a single label + + Arguments: + label: A black-and-white pillow Image object + + kwargs: + length: The length of the label (in mm) + width: The width of the label (in mm) + + """ + + # Unimplemented (to be implemented by the particular plugin class) + MixinNotImplementedError('This Plugin must implement a `print_label` method') diff --git a/InvenTree/plugin/builtin/action/simpleactionplugin.py b/InvenTree/plugin/builtin/action/simpleactionplugin.py index 01a0829887..d2a321789d 100644 --- a/InvenTree/plugin/builtin/action/simpleactionplugin.py +++ b/InvenTree/plugin/builtin/action/simpleactionplugin.py @@ -1,15 +1,16 @@ # -*- coding: utf-8 -*- -"""sample implementation for ActionPlugin""" -from plugin.action import ActionPlugin +"""sample implementation for ActionMixin""" +from plugin import InvenTreePlugin +from plugin.mixins import ActionMixin -class SimpleActionPlugin(ActionPlugin): +class SimpleActionPlugin(ActionMixin, InvenTreePlugin): """ An EXTREMELY simple action plugin which demonstrates - the capability of the ActionPlugin class + the capability of the ActionMixin class """ - PLUGIN_NAME = "SimpleActionPlugin" + NAME = "SimpleActionPlugin" ACTION_NAME = "simple" def perform_action(self): diff --git a/InvenTree/plugin/builtin/action/test_samples_action.py b/InvenTree/plugin/builtin/action/test_simpleactionplugin.py similarity index 100% rename from InvenTree/plugin/builtin/action/test_samples_action.py rename to InvenTree/plugin/builtin/action/test_simpleactionplugin.py diff --git a/InvenTree/plugin/builtin/barcodes/inventree_barcode.py b/InvenTree/plugin/builtin/barcodes/inventree_barcode.py index 939bf12c08..f9f498ccd8 100644 --- a/InvenTree/plugin/builtin/barcodes/inventree_barcode.py +++ b/InvenTree/plugin/builtin/barcodes/inventree_barcode.py @@ -13,7 +13,7 @@ references model objects actually exist in the database. import json -from plugin import IntegrationPluginBase +from plugin import InvenTreePlugin from plugin.mixins import BarcodeMixin from stock.models import StockItem, StockLocation @@ -22,9 +22,9 @@ from part.models import Part from rest_framework.exceptions import ValidationError -class InvenTreeBarcodePlugin(BarcodeMixin, IntegrationPluginBase): +class InvenTreeBarcodePlugin(BarcodeMixin, InvenTreePlugin): - PLUGIN_NAME = "InvenTreeBarcode" + NAME = "InvenTreeBarcode" def validate(self): """ diff --git a/InvenTree/plugin/builtin/integration/core_notifications.py b/InvenTree/plugin/builtin/integration/core_notifications.py index a26c69e877..67b61eea3b 100644 --- a/InvenTree/plugin/builtin/integration/core_notifications.py +++ b/InvenTree/plugin/builtin/integration/core_notifications.py @@ -5,7 +5,7 @@ from django.utils.translation import ugettext_lazy as _ from allauth.account.models import EmailAddress -from plugin import IntegrationPluginBase +from plugin import InvenTreePlugin from plugin.mixins import BulkNotificationMethod, SettingsMixin import InvenTree.tasks @@ -15,12 +15,12 @@ class PlgMixin: return CoreNotificationsPlugin -class CoreNotificationsPlugin(SettingsMixin, IntegrationPluginBase): +class CoreNotificationsPlugin(SettingsMixin, InvenTreePlugin): """ Core notification methods for InvenTree """ - PLUGIN_NAME = "CoreNotificationsPlugin" + NAME = "CoreNotificationsPlugin" AUTHOR = _('InvenTree contributors') DESCRIPTION = _('Integrated outgoing notificaton methods') diff --git a/InvenTree/plugin/events.py b/InvenTree/plugin/events.py index 043f3b97a3..193a1805fe 100644 --- a/InvenTree/plugin/events.py +++ b/InvenTree/plugin/events.py @@ -1,239 +1,9 @@ """ -Functions for triggering and responding to server side events +Import helper for events """ -# -*- coding: utf-8 -*- -from __future__ import unicode_literals +from plugin.base.event.events import trigger_event -import logging - -from django.utils.translation import gettext_lazy as _ - -from django.conf import settings -from django.db import transaction -from django.db.models.signals import post_save, post_delete -from django.dispatch.dispatcher import receiver - -from common.models import InvenTreeSetting -import common.notifications - -from InvenTree.ready import canAppAccessDatabase, isImportingData -from InvenTree.tasks import offload_task - -from plugin.registry import registry - - -logger = logging.getLogger('inventree') - - -def trigger_event(event, *args, **kwargs): - """ - Trigger an event with optional arguments. - - This event will be stored in the database, - and the worker will respond to it later on. - """ - - if not settings.PLUGINS_ENABLED: - # Do nothing if plugins are not enabled - return - - if not canAppAccessDatabase(): - logger.debug(f"Ignoring triggered event '{event}' - database not ready") - return - - logger.debug(f"Event triggered: '{event}'") - - offload_task( - 'plugin.events.register_event', - event, - *args, - **kwargs - ) - - -def register_event(event, *args, **kwargs): - """ - Register the event with any interested plugins. - - Note: This function is processed by the background worker, - as it performs multiple database access operations. - """ - - logger.debug(f"Registering triggered event: '{event}'") - - # Determine if there are any plugins which are interested in responding - if settings.PLUGIN_TESTING or InvenTreeSetting.get_setting('ENABLE_PLUGINS_EVENTS'): - - with transaction.atomic(): - - for slug, plugin in registry.plugins.items(): - - if plugin.mixin_enabled('events'): - - config = plugin.plugin_config() - - if config and config.active: - - logger.debug(f"Registering callback for plugin '{slug}'") - - # Offload a separate task for each plugin - offload_task( - 'plugin.events.process_event', - slug, - event, - *args, - **kwargs - ) - - -def process_event(plugin_slug, event, *args, **kwargs): - """ - Respond to a triggered event. - - This function is run by the background worker process. - - This function may queue multiple functions to be handled by the background worker. - """ - - logger.info(f"Plugin '{plugin_slug}' is processing triggered event '{event}'") - - plugin = registry.plugins.get(plugin_slug, None) - - if plugin is None: - logger.error(f"Could not find matching plugin for '{plugin_slug}'") - return - - plugin.process_event(event, *args, **kwargs) - - -def allow_table_event(table_name): - """ - Determine if an automatic event should be fired for a given table. - We *do not* want events to be fired for some tables! - """ - - if isImportingData(): - # Prevent table events during the data import process - return False - - table_name = table_name.lower().strip() - - # Ignore any tables which start with these prefixes - ignore_prefixes = [ - 'account_', - 'auth_', - 'authtoken_', - 'django_', - 'error_', - 'exchange_', - 'otp_', - 'plugin_', - 'socialaccount_', - 'user_', - 'users_', - ] - - if any([table_name.startswith(prefix) for prefix in ignore_prefixes]): - return False - - ignore_tables = [ - 'common_notificationentry', - 'common_webhookendpoint', - 'common_webhookmessage', - ] - - if table_name in ignore_tables: - return False - - return True - - -@receiver(post_save) -def after_save(sender, instance, created, **kwargs): - """ - Trigger an event whenever a database entry is saved - """ - - table = sender.objects.model._meta.db_table - - instance_id = getattr(instance, 'id', None) - - if instance_id is None: - return - - if not allow_table_event(table): - return - - if created: - trigger_event( - f'{table}.created', - id=instance.id, - model=sender.__name__, - ) - else: - trigger_event( - f'{table}.saved', - id=instance.id, - model=sender.__name__, - ) - - -@receiver(post_delete) -def after_delete(sender, instance, **kwargs): - """ - Trigger an event whenever a database entry is deleted - """ - - table = sender.objects.model._meta.db_table - - if not allow_table_event(table): - return - - trigger_event( - f'{table}.deleted', - model=sender.__name__, - ) - - -def print_label(plugin_slug, label_image, label_instance=None, user=None): - """ - Print label with the provided plugin. - - This task is nominally handled by the background worker. - - If the printing fails (throws an exception) then the user is notified. - - Arguments: - plugin_slug: The unique slug (key) of the plugin - label_image: A PIL.Image image object to be printed - """ - - logger.info(f"Plugin '{plugin_slug}' is printing a label") - - plugin = registry.plugins.get(plugin_slug, None) - - if plugin is None: - logger.error(f"Could not find matching plugin for '{plugin_slug}'") - return - - try: - plugin.print_label(label_image, width=label_instance.width, height=label_instance.height) - except Exception as e: - # Plugin threw an error - notify the user who attempted to print - - ctx = { - 'name': _('Label printing failed'), - 'message': str(e), - } - - logger.error(f"Label printing failed: Sending notification to user '{user}'") - - # Throw an error against the plugin instance - common.notifications.trigger_notifaction( - plugin.plugin_config(), - 'label.printing_failed', - targets=[user], - context=ctx, - delivery_methods=[common.notifications.UIMessageNotification] - ) +__all__ = [ + trigger_event, +] diff --git a/InvenTree/plugin/helpers.py b/InvenTree/plugin/helpers.py index f1753b1b45..1217fa4d47 100644 --- a/InvenTree/plugin/helpers.py +++ b/InvenTree/plugin/helpers.py @@ -8,12 +8,17 @@ import sysconfig import traceback import inspect import pkgutil +import logging +from django import template from django.conf import settings from django.core.exceptions import AppRegistryNotReady from django.db.utils import IntegrityError +logger = logging.getLogger('inventree') + + # region logging / errors class IntegrationPluginError(Exception): """ @@ -200,7 +205,7 @@ def get_plugins(pkg, baseclass): Return a list of all modules under a given package. - Modules must be a subclass of the provided 'baseclass' - - Modules must have a non-empty PLUGIN_NAME parameter + - Modules must have a non-empty NAME parameter """ plugins = [] @@ -212,8 +217,32 @@ def get_plugins(pkg, baseclass): # Iterate through each class in the module for item in get_classes(mod): plugin = item[1] - if issubclass(plugin, baseclass) and plugin.PLUGIN_NAME: + if issubclass(plugin, baseclass) and plugin.NAME: plugins.append(plugin) return plugins # endregion + + +# region templates +def render_template(plugin, template_file, context=None): + """ + Locate and render a template file, available in the global template context. + """ + + try: + tmp = template.loader.get_template(template_file) + except template.TemplateDoesNotExist: + logger.error(f"Plugin {plugin.slug} could not locate template '{template_file}'") + + return f""" +