django-orchestra/orchestra/apps/domains/backends.py

150 lines
5.3 KiB
Python
Raw Normal View History

2014-10-17 20:03:41 +00:00
import re
import textwrap
2014-05-08 16:59:35 +00:00
from django.utils.translation import ugettext_lazy as _
2014-05-08 16:59:35 +00:00
2014-07-09 16:17:43 +00:00
from orchestra.apps.orchestration import ServiceController
2014-10-04 13:23:04 +00:00
from orchestra.utils.python import AttrDict
2014-05-08 16:59:35 +00:00
from . import settings
2014-05-08 16:59:35 +00:00
2014-07-09 16:17:43 +00:00
class Bind9MasterDomainBackend(ServiceController):
2014-05-08 16:59:35 +00:00
verbose_name = _("Bind9 master domain")
model = 'domains.Domain'
related_models = (
('domains.Record', 'domain__origin'),
('domains.Domain', 'origin'),
)
2014-10-10 14:39:46 +00:00
ignore_fields = ['serial']
2014-05-08 16:59:35 +00:00
@classmethod
def is_main(cls, obj):
""" work around Domain.top self relationship """
if super(Bind9MasterDomainBackend, cls).is_main(obj):
return not obj.top
def save(self, domain):
context = self.get_context(domain)
domain.refresh_serial()
context['zone'] = ';; %(banner)s\n' % context
context['zone'] += domain.render_zone()
2014-10-17 13:41:08 +00:00
self.append(textwrap.dedent("""\
2014-10-17 20:03:41 +00:00
echo -e '%(zone)s' > %(zone_path)s.tmp
diff -N -I'^\s*;;' %(zone_path)s %(zone_path)s.tmp || UPDATED=1
mv %(zone_path)s.tmp %(zone_path)s""" % context
2014-10-17 13:41:08 +00:00
))
2014-05-08 16:59:35 +00:00
self.update_conf(context)
def update_conf(self, context):
2014-10-17 13:41:08 +00:00
self.append(textwrap.dedent("""\
2014-10-17 20:03:41 +00:00
sed '/zone "%(name)s".*/,/^\s*};\s*$/!d' %(conf_path)s | diff -B -I"^\s*//" - <(echo '%(conf)s') || {
2014-10-20 15:51:24 +00:00
sed -i -e '/zone\s\s*"%(name)s".*/,/^\s*};/d' \\
2014-10-17 20:03:41 +00:00
-e 'N; /^\\n$/d; P; D' %(conf_path)s
echo '%(conf)s' >> %(conf_path)s
2014-10-17 13:41:08 +00:00
UPDATED=1
}""" % context
))
2014-10-17 20:03:41 +00:00
# Delete ex-top-domains that are now subdomains
self.append(textwrap.dedent("""\
2014-10-20 15:51:24 +00:00
sed -i -e '/zone\s\s*".*\.%(name)s".*/,/^\s*};\s*$/d' \\
2014-10-17 20:03:41 +00:00
-e 'N; /^\\n$/d; P; D' %(conf_path)s""" % context
))
if 'zone_path' in context:
context['zone_subdomains_path'] = re.sub(r'^(.*/)', r'\1*.', context['zone_path'])
self.append('rm -f %(zone_subdomains_path)s' % context)
2014-05-08 16:59:35 +00:00
def delete(self, domain):
context = self.get_context(domain)
self.append('rm -f %(zone_path)s;' % context)
self.delete_conf(context)
def delete_conf(self, context):
2014-10-17 13:09:56 +00:00
if context['name'][0] in ('*', '_'):
# These can never be top level domains
return
2014-10-17 13:41:08 +00:00
self.append(textwrap.dedent("""\
2014-10-20 15:51:24 +00:00
sed -e '/zone\s\s*"%(name)s".*/,/^\s*};\s*$/d' \\
2014-10-17 20:03:41 +00:00
-e 'N; /^\\n$/d; P; D' %(conf_path)s > %(conf_path)s.tmp""" % context
2014-10-04 09:29:18 +00:00
))
2014-10-17 13:57:34 +00:00
self.append('diff -B -I"^\s*//" %(conf_path)s.tmp %(conf_path)s || UPDATED=1' % context)
2014-05-08 16:59:35 +00:00
self.append('mv %(conf_path)s.tmp %(conf_path)s' % context)
def commit(self):
""" reload bind if needed """
self.append('[[ $UPDATED == 1 ]] && service bind9 reload')
def get_servers(self, domain, backend):
2014-10-04 13:23:04 +00:00
from orchestra.apps.orchestration.models import Route
operation = AttrDict(backend=backend, action='save', instance=domain)
servers = []
for server in Route.get_servers(operation):
servers.append(server.get_ip())
return servers
2014-10-04 09:29:18 +00:00
def get_slaves(self, domain):
return self.get_servers(domain, Bind9SlaveDomainBackend)
2014-05-08 16:59:35 +00:00
def get_context(self, domain):
context = {
'name': domain.name,
'zone_path': settings.DOMAINS_ZONE_PATH % {'name': domain.name},
2014-10-03 14:02:11 +00:00
'subdomains': domain.subdomains.all(),
2014-05-08 16:59:35 +00:00
'banner': self.get_banner(),
2014-10-20 10:57:16 +00:00
'slaves': '; '.join(self.get_slaves(domain)) or '"none"',
2014-05-08 16:59:35 +00:00
}
context.update({
'conf_path': settings.DOMAINS_MASTERS_PATH,
'conf': textwrap.dedent("""
zone "%(name)s" {
// %(banner)s
type master;
file "%(zone_path)s";
allow-transfer { %(slaves)s; };
2014-10-20 10:57:16 +00:00
also-notify { %(slaves)s; };
};""" % context)
2014-05-08 16:59:35 +00:00
})
return context
class Bind9SlaveDomainBackend(Bind9MasterDomainBackend):
verbose_name = _("Bind9 slave domain")
2014-09-24 20:09:41 +00:00
related_models = (
('domains.Domain', 'origin'),
)
2014-05-13 13:46:40 +00:00
2014-05-08 16:59:35 +00:00
def save(self, domain):
context = self.get_context(domain)
self.update_conf(context)
def delete(self, domain):
context = self.get_context(domain)
self.delete_conf(context)
2014-10-04 09:29:18 +00:00
2014-05-08 16:59:35 +00:00
def commit(self):
""" ideally slave should be restarted after master """
self.append('[[ $UPDATED == 1 ]] && { sleep 1 && service bind9 reload; } &')
2014-10-04 09:29:18 +00:00
def get_masters(self, domain):
return self.get_servers(domain, Bind9MasterDomainBackend)
2014-05-08 16:59:35 +00:00
def get_context(self, domain):
context = {
'name': domain.name,
'banner': self.get_banner(),
'subdomains': domain.subdomains.all(),
'masters': '; '.join(self.get_masters(domain)) or '"none"',
2014-05-08 16:59:35 +00:00
}
context.update({
'conf_path': settings.DOMAINS_SLAVES_PATH,
'conf': textwrap.dedent("""
zone "%(name)s" {
// %(banner)s
type slave;
file "%(name)s";
masters { %(masters)s; };
allow-notify { %(masters)s; };
};""" % context)
2014-05-08 16:59:35 +00:00
})
return context