from django import forms from django.contrib import admin from django.urls import reverse from django.utils.encoding import force_str from django.utils.safestring import mark_safe from django.utils.translation import gettext, gettext_lazy as _ from django.shortcuts import resolve_url from django.contrib.admin.templatetags.admin_urls import admin_urlname from orchestra.admin import ExtendedModelAdmin from orchestra.admin.utils import admin_link, get_modeladmin from orchestra.contrib.accounts.actions import list_accounts from orchestra.contrib.accounts.admin import AccountAdminMixin from orchestra.contrib.systemusers.models import WebappUsers from orchestra.forms.widgets import DynamicHelpTextSelect from orchestra.plugins.admin import SelectPluginAdminMixin, display_plugin_field from orchestra.utils.html import get_on_site_link from orchestra.settings import NEW_SERVERS from .filters import HasWebsiteListFilter, DetailListFilter from .models import WebApp, WebAppOption from .options import AppOption from .types import AppType from django.db.models.signals import post_save from django.dispatch import receiver class WebAppOptionForm(forms.ModelForm): class Meta: model = WebAppOption fields = '__all__' # en las app de moodle el public-root sera siempre moodle def clean(self): data = self.cleaned_data webapp = self.cleaned_data.get("webapp") if webapp.type == 'moodle-php': if self.cleaned_data.get("name") == 'public-root': data['value'] = 'moodle' data['DELETE'] = False return data class WebAppOptionInline(admin.TabularInline): model = WebAppOption extra = 1 form = WebAppOptionForm OPTIONS_HELP_TEXT = { op.name: force_str(op.help_text) for op in AppOption.get_plugins() } class Media: css = { 'all': ('orchestra/css/hide-inline-id.css',) } def formfield_for_dbfield(self, db_field, **kwargs): if db_field.name == 'value': kwargs['widget'] = forms.TextInput(attrs={'size':'100'}) if db_field.name == 'name': if self.parent_object: plugin = self.parent_object.type_class else: request = kwargs['request'] webapp_modeladmin = get_modeladmin(self.parent_model) plugin_value = webapp_modeladmin.get_plugin_value(request) plugin = AppType.get(plugin_value) kwargs['choices'] = plugin.get_group_options_choices() # Help text based on select widget target = 'this.id.replace("name", "value")' kwargs['widget'] = DynamicHelpTextSelect(target, self.OPTIONS_HELP_TEXT) return super(WebAppOptionInline, self).formfield_for_dbfield(db_field, **kwargs) class WebAppAdmin(SelectPluginAdminMixin, AccountAdminMixin, ExtendedModelAdmin): list_display = ( 'name', 'display_type', 'display_detail', 'display_websites', 'account_link', 'target_server', ) list_filter = ('type', HasWebsiteListFilter, DetailListFilter, 'target_server') inlines = [WebAppOptionInline] readonly_fields = ('account_link',) change_readonly_fields = ('name', 'type', 'display_websites', 'display_sftpuser', 'target_server',) search_fields = ('name', 'account__username', 'data', 'website__domains__name') list_prefetch_related = ('content_set__website', 'content_set__website__domains') plugin = AppType plugin_field = 'type' plugin_title = _("Web application type") actions = (list_accounts,) display_type = display_plugin_field('type') def display_sftpuser(self, obj): salida = "" if obj.sftpuser is None: salida = None else: url = resolve_url(admin_urlname(WebappUsers._meta, 'change'), obj.sftpuser.id) salida += f'<a href="{url}">{obj.sftpuser}</a> <br />' return mark_safe(salida) display_sftpuser.short_description = _("user sftp") @mark_safe def display_websites(self, webapp): websites = [] for content in webapp.content_set.all(): site_url = content.get_absolute_url() site_link = get_on_site_link(site_url) website = content.website #name = "%s on %s %s" % (website.name, content.path, site_link) name = "%s on %s" % (website.name, content.path) link = admin_link(display=name)(website) websites.append(link) if not websites: add_url = reverse('admin:websites_website_add') add_url += '?account=%s' % webapp.account_id plus = '<strong style="color:green; font-size:12px">+</strong>' websites.append('<a href="%s">%s%s</a>' % (add_url, plus, gettext("Add website"))) return '<br>'.join(websites) display_websites.short_description = _("web sites") def display_detail(self, webapp): try: return webapp.type_instance.get_detail() except KeyError: return mark_safe("<span style='color:red;'>Not available</span>") display_detail.short_description = _("detail") def save_model(self, request, obj, form, change): if not change: user = form.cleaned_data.get('username') server = form.cleaned_data.get('target_server') if user and server.name in NEW_SERVERS: user = WebappUsers( username=form.cleaned_data['username'], account_id=obj.account.pk, target_server=form.cleaned_data['target_server'], home=form.cleaned_data['name'] ) user.set_password(form.cleaned_data["password1"]) user.save() obj.sftpuser = user super(WebAppAdmin, self).save_model(request, obj, form, change) # fuerza a las app Moodle a crear public-root moodle def response_add(self, request, obj, post_url_continue=None): if obj.type == 'moodle-php': mywebapp = WebApp.objects.get(id=obj.id) WebAppOption.objects.update_or_create( webapp=mywebapp, name='public-root', defaults={'value':'moodle'} ) return super().response_add(request, obj, post_url_continue) admin.site.register(WebApp, WebAppAdmin)