276 lines
7.9 KiB
Python
Raw Normal View History

import logging
2015-12-13 16:34:12 +07:00
import re
import json
import requests
import ipaddress
import idna
2016-07-13 21:33:21 +07:00
2020-05-25 14:16:33 +01:00
from collections.abc import Iterable
from distutils.version import StrictVersion
2018-03-30 13:49:35 +07:00
from urllib.parse import urlparse
2018-03-30 13:49:35 +07:00
2015-12-13 16:34:12 +07:00
def auth_from_url(url):
auth = None
2018-03-30 13:49:35 +07:00
parsed_url = urlparse(url).netloc
2015-12-13 16:34:12 +07:00
if '@' in parsed_url:
auth = parsed_url.split('@')[0].split(':')
auth = requests.auth.HTTPBasicAuth(auth[0], auth[1])
return auth
def fetch_remote(remote_url,
method='GET',
data=None,
accept=None,
params=None,
timeout=None,
headers=None,
verify=True):
2015-12-13 16:34:12 +07:00
if data is not None and type(data) != str:
data = json.dumps(data)
verify = bool(verify) # enforce type boolean
2015-12-13 16:34:12 +07:00
our_headers = {
'user-agent': 'powerdnsadmin/0',
'pragma': 'no-cache',
'cache-control': 'no-cache'
}
if accept is not None:
our_headers['accept'] = accept
if headers is not None:
our_headers.update(headers)
r = requests.request(method,
remote_url,
headers=headers,
verify=verify,
auth=auth_from_url(remote_url),
timeout=timeout,
data=data,
params=params)
logging.debug(
'Querying remote server "{0}" ({1}) finished with code {2} (took {3}s)'
.format(remote_url, method, r.status_code, r.elapsed.total_seconds()))
2015-12-13 16:34:12 +07:00
try:
if r.status_code not in (200, 201, 204, 400, 409, 422):
2015-12-13 16:34:12 +07:00
r.raise_for_status()
except Exception as e:
msg = "Returned status {0} and content {1}".format(r.status_code, r.text)
raise RuntimeError('Error while fetching {0}. {1}'.format(
remote_url, msg))
2015-12-13 16:34:12 +07:00
return r
def fetch_json(remote_url,
method='GET',
data=None,
params=None,
headers=None,
timeout=None,
verify=True):
r = fetch_remote(remote_url,
method=method,
data=data,
params=params,
headers=headers,
timeout=timeout,
verify=verify,
2018-03-28 01:52:48 +02:00
accept='application/json; q=1')
2015-12-13 16:34:12 +07:00
if method == "DELETE":
return True
if r.status_code == 204:
return {}
2019-12-04 11:50:46 +07:00
elif r.status_code == 409:
return {
'error': 'Resource already exists or conflict',
'http_code': r.status_code
}
2015-12-13 16:34:12 +07:00
try:
assert ('json' in r.headers['content-type'])
2015-12-13 16:34:12 +07:00
except Exception as e:
raise RuntimeError(
'Error while fetching {0}'.format(remote_url)) from e
2015-12-13 16:34:12 +07:00
# don't use r.json here, as it will read from r.text, which will trigger
# content encoding auto-detection in almost all cases, WHICH IS EXTREMELY
# SLOOOOOOOOOOOOOOOOOOOOOOW. just don't.
data = None
try:
2018-03-30 13:49:35 +07:00
data = json.loads(r.content.decode('utf-8'))
2021-10-30 21:06:44 +02:00
except UnicodeDecodeError:
# If the decoding fails, switch to slower but probably working .json()
try:
logging.warning("UTF-8 content.decode failed, switching to slower .json method")
data = r.json()
except Exception as e:
raise e
2018-03-30 13:49:35 +07:00
except Exception as e:
raise RuntimeError(
'Error while loading JSON data from {0}'.format(remote_url)) from e
2015-12-13 16:34:12 +07:00
return data
def display_record_name(data):
record_name, domain_name = data
if record_name == domain_name:
return '@'
2015-12-13 16:34:12 +07:00
else:
return re.sub('\.{}$'.format(domain_name), '', record_name)
2018-03-30 13:49:35 +07:00
2015-12-13 16:34:12 +07:00
def display_master_name(data):
"""
input data: "[u'127.0.0.1', u'8.8.8.8']"
"""
matches = re.findall(r'\'(.+?)\'', data)
return ", ".join(matches)
2018-03-30 13:49:35 +07:00
def format_zone_type(data):
"""Formats the given zone type for modern social standards."""
data = str(data).lower()
if data == 'master':
data = 'primary'
elif data == 'slave':
data = 'secondary'
return data.title()
2015-12-13 16:34:12 +07:00
def display_time(amount, units='s', remove_seconds=True):
"""
Convert timestamp to normal time format
"""
amount = int(amount)
INTERVALS = [(lambda mlsec: divmod(mlsec, 1000), 'ms'),
(lambda seconds: divmod(seconds, 60), 's'),
(lambda minutes: divmod(minutes, 60), 'm'),
(lambda hours: divmod(hours, 24), 'h'),
(lambda days: divmod(days, 7), 'D'),
(lambda weeks: divmod(weeks, 4), 'W'),
(lambda years: divmod(years, 12), 'M'),
(lambda decades: divmod(decades, 10), 'Y')]
2015-12-13 16:34:12 +07:00
for index_start, (interval, unit) in enumerate(INTERVALS):
if unit == units:
break
amount_abrev = []
last_index = 0
amount_temp = amount
for index, (formula,
abrev) in enumerate(INTERVALS[index_start:len(INTERVALS)]):
2015-12-13 16:34:12 +07:00
divmod_result = formula(amount_temp)
amount_temp = divmod_result[0]
amount_abrev.append((divmod_result[1], abrev))
if divmod_result[1] > 0:
last_index = index
amount_abrev_partial = amount_abrev[0:last_index + 1]
2015-12-13 16:34:12 +07:00
amount_abrev_partial.reverse()
final_string = ''
for amount, abrev in amount_abrev_partial:
final_string += str(amount) + abrev + ' '
if remove_seconds and 'm' in final_string:
final_string = final_string[:final_string.rfind(' ')]
return final_string[:final_string.rfind(' ')]
return final_string
2018-03-30 13:49:35 +07:00
def pdns_api_extended_uri(version):
"""
Check the pdns version
"""
if StrictVersion(version) >= StrictVersion('4.0.0'):
return "/api/v1"
else:
return ""
2016-07-13 21:33:21 +07:00
2018-03-30 13:49:35 +07:00
def display_setting_state(value):
if value == 1:
return "ON"
elif value == 0:
return "OFF"
else:
return "UNKNOWN"
def validate_ipaddress(address):
try:
ip = ipaddress.ip_address(address)
except ValueError:
pass
else:
if isinstance(ip, (ipaddress.IPv4Address, ipaddress.IPv6Address)):
return [ip]
return []
2019-12-14 14:47:21 +07:00
def pretty_json(data):
return json.dumps(data, sort_keys=True, indent=4)
2020-05-25 14:16:33 +01:00
def ensure_list(l):
if not l:
l = []
elif not isinstance(l, Iterable) or isinstance(l, str):
l = [l]
yield from l
2020-06-19 08:47:51 +07:00
2022-12-23 08:23:14 +08:00
def pretty_domain_name(domain_name):
# Add a debugging statement to print out the domain name
print("Received zone name:", domain_name)
2022-12-23 08:23:14 +08:00
# Check if the domain name is encoded using Punycode
if domain_name.endswith('.xn--'):
try:
# Decode the domain name using the idna library
domain_name = idna.decode(domain_name)
except Exception as e:
# If the decoding fails, raise an exception with more information
raise Exception('Cannot decode IDN zone: {}'.format(e))
2022-12-23 08:23:14 +08:00
# Return the "pretty" version of the zone name
2022-12-23 08:23:14 +08:00
return domain_name
def to_idna(value, action):
splits = value.split('.')
result = []
if action == 'encode':
for split in splits:
try:
# Try encoding to idna
if not split.startswith('_') and not split.startswith('-'):
result.append(idna.encode(split).decode())
else:
result.append(split)
except idna.IDNAError:
result.append(split)
elif action == 'decode':
for split in splits:
if not split.startswith('_') and not split.startswith('--'):
result.append(idna.decode(split))
else:
result.append(split)
else:
raise Exception('No valid action received')
return '.'.join(result)
def format_datetime(value, format_str="%Y-%m-%d %I:%M %p"):
"""Format a date time to (Default): YYYY-MM-DD HH:MM P"""
if value is None:
return ""
return value.strftime(format_str)