Improved resource monitoring
This commit is contained in:
parent
1cd3092673
commit
9b9abc3c91
3
TODO.md
3
TODO.md
|
@ -47,3 +47,6 @@ Remember that, as always with QuerySets, any subsequent chained methods which im
|
||||||
|
|
||||||
* passlib; nano /usr/local/lib/python2.7/dist-packages/passlib/ext/django/utils.py SortedDict -> collections.OrderedDict
|
* passlib; nano /usr/local/lib/python2.7/dist-packages/passlib/ext/django/utils.py SortedDict -> collections.OrderedDict
|
||||||
* pip install pyinotify
|
* pip install pyinotify
|
||||||
|
|
||||||
|
|
||||||
|
* Backend.operations dynamically generated based on defined methods
|
||||||
|
|
|
@ -8,12 +8,13 @@ from django.utils.encoding import force_text
|
||||||
|
|
||||||
|
|
||||||
def action_with_confirmation(action_name, extra_context={},
|
def action_with_confirmation(action_name, extra_context={},
|
||||||
template='admin/controller/generic_confirmation.html'):
|
template='admin/orchestra/generic_confirmation.html'):
|
||||||
"""
|
"""
|
||||||
Generic pattern for actions that needs confirmation step
|
Generic pattern for actions that needs confirmation step
|
||||||
If custom template is provided the form must contain:
|
If custom template is provided the form must contain:
|
||||||
<input type="hidden" name="post" value="generic_confirmation" />
|
<input type="hidden" name="post" value="generic_confirmation" />
|
||||||
"""
|
"""
|
||||||
|
|
||||||
def decorator(func, extra_context=extra_context, template=template):
|
def decorator(func, extra_context=extra_context, template=template):
|
||||||
@wraps(func, assigned=available_attrs(func))
|
@wraps(func, assigned=available_attrs(func))
|
||||||
def inner(modeladmin, request, queryset):
|
def inner(modeladmin, request, queryset):
|
||||||
|
|
|
@ -1,18 +1,25 @@
|
||||||
from orchestra.apps.orchestration import ServiceBackend
|
from django.utils.translation import ugettext_lazy as _
|
||||||
|
|
||||||
|
from orchestra.apps.orchestration import ServiceController
|
||||||
|
from orchestra.apps.resources import ServiceMonitor
|
||||||
|
|
||||||
from . import settings
|
from . import settings
|
||||||
|
|
||||||
|
|
||||||
class MySQLDBBackend(ServiceBackend):
|
class MySQLDBBackend(ServiceController):
|
||||||
verbose_name = "MySQL database"
|
verbose_name = "MySQL database"
|
||||||
model = 'databases.Database'
|
model = 'databases.Database'
|
||||||
|
|
||||||
def save(self, database):
|
def save(self, database):
|
||||||
if database.type == database.MYSQL:
|
if database.type == database.MYSQL:
|
||||||
context = self.get_context(database)
|
context = self.get_context(database)
|
||||||
self.append("mysql -e 'CREATE DATABASE `%(database)s`;'" % context)
|
self.append(
|
||||||
self.append("mysql -e 'GRANT ALL PRIVILEGES ON `%(database)s`.* "
|
"mysql -e 'CREATE DATABASE `%(database)s`;'" % context
|
||||||
" TO \"%(owner)s\"@\"%(host)s\" WITH GRANT OPTION;'" % context)
|
)
|
||||||
|
self.append(
|
||||||
|
"mysql -e 'GRANT ALL PRIVILEGES ON `%(database)s`.* "
|
||||||
|
" TO \"%(owner)s\"@\"%(host)s\" WITH GRANT OPTION;'" % context
|
||||||
|
)
|
||||||
|
|
||||||
def delete(self, database):
|
def delete(self, database):
|
||||||
if database.type == database.MYSQL:
|
if database.type == database.MYSQL:
|
||||||
|
@ -30,21 +37,27 @@ class MySQLDBBackend(ServiceBackend):
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
class MySQLUserBackend(ServiceBackend):
|
class MySQLUserBackend(ServiceController):
|
||||||
verbose_name = "MySQL user"
|
verbose_name = "MySQL user"
|
||||||
model = 'databases.DatabaseUser'
|
model = 'databases.DatabaseUser'
|
||||||
|
|
||||||
def save(self, database):
|
def save(self, database):
|
||||||
if database.type == database.MYSQL:
|
if database.type == database.MYSQL:
|
||||||
context = self.get_context(database)
|
context = self.get_context(database)
|
||||||
self.append("mysql -e 'CREATE USER \"%(username)s\"@\"%(host)s\";'" % context)
|
self.append(
|
||||||
self.append("mysql -e 'UPDATE mysql.user SET Password=\"%(password)s\" "
|
"mysql -e 'CREATE USER \"%(username)s\"@\"%(host)s\";'" % context
|
||||||
" WHERE User=\"%(username)s\";'" % context)
|
)
|
||||||
|
self.append(
|
||||||
|
"mysql -e 'UPDATE mysql.user SET Password=\"%(password)s\" "
|
||||||
|
" WHERE User=\"%(username)s\";'" % context
|
||||||
|
)
|
||||||
|
|
||||||
def delete(self, database):
|
def delete(self, database):
|
||||||
if database.type == database.MYSQL:
|
if database.type == database.MYSQL:
|
||||||
context = self.get_context(database)
|
context = self.get_context(database)
|
||||||
self.append("mysql -e 'DROP USER \"%(username)s\"@\"%(host)s\";'" % context)
|
self.append(
|
||||||
|
"mysql -e 'DROP USER \"%(username)s\"@\"%(host)s\";'" % context
|
||||||
|
)
|
||||||
|
|
||||||
def get_context(self, database):
|
def get_context(self, database):
|
||||||
return {
|
return {
|
||||||
|
@ -54,7 +67,12 @@ class MySQLUserBackend(ServiceBackend):
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
class MySQLPermissionBackend(ServiceBackend):
|
class MySQLPermissionBackend(ServiceController):
|
||||||
model = 'databases.UserDatabaseRelation'
|
model = 'databases.UserDatabaseRelation'
|
||||||
verbose_name = "MySQL permission"
|
verbose_name = "MySQL permission"
|
||||||
|
|
||||||
|
|
||||||
|
class MysqlDisk(ServiceMonitor):
|
||||||
|
model = 'database.Database'
|
||||||
|
resource = ServiceMonitor.DISK
|
||||||
|
verbose_name = _("MySQL disk")
|
||||||
|
|
|
@ -4,10 +4,10 @@ from django.utils.translation import ugettext_lazy as _
|
||||||
|
|
||||||
from . import settings
|
from . import settings
|
||||||
|
|
||||||
from orchestra.apps.orchestration import ServiceBackend
|
from orchestra.apps.orchestration import ServiceController
|
||||||
|
|
||||||
|
|
||||||
class Bind9MasterDomainBackend(ServiceBackend):
|
class Bind9MasterDomainBackend(ServiceController):
|
||||||
verbose_name = _("Bind9 master domain")
|
verbose_name = _("Bind9 master domain")
|
||||||
model = 'domains.Domain'
|
model = 'domains.Domain'
|
||||||
related_models = (
|
related_models = (
|
||||||
|
|
|
@ -1,11 +1,41 @@
|
||||||
from django.template import Template, Context
|
from django.template import Template, Context
|
||||||
|
from django.utils.translation import ugettext_lazy as _
|
||||||
|
|
||||||
from orchestra.apps.orchestration import ServiceBackend
|
from orchestra.apps.orchestration import ServiceController
|
||||||
|
from orchestra.apps.resources import ServiceMonitor
|
||||||
|
|
||||||
|
|
||||||
class MailmanBackend(ServiceBackend):
|
class MailmanBackend(ServiceController):
|
||||||
verbose_name = "Mailman"
|
verbose_name = "Mailman"
|
||||||
model = 'lists.List'
|
model = 'lists.List'
|
||||||
|
|
||||||
def save(self, mailinglist):
|
|
||||||
pass
|
class MailmanTraffic(ServiceMonitor):
|
||||||
|
model = 'lists.List'
|
||||||
|
resource = ServiceMonitor.TRAFFIC
|
||||||
|
|
||||||
|
def process(self, output):
|
||||||
|
for line in output.readlines():
|
||||||
|
listname, value = line.strip().slpit()
|
||||||
|
|
||||||
|
def monitor(self, mailinglist):
|
||||||
|
self.append(
|
||||||
|
"LISTS=$(grep -v 'post to mailman' /var/log/mailman/post"
|
||||||
|
" | grep size | cut -d'<' -f2 | cut -d'>' -f1 | sort | uniq"
|
||||||
|
" | while read line; do \n"
|
||||||
|
" grep \"$line\" post | head -n1 | awk {'print $8\" \"$11'}"
|
||||||
|
" | sed 's/size=//' | sed 's/,//'\n"
|
||||||
|
"done)"
|
||||||
|
)
|
||||||
|
self.append(
|
||||||
|
'SUBS=""\n'
|
||||||
|
'while read LIST; do\n'
|
||||||
|
' NAME=$(echo "$LIST" | awk {\'print $1\'})\n'
|
||||||
|
' SIZE=$(echo "$LIST" | awk {\'print $2\'})\n'
|
||||||
|
' if [[ ! $(echo -e "$SUBS" | grep "$NAME") ]]; then\n'
|
||||||
|
' SUBS="${SUBS}${NAME} $(list_members "$NAME" | wc -l)\n"\n'
|
||||||
|
' fi\n'
|
||||||
|
' SUBSCRIBERS=$(echo -e "$SUBS" | grep "$NAME" | awk {\'print $2\'})\n'
|
||||||
|
' echo "$NAME $(($SUBSCRIBERS*$SIZE))"\n'
|
||||||
|
'done <<< "$LISTS"'
|
||||||
|
)
|
||||||
|
|
|
@ -1 +1 @@
|
||||||
from .backends import ServiceBackend
|
from .backends import ServiceBackend, ServiceController
|
||||||
|
|
|
@ -22,10 +22,11 @@ STATE_COLORS = {
|
||||||
|
|
||||||
class RouteAdmin(admin.ModelAdmin):
|
class RouteAdmin(admin.ModelAdmin):
|
||||||
list_display = [
|
list_display = [
|
||||||
'id', 'backend', 'host', 'match', 'display_model', 'is_active'
|
'id', 'backend', 'host', 'match', 'display_model', 'display_actions',
|
||||||
|
'is_active'
|
||||||
]
|
]
|
||||||
list_editable = ['backend', 'host', 'match', 'is_active']
|
list_editable = ['backend', 'host', 'match', 'is_active']
|
||||||
list_filter = ['backend', 'host', 'is_active']
|
list_filter = ['host', 'is_active', 'backend']
|
||||||
|
|
||||||
def display_model(self, route):
|
def display_model(self, route):
|
||||||
try:
|
try:
|
||||||
|
@ -35,6 +36,14 @@ class RouteAdmin(admin.ModelAdmin):
|
||||||
display_model.short_description = _("model")
|
display_model.short_description = _("model")
|
||||||
display_model.allow_tags = True
|
display_model.allow_tags = True
|
||||||
|
|
||||||
|
def display_actions(self, route):
|
||||||
|
try:
|
||||||
|
return '<br>'.join(route.get_backend().get_actions())
|
||||||
|
except KeyError:
|
||||||
|
return "<span style='color: red;'>NOT AVAILABLE</span>"
|
||||||
|
display_actions.short_description = _("actions")
|
||||||
|
display_actions.allow_tags = True
|
||||||
|
|
||||||
|
|
||||||
class BackendOperationInline(admin.TabularInline):
|
class BackendOperationInline(admin.TabularInline):
|
||||||
model = BackendOperation
|
model = BackendOperation
|
||||||
|
|
|
@ -23,6 +23,7 @@ class ServiceBackend(object):
|
||||||
function_method = methods.Python
|
function_method = methods.Python
|
||||||
type = 'task' # 'sync'
|
type = 'task' # 'sync'
|
||||||
ignore_fields = []
|
ignore_fields = []
|
||||||
|
actions = []
|
||||||
|
|
||||||
# TODO type: 'script', execution:'task'
|
# TODO type: 'script', execution:'task'
|
||||||
|
|
||||||
|
@ -37,6 +38,10 @@ class ServiceBackend(object):
|
||||||
def __init__(self):
|
def __init__(self):
|
||||||
self.cmds = []
|
self.cmds = []
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def get_actions(cls):
|
||||||
|
return [ action for action in cls.actions if action in dir(cls) ]
|
||||||
|
|
||||||
@classmethod
|
@classmethod
|
||||||
def get_name(cls):
|
def get_name(cls):
|
||||||
return cls.__name__
|
return cls.__name__
|
||||||
|
@ -68,7 +73,7 @@ class ServiceBackend(object):
|
||||||
choices = []
|
choices = []
|
||||||
for b in backends:
|
for b in backends:
|
||||||
# don't evaluate b.verbose_name ugettext_lazy
|
# don't evaluate b.verbose_name ugettext_lazy
|
||||||
verbose = getattr(b.verbose_name, '_proxy____args', [None])
|
verbose = getattr(b.verbose_name, '_proxy____args', [b.verbose_name])
|
||||||
if verbose[0]:
|
if verbose[0]:
|
||||||
verbose = b.verbose_name
|
verbose = b.verbose_name
|
||||||
else:
|
else:
|
||||||
|
@ -110,3 +115,12 @@ class ServiceBackend(object):
|
||||||
the service once in bulk operations
|
the service once in bulk operations
|
||||||
"""
|
"""
|
||||||
pass
|
pass
|
||||||
|
|
||||||
|
|
||||||
|
class ServiceController(ServiceBackend):
|
||||||
|
actions = ('save', 'delete')
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def get_backends(cls):
|
||||||
|
""" filter controller classes """
|
||||||
|
return [ plugin for plugin in cls.plugins if ServiceController in plugin.__mro__ ]
|
||||||
|
|
|
@ -76,16 +76,14 @@ class BackendOperation(models.Model):
|
||||||
"""
|
"""
|
||||||
Encapsulates an operation, storing its related object, the action and the backend.
|
Encapsulates an operation, storing its related object, the action and the backend.
|
||||||
"""
|
"""
|
||||||
SAVE = 'save'
|
|
||||||
DELETE = 'delete'
|
DELETE = 'delete'
|
||||||
ACTIONS = (
|
SAVE = 'save'
|
||||||
(SAVE, _("save")),
|
MONITOR = 'monitor'
|
||||||
(DELETE, _("delete")),
|
|
||||||
)
|
|
||||||
|
|
||||||
log = models.ForeignKey('orchestration.BackendLog', related_name='operations')
|
log = models.ForeignKey('orchestration.BackendLog', related_name='operations')
|
||||||
|
# TODO backend and backend_class() (like content_type)
|
||||||
backend_class = models.CharField(_("backend"), max_length=256)
|
backend_class = models.CharField(_("backend"), max_length=256)
|
||||||
action = models.CharField(_("action"), max_length=64, choices=ACTIONS)
|
action = models.CharField(_("action"), max_length=64)
|
||||||
content_type = models.ForeignKey(ContentType)
|
content_type = models.ForeignKey(ContentType)
|
||||||
object_id = models.PositiveIntegerField()
|
object_id = models.PositiveIntegerField()
|
||||||
instance = generic.GenericForeignKey('content_type', 'object_id')
|
instance = generic.GenericForeignKey('content_type', 'object_id')
|
||||||
|
@ -149,14 +147,21 @@ class Route(models.Model):
|
||||||
|
|
||||||
@classmethod
|
@classmethod
|
||||||
def get_servers(cls, operation):
|
def get_servers(cls, operation):
|
||||||
backend_name = operation.backend.get_name()
|
# TODO use cached data sctructure and refactor
|
||||||
|
backend = operation.backend
|
||||||
|
servers = []
|
||||||
try:
|
try:
|
||||||
routes = cls.objects.filter(is_active=True, backend=backend_name)
|
routes = cls.objects.filter(is_active=True, backend=backend.get_name())
|
||||||
except cls.DoesNotExist:
|
except cls.DoesNotExist:
|
||||||
return []
|
return servers
|
||||||
safe_locals = { 'instance': operation.instance }
|
safe_locals = {
|
||||||
pks = [ route.pk for route in routes.all() if eval(route.match, safe_locals) ]
|
'instance': operation.instance
|
||||||
return [ route.host for route in routes.filter(pk__in=pks) ]
|
}
|
||||||
|
actions = backend.get_actions()
|
||||||
|
for route in routes:
|
||||||
|
if operation.action in actions and eval(route.match, safe_locals):
|
||||||
|
servers.append(route.host)
|
||||||
|
return servers
|
||||||
|
|
||||||
def get_backend(self):
|
def get_backend(self):
|
||||||
for backend in ServiceBackend.get_backends():
|
for backend in ServiceBackend.get_backends():
|
||||||
|
|
|
@ -1 +1,4 @@
|
||||||
|
from .backends import ServiceMonitor
|
||||||
|
|
||||||
|
|
||||||
default_app_config = 'orchestra.apps.resources.apps.ResourcesConfig'
|
default_app_config = 'orchestra.apps.resources.apps.ResourcesConfig'
|
||||||
|
|
|
@ -1,5 +1,3 @@
|
||||||
import sys
|
|
||||||
|
|
||||||
from django.contrib import admin
|
from django.contrib import admin
|
||||||
from django.contrib.contenttypes import generic
|
from django.contrib.contenttypes import generic
|
||||||
from django.utils.functional import cached_property
|
from django.utils.functional import cached_property
|
||||||
|
@ -7,9 +5,10 @@ from django.utils.translation import ugettext_lazy as _
|
||||||
|
|
||||||
from orchestra.admin.filters import UsedContentTypeFilter
|
from orchestra.admin.filters import UsedContentTypeFilter
|
||||||
from orchestra.admin.utils import insertattr, get_modeladmin
|
from orchestra.admin.utils import insertattr, get_modeladmin
|
||||||
|
from orchestra.utils import running_syncdb
|
||||||
|
|
||||||
from .forms import ResourceForm
|
from .forms import ResourceForm
|
||||||
from .models import Resource, ResourceAllocation, Monitor, MonitorData
|
from .models import Resource, ResourceData, MonitorData
|
||||||
|
|
||||||
|
|
||||||
class ResourceAdmin(admin.ModelAdmin):
|
class ResourceAdmin(admin.ModelAdmin):
|
||||||
|
@ -26,30 +25,24 @@ class ResourceAdmin(admin.ModelAdmin):
|
||||||
resources = obj.content_type.resource_set.filter(is_active=True)
|
resources = obj.content_type.resource_set.filter(is_active=True)
|
||||||
inlines = []
|
inlines = []
|
||||||
for inline in modeladmin.inlines:
|
for inline in modeladmin.inlines:
|
||||||
if inline.model is ResourceAllocation:
|
if inline.model is ResourceData:
|
||||||
inline = resource_inline_factory(resources)
|
inline = resource_inline_factory(resources)
|
||||||
inlines.append(inline)
|
inlines.append(inline)
|
||||||
modeladmin.inlines = inlines
|
modeladmin.inlines = inlines
|
||||||
|
|
||||||
|
|
||||||
class ResourceAllocationAdmin(admin.ModelAdmin):
|
class ResourceDataAdmin(admin.ModelAdmin):
|
||||||
list_display = ('id', 'resource', 'content_object', 'value')
|
list_display = ('id', 'resource', 'used', 'allocated', 'last_update',) # TODO content_object
|
||||||
list_filter = ('resource',)
|
list_filter = ('resource',)
|
||||||
|
|
||||||
|
|
||||||
class MonitorAdmin(admin.ModelAdmin):
|
|
||||||
list_display = ('backend', 'resource', 'crontab')
|
|
||||||
list_filter = ('backend', 'resource')
|
|
||||||
|
|
||||||
|
|
||||||
class MonitorDataAdmin(admin.ModelAdmin):
|
class MonitorDataAdmin(admin.ModelAdmin):
|
||||||
list_display = ('id', 'monitor', 'content_object', 'date', 'value')
|
list_display = ('id', 'monitor', 'date', 'value') # TODO content_object
|
||||||
list_filter = ('monitor',)
|
list_filter = ('monitor',)
|
||||||
|
|
||||||
|
|
||||||
admin.site.register(Resource, ResourceAdmin)
|
admin.site.register(Resource, ResourceAdmin)
|
||||||
admin.site.register(ResourceAllocation, ResourceAllocationAdmin)
|
admin.site.register(ResourceData, ResourceDataAdmin)
|
||||||
admin.site.register(Monitor, MonitorAdmin)
|
|
||||||
admin.site.register(MonitorData, MonitorDataAdmin)
|
admin.site.register(MonitorData, MonitorDataAdmin)
|
||||||
|
|
||||||
|
|
||||||
|
@ -68,7 +61,7 @@ def resource_inline_factory(resources):
|
||||||
return forms
|
return forms
|
||||||
|
|
||||||
class ResourceInline(generic.GenericTabularInline):
|
class ResourceInline(generic.GenericTabularInline):
|
||||||
model = ResourceAllocation
|
model = ResourceData
|
||||||
verbose_name_plural = _("resources")
|
verbose_name_plural = _("resources")
|
||||||
form = ResourceForm
|
form = ResourceForm
|
||||||
formset = ResourceInlineFormSet
|
formset = ResourceInlineFormSet
|
||||||
|
@ -84,7 +77,7 @@ def resource_inline_factory(resources):
|
||||||
|
|
||||||
return ResourceInline
|
return ResourceInline
|
||||||
|
|
||||||
if not 'migrate' in sys.argv and not 'syncdb' in sys.argv:
|
if not running_syncdb():
|
||||||
# not run during syncdb
|
# not run during syncdb
|
||||||
for resources in Resource.group_by_content_type():
|
for resources in Resource.group_by_content_type():
|
||||||
inline = resource_inline_factory(resources)
|
inline = resource_inline_factory(resources)
|
||||||
|
|
|
@ -1,6 +1,8 @@
|
||||||
from django.apps import AppConfig
|
from django.apps import AppConfig
|
||||||
from django.contrib.contenttypes import generic
|
from django.contrib.contenttypes import generic
|
||||||
|
|
||||||
|
from orchestra.utils import running_syncdb
|
||||||
|
|
||||||
|
|
||||||
class ResourcesConfig(AppConfig):
|
class ResourcesConfig(AppConfig):
|
||||||
name = 'orchestra.apps.resources'
|
name = 'orchestra.apps.resources'
|
||||||
|
@ -9,7 +11,8 @@ class ResourcesConfig(AppConfig):
|
||||||
def ready(self):
|
def ready(self):
|
||||||
from .models import Resource
|
from .models import Resource
|
||||||
# TODO execute on Resource.save()
|
# TODO execute on Resource.save()
|
||||||
relation = generic.GenericRelation('resources.ResourceAllocation')
|
if not running_syncdb():
|
||||||
|
relation = generic.GenericRelation('resources.ResourceData')
|
||||||
for resources in Resource.group_by_content_type():
|
for resources in Resource.group_by_content_type():
|
||||||
model = resources[0].content_type.model_class()
|
model = resources[0].content_type.model_class()
|
||||||
model.add_to_class('allocations', relation)
|
model.add_to_class('resources', relation)
|
||||||
|
|
|
@ -0,0 +1,27 @@
|
||||||
|
from orchestra.apps.orchestration import ServiceBackend
|
||||||
|
|
||||||
|
|
||||||
|
class ServiceMonitor(ServiceBackend):
|
||||||
|
TRAFFIC = 'traffic'
|
||||||
|
DISK = 'disk'
|
||||||
|
MEMORY = 'memory'
|
||||||
|
CPU = 'cpu'
|
||||||
|
|
||||||
|
actions = ('monitor', 'resource_exceeded', 'resource_recovery')
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def get_backends(cls):
|
||||||
|
""" filter monitor classes """
|
||||||
|
return [plugin for plugin in cls.plugins if ServiceMonitor in plugin.__mro__]
|
||||||
|
|
||||||
|
def store(self, stdout):
|
||||||
|
""" object_id value """
|
||||||
|
for line in stdout.readlines():
|
||||||
|
line = line.strip()
|
||||||
|
object_id, value = line.split()
|
||||||
|
# TODO date
|
||||||
|
MonitorHistory.store(self.model, object_id, value, date)
|
||||||
|
|
||||||
|
def execute(self, server):
|
||||||
|
log = super(MonitorBackend, self).execute(server)
|
||||||
|
return log
|
|
@ -7,23 +7,33 @@ from orchestra.forms.widgets import ShowTextWidget, ReadOnlyWidget
|
||||||
class ResourceForm(forms.ModelForm):
|
class ResourceForm(forms.ModelForm):
|
||||||
verbose_name = forms.CharField(label=_("Name"), widget=ShowTextWidget(bold=True),
|
verbose_name = forms.CharField(label=_("Name"), widget=ShowTextWidget(bold=True),
|
||||||
required=False)
|
required=False)
|
||||||
current = forms.CharField(label=_("Current"), widget=ShowTextWidget(),
|
used = forms.IntegerField(label=_("Used"), widget=ShowTextWidget(),
|
||||||
required=False)
|
required=False)
|
||||||
value = forms.CharField(label=_("Allocation"))
|
last_update = forms.CharField(label=_("Last update"), widget=ShowTextWidget(),
|
||||||
|
required=False)
|
||||||
|
allocated = forms.IntegerField(label=_("Allocated"))
|
||||||
|
|
||||||
class Meta:
|
class Meta:
|
||||||
fields = ('verbose_name', 'current', 'value',)
|
fields = ('verbose_name', 'used', 'last_update', 'allocated',)
|
||||||
|
|
||||||
def __init__(self, *args, **kwargs):
|
def __init__(self, *args, **kwargs):
|
||||||
self.resource = kwargs.pop('resource', None)
|
self.resource = kwargs.pop('resource', None)
|
||||||
super(ResourceForm, self).__init__(*args, **kwargs)
|
super(ResourceForm, self).__init__(*args, **kwargs)
|
||||||
if self.resource:
|
if self.resource:
|
||||||
self.fields['verbose_name'].initial = self.resource.verbose_name
|
self.fields['verbose_name'].initial = self.resource.verbose_name
|
||||||
self.fields['current'].initial = self.resource.get_current()
|
self.fields['used'].initial = self.resource.get_current()
|
||||||
if self.resource.ondemand:
|
if self.resource.ondemand:
|
||||||
self.fields['value'].widget = ReadOnlyWidget('')
|
self.fields['allocated'].required = False
|
||||||
|
self.fields['allocated'].widget = ReadOnlyWidget(None, '')
|
||||||
else:
|
else:
|
||||||
self.fields['value'].initial = self.resource.default_allocation
|
self.fields['allocated'].required = True
|
||||||
|
self.fields['allocated'].initial = self.resource.default_allocation
|
||||||
|
|
||||||
|
def has_changed(self):
|
||||||
|
""" Make sure resourcedata objects are created for all resources """
|
||||||
|
if not self.instance.pk:
|
||||||
|
return True
|
||||||
|
return super(ResourceForm, self).has_changed()
|
||||||
|
|
||||||
def save(self, *args, **kwargs):
|
def save(self, *args, **kwargs):
|
||||||
self.instance.resource_id = self.resource.pk
|
self.instance.resource_id = self.resource.pk
|
||||||
|
|
|
@ -7,13 +7,25 @@ from django.core import validators
|
||||||
from django.utils.translation import ugettext_lazy as _
|
from django.utils.translation import ugettext_lazy as _
|
||||||
from djcelery.models import PeriodicTask, CrontabSchedule
|
from djcelery.models import PeriodicTask, CrontabSchedule
|
||||||
|
|
||||||
|
from orchestra.models.fields import MultiSelectField
|
||||||
from orchestra.utils.apps import autodiscover
|
from orchestra.utils.apps import autodiscover
|
||||||
|
|
||||||
|
from .backends import ServiceMonitor
|
||||||
|
|
||||||
|
|
||||||
class Resource(models.Model):
|
class Resource(models.Model):
|
||||||
MONTHLY = 'MONTHLY'
|
"""
|
||||||
|
Defines a resource, a resource is basically an interpretation of data
|
||||||
|
gathered by a Monitor
|
||||||
|
"""
|
||||||
|
|
||||||
|
LAST = 'LAST'
|
||||||
|
MONTHLY_SUM = 'MONTHLY_SUM'
|
||||||
|
MONTHLY_AVG = 'MONTHLY_AVG'
|
||||||
PERIODS = (
|
PERIODS = (
|
||||||
(MONTHLY, _('Monthly')),
|
(LAST, _("Last")),
|
||||||
|
(MONTHLY_SUM, _("Monthly Sum")),
|
||||||
|
(MONTHLY_AVG, _("Monthly Average")),
|
||||||
)
|
)
|
||||||
|
|
||||||
name = models.CharField(_("name"), max_length=32, unique=True,
|
name = models.CharField(_("name"), max_length=32, unique=True,
|
||||||
|
@ -24,11 +36,14 @@ class Resource(models.Model):
|
||||||
verbose_name = models.CharField(_("verbose name"), max_length=256, unique=True)
|
verbose_name = models.CharField(_("verbose name"), max_length=256, unique=True)
|
||||||
content_type = models.ForeignKey(ContentType) # TODO filter by servicE?
|
content_type = models.ForeignKey(ContentType) # TODO filter by servicE?
|
||||||
period = models.CharField(_("period"), max_length=16, choices=PERIODS,
|
period = models.CharField(_("period"), max_length=16, choices=PERIODS,
|
||||||
default=MONTHLY)
|
default=LAST)
|
||||||
ondemand = models.BooleanField(default=False)
|
ondemand = models.BooleanField(_("on demand"), default=False)
|
||||||
default_allocation = models.PositiveIntegerField(null=True, blank=True)
|
default_allocation = models.PositiveIntegerField(_("default allocation"),
|
||||||
is_active = models.BooleanField(default=True)
|
null=True, blank=True)
|
||||||
disable_trigger = models.BooleanField(default=False)
|
is_active = models.BooleanField(_("is active"), default=True)
|
||||||
|
disable_trigger = models.BooleanField(_("disable trigger"), default=False)
|
||||||
|
monitors = MultiSelectField(_("monitors"), max_length=256,
|
||||||
|
choices=ServiceMonitor.get_choices())
|
||||||
|
|
||||||
def __unicode__(self):
|
def __unicode__(self):
|
||||||
return self.name
|
return self.name
|
||||||
|
@ -53,47 +68,58 @@ class Resource(models.Model):
|
||||||
today = datetime.date.today()
|
today = datetime.date.today()
|
||||||
result = 0
|
result = 0
|
||||||
has_result = False
|
has_result = False
|
||||||
for monitor in self.monitors.all():
|
for monitor in self.monitors:
|
||||||
|
dataset = MonitorData.objects.filter(monitor=monitor)
|
||||||
|
if self.period == self.MONTHLY_AVG:
|
||||||
|
try:
|
||||||
|
last = dataset.latest()
|
||||||
|
except MonitorData.DoesNotExist:
|
||||||
|
continue
|
||||||
has_result = True
|
has_result = True
|
||||||
if self.period == self.MONTHLY:
|
epoch = datetime(year=today.year, month=today.month, day=1)
|
||||||
data = monitor.dataset.filter(date__year=today.year,
|
total = (epoch-last.date).total_seconds()
|
||||||
|
dataset = dataset.filter(date__year=today.year,
|
||||||
date__month=today.month)
|
date__month=today.month)
|
||||||
result += data.aggregate(models.Sum('value'))['value__sum']
|
for data in dataset:
|
||||||
|
slot = (previous-data.date).total_seconds()
|
||||||
|
result += data.value * slot/total
|
||||||
|
elif self.period == self.MONTHLY_SUM:
|
||||||
|
data = dataset.filter(date__year=today.year,
|
||||||
|
date__month=today.month)
|
||||||
|
value = data.aggregate(models.Sum('value'))['value__sum']
|
||||||
|
if value:
|
||||||
|
has_result = True
|
||||||
|
result += value
|
||||||
|
elif self.period == self.LAST:
|
||||||
|
try:
|
||||||
|
result += dataset.latest().value
|
||||||
|
except MonitorData.DoesNotExist:
|
||||||
|
continue
|
||||||
|
has_result = True
|
||||||
else:
|
else:
|
||||||
raise NotImplementedError("%s support not implemented" % self.period)
|
raise NotImplementedError("%s support not implemented" % self.period)
|
||||||
return result if has_result else None
|
return result if has_result else None
|
||||||
|
|
||||||
|
|
||||||
class ResourceAllocation(models.Model):
|
class ResourceData(models.Model):
|
||||||
|
""" Stores computed resource usage and allocation """
|
||||||
resource = models.ForeignKey(Resource)
|
resource = models.ForeignKey(Resource)
|
||||||
content_type = models.ForeignKey(ContentType)
|
content_type = models.ForeignKey(ContentType)
|
||||||
object_id = models.PositiveIntegerField()
|
object_id = models.PositiveIntegerField()
|
||||||
value = models.PositiveIntegerField()
|
used = models.PositiveIntegerField(null=True)
|
||||||
|
last_update = models.DateTimeField(null=True)
|
||||||
|
allocated = models.PositiveIntegerField(null=True)
|
||||||
|
|
||||||
content_object = generic.GenericForeignKey()
|
content_object = generic.GenericForeignKey()
|
||||||
|
|
||||||
class Meta:
|
class Meta:
|
||||||
unique_together = ('resource', 'content_type', 'object_id')
|
unique_together = ('resource', 'content_type', 'object_id')
|
||||||
|
verbose_name_plural = _("resource data")
|
||||||
|
|
||||||
autodiscover('monitors')
|
|
||||||
|
|
||||||
|
|
||||||
class Monitor(models.Model):
|
|
||||||
backend = models.CharField(_("backend"), max_length=256,)
|
|
||||||
# choices=MonitorBackend.get_choices())
|
|
||||||
resource = models.ForeignKey(Resource, related_name='monitors')
|
|
||||||
crontab = models.ForeignKey(CrontabSchedule)
|
|
||||||
|
|
||||||
class Meta:
|
|
||||||
unique_together=('backend', 'resource')
|
|
||||||
|
|
||||||
def __unicode__(self):
|
|
||||||
return self.backend
|
|
||||||
|
|
||||||
|
|
||||||
class MonitorData(models.Model):
|
class MonitorData(models.Model):
|
||||||
monitor = models.ForeignKey(Monitor, related_name='dataset')
|
""" Stores monitored data """
|
||||||
|
monitor = models.CharField(_("monitor"), max_length=256,
|
||||||
|
choices=ServiceMonitor.get_choices())
|
||||||
content_type = models.ForeignKey(ContentType)
|
content_type = models.ForeignKey(ContentType)
|
||||||
object_id = models.PositiveIntegerField()
|
object_id = models.PositiveIntegerField()
|
||||||
date = models.DateTimeField(auto_now_add=True)
|
date = models.DateTimeField(auto_now_add=True)
|
||||||
|
@ -101,5 +127,9 @@ class MonitorData(models.Model):
|
||||||
|
|
||||||
content_object = generic.GenericForeignKey()
|
content_object = generic.GenericForeignKey()
|
||||||
|
|
||||||
|
class Meta:
|
||||||
|
get_latest_by = 'date'
|
||||||
|
verbose_name_plural = _("monitor data")
|
||||||
|
|
||||||
def __unicode__(self):
|
def __unicode__(self):
|
||||||
return str(self.monitor)
|
return str(self.monitor)
|
||||||
|
|
|
@ -1,27 +1,24 @@
|
||||||
from rest_framework import serializers
|
from rest_framework import serializers
|
||||||
|
|
||||||
from orchestra.api import router
|
from orchestra.api import router
|
||||||
|
from orchestra.utils import running_syncdb
|
||||||
|
|
||||||
from .models import Resource, ResourceAllocation
|
from .models import Resource, ResourceData
|
||||||
|
|
||||||
|
|
||||||
class ResourceSerializer(serializers.ModelSerializer):
|
class ResourceSerializer(serializers.ModelSerializer):
|
||||||
name = serializers.SerializerMethodField('get_name')
|
name = serializers.SerializerMethodField('get_name')
|
||||||
current = serializers.SerializerMethodField('get_current')
|
|
||||||
allocation = serializers.IntegerField(source='value')
|
|
||||||
|
|
||||||
class Meta:
|
class Meta:
|
||||||
model = ResourceAllocation
|
model = ResourceData
|
||||||
fields = ('name', 'current', 'allocation')
|
fields = ('name', 'used', 'allocated')
|
||||||
|
read_only_fields = ('used',)
|
||||||
|
|
||||||
def get_name(self, instance):
|
def get_name(self, instance):
|
||||||
return instance.resource.name
|
return instance.resource.name
|
||||||
|
|
||||||
def get_current(self, instance):
|
|
||||||
return instance.resource.get_current()
|
|
||||||
|
|
||||||
|
if not running_syncdb():
|
||||||
for resources in Resource.group_by_content_type():
|
for resources in Resource.group_by_content_type():
|
||||||
model = resources[0].content_type.model_class()
|
model = resources[0].content_type.model_class()
|
||||||
router.insert(model, 'resources', ResourceSerializer, required=False,
|
router.insert(model, 'resources', ResourceSerializer, required=False)
|
||||||
source='allocations')
|
|
||||||
|
|
|
@ -0,0 +1,14 @@
|
||||||
|
from celery import shared_task
|
||||||
|
|
||||||
|
from .backends import ServiceMonitor
|
||||||
|
|
||||||
|
|
||||||
|
@shared_task
|
||||||
|
def monitor(backend_name):
|
||||||
|
routes = Route.objects.filter(is_active=True, backend=backend_name)
|
||||||
|
for route in routes:
|
||||||
|
pass
|
||||||
|
for backend in ServiceMonitor.get_backends():
|
||||||
|
if backend.get_name() == backend_name:
|
||||||
|
# TODO execute monitor BackendOperation
|
||||||
|
pass
|
|
@ -15,7 +15,7 @@ from .roles.filters import role_list_filter_factory
|
||||||
|
|
||||||
|
|
||||||
class UserAdmin(AccountAdminMixin, auth.UserAdmin, ExtendedModelAdmin):
|
class UserAdmin(AccountAdminMixin, auth.UserAdmin, ExtendedModelAdmin):
|
||||||
list_display = ('username', 'is_main')
|
list_display = ('username', 'display_is_main')
|
||||||
list_filter = ('is_staff', 'is_superuser', 'is_active')
|
list_filter = ('is_staff', 'is_superuser', 'is_active')
|
||||||
fieldsets = (
|
fieldsets = (
|
||||||
(None, {
|
(None, {
|
||||||
|
@ -25,7 +25,7 @@ class UserAdmin(AccountAdminMixin, auth.UserAdmin, ExtendedModelAdmin):
|
||||||
'fields': ('first_name', 'last_name', 'email')
|
'fields': ('first_name', 'last_name', 'email')
|
||||||
}),
|
}),
|
||||||
(_("Permissions"), {
|
(_("Permissions"), {
|
||||||
'fields': ('is_active', 'is_staff', 'is_superuser', 'is_admin', 'is_main')
|
'fields': ('is_active', 'is_staff', 'is_superuser', 'display_is_main')
|
||||||
}),
|
}),
|
||||||
(_("Important dates"), {
|
(_("Important dates"), {
|
||||||
'fields': ('last_login', 'date_joined')
|
'fields': ('last_login', 'date_joined')
|
||||||
|
@ -38,7 +38,7 @@ class UserAdmin(AccountAdminMixin, auth.UserAdmin, ExtendedModelAdmin):
|
||||||
}),
|
}),
|
||||||
)
|
)
|
||||||
search_fields = ['username', 'account__user__username']
|
search_fields = ['username', 'account__user__username']
|
||||||
readonly_fields = ('is_main', 'account_link')
|
readonly_fields = ('display_is_main', 'account_link')
|
||||||
change_readonly_fields = ('username',)
|
change_readonly_fields = ('username',)
|
||||||
filter_horizontal = ()
|
filter_horizontal = ()
|
||||||
add_form = UserCreationForm
|
add_form = UserCreationForm
|
||||||
|
@ -46,10 +46,10 @@ class UserAdmin(AccountAdminMixin, auth.UserAdmin, ExtendedModelAdmin):
|
||||||
roles = []
|
roles = []
|
||||||
ordering = ('-id',)
|
ordering = ('-id',)
|
||||||
|
|
||||||
|
def display_is_main(self, instance):
|
||||||
def is_main(self, user):
|
return instance.is_main
|
||||||
return user.account.user == user
|
display_is_main.short_description = _("is main")
|
||||||
is_main.boolean = True
|
display_is_main.boolean = True
|
||||||
|
|
||||||
def get_urls(self):
|
def get_urls(self):
|
||||||
""" Returns the additional urls for the change view links """
|
""" Returns the additional urls for the change view links """
|
||||||
|
|
|
@ -1,11 +1,12 @@
|
||||||
from django.utils.translation import ugettext_lazy as _
|
from django.utils.translation import ugettext_lazy as _
|
||||||
|
|
||||||
from orchestra.apps.orchestration import ServiceBackend
|
from orchestra.apps.orchestration import ServiceController
|
||||||
|
from orchestra.apps.resources import ServiceMonitor
|
||||||
|
|
||||||
from . import settings
|
from . import settings
|
||||||
|
|
||||||
|
|
||||||
class SystemUserBackend(ServiceBackend):
|
class SystemUserBackend(ServiceController):
|
||||||
verbose_name = _("System User")
|
verbose_name = _("System User")
|
||||||
model = 'users.User'
|
model = 'users.User'
|
||||||
ignore_fields = ['last_login']
|
ignore_fields = ['last_login']
|
||||||
|
@ -39,3 +40,22 @@ class SystemUserBackend(ServiceBackend):
|
||||||
}
|
}
|
||||||
context['home'] = settings.USERS_SYSTEMUSER_HOME % context
|
context['home'] = settings.USERS_SYSTEMUSER_HOME % context
|
||||||
return context
|
return context
|
||||||
|
|
||||||
|
|
||||||
|
class SystemUserDisk(ServiceMonitor):
|
||||||
|
model = 'users.User'
|
||||||
|
resource = ServiceMonitor.DISK
|
||||||
|
verbose_name = _('System user disk')
|
||||||
|
|
||||||
|
def monitor(self, user):
|
||||||
|
context = self.get_context(user)
|
||||||
|
self.append("du -s %(home)s | {\n"
|
||||||
|
" read value\n"
|
||||||
|
" echo '%(username)s' $value\n"
|
||||||
|
"}" % context)
|
||||||
|
|
||||||
|
def process(self, output):
|
||||||
|
# TODO transaction
|
||||||
|
for line in output.readlines():
|
||||||
|
username, value = line.strip().slpit()
|
||||||
|
History.store(object_id=user_id, value=value)
|
||||||
|
|
|
@ -36,6 +36,10 @@ class User(auth.AbstractBaseUser):
|
||||||
USERNAME_FIELD = 'username'
|
USERNAME_FIELD = 'username'
|
||||||
REQUIRED_FIELDS = ['email']
|
REQUIRED_FIELDS = ['email']
|
||||||
|
|
||||||
|
@property
|
||||||
|
def is_main(self):
|
||||||
|
return self.account.user == self
|
||||||
|
|
||||||
def get_full_name(self):
|
def get_full_name(self):
|
||||||
full_name = '%s %s' % (self.first_name, self.last_name)
|
full_name = '%s %s' % (self.first_name, self.last_name)
|
||||||
return full_name.strip() or self.username
|
return full_name.strip() or self.username
|
||||||
|
|
|
@ -3,12 +3,13 @@ import os
|
||||||
from django.utils import timezone
|
from django.utils import timezone
|
||||||
from django.utils.translation import ugettext_lazy as _
|
from django.utils.translation import ugettext_lazy as _
|
||||||
|
|
||||||
from orchestra.apps.orchestration import ServiceBackend
|
from orchestra.apps.orchestration import ServiceController
|
||||||
|
from orchestra.apps.resources import ServiceMonitor
|
||||||
|
|
||||||
from . import settings
|
from . import settings
|
||||||
|
|
||||||
|
|
||||||
class MailSystemUserBackend(ServiceBackend):
|
class MailSystemUserBackend(ServiceController):
|
||||||
verbose_name = _("Mail system user")
|
verbose_name = _("Mail system user")
|
||||||
model = 'mail.Mailbox'
|
model = 'mail.Mailbox'
|
||||||
|
|
||||||
|
@ -62,7 +63,7 @@ class MailSystemUserBackend(ServiceBackend):
|
||||||
return context
|
return context
|
||||||
|
|
||||||
|
|
||||||
class PostfixAddressBackend(ServiceBackend):
|
class PostfixAddressBackend(ServiceController):
|
||||||
verbose_name = _("Postfix address")
|
verbose_name = _("Postfix address")
|
||||||
model = 'mail.Address'
|
model = 'mail.Address'
|
||||||
|
|
||||||
|
@ -132,12 +133,12 @@ class PostfixAddressBackend(ServiceBackend):
|
||||||
return context
|
return context
|
||||||
|
|
||||||
|
|
||||||
class AutoresponseBackend(ServiceBackend):
|
class AutoresponseBackend(ServiceController):
|
||||||
verbose_name = _("Mail autoresponse")
|
verbose_name = _("Mail autoresponse")
|
||||||
model = 'mail.Autoresponse'
|
model = 'mail.Autoresponse'
|
||||||
|
|
||||||
def save(self, autoresponse):
|
|
||||||
pass
|
|
||||||
|
|
||||||
def delete(self, autoresponse):
|
class MailDisk(ServiceMonitor):
|
||||||
pass
|
model = 'email.Mailbox'
|
||||||
|
resource = ServiceMonitor.DISK
|
||||||
|
verbose_name = _("Mail disk")
|
||||||
|
|
|
@ -0,0 +1,20 @@
|
||||||
|
from django.utils.translation import ugettext_lazy as _
|
||||||
|
|
||||||
|
from orchestra.apps.resources import ServiceMonitor
|
||||||
|
|
||||||
|
|
||||||
|
class OpenVZDisk(ServiceMonitor):
|
||||||
|
model = 'vps.VPS'
|
||||||
|
resource = ServiceMonitor.DISK
|
||||||
|
|
||||||
|
|
||||||
|
class OpenVZMemory(ServiceMonitor):
|
||||||
|
model = 'vps.VPS'
|
||||||
|
resource = ServiceMonitor.MEMORY
|
||||||
|
|
||||||
|
|
||||||
|
class OpenVZTraffic(ServiceMonitor):
|
||||||
|
model = 'vps.VPS'
|
||||||
|
resource = ServiceMonitor.TRAFFIC
|
||||||
|
|
||||||
|
|
|
@ -1,12 +1,9 @@
|
||||||
from django.utils.translation import ugettext_lazy as _
|
from django.utils.translation import ugettext_lazy as _
|
||||||
|
|
||||||
from orchestra.apps.orchestration import ServiceBackend
|
from orchestra.apps.orchestration import ServiceController
|
||||||
|
|
||||||
from . import WebAppServiceMixin
|
from . import WebAppServiceMixin
|
||||||
|
|
||||||
|
|
||||||
class AwstatsBackend(WebAppServiceMixin, ServiceBackend):
|
class AwstatsBackend(WebAppServiceMixin, ServiceController):
|
||||||
verbose_name = _("Awstats")
|
verbose_name = _("Awstats")
|
||||||
|
|
||||||
def save(self, webapp):
|
|
||||||
pass
|
|
||||||
|
|
|
@ -1,11 +1,12 @@
|
||||||
from django.utils.translation import ugettext_lazy as _
|
from django.utils.translation import ugettext_lazy as _
|
||||||
|
|
||||||
from orchestra.apps.orchestration import ServiceBackend
|
from orchestra.apps.orchestration import ServiceController
|
||||||
|
|
||||||
from . import WebAppServiceMixin
|
from . import WebAppServiceMixin
|
||||||
from .. import settings
|
from .. import settings
|
||||||
|
|
||||||
class DokuWikiMuBackend(WebAppServiceMixin, ServiceBackend):
|
|
||||||
|
class DokuWikiMuBackend(WebAppServiceMixin, ServiceController):
|
||||||
verbose_name = _("DokuWiki multisite")
|
verbose_name = _("DokuWiki multisite")
|
||||||
|
|
||||||
def save(self, webapp):
|
def save(self, webapp):
|
||||||
|
|
|
@ -2,13 +2,13 @@ import os
|
||||||
|
|
||||||
from django.utils.translation import ugettext_lazy as _
|
from django.utils.translation import ugettext_lazy as _
|
||||||
|
|
||||||
from orchestra.apps.orchestration import ServiceBackend
|
from orchestra.apps.orchestration import ServiceController
|
||||||
|
|
||||||
from . import WebAppServiceMixin
|
from . import WebAppServiceMixin
|
||||||
from .. import settings
|
from .. import settings
|
||||||
|
|
||||||
|
|
||||||
class DrupalMuBackend(WebAppServiceMixin, ServiceBackend):
|
class DrupalMuBackend(WebAppServiceMixin, ServiceController):
|
||||||
verbose_name = _("Drupal multisite")
|
verbose_name = _("Drupal multisite")
|
||||||
|
|
||||||
def save(self, webapp):
|
def save(self, webapp):
|
||||||
|
|
|
@ -2,13 +2,13 @@ import os
|
||||||
|
|
||||||
from django.utils.translation import ugettext_lazy as _
|
from django.utils.translation import ugettext_lazy as _
|
||||||
|
|
||||||
from orchestra.apps.orchestration import ServiceBackend
|
from orchestra.apps.orchestration import ServiceController
|
||||||
|
|
||||||
from . import WebAppServiceMixin
|
from . import WebAppServiceMixin
|
||||||
from .. import settings
|
from .. import settings
|
||||||
|
|
||||||
|
|
||||||
class PHPFcgidBackend(WebAppServiceMixin, ServiceBackend):
|
class PHPFcgidBackend(WebAppServiceMixin, ServiceController):
|
||||||
verbose_name = _("PHP-Fcgid")
|
verbose_name = _("PHP-Fcgid")
|
||||||
|
|
||||||
def save(self, webapp):
|
def save(self, webapp):
|
||||||
|
|
|
@ -3,13 +3,13 @@ import os
|
||||||
from django.template import Template, Context
|
from django.template import Template, Context
|
||||||
from django.utils.translation import ugettext_lazy as _
|
from django.utils.translation import ugettext_lazy as _
|
||||||
|
|
||||||
from orchestra.apps.orchestration import ServiceBackend
|
from orchestra.apps.orchestration import ServiceController
|
||||||
|
|
||||||
from . import WebAppServiceMixin
|
from . import WebAppServiceMixin
|
||||||
from .. import settings
|
from .. import settings
|
||||||
|
|
||||||
|
|
||||||
class PHPFPMBackend(WebAppServiceMixin, ServiceBackend):
|
class PHPFPMBackend(WebAppServiceMixin, ServiceController):
|
||||||
verbose_name = _("PHP-FPM")
|
verbose_name = _("PHP-FPM")
|
||||||
|
|
||||||
def save(self, webapp):
|
def save(self, webapp):
|
||||||
|
|
|
@ -1,11 +1,11 @@
|
||||||
from django.utils.translation import ugettext_lazy as _
|
from django.utils.translation import ugettext_lazy as _
|
||||||
|
|
||||||
from orchestra.apps.orchestration import ServiceBackend
|
from orchestra.apps.orchestration import ServiceController
|
||||||
|
|
||||||
from . import WebAppServiceMixin
|
from . import WebAppServiceMixin
|
||||||
|
|
||||||
|
|
||||||
class StaticBackend(WebAppServiceMixin, ServiceBackend):
|
class StaticBackend(WebAppServiceMixin, ServiceController):
|
||||||
verbose_name = _("Static")
|
verbose_name = _("Static")
|
||||||
|
|
||||||
def save(self, webapp):
|
def save(self, webapp):
|
||||||
|
|
|
@ -4,13 +4,13 @@ import sys
|
||||||
import requests
|
import requests
|
||||||
from django.utils.translation import ugettext_lazy as _
|
from django.utils.translation import ugettext_lazy as _
|
||||||
|
|
||||||
from orchestra.apps.orchestration import ServiceBackend
|
from orchestra.apps.orchestration import ServiceController
|
||||||
|
|
||||||
from . import WebAppServiceMixin
|
from . import WebAppServiceMixin
|
||||||
from .. import settings
|
from .. import settings
|
||||||
|
|
||||||
|
|
||||||
class WordpressMuBackend(WebAppServiceMixin, ServiceBackend):
|
class WordpressMuBackend(WebAppServiceMixin, ServiceController):
|
||||||
verbose_name = _("Wordpress multisite")
|
verbose_name = _("Wordpress multisite")
|
||||||
|
|
||||||
@property
|
@property
|
||||||
|
|
|
@ -3,12 +3,13 @@ import os
|
||||||
from django.template import Template, Context
|
from django.template import Template, Context
|
||||||
from django.utils.translation import ugettext_lazy as _
|
from django.utils.translation import ugettext_lazy as _
|
||||||
|
|
||||||
from orchestra.apps.orchestration import ServiceBackend
|
from orchestra.apps.orchestration import ServiceController
|
||||||
|
from orchestra.apps.resources import ServiceMonitor
|
||||||
|
|
||||||
from .. import settings
|
from .. import settings
|
||||||
|
|
||||||
|
|
||||||
class Apache2Backend(ServiceBackend):
|
class Apache2Backend(ServiceController):
|
||||||
model = 'websites.Website'
|
model = 'websites.Website'
|
||||||
related_models = (('websites.Content', 'website'),)
|
related_models = (('websites.Content', 'website'),)
|
||||||
verbose_name = _("Apache 2")
|
verbose_name = _("Apache 2")
|
||||||
|
@ -173,3 +174,58 @@ class Apache2Backend(ServiceBackend):
|
||||||
'fpm_port': content.webapp.get_fpm_port(),
|
'fpm_port': content.webapp.get_fpm_port(),
|
||||||
})
|
})
|
||||||
return context
|
return context
|
||||||
|
|
||||||
|
|
||||||
|
class Apache2Traffic(ServiceMonitor):
|
||||||
|
model = 'websites.Website'
|
||||||
|
resource = ServiceMonitor.TRAFFIC
|
||||||
|
verbose_name = _("Apache 2 Traffic")
|
||||||
|
|
||||||
|
def monitor(self, site):
|
||||||
|
context = self.get_context(site)
|
||||||
|
self.append("""
|
||||||
|
awk 'BEGIN {
|
||||||
|
ini = "%(start_date)s";
|
||||||
|
end = "%(end_date)s";
|
||||||
|
|
||||||
|
months["Jan"]="01";
|
||||||
|
months["Feb"]="02";
|
||||||
|
months["Mar"]="03";
|
||||||
|
months["Apr"]="04";
|
||||||
|
months["May"]="05";
|
||||||
|
months["Jun"]="06";
|
||||||
|
months["Jul"]="07";
|
||||||
|
months["Aug"]="08";
|
||||||
|
months["Sep"]="09";
|
||||||
|
months["Oct"]="10";
|
||||||
|
months["Nov"]="11";
|
||||||
|
months["Dec"]="12";
|
||||||
|
} {
|
||||||
|
date = substr($4,2)
|
||||||
|
year = substr(date,8,4)
|
||||||
|
month = months[substr(date,4,3)];
|
||||||
|
day = substr(date,1,2)
|
||||||
|
hour = substr(date,13,2)
|
||||||
|
minute = substr(date,16,2)
|
||||||
|
second = substr(date,19,2);
|
||||||
|
line_date = year month day hour minute second
|
||||||
|
if ( line_date > ini && line_date < end)
|
||||||
|
if ( $10 == "" )
|
||||||
|
sum+=$9
|
||||||
|
else
|
||||||
|
sum+=$10;
|
||||||
|
} END {
|
||||||
|
print sum;
|
||||||
|
}' %(log_file)s | {
|
||||||
|
read value
|
||||||
|
echo %(site_name)s $value
|
||||||
|
}
|
||||||
|
""" % context)
|
||||||
|
|
||||||
|
def get_context(self, site):
|
||||||
|
return {
|
||||||
|
'log_file': os.path.join(settings.WEBSITES_BASE_APACHE_LOGS, site.unique_name),
|
||||||
|
'start_date': '',
|
||||||
|
'end_date': '',
|
||||||
|
'site_name': '',
|
||||||
|
}
|
||||||
|
|
|
@ -3,12 +3,12 @@ from functools import partial
|
||||||
|
|
||||||
from django.utils.translation import ugettext_lazy as _
|
from django.utils.translation import ugettext_lazy as _
|
||||||
|
|
||||||
from orchestra.apps.orchestration import ServiceBackend
|
from orchestra.apps.orchestration import ServiceController
|
||||||
|
|
||||||
from .. import settings
|
from .. import settings
|
||||||
|
|
||||||
|
|
||||||
class WebalizerBackend(ServiceBackend):
|
class WebalizerBackend(ServiceController):
|
||||||
verbose_name = _("Webalizer")
|
verbose_name = _("Webalizer")
|
||||||
model = 'websites.Content'
|
model = 'websites.Content'
|
||||||
|
|
||||||
|
|
|
@ -1,291 +0,0 @@
|
||||||
from . import settings
|
|
||||||
|
|
||||||
|
|
||||||
class ServiceBackend(object):
|
|
||||||
"""
|
|
||||||
Service management backend base class
|
|
||||||
|
|
||||||
It uses the _unit of work_ design principle, which allows bulk operations to
|
|
||||||
be conviniently supported. Each backend generates the configuration for all
|
|
||||||
the changes of all modified objects, reloading the daemon just once.
|
|
||||||
"""
|
|
||||||
verbose_name = None
|
|
||||||
model = None
|
|
||||||
related_models = () # ((model, accessor__attribute),)
|
|
||||||
script_method = methods.BashSSH
|
|
||||||
function_method = methods.Python
|
|
||||||
type = 'task' # 'sync'
|
|
||||||
ignore_fields = []
|
|
||||||
|
|
||||||
# TODO type: 'script', execution:'task'
|
|
||||||
|
|
||||||
__metaclass__ = plugins.PluginMount
|
|
||||||
|
|
||||||
def __unicode__(self):
|
|
||||||
return type(self).__name__
|
|
||||||
|
|
||||||
def __str__(self):
|
|
||||||
return unicode(self)
|
|
||||||
|
|
||||||
def __init__(self):
|
|
||||||
self.cmds = []
|
|
||||||
|
|
||||||
@classmethod
|
|
||||||
def get_name(cls):
|
|
||||||
return cls.__name__
|
|
||||||
|
|
||||||
@classmethod
|
|
||||||
def is_main(cls, obj):
|
|
||||||
opts = obj._meta
|
|
||||||
return cls.model == '%s.%s' % (opts.app_label, opts.object_name)
|
|
||||||
|
|
||||||
@classmethod
|
|
||||||
def get_related(cls, obj):
|
|
||||||
opts = obj._meta
|
|
||||||
model = '%s.%s' % (opts.app_label, opts.object_name)
|
|
||||||
for rel_model, field in cls.related_models:
|
|
||||||
if rel_model == model:
|
|
||||||
related = obj
|
|
||||||
for attribute in field.split('__'):
|
|
||||||
related = getattr(related, attribute)
|
|
||||||
return related
|
|
||||||
return None
|
|
||||||
|
|
||||||
@classmethod
|
|
||||||
def get_backends(cls):
|
|
||||||
return cls.plugins
|
|
||||||
|
|
||||||
@classmethod
|
|
||||||
def get_choices(cls):
|
|
||||||
backends = cls.get_backends()
|
|
||||||
choices = ( (b.get_name(), b.verbose_name or b.get_name()) for b in backends )
|
|
||||||
return sorted(choices, key=lambda e: e[1])
|
|
||||||
|
|
||||||
def get_banner(self):
|
|
||||||
time = datetime.now().strftime("%h %d, %Y %I:%M:%S")
|
|
||||||
return "Generated by Orchestra %s" % time
|
|
||||||
|
|
||||||
def append(self, *cmd):
|
|
||||||
# aggregate commands acording to its execution method
|
|
||||||
if isinstance(cmd[0], basestring):
|
|
||||||
method = self.script_method
|
|
||||||
cmd = cmd[0]
|
|
||||||
else:
|
|
||||||
method = self.function_method
|
|
||||||
cmd = partial(*cmd)
|
|
||||||
if not self.cmds or self.cmds[-1][0] != method:
|
|
||||||
self.cmds.append((method, [cmd]))
|
|
||||||
else:
|
|
||||||
self.cmds[-1][1].append(cmd)
|
|
||||||
|
|
||||||
def execute(self, server):
|
|
||||||
from .models import BackendLog
|
|
||||||
state = BackendLog.STARTED if self.cmds else BackendLog.SUCCESS
|
|
||||||
log = BackendLog.objects.create(backend=self.get_name(), state=state, server=server)
|
|
||||||
for method, cmds in self.cmds:
|
|
||||||
method(log, server, cmds)
|
|
||||||
if log.state != BackendLog.SUCCESS:
|
|
||||||
break
|
|
||||||
return log
|
|
||||||
|
|
||||||
|
|
||||||
def ServiceController(ServiceBackend):
|
|
||||||
def save(self, obj)
|
|
||||||
raise NotImplementedError
|
|
||||||
|
|
||||||
def delete(self, obj):
|
|
||||||
raise NotImplementedError
|
|
||||||
|
|
||||||
def commit(self):
|
|
||||||
"""
|
|
||||||
apply the configuration, usually reloading a service
|
|
||||||
reloading a service is done in a separated method in order to reload
|
|
||||||
the service once in bulk operations
|
|
||||||
"""
|
|
||||||
pass
|
|
||||||
|
|
||||||
|
|
||||||
class ServiceMonitor(ServiceBackend):
|
|
||||||
TRAFFIC = 'traffic'
|
|
||||||
DISK = 'disk'
|
|
||||||
MEMORY = 'memory'
|
|
||||||
CPU = 'cpu'
|
|
||||||
|
|
||||||
def prepare(self):
|
|
||||||
pass
|
|
||||||
|
|
||||||
def store(self, stdout):
|
|
||||||
""" object_id value """
|
|
||||||
for line in stdout.readlines():
|
|
||||||
line = line.strip()
|
|
||||||
object_id, value = line.split()
|
|
||||||
# TODO date
|
|
||||||
MonitorHistory.store(self.model, object_id, value, date)
|
|
||||||
|
|
||||||
def monitor(self, obj):
|
|
||||||
raise NotImplementedError
|
|
||||||
|
|
||||||
def trigger(self, obj):
|
|
||||||
raise NotImplementedError
|
|
||||||
|
|
||||||
def execute(self, server):
|
|
||||||
log = super(MonitorBackend, self).execute(server)
|
|
||||||
|
|
||||||
return log
|
|
||||||
|
|
||||||
|
|
||||||
class AccountDisk(MonitorBackend):
|
|
||||||
model = 'accounts.Account'
|
|
||||||
resource = MonitorBackend.DISK
|
|
||||||
verbose_name = 'Disk'
|
|
||||||
|
|
||||||
def monitor(self, user):
|
|
||||||
context = self.get_context(user)
|
|
||||||
self.append("du -s %(home)s | {\n"
|
|
||||||
" read value\n"
|
|
||||||
" echo '%(username)s' $value\n"
|
|
||||||
"}" % context)
|
|
||||||
|
|
||||||
def process(self, output):
|
|
||||||
# TODO transaction
|
|
||||||
for line in output.readlines():
|
|
||||||
username, value = line.strip().slpit()
|
|
||||||
History.store(object_id=user_id, value=value)
|
|
||||||
|
|
||||||
|
|
||||||
class MailmanTraffic(MonitorBackend):
|
|
||||||
model = 'lists.List'
|
|
||||||
resource = MonitorBackend.TRAFFIC
|
|
||||||
|
|
||||||
def process(self, output):
|
|
||||||
for line in output.readlines():
|
|
||||||
listname, value = line.strip().slpit()
|
|
||||||
|
|
||||||
def monitor(self, mailinglist):
|
|
||||||
self.append("LISTS=$(grep -v 'post to mailman' /var/log/mailman/post"
|
|
||||||
" | grep size | cut -d'<' -f2 | cut -d'>' -f1 | sort | uniq"
|
|
||||||
" | while read line; do \n"
|
|
||||||
" grep \"$line\" post | head -n1 | awk {'print $8\" \"$11'}"
|
|
||||||
" | sed 's/size=//' | sed 's/,//'\n"
|
|
||||||
"done)")
|
|
||||||
self.append('SUBS=""\n'
|
|
||||||
'while read LIST; do\n'
|
|
||||||
' NAME=$(echo "$LIST" | awk {\'print $1\'})\n'
|
|
||||||
' SIZE=$(echo "$LIST" | awk {\'print $2\'})\n'
|
|
||||||
' if [[ ! $(echo -e "$SUBS" | grep "$NAME") ]]; then\n'
|
|
||||||
' SUBS="${SUBS}${NAME} $(list_members "$NAME" | wc -l)\n"\n'
|
|
||||||
' fi\n'
|
|
||||||
' SUBSCRIBERS=$(echo -e "$SUBS" | grep "$NAME" | awk {\'print $2\'})\n'
|
|
||||||
' echo "$NAME $(($SUBSCRIBERS*$SIZE))"\n'
|
|
||||||
'done <<< "$LISTS"')
|
|
||||||
|
|
||||||
|
|
||||||
class MailDisk(MonitorBackend):
|
|
||||||
model = 'email.Mailbox'
|
|
||||||
resource = MonitorBackend.DISK
|
|
||||||
verbose_name = _("Mail disk")
|
|
||||||
|
|
||||||
def process(self, output):
|
|
||||||
pass
|
|
||||||
|
|
||||||
def monitor(self, mail):
|
|
||||||
pass
|
|
||||||
|
|
||||||
|
|
||||||
class MysqlDisk(MonitorBackend):
|
|
||||||
model = 'database.Database'
|
|
||||||
resource = MonitorBackend.DISK
|
|
||||||
verbose_name = _("MySQL disk")
|
|
||||||
|
|
||||||
def process(self, output):
|
|
||||||
pass
|
|
||||||
|
|
||||||
def monitor(self, db):
|
|
||||||
pass
|
|
||||||
|
|
||||||
|
|
||||||
class OpenVZDisk(MonitorBackend):
|
|
||||||
model = 'vps.VPS'
|
|
||||||
resource = MonitorBackend.DISK
|
|
||||||
|
|
||||||
|
|
||||||
class OpenVZMemory(MonitorBackend):
|
|
||||||
model = 'vps.VPS'
|
|
||||||
resource = MonitorBackend.MEMORY
|
|
||||||
|
|
||||||
|
|
||||||
class OpenVZTraffic(MonitorBackend):
|
|
||||||
model = 'vps.VPS'
|
|
||||||
resource = MonitorBackend.TRAFFIC
|
|
||||||
|
|
||||||
|
|
||||||
class Apache2Traffic(MonitorBackend):
|
|
||||||
model = 'websites.Website'
|
|
||||||
resource = MonitorBackend.TRAFFIC
|
|
||||||
verbose_name = _("Apache2 Traffic")
|
|
||||||
|
|
||||||
def monitor(self, site):
|
|
||||||
context = self.get_context(site)
|
|
||||||
self.append("""
|
|
||||||
awk 'BEGIN {
|
|
||||||
ini = "%(start_date)s";
|
|
||||||
end = "%(end_date)s";
|
|
||||||
|
|
||||||
months["Jan"]="01";
|
|
||||||
months["Feb"]="02";
|
|
||||||
months["Mar"]="03";
|
|
||||||
months["Apr"]="04";
|
|
||||||
months["May"]="05";
|
|
||||||
months["Jun"]="06";
|
|
||||||
months["Jul"]="07";
|
|
||||||
months["Aug"]="08";
|
|
||||||
months["Sep"]="09";
|
|
||||||
months["Oct"]="10";
|
|
||||||
months["Nov"]="11";
|
|
||||||
months["Dec"]="12";
|
|
||||||
} {
|
|
||||||
date = substr($4,2)
|
|
||||||
year = substr(date,8,4)
|
|
||||||
month = months[substr(date,4,3)];
|
|
||||||
day = substr(date,1,2)
|
|
||||||
hour = substr(date,13,2)
|
|
||||||
minute = substr(date,16,2)
|
|
||||||
second = substr(date,19,2);
|
|
||||||
line_date = year month day hour minute second
|
|
||||||
if ( line_date > ini && line_date < end)
|
|
||||||
if ( $10 == "" )
|
|
||||||
sum+=$9
|
|
||||||
else
|
|
||||||
sum+=$10;
|
|
||||||
} END {
|
|
||||||
print sum;
|
|
||||||
}' %(log_file)s | {
|
|
||||||
read value
|
|
||||||
echo %(site_name)s $value
|
|
||||||
}
|
|
||||||
""" % context)
|
|
||||||
|
|
||||||
def trigger(self, site):
|
|
||||||
pass
|
|
||||||
|
|
||||||
def get_context(self, site):
|
|
||||||
return {
|
|
||||||
'log_file': os.path.join(settings.WEBSITES_BASE_APACHE_LOGS, site.unique_name)
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
# start_date and end_date expected format: YYYYMMDDhhmmss
|
|
||||||
|
|
||||||
function get_traffic(){
|
|
||||||
|
|
||||||
|
|
||||||
RESULT=$(get_traffic)
|
|
||||||
|
|
||||||
if [[ $RESULT ]]; then
|
|
||||||
echo $RESULT
|
|
||||||
else
|
|
||||||
echo 0
|
|
||||||
fi
|
|
||||||
|
|
||||||
return 0
|
|
||||||
|
|
|
@ -4,6 +4,11 @@ from django.utils.encoding import force_text
|
||||||
|
|
||||||
|
|
||||||
class ShowTextWidget(forms.Widget):
|
class ShowTextWidget(forms.Widget):
|
||||||
|
def __init__(self, *args, **kwargs):
|
||||||
|
for kwarg in ['bold', 'warning', 'hidden']:
|
||||||
|
setattr(self, kwarg, kwargs.pop(kwarg, False))
|
||||||
|
super(ShowTextWidget, self).__init__(*args, **kwargs)
|
||||||
|
|
||||||
def render(self, name, value, attrs):
|
def render(self, name, value, attrs):
|
||||||
value = force_text(value)
|
value = force_text(value)
|
||||||
if value is None:
|
if value is None:
|
||||||
|
@ -20,11 +25,6 @@ class ShowTextWidget(forms.Widget):
|
||||||
final_value = u'%s<input type="hidden" name="%s" value="%s"/>' % (final_value, name, value)
|
final_value = u'%s<input type="hidden" name="%s" value="%s"/>' % (final_value, name, value)
|
||||||
return mark_safe(final_value)
|
return mark_safe(final_value)
|
||||||
|
|
||||||
def __init__(self, *args, **kwargs):
|
|
||||||
for kwarg in ['bold', 'warning', 'hidden']:
|
|
||||||
setattr(self, kwarg, kwargs.pop(kwarg, False))
|
|
||||||
super(ShowTextWidget, self).__init__(*args, **kwargs)
|
|
||||||
|
|
||||||
def _has_changed(self, initial, data):
|
def _has_changed(self, initial, data):
|
||||||
return False
|
return False
|
||||||
|
|
||||||
|
|
|
@ -54,4 +54,4 @@ class MultiSelectField(models.CharField):
|
||||||
|
|
||||||
if isinstalled('south'):
|
if isinstalled('south'):
|
||||||
from south.modelsinspector import add_introspection_rules
|
from south.modelsinspector import add_introspection_rules
|
||||||
add_introspection_rules([], ["^controller\.models\.fields\.MultiSelectField"])
|
add_introspection_rules([], ["^orchestra\.models\.fields\.MultiSelectField"])
|
||||||
|
|
|
@ -5,18 +5,18 @@ body {
|
||||||
|
|
||||||
#header #branding h1 {
|
#header #branding h1 {
|
||||||
margin: 0;
|
margin: 0;
|
||||||
padding: 5px 10px;
|
padding: 2px 10px;
|
||||||
background: transparent url(/static/orchestra/images/orchestra-logo.png) 10px 5px no-repeat;
|
background: transparent url(/static/orchestra/images/orchestra-logo.png) 10px 2px no-repeat;
|
||||||
text-indent: 0;
|
text-indent: 0;
|
||||||
height: 31px;
|
height: 31px;
|
||||||
font-size: 18px;
|
font-size: 16px;
|
||||||
font-weight: bold;
|
/* font-weight: bold;*/
|
||||||
padding-left: 50px;
|
padding-left: 50px;
|
||||||
line-height: 30px;
|
line-height: 30px;
|
||||||
}
|
}
|
||||||
|
|
||||||
#branding h1, #branding h1 a:link, #branding h1 a:visited {
|
#branding h1, #branding h1 a:link, #branding h1 a:visited {
|
||||||
color: #707070;
|
color: #555;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
Binary file not shown.
Before Width: | Height: | Size: 1.9 KiB After Width: | Height: | Size: 1.8 KiB |
|
@ -16,8 +16,8 @@
|
||||||
inkscape:version="0.48.3.1 r9886"
|
inkscape:version="0.48.3.1 r9886"
|
||||||
sodipodi:docname="orchestra-logo.svg"
|
sodipodi:docname="orchestra-logo.svg"
|
||||||
inkscape:export-filename="/home/glic3rinu/orchestra/django-orchestra/orchestra/static/orchestra/images/orchestra-logo.png"
|
inkscape:export-filename="/home/glic3rinu/orchestra/django-orchestra/orchestra/static/orchestra/images/orchestra-logo.png"
|
||||||
inkscape:export-xdpi="90"
|
inkscape:export-xdpi="81.290321"
|
||||||
inkscape:export-ydpi="90">
|
inkscape:export-ydpi="81.290321">
|
||||||
<defs
|
<defs
|
||||||
id="defs4">
|
id="defs4">
|
||||||
<linearGradient
|
<linearGradient
|
||||||
|
|
Before Width: | Height: | Size: 27 KiB After Width: | Height: | Size: 27 KiB |
|
@ -9,7 +9,7 @@ register = template.Library()
|
||||||
|
|
||||||
|
|
||||||
@register.simple_tag(name="version")
|
@register.simple_tag(name="version")
|
||||||
def controller_version():
|
def orchestra_version():
|
||||||
return get_version()
|
return get_version()
|
||||||
|
|
||||||
|
|
||||||
|
|
|
@ -1,3 +1,4 @@
|
||||||
|
import sys
|
||||||
import urlparse
|
import urlparse
|
||||||
|
|
||||||
from django.core.mail import EmailMultiAlternatives
|
from django.core.mail import EmailMultiAlternatives
|
||||||
|
@ -37,3 +38,6 @@ def send_email_template(template, context, to, email_from=None, html=None):
|
||||||
msg.attach_alternative(html_message, "text/html")
|
msg.attach_alternative(html_message, "text/html")
|
||||||
msg.send()
|
msg.send()
|
||||||
|
|
||||||
|
|
||||||
|
def running_syncdb():
|
||||||
|
return 'migrate' in sys.argv or 'syncdb' in sys.argv
|
||||||
|
|
Loading…
Reference in New Issue