| # Licensed to the Apache Software Foundation (ASF) under one or more |
| # contributor license agreements. See the NOTICE file distributed with |
| # this work for additional information regarding copyright ownership. |
| # The ASF licenses this file to You 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 sys |
| import unittest |
| |
| try: |
| import simplejson as json |
| except ImportError: |
| import json # NOQA |
| |
| from libcloud.utils.py3 import httplib |
| |
| from libcloud.common.types import InvalidCredsError |
| from libcloud.compute.base import NodeImage |
| from libcloud.compute.drivers.digitalocean import DigitalOceanNodeDriver |
| |
| from libcloud.test import LibcloudTestCase, MockHttpTestCase |
| from libcloud.test.file_fixtures import ComputeFileFixtures |
| from libcloud.test.secrets import DIGITALOCEAN_v1_PARAMS |
| from libcloud.test.secrets import DIGITALOCEAN_v2_PARAMS |
| |
| |
| # class DigitalOceanTests(unittest.TestCase, TestCaseMixin): |
| class DigitalOcean_v1_Tests(LibcloudTestCase): |
| |
| def setUp(self): |
| DigitalOceanNodeDriver.connectionCls.conn_classes = \ |
| (None, DigitalOceanMockHttp) |
| DigitalOceanMockHttp.type = None |
| self.driver = DigitalOceanNodeDriver(*DIGITALOCEAN_v1_PARAMS, |
| api_version='v1') |
| |
| def test_v1_uses_v2_key(self): |
| self.assertRaises(InvalidCredsError, DigitalOceanNodeDriver, |
| *DIGITALOCEAN_v2_PARAMS, api_version='v1') |
| |
| def test_authentication(self): |
| DigitalOceanMockHttp.type = 'UNAUTHORIZED_CLIENT' |
| self.assertRaises(InvalidCredsError, self.driver.list_nodes) |
| |
| def test_list_images_success(self): |
| images = self.driver.list_images() |
| self.assertTrue(len(images) >= 1) |
| |
| image = images[0] |
| self.assertTrue(image.id is not None) |
| self.assertTrue(image.name is not None) |
| |
| def test_list_sizes_success(self): |
| sizes = self.driver.list_sizes() |
| self.assertTrue(len(sizes) >= 1) |
| |
| size = sizes[0] |
| self.assertTrue(size.id is not None) |
| self.assertEqual(size.name, '512MB') |
| self.assertEqual(size.ram, 512) |
| |
| size = sizes[4] |
| self.assertTrue(size.id is not None) |
| self.assertEqual(size.name, '8GB') |
| self.assertEqual(size.ram, 8 * 1024) |
| |
| def test_list_locations_success(self): |
| locations = self.driver.list_locations() |
| self.assertTrue(len(locations) >= 1) |
| |
| location = locations[0] |
| self.assertEqual(location.id, '1') |
| self.assertEqual(location.name, 'New York 1') |
| |
| def test_list_nodes_success(self): |
| nodes = self.driver.list_nodes() |
| self.assertEqual(len(nodes), 1) |
| self.assertEqual(nodes[0].name, 'test-2') |
| self.assertEqual(nodes[0].public_ips, []) |
| self.assertEqual(nodes[0].extra['image_id'], 1601) |
| self.assertEqual(nodes[0].extra['size_id'], 66) |
| |
| def test_list_nodes_does_not_support_created_datetime(self): |
| nodes = self.driver.list_nodes() |
| self.assertIsNone(nodes[0].created_at) |
| |
| def test_create_node_invalid_size(self): |
| image = NodeImage(id='invalid', name=None, driver=self.driver) |
| size = self.driver.list_sizes()[0] |
| location = self.driver.list_locations()[0] |
| |
| DigitalOceanMockHttp.type = 'INVALID_IMAGE' |
| expected_msg = \ |
| r'You specified an invalid image for Droplet creation. \(code: (404|HTTPStatus.NOT_FOUND)\)' |
| self.assertRaisesRegexp(Exception, expected_msg, |
| self.driver.create_node, |
| name='test', size=size, image=image, |
| location=location) |
| |
| def test_reboot_node_success(self): |
| node = self.driver.list_nodes()[0] |
| result = self.driver.reboot_node(node) |
| self.assertTrue(result) |
| |
| def test_destroy_node_success(self): |
| node = self.driver.list_nodes()[0] |
| result = self.driver.destroy_node(node) |
| self.assertTrue(result) |
| |
| def test_ex_rename_node_success(self): |
| node = self.driver.list_nodes()[0] |
| result = self.driver.ex_rename_node(node, 'fedora helios') |
| self.assertTrue(result) |
| |
| def test_list_key_pairs(self): |
| keys = self.driver.list_key_pairs() |
| self.assertEqual(len(keys), 1) |
| |
| self.assertEqual(keys[0].extra['id'], 7717) |
| self.assertEqual(keys[0].name, 'test1') |
| self.assertEqual(keys[0].public_key, None) |
| |
| def test_ex_list_ssh_keys(self): |
| keys = self.driver.ex_list_ssh_keys() |
| self.assertEqual(len(keys), 1) |
| |
| self.assertEqual(keys[0].id, 7717) |
| self.assertEqual(keys[0].name, 'test1') |
| self.assertEqual(keys[0].pub_key, None) |
| |
| def test_delete_key_pair(self): |
| key = self.driver.list_key_pairs()[0] |
| result = self.driver.delete_key_pair(key) |
| self.assertTrue(result) |
| |
| def test_ex_destroy_ssh_key(self): |
| key = self.driver.ex_list_ssh_keys()[0] |
| result = self.driver.ex_destroy_ssh_key(key.id) |
| self.assertTrue(result) |
| |
| |
| class DigitalOceanMockHttp(MockHttpTestCase): |
| fixtures = ComputeFileFixtures('digitalocean_v1') |
| |
| def _v1_regions(self, method, url, body, headers): |
| body = self.fixtures.load('list_locations.json') |
| return (httplib.OK, body, {}, httplib.responses[httplib.OK]) |
| |
| def _v1_images(self, method, url, body, headers): |
| body = self.fixtures.load('list_images.json') |
| return (httplib.OK, body, {}, httplib.responses[httplib.OK]) |
| |
| def _v1_sizes(self, method, url, body, headers): |
| body = self.fixtures.load('list_sizes.json') |
| return (httplib.OK, body, {}, httplib.responses[httplib.OK]) |
| |
| def _v1_droplets(self, method, url, body, headers): |
| body = self.fixtures.load('list_nodes.json') |
| return (httplib.OK, body, {}, httplib.responses[httplib.OK]) |
| |
| def _v1_droplets_new_INVALID_IMAGE(self, method, url, body, headers): |
| # reboot_node |
| body = self.fixtures.load('error_invalid_image.json') |
| return (httplib.NOT_FOUND, body, {}, |
| httplib.responses[httplib.NOT_FOUND]) |
| |
| def _v1_droplets_119461_reboot(self, method, url, body, headers): |
| # reboot_node |
| body = self.fixtures.load('reboot_node.json') |
| return (httplib.OK, body, {}, httplib.responses[httplib.OK]) |
| |
| def _v1_droplets_119461_destroy(self, method, url, body, headers): |
| # destroy_node |
| self.assertUrlContainsQueryParams(url, {'scrub_data': '1'}) |
| body = self.fixtures.load('destroy_node.json') |
| return (httplib.OK, body, {}, httplib.responses[httplib.OK]) |
| |
| def _v1_droplets_119461_rename(self, method, url, body, headers): |
| # reboot_node |
| self.assertUrlContainsQueryParams(url, {'name': 'fedora helios'}) |
| body = self.fixtures.load('ex_rename_node.json') |
| return (httplib.OK, body, {}, httplib.responses[httplib.OK]) |
| |
| def _v1_ssh_keys(self, method, url, body, headers): |
| body = self.fixtures.load('ex_list_ssh_keys.json') |
| return (httplib.OK, body, {}, httplib.responses[httplib.OK]) |
| |
| def _v1_ssh_keys_7717_destroy(self, method, url, body, headers): |
| # destroy_ssh_key |
| body = self.fixtures.load('ex_destroy_ssh_key.json') |
| return (httplib.OK, body, {}, httplib.responses[httplib.OK]) |
| |
| def _v1_droplets_UNAUTHORIZED_CLIENT(self, method, url, body, headers): |
| body = self.fixtures.load('error.txt') |
| return (httplib.FOUND, body, {}, httplib.responses[httplib.FOUND]) |
| |
| if __name__ == '__main__': |
| sys.exit(unittest.main()) |