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'{obj.sftpuser}
'
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 = '+'
websites.append('%s%s' % (add_url, plus, gettext("Add website")))
return '
'.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("Not available")
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)