# Copyright 2014 OpenStack LLC.
# All Rights Reserved.
#
#    Licensed under the Apache License, Version 2.0 (the "License"); you may
#    not use this file except in compliance with the License. You may obtain
#    a copy of the License at
#
#         http://www.apache.org/licenses/LICENSE-2.0
#
#    Unless required by applicable law or agreed to in writing, software
#    distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
#    WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
#    License for the specific language governing permissions and limitations
#    under the License.
import re
import unittest
import os
import copy
from unittest import mock

from infoblox_client import objects
import infoblox_client.exceptions as ib_ex

REC = 'ZG5zLmJpbmRfbXgkLjQuY29tLm15X3pvbmUuZGVtby5teC5kZW1vLm15X3pvbmUuY29tLjE'

DEFAULT_HOST_RECORD = {
    '_ref': 'record:host/ZG5zLmhvc3QkLl9kZWZhdWx0LmNvbS5nbG9iYWwuY22NA'
            ':test_host_name.testsubnet.cloud.global.com/default',
    'ipv4addrs': [{
        'configure_for_dhcp': False,
        '_ref': 'record:host_ipv4addr/lMmQ3ZjkuM4Zj5Mi00Y2:22.0.0.2/'
                'test_host_name.testsubnet.cloud.global.com/default',
        'ipv4addr': '22.0.0.2',
        'mac': 'fa:16:3e:29:87:70',
        'host': '2c8f8e97-0d92-4cac-a350-096ff2b79.cloud.global.com'}],
    'extattrs': {
        'Account': {'value': '8a21c40495f04f30a1b2dc6fd1d9ed1a'},
        'Cloud API Owned': {'value': 'True'},
        'VM ID': {'value': 'None'},
        'IP Type': {'value': 'Fixed'},
        'CMP Type': {'value': 'OpenStack'},
        'Port ID': {'value': '136ef9ad-9c88-41ea-9fa6-bd48d8ec789a'},
        'Tenant ID': {'value': '00fd80791dee4112bb538c872b206d4c'}}
}

DEFAULT_MX_RECORD = {
    '_ref': 'record:mx/%s'
            'mx.demo.my_zone.com/my_dns_view' % REC,
    'view': 'my_dns_view',
    'name': 'mx.demo.my_zone.com',
    'preference': '1',
    'mail_exchanger': 'demo.my_zone.com'
}

DEFAULT_TXT_RECORD = {
    '_ref': 'record:txt/%s'
            'text_test.my_zone.com/my_dns_view' % REC,
    'view': 'my_dns_view',
    'name': 'text_test.my_zone.com',
    'text': 'hello_test'
}


class TestObjects(unittest.TestCase):
    @staticmethod
    def _create_infoblox_csv():
        file_data = [
            'header-network,address,netmask,comment'
            'network,10.10.10.0,255.255.255.0,test1',
            'network,10.10.11.0,255.255.255.0,test2'
        ]
        with open('tests/ibx_networks.csv', 'w') as fh:
            fh.write('\n'.join(file_data))
            fh.close()

    @staticmethod
    def _delete_infoblox_csv():
        if os.path.exists('tests/ibx_networks.csv'):
            os.unlink('tests/ibx_networks.csv')

    def _mock_connector(self, get_object=None, create_object=None,
                        delete_object=None):
        connector = mock.Mock()
        connector.get_object.return_value = get_object
        connector.create_object.return_value = create_object
        connector.delete_object.return_value = delete_object
        return connector

    def test_search_network(self):
        connector = self._mock_connector()

        objects.Network.search(connector,
                               network_view='some-view',
                               cidr='192.68.1.0/20')
        connector.get_object.assert_called_once_with(
            'network',
            {'network_view': 'some-view', 'network': '192.68.1.0/20'},
            extattrs=None, force_proxy=False, return_fields=mock.ANY,
            paging=False, max_results=None)

    def test_search_network_v6(self):
        connector = self._mock_connector()

        objects.Network.search(connector,
                               network_view='some-view',
                               cidr='fffe:2312::/64')
        connector.get_object.assert_called_once_with(
            'ipv6network',
            {'network_view': 'some-view', 'network': 'fffe:2312::/64'},
            extattrs=None, force_proxy=False, return_fields=mock.ANY,
            paging=False, max_results=None)

    def test_search_network_v6_using_network_field(self):
        connector = self._mock_connector()

        objects.Network.search(connector,
                               network_view='some-view',
                               network='fffe:2312::/64')
        connector.get_object.assert_called_once_with(
            'ipv6network',
            {'network_view': 'some-view', 'network': 'fffe:2312::/64'},
            extattrs=None, force_proxy=False, return_fields=mock.ANY,
            paging=False, max_results=None)

    def test_search_network_with_grid_dhcp_members(self):
        found = {
            '_ref': 'network/ZG5zLm5ldHdvcmskMTAuMC4zMi4wLzI0LzA:10.0.32.0/24/default',
            'members': [
                {'_struct': 'dhcpmember', 'ipv4addr': '192.168.10.67', 'name': 'dhcp01.example.com'},
                {'_struct': 'dhcpmember', 'ipv4addr': '192.168.11.67', 'name': 'dhcp02.example.com'}
            ]
        }
        connector = self._mock_connector(get_object=[found])

        network = objects.Network.search(
            connector,
            network_view='some-view',
            network='10.0.32.0/24',
            return_fields=['members']
        )
        connector.get_object.assert_called_once_with(
            'network',
            {'network_view': 'some-view', 'network': '10.0.32.0/24'},
            extattrs=None, force_proxy=False, return_fields=['members'],
            paging=False, max_results=None
        )
        self.assertIsInstance(network.members[0], objects.Dhcpmember)
        self.assertIsInstance(network.members[1], objects.Dhcpmember)
        self.assertEqual('dhcpmember', network.members[0]._struct)
        self.assertEqual('dhcpmember', network.members[1]._struct)
        self.assertEqual('192.168.10.67', network.members[0].ipv4addr)
        self.assertEqual('192.168.11.67', network.members[1].ipv4addr)

    def test_search_network_with_ms_dhcp_members(self):
        found = {
            '_ref': 'network/ZG5zLm5ldHdvcmskMTAuMC4zMi4wLzI0LzA:10.0.32.0/24/default',
            'members': [
                {'_struct': 'msdhcpserver', 'ipv4addr': '192.168.10.67', 'name': 'dhcp01.example.com'},
                {'_struct': 'msdhcpserver', 'ipv4addr': '192.168.11.67', 'name': 'dhcp02.example.com'}
            ]
        }
        connector = self._mock_connector(get_object=[found])

        network = objects.Network.search(
            connector,
            network_view='some-view',
            network='10.0.32.0/24',
            return_fields=['members']
        )
        connector.get_object.assert_called_once_with(
            'network',
            {'network_view': 'some-view', 'network': '10.0.32.0/24'},
            extattrs=None, force_proxy=False, return_fields=['members'],
            paging=False, max_results=None
        )
        self.assertIsInstance(network.members[0], objects.Dhcpmember)
        self.assertIsInstance(network.members[1], objects.Dhcpmember)
        self.assertEqual('msdhcpserver', network.members[0]._struct)
        self.assertEqual('msdhcpserver', network.members[1]._struct)
        self.assertEqual('192.168.10.67', network.members[0].ipv4addr)
        self.assertEqual('192.168.11.67', network.members[1].ipv4addr)

    def test_search_network_with_results(self):
        found = {"_ref": "network/ZG5zLm5ldHdvcmskMTAuMzkuMTEuMC8yNC8w"
                         ":10.39.11.0/24/default",
                 "network_view": 'some-view',
                 "network": '192.68.1.0/20'}
        connector = self._mock_connector(get_object=[found])

        network = objects.Network.search(connector,
                                         network_view='some-view',
                                         cidr='192.68.1.0/20')
        connector.get_object.assert_called_once_with(
            'network',
            {'network_view': 'some-view', 'network': '192.68.1.0/20'},
            extattrs=None, force_proxy=False, return_fields=mock.ANY,
            paging=False, max_results=None)
        self.assertEqual('192.68.1.0/20', network.network)
        self.assertEqual('some-view', network.network_view)
        # verify aliased fields works too
        self.assertEqual('192.68.1.0/20', network.cidr)

    def test_create_IP(self):
        ip = objects.IP.create(ip='192.168.1.12', mac='4a:ac:de:12:34:45')
        self.assertIsInstance(ip, objects.IPv4)
        self.assertEqual('192.168.1.12', ip.ip)
        self.assertEqual('192.168.1.12', ip.ipv4addr)
        self.assertEqual('4a:ac:de:12:34:45', ip.mac)
        self.assertEqual(None, ip.configure_for_dhcp)
        self.assertEqual(None, ip.host)

    def test_Create_MX_Record(self):
        mock_record = DEFAULT_MX_RECORD
        mx_record_copy = copy.deepcopy(mock_record)
        connector = self._mock_connector(create_object=mx_record_copy)
        mx = objects.MXRecord.create(connector, name='mx.demo.my_zone.com',
                                     mail_exchanger='demo.my_zone.com',
                                     view='my_dns_view', preference=1)
        self.assertIsInstance(mx, objects.MXRecord)
        connector.create_object.assert_called_once_with(
            'record:mx',
            {'mail_exchanger': 'demo.my_zone.com',
             'name': 'mx.demo.my_zone.com',
             'preference': 1,
             'view': 'my_dns_view'
             }, ['extattrs', 'mail_exchanger', 'name', 'preference', 'view'])

    def test_update_MX_Record(self):
        mx_record_copy = [
            {'_ref': 'record:mx/%s' % REC,
             'name': 'mx.demo.my_zone.com',
             'preference': 1,
             'mail_exchanger': 'demo.my_zone.com'}]

        connector = self._mock_connector(get_object=mx_record_copy)
        mx = objects.MXRecord.create(
            connector, name='mx1.demo.my_zone.com',
            mail_exchanger='demo2.my_zone.com',
            preference=1,
            update_if_exists=True)
        connector.update_object.assert_called_once_with(
            mx_record_copy[0]['_ref'],
            {'mail_exchanger': 'demo2.my_zone.com',
             'name': 'mx1.demo.my_zone.com', 'preference': 1},
            ['extattrs', 'mail_exchanger', 'name', 'preference', 'view'])

    def test_search_and_delete_MX_Record(self):
        mx_record_copy = copy.deepcopy(DEFAULT_MX_RECORD)
        connector = self._mock_connector(get_object=[mx_record_copy])

        mx_record = objects.MXRecord.search(connector,
                                            view='some_view',
                                            name='some_name')
        connector.get_object.assert_called_once_with(
            'record:mx', {'view': 'some_view',
                          'name': 'some_name'},
            extattrs=None, force_proxy=False, max_results=None, paging=False,
            return_fields=['extattrs', 'mail_exchanger', 'name', 'preference', 'view'])
        mx_record.delete()
        connector.delete_object.assert_called_once_with(
            DEFAULT_MX_RECORD['_ref'])

    def test_create_host_record_with_ttl(self):
        mock_record = DEFAULT_HOST_RECORD
        host_record_copy = copy.deepcopy(mock_record)
        connector = self._mock_connector(create_object=host_record_copy)

        ip = objects.IP.create(ip='22.0.0.2', mac='fa:16:3e:29:87:70')
        self.assertIsInstance(ip, objects.IPv4)

        host_record = objects.HostRecord.create(connector,
                                                ttl=42,
                                                view='some-dns-view',
                                                ip=[ip])
        self.assertIsInstance(host_record, objects.HostRecordV4)
        connector.create_object.assert_called_once_with(
            'record:host',
            {'ttl': 42,
             'ipv4addrs': [
                 {'mac': 'fa:16:3e:29:87:70',
                  'ipv4addr': '22.0.0.2'}],
             'view': 'some-dns-view'},
            ['extattrs', 'ipv4addrs', 'name', 'view', 'aliases'])

    def test_create_host_record_with_ip(self):
        mock_record = DEFAULT_HOST_RECORD
        host_record_copy = copy.deepcopy(mock_record)
        connector = self._mock_connector(create_object=host_record_copy)

        ip = objects.IP.create(ip='22.0.0.2', mac='fa:16:3e:29:87:70')
        self.assertIsInstance(ip, objects.IPv4)

        host_record = objects.HostRecord.create(connector,
                                                view='some-dns-view',
                                                ip=[ip])
        # Validate that ip object was converted to simple ip
        # as a string representation for searching
        connector.get_object.assert_called_once_with(
            'record:host',
            {'view': 'some-dns-view', 'ipv4addr': '22.0.0.2'},
            return_fields=mock.ANY)
        # Validate create_object call
        ip_dict = {'ipv4addr': '22.0.0.2', 'mac': 'fa:16:3e:29:87:70'}
        connector.create_object.assert_called_once_with(
            'record:host',
            {'view': 'some-dns-view', 'ipv4addrs': [ip_dict]}, mock.ANY)
        self.assertIsInstance(host_record, objects.HostRecordV4)
        # validate nios reply was parsed correctly
        self.assertEqual(mock_record['_ref'], host_record._ref)
        nios_ip = host_record.ipv4addrs[0]
        self.assertIsInstance(ip, objects.IPv4)
        self.assertEqual(mock_record['ipv4addrs'][0]['mac'], nios_ip.mac)
        self.assertEqual(mock_record['ipv4addrs'][0]['ipv4addr'],
                         nios_ip.ipv4addr)
        self.assertEqual(mock_record['ipv4addrs'][0]['host'],
                         nios_ip.host)
        self.assertEqual(mock_record['ipv4addrs'][0]['configure_for_dhcp'],
                         nios_ip.configure_for_dhcp)
        # Validate 'host' field is not send on update
        new_ip = objects.IP.create(ip='22.0.0.10', mac='fa:16:3e:29:87:71',
                                   configure_for_dhcp=False)
        host_record.ip.append(new_ip)
        host_record.extattrs = {}
        host_record.update()
        ip_dict['configure_for_dhcp'] = False
        ip_dict_new = {'ipv4addr': '22.0.0.10', 'mac': 'fa:16:3e:29:87:71',
                       'configure_for_dhcp': False}
        connector.update_object.assert_called_once_with(
            host_record.ref,
            {'ipv4addrs': [ip_dict, ip_dict_new],
             'extattrs': {}}, mock.ANY)

    def test_search_and_delete_host_record(self):
        host_record_copy = copy.deepcopy(DEFAULT_HOST_RECORD)
        connector = self._mock_connector(get_object=[host_record_copy])

        host_record = objects.HostRecord.search(connector,
                                                view='some-dns-view',
                                                ip='192.168.15.20',
                                                network_view='test-netview')
        connector.get_object.assert_called_once_with(
            'record:host',
            {'view': 'some-dns-view', 'ipv4addr': '192.168.15.20',
             'network_view': 'test-netview'},
            extattrs=None, force_proxy=False, return_fields=mock.ANY,
            paging=False, max_results=None)

        # Validate extattrs in host_record are converted to EA object
        self.assertIsInstance(host_record.extattrs, objects.EA)

        host_record.delete()
        connector.delete_object.assert_called_once_with(
            DEFAULT_HOST_RECORD['_ref'])

    def test_create_fixed_address(self):
        mock_fixed_address = {
            '_ref': 'fixedaddress/ZG5zLmhvc3QkLl9kZWZhdWx0LmNvbS5nbG9iYWw2NA',
            'ipv4addr': '192.168.1.15',
            'mac': 'aa:ac:cd:11:22:33',
        }
        connector = self._mock_connector(create_object=mock_fixed_address)

        fixed_addr = objects.FixedAddress.create(
            connector,
            ip='192.168.1.15',
            network_view='some-view',
            mac='aa:ac:cd:11:22:33',
            ms_server={'_struct': 'msdhcpserver',
                       'ipv4addr': '192.168.1.0'})
        connector.get_object.assert_called_once_with(
            'fixedaddress',
            {'network_view': 'some-view', 'ipv4addr': '192.168.1.15',
             'mac': 'aa:ac:cd:11:22:33'},
            return_fields=mock.ANY)
        self.assertIsInstance(fixed_addr, objects.FixedAddressV4)
        connector.create_object.assert_called_once_with(
            'fixedaddress',
            {'network_view': 'some-view',
             'ipv4addr': '192.168.1.15',
             'mac': 'aa:ac:cd:11:22:33',
             'ms_server': {'_struct': 'msdhcpserver',
                           'ipv4addr': '192.168.1.0'}}, mock.ANY)

    def test_create_fixed_address_v6(self):
        mock_fixed_address = {
            '_ref': 'ipv6fixedaddress/ZG5zLmhvc3QkLl9kZWZhdWx0LmNvbS5nbG9iYA',
            'ipv6addr': 'fffe:1234:1234::1',
            'duid': '00:23:97:49:aa:ac:cd:11:22:33',
        }
        connector = self._mock_connector(create_object=mock_fixed_address)

        fixed_addr = objects.FixedAddress.create(
            connector,
            ip='fffe:1234:1234::1',
            network_view='some-view',
            mac='aa:ac:cd:11:22:33',
            ms_server={'_struct': 'msdhcpserver',
                       'ipv4addr': '192.168.1.0'})

        self.assertIsInstance(fixed_addr, objects.FixedAddressV6)
        self.assertEqual(mock_fixed_address['duid'], fixed_addr.duid)

        connector.get_object.assert_called_once_with(
            'ipv6fixedaddress',
            {'duid': mock.ANY, 'ipv6addr': 'fffe:1234:1234::1',
             'network_view': 'some-view'},
            return_fields=mock.ANY)
        connector.create_object.assert_called_once_with(
            'ipv6fixedaddress',
            {'duid': mock.ANY, 'ipv6addr': 'fffe:1234:1234::1',
             'network_view': 'some-view'}, mock.ANY)

    @mock.patch('infoblox_client.utils.generate_duid')
    def test_fixed_address_v6(self, generate):
        mac = 'aa:ac:cd:11:22:33'
        duid = '00:0a:d3:9b:aa:ac:cd:11:22:33'
        generate.return_value = duid
        connector = self._mock_connector()
        fixed_addr = objects.FixedAddress(
            connector,
            ip='fffe:1234:1234::1',
            network_view='some-view',
            mac=mac)
        self.assertIsInstance(fixed_addr, objects.FixedAddressV6)
        self.assertEqual(mac, fixed_addr.mac)
        self.assertEqual(duid, fixed_addr.duid)
        generate.assert_called_once_with(mac)

    def test_search_ipaddress(self):
        ip_mock = [{'_ref': ('ipv4address/Li5pcHY0X2FkZHJlc3MkMTky'
                             'LjE2OC4xLjEwLzE:192.168.1.10/my_view'),
                    'objects': ['ref_1', 'ref_2']}]
        connector = self._mock_connector(get_object=ip_mock)
        ip = objects.IPAddress.search(connector,
                                      network_view='some_view',
                                      ip_address='192.168.1.5')
        payload = {'network_view': 'some_view', 'ip_address': '192.168.1.5'}
        connector.get_object.assert_called_once_with(
            'ipv4address', payload, return_fields=mock.ANY,
            paging=False, extattrs=None, force_proxy=mock.ANY, max_results=None)
        self.assertIsInstance(ip, objects.IPv4Address)
        self.assertEqual(ip_mock[0]['objects'], ip.objects)

    def test__process_value(self):
        data = (([1, 2, 3], ['1', '2', '3']),
                ((1, 2), ['1', '2']),
                (1, '1'),
                ('string', 'string'))
        for input, output in data:
            self.assertEqual(output, objects.EA._process_value(str, input))

    def test_ea_parse_generate(self):
        eas = {'Subnet ID': {'value': 'some-id'},
               'Tenant Name': {'value': 'tenant-name'},
               'Cloud API Owned': {'value': 'True'},
               'Some EA': {'value': 'False'},
               'Zero EA': {'value': '0'}}
        ea = objects.EA.from_dict(eas)
        self.assertIsInstance(ea, objects.EA)
        # validate True and False are converted to booleans
        self.assertEqual(True, ea.get('Cloud API Owned'))
        self.assertEqual(False, ea.get('Some EA'))
        self.assertEqual('0', ea.get('Zero EA'))
        self.assertEqual(eas, ea.to_dict())

    def test_ea_to_dict(self):
        ea = {'Subnet ID': 'some-id',
              'Tenant Name': 'tenant-name',
              'Cloud API Owned': 'True',
              'DNS Record Types': ['record_a', 'record_ptr'],
              'False String EA': 'False',
              'Empty String EA': '',
              'False EA': False,
              'Zero EA': 0,
              'None EA': None,
              'None String EA': 'None',
              'Empty List EA': [],
              'Zero String EA': '0'}
        processed_ea = {'Subnet ID': 'some-id',
                        'Tenant Name': 'tenant-name',
                        'Cloud API Owned': 'True',
                        'DNS Record Types': ['record_a', 'record_ptr'],
                        'False String EA': 'False',
                        'False EA': 'False',
                        'Zero EA': '0',
                        'None String EA': 'None',
                        'Zero String EA': '0'}
        ea_exist = ['Subnet ID',
                    'Tenant Name',
                    'Cloud API Owned',
                    'DNS Record Types',
                    'False String EA',
                    'False EA',
                    'Zero EA',
                    'None String EA',
                    'Zero String EA']
        ea_purged = ['Empty String EA',
                     'None EA',
                     'Empty List EA']
        ea_dict = objects.EA(ea).to_dict()
        self.assertIsInstance(ea_dict, dict)
        for key in ea_exist:
            self.assertEqual(True, key in ea_dict)
        for key in ea_purged:
            self.assertEqual(False, key in ea_dict)
        for key in processed_ea:
            self.assertEqual(processed_ea[key], ea_dict.get(key).get('value'))

    def test_ea_returns_none(self):
        for ea in (None, '', 0):
            self.assertEqual(None, objects.EA.from_dict(ea))

    def test_ea_set_get(self):
        ea = objects.EA()
        ea_name = 'Subnet ID'
        id = 'subnet-id'
        generated_eas = {ea_name: {'value': id}}
        ea.set(ea_name, id)
        self.assertEqual(id, ea.get(ea_name))
        self.assertEqual(generated_eas, ea.to_dict())

    def test_ea_returns_ea_dict(self):
        ea_dict = {'Subnet ID': 'some-id'}
        ea = objects.EA(ea_dict)
        ea_dict_from_EA_object = ea.ea_dict
        self.assertEqual(ea_dict, ea_dict_from_EA_object)
        # Make sure a copy of dict is returned,
        # and updating returned value do not affect EA object
        ea_dict_from_EA_object['Subnet ID'] = 'another-id'
        self.assertEqual('some-id', ea.get('Subnet ID'))

    def test_update_from_dict(self):
        net = objects.Network(mock.Mock(), network='192.168.1.0/24')
        self.assertEqual(None, net._ref)

        reply = {'_ref': 'network/asdwdqwecaszxcrqqwe',
                 'network': '192.168.100.0/24',
                 'network_view': 'default'}
        net.update_from_dict(reply, only_ref=True)
        self.assertEqual(reply['_ref'], net._ref)
        self.assertEqual('192.168.1.0/24', net.network)
        self.assertEqual(None, net.network_view)

    def test_create_fails_on_multiple_api_objects(self):
        """
        If multiple objects are returned by the API, create should raise
        exception.
        """
        a_records = [{'_ref': 'record:a/Awsdrefsasdwqoijvoriibtrni',
                      'ip': '192.168.1.52',
                      'name': 'record1'},
                     {'_ref': 'record:a/Awsdrefsasdwqoijvoriibtrna',
                      'ip': '192.168.1.52',
                      'name': 'record2'}]

        connector = self._mock_connector(get_object=a_records)

        with self.assertRaises(ib_ex.InfobloxFetchGotMultipleObjects):
            objects.ARecordBase.create(connector,
                                       ip='192.168.1.52',
                                       view='view')

        connector.get_object.assert_called_once_with(
            'record:a',
            {'view': 'view', 'ipv4addr': '192.168.1.52'},
            return_fields=[])

    def test_update_fields_on_create(self):
        a_record = [{'_ref': 'record:a/Awsdrefsasdwqoijvoriibtrni',
                     'ip': '192.168.1.52',
                     'name': 'a_record',
                     'comment': 'test_comment'}]
        connector = self._mock_connector(get_object=a_record)
        objects.ARecordBase.create(connector,
                                   ip='192.168.1.52',
                                   view='view',
                                   comment='new_test_comment',
                                   update_if_exists=True)
        connector.get_object.assert_called_once_with(
            'record:a',
            {'view': 'view', 'ipv4addr': '192.168.1.52'},
            return_fields=[])
        connector.update_object.assert_called_once_with(
            a_record[0]['_ref'],
            {'ipv4addr': '192.168.1.52', 'comment': 'new_test_comment'},
            mock.ANY)

    def test_update_fields_on_create_v6(self):
        aaaa_record = [{'_ref': 'record:aaaa/Awsdrefsasdwqoijvoriibtrni',
                        'ip': '2001:610:240:22::c100:68b',
                        'name': 'aaaa_record',
                        'comment': "test_comment"}]
        connector = self._mock_connector(get_object=aaaa_record)
        objects.ARecordBase.create(connector,
                                   ip='2001:610:240:22::c100:68b',
                                   view='view',
                                   comment='new_test_comment',
                                   update_if_exists=True)
        connector.get_object.assert_called_once_with(
            'record:aaaa',
            {'view': 'view', 'ipv6addr': '2001:610:240:22::c100:68b'},
            return_fields=[])
        connector.update_object.assert_called_once_with(
            aaaa_record[0]['_ref'],
            {'comment': 'new_test_comment', 'ipv6addr': '2001:610:240:22::c100:68b'},
            mock.ANY)

    def test_ip_version(self):
        conn = mock.Mock()
        net_v4 = objects.Network(conn, network='192.168.1.0/24')
        self.assertEqual(4, net_v4.ip_version)
        net_v6 = objects.Network(conn, network='fffe::/64')
        self.assertEqual(6, net_v6.ip_version)

    def test_get_tenant(self):
        id = 'tenant_id'
        fake_tenant = {
            '_ref': 'grid:cloudapi:tenant/ZG5zLm5ldHdvcmskMTAuMzk',
            'id': id,
            'name': 'Tenant Name',
            'comment': 'Some comment'}
        conn = self._mock_connector(get_object=[fake_tenant])
        tenant = objects.Tenant.search(conn, id=id)
        conn.get_object.assert_called_once_with(
            'grid:cloudapi:tenant', {'id': id},
            return_fields=mock.ANY, extattrs=None, force_proxy=mock.ANY,
            paging=False, max_results=None)
        self.assertEqual(fake_tenant['id'], tenant.id)
        self.assertEqual(fake_tenant['name'], tenant.name)
        self.assertEqual(fake_tenant['comment'], tenant.comment)

    def test__remap_fields_support_unknown_fields(self):
        data = {'host_name': 'cp.com',
                'unknown_field': 'some_data'}
        self.assertEqual(data, objects.Member._remap_fields(data))

    def test_TXT_Record(self):
        mock_record = DEFAULT_TXT_RECORD
        txt_record_copy = copy.deepcopy(mock_record)
        connector = self._mock_connector(create_object=txt_record_copy)
        txt = objects.TXTRecord.create(connector, name='text_test.my_zone.com',
                                       text='hello_text',
                                       view='my_dns_view')
        self.assertIsInstance(txt, objects.TXTRecord)
        connector.create_object.assert_called_once_with(
            'record:txt',
            {'name': 'text_test.my_zone.com',
             'text': 'hello_text',
             'view': 'my_dns_view',
             }, ['extattrs', 'name', 'text', 'view'])

    def test_call_upload_file(self):
        upload_file_path = '/http_direct_file_io/req_id-UPLOAD-0302163936014609/ibx_networks.csv'
        upload_url = 'https://infoblox.example.org' + upload_file_path
        self._create_infoblox_csv()
        with open('tests/ibx_networks.csv', 'r') as fh:
            data = fh.read()
            fh.close()
        payload = dict(file=data)
        connector = self._mock_connector()
        fo = objects.Fileop(connector)
        result = fo.upload_file(upload_url, payload)
        self.assertIsInstance(fo, objects.Fileop)
        self.assertTrue(result)
        # clean up and remove csv file
        self._delete_infoblox_csv()

    def test__search_non_searchable_fields(self):
        """
        Method InfobloxObject._search should raise the
        ib_ex.InfobloxFieldNotSearchable error, if non-searchable
        fields are used.
        """
        connector = self._mock_connector()

        with self.assertRaises(ib_ex.InfobloxFieldNotSearchable) as e:
            objects.AAAARecord._search(
                connector,
                # Use non-searchable field for search
                use_ttl=True,
            )

        self.assertTrue(
            re.match(
                # For Python 3.x and 2.x string repr of dict keys may differ.
                # This regex matches both representations.
                r"^Field is not searchable: use_ttl",
                str(e.exception),
            ),
            "Exception string '%s' doesn't match test regexp" % e.exception
        )

    def test_member_searchable_ipv4(self):
        """
        Validates if Member object can be searched by ipv4_address
        """
        connector = self._mock_connector()
        objects.Member.search(connector, ipv4_address='10.0.3.5')
        connector.get_object.assert_called_once_with(
            "member",
            {"ipv4_address": "10.0.3.5"},
            extattrs=None,
            force_proxy=False,
            return_fields=mock.ANY,
            paging=False,
            max_results=None,
        )

    def test_member_searchable_ipv6(self):
        """
        Validates if Member object can be searched by ipv6_address
        """
        connector = self._mock_connector()
        objects.Member.search(connector, ipv6_address='fffe:1234:1234::1')
        connector.get_object.assert_called_once_with(
            "member",
            {"ipv6_address": "fffe:1234:1234::1"},
            extattrs=None,
            force_proxy=False,
            return_fields=mock.ANY,
            paging=False,
            max_results=None,
        )
