summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorJenkins <jenkins@review.openstack.org>2014-05-06 13:25:00 +0000
committerGerrit Code Review <review@openstack.org>2014-05-06 13:25:00 +0000
commit4e85183019da3c1a4b350d4b426b43de5a492727 (patch)
treece92fc474a3547eff2cf554e3256ef8f11c2aa93
parentb1ca7056b106638151f8bde45a4c7764633c139b (diff)
parenteb94ac076dec7e08f4e6c4c55d7c6f60b1b1cbfd (diff)
downloadpython-swiftclient-4e85183019da3c1a4b350d4b426b43de5a492727.tar.gz
Merge "Add functional tests for python-swiftclient"
-rwxr-xr-x.functests8
-rwxr-xr-x.unittests2
-rw-r--r--swiftclient/exceptions.py4
-rw-r--r--tests/functional/__init__.py0
-rw-r--r--tests/functional/test_swiftclient.py289
-rw-r--r--tests/sample.conf17
-rw-r--r--tests/unit/__init__.py0
-rw-r--r--tests/unit/test_command_helpers.py (renamed from tests/test_command_helpers.py)0
-rw-r--r--tests/unit/test_multithreading.py (renamed from tests/test_multithreading.py)0
-rw-r--r--tests/unit/test_swiftclient.py (renamed from tests/test_swiftclient.py)0
-rw-r--r--tests/unit/test_utils.py (renamed from tests/test_utils.py)0
-rw-r--r--tests/unit/utils.py (renamed from tests/utils.py)0
-rw-r--r--tox.ini2
13 files changed, 320 insertions, 2 deletions
diff --git a/.functests b/.functests
new file mode 100755
index 0000000..05d9002
--- /dev/null
+++ b/.functests
@@ -0,0 +1,8 @@
+#!/bin/bash
+set -e
+
+python setup.py testr --coverage --testr-args="--concurrency=1 tests.functional"
+RET=$?
+coverage report -m
+rm -f .coverage
+exit $RET
diff --git a/.unittests b/.unittests
index bf5b027..5d935b1 100755
--- a/.unittests
+++ b/.unittests
@@ -1,7 +1,7 @@
#!/bin/bash
set -e
-python setup.py testr --coverage
+python setup.py testr --coverage --testr-args="tests.unit"
RET=$?
coverage report -m
rm -f .coverage
diff --git a/swiftclient/exceptions.py b/swiftclient/exceptions.py
index 9a77672..56c696f 100644
--- a/swiftclient/exceptions.py
+++ b/swiftclient/exceptions.py
@@ -66,3 +66,7 @@ class ClientException(Exception):
b += ' [first 60 chars of response] %s' \
% self.http_response_content[:60]
return b and '%s: %s' % (a, b) or a
+
+
+class SkipTest(Exception):
+ pass
diff --git a/tests/functional/__init__.py b/tests/functional/__init__.py
new file mode 100644
index 0000000..e69de29
--- /dev/null
+++ b/tests/functional/__init__.py
diff --git a/tests/functional/test_swiftclient.py b/tests/functional/test_swiftclient.py
new file mode 100644
index 0000000..891ba54
--- /dev/null
+++ b/tests/functional/test_swiftclient.py
@@ -0,0 +1,289 @@
+# Copyright (c) 2014 Christian Schwede <christian.schwede@enovance.com>
+#
+# 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 ConfigParser
+import os
+import StringIO
+import testtools
+import time
+import types
+import unittest
+
+import swiftclient
+
+
+class TestFunctional(testtools.TestCase):
+
+ def __init__(self, *args, **kwargs):
+ super(TestFunctional, self).__init__(*args, **kwargs)
+ self.skip_tests = False
+ self._get_config()
+
+ self.test_data = '42' * 10
+ self.etag = '2704306ec982238d85d4b235c925d58e'
+
+ self.containername = "functional-tests-container-%s" % int(time.time())
+ self.containername_2 = self.containername + '_second'
+ self.containername_3 = self.containername + '_third'
+ self.objectname = "functional-tests-object-%s" % int(time.time())
+ self.objectname_2 = self.objectname + '_second'
+
+ def _get_config(self):
+ config_file = os.environ.get('SWIFT_TEST_CONFIG_FILE',
+ '/etc/swift/test.conf')
+ config = ConfigParser.SafeConfigParser({'auth_version': '1'})
+ config.read(config_file)
+ if config.has_section('func_test'):
+ auth_host = config.get('func_test', 'auth_host')
+ auth_port = config.getint('func_test', 'auth_port')
+ auth_ssl = config.getboolean('func_test', 'auth_ssl')
+ auth_prefix = config.get('func_test', 'auth_prefix')
+ self.auth_version = config.get('func_test', 'auth_version')
+ self.account = config.get('func_test', 'account')
+ self.username = config.get('func_test', 'username')
+ self.password = config.get('func_test', 'password')
+ self.auth_url = ""
+ if auth_ssl:
+ self.auth_url += "https://"
+ else:
+ self.auth_url += "http://"
+ self.auth_url += "%s:%s%s" % (auth_host, auth_port, auth_prefix)
+ if self.auth_version == "1":
+ self.auth_url += 'v1.0'
+ self.account_username = "%s:%s" % (self.account, self.username)
+
+ else:
+ self.skip_tests = True
+
+ def setUp(self):
+ super(TestFunctional, self).setUp()
+ if self.skip_tests:
+ raise swiftclient.exceptions.SkipTest(
+ 'SKIPPING FUNCTIONAL TESTS DUE TO NO CONFIG')
+
+ self.conn = swiftclient.Connection(
+ self.auth_url, self.account_username, self.password,
+ auth_version=self.auth_version)
+
+ self.conn.put_container(self.containername)
+ self.conn.put_container(self.containername_2)
+ self.conn.put_object(
+ self.containername, self.objectname, self.test_data)
+ self.conn.put_object(
+ self.containername, self.objectname_2, self.test_data)
+
+ def tearDown(self):
+ super(TestFunctional, self).tearDown()
+ for obj in [self.objectname, self.objectname_2]:
+ try:
+ self.conn.delete_object(self.containername, obj)
+ except swiftclient.ClientException:
+ pass
+
+ for container in [self.containername,
+ self.containername_2,
+ self.containername_3,
+ self.containername + '_segments']:
+ try:
+ self.conn.delete_container(container)
+ except swiftclient.ClientException:
+ pass
+
+ def _check_account_headers(self, headers):
+ self.assertTrue(headers.get('content-length'))
+ self.assertTrue(headers.get('x-account-object-count'))
+ self.assertTrue(headers.get('x-timestamp'))
+ self.assertTrue(headers.get('x-trans-id'))
+ self.assertTrue(headers.get('date'))
+ self.assertTrue(headers.get('x-account-bytes-used'))
+ self.assertTrue(headers.get('x-account-container-count'))
+ self.assertTrue(headers.get('content-type'))
+ self.assertTrue(headers.get('accept-ranges'))
+
+ def test_stat_account(self):
+ headers = self.conn.head_account()
+ self._check_account_headers(headers)
+
+ def test_list_account(self):
+ headers, containers = self.conn.get_account()
+ self._check_account_headers(headers)
+
+ self.assertTrue(len(containers))
+ test_container = [c
+ for c in containers
+ if c.get('name') == self.containername][0]
+ self.assertTrue(test_container.get('bytes') >= 0)
+ self.assertTrue(test_container.get('count') >= 0)
+
+ # Check if list limit is working
+ headers, containers = self.conn.get_account(limit=1)
+ self.assertEqual(1, len(containers))
+
+ # Check full listing
+ headers, containers = self.conn.get_account(limit=1, full_listing=True)
+ self.assertTrue(len(containers) >= 2) # there might be more containers
+
+ # Test marker
+ headers, containers = self.conn.get_account(marker=self.containername)
+ self.assertTrue(len(containers) >= 1)
+ self.assertEqual(self.containername_2, containers[0].get('name'))
+
+ def _check_container_headers(self, headers):
+ self.assertTrue(headers.get('content-length'))
+ self.assertTrue(headers.get('x-container-object-count'))
+ self.assertTrue(headers.get('x-timestamp'))
+ self.assertTrue(headers.get('x-trans-id'))
+ self.assertTrue(headers.get('date'))
+ self.assertTrue(headers.get('x-container-bytes-used'))
+ self.assertTrue(headers.get('x-container-object-count'))
+ self.assertTrue(headers.get('content-type'))
+ self.assertTrue(headers.get('accept-ranges'))
+
+ def test_stat_container(self):
+ headers = self.conn.head_container(self.containername)
+ self._check_container_headers(headers)
+
+ def test_list_container(self):
+ headers, objects = self.conn.get_container(self.containername)
+ self._check_container_headers(headers)
+ self.assertTrue(len(objects))
+ test_object = [o
+ for o in objects
+ if o.get('name') == self.objectname][0]
+ self.assertEqual(len(self.test_data), test_object.get('bytes'))
+ self.assertEqual(self.etag, test_object.get('hash'))
+ self.assertEqual('application/octet-stream',
+ test_object.get('content_type'))
+
+ # Check if list limit is working
+ headers, objects = self.conn.get_container(self.containername, limit=1)
+ self.assertEqual(1, len(objects))
+
+ # Check full listing
+ headers, objects = self.conn.get_container(
+ self.containername, limit=1, full_listing=True)
+ self.assertEqual(2, len(objects))
+
+ # Test marker
+ headers, objects = self.conn.get_container(
+ self.containername, marker=self.objectname)
+ self.assertEqual(1, len(objects))
+ self.assertEqual(self.objectname_2, objects[0].get('name'))
+
+ def test_create_container(self):
+ self.conn.put_container(self.containername_3)
+ self.assertTrue(self.conn.head_container(self.containername_3))
+
+ def test_delete(self):
+ self.conn.delete_object(self.containername, self.objectname)
+ self.conn.delete_object(self.containername, self.objectname_2)
+ self.conn.delete_container(self.containername)
+
+ # Container HEAD will raise an exception if container doesn't exist
+ # which is only possible if previous requests succeeded
+ self.assertRaises(
+ swiftclient.ClientException,
+ self.conn.head_container,
+ self.containername)
+
+ def test_upload_object(self):
+ # Object with content from string
+ self.conn.put_object(
+ self.containername, self.objectname, contents=self.test_data)
+ hdrs = self.conn.head_object(self.containername, self.objectname)
+ self.assertEqual(str(len(self.test_data)),
+ hdrs.get('content-length'))
+ self.assertEqual(self.etag, hdrs.get('etag'))
+ self.assertEqual('application/octet-stream',
+ hdrs.get('content-type'))
+
+ # Same but with content-length
+ self.conn.put_object(
+ self.containername, self.objectname,
+ contents=self.test_data, content_length=len(self.test_data))
+ hdrs = self.conn.head_object(self.containername, self.objectname)
+ self.assertEqual(str(len(self.test_data)),
+ hdrs.get('content-length'))
+ self.assertEqual(self.etag, hdrs.get('etag'))
+ self.assertEqual('application/octet-stream', hdrs.get('content-type'))
+
+ # Content from File-like object
+ fileobj = StringIO.StringIO(self.test_data)
+ self.conn.put_object(
+ self.containername, self.objectname, contents=fileobj)
+ hdrs = self.conn.head_object(self.containername, self.objectname)
+ self.assertEqual(str(len(self.test_data)),
+ hdrs.get('content-length'))
+ self.assertEqual(self.etag, hdrs.get('etag'))
+ self.assertEqual('application/octet-stream', hdrs.get('content-type'))
+
+ # Content from File-like object, but read in chunks
+ fileobj = StringIO.StringIO(self.test_data)
+ self.conn.put_object(
+ self.containername, self.objectname,
+ contents=fileobj, content_length=len(self.test_data),
+ chunk_size=10)
+ hdrs = self.conn.head_object(self.containername, self.objectname)
+ self.assertEqual(str(len(self.test_data)),
+ hdrs.get('content-length'))
+ self.assertEqual(self.etag, hdrs.get('etag'))
+ self.assertEqual('application/octet-stream', hdrs.get('content-type'))
+
+ # Wrong etag arg, should raise an exception
+ self.assertRaises(
+ swiftclient.ClientException,
+ self.conn.put_object,
+ self.containername, self.objectname,
+ contents=self.test_data, etag='invalid')
+
+ def test_download_object(self):
+ # Download whole object
+ hdrs, body = self.conn.get_object(self.containername, self.objectname)
+ self.assertEqual(self.test_data, body)
+
+ # Download in chunks, should return a generator
+ hdrs, body = self.conn.get_object(
+ self.containername, self.objectname,
+ resp_chunk_size=10)
+ self.assertTrue(isinstance(body, types.GeneratorType))
+ self.assertEqual(self.test_data, ''.join(body))
+
+ def test_post_account(self):
+ self.conn.post_account({'x-account-meta-data': 'Something'})
+ headers = self.conn.head_account()
+ self.assertEqual('Something', headers.get('x-account-meta-data'))
+
+ def test_post_container(self):
+ self.conn.post_container(
+ self.containername, {'x-container-meta-color': 'Something'})
+
+ headers = self.conn.head_container(self.containername)
+ self.assertEqual('Something', headers.get('x-container-meta-color'))
+
+ def test_post_object(self):
+ self.conn.post_object(self.containername,
+ self.objectname,
+ {'x-object-meta-color': 'Something'})
+
+ headers = self.conn.head_object(self.containername, self.objectname)
+ self.assertEqual('Something', headers.get('x-object-meta-color'))
+
+ def test_get_capabilities(self):
+ resp = self.conn.get_capabilities()
+ self.assertTrue(resp.get('swift'))
+
+
+if __name__ == '__main__':
+ unittest.main()
diff --git a/tests/sample.conf b/tests/sample.conf
new file mode 100644
index 0000000..4578ab7
--- /dev/null
+++ b/tests/sample.conf
@@ -0,0 +1,17 @@
+[func_test]
+# sample config
+auth_host = 127.0.0.1
+auth_port = 8080
+auth_ssl = no
+auth_prefix = /auth/
+## sample config for Swift with Keystone
+#auth_version = 2
+#auth_host = localhost
+#auth_port = 5000
+#auth_ssl = no
+#auth_prefix = /v2.0/
+
+# Primary functional test account (needs admin access to the account)
+account = test
+username = tester
+password = testing
diff --git a/tests/unit/__init__.py b/tests/unit/__init__.py
new file mode 100644
index 0000000..e69de29
--- /dev/null
+++ b/tests/unit/__init__.py
diff --git a/tests/test_command_helpers.py b/tests/unit/test_command_helpers.py
index af81601..af81601 100644
--- a/tests/test_command_helpers.py
+++ b/tests/unit/test_command_helpers.py
diff --git a/tests/test_multithreading.py b/tests/unit/test_multithreading.py
index 875e43a..875e43a 100644
--- a/tests/test_multithreading.py
+++ b/tests/unit/test_multithreading.py
diff --git a/tests/test_swiftclient.py b/tests/unit/test_swiftclient.py
index 4e83a2d..4e83a2d 100644
--- a/tests/test_swiftclient.py
+++ b/tests/unit/test_swiftclient.py
diff --git a/tests/test_utils.py b/tests/unit/test_utils.py
index d9d74c5..d9d74c5 100644
--- a/tests/test_utils.py
+++ b/tests/unit/test_utils.py
diff --git a/tests/utils.py b/tests/unit/utils.py
index 914bc8f..914bc8f 100644
--- a/tests/utils.py
+++ b/tests/unit/utils.py
diff --git a/tox.ini b/tox.ini
index 5cee4cd..f88505c 100644
--- a/tox.ini
+++ b/tox.ini
@@ -10,7 +10,7 @@ setenv = VIRTUAL_ENV={envdir}
deps = -r{toxinidir}/requirements.txt
-r{toxinidir}/test-requirements.txt
-commands = python setup.py testr --testr-args="{posargs}"
+commands = python setup.py testr --testr-args="{posargs} tests.unit"
[testenv:pypy]
deps = setuptools<3.2