mirror of
				https://github.com/github/octodns.git
				synced 2024-05-11 05:55:00 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			369 lines
		
	
	
		
			13 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			369 lines
		
	
	
		
			13 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
#
 | 
						|
#
 | 
						|
#
 | 
						|
 | 
						|
from __future__ import absolute_import, division, print_function, \
 | 
						|
    unicode_literals
 | 
						|
 | 
						|
from logging import getLogger
 | 
						|
from itertools import chain
 | 
						|
from collections import OrderedDict, defaultdict
 | 
						|
from nsone import NSONE
 | 
						|
from nsone.rest.errors import RateLimitException, ResourceException
 | 
						|
from pycountry_convert import country_alpha2_to_continent_code
 | 
						|
from time import sleep
 | 
						|
 | 
						|
from six import text_type
 | 
						|
 | 
						|
from ..record import Record
 | 
						|
from .base import BaseProvider
 | 
						|
 | 
						|
 | 
						|
class Ns1Provider(BaseProvider):
 | 
						|
    '''
 | 
						|
    Ns1 provider
 | 
						|
 | 
						|
    nsone:
 | 
						|
        class: octodns.provider.ns1.Ns1Provider
 | 
						|
        api_key: env/NS1_API_KEY
 | 
						|
    '''
 | 
						|
    SUPPORTS_GEO = True
 | 
						|
    SUPPORTS_DYNAMIC = False
 | 
						|
    SUPPORTS = set(('A', 'AAAA', 'ALIAS', 'CAA', 'CNAME', 'MX', 'NAPTR',
 | 
						|
                    'NS', 'PTR', 'SPF', 'SRV', 'TXT'))
 | 
						|
 | 
						|
    ZONE_NOT_FOUND_MESSAGE = 'server error: zone not found'
 | 
						|
 | 
						|
    def __init__(self, id, api_key, *args, **kwargs):
 | 
						|
        self.log = getLogger('Ns1Provider[{}]'.format(id))
 | 
						|
        self.log.debug('__init__: id=%s, api_key=***', id)
 | 
						|
        super(Ns1Provider, self).__init__(id, *args, **kwargs)
 | 
						|
        self._client = NSONE(apiKey=api_key)
 | 
						|
 | 
						|
    def _data_for_A(self, _type, record):
 | 
						|
        # record meta (which would include geo information is only
 | 
						|
        # returned when getting a record's detail, not from zone detail
 | 
						|
        geo = defaultdict(list)
 | 
						|
        data = {
 | 
						|
            'ttl': record['ttl'],
 | 
						|
            'type': _type,
 | 
						|
        }
 | 
						|
        values, codes = [], []
 | 
						|
        if 'answers' not in record:
 | 
						|
            values = record['short_answers']
 | 
						|
        for answer in record.get('answers', []):
 | 
						|
            meta = answer.get('meta', {})
 | 
						|
            if meta:
 | 
						|
                # country + state and country + province are allowed
 | 
						|
                # in that case though, supplying a state/province would
 | 
						|
                # be redundant since the country would supercede in when
 | 
						|
                # resolving the record.  it is syntactically valid, however.
 | 
						|
                country = meta.get('country', [])
 | 
						|
                us_state = meta.get('us_state', [])
 | 
						|
                ca_province = meta.get('ca_province', [])
 | 
						|
                for cntry in country:
 | 
						|
                    con = country_alpha2_to_continent_code(cntry)
 | 
						|
                    key = '{}-{}'.format(con, cntry)
 | 
						|
                    geo[key].extend(answer['answer'])
 | 
						|
                for state in us_state:
 | 
						|
                    key = 'NA-US-{}'.format(state)
 | 
						|
                    geo[key].extend(answer['answer'])
 | 
						|
                for province in ca_province:
 | 
						|
                    key = 'NA-CA-{}'.format(province)
 | 
						|
                    geo[key].extend(answer['answer'])
 | 
						|
                for code in meta.get('iso_region_code', []):
 | 
						|
                    key = code
 | 
						|
                    geo[key].extend(answer['answer'])
 | 
						|
            else:
 | 
						|
                values.extend(answer['answer'])
 | 
						|
                codes.append([])
 | 
						|
        values = [text_type(x) for x in values]
 | 
						|
        geo = OrderedDict(
 | 
						|
            {text_type(k): [text_type(x) for x in v] for k, v in geo.items()}
 | 
						|
        )
 | 
						|
        data['values'] = values
 | 
						|
        data['geo'] = geo
 | 
						|
        return data
 | 
						|
 | 
						|
    _data_for_AAAA = _data_for_A
 | 
						|
 | 
						|
    def _data_for_SPF(self, _type, record):
 | 
						|
        values = [v.replace(';', '\\;') for v in record['short_answers']]
 | 
						|
        return {
 | 
						|
            'ttl': record['ttl'],
 | 
						|
            'type': _type,
 | 
						|
            'values': values
 | 
						|
        }
 | 
						|
 | 
						|
    _data_for_TXT = _data_for_SPF
 | 
						|
 | 
						|
    def _data_for_CAA(self, _type, record):
 | 
						|
        values = []
 | 
						|
        for answer in record['short_answers']:
 | 
						|
            flags, tag, value = answer.split(' ', 2)
 | 
						|
            values.append({
 | 
						|
                'flags': flags,
 | 
						|
                'tag': tag,
 | 
						|
                'value': value,
 | 
						|
            })
 | 
						|
        return {
 | 
						|
            'ttl': record['ttl'],
 | 
						|
            'type': _type,
 | 
						|
            'values': values,
 | 
						|
        }
 | 
						|
 | 
						|
    def _data_for_CNAME(self, _type, record):
 | 
						|
        try:
 | 
						|
            value = record['short_answers'][0]
 | 
						|
        except IndexError:
 | 
						|
            value = None
 | 
						|
        return {
 | 
						|
            'ttl': record['ttl'],
 | 
						|
            'type': _type,
 | 
						|
            'value': value,
 | 
						|
        }
 | 
						|
 | 
						|
    _data_for_ALIAS = _data_for_CNAME
 | 
						|
    _data_for_PTR = _data_for_CNAME
 | 
						|
 | 
						|
    def _data_for_MX(self, _type, record):
 | 
						|
        values = []
 | 
						|
        for answer in record['short_answers']:
 | 
						|
            preference, exchange = answer.split(' ', 1)
 | 
						|
            values.append({
 | 
						|
                'preference': preference,
 | 
						|
                'exchange': exchange,
 | 
						|
            })
 | 
						|
        return {
 | 
						|
            'ttl': record['ttl'],
 | 
						|
            'type': _type,
 | 
						|
            'values': values,
 | 
						|
        }
 | 
						|
 | 
						|
    def _data_for_NAPTR(self, _type, record):
 | 
						|
        values = []
 | 
						|
        for answer in record['short_answers']:
 | 
						|
            order, preference, flags, service, regexp, replacement = \
 | 
						|
                answer.split(' ', 5)
 | 
						|
            values.append({
 | 
						|
                'flags': flags,
 | 
						|
                'order': order,
 | 
						|
                'preference': preference,
 | 
						|
                'regexp': regexp,
 | 
						|
                'replacement': replacement,
 | 
						|
                'service': service,
 | 
						|
            })
 | 
						|
        return {
 | 
						|
            'ttl': record['ttl'],
 | 
						|
            'type': _type,
 | 
						|
            'values': values,
 | 
						|
        }
 | 
						|
 | 
						|
    def _data_for_NS(self, _type, record):
 | 
						|
        return {
 | 
						|
            'ttl': record['ttl'],
 | 
						|
            'type': _type,
 | 
						|
            'values': [a if a.endswith('.') else '{}.'.format(a)
 | 
						|
                       for a in record['short_answers']],
 | 
						|
        }
 | 
						|
 | 
						|
    def _data_for_SRV(self, _type, record):
 | 
						|
        values = []
 | 
						|
        for answer in record['short_answers']:
 | 
						|
            priority, weight, port, target = answer.split(' ', 3)
 | 
						|
            values.append({
 | 
						|
                'priority': priority,
 | 
						|
                'weight': weight,
 | 
						|
                'port': port,
 | 
						|
                'target': target,
 | 
						|
            })
 | 
						|
        return {
 | 
						|
            'ttl': record['ttl'],
 | 
						|
            'type': _type,
 | 
						|
            'values': values,
 | 
						|
        }
 | 
						|
 | 
						|
    def populate(self, zone, target=False, lenient=False):
 | 
						|
        self.log.debug('populate: name=%s, target=%s, lenient=%s',
 | 
						|
                       zone.name,
 | 
						|
                       target, lenient)
 | 
						|
 | 
						|
        try:
 | 
						|
            nsone_zone = self._client.loadZone(zone.name[:-1])
 | 
						|
            records = nsone_zone.data['records']
 | 
						|
 | 
						|
            # change answers for certain types to always be absolute
 | 
						|
            for record in records:
 | 
						|
                if record['type'] in ['ALIAS', 'CNAME', 'MX', 'NS', 'PTR',
 | 
						|
                                      'SRV']:
 | 
						|
                    for i, a in enumerate(record['short_answers']):
 | 
						|
                        if not a.endswith('.'):
 | 
						|
                            record['short_answers'][i] = '{}.'.format(a)
 | 
						|
 | 
						|
            geo_records = nsone_zone.search(has_geo=True)
 | 
						|
            exists = True
 | 
						|
        except ResourceException as e:
 | 
						|
            if e.message != self.ZONE_NOT_FOUND_MESSAGE:
 | 
						|
                raise
 | 
						|
            records = []
 | 
						|
            geo_records = []
 | 
						|
            exists = False
 | 
						|
 | 
						|
        before = len(zone.records)
 | 
						|
        # geo information isn't returned from the main endpoint, so we need
 | 
						|
        # to query for all records with geo information
 | 
						|
        zone_hash = {}
 | 
						|
        for record in chain(records, geo_records):
 | 
						|
            _type = record['type']
 | 
						|
            if _type not in self.SUPPORTS:
 | 
						|
                continue
 | 
						|
            data_for = getattr(self, '_data_for_{}'.format(_type))
 | 
						|
            name = zone.hostname_from_fqdn(record['domain'])
 | 
						|
            record = Record.new(zone, name, data_for(_type, record),
 | 
						|
                                source=self, lenient=lenient)
 | 
						|
            zone_hash[(_type, name)] = record
 | 
						|
        [zone.add_record(r, lenient=lenient) for r in zone_hash.values()]
 | 
						|
        self.log.info('populate:   found %s records, exists=%s',
 | 
						|
                      len(zone.records) - before, exists)
 | 
						|
        return exists
 | 
						|
 | 
						|
    def _params_for_A(self, record):
 | 
						|
        params = {'answers': record.values, 'ttl': record.ttl}
 | 
						|
        if hasattr(record, 'geo'):
 | 
						|
            # purposefully set non-geo answers to have an empty meta,
 | 
						|
            # so that we know we did this on purpose if/when troubleshooting
 | 
						|
            params['answers'] = [{"answer": [x], "meta": {}}
 | 
						|
                                 for x in record.values]
 | 
						|
            has_country = False
 | 
						|
            for iso_region, target in record.geo.items():
 | 
						|
                key = 'iso_region_code'
 | 
						|
                value = iso_region
 | 
						|
                if not has_country and \
 | 
						|
                   len(value.split('-')) > 1:  # pragma: nocover
 | 
						|
                    has_country = True
 | 
						|
                for answer in target.values:
 | 
						|
                    params['answers'].append(
 | 
						|
                        {
 | 
						|
                            'answer': [answer],
 | 
						|
                            'meta': {key: [value]},
 | 
						|
                        },
 | 
						|
                    )
 | 
						|
            params['filters'] = []
 | 
						|
            if has_country:
 | 
						|
                params['filters'].append(
 | 
						|
                    {"filter": "shuffle", "config": {}}
 | 
						|
                )
 | 
						|
                params['filters'].append(
 | 
						|
                    {"filter": "geotarget_country", "config": {}}
 | 
						|
                )
 | 
						|
                params['filters'].append(
 | 
						|
                    {"filter": "select_first_n",
 | 
						|
                     "config": {"N": 1}}
 | 
						|
                )
 | 
						|
        self.log.debug("params for A: %s", params)
 | 
						|
        return params
 | 
						|
 | 
						|
    _params_for_AAAA = _params_for_A
 | 
						|
    _params_for_NS = _params_for_A
 | 
						|
 | 
						|
    def _params_for_SPF(self, record):
 | 
						|
        # NS1 seems to be the only provider that doesn't want things
 | 
						|
        # escaped in values so we have to strip them here and add
 | 
						|
        # them when going the other way
 | 
						|
        values = [v.replace('\\;', ';') for v in record.values]
 | 
						|
        return {'answers': values, 'ttl': record.ttl}
 | 
						|
 | 
						|
    _params_for_TXT = _params_for_SPF
 | 
						|
 | 
						|
    def _params_for_CAA(self, record):
 | 
						|
        values = [(v.flags, v.tag, v.value) for v in record.values]
 | 
						|
        return {'answers': values, 'ttl': record.ttl}
 | 
						|
 | 
						|
    def _params_for_CNAME(self, record):
 | 
						|
        return {'answers': [record.value], 'ttl': record.ttl}
 | 
						|
 | 
						|
    _params_for_ALIAS = _params_for_CNAME
 | 
						|
    _params_for_PTR = _params_for_CNAME
 | 
						|
 | 
						|
    def _params_for_MX(self, record):
 | 
						|
        values = [(v.preference, v.exchange) for v in record.values]
 | 
						|
        return {'answers': values, 'ttl': record.ttl}
 | 
						|
 | 
						|
    def _params_for_NAPTR(self, record):
 | 
						|
        values = [(v.order, v.preference, v.flags, v.service, v.regexp,
 | 
						|
                   v.replacement) for v in record.values]
 | 
						|
        return {'answers': values, 'ttl': record.ttl}
 | 
						|
 | 
						|
    def _params_for_SRV(self, record):
 | 
						|
        values = [(v.priority, v.weight, v.port, v.target)
 | 
						|
                  for v in record.values]
 | 
						|
        return {'answers': values, 'ttl': record.ttl}
 | 
						|
 | 
						|
    def _get_name(self, record):
 | 
						|
        return record.fqdn[:-1] if record.name == '' else record.name
 | 
						|
 | 
						|
    def _apply_Create(self, nsone_zone, change):
 | 
						|
        new = change.new
 | 
						|
        name = self._get_name(new)
 | 
						|
        _type = new._type
 | 
						|
        params = getattr(self, '_params_for_{}'.format(_type))(new)
 | 
						|
        meth = getattr(nsone_zone, 'add_{}'.format(_type))
 | 
						|
        try:
 | 
						|
            meth(name, **params)
 | 
						|
        except RateLimitException as e:
 | 
						|
            period = float(e.period)
 | 
						|
            self.log.warn('_apply_Create: rate limit encountered, pausing '
 | 
						|
                          'for %ds and trying again', period)
 | 
						|
            sleep(period)
 | 
						|
            meth(name, **params)
 | 
						|
 | 
						|
    def _apply_Update(self, nsone_zone, change):
 | 
						|
        existing = change.existing
 | 
						|
        name = self._get_name(existing)
 | 
						|
        _type = existing._type
 | 
						|
        record = nsone_zone.loadRecord(name, _type)
 | 
						|
        new = change.new
 | 
						|
        params = getattr(self, '_params_for_{}'.format(_type))(new)
 | 
						|
        try:
 | 
						|
            record.update(**params)
 | 
						|
        except RateLimitException as e:
 | 
						|
            period = float(e.period)
 | 
						|
            self.log.warn('_apply_Update: rate limit encountered, pausing '
 | 
						|
                          'for %ds and trying again', period)
 | 
						|
            sleep(period)
 | 
						|
            record.update(**params)
 | 
						|
 | 
						|
    def _apply_Delete(self, nsone_zone, change):
 | 
						|
        existing = change.existing
 | 
						|
        name = self._get_name(existing)
 | 
						|
        _type = existing._type
 | 
						|
        record = nsone_zone.loadRecord(name, _type)
 | 
						|
        try:
 | 
						|
            record.delete()
 | 
						|
        except RateLimitException as e:
 | 
						|
            period = float(e.period)
 | 
						|
            self.log.warn('_apply_Delete: rate limit encountered, pausing '
 | 
						|
                          'for %ds and trying again', period)
 | 
						|
            sleep(period)
 | 
						|
            record.delete()
 | 
						|
 | 
						|
    def _apply(self, plan):
 | 
						|
        desired = plan.desired
 | 
						|
        changes = plan.changes
 | 
						|
        self.log.debug('_apply: zone=%s, len(changes)=%d', desired.name,
 | 
						|
                       len(changes))
 | 
						|
 | 
						|
        domain_name = desired.name[:-1]
 | 
						|
        try:
 | 
						|
            nsone_zone = self._client.loadZone(domain_name)
 | 
						|
        except ResourceException as e:
 | 
						|
            if e.message != self.ZONE_NOT_FOUND_MESSAGE:
 | 
						|
                raise
 | 
						|
            self.log.debug('_apply:   no matching zone, creating')
 | 
						|
            nsone_zone = self._client.createZone(domain_name)
 | 
						|
 | 
						|
        for change in changes:
 | 
						|
            class_name = change.__class__.__name__
 | 
						|
            getattr(self, '_apply_{}'.format(class_name))(nsone_zone,
 | 
						|
                                                          change)
 |