mirror of
https://github.com/inventree/InvenTree.git
synced 2025-05-03 05:48:47 +00:00
* Cleanup plugin settings page - Template adjustments * Activate plugin directly via API * Update plugin activate endpoint - Allow plugin to be deactivated also - Default value = True if not provided * Update front-end / js - Allow same JS method to either enable or disable a plugin * Hide info for plugins which are not active * remove duplicated column * Tweak serializer docstring * Fix typo * Add extra data to plugin serializer - is_builtin - is_sample * Some backend cleanup - Don't stringify null values - Don't replace None with "Unavailable" * front-end table for rendering plugins * Change default sorting - Show active plugins first * Fix button callback * Remove old template * JS linting * More linting
277 lines
8.2 KiB
Python
277 lines
8.2 KiB
Python
"""API for the plugin app."""
|
|
|
|
from django.urls import include, path, re_path
|
|
|
|
from django_filters.rest_framework import DjangoFilterBackend
|
|
from rest_framework import permissions, status
|
|
from rest_framework.exceptions import NotFound
|
|
from rest_framework.response import Response
|
|
|
|
import plugin.serializers as PluginSerializers
|
|
from common.api import GlobalSettingsPermissions
|
|
from InvenTree.filters import SEARCH_ORDER_FILTER
|
|
from InvenTree.mixins import (CreateAPI, ListAPI, RetrieveUpdateAPI,
|
|
RetrieveUpdateDestroyAPI, UpdateAPI)
|
|
from InvenTree.permissions import IsSuperuser
|
|
from plugin.base.action.api import ActionPluginView
|
|
from plugin.base.barcodes.api import barcode_api_urls
|
|
from plugin.base.locate.api import LocatePluginView
|
|
from plugin.models import PluginConfig, PluginSetting
|
|
from plugin.plugin import InvenTreePlugin
|
|
|
|
|
|
class PluginList(ListAPI):
|
|
"""API endpoint for list of PluginConfig objects.
|
|
|
|
- GET: Return a list of all PluginConfig objects
|
|
"""
|
|
|
|
# Allow any logged in user to read this endpoint
|
|
# This is necessary to allow certain functionality,
|
|
# e.g. determining which label printing plugins are available
|
|
permission_classes = [permissions.IsAuthenticated]
|
|
|
|
serializer_class = PluginSerializers.PluginConfigSerializer
|
|
queryset = PluginConfig.objects.all()
|
|
|
|
def filter_queryset(self, queryset):
|
|
"""Filter for API requests.
|
|
|
|
Filter by mixin with the `mixin` flag
|
|
"""
|
|
queryset = super().filter_queryset(queryset)
|
|
|
|
params = self.request.query_params
|
|
|
|
# Filter plugins which support a given mixin
|
|
mixin = params.get('mixin', None)
|
|
|
|
if mixin:
|
|
matches = []
|
|
|
|
for result in queryset:
|
|
if mixin in result.mixins().keys():
|
|
matches.append(result.pk)
|
|
|
|
queryset = queryset.filter(pk__in=matches)
|
|
|
|
return queryset
|
|
|
|
filter_backends = SEARCH_ORDER_FILTER
|
|
|
|
filterset_fields = [
|
|
'active',
|
|
]
|
|
|
|
ordering_fields = [
|
|
'key',
|
|
'name',
|
|
'active',
|
|
]
|
|
|
|
ordering = [
|
|
'-active',
|
|
'name',
|
|
'key',
|
|
]
|
|
|
|
search_fields = [
|
|
'key',
|
|
'name',
|
|
]
|
|
|
|
|
|
class PluginDetail(RetrieveUpdateDestroyAPI):
|
|
"""API detail endpoint for PluginConfig object.
|
|
|
|
get:
|
|
Return a single PluginConfig object
|
|
|
|
post:
|
|
Update a PluginConfig
|
|
|
|
delete:
|
|
Remove a PluginConfig
|
|
"""
|
|
|
|
queryset = PluginConfig.objects.all()
|
|
serializer_class = PluginSerializers.PluginConfigSerializer
|
|
|
|
|
|
class PluginInstall(CreateAPI):
|
|
"""Endpoint for installing a new plugin."""
|
|
|
|
queryset = PluginConfig.objects.none()
|
|
serializer_class = PluginSerializers.PluginConfigInstallSerializer
|
|
|
|
def create(self, request, *args, **kwargs):
|
|
"""Install a plugin via the API"""
|
|
# Clean up input data
|
|
data = self.clean_data(request.data)
|
|
|
|
serializer = self.get_serializer(data=data)
|
|
serializer.is_valid(raise_exception=True)
|
|
result = self.perform_create(serializer)
|
|
result['input'] = serializer.data
|
|
headers = self.get_success_headers(serializer.data)
|
|
return Response(result, status=status.HTTP_201_CREATED, headers=headers)
|
|
|
|
def perform_create(self, serializer):
|
|
"""Saving the serializer instance performs plugin installation"""
|
|
return serializer.save()
|
|
|
|
|
|
class PluginActivate(UpdateAPI):
|
|
"""Endpoint for activating a plugin.
|
|
|
|
- PATCH: Activate a plugin
|
|
|
|
Pass a boolean value for the 'active' field.
|
|
If not provided, it is assumed to be True,
|
|
and the plugin will be activated.
|
|
"""
|
|
|
|
queryset = PluginConfig.objects.all()
|
|
serializer_class = PluginSerializers.PluginActivateSerializer
|
|
permission_classes = [IsSuperuser, ]
|
|
|
|
def get_object(self):
|
|
"""Returns the object for the view."""
|
|
if self.request.data.get('pk', None):
|
|
return self.queryset.get(pk=self.request.data.get('pk'))
|
|
return super().get_object()
|
|
|
|
def perform_update(self, serializer):
|
|
"""Activate the plugin."""
|
|
|
|
serializer.save()
|
|
|
|
|
|
class PluginSettingList(ListAPI):
|
|
"""List endpoint for all plugin related settings.
|
|
|
|
- read only
|
|
- only accessible by staff users
|
|
"""
|
|
|
|
queryset = PluginSetting.objects.all()
|
|
serializer_class = PluginSerializers.PluginSettingSerializer
|
|
|
|
permission_classes = [
|
|
GlobalSettingsPermissions,
|
|
]
|
|
|
|
filter_backends = [
|
|
DjangoFilterBackend,
|
|
]
|
|
|
|
filterset_fields = [
|
|
'plugin__active',
|
|
'plugin__key',
|
|
]
|
|
|
|
|
|
def check_plugin(plugin_slug: str, plugin_pk: int) -> InvenTreePlugin:
|
|
"""Check that a plugin for the provided slug exists and get the config.
|
|
|
|
Args:
|
|
plugin_slug (str): Slug for plugin.
|
|
plugin_pk (int): Primary key for plugin.
|
|
|
|
Raises:
|
|
NotFound: If plugin is not installed
|
|
NotFound: If plugin is not correctly registered
|
|
NotFound: If plugin is not active
|
|
|
|
Returns:
|
|
InvenTreePlugin: The config object for the provided plugin.
|
|
"""
|
|
# Make sure that a plugin reference is specified
|
|
if plugin_slug is None and plugin_pk is None:
|
|
raise NotFound(detail="Plugin not specified")
|
|
|
|
# Define filter
|
|
filter = {}
|
|
if plugin_slug:
|
|
filter['key'] = plugin_slug
|
|
elif plugin_pk:
|
|
filter['pk'] = plugin_pk
|
|
ref = plugin_slug or plugin_pk
|
|
|
|
# Check that the 'plugin' specified is valid
|
|
try:
|
|
plugin_cgf = PluginConfig.objects.get(**filter)
|
|
except PluginConfig.DoesNotExist:
|
|
raise NotFound(detail=f"Plugin '{ref}' not installed")
|
|
|
|
if plugin_cgf is None:
|
|
# This only occurs if the plugin mechanism broke
|
|
raise NotFound(detail=f"Plugin '{ref}' not found") # pragma: no cover
|
|
|
|
# Check that the plugin is activated
|
|
if not plugin_cgf.active:
|
|
raise NotFound(detail=f"Plugin '{ref}' is not active")
|
|
|
|
return plugin_cgf.plugin
|
|
|
|
|
|
class PluginSettingDetail(RetrieveUpdateAPI):
|
|
"""Detail endpoint for a plugin-specific setting.
|
|
|
|
Note that these cannot be created or deleted via the API
|
|
"""
|
|
|
|
queryset = PluginSetting.objects.all()
|
|
serializer_class = PluginSerializers.PluginSettingSerializer
|
|
|
|
def get_object(self):
|
|
"""Lookup the plugin setting object, based on the URL.
|
|
|
|
The URL provides the 'slug' of the plugin, and the 'key' of the setting.
|
|
Both the 'slug' and 'key' must be valid, else a 404 error is raised
|
|
"""
|
|
key = self.kwargs['key']
|
|
|
|
# Look up plugin
|
|
plugin = check_plugin(plugin_slug=self.kwargs.get('plugin'), plugin_pk=self.kwargs.get('pk'))
|
|
|
|
settings = getattr(plugin, 'settings', {})
|
|
|
|
if key not in settings:
|
|
raise NotFound(detail=f"Plugin '{plugin.slug}' has no setting matching '{key}'")
|
|
|
|
return PluginSetting.get_setting_object(key, plugin=plugin.plugin_config())
|
|
|
|
# Staff permission required
|
|
permission_classes = [
|
|
GlobalSettingsPermissions,
|
|
]
|
|
|
|
|
|
plugin_api_urls = [
|
|
re_path(r'^action/', ActionPluginView.as_view(), name='api-action-plugin'),
|
|
re_path(r'^barcode/', include(barcode_api_urls)),
|
|
re_path(r'^locate/', LocatePluginView.as_view(), name='api-locate-plugin'),
|
|
re_path(r'^plugins/', include([
|
|
# Plugin settings URLs
|
|
re_path(r'^settings/', include([
|
|
re_path(r'^(?P<plugin>[-\w]+)/(?P<key>\w+)/', PluginSettingDetail.as_view(), name='api-plugin-setting-detail'), # Used for admin interface
|
|
re_path(r'^.*$', PluginSettingList.as_view(), name='api-plugin-setting-list'),
|
|
])),
|
|
|
|
# Detail views for a single PluginConfig item
|
|
path(r'<int:pk>/', include([
|
|
re_path(r'^settings/(?P<key>\w+)/', PluginSettingDetail.as_view(), name='api-plugin-setting-detail-pk'),
|
|
re_path(r'^activate/', PluginActivate.as_view(), name='api-plugin-detail-activate'),
|
|
re_path(r'^.*$', PluginDetail.as_view(), name='api-plugin-detail'),
|
|
])),
|
|
|
|
# Plugin management
|
|
re_path(r'^install/', PluginInstall.as_view(), name='api-plugin-install'),
|
|
re_path(r'^activate/', PluginActivate.as_view(), name='api-plugin-activate'),
|
|
|
|
# Anything else
|
|
re_path(r'^.*$', PluginList.as_view(), name='api-plugin-list'),
|
|
]))
|
|
]
|