Skip to content

Commit

Permalink
Plugin missing fix (inventree#5653)
Browse files Browse the repository at this point in the history
* Add 'is_installed' property to PluginConfig model

- Allow us to show users which plugins are "outdated"

* Cleanup plugin table

- Display clearly if a plugin is no longer installed

* Fixes for plugin table

* Revert change due to failing CI
  • Loading branch information
SchrodingersGat authored Oct 3, 2023
1 parent 88314da commit 78905a4
Show file tree
Hide file tree
Showing 5 changed files with 50 additions and 24 deletions.
12 changes: 12 additions & 0 deletions InvenTree/plugin/api.py
Original file line number Diff line number Diff line change
Expand Up @@ -83,6 +83,18 @@ def filter_queryset(self, queryset):

queryset = queryset.filter(pk__in=matches)

# Filter queryset by 'installed' flag
if 'installed' in params:
installed = str2bool(params['installed'])

matches = []

for result in queryset:
if result.is_installed() == installed:
matches.append(result.pk)

queryset = queryset.filter(pk__in=matches)

return queryset

filter_backends = SEARCH_ORDER_FILTER
Expand Down
17 changes: 13 additions & 4 deletions InvenTree/plugin/models.py
Original file line number Diff line number Diff line change
Expand Up @@ -104,9 +104,9 @@ def get_plugin_meta(name):
self.plugin: InvenTreePlugin = plugin

def __getstate__(self):
"""Customize pickeling behaviour."""
"""Customize pickling behavior."""
state = super().__getstate__()
state.pop("plugin", None) # plugin cannot be pickelt in some circumstances when used with drf views, remove it (#5408)
state.pop("plugin", None) # plugin cannot be pickled in some circumstances when used with drf views, remove it (#5408)
return state

def save(self, force_insert=False, force_update=False, *args, **kwargs):
Expand All @@ -120,14 +120,23 @@ def save(self, force_insert=False, force_update=False, *args, **kwargs):
self.active = True

if not reload:
if (self.active is False and self.__org_active is True) or \
(self.active is True and self.__org_active is False):
if self.active != self.__org_active:
if settings.PLUGIN_TESTING:
warnings.warn('A reload was triggered', stacklevel=2)
registry.reload_plugins()

return ret

@admin.display(boolean=True, description=_('Installed'))
def is_installed(self) -> bool:
"""Simple check to determine if this plugin is installed.
A plugin might not be installed if it has been removed from the system,
but the PluginConfig associated with it still exists.
"""

return self.plugin is not None

@admin.display(boolean=True, description=_('Sample plugin'))
def is_sample(self) -> bool:
"""Is this plugin a sample app?"""
Expand Down
2 changes: 2 additions & 0 deletions InvenTree/plugin/serializers.py
Original file line number Diff line number Diff line change
Expand Up @@ -56,12 +56,14 @@ class Meta:
'mixins',
'is_builtin',
'is_sample',
'is_installed',
]

read_only_fields = [
'key',
'is_builtin',
'is_sample',
'is_installed',
]

meta = serializers.DictField(read_only=True)
Expand Down
39 changes: 19 additions & 20 deletions InvenTree/templates/js/translated/plugin.js
Original file line number Diff line number Diff line change
Expand Up @@ -45,34 +45,24 @@ function loadPluginTable(table, options={}) {
return '{% trans "No plugins found" %}';
},
columns: [
{
field: 'active',
title: '',
sortable: true,
formatter: function(value, row) {
if (row.active) {
return `<span class='fa fa-check-circle icon-green' title='{% trans "This plugin is active" %}'></span>`;
} else {
return `<span class='fa fa-times-circle icon-red' title ='{% trans "This plugin is not active" %}'></span>`;
}
}
},
{
field: 'name',
title: '{% trans "Plugin Description" %}',
title: '{% trans "Plugin" %}',
sortable: true,
switchable: false,
formatter: function(value, row) {
let html = '';

if (row.active) {
html += `<strong>${value}</strong>`;
if (row.meta && row.meta.description) {
html += ` - <small>${row.meta.description}</small>`;
}
if (!row.is_installed) {
html += `<span class='fa fa-question-circle' title='{% trans "This plugin is no longer installed" %}'></span>`;
} else if (row.active) {
html += `<span class='fa fa-check-circle icon-green' title='{% trans "This plugin is active" %}'></span>`;
} else {
html += `<em>${value}</em>`;
html += `<span class='fa fa-times-circle icon-red' title ='{% trans "This plugin is installed but not active" %}'></span>`;
}

html += `&nbsp;<span>${value}</span>`;

if (row.is_builtin) {
html += `<span class='badge bg-success rounded-pill badge-right'>{% trans "Builtin" %}</span>`;
}
Expand All @@ -84,6 +74,12 @@ function loadPluginTable(table, options={}) {
return html;
}
},
{
field: 'meta.description',
title: '{% trans "Description" %}',
sortable: false,
switchable: true,
},
{
field: 'meta.version',
title: '{% trans "Version" %}',
Expand All @@ -104,15 +100,18 @@ function loadPluginTable(table, options={}) {
{
field: 'meta.author',
title: '{% trans "Author" %}',
sortable: false,
},
{
field: 'actions',
title: '',
switchable: false,
sortable: false,
formatter: function(value, row) {
let buttons = '';

// Check if custom plugins are enabled for this instance
if (options.custom && !row.is_builtin) {
if (options.custom && !row.is_builtin && row.is_installed) {
if (row.active) {
buttons += makeIconButton('fa-stop-circle icon-red', 'btn-plugin-disable', row.pk, '{% trans "Disable Plugin" %}');
} else {
Expand Down
4 changes: 4 additions & 0 deletions InvenTree/templates/js/translated/table_filters.js
Original file line number Diff line number Diff line change
Expand Up @@ -471,6 +471,10 @@ function getPluginTableFilters() {
type: 'bool',
title: '{% trans "Sample" %}',
},
installed: {
type: 'bool',
title: '{% trans "Installed" %}'
},
};
}

Expand Down

0 comments on commit 78905a4

Please sign in to comment.