Random fixes
This commit is contained in:
parent
11f4a7de60
commit
6715f2ee2b
3
TODO.md
3
TODO.md
|
@ -384,3 +384,6 @@ Case
|
||||||
# Deprecate orchestra start/stop/restart services management commands?
|
# Deprecate orchestra start/stop/restart services management commands?
|
||||||
|
|
||||||
# Enable/disable ignore period orders list filter
|
# Enable/disable ignore period orders list filter
|
||||||
|
|
||||||
|
|
||||||
|
# Modsecurity rules template by cms (wordpress, joomla, dokuwiki (973337 973338 973347 958057), ...
|
||||||
|
|
|
@ -37,13 +37,13 @@ view_bill.url_name = 'view'
|
||||||
|
|
||||||
@transaction.atomic
|
@transaction.atomic
|
||||||
def close_bills(modeladmin, request, queryset, action='close_bills'):
|
def close_bills(modeladmin, request, queryset, action='close_bills'):
|
||||||
queryset = queryset.filter(is_open=True)
|
# Validate bills
|
||||||
if not queryset:
|
|
||||||
messages.warning(request, _("Selected bills should be in open state"))
|
|
||||||
return
|
|
||||||
for bill in queryset:
|
for bill in queryset:
|
||||||
if not validate_contact(request, bill):
|
if not validate_contact(request, bill):
|
||||||
return
|
return False
|
||||||
|
if not bill.is_open:
|
||||||
|
messages.warning(request, _("Selected bills should be in open state"))
|
||||||
|
return False
|
||||||
SelectSourceFormSet = adminmodelformset_factory(modeladmin, SelectSourceForm, extra=0)
|
SelectSourceFormSet = adminmodelformset_factory(modeladmin, SelectSourceForm, extra=0)
|
||||||
formset = SelectSourceFormSet(queryset=queryset)
|
formset = SelectSourceFormSet(queryset=queryset)
|
||||||
if request.POST.get('post') == 'generic_confirmation':
|
if request.POST.get('post') == 'generic_confirmation':
|
||||||
|
@ -143,9 +143,13 @@ download_bills.url_name = 'download'
|
||||||
|
|
||||||
def close_send_download_bills(modeladmin, request, queryset):
|
def close_send_download_bills(modeladmin, request, queryset):
|
||||||
response = close_bills(modeladmin, request, queryset, action='close_send_download_bills')
|
response = close_bills(modeladmin, request, queryset, action='close_send_download_bills')
|
||||||
|
if response is False:
|
||||||
|
# Not a valid contact or closed bill
|
||||||
|
return
|
||||||
if request.POST.get('post') == 'generic_confirmation':
|
if request.POST.get('post') == 'generic_confirmation':
|
||||||
response = send_bills_action(modeladmin, request, queryset)
|
response = send_bills_action(modeladmin, request, queryset)
|
||||||
if response is False:
|
if response is False:
|
||||||
|
# Not a valid contact
|
||||||
return
|
return
|
||||||
return download_bills(modeladmin, request, queryset)
|
return download_bills(modeladmin, request, queryset)
|
||||||
return response
|
return response
|
||||||
|
|
|
@ -80,15 +80,20 @@ class MessageAdmin(admin.ModelAdmin):
|
||||||
part = email.message_from_string(instance.content)
|
part = email.message_from_string(instance.content)
|
||||||
payload = part.get_payload()
|
payload = part.get_payload()
|
||||||
if isinstance(payload, list):
|
if isinstance(payload, list):
|
||||||
for part in payload:
|
for cpart in payload:
|
||||||
payload = part.get_payload()
|
cpayload = cpart.get_payload()
|
||||||
if part.get_content_type() == 'text/html':
|
if cpart.get_content_type().startswith('text/'):
|
||||||
|
part = cpart
|
||||||
|
payload = cpayload
|
||||||
|
if cpart.get_content_type() == 'text/html':
|
||||||
payload = '<div style="padding-left:110px">%s</div>' % payload
|
payload = '<div style="padding-left:110px">%s</div>' % payload
|
||||||
# prioritize HTML
|
# prioritize HTML
|
||||||
break
|
break
|
||||||
if part.get('Content-Transfer-Encoding') == 'base64':
|
if part.get('Content-Transfer-Encoding') == 'base64':
|
||||||
payload = base64.b64decode(payload)
|
payload = base64.b64decode(payload)
|
||||||
payload = payload.decode(part.get_charsets()[0])
|
charset = part.get_charsets()[0]
|
||||||
|
if charset:
|
||||||
|
payload = payload.decode(charset)
|
||||||
if part.get_content_type() == 'text/plain':
|
if part.get_content_type() == 'text/plain':
|
||||||
payload = payload.replace('\n', '<br>').replace(' ', ' ')
|
payload = payload.replace('\n', '<br>').replace(' ', ' ')
|
||||||
return payload
|
return payload
|
||||||
|
|
|
@ -39,11 +39,11 @@ class MetricStorageInline(admin.TabularInline):
|
||||||
if change_view:
|
if change_view:
|
||||||
qs = qs.order_by('-id')
|
qs = qs.order_by('-id')
|
||||||
try:
|
try:
|
||||||
tenth_id = qs.values_list('id', flat=True)[9]
|
tenth_id = qs.filter(order_id=self.parent_object.pk).values_list('id', flat=True)[9]
|
||||||
except IndexError:
|
except IndexError:
|
||||||
pass
|
pass
|
||||||
else:
|
else:
|
||||||
return qs.filter(pk__lte=tenth_id)
|
return qs.filter(pk__gte=tenth_id)
|
||||||
return qs
|
return qs
|
||||||
|
|
||||||
|
|
||||||
|
|
|
@ -181,7 +181,7 @@ class Order(models.Model):
|
||||||
return import_class(settings.ORDERS_BILLING_BACKEND)()
|
return import_class(settings.ORDERS_BILLING_BACKEND)()
|
||||||
|
|
||||||
def clean(self):
|
def clean(self):
|
||||||
if self.billed_on < self.registered_on:
|
if self.billed_on and self.billed_on < self.registered_on:
|
||||||
raise ValidationError(_("Billed date can not be earlier than registered on."))
|
raise ValidationError(_("Billed date can not be earlier than registered on."))
|
||||||
if self.billed_until and not self.billed_on:
|
if self.billed_until and not self.billed_on:
|
||||||
raise ValidationError(_("Billed on is missing while billed until is being provided."))
|
raise ValidationError(_("Billed on is missing while billed until is being provided."))
|
||||||
|
|
|
@ -24,7 +24,8 @@ def get_history_data(queryset):
|
||||||
'objects': [],
|
'objects': [],
|
||||||
}
|
}
|
||||||
resources[resource] = (options, aggregation)
|
resources[resource] = (options, aggregation)
|
||||||
|
if aggregation.aggregated_history:
|
||||||
|
needs_aggregation = True
|
||||||
monitors = []
|
monitors = []
|
||||||
scale = options['scale']
|
scale = options['scale']
|
||||||
all_dates = options['dates']
|
all_dates = options['dates']
|
||||||
|
@ -32,7 +33,6 @@ def get_history_data(queryset):
|
||||||
datasets = {}
|
datasets = {}
|
||||||
for content_object, datas in aggregation.aggregate_history(dataset):
|
for content_object, datas in aggregation.aggregate_history(dataset):
|
||||||
if aggregation.aggregated_history:
|
if aggregation.aggregated_history:
|
||||||
needs_aggregation = True
|
|
||||||
serie = {}
|
serie = {}
|
||||||
for data in datas:
|
for data in datas:
|
||||||
value = round(float(data.value)/scale, 3) if data.value is not None else None
|
value = round(float(data.value)/scale, 3) if data.value is not None else None
|
||||||
|
@ -54,9 +54,9 @@ def get_history_data(queryset):
|
||||||
})
|
})
|
||||||
options['objects'].append({
|
options['objects'].append({
|
||||||
'object_name': rdata.content_object_repr,
|
'object_name': rdata.content_object_repr,
|
||||||
'current': round(float(rdata.used), 3),
|
'current': round(float(rdata.used or 0), 3),
|
||||||
'allocated': float(rdata.allocated) if rdata.allocated is not None else None,
|
'allocated': float(rdata.allocated) if rdata.allocated is not None else None,
|
||||||
'updated_at': rdata.updated_at.isoformat(),
|
'updated_at': rdata.updated_at.isoformat() if rdata.updated_at else None,
|
||||||
'monitors': monitors,
|
'monitors': monitors,
|
||||||
})
|
})
|
||||||
if needs_aggregation:
|
if needs_aggregation:
|
||||||
|
|
|
@ -18,11 +18,11 @@ class PHPListForm(SaaSPasswordForm):
|
||||||
admin_username = forms.CharField(label=_("Admin username"), required=False,
|
admin_username = forms.CharField(label=_("Admin username"), required=False,
|
||||||
widget=SpanWidget(display='admin'))
|
widget=SpanWidget(display='admin'))
|
||||||
database = forms.CharField(label=_("Database"), required=False,
|
database = forms.CharField(label=_("Database"), required=False,
|
||||||
help_text=_("Database used for this instance."),
|
help_text=_("Database dedicated to this phpList instance."),
|
||||||
widget=SpanWidget(display=settings.SAAS_PHPLIST_DB_NAME.replace(
|
widget=SpanWidget(display=settings.SAAS_PHPLIST_DB_NAME.replace(
|
||||||
'%(', '<').replace(')s', '>')))
|
'%(', '<').replace(')s', '>')))
|
||||||
mailbox = forms.CharField(label=_("Bounces mailbox"), required=False,
|
mailbox = forms.CharField(label=_("Bounces mailbox"), required=False,
|
||||||
help_text=_("Mailbox used for reciving bounces."),
|
help_text=_("Dedicated mailbox used for reciving bounces."),
|
||||||
widget=SpanWidget(display=settings.SAAS_PHPLIST_BOUNCES_MAILBOX_NAME.replace(
|
widget=SpanWidget(display=settings.SAAS_PHPLIST_BOUNCES_MAILBOX_NAME.replace(
|
||||||
'%(', '<').replace(')s', '>')))
|
'%(', '<').replace(')s', '>')))
|
||||||
|
|
||||||
|
|
|
@ -374,7 +374,7 @@ class ServiceHandler(plugins.Plugin, metaclass=plugins.PluginMount):
|
||||||
priced[order][0] += price
|
priced[order][0] += price
|
||||||
priced[order][1] += cprice
|
priced[order][1] += cprice
|
||||||
else:
|
else:
|
||||||
priced[order] = (price, cprice)
|
priced[order] = [price, cprice]
|
||||||
lines = []
|
lines = []
|
||||||
for order, prices in priced.items():
|
for order, prices in priced.items():
|
||||||
if hasattr(order, 'new_billed_until'):
|
if hasattr(order, 'new_billed_until'):
|
||||||
|
|
|
@ -11,6 +11,7 @@ SYSTEMUSERS_SHELLS = Setting('SYSTEMUSERS_SHELLS',
|
||||||
(
|
(
|
||||||
('/dev/null', _("No shell, FTP only")),
|
('/dev/null', _("No shell, FTP only")),
|
||||||
('/bin/rssh', _("No shell, SFTP/RSYNC only")),
|
('/bin/rssh', _("No shell, SFTP/RSYNC only")),
|
||||||
|
('/usr/bin/git-shell', _("No shell, GIT only")),
|
||||||
('/bin/bash', "/bin/bash"),
|
('/bin/bash', "/bin/bash"),
|
||||||
('/bin/sh', "/bin/sh"),
|
('/bin/sh', "/bin/sh"),
|
||||||
),
|
),
|
||||||
|
@ -28,6 +29,7 @@ SYSTEMUSERS_DISABLED_SHELLS = Setting('SYSTEMUSERS_DISABLED_SHELLS',
|
||||||
default=(
|
default=(
|
||||||
'/dev/null',
|
'/dev/null',
|
||||||
'/bin/rssh',
|
'/bin/rssh',
|
||||||
|
'/usr/bin/git-shell',
|
||||||
),
|
),
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
|
@ -103,6 +103,11 @@ class PHPEnableFunctions(PHPAppOption):
|
||||||
regex = r'^[\w\.,-]+$'
|
regex = r'^[\w\.,-]+$'
|
||||||
comma_separated = True
|
comma_separated = True
|
||||||
|
|
||||||
|
def validate(self):
|
||||||
|
# Clean value removing spaces
|
||||||
|
self.instance.value = self.instance.value.replace(' ', '')
|
||||||
|
super(PHPEnableFunctions, self).validate()
|
||||||
|
|
||||||
|
|
||||||
class PHPAllowURLInclude(PHPAppOption):
|
class PHPAllowURLInclude(PHPAppOption):
|
||||||
name = 'allow_url_include'
|
name = 'allow_url_include'
|
||||||
|
|
|
@ -38,9 +38,7 @@ class Apache2Backend(ServiceController):
|
||||||
'WEBSITES_SAAS_DIRECTIVES',
|
'WEBSITES_SAAS_DIRECTIVES',
|
||||||
))
|
))
|
||||||
|
|
||||||
def render_virtual_host(self, site, context, ssl=False):
|
def get_extra_conf(self, site, context, ssl=False):
|
||||||
context['port'] = self.HTTPS_PORT if ssl else self.HTTP_PORT
|
|
||||||
context['vhost_set_fcgid'] = False
|
|
||||||
extra_conf = self.get_content_directives(site, context)
|
extra_conf = self.get_content_directives(site, context)
|
||||||
directives = site.get_directives()
|
directives = site.get_directives()
|
||||||
if ssl:
|
if ssl:
|
||||||
|
@ -54,7 +52,12 @@ class Apache2Backend(ServiceController):
|
||||||
extra_conf.append((location, directive % settings_context))
|
extra_conf.append((location, directive % settings_context))
|
||||||
# Order extra conf directives based on directives (longer first)
|
# Order extra conf directives based on directives (longer first)
|
||||||
extra_conf = sorted(extra_conf, key=lambda a: len(a[0]), reverse=True)
|
extra_conf = sorted(extra_conf, key=lambda a: len(a[0]), reverse=True)
|
||||||
context['extra_conf'] = '\n'.join([conf for location, conf in extra_conf])
|
return '\n'.join([conf for location, conf in extra_conf])
|
||||||
|
|
||||||
|
def render_virtual_host(self, site, context, ssl=False):
|
||||||
|
context['port'] = self.HTTPS_PORT if ssl else self.HTTP_PORT
|
||||||
|
context['vhost_set_fcgid'] = False
|
||||||
|
context['extra_conf'] = self.get_extra_conf(site, context, ssl)
|
||||||
return Template(textwrap.dedent("""\
|
return Template(textwrap.dedent("""\
|
||||||
<VirtualHost{% for ip in ips %} {{ ip }}:{{ port }}{% endfor %}>
|
<VirtualHost{% for ip in ips %} {{ ip }}:{{ port }}{% endfor %}>
|
||||||
IncludeOptional /etc/apache2/site[s]-override/{{ site_unique_name }}.con[f]
|
IncludeOptional /etc/apache2/site[s]-override/{{ site_unique_name }}.con[f]
|
||||||
|
|
|
@ -169,6 +169,7 @@ class SecEngine(SecRuleRemove):
|
||||||
verbose_name = _("SecRuleEngine Off")
|
verbose_name = _("SecRuleEngine Off")
|
||||||
help_text = _("URL path with disabled modsecurity engine.")
|
help_text = _("URL path with disabled modsecurity engine.")
|
||||||
regex = r'^/[^ ]*$'
|
regex = r'^/[^ ]*$'
|
||||||
|
unique_location = False
|
||||||
|
|
||||||
|
|
||||||
class WordPressSaaS(SiteDirective):
|
class WordPressSaaS(SiteDirective):
|
||||||
|
|
|
@ -109,8 +109,8 @@ WEBSITES_TRAFFIC_IGNORE_HOSTS = Setting('WEBSITES_TRAFFIC_IGNORE_HOSTS',
|
||||||
WEBSITES_SAAS_DIRECTIVES = Setting('WEBSITES_SAAS_DIRECTIVES',
|
WEBSITES_SAAS_DIRECTIVES = Setting('WEBSITES_SAAS_DIRECTIVES',
|
||||||
{
|
{
|
||||||
'wordpress-saas': ('fpm', '/var/run/fpm/pangea-5.4-fpm.sock', '/home/httpd/wordpress-mu/'),
|
'wordpress-saas': ('fpm', '/var/run/fpm/pangea-5.4-fpm.sock', '/home/httpd/wordpress-mu/'),
|
||||||
'drupal-saas': ('fpm', '/var/run/fpm/pangea-5.4-fpm.sock','/home/httpd/drupal-mu/'),
|
'drupal-saas': ('fpm', '/var/run/fpm/pangea-5.4-fpm.sock', '/home/httpd/drupal-mu/'),
|
||||||
'dokuwiki-saas': ('fpm', '/var/run/fpm/pangea-5.4-fpm.sock','/home/httpd/moodle-mu/'),
|
'dokuwiki-saas': ('fpm', '/var/run/fpm/pangea-5.4-fpm.sock', '/home/httpd/moodle-mu/'),
|
||||||
},
|
},
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue