Adjustment in domain's record applying

This commit is contained in:
Khanh Ngo 2019-12-14 14:47:21 +07:00
parent 65d4acc6c5
commit 14658d797e
No known key found for this signature in database
GPG Key ID: D5FAA6A16150E49E
5 changed files with 401 additions and 363 deletions

View File

@ -5,7 +5,6 @@ import hashlib
import ipaddress import ipaddress
import os import os
# from app import app
from distutils.version import StrictVersion from distutils.version import StrictVersion
from urllib.parse import urlparse from urllib.parse import urlparse
from datetime import datetime, timedelta from datetime import datetime, timedelta
@ -300,6 +299,10 @@ def validate_ipaddress(address):
return [] return []
def pretty_json(data):
return json.dumps(data, sort_keys=True, indent=4)
class customBoxes: class customBoxes:
boxes = { boxes = {
"reverse": (" ", " "), "reverse": (" ", " "),

View File

@ -7,6 +7,7 @@ from distutils.version import StrictVersion
from flask import current_app from flask import current_app
from urllib.parse import urljoin from urllib.parse import urljoin
from distutils.util import strtobool from distutils.util import strtobool
from itertools import groupby
from .. import utils from .. import utils
from .base import db from .base import db
@ -89,14 +90,34 @@ class Record(object):
return jdata return jdata
def add(self, domain): def get_rrsets(self, domain):
""" """
Add a record to domain Query domain's rrsets via PDNS API
""" """
# validate record first headers = {}
r = self.get_record_data(domain) headers['X-API-Key'] = self.PDNS_API_KEY
records = r['records'] try:
check = list(filter(lambda check: check['name'] == self.name, records)) jdata = utils.fetch_json(urljoin(
self.PDNS_STATS_URL, self.API_EXTENDED_URL +
'/servers/localhost/zones/{0}'.format(domain)),
timeout=int(
Setting().get('pdns_api_timeout')),
headers=headers)
except Exception as e:
current_app.logger.error(
"Cannot fetch domain's record data from remote powerdns api. DETAIL: {0}"
.format(e))
return False
return jdata['rrsets']
def add(self, domain_name, rrset):
"""
Add a record to a domain (a reverse domain name)
"""
# Validate record first
rrsets = self.get_rrsets(domain_name)
check = list(filter(lambda check: check['name'] == self.name, rrsets))
if check: if check:
r = check[0] r = check[0]
if r['type'] in ('A', 'AAAA', 'CNAME'): if r['type'] in ('A', 'AAAA', 'CNAME'):
@ -106,315 +127,288 @@ class Record(object):
'Record already exists with type "A", "AAAA" or "CNAME"' 'Record already exists with type "A", "AAAA" or "CNAME"'
} }
# continue if the record is ready to be added # Continue if the record is ready to be added
headers = {} headers = {}
headers['X-API-Key'] = self.PDNS_API_KEY headers['X-API-Key'] = self.PDNS_API_KEY
if self.NEW_SCHEMA: # if self.NEW_SCHEMA:
data = { # data = {
"rrsets": [{ # "rrsets": [{
"name": # "name":
self.name.rstrip('.') + '.', # self.name.rstrip('.') + '.',
"type": # "type":
self.type, # self.type,
"changetype": # "changetype":
"REPLACE", # "REPLACE",
"ttl": # "ttl":
self.ttl, # self.ttl,
"records": [{ # "records": [{
"content": self.data, # "content": self.data,
"disabled": self.status, # "disabled": self.status,
}], # }],
"comments": # "comments":
[self.comment_data] if self.comment_data else [] # [self.comment_data] if self.comment_data else []
}] # }]
} # }
else: # else:
data = { # data = {
"rrsets": [{ # "rrsets": [{
"name": # "name":
self.name, # self.name,
"type": # "type":
self.type, # self.type,
"changetype": # "changetype":
"REPLACE", # "REPLACE",
"records": [{ # "records": [{
"content": self.data, # "content": self.data,
"disabled": self.status, # "disabled": self.status,
"name": self.name, # "name": self.name,
"ttl": self.ttl, # "ttl": self.ttl,
"type": self.type # "type": self.type
}], # }],
"comments": # "comments":
[self.comment_data] if self.comment_data else [] # [self.comment_data] if self.comment_data else []
}] # }]
} # }
try: try:
jdata = utils.fetch_json(urljoin( jdata = utils.fetch_json(urljoin(
self.PDNS_STATS_URL, self.API_EXTENDED_URL + self.PDNS_STATS_URL, self.API_EXTENDED_URL +
'/servers/localhost/zones/{0}'.format(domain)), '/servers/localhost/zones/{0}'.format(domain_name)),
headers=headers, headers=headers,
timeout=int( timeout=int(
Setting().get('pdns_api_timeout')), Setting().get('pdns_api_timeout')),
method='PATCH', method='PATCH',
data=data) data=rrset)
current_app.logger.debug(jdata) current_app.logger.debug(jdata)
return {'status': 'ok', 'msg': 'Record was added successfully'} return {'status': 'ok', 'msg': 'Record was added successfully'}
except Exception as e: except Exception as e:
current_app.logger.error( current_app.logger.error(
"Cannot add record {0}/{1}/{2} to domain {3}. DETAIL: {4}". "Cannot add record {0}/{1}/{2} to domain {3}. DETAIL: {4}".
format(self.name, self.type, self.data, domain, e)) format(self.name, self.type, self.data, domain_name, e))
return { return {
'status': 'error', 'status': 'error',
'msg': 'msg':
'There was something wrong, please contact administrator' 'There was something wrong, please contact administrator'
} }
def compare(self, domain_name, new_records): def merge_rrsets(self, rrsets):
""" """
Compare new records with current powerdns record data Merge the rrsets that has same "name" and
Input is a list of hashes (records) "type".
Return: a new rrest which has multiple "records"
and "comments"
""" """
# get list of current records we have in powerdns if not rrsets:
current_records = self.get_record_data(domain_name)['records'] raise Exception("Empty rrsets to merge")
elif len(rrsets) == 1:
# It is unique rrest already
return rrsets[0]
else:
# Merge rrsets into one
rrest = rrsets[0]
for r in rrsets[1:]:
rrest['records'] = rrest['records'] + r['records']
rrest['comments'] = rrest['comments'] + r['comments']
return rrest
# convert them to list of list (just has [name, type]) instead of list of hash def build_rrsets(self, domain_name, submitted_records):
# to compare easier
list_current_records = [[x['name'], x['type']]
for x in current_records]
list_new_records = [[x['name'], x['type']] for x in new_records]
# get list of deleted records
# they are the records which exist in list_current_records but not in list_new_records
list_deleted_records = [
x for x in list_current_records if x not in list_new_records
]
# convert back to list of hash
deleted_records = [
x for x in current_records
if [x['name'], x['type']] in list_deleted_records and (
x['type'] in Setting().get_records_allow_to_edit()
and x['type'] != 'SOA')
]
# return a tuple
return deleted_records, new_records
def apply(self, domain, post_records):
""" """
Apply record changes to domain Build rrsets from the datatable's records
Args:
domain_name(str): The zone name
submitted_records(list): List of records submitted from PDA datatable
Returns:
transformed_rrsets(list): List of rrests converted from PDA datatable
""" """
records = [] rrsets = []
for r in post_records: for record in submitted_records:
r_name = domain if r['record_name'] in [ # Format the record name
record_name = "{}.{}.".format(
record["record_name"],
domain_name) if record["record_name"] not in [
'@', '' '@', ''
] else r['record_name'] + '.' + domain ] else domain_name + '.'
r_type = r['record_type']
if self.PRETTY_IPV6_PTR: # only if activated
if self.NEW_SCHEMA: # only if new schema
if r_type == 'PTR': # only ptr
if ':' in r['record_name']: # dirty ipv6 check
r_name = r['record_name']
r_data = domain if r_type == 'CNAME' and r['record_data'] in [ # Format the record content, it musts end
'@', '' # with a dot character if in following types
] else r['record_data'] if record["record_type"] in [
'MX', 'CNAME', 'SRV', 'NS'
] and record["record_data"].strip()[-1:] != '.':
record["record_data"] += '.'
record = { record_conntent = {
"name": r_name, "content": record["record_data"],
"type": r_type,
"content": r_data,
"disabled": "disabled":
True if r['record_status'] == 'Disabled' else False, False if record['record_status'] == 'Active' else True
"ttl": int(r['record_ttl']) if r['record_ttl'] else 3600,
"comment_data": r['comment_data']
}
records.append(record)
deleted_records, new_records = self.compare(domain, records)
records = []
for r in deleted_records:
r_name = r['name'].rstrip(
'.') + '.' if self.NEW_SCHEMA else r['name']
r_type = r['type']
if self.PRETTY_IPV6_PTR: # only if activated
if self.NEW_SCHEMA: # only if new schema
if r_type == 'PTR': # only ptr
if ':' in r['name']: # dirty ipv6 check
r_name = dns.reversename.from_address(
r['name']).to_text()
record = {
"name": r_name,
"type": r_type,
"changetype": "DELETE",
"records": []
}
records.append(record)
postdata_for_delete = {"rrsets": records}
records = []
for r in new_records:
if self.NEW_SCHEMA:
r_name = r['name'].rstrip('.') + '.'
r_type = r['type']
if self.PRETTY_IPV6_PTR: # only if activated
if r_type == 'PTR': # only ptr
if ':' in r['name']: # dirty ipv6 check
r_name = r['name']
record = {
"name":
r_name,
"type":
r_type,
"changetype":
"REPLACE",
"ttl":
r['ttl'],
"records": [{
"content": r['content'],
"disabled": r['disabled']
}],
"comments":
r['comment_data']
}
else:
record = {
"name":
r['name'],
"type":
r['type'],
"changetype":
"REPLACE",
"records": [{
"content": r['content'],
"disabled": r['disabled'],
"name": r['name'],
"ttl": r['ttl'],
"type": r['type'],
"priority":
10, # priority field for pdns 3.4.1. https://doc.powerdns.com/md/authoritative/upgrading/
}],
"comments":
r['comment_data']
} }
records.append(record) # Format the comment
record_comments = [{
"content": record["record_comment"],
"account": ""
}] if record["record_comment"] else []
# Adjustment to add multiple records which described in # Add the formatted record to rrsets list
# https://github.com/ngoduykhanh/PowerDNS-Admin/issues/5#issuecomment-181637576 rrsets.append({
final_records = [] "name": record_name,
records = sorted(records, "type": record["record_type"],
key=lambda item: "ttl": int(record["record_ttl"]),
(item["name"], item["type"], item["changetype"])) "records": [record_conntent],
for key, group in itertools.groupby( "comments": record_comments
records, lambda item:
(item["name"], item["type"], item["changetype"])):
if self.NEW_SCHEMA:
r_name = key[0]
r_type = key[1]
r_changetype = key[2]
if self.PRETTY_IPV6_PTR: # only if activated
if r_type == 'PTR': # only ptr
if ':' in r_name: # dirty ipv6 check
r_name = dns.reversename.from_address(
r_name).to_text()
new_record = {
"name": r_name,
"type": r_type,
"changetype": r_changetype,
"ttl": None,
"records": []
}
for item in group:
temp_content = item['records'][0]['content']
temp_disabled = item['records'][0]['disabled']
if key[1] in ['MX', 'CNAME', 'SRV', 'NS']:
if temp_content.strip()[-1:] != '.':
temp_content += '.'
if new_record['ttl'] is None:
new_record['ttl'] = item['ttl']
new_record['records'].append({
"content": temp_content,
"disabled": temp_disabled
})
new_record['comments'] = item['comments']
final_records.append(new_record)
else:
final_records.append({
"name":
key[0],
"type":
key[1],
"changetype":
key[2],
"records": [{
"content": item['records'][0]['content'],
"disabled": item['records'][0]['disabled'],
"name": key[0],
"ttl": item['records'][0]['ttl'],
"type": key[1],
"priority": 10,
} for item in group]
}) })
postdata_for_new = {"rrsets": final_records} # Group the records which has the same name and type.
# The rrest then has multiple records inside.
transformed_rrsets = []
# Sort the list before using groupby
rrsets = sorted(rrsets, key=lambda r: (r['name'], r['type']))
groups = groupby(rrsets, key=lambda r: (r['name'], r['type']))
for k, v in groups:
group = list(v)
transformed_rrsets.append(self.merge_rrsets(group))
return transformed_rrsets
def compare(self, domain_name, submitted_records):
"""
Compare the submitted records with PDNS's actual data
Args:
domain_name(str): The zone name
submitted_records(list): List of records submitted from PDA datatable
Returns:
new_rrsets(list): List of rrests to be added
del_rrsets(list): List of rrests to be deleted
"""
# Create submitted rrsets from submitted records
submitted_rrsets = self.build_rrsets(domain_name, submitted_records)
current_app.logger.debug( current_app.logger.debug(
"postdata_for_new: {}".format(postdata_for_new)) "submitted_rrsets_data: \n{}".format(utils.pretty_json(submitted_rrsets)))
# Current domain's rrsets in PDNS
current_rrsets = self.get_rrsets(domain_name)
current_app.logger.debug("current_rrsets_data: \n{}".format(
utils.pretty_json(current_rrsets)))
# Remove comment's 'modified_at' key
# PDNS API always return the comments with modified_at
# info, we have to remove it to be able to do the dict
# comparison between current and submitted rrsets
for r in current_rrsets:
for comment in r['comments']:
del comment['modified_at']
# List of rrsets to be added
new_rrsets = {"rrsets": []}
for r in submitted_rrsets:
if r not in current_rrsets and r['type'] in Setting(
).get_records_allow_to_edit():
r['changetype'] = 'REPLACE'
new_rrsets["rrsets"].append(r)
# List of rrsets to be removed
del_rrsets = {"rrsets": []}
for r in current_rrsets:
if r not in submitted_rrsets and r['type'] in Setting(
).get_records_allow_to_edit():
r['changetype'] = 'DELETE'
del_rrsets["rrsets"].append(r)
current_app.logger.debug("new_rrsets: \n{}".format(utils.pretty_json(new_rrsets)))
current_app.logger.debug("del_rrsets: \n{}".format(utils.pretty_json(del_rrsets)))
return new_rrsets, del_rrsets
def apply(self, domain_name, submitted_records):
"""
Apply record changes to a domain. This function
will make 2 calls to the PDNS API to DELETE and
REPLACE records (rrests)
"""
current_app.logger.debug( current_app.logger.debug(
"postdata_for_delete: {}".format(postdata_for_delete)) "submitted_records: {}".format(submitted_records))
current_app.logger.info(
urljoin( # Get the list of rrsets to be added and deleted
self.PDNS_STATS_URL, self.API_EXTENDED_URL + new_rrsets, del_rrsets = self.compare(domain_name, submitted_records)
'/servers/localhost/zones/{0}'.format(domain)))
# records = []
# for r in deleted_records:
# r_name = r['name'].rstrip(
# '.') + '.' if self.NEW_SCHEMA else r['name']
# r_type = r['type']
# if self.PRETTY_IPV6_PTR: # only if activated
# if self.NEW_SCHEMA: # only if new schema
# if r_type == 'PTR': # only ptr
# if ':' in r['name']: # dirty ipv6 check
# r_name = dns.reversename.from_address(
# r['name']).to_text()
# record = {
# "name": r_name,
# "type": r_type,
# "changetype": "DELETE",
# "records": []
# }
# records.append(record)
# postdata_for_delete = {"rrsets": records}
# records = []
# for r in new_records:
# if self.NEW_SCHEMA:
# r_name = r['name'].rstrip('.') + '.'
# r_type = r['type']
# if self.PRETTY_IPV6_PTR: # only if activated
# if r_type == 'PTR': # only ptr
# if ':' in r['name']: # dirty ipv6 check
# r_name = r['name']
# Submit the changes to PDNS API
try: try:
headers = {} headers = {}
headers['X-API-Key'] = self.PDNS_API_KEY headers['X-API-Key'] = self.PDNS_API_KEY
if del_rrsets["rrsets"]:
jdata1 = utils.fetch_json(urljoin( jdata1 = utils.fetch_json(urljoin(
self.PDNS_STATS_URL, self.API_EXTENDED_URL + self.PDNS_STATS_URL, self.API_EXTENDED_URL +
'/servers/localhost/zones/{0}'.format(domain)), '/servers/localhost/zones/{0}'.format(domain_name)),
headers=headers, headers=headers,
method='PATCH', method='PATCH',
data=postdata_for_delete) data=del_rrsets)
jdata2 = utils.fetch_json(urljoin(
self.PDNS_STATS_URL, self.API_EXTENDED_URL +
'/servers/localhost/zones/{0}'.format(domain)),
headers=headers,
timeout=int(
Setting().get('pdns_api_timeout')),
method='PATCH',
data=postdata_for_new)
if 'error' in jdata1.keys(): if 'error' in jdata1.keys():
current_app.logger.error('Cannot apply record changes.') current_app.logger.error(
current_app.logger.debug(jdata1['error']) 'Cannot apply record changes with deleting rrsets step. PDNS error: {}'
.format(jdata1['error']))
return {'status': 'error', 'msg': jdata1['error']} return {'status': 'error', 'msg': jdata1['error']}
elif 'error' in jdata2.keys():
current_app.logger.error('Cannot apply record changes.') if new_rrsets["rrsets"]:
current_app.logger.debug(jdata2['error']) jdata2 = utils.fetch_json(
urljoin(
self.PDNS_STATS_URL, self.API_EXTENDED_URL +
'/servers/localhost/zones/{0}'.format(domain_name)),
headers=headers,
timeout=int(Setting().get('pdns_api_timeout')),
method='PATCH',
data=new_rrsets)
if 'error' in jdata2.keys():
current_app.logger.error(
'Cannot apply record changes with adding rrsets step. PDNS error: {}'
.format(jdata2['error']))
return {'status': 'error', 'msg': jdata2['error']} return {'status': 'error', 'msg': jdata2['error']}
else:
self.auto_ptr(domain, new_records, deleted_records) self.auto_ptr(domain_name, new_rrsets, del_rrsets)
self.update_db_serial(domain) self.update_db_serial(domain_name)
current_app.logger.info('Record was applied successfully.') current_app.logger.info('Record was applied successfully.')
return { return {'status': 'ok', 'msg': 'Record was applied successfully', 'data': (new_rrsets, del_rrsets)}
'status': 'ok',
'msg': 'Record was applied successfully'
}
except Exception as e: except Exception as e:
current_app.logger.error( current_app.logger.error(
"Cannot apply record changes to domain {0}. Error: {1}".format( "Cannot apply record changes to domain {0}. Error: {1}".format(
domain, e)) domain_name, e))
current_app.logger.debug(traceback.format_exc()) current_app.logger.debug(traceback.format_exc())
return { return {
'status': 'error', 'status': 'error',
@ -422,48 +416,93 @@ class Record(object):
'There was something wrong, please contact administrator' 'There was something wrong, please contact administrator'
} }
def auto_ptr(self, domain, new_records, deleted_records): def auto_ptr(self, domain_name, new_rrsets, del_rrsets):
""" """
Add auto-ptr records Add auto-ptr records
""" """
domain_obj = Domain.query.filter(Domain.name == domain).first() # Check if auto_ptr is enabled for this domain
domain_auto_ptr = DomainSetting.query.filter( auto_ptr_enabled = False
if Setting().get('auto_ptr'):
auto_ptr_enabled = True
else:
domain_obj = Domain.query.filter(Domain.name == domain_name).first()
domain_setting = DomainSetting.query.filter(
DomainSetting.domain == domain_obj).filter( DomainSetting.domain == domain_obj).filter(
DomainSetting.setting == 'auto_ptr').first() DomainSetting.setting == 'auto_ptr').first()
domain_auto_ptr = strtobool( auto_ptr_enabled = strtobool(
domain_auto_ptr.value) if domain_auto_ptr else False domain_setting.value) if domain_setting else False
system_auto_ptr = Setting().get('auto_ptr') # If it is enabled, we create/delete the PTR records automatically
if auto_ptr_enabled:
if system_auto_ptr or domain_auto_ptr:
try: try:
RECORD_TYPE_TO_PTR = ['A', 'AAAA']
new_rrsets = new_rrsets['rrsets']
del_rrsets = del_rrsets['rrsets']
if not new_rrsets and not del_rrsets:
msg = 'No changes detected. Skipping auto ptr...'
current_app.logger.info(msg)
return {'status': 'ok', 'msg': msg}
new_rrsets = [
r for r in new_rrsets if r['type'] in RECORD_TYPE_TO_PTR
]
del_rrsets = [
r for r in del_rrsets if r['type'] in RECORD_TYPE_TO_PTR
]
d = Domain() d = Domain()
for r in new_records: for r in new_rrsets:
if r['type'] in ['A', 'AAAA']: for record in r['records']:
r_name = r['name'] + '.' # Format the reverse record name
r_content = r['content'] # It is the reverse of forward record's content.
reverse_host_address = dns.reversename.from_address( reverse_host_address = dns.reversename.from_address(
r_content).to_text() record['content']).to_text()
# Create the reverse domain name in PDNS
domain_reverse_name = d.get_reverse_domain_name( domain_reverse_name = d.get_reverse_domain_name(
reverse_host_address) reverse_host_address)
d.create_reverse_domain(domain, domain_reverse_name) d.create_reverse_domain(domain_name,
self.name = dns.reversename.from_address( domain_reverse_name)
r_content).to_text().rstrip('.')
self.type = 'PTR' # Build the rrset for reverse zone updating
self.status = r['disabled'] rrset_data = [{
self.ttl = r['ttl'] "changetype":
self.data = r_name "REPLACE",
self.add(domain_reverse_name) "name":
for r in deleted_records: reverse_host_address,
if r['type'] in ['A', 'AAAA']: "ttl":
r_content = r['content'] r['ttl'],
"type":
"PTR",
"records": [{
"content": r['name'],
"disabled": record['disabled']
}],
"comments": []
}]
# Format the rrset
rrset = {"rrsets": rrset_data}
self.add(domain_reverse_name, rrset)
for r in del_rrsets:
for record in r['records']:
# Format the reverse record name
# It is the reverse of forward record's content.
reverse_host_address = dns.reversename.from_address( reverse_host_address = dns.reversename.from_address(
r_content).to_text() record['content']).to_text()
# Create the reverse domain name in PDNS
domain_reverse_name = d.get_reverse_domain_name( domain_reverse_name = d.get_reverse_domain_name(
reverse_host_address) reverse_host_address)
d.create_reverse_domain(domain_name,
domain_reverse_name)
# Delete the reverse zone
self.name = reverse_host_address self.name = reverse_host_address
self.type = 'PTR' self.type = 'PTR'
self.data = r_content self.data = record['content']
self.delete(domain_reverse_name) self.delete(domain_reverse_name)
return { return {
'status': 'ok', 'status': 'ok',
@ -472,7 +511,7 @@ class Record(object):
except Exception as e: except Exception as e:
current_app.logger.error( current_app.logger.error(
"Cannot update auto-ptr record changes to domain {0}. Error: {1}" "Cannot update auto-ptr record changes to domain {0}. Error: {1}"
.format(domain, e)) .format(domain_name, e))
current_app.logger.debug(traceback.format_exc()) current_app.logger.debug(traceback.format_exc())
return { return {
'status': 'status':

View File

@ -5,6 +5,7 @@ from distutils.version import StrictVersion
from flask import Blueprint, render_template, make_response, url_for, current_app, request, redirect, abort, jsonify from flask import Blueprint, render_template, make_response, url_for, current_app, request, redirect, abort, jsonify
from flask_login import login_required, current_user from flask_login import login_required, current_user
from ..lib.utils import pretty_json
from ..decorators import can_create_domain, operator_role_required, can_access_domain, can_configure_dnssec from ..decorators import can_create_domain, operator_role_required, can_access_domain, can_configure_dnssec
from ..models.user import User from ..models.user import User
from ..models.account import Account from ..models.account import Account
@ -27,17 +28,17 @@ domain_bp = Blueprint('domain',
@login_required @login_required
@can_access_domain @can_access_domain
def domain(domain_name): def domain(domain_name):
r = Record() # Validate the domain existing in the local DB
domain = Domain.query.filter(Domain.name == domain_name).first() domain = Domain.query.filter(Domain.name == domain_name).first()
if not domain: if not domain:
abort(404) abort(404)
# query domain info from PowerDNS API # Query domain's rrsets from PowerDNS API
zone_info = r.get_record_data(domain.name) rrsets = Record().get_rrsets(domain.name)
if zone_info: current_app.logger.debug("Fetched rrests: \n{}".format(pretty_json(rrsets)))
jrecords = zone_info['records']
else: # API server might be down, misconfigured
# can not get any record, API server might be down if not rrsets:
abort(500) abort(500)
quick_edit = Setting().get('record_quick_edit') quick_edit = Setting().get('record_quick_edit')
@ -49,45 +50,40 @@ def domain(domain_name):
ttl_options = Setting().get_ttl_options() ttl_options = Setting().get_ttl_options()
records = [] records = []
# Render the "records" to display in HTML datatable
#
# BUG: If we have multiple records with the same name
# and each record has its own comment, the display of
# [record-comment] may not consistent because PDNS API
# returns the rrsets (records, comments) has different
# order than its database records.
# TODO:
# - Find a way to make it consistent, or
# - Only allow one comment for that case
if StrictVersion(Setting().get('pdns_version')) >= StrictVersion('4.0.0'): if StrictVersion(Setting().get('pdns_version')) >= StrictVersion('4.0.0'):
for jr in jrecords: for r in rrsets:
if jr['type'] in records_allow_to_edit: if r['type'] in records_allow_to_edit:
for subrecord in jr['records']: index = 0
record = RecordEntry(name=jr['name'], for record in r['records']:
type=jr['type'], record_entry = RecordEntry(
status='Disabled' if name=r['name'].rstrip('.'),
subrecord['disabled'] else 'Active', type=r['type'],
ttl=jr['ttl'], status='Disabled' if record['disabled'] else 'Active',
data=subrecord['content'], ttl=r['ttl'],
comment=jr['comment_data']['content'], data=record['content'],
comment=r['comments'][index]['content'] if r['comments'] else '',
is_allowed_edit=True) is_allowed_edit=True)
records.append(record) index += 1
if not re.search('ip6\.arpa|in-addr\.arpa$', domain_name): records.append(record_entry)
editable_records = forward_records_allow_to_edit else:
else: # Unsupported version
editable_records = reverse_records_allow_to_edit abort(500)
return render_template('domain.html',
domain=domain,
records=records,
editable_records=editable_records,
quick_edit=quick_edit,
ttl_options=ttl_options)
else:
for jr in jrecords:
if jr['type'] in records_allow_to_edit:
record = RecordEntry(
name=jr['name'],
type=jr['type'],
status='Disabled' if jr['disabled'] else 'Active',
ttl=jr['ttl'],
data=jr['content'],
comment=jr['comment_data']['content'],
is_allowed_edit=True)
records.append(record)
if not re.search('ip6\.arpa|in-addr\.arpa$', domain_name): if not re.search('ip6\.arpa|in-addr\.arpa$', domain_name):
editable_records = forward_records_allow_to_edit editable_records = forward_records_allow_to_edit
else: else:
editable_records = reverse_records_allow_to_edit editable_records = reverse_records_allow_to_edit
return render_template('domain.html', return render_template('domain.html',
domain=domain, domain=domain,
records=records, records=records,
@ -354,19 +350,16 @@ def change_account(domain_name):
@login_required @login_required
@can_access_domain @can_access_domain
def record_apply(domain_name): def record_apply(domain_name):
#TODO: filter removed records / name modified records.
try: try:
jdata = request.json jdata = request.json
submitted_serial = jdata['serial'] submitted_serial = jdata['serial']
submitted_record = jdata['record'] submitted_record = jdata['record']
domain = Domain.query.filter(Domain.name == domain_name).first() domain = Domain.query.filter(Domain.name == domain_name).first()
current_app.logger.debug(
'Your submitted serial: {0}'.format(submitted_serial)) if domain:
current_app.logger.debug('Current domain serial: {0}'.format( current_app.logger.debug('Current domain serial: {0}'.format(
domain.serial)) domain.serial))
if domain:
if int(submitted_serial) != domain.serial: if int(submitted_serial) != domain.serial:
return make_response( return make_response(
jsonify({ jsonify({
@ -384,26 +377,17 @@ def record_apply(domain_name):
'Domain name {0} does not exist'.format(domain_name) 'Domain name {0} does not exist'.format(domain_name)
}), 404) }), 404)
# Modify the record's comment data. We append
# the "current_user" into account field as it
# a field with user-defined meaning
for sr in submitted_record:
if sr.get('record_comment'):
sr['comment_data'] = [{
'content': sr['record_comment'],
'account': current_user.username
}]
else:
sr['comment_data'] = []
r = Record() r = Record()
result = r.apply(domain_name, submitted_record) result = r.apply(domain_name, submitted_record)
if result['status'] == 'ok': if result['status'] == 'ok':
jdata.pop('_csrf_token',
None) # don't store csrf token in the history.
history = History( history = History(
msg='Apply record changes to domain {0}'.format(domain_name), msg='Apply record changes to domain {0}'.format(domain_name),
detail=str(json.dumps(jdata)), detail=str(
json.dumps({
"domain": domain_name,
"add_rrests": result['data'][0]['rrsets'],
"del_rrests": result['data'][1]['rrsets']
})),
created_by=current_user.username) created_by=current_user.username)
history.add() history.add()
return make_response(jsonify(result), 200) return make_response(jsonify(result), 200)

View File

@ -335,6 +335,9 @@
mx_server = modal.find('#mx_server').val(); mx_server = modal.find('#mx_server').val();
mx_priority = modal.find('#mx_priority').val(); mx_priority = modal.find('#mx_priority').val();
data = mx_priority + " " + mx_server; data = mx_priority + " " + mx_server;
if (data && !data.endsWith('.')) {
data = data + '.'
}
record_data.val(data); record_data.val(data);
modal.modal('hide'); modal.modal('hide');
}) })
@ -370,6 +373,9 @@
srv_port = modal.find('#srv_port').val(); srv_port = modal.find('#srv_port').val();
srv_target = modal.find('#srv_target').val(); srv_target = modal.find('#srv_target').val();
data = srv_priority + " " + srv_weight + " " + srv_port + " " + srv_target; data = srv_priority + " " + srv_weight + " " + srv_port + " " + srv_target;
if (data && !data.endsWith('.')) {
data = data + '.'
}
record_data.val(data); record_data.val(data);
modal.modal('hide'); modal.modal('hide');
}) })

View File

@ -313,6 +313,9 @@
mx_server = modal.find('#mx_server').val(); mx_server = modal.find('#mx_server').val();
mx_priority = modal.find('#mx_priority').val(); mx_priority = modal.find('#mx_priority').val();
data = mx_priority + " " + mx_server; data = mx_priority + " " + mx_server;
if (data && !data.endsWith('.')) {
data = data + '.'
}
record_data.val(data); record_data.val(data);
modal.modal('hide'); modal.modal('hide');
}) })
@ -348,6 +351,9 @@
srv_port = modal.find('#srv_port').val(); srv_port = modal.find('#srv_port').val();
srv_target = modal.find('#srv_target').val(); srv_target = modal.find('#srv_target').val();
data = srv_priority + " " + srv_weight + " " + srv_port + " " + srv_target; data = srv_priority + " " + srv_weight + " " + srv_port + " " + srv_target;
if (data && !data.endsWith('.')) {
data = data + '.'
}
record_data.val(data); record_data.val(data);
modal.modal('hide'); modal.modal('hide');
}) })