Fixes data and content-type headers in no content POST requests for objects
[pithos] / pithos / api / tests.py
index f39117e..6ec235c 100644 (file)
-from django.test.client import Client\r
-from django.test import TestCase\r
-from django.utils import simplejson as json\r
-from xml.dom import minidom\r
-import types\r
-import hashlib\r
-import os\r
-\r
-\r
-class AaiClient(Client):\r
-    def request(self, **request):\r
-        request['HTTP_X_AUTH_TOKEN'] = '46e427d657b20defe352804f0eb6f8a2'\r
-        return super(AaiClient, self).request(**request)\r
-\r
-\r
-class BaseTestCase(TestCase):\r
-    #TODO unauthorized request\r
-    def setUp(self):\r
-        self.client = AaiClient()\r
-        self.headers = {\r
-            'account':(\r
-                'X-Account-Container-Count',\r
-                'X-Account-Bytes-Used',\r
-                'Last-Modified',),\r
-            'container':(\r
-                'X-Container-Object-Count',\r
-                'X-Container-Bytes-Used',\r
-                'Last-Modified',),\r
-            'object':(\r
-                'ETag',\r
-                'Content-Length',\r
-                'Content-Type',\r
-                'Content-Encoding',\r
-                'Last-Modified',)}\r
-        self.contentTypes = {'xml':'application/xml',\r
-                             'json':'application/json',\r
-                             '':'text/plain'}\r
-        self.extended = {\r
-            'container':(\r
-                'name',\r
-                'count',\r
-                'bytes',\r
-                'last_modified'),\r
-            'object':(\r
-                'name',\r
-                'hash',\r
-                'bytes',\r
-                'content_type',\r
-                'content_encoding',\r
-                'last_modified',)}\r
-        self.return_codes = (400, 401, 404, 503,)\r
-    \r
-    def assertFault(self, response, status_code, name):\r
-        self.assertEqual(response.status_code, status_code)\r
-    \r
-    def assertBadRequest(self, response):\r
-        self.assertFault(response, 400, 'badRequest')\r
-    \r
-    def assertItemNotFound(self, response):\r
-        self.assertFault(response, 404, 'itemNotFound')\r
-\r
-    def assertUnauthorized(self, response):\r
-        self.assertFault(response, 401, 'unauthorized')\r
-\r
-    def assertServiceUnavailable(self, response):\r
-        self.assertFault(response, 503, 'serviceUnavailable')\r
-\r
-    def assertNonEmpty(self, response):\r
-        self.assertFault(response, 409, 'nonEmpty')\r
-\r
-    def assert_status(self, response, codes):\r
-        l = [elem for elem in self.return_codes]\r
-        if type(codes) == types.ListType:\r
-            l.extend(codes)\r
-        else:\r
-            l.append(codes)\r
-        self.assertTrue(response.status_code in l)\r
-\r
-    def get_account_meta(self, account):\r
-        path = '/v1/%s' % account\r
-        response = self.client.head(path)\r
-        self.assert_status(response, 204)\r
-        self.assert_headers(response, 'account')\r
-        return response\r
-\r
-    def list_containers(self, account, limit=10000, marker='', format=''):\r
-        params = locals()\r
-        params.pop('self')\r
-        params.pop('account')\r
-        path = '/v1/%s' % account\r
-        response = self.client.get(path, params)\r
-        self.assert_status(response, [200, 204])\r
-        response.content = response.content.strip()\r
-        if format:\r
-            self.assert_extended(response, format, 'container', limit)\r
-        else:\r
-            names = get_content_splitted(response)\r
-            self.assertTrue(len(names) <= limit)\r
-        return response\r
-\r
-    def update_account_meta(self, account, **metadata):\r
-        path = '/v1/%s' % account\r
-        response = self.client.post(path, **metadata)\r
-        response.content = response.content.strip()\r
-        self.assert_status(response, 202)\r
-        return response\r
-\r
-    def get_container_meta(self, account, container):\r
-        params = locals()\r
-        params.pop('self')\r
-        params.pop('account')\r
-        params.pop('container')\r
-        path = '/v1/%s/%s' %(account, container)\r
-        response = self.client.head(path, params)\r
-        response.content = response.content.strip()\r
-        self.assert_status(response, 204)\r
-        if response.status_code == 204:\r
-            self.assert_headers(response, 'container')\r
-        return response\r
-\r
-    def list_objects(self, account, container, limit=10000, marker='', prefix='', format='', path='', delimiter=''):\r
-        params = locals()\r
-        params.pop('self')\r
-        params.pop('account')\r
-        params.pop('container')\r
-        path = '/v1/%s/%s' % (account, container)\r
-        response = self.client.get(path, params)\r
-        response.content = response.content.strip()\r
-        if format:\r
-            self.assert_extended(response, format, 'object', limit)\r
-        self.assert_status(response, [200, 204])\r
-        return response\r
-\r
-    def create_container(self, account, name, **meta):\r
-        path = '/v1/%s/%s' %(account, name)\r
-        response = self.client.put(path, **meta)\r
-        response.content = response.content.strip()\r
-        self.assert_status(response, [201, 202])\r
-        return response\r
-\r
-    def update_container_meta(self, account, name, **meta):\r
-        path = '/v1/%s/%s' %(account, name)\r
-        response = self.client.post(path, data={}, content_type='text/xml', follow=False, **meta)\r
-        response.content = response.content.strip()\r
-        self.assert_status(response, 202)\r
-        return response\r
-\r
-    def delete_container(self, account, container):\r
-        path = '/v1/%s/%s' %(account, container)\r
-        response = self.client.delete(path)\r
-        response.content = response.content.strip()\r
-        self.assert_status(response, [204, 409])\r
-        return response\r
-\r
-    def get_object_meta(self, account, container, name):\r
-        path = '/v1/%s/%s/%s' %(account, container, name)\r
-        response = self.client.head(path)\r
-        response.content = response.content.strip()\r
-        self.assert_status(response, 204)\r
-        return response\r
-\r
-    def get_object(self, account, container, name, **headers):\r
-        path = '/v1/%s/%s/%s' %(account, container, name)\r
-        response = self.client.get(path, **headers)\r
-        response.content = response.content.strip()\r
-        self.assert_status(response, [200, 206, 304, 412, 416])\r
-        if response.status_code in [200, 206]:\r
-            self.assert_headers(response, 'object')\r
-        return response\r
-\r
-    def upload_object(self, account, container, name, data, content_type='application/json', **headers):\r
-        path = '/v1/%s/%s/%s' %(account, container, name)\r
-        response = self.client.put(path, data, content_type, **headers)\r
-        response.content = response.content.strip()\r
-        self.assert_status(response, [201, 411, 422])\r
-        if response.status_code == 201:\r
-            self.assertTrue(response['Etag'])\r
-        return response\r
-\r
-    def copy_object(self, account, container, name, src, **headers):\r
-        path = '/v1/%s/%s/%s' %(account, container, name)\r
-        headers['X-Copy-From'] = src\r
-        response = self.client.put(path, **headers)\r
-        response.content = response.content.strip()\r
-        self.assert_status(response, 201)\r
-        return response\r
-\r
-    def move_object(self, account, container, name, **headers):\r
-        path = '/v1/%s/%s/%s' % account, container, name\r
-        response = self.client.move(path, **headers)\r
-        response.content = response.content.strip()\r
-        self.assert_status(response, 201)\r
-        return response\r
-\r
-    def update_object_meta(self, account, container, name, **headers):\r
-        path = '/v1/%s/%s/%s' %(account, container, name)\r
-        response = self.client.post(path, **headers)\r
-        response.content = response.content.strip()\r
-        self.assert_status(response, 202)\r
-        return response\r
-\r
-    def delete_object(self, account, container, name):\r
-        path = '/v1/%s/%s/%s' %(account, container, name)\r
-        response = self.client.delete(path)\r
-        response.content = response.content.strip()\r
-        self.assert_status(response, 204)\r
-        return response\r
-\r
-    def assert_headers(self, response, type):\r
-        for item in self.headers[type]:\r
-            self.assertTrue(response[item])\r
-\r
-    def assert_extended(self, response, format, type, size):\r
-        self.assertEqual(response['Content-Type'].find(self.contentTypes[format]), 0)\r
-        if format == 'xml':\r
-            self.assert_xml(response, type, size)\r
-        elif format == 'json':\r
-            self.assert_json(response, type, size)\r
-\r
-    def assert_json(self, response, type, size):\r
-        convert = lambda s: s.lower()\r
-        info = [convert(elem) for elem in self.extended[type]]\r
-        data = json.loads(response.content)\r
-        self.assertTrue(len(data) <= size)\r
-        for item in info:\r
-            for i in data:\r
-                if 'subdir' in i.keys():\r
-                    continue\r
-                self.assertTrue(item in i.keys())\r
-\r
-    def assert_xml(self, response, type, size):\r
-        convert = lambda s: s.lower()\r
-        info = [convert(elem) for elem in self.extended[type]]\r
-        xml = minidom.parseString(response.content)\r
-        for item in info:\r
-            nodes = xml.getElementsByTagName(item)\r
-            self.assertTrue(nodes)\r
-            self.assertTrue(len(nodes) <= size)\r
-\r
-class ListContainers(BaseTestCase):\r
-    def setUp(self):\r
-        BaseTestCase.setUp(self)\r
-        self.account = 'test'\r
-        #create some containers\r
-        self.containers = ['apples', 'bananas', 'kiwis', 'oranges', 'pears']\r
-        for item in self.containers:\r
-            self.create_container(self.account, item)\r
-\r
-    def tearDown(self):\r
-        for c in get_content_splitted(self.list_containers(self.account)):\r
-            response = self.delete_container(self.account, c)\r
-\r
-    def test_list(self):\r
-        #list containers\r
-        response = self.list_containers(self.account)\r
-        containers = get_content_splitted(response)\r
-        self.assertEquals(self.containers, containers)\r
-\r
-    #def test_list_204(self):\r
-    #    response = self.list_containers('non-existing-account')\r
-    #    self.assertEqual(response.status_code, 204)\r
-\r
-    def test_list_with_limit(self):\r
-        limit = 2\r
-        response = self.list_containers(self.account, limit=limit)\r
-        containers = get_content_splitted(response)\r
-        self.assertEquals(len(containers), limit)\r
-        self.assertEquals(self.containers[:2], containers)\r
-\r
-    def test_list_with_marker(self):\r
-        limit = 2\r
-        marker = 'bananas'\r
-        response = self.list_containers(self.account, limit=limit, marker=marker)\r
-        containers =  get_content_splitted(response)\r
-        i = self.containers.index(marker) + 1\r
-        self.assertEquals(self.containers[i:(i+limit)], containers)\r
-        \r
-        marker = 'oranges'\r
-        response = self.list_containers(self.account, limit=limit, marker=marker)\r
-        containers = get_content_splitted(response)\r
-        i = self.containers.index(marker) + 1\r
-        self.assertEquals(self.containers[i:(i+limit)], containers)\r
-\r
-    def test_extended_list(self):\r
-        self.list_containers(self.account, limit=3, format='xml')\r
-        self.list_containers(self.account, limit=3, format='json')\r
-\r
-    def test_list_json_with_marker(self):\r
-        limit = 2\r
-        marker = 'bananas'\r
-        response = self.list_containers(self.account, limit=limit, marker=marker, format='json')\r
-        containers = json.loads(response.content)\r
-        self.assertEqual(containers[0]['name'], 'kiwis')\r
-        self.assertEqual(containers[1]['name'], 'oranges')\r
-\r
-    def test_list_xml_with_marker(self):\r
-        limit = 2\r
-        marker = 'oranges'\r
-        response = self.list_containers(self.account, limit=limit, marker=marker, format='xml')\r
-        xml = minidom.parseString(response.content)\r
-        nodes = xml.getElementsByTagName('name')\r
-        self.assertEqual(len(nodes), 1)\r
-        self.assertEqual(nodes[0].childNodes[0].data, 'pears')\r
-\r
-class AccountMetadata(BaseTestCase):\r
-    def setUp(self):\r
-        BaseTestCase.setUp(self)\r
-        self.account = 'test'\r
-        self.containers = ['apples', 'bananas', 'kiwis', 'oranges', 'pears']\r
-        for item in self.containers:\r
-            self.create_container(self.account, item)\r
-\r
-    def tearDown(self):\r
-        for c in  get_content_splitted(self.list_containers(self.account)):\r
-            self.delete_container(self.account, c)\r
-\r
-    def test_get_account_meta(self):\r
-        response = self.get_account_meta(self.account)\r
-        r2 = self.list_containers(self.account)\r
-        containers =  get_content_splitted(r2)\r
-        self.assertEqual(response['X-Account-Container-Count'], str(len(containers)))\r
-        size = 0\r
-        for c in containers:\r
-            r = self.get_container_meta(self.account, c)\r
-            size = size + int(r['X-Container-Bytes-Used'])\r
-        self.assertEqual(response['X-Account-Bytes-Used'], str(size))\r
-\r
-    #def test_get_account_401(self):\r
-    #    response = self.get_account_meta('non-existing-account')\r
-    #    print response\r
-    #    self.assertEqual(response.status_code, 401)\r
-\r
-    def test_update_meta(self):\r
-        meta = {'HTTP_X_ACCOUNT_META_TEST':'test', 'HTTP_X_ACCOUNT_META_TOST':'tost'}\r
-        response = self.update_account_meta(self.account, **meta)\r
-        response = self.get_account_meta(self.account)\r
-        for k,v in meta.items():\r
-            key = '-'.join(elem.capitalize() for elem in k.split('_')[1:])\r
-            self.assertTrue(response[key])\r
-            self.assertEqual(response[key], v)\r
-\r
-    #def test_invalid_account_update_meta(self):\r
-    #    with AssertInvariant(self.get_account_meta, self.account):\r
-    #        meta = {'HTTP_X_ACCOUNT_META_TEST':'test', 'HTTP_X_ACCOUNT_META_TOST':'tost'}\r
-    #        response = self.update_account_meta('non-existing-account', **meta)\r
-\r
-class ListObjects(BaseTestCase):\r
-    def setUp(self):\r
-        BaseTestCase.setUp(self)\r
-        self.account = 'test'\r
-        self.container = ['pears', 'apples']\r
-        self.create_container(self.account, self.container[0])\r
-        self.l = [\r
-            {'name':'kate.jpg',\r
-             'meta':{'HTTP_ETAG':'99914b932bd37a50b983c5e7c90ae93b',\r
-                     'HTTP_CONTENT-LENGTH':23,\r
-                     'HTTP_CONTENT_TYPE':'image/jpeg',\r
-                     'HTTP_CONTENT_ENCODING':'utf8',\r
-                     'HTTP_X_OBJECT_MANIFEST':123,\r
-                     'HTTP_X_OBJECT_META_TEST':'test1'}},\r
-            {'name':'kate_beckinsale.jpg',\r
-             'meta':{'HTTP_ETAG':'99914b932bd37a50b983c5e7c90ae93b',\r
-                     'HTTP_CONTENT-LENGTH':23,\r
-                     'HTTP_CONTENT_TYPE':'image/jpeg',\r
-                     'HTTP_CONTENT_ENCODING':'utf8',\r
-                     'HTTP_X_OBJECT_MANIFEST':123,\r
-                     'HTTP_X_OBJECT_META_TEST':'test1'}},\r
-            {'name':'How To Win Friends And Influence People.pdf',\r
-             'meta':{'HTTP_ETAG':'99914b932bd37a50b983c5e7c90ae93b',\r
-                     'HTTP_CONTENT-LENGTH':28,\r
-                     'HTTP_CONTENT_TYPE':'application/pdf',\r
-                     'HTTP_CONTENT_ENCODING':'utf8',\r
-                     'HTTP_X_OBJECT_MANIFEST':123,\r
-                     'HTTP_X_OBJECT_META_TEST':'test2'}},\r
-            {'name':'moms_birthday.jpg',\r
-             'meta':{'HTTP_ETAG':'99914b932bd37a50b983c5e7c90ae93b',\r
-                     'HTTP_CONTENT-LENGTH':255, 'HTTP_CONTENT_TYPE':'image/jpeg',\r
-                     'HTTP_CONTENT_ENCODING':'utf8',\r
-                     'HTTP_X_OBJECT_MANIFEST':123,\r
-                     'HTTP_X_OBJECT_META_TEST':'test3'}},\r
-            {'name':'poodle_strut.mov',\r
-             'meta':{'HTTP_ETAG':'99914b932bd37a50b983c5e7c90ae93b',\r
-                     'HTTP_CONTENT-LENGTH':23,\r
-                     'HTTP_CONTENT_TYPE':'image/jpeg',\r
-                     'HTTP_CONTENT_ENCODING':'utf8',\r
-                     'HTTP_X_OBJECT_MANIFEST':123,\r
-                     'HTTP_X_OBJECT_META_TEST':'test4'}},\r
-            {'name':'Disturbed - Down With The Sickness.mp3',\r
-             'meta':{'HTTP_ETAG':'99914b932bd37a50b983c5e7c90ae93b',\r
-                     'HTTP_CONTENT-LENGTH':23,\r
-                     'HTTP_CONTENT_TYPE':'image/jpeg',\r
-                     'HTTP_CONTENT_ENCODING':'utf8',\r
-                     'HTTP_X_OBJECT_MANIFEST':123,\r
-                     'HTTP_X_OBJECT_META_TEST':'test5'}},\r
-            {'name':'army_of_darkness.avi',\r
-             'meta':{'HTTP_ETAG':'99914b932bd37a50b983c5e7c90ae93b',\r
-                     'HTTP_CONTENT-LENGTH':23,\r
-                     'HTTP_CONTENT_TYPE':'image/jpeg',\r
-                     'HTTP_CONTENT_ENCODING':'utf8',\r
-                     'HTTP_X_OBJECT_MANIFEST':123,\r
-                     'HTTP_X_OBJECT_META_':'test6'}},\r
-            {'name':'the_mad.avi',\r
-             'meta':{'HTTP_ETAG':'wrong-hash',\r
-                     'HTTP_CONTENT-LENGTH':23,\r
-                     'HTTP_CONTENT_TYPE':'image/jpeg',\r
-                     'HTTP_CONTENT_ENCODING':'utf8',\r
-                     'HTTP_X_OBJECT_MANIFEST':123,\r
-                     'HTTP_X_OBJECT_META_TEST':'test7'}}\r
-        ]\r
-        for item in self.l[:-1]:\r
-            response = self.upload_object(self.account, self.container[0], item['name'], json.dumps({}), **item['meta'])\r
-        self.create_container(self.account, self.container[1])\r
-        l = [\r
-            {'name':'photos/animals/dogs/poodle.jpg',\r
-             'meta':{'HTTP_ETAG':'99914b932bd37a50b983c5e7c90ae93b',\r
-                     'HTTP_CONTENT-LENGTH':23,\r
-                     'HTTP_CONTENT_TYPE':'image/jpeg',\r
-                     'HTTP_CONTENT_ENCODING':'utf8',\r
-                     'HTTP_X_OBJECT_MANIFEST':123,\r
-                     'HTTP_X_OBJECT_META_TEST':'test1'}},\r
-            {'name':'photos/animals/dogs/terrier.jpg',\r
-             'meta':{'HTTP_ETAG':'99914b932bd37a50b983c5e7c90ae93b',\r
-                     'HTTP_CONTENT-LENGTH':23,\r
-                     'HTTP_CONTENT_TYPE':'image/jpeg',\r
-                     'HTTP_CONTENT_ENCODING':'utf8',\r
-                     'HTTP_X_OBJECT_MANIFEST':123,\r
-                     'HTTP_X_OBJECT_META_TEST':'test1'}},\r
-            {'name':'photos/animals/cats/persian.jpg',\r
-             'meta':{'HTTP_ETAG':'99914b932bd37a50b983c5e7c90ae93b',\r
-                     'HTTP_CONTENT-LENGTH':28,\r
-                     'HTTP_CONTENT_TYPE':'application/pdf',\r
-                     'HTTP_CONTENT_ENCODING':'utf8',\r
-                     'HTTP_X_OBJECT_MANIFEST':123,\r
-                     'HTTP_X_OBJECT_META_TEST':'test2'}},\r
-            {'name':'photos/animals/cats/siamese.jpg',\r
-             'meta':{'HTTP_ETAG':'99914b932bd37a50b983c5e7c90ae93b',\r
-                     'HTTP_CONTENT-LENGTH':255,\r
-                     'HTTP_CONTENT_TYPE':'image/jpeg',\r
-                     'HTTP_CONTENT_ENCODING':'utf8',\r
-                     'HTTP_X_OBJECT_MANIFEST':123,\r
-                     'HTTP_X_OBJECT_META_TEST':'test3'}},\r
-            {'name':'photos/plants/fern.jpg',\r
-             'meta':{'HTTP_ETAG':'99914b932bd37a50b983c5e7c90ae93b',\r
-                     'HTTP_CONTENT-LENGTH':23,\r
-                     'HTTP_CONTENT_TYPE':'image/jpeg',\r
-                     'HTTP_CONTENT_ENCODING':'utf8',\r
-                     'HTTP_X_OBJECT_MANIFEST':123,\r
-                     'HTTP_X_OBJECT_META_TEST':'test4'}},\r
-            {'name':'photos/plants/rose.jpg',\r
-             'meta':{'HTTP_ETAG':'99914b932bd37a50b983c5e7c90ae93b',\r
-                     'HTTP_CONTENT-LENGTH':23,\r
-                     'HTTP_CONTENT_TYPE':'image/jpeg',\r
-                     'HTTP_CONTENT_ENCODING':'utf8',\r
-                     'HTTP_X_OBJECT_MANIFEST':123,\r
-                     'XHTTP__OBJECT_META_TEST':'test5'}},\r
-            {'name':'photos/me.jpg',\r
-             'meta':{'HTTP_ETAG':'99914b932bd37a50b983c5e7c90ae93b',\r
-                     'HTTP_CONTENT-LENGTH':23,\r
-                     'HTTP_CONTENT_TYPE':'image/jpeg',\r
-                     'HTTP_CONTENT_ENCODING':'utf8',\r
-                     'X_OBJECT_MANIFEST':123,\r
-                     'HTTP_X_OBJECT_META_TEST':'test6'}},\r
-        ]\r
-        for item in l:\r
-            response = self.upload_object(self.account, self.container[1], item['name'], json.dumps({}), **item['meta'])\r
-\r
-    def tearDown(self):\r
-        for c in self.container:\r
-            self.delete_container_recursively(c)\r
-\r
-    def delete_container_recursively(self, c):\r
-        for obj in get_content_splitted(self.list_objects(self.account, c)):\r
-            self.delete_object(self.account, c, obj)\r
-        self.delete_container(self.account, c)\r
-\r
-    def test_list_objects(self):\r
-        response = self.list_objects(self.account, self.container[0])\r
-        objects = get_content_splitted(response)\r
-        l = [elem['name'] for elem in self.l[:-1]]\r
-        l.sort()\r
-        self.assertEqual(objects, l)\r
-\r
-    def test_list_objects_with_limit_marker(self):\r
-        response = self.list_objects(self.account, self.container[0], limit=2)\r
-        objects = get_content_splitted(response)\r
-        l = [elem['name'] for elem in self.l[:-1]]\r
-        l.sort()\r
-        self.assertEqual(objects, l[:2])\r
-        \r
-        response = self.list_objects(self.account, self.container[0], limit=4, marker='How To Win Friends And Influence People.pdf')\r
-        objects = get_content_splitted(response)\r
-        l = [elem['name'] for elem in self.l[:-1]]\r
-        l.sort()\r
-        self.assertEqual(objects, l[2:6])\r
-        \r
-        response = self.list_objects(self.account, self.container[0], limit=4, marker='moms_birthday.jpg')\r
-        objects = get_content_splitted(response)\r
-        l = [elem['name'] for elem in self.l[:-1]]\r
-        l.sort()\r
-        self.assertEqual(objects, l[-1:])\r
-\r
-    def test_list_pseudo_hierarchical_folders(self):\r
-        response = self.list_objects(self.account, self.container[1], prefix='photos', delimiter='/')\r
-        objects = get_content_splitted(response)\r
-        self.assertEquals(['photos/animals/', 'photos/me.jpg', 'photos/plants/'], objects)\r
-        \r
-        response = self.list_objects(self.account, self.container[1], prefix='photos/animals', delimiter='/')\r
-        objects = get_content_splitted(response)\r
-        self.assertEquals(['photos/animals/cats/', 'photos/animals/dogs/'], objects)\r
-        \r
-        response = self.list_objects(self.account, self.container[1], path='photos')\r
-        objects = get_content_splitted(response)\r
-        self.assertEquals(['photos/me.jpg'], objects)\r
-\r
-    def test_extended_list_json(self):\r
-        response = self.list_objects(self.account, self.container[1], format='json', limit=2, prefix='photos/animals', delimiter='/')\r
-        objects = json.loads(response.content)\r
-        self.assertEqual(objects[0]['subdir'], 'photos/animals/cats/')\r
-        self.assertEqual(objects[1]['subdir'], 'photos/animals/dogs/')\r
-\r
-    def test_extended_list_xml(self):\r
-        response = self.list_objects(self.account, self.container[1], format='xml', limit=4, prefix='photos', delimiter='/')\r
-        xml = minidom.parseString(response.content)\r
-        dirs = xml.getElementsByTagName('subdir')\r
-        self.assertEqual(len(dirs), 2)\r
-        self.assertEqual(dirs[0].attributes['name'].value, 'photos/animals/')\r
-        self.assertEqual(dirs[1].attributes['name'].value, 'photos/plants/')\r
-        \r
-        objects = xml.getElementsByTagName('name')\r
-        self.assertEqual(len(objects), 1)\r
-        self.assertEqual(objects[0].childNodes[0].data, 'photos/me.jpg')\r
-\r
-class ContainerMeta(BaseTestCase):\r
-    def setUp(self):\r
-        BaseTestCase.setUp(self)\r
-        self.account = 'test'\r
-        self.container = 'apples'\r
-        self.create_container(self.account, self.container)\r
-\r
-    def tearDown(self):\r
-        self.delete_container(self.account, self.container)\r
-\r
-    def test_update_meta(self):\r
-        meta = {'HTTP_X_CONTAINER_META_TEST':'test33', 'HTTP_X_CONTAINER_META_TOST':'tost22'}\r
-        response = self.update_container_meta(self.account, self.container, **meta)\r
-        response = self.get_container_meta(self.account, self.container)\r
-        for k,v in meta.items():\r
-            key = '-'.join(elem.capitalize() for elem in k.split('_')[1:])\r
-            self.assertTrue(response[key])\r
-            self.assertEqual(response[key], v)\r
-\r
-class CreateContainer(BaseTestCase):\r
-    def setUp(self):\r
-        BaseTestCase.setUp(self)\r
-        self.account = 'test'\r
-        self.containers = ['c1', 'c2']\r
-\r
-    def tearDown(self):\r
-        for c in self.containers:\r
-            r = self.delete_container(self.account, c)\r
-\r
-    def test_create(self):\r
-        response = self.create_container(self.account, self.containers[0])\r
-        if response.status_code == 201:\r
-            response = self.list_containers(self.account)\r
-            self.assertTrue(self.containers[0] in get_content_splitted(response))\r
-            r = self.get_container_meta(self.account, self.containers[0])\r
-            self.assertEqual(r.status_code, 204)\r
-\r
-    def test_create_twice(self):\r
-        response = self.create_container(self.account, self.containers[0])\r
-        if response.status_code == 201:\r
-            self.assertTrue(self.create_container(self.account, self.containers[0]).status_code, 202)\r
-\r
-class DeleteContainer(BaseTestCase):\r
-    def setUp(self):\r
-        BaseTestCase.setUp(self)\r
-        self.account = 'test'\r
-        self.containers = ['c1', 'c2']\r
-        for c in self.containers:\r
-            self.create_container(self.account, c)\r
-        obj = {'name':'kate.jpg',\r
-               'meta':{'HTTP_ETAG':'99914b932bd37a50b983c5e7c90ae93b',\r
-                       'HTTP_CONTENT-LENGTH':23,\r
-                       'HTTP_CONTENT_TYPE':'image/jpeg',\r
-                       'HTTP_CONTENT_ENCODING':'utf8',\r
-                       'HTTP_X_OBJECT_MANIFEST':123,\r
-                       'HTTP_X_OBJECT_META_TEST':'test1'\r
-                       }\r
-                }\r
-        self.upload_object(self.account, self.containers[1], obj['name'], json.dumps({}), **obj['meta'])\r
-\r
-    def tearDown(self):\r
-        for c in self.containers:\r
-            for o in get_content_splitted(self.list_objects(self.account, c)):\r
-                self.delete_object(self.account, c, o)\r
-            self.delete_container(self.account, c)\r
-\r
-    def test_delete(self):\r
-        r = self.delete_container(self.account, self.containers[0])\r
-        self.assertEqual(r.status_code, 204)\r
-\r
-    def test_delete_non_empty(self):\r
-        r = self.delete_container(self.account, self.containers[1])\r
-        self.assertNonEmpty(r)\r
-\r
-    def test_delete_invalid(self):\r
-        self.assertItemNotFound(self.delete_container(self.account, 'c3'))\r
-\r
-class GetObjects(BaseTestCase):\r
-    def setUp(self):\r
-        BaseTestCase.setUp(self)\r
-        self.account = 'test'\r
-        self.containers = ['c1', 'c2']\r
-        for c in self.containers:\r
-            self.create_container(self.account, c)\r
-        self.obj = {'name':'kate.jpg',\r
-               'meta':{'HTTP_ETAG':'99914b932bd37a50b983c5e7c90ae93b',\r
-                       'HTTP_CONTENT-LENGTH':23,\r
-                       'HTTP_CONTENT_TYPE':'image/jpeg',\r
-                       'HTTP_CONTENT_ENCODING':'utf8',\r
-                       'HTTP_X_OBJECT_MANIFEST':123,\r
-                       'HTTP_X_OBJECT_META_TEST':'test1'\r
-                       }\r
-                }\r
-        self.upload_object(self.account, self.containers[1], self.obj['name'], json.dumps({}), **self.obj['meta'])\r
-\r
-    def tearDown(self):\r
-        for c in self.containers:\r
-            for o in get_content_splitted(self.list_objects(self.account, c)):\r
-                self.delete_object(self.account, c, o)\r
-            self.delete_container(self.account, c)\r
-\r
-    def test_get(self):\r
-        r = self.get_object(self.account, self.containers[1], self.obj['name'])\r
-        self.assertEqual(r.status_code, 200)\r
-\r
-    def test_get_invalid(self):\r
-        r = self.get_object(self.account, self.containers[0], self.obj['name'])\r
-        self.assertItemNotFound(r)\r
-\r
-    def test_get_with_range(self):\r
-        return\r
-    \r
-    def test_get_with_if_match(self):\r
-        return\r
-    \r
-    def test_get_with_if_none_match(self):\r
-        return\r
-    \r
-    def test_get_with_if_modified_since(self):\r
-        return\r
-    \r
-    def test_get_with_if_unmodified_since(self):\r
-        return\r
-    \r
-    def test_get_with_several_headers(self):\r
-        return\r
-\r
-class UploadObject(BaseTestCase):\r
-    def setUp(self):\r
-        BaseTestCase.setUp(self)\r
-        self.account = 'test'\r
-        self.container = 'c1'\r
-        self.create_container(self.account, self.container)\r
-\r
-    def tearDown(self):\r
-        for o in get_content_splitted(self.list_objects(self.account, self.container)):\r
-            self.delete_object(self.account, self.container, o)\r
-        self.delete_container(self.account, self.container)\r
-    \r
-    def test_upload(self):\r
-        filename = './api/tests.py'\r
-        f = open(filename, 'r')\r
-        data = f.read()\r
-        hash = compute_hash(data)\r
-        meta={'HTTP_ETAG':hash,\r
-              'HTTP_CONTENT-LENGTH':os.path.getsize(filename),\r
-              'HTTP_CONTENT_TYPE':'text/x-java',\r
-              'HTTP_CONTENT_ENCODING':'us-ascii',\r
-              'HTTP_X_OBJECT_MANIFEST':123,\r
-              'HTTP_X_OBJECT_META_TEST':'test1'\r
-              }\r
-        r = self.upload_object(self.account, self.container, filename, data, content_type='plain/text',**meta)\r
-        self.assertEqual(r.status_code, 201)\r
-        \r
-class AssertInvariant(object):\r
-    def __init__(self, callable, *args, **kwargs):\r
-        self.callable = callable\r
-        self.args = args\r
-        self.kwargs = kwargs\r
-\r
-    def __enter__(self):\r
-        self.value = self.callable(*self.args, **self.kwargs)\r
-        return self.value\r
-\r
-    def __exit__(self, type, value, tb):\r
-        assert self.value == self.callable(*self.args, **self.kwargs)\r
-\r
-def get_content_splitted(response):\r
-    if response:\r
-        return response.content.split('\n')\r
-\r
-def compute_hash(data):\r
-    md5 = hashlib.md5()\r
-    offset = 0\r
-    md5.update(data)\r
-    return md5.hexdigest().lower()
\ No newline at end of file
+# Copyright 2011 GRNET S.A. All rights reserved.
+# 
+# Redistribution and use in source and binary forms, with or
+# without modification, are permitted provided that the following
+# conditions are met:
+# 
+#   1. Redistributions of source code must retain the above
+#      copyright notice, this list of conditions and the following
+#      disclaimer.
+# 
+#   2. Redistributions in binary form must reproduce the above
+#      copyright notice, this list of conditions and the following
+#      disclaimer in the documentation and/or other materials
+#      provided with the distribution.
+# 
+# THIS SOFTWARE IS PROVIDED BY GRNET S.A. ``AS IS'' AND ANY EXPRESS
+# OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
+# WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
+# PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL GRNET S.A OR
+# CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
+# SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
+# LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF
+# USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED
+# AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
+# LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN
+# ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
+# POSSIBILITY OF SUCH DAMAGE.
+# 
+# The views and conclusions contained in the software and
+# documentation are those of the authors and should not be
+# interpreted as representing official policies, either expressed
+# or implied, of GRNET S.A.
+
+from django.test.client import Client
+from django.test import TestCase
+from django.utils import simplejson as json
+from xml.dom import minidom
+import types
+import hashlib
+import os
+import mimetypes
+import random
+import datetime
+import string
+from pithos.backends import backend
+
+DATE_FORMATS = ["%a %b %d %H:%M:%S %Y",
+                "%A, %d-%b-%y %H:%M:%S GMT",
+                "%a, %d %b %Y %H:%M:%S GMT"]
+
+class AaiClient(Client):
+    def request(self, **request):
+        request['HTTP_X_AUTH_TOKEN'] = '0000'
+        return super(AaiClient, self).request(**request)
+
+class BaseTestCase(TestCase):
+    #TODO unauthorized request
+    def setUp(self):
+        self.client = AaiClient()
+        self.headers = {
+            'account':(
+                'X-Account-Container-Count',
+                'X-Account-Bytes-Used',
+                'Last-Modified',
+                'Content-Length',
+                'Date',
+                'Content-Type',),
+            'container':(
+                'X-Container-Object-Count',
+                'X-Container-Bytes-Used',
+                'Content-Type',
+                'Last-Modified',
+                'Content-Length',
+                'Date',
+                'X-Container-Block-Size',
+                'X-Container-Block-Hash',),
+            'object':(
+                'ETag',
+                'Content-Length',
+                'Content-Type',
+                'Content-Encoding',
+                'Last-Modified',
+                'Date',
+                'X-Object-Manifest',
+                'Content-Range',
+                'X-Object-Modified-By',
+                'X-Object-Version',
+                'X-Object-Version-Timestamp',)}
+        self.contentTypes = {'xml':'application/xml',
+                             'json':'application/json',
+                             '':'text/plain'}
+        self.extended = {
+            'container':(
+                'name',
+                'count',
+                'bytes',
+                'last_modified'),
+            'object':(
+                'name',
+                'hash',
+                'bytes',
+                'content_type',
+                'content_encoding',
+                'last_modified',)}
+        self.return_codes = (400, 401, 404, 503,)
+
+    def assertFault(self, response, status_code, name):
+        self.assertEqual(response.status_code, status_code)
+
+    def assertBadRequest(self, response):
+        self.assertFault(response, 400, 'badRequest')
+
+    def assertItemNotFound(self, response):
+        self.assertFault(response, 404, 'itemNotFound')
+
+    def assertUnauthorized(self, response):
+        self.assertFault(response, 401, 'unauthorized')
+
+    def assertServiceUnavailable(self, response):
+        self.assertFault(response, 503, 'serviceUnavailable')
+
+    def assertNonEmpty(self, response):
+        self.assertFault(response, 409, 'nonEmpty')
+
+    def assert_status(self, response, codes):
+        l = [elem for elem in self.return_codes]
+        if type(codes) == types.ListType:
+            l.extend(codes)
+        else:
+            l.append(codes)
+        self.assertTrue(response.status_code in l)
+
+    def get_account_meta(self, account, exp_meta={}):
+        path = '/v1/%s' % account
+        response = self.client.head(path)
+        self.assert_status(response, 204)
+        self.assert_headers(response, 'account', exp_meta)
+        return response
+
+    def list_containers(self, account, limit=10000, marker='', format='',
+                        **headers):
+        params = locals()
+        params.pop('self')
+        params.pop('account')
+        path = '/v1/%s' % account
+        response = self.client.get(path, params, **headers)
+        self.assert_status(response, [200, 204, 304, 412])
+        response.content = response.content.strip()
+        if format:
+            self.assert_extended(response, format, 'container', limit)
+        else:
+            names = get_content_splitted(response)
+            self.assertTrue(len(names) <= limit)
+        return response
+
+    def update_account_meta(self, account, **metadata):
+        path = '/v1/%s' % account
+        response = self.client.post(path, **metadata)
+        response.content = response.content
+        self.assert_status(response, 202)
+        return response
+
+    def get_container_meta(self, account, container, exp_meta={}):
+        params = locals()
+        params.pop('self')
+        params.pop('account')
+        params.pop('container')
+        path = '/v1/%s/%s' %(account, container)
+        response = self.client.head(path, params)
+        response.content = response.content
+        self.assert_status(response, 204)
+        if response.status_code == 204:
+            self.assert_headers(response, 'container', exp_meta)
+        return response
+
+    def list_objects(self, account, container, limit=10000, marker='',
+                     prefix='', format='', path='', delimiter='', meta='',
+                     **headers):
+        params = locals()
+        params.pop('self')
+        params.pop('account')
+        params.pop('container')
+        path = '/v1/%s/%s' % (account, container)
+        response = self.client.get(path, params, **headers)
+        response.content = response.content.strip()
+        if format:
+            self.assert_extended(response, format, 'object', limit)
+        self.assert_status(response, [200, 204, 304, 412])
+        return response
+
+    def create_container(self, account, name, **meta):
+        path = '/v1/%s/%s' %(account, name)
+        response = self.client.put(path, **meta)
+        response.content = response.content
+        self.assert_status(response, [201, 202])
+        return response
+
+    def update_container_meta(self, account, name, **meta):
+        path = '/v1/%s/%s' %(account, name)
+        response = self.client.post(path,
+                                    data=None,
+                                    content_type='text/xml',
+                                    follow=False, **meta)
+        response.content = response.content
+        self.assert_status(response, 202)
+        return response
+
+    def delete_container(self, account, container):
+        path = '/v1/%s/%s' %(account, container)
+        response = self.client.delete(path)
+        response.content = response.content
+        self.assert_status(response, [204, 409])
+        return response
+
+    def get_object_meta(self, account, container, name):
+        path = '/v1/%s/%s/%s' %(account, container, name)
+        response = self.client.head(path)
+        response.content = response.content
+        self.assert_status(response, 200)
+        return response
+
+    def get_object(self, account, container, name, format='', **headers):
+        path = '/v1/%s/%s/%s' %(account, container, name)
+        response = self.client.get(path, {'format':format}, **headers)
+        response.content = response.content
+        self.assert_status(response, [200, 206, 304, 412, 416])
+        if response.status_code in [200, 206]:
+            self.assert_headers(response, 'object')
+        return response
+
+    def upload_object(self, account, container, name, data, content_type='',
+                      **headers):
+        path = '/v1/%s/%s/%s' %(account, container, name)
+        response = self.client.put(path, data, content_type, **headers)
+        response.content = response.content
+        self.assert_status(response, [201, 411, 422])
+        if response.status_code == 201:
+            self.assertTrue(response['Etag'])
+        return response
+
+    def copy_object(self, account, container, name, src, **headers):
+        path = '/v1/%s/%s/%s' %(account, container, name)
+        headers['HTTP_X_COPY_FROM'] = src
+        response = self.client.put(path, **headers)
+        response.content = response.content
+        self.assert_status(response, 201)
+        return response
+
+    def move_object(self, account, container, name, src, **headers):
+        path = '/v1/%s/%s/%s' % (account, container, name)
+        headers['HTTP_X_MOVE_FROM'] = src
+        response = self.client.put(path, **headers)
+        response.content = response.content
+        self.assert_status(response, 201)
+        return response
+
+    def update_object(self, account, container, name, data='',
+                      content_type='', **headers):
+        path = '/v1/%s/%s/%s' %(account, container, name)
+        response = self.client.post(path, data, content_type, **headers)
+        response.content = response.content
+        self.assert_status(response, [202, 204, 416])
+        return response
+
+    def delete_object(self, account, container, name):
+        path = '/v1/%s/%s/%s' %(account, container, name)
+        response = self.client.delete(path)
+        response.content = response.content
+        self.assert_status(response, 204)
+        return response
+
+    def assert_headers(self, response, type, exp_meta={}):
+        entities = ['Account', 'Container', 'Container-Object', 'Object']
+        user_defined_meta = ['X-%s-Meta' %elem for elem in entities]
+        headers = [item for item in response._headers.values()]
+        t = tuple(user_defined_meta)
+        system_headers = [h for h in headers if not h[0].startswith(t)]
+        for h in system_headers:
+            self.assertTrue(h[0] in self.headers[type])
+            if exp_meta:
+                self.assertEqual(h[1], exp_meta[h[0]])
+
+    def assert_extended(self, response, format, type, size):
+        exp_content_type = self.contentTypes[format]
+        self.assertEqual(response['Content-Type'].find(exp_content_type), 0)
+        if format == 'xml':
+            self.assert_xml(response, type, size)
+        elif format == 'json':
+            self.assert_json(response, type, size)
+
+    def assert_json(self, response, type, size):
+        convert = lambda s: s.lower()
+        info = [convert(elem) for elem in self.extended[type]]
+        data = json.loads(response.content)
+        self.assertTrue(len(data) <= size)
+        for item in info:
+            for i in data:
+                if 'subdir' in i.keys():
+                    continue
+                self.assertTrue(item in i.keys())
+
+    def assert_xml(self, response, type, size):
+        convert = lambda s: s.lower()
+        info = [convert(elem) for elem in self.extended[type]]
+        try:
+            info.remove('content_encoding')
+        except ValueError:
+            pass
+        xml = minidom.parseString(response.content)
+        for item in info:
+            nodes = xml.getElementsByTagName(item)
+            self.assertTrue(nodes)
+            self.assertTrue(len(nodes) <= size)
+            
+
+    def upload_os_file(self, account, container, fullpath, meta={}):
+        try:
+            f = open(fullpath, 'r')
+            data = f.read()
+            name = os.path.split(fullpath)[-1]
+            return self.upload_data(account, container, name, data)    
+        except IOError:
+            return
+
+    def upload_random_data(self, account, container, name, length=1024,
+                           meta={}):
+        data = get_random_data(length)
+        return self.upload_data(account, container, name, data, meta)
+
+    def upload_data(self, account, container, name, data, meta={}):
+        obj = {}
+        obj['name'] = name
+        try:
+            obj['data'] = data
+            obj['hash'] = compute_md5_hash(obj['data'])
+            meta.update({'HTTP_X_OBJECT_META_TEST':'test1',
+                         'HTTP_ETAG':obj['hash']})
+            type, enc = mimetypes.guess_type(name)
+            meta['HTTP_CONTENT_TYPE'] = type and type or 'plain/text'
+            if enc:
+                meta['HTTP_CONTENT_ENCODING'] = enc
+            
+            obj['meta'] = meta
+            r = self.upload_object(account,
+                               container,
+                               obj['name'],
+                               obj['data'],
+                               meta['HTTP_CONTENT_TYPE'],
+                               **meta)
+            if r.status_code == 201:
+                return obj
+        except IOError:
+            return
+
+class AccountHead(BaseTestCase):
+    def setUp(self):
+        BaseTestCase.setUp(self)
+        self.account = 'test'
+        self.containers = ['apples', 'bananas', 'kiwis', 'oranges', 'pears']
+        for item in self.containers:
+            self.create_container(self.account, item)
+
+    def tearDown(self):
+        for c in  get_content_splitted(self.list_containers(self.account)):
+            self.delete_container(self.account, c)
+
+    def test_get_account_meta(self):
+        response = self.get_account_meta(self.account)
+        r2 = self.list_containers(self.account)
+        containers =  get_content_splitted(r2)
+        l = str(len(containers))
+        self.assertEqual(response['X-Account-Container-Count'], l)
+        size = 0
+        for c in containers:
+            r = self.get_container_meta(self.account, c)
+            size = size + int(r['X-Container-Bytes-Used'])
+        self.assertEqual(response['X-Account-Bytes-Used'], str(size))
+
+    #def test_get_account_401(self):
+    #    response = self.get_account_meta('non-existing-account')
+    #    print response
+    #    self.assertEqual(response.status_code, 401)
+
+class AccountGet(BaseTestCase):
+    def setUp(self):
+        BaseTestCase.setUp(self)
+        self.account = 'test'
+        #create some containers
+        self.containers = ['apples', 'bananas', 'kiwis', 'oranges', 'pears']
+        for item in self.containers:
+            self.create_container(self.account, item)
+
+    def tearDown(self):
+        for c in get_content_splitted(self.list_containers(self.account)):
+            response = self.delete_container(self.account, c)
+
+    def test_list(self):
+        #list containers
+        response = self.list_containers(self.account)
+        containers = get_content_splitted(response)
+        self.assertEquals(self.containers, containers)
+
+    #def test_list_204(self):
+    #    response = self.list_containers('non-existing-account')
+    #    self.assertEqual(response.status_code, 204)
+
+    def test_list_with_limit(self):
+        limit = 2
+        response = self.list_containers(self.account, limit=limit)
+        containers = get_content_splitted(response)
+        self.assertEquals(len(containers), limit)
+        self.assertEquals(self.containers[:2], containers)
+
+    def test_list_with_marker(self):
+        l = 2
+        m = 'bananas'
+        response = self.list_containers(self.account, limit=l, marker=m)
+        containers =  get_content_splitted(response)
+        i = self.containers.index(m) + 1
+        self.assertEquals(self.containers[i:(i+l)], containers)
+        
+        m = 'oranges'
+        response = self.list_containers(self.account, limit=l, marker=m)
+        containers = get_content_splitted(response)
+        i = self.containers.index(m) + 1
+        self.assertEquals(self.containers[i:(i+l)], containers)
+
+    #def test_extended_list(self):
+    #    self.list_containers(self.account, limit=3, format='xml')
+    #    self.list_containers(self.account, limit=3, format='json')
+
+    def test_list_json_with_marker(self):
+        l = 2
+        m = 'bananas'
+        response = self.list_containers(self.account, limit=l, marker=m,
+                                        format='json')
+        containers = json.loads(response.content)
+        self.assertEqual(containers[0]['name'], 'kiwis')
+        self.assertEqual(containers[1]['name'], 'oranges')
+
+    def test_list_xml_with_marker(self):
+        l = 2
+        m = 'oranges'
+        response = self.list_containers(self.account, limit=l, marker=m,
+                                        format='xml')
+        xml = minidom.parseString(response.content)
+        nodes = xml.getElementsByTagName('name')
+        self.assertEqual(len(nodes), 1)
+        self.assertEqual(nodes[0].childNodes[0].data, 'pears')
+
+    def test_if_modified_since(self):
+        t = datetime.datetime.utcnow()
+        t2 = t - datetime.timedelta(minutes=10)
+        
+        #add a new container
+        self.create_container(self.account,
+                              'dummy')
+
+        for f in DATE_FORMATS:
+            past = t2.strftime(f)
+            
+            headers = {'HTTP_IF_MODIFIED_SINCE':'%s' %past}
+            r = self.list_containers(self.account, **headers)
+            
+            #assert get success
+            self.assertEqual(r.status_code, 200)
+
+    def test_if_modified_since_invalid_date(self):
+        headers = {'HTTP_IF_MODIFIED_SINCE':''}
+        r = self.list_containers(self.account, **headers)
+            
+        #assert get success
+        self.assertEqual(r.status_code, 200)
+
+    def test_if_not_modified_since(self):
+        now = datetime.datetime.utcnow()
+        since = now + datetime.timedelta(1)
+        
+        for f in DATE_FORMATS:
+            headers = {'HTTP_IF_MODIFIED_SINCE':'%s' %since.strftime(f)}
+            r = self.list_containers(self.account, **headers)
+            
+            #assert not modified
+            self.assertEqual(r.status_code, 304)
+
+    def test_if_unmodified_since(self):
+        now = datetime.datetime.utcnow()
+        since = now + datetime.timedelta(1)
+        
+        for f in DATE_FORMATS:
+            headers = {'HTTP_IF_UNMODIFIED_SINCE':'%s' %since.strftime(f)}
+            r = self.list_containers(self.account, **headers)
+            
+            #assert success
+            self.assertEqual(r.status_code, 200)
+            self.assertEqual(self.containers, get_content_splitted(r))
+
+    def test_if_unmodified_since_precondition_failed(self):
+        t = datetime.datetime.utcnow()
+        t2 = t - datetime.timedelta(minutes=10)
+        
+        #add a new container
+        self.create_container(self.account,
+                              'dummy')
+        
+        for f in DATE_FORMATS:
+            past = t2.strftime(f)
+            
+            headers = {'HTTP_IF_UNMODIFIED_SINCE':'%s' %past}
+            r = self.list_containers(self.account, **headers)
+            
+            #assert get success
+            self.assertEqual(r.status_code, 412)
+
+class AccountPost(BaseTestCase):
+    def setUp(self):
+        BaseTestCase.setUp(self)
+        self.account = 'test'
+        self.containers = ['apples', 'bananas', 'kiwis', 'oranges', 'pears']
+        for item in self.containers:
+            self.create_container(self.account, item)
+
+    def tearDown(self):
+        for c in  get_content_splitted(self.list_containers(self.account)):
+            self.delete_container(self.account, c)
+
+    def test_update_meta(self):
+        meta = {'HTTP_X_ACCOUNT_META_TEST':'test',
+                'HTTP_X_ACCOUNT_META_TOST':'tost'}
+        response = self.update_account_meta(self.account, **meta)
+        response = self.get_account_meta(self.account)
+        for k,v in meta.items():
+            key = '-'.join(elem.capitalize() for elem in k.split('_')[1:])
+            self.assertTrue(response[key])
+            self.assertEqual(response[key], v)
+
+    #def test_invalid_account_update_meta(self):
+    #    with AssertInvariant(self.get_account_meta, self.account):
+    #        meta = {'HTTP_X_ACCOUNT_META_TEST':'test',
+    #               'HTTP_X_ACCOUNT_META_TOST':'tost'}
+    #        response = self.update_account_meta('non-existing-account', **meta)
+
+class ContainerHead(BaseTestCase):
+    def setUp(self):
+        BaseTestCase.setUp(self)
+        self.account = 'test'
+        self.container = 'apples'
+        self.create_container(self.account, self.container)
+
+    def tearDown(self):
+        for o in self.list_objects(self.account, self.container):
+            self.delete_object(self.account, self.container, o)
+        self.delete_container(self.account, self.container)
+
+    def test_get_meta(self):
+        headers = {'HTTP_X_OBJECT_META_TRASH':'true'}
+        t1 = datetime.datetime.utcnow()
+        o = self.upload_random_data(self.account,
+                                self.container,
+                                'McIntosh.jpg',
+                                meta=headers)
+        if o:
+            r = self.get_container_meta(self.account,
+                                        self.container)
+            self.assertEqual(r['X-Container-Object-Count'], '1')
+            self.assertEqual(r['X-Container-Bytes-Used'], str(len(o['data'])))
+            t2 = datetime.datetime.strptime(r['Last-Modified'], DATE_FORMATS[2])
+            delta = (t2 - t1)
+            threashold = datetime.timedelta(seconds=1) 
+            self.assertTrue(delta < threashold)
+            self.assertTrue(r['X-Container-Object-Meta'])
+            self.assertTrue('Trash' in r['X-Container-Object-Meta'])
+
+class ContainerGet(BaseTestCase):
+    def setUp(self):
+        BaseTestCase.setUp(self)
+        self.account = 'test'
+        self.container = ['pears', 'apples']
+        for c in self.container:
+            self.create_container(self.account, c)
+        self.obj = []
+        for o in o_names[:8]:
+            self.obj.append(self.upload_random_data(self.account,
+                                                    self.container[0],
+                                                    o))
+        for o in o_names[8:]:
+            self.obj.append(self.upload_random_data(self.account,
+                                                    self.container[1],
+                                                    o))
+
+    def tearDown(self):
+        for c in self.container:
+            for obj in get_content_splitted(self.list_objects(self.account, c)):
+                self.delete_object(self.account, c, obj)
+            self.delete_container(self.account, c)
+
+    def test_list_objects(self):
+        response = self.list_objects(self.account, self.container[0])
+        objects = get_content_splitted(response)
+        l = [elem['name'] for elem in self.obj[:8]]
+        l.sort()
+        self.assertEqual(objects, l)
+
+    def test_list_objects_with_limit_marker(self):
+        response = self.list_objects(self.account, self.container[0], limit=2)
+        objects = get_content_splitted(response)
+        l = [elem['name'] for elem in self.obj[:8]]
+        l.sort()
+        self.assertEqual(objects, l[:2])
+        
+        markers = ['How To Win Friends And Influence People.pdf',
+                   'moms_birthday.jpg']
+        limit = 4
+        for m in markers:
+            response = self.list_objects(self.account, self.container[0],
+                                         limit=limit, marker=m)
+            objects = get_content_splitted(response)
+            l = [elem['name'] for elem in self.obj[:8]]
+            l.sort()
+            start = l.index(m) + 1
+            end = start + limit
+            end = len(l) >= end and end or len(l)
+            self.assertEqual(objects, l[start:end])
+
+    def test_list_pseudo_hierarchical_folders(self):
+        response = self.list_objects(self.account, self.container[1],
+                                     prefix='photos', delimiter='/')
+        objects = get_content_splitted(response)
+        self.assertEquals(['photos/animals/', 'photos/me.jpg',
+                           'photos/plants/'], objects)
+        
+        response = self.list_objects(self.account, self.container[1],
+                                     prefix='photos/animals', delimiter='/')
+        objs = get_content_splitted(response)
+        l = ['photos/animals/cats/', 'photos/animals/dogs/']
+        self.assertEquals(l, objs)
+        
+        response = self.list_objects(self.account, self.container[1],
+                                     path='photos')
+        objects = get_content_splitted(response)
+        self.assertEquals(['photos/me.jpg'], objects)
+
+    def test_extended_list_json(self):
+        response = self.list_objects(self.account,
+                                     self.container[1],
+                                     format='json', limit=2,
+                                     prefix='photos/animals',
+                                     delimiter='/')
+        objects = json.loads(response.content)
+        self.assertEqual(objects[0]['subdir'], 'photos/animals/cats/')
+        self.assertEqual(objects[1]['subdir'], 'photos/animals/dogs/')
+
+    def test_extended_list_xml(self):
+        response = self.list_objects(self.account, self.container[1],
+                                     format='xml', limit=4, prefix='photos',
+                                     delimiter='/')
+        xml = minidom.parseString(response.content)
+        dirs = xml.getElementsByTagName('subdir')
+        self.assertEqual(len(dirs), 2)
+        self.assertEqual(dirs[0].attributes['name'].value, 'photos/animals/')
+        self.assertEqual(dirs[1].attributes['name'].value, 'photos/plants/')
+        
+        objects = xml.getElementsByTagName('name')
+        self.assertEqual(len(objects), 1)
+        self.assertEqual(objects[0].childNodes[0].data, 'photos/me.jpg')
+
+    def test_list_meta_double_matching(self):
+        meta = {'HTTP_X_OBJECT_META_QUALITY':'aaa',
+                'HTTP_X_OBJECT_META_STOCK':'true'}
+        r = self.update_object(self.account,
+                                    self.container[0],
+                                    self.obj[0]['name'],
+                                    
+                                    **meta)
+        r = self.list_objects(self.account,
+                          self.container[0],
+                          meta='Quality,Stock')
+        self.assertEqual(r.status_code, 200)
+        obj = get_content_splitted(r)
+        self.assertEqual(len(obj), 1)
+        self.assertTrue(obj, self.obj[0]['name'])
+
+    def test_list_using_meta(self):
+        meta = {'HTTP_X_OBJECT_META_QUALITY':'aaa'}
+        for o in self.obj[:2]:
+            r = self.update_object(self.account,
+                                    self.container[0],
+                                    o['name'],
+                                    **meta)
+        meta = {'HTTP_X_OBJECT_META_STOCK':'true'}
+        for o in self.obj[3:5]:
+            r = self.update_object(self.account,
+                                    self.container[0],
+                                    o['name'],
+                                    **meta)
+        
+        r = self.list_objects(self.account,
+                          self.container[0],
+                          meta='Quality')
+        self.assertEqual(r.status_code, 200)
+        obj = get_content_splitted(r)
+        self.assertEqual(len(obj), 2)
+        self.assertTrue(obj, [o['name'] for o in self.obj[:2]])
+        
+        # test case insensitive
+        r = self.list_objects(self.account,
+                          self.container[0],
+                          meta='quality')
+        self.assertEqual(r.status_code, 200)
+        obj = get_content_splitted(r)
+        self.assertEqual(len(obj), 2)
+        self.assertTrue(obj, [o['name'] for o in self.obj[:2]])
+        
+        # test multiple matches
+        r = self.list_objects(self.account,
+                          self.container[0],
+                          meta='Quality,Stock')
+        self.assertEqual(r.status_code, 200)
+        obj = get_content_splitted(r)
+        self.assertEqual(len(obj), 4)
+        self.assertTrue(obj, [o['name'] for o in self.obj[:4]])
+        
+        # test non 1-1 multiple match
+        r = self.list_objects(self.account,
+                          self.container[0],
+                          meta='Quality,aaaa')
+        self.assertEqual(r.status_code, 200)
+        obj = get_content_splitted(r)
+        self.assertEqual(len(obj), 2)
+        self.assertTrue(obj, [o['name'] for o in self.obj[:2]])
+
+    def test_if_modified_since(self):
+        t = datetime.datetime.utcnow()
+        t2 = t - datetime.timedelta(minutes=10)
+        
+        #add a new container
+        self.upload_random_data(self.account,
+                                self.container[0],
+                                'dummy.txt')
+
+        for f in DATE_FORMATS:
+            past = t2.strftime(f)
+            
+            headers = {'HTTP_IF_MODIFIED_SINCE':'%s' %past}
+            r = self.list_objects(self.account,
+                                  self.container[0], **headers)
+            
+            #assert get success
+            self.assertEqual(r.status_code, 200)
+
+    def test_if_modified_since_invalid_date(self):
+        headers = {'HTTP_IF_MODIFIED_SINCE':''}
+        r = self.list_objects(self.account,
+                              self.container[0], **headers)
+        
+        #assert get success
+        self.assertEqual(r.status_code, 200)
+
+    def test_if_not_modified_since(self):
+        now = datetime.datetime.utcnow()
+        since = now + datetime.timedelta(1)
+        
+        for f in DATE_FORMATS:
+            headers = {'HTTP_IF_MODIFIED_SINCE':'%s' %since.strftime(f)}
+            r = self.list_objects(self.account,
+                              self.container[0], **headers)
+        
+            #assert not modified
+            self.assertEqual(r.status_code, 304)
+
+    def test_if_unmodified_since(self):
+        now = datetime.datetime.utcnow()
+        since = now + datetime.timedelta(1)
+        
+        for f in DATE_FORMATS:
+            headers = {'HTTP_IF_UNMODIFIED_SINCE':'%s' %since.strftime(f)}
+            r = self.list_objects(self.account,
+                              self.container[0], **headers)
+        
+            #assert success
+            self.assertEqual(r.status_code, 200)
+            objlist = self.list_objects(self.account, self.container[0])
+            self.assertEqual(get_content_splitted(r),
+                             get_content_splitted(objlist))
+
+    def test_if_unmodified_since_precondition_failed(self):
+        t = datetime.datetime.utcnow()
+        t2 = t - datetime.timedelta(minutes=10)
+        
+        #add a new container
+        self.create_container(self.account,
+                              'dummy')
+
+        for f in DATE_FORMATS:
+            past = t2.strftime(f)
+            
+            headers = {'HTTP_IF_UNMODIFIED_SINCE':'%s' %past}
+            r = self.list_objects(self.account,
+                              self.container[0], **headers)
+        
+            #assert get success
+            self.assertEqual(r.status_code, 412)
+
+class ContainerPut(BaseTestCase):
+    def setUp(self):
+        BaseTestCase.setUp(self)
+        self.account = 'test'
+        self.containers = ['c1', 'c2']
+
+    def tearDown(self):
+        for c in self.containers:
+            r = self.delete_container(self.account, c)
+
+    def test_create(self):
+        response = self.create_container(self.account, self.containers[0])
+        if response.status_code == 201:
+            response = self.list_containers(self.account)
+            content = get_content_splitted(response)
+            self.assertTrue(self.containers[0] in content)
+            r = self.get_container_meta(self.account, self.containers[0])
+            self.assertEqual(r.status_code, 204)
+
+    def test_create_twice(self):
+        response = self.create_container(self.account, self.containers[0])
+        if response.status_code == 201:
+            r = self.create_container(self.account, self.containers[0])
+            self.assertTrue(r.status_code, 202)
+
+class ContainerPost(BaseTestCase):
+    def setUp(self):
+        BaseTestCase.setUp(self)
+        self.account = 'test'
+        self.container = 'apples'
+        self.create_container(self.account, self.container)
+
+    def tearDown(self):
+        for o in self.list_objects(self.account, self.container):
+            self.delete_object(self.account, self.container, o)
+        self.delete_container(self.account, self.container)
+
+    def test_update_meta(self):
+        meta = {'HTTP_X_CONTAINER_META_TEST':'test33',
+                'HTTP_X_CONTAINER_META_TOST':'tost22'}
+        response = self.update_container_meta(self.account, self.container,
+                                              **meta)
+        response = self.get_container_meta(self.account, self.container)
+        for k,v in meta.items():
+            key = '-'.join(elem.capitalize() for elem in k.split('_')[1:])
+            self.assertTrue(response[key])
+            self.assertEqual(response[key], v)
+
+class ContainerDelete(BaseTestCase):
+    def setUp(self):
+        BaseTestCase.setUp(self)
+        self.account = 'test'
+        self.containers = ['c1', 'c2']
+        for c in self.containers:
+            self.create_container(self.account, c)
+        self.upload_random_data(self.account,
+                                self.containers[1],
+                                'nice.jpg')
+
+    def tearDown(self):
+        for c in self.containers:
+            for o in get_content_splitted(self.list_objects(self.account, c)):
+                self.delete_object(self.account, c, o)
+            self.delete_container(self.account, c)
+
+    def test_delete(self):
+        r = self.delete_container(self.account, self.containers[0])
+        self.assertEqual(r.status_code, 204)
+
+    def test_delete_non_empty(self):
+        r = self.delete_container(self.account, self.containers[1])
+        self.assertNonEmpty(r)
+
+    def test_delete_invalid(self):
+        self.assertItemNotFound(self.delete_container(self.account, 'c3'))
+
+class ObjectHead(BaseTestCase):
+    pass
+
+class ObjectGet(BaseTestCase):
+    def setUp(self):
+        BaseTestCase.setUp(self)
+        self.account = 'test'
+        self.containers = ['c1', 'c2']
+        #create some containers
+        for c in self.containers:
+            self.create_container(self.account, c)
+        
+        #upload a file
+        self.objects = []
+        self.objects.append(self.upload_os_file(self.account,
+                            self.containers[1],
+                            './api/tests.py'))
+        self.objects.append(self.upload_os_file(self.account,
+                            self.containers[1],
+                            'settings.py'))
+
+    def tearDown(self):
+        for c in self.containers:
+            for o in get_content_splitted(self.list_objects(self.account, c)):
+                self.delete_object(self.account, c, o)
+            self.delete_container(self.account, c)
+
+    def test_get(self):
+        #perform get
+        r = self.get_object(self.account,
+                            self.containers[1],
+                            self.objects[0]['name'],
+                            self.objects[0]['meta'])
+        #assert success
+        self.assertEqual(r.status_code, 200)
+        
+        #assert content-type
+        self.assertEqual(r['Content-Type'],
+                         self.objects[0]['meta']['HTTP_CONTENT_TYPE'])
+
+    def test_get_invalid(self):
+        r = self.get_object(self.account,
+                            self.containers[0],
+                            self.objects[0]['name'])
+        self.assertItemNotFound(r)
+
+    def test_get_partial(self):
+        #perform get with range
+        headers = {'HTTP_RANGE':'bytes=0-499'}
+        r = self.get_object(self.account,
+                        self.containers[1],
+                        self.objects[0]['name'],
+                        **headers)
+        
+        #assert successful partial content
+        self.assertEqual(r.status_code, 206)
+        
+        #assert content-type
+        self.assertEqual(r['Content-Type'],
+                         self.objects[0]['meta']['HTTP_CONTENT_TYPE'])
+        
+        #assert content length
+        self.assertEqual(int(r['Content-Length']), 500)
+        
+        #assert content
+        self.assertEqual(self.objects[0]['data'][:500], r.content)
+
+    def test_get_final_500(self):
+        #perform get with range
+        headers = {'HTTP_RANGE':'bytes=-500'}
+        r = self.get_object(self.account,
+                        self.containers[1],
+                        self.objects[0]['name'],
+                        **headers)
+        
+        #assert successful partial content
+        self.assertEqual(r.status_code, 206)
+        
+        #assert content-type
+        self.assertEqual(r['Content-Type'],
+                         self.objects[0]['meta']['HTTP_CONTENT_TYPE'])
+        
+        #assert content length
+        self.assertEqual(int(r['Content-Length']), 500)
+        
+        #assert content
+        self.assertTrue(self.objects[0]['data'][-500:], r.content)
+
+    def test_get_rest(self):
+        #perform get with range
+        offset = len(self.objects[0]['data']) - 500
+        headers = {'HTTP_RANGE':'bytes=%s-' %offset}
+        r = self.get_object(self.account,
+                        self.containers[1],
+                        self.objects[0]['name'],
+                        **headers)
+        
+        #assert successful partial content
+        self.assertEqual(r.status_code, 206)
+        
+        #assert content-type
+        self.assertEqual(r['Content-Type'],
+                         self.objects[0]['meta']['HTTP_CONTENT_TYPE'])
+        
+        #assert content length
+        self.assertEqual(int(r['Content-Length']), 500)
+        
+        #assert content
+        self.assertTrue(self.objects[0]['data'][-500:], r.content)
+
+    def test_get_range_not_satisfiable(self):
+        #perform get with range
+        offset = len(self.objects[0]['data']) + 1
+        headers = {'HTTP_RANGE':'bytes=0-%s' %offset}
+        r = self.get_object(self.account,
+                        self.containers[1],
+                        self.objects[0]['name'],
+                        **headers)
+        
+        #assert Range Not Satisfiable
+        self.assertEqual(r.status_code, 416)
+
+    def test_multiple_range(self):
+        #perform get with multiple range
+        ranges = ['0-499', '-500', '1000-']
+        headers = {'HTTP_RANGE' : 'bytes=%s' % ','.join(ranges)}
+        r = self.get_object(self.account,
+                        self.containers[1],
+                        self.objects[0]['name'],
+                        **headers)
+        
+        # assert partial content
+        self.assertEqual(r.status_code, 206)
+        
+        # assert Content-Type of the reply will be multipart/byteranges
+        self.assertTrue(r['Content-Type'])
+        content_type_parts = r['Content-Type'].split()
+        self.assertEqual(content_type_parts[0], ('multipart/byteranges;'))
+        
+        boundary = '--%s' %content_type_parts[1].split('=')[-1:][0]
+        cparts = r.content.split(boundary)[1:-1]
+        
+        # assert content parts are exactly 2
+        self.assertEqual(len(cparts), len(ranges))
+        
+        # for each content part assert headers
+        i = 0
+        for cpart in cparts:
+            content = cpart.split('\r\n')
+            headers = content[1:3]
+            content_range = headers[0].split(': ')
+            self.assertEqual(content_range[0], 'Content-Range')
+            
+            r = ranges[i].split('-')
+            if not r[0] and not r[1]:
+                pass
+            elif not r[0]:
+                start = len(self.objects[0]['data']) - int(r[1])
+                end = len(self.objects[0]['data'])
+            elif not r[1]:
+                start = int(r[0])
+                end = len(self.objects[0]['data'])
+            else:
+                start = int(r[0])
+                end = int(r[1]) + 1
+            fdata = self.objects[0]['data'][start:end]
+            sdata = '\r\n'.join(content[4:-1])
+            self.assertEqual(len(fdata), len(sdata))
+            self.assertEquals(fdata, sdata)
+            i+=1
+
+    def test_multiple_range_not_satisfiable(self):
+        #perform get with multiple range
+        out_of_range = len(self.objects[0]['data']) + 1
+        ranges = ['0-499', '-500', '%d-' %out_of_range]
+        headers = {'HTTP_RANGE' : 'bytes=%s' % ','.join(ranges)}
+        r = self.get_object(self.account,
+                        self.containers[1],
+                        self.objects[0]['name'],
+                        **headers)
+        
+        # assert partial content
+        self.assertEqual(r.status_code, 416)
+
+    def test_get_with_if_match(self):
+        #perform get with If-Match
+        headers = {'HTTP_IF_MATCH':self.objects[0]['hash']}
+        r = self.get_object(self.account,
+                        self.containers[1],
+                        self.objects[0]['name'],
+                        **headers)
+        #assert get success
+        self.assertEqual(r.status_code, 200)
+        
+        #assert content-type
+        self.assertEqual(r['Content-Type'],
+                         self.objects[0]['meta']['HTTP_CONTENT_TYPE'])
+        
+        #assert response content
+        self.assertEqual(self.objects[0]['data'].strip(), r.content.strip())
+
+    def test_get_with_if_match_star(self):
+        #perform get with If-Match *
+        headers = {'HTTP_IF_MATCH':'*'}
+        r = self.get_object(self.account,
+                        self.containers[1],
+                        self.objects[0]['name'],
+                        **headers)
+        #assert get success
+        self.assertEqual(r.status_code, 200)
+        
+        #assert content-type
+        self.assertEqual(r['Content-Type'],
+                         self.objects[0]['meta']['HTTP_CONTENT_TYPE'])
+        
+        #assert response content
+        self.assertEqual(self.objects[0]['data'].strip(), r.content.strip())
+
+    def test_get_with_multiple_if_match(self):
+        #perform get with If-Match
+        etags = [i['hash'] for i in self.objects if i]
+        etags = ','.join('"%s"' % etag for etag in etags)
+        headers = {'HTTP_IF_MATCH':etags}
+        r = self.get_object(self.account,
+                        self.containers[1],
+                        self.objects[0]['name'],
+                        **headers)
+        #assert get success
+        self.assertEqual(r.status_code, 200)
+        
+        #assert content-type
+        self.assertEqual(r['Content-Type'],
+                         self.objects[0]['meta']['HTTP_CONTENT_TYPE'])
+        
+        #assert content-type
+        self.assertEqual(r['Content-Type'],
+                         self.objects[0]['meta']['HTTP_CONTENT_TYPE'])
+        
+        #assert response content
+        self.assertEqual(self.objects[0]['data'].strip(), r.content.strip())
+
+    def test_if_match_precondition_failed(self):
+        #perform get with If-Match
+        headers = {'HTTP_IF_MATCH':'123'}
+        r = self.get_object(self.account,
+                        self.containers[1],
+                        self.objects[0]['name'],
+                        **headers)
+        #assert precondition failed 
+        self.assertEqual(r.status_code, 412)
+
+    def test_if_none_match(self):
+        #perform get with If-None-Match
+        headers = {'HTTP_IF_NONE_MATCH':'123'}
+        r = self.get_object(self.account,
+                        self.containers[1],
+                        self.objects[0]['name'],
+                        **headers)
+        
+        #assert get success
+        self.assertEqual(r.status_code, 200)
+        
+        #assert content-type
+        self.assertEqual(r['Content-Type'],
+                         self.objects[0]['meta']['HTTP_CONTENT_TYPE'])
+
+    def test_if_none_match(self):
+        #perform get with If-None-Match *
+        headers = {'HTTP_IF_NONE_MATCH':'*'}
+        r = self.get_object(self.account,
+                        self.containers[1],
+                        self.objects[0]['name'],
+                        **headers)
+        
+        #assert get success
+        self.assertEqual(r.status_code, 304)
+
+    def test_if_none_match_not_modified(self):
+        #perform get with If-None-Match
+        headers = {'HTTP_IF_NONE_MATCH':'%s' %self.objects[0]['hash']}
+        r = self.get_object(self.account,
+                        self.containers[1],
+                        self.objects[0]['name'],
+                        **headers)
+        
+        #assert not modified
+        self.assertEqual(r.status_code, 304)
+        self.assertEqual(r['ETag'], self.objects[0]['hash'])
+
+    def test_if_modified_since(self):
+        t = datetime.datetime.utcnow()
+        t2 = t - datetime.timedelta(minutes=10)
+        
+        #modify the object
+        self.upload_object(self.account,
+                           self.containers[1],
+                           self.objects[0]['name'],
+                           self.objects[0]['data'][:200])
+        
+        for f in DATE_FORMATS:
+            past = t2.strftime(f)
+            
+            headers = {'HTTP_IF_MODIFIED_SINCE':'%s' %past}
+            r = self.get_object(self.account,
+                        self.containers[1],
+                        self.objects[0]['name'],
+                        **headers)
+            
+            #assert get success
+            self.assertEqual(r.status_code, 200)
+            
+            #assert content-type
+            self.assertEqual(r['Content-Type'],
+                             self.objects[0]['meta']['HTTP_CONTENT_TYPE'])   
+
+    def test_if_modified_since_invalid_date(self):
+        headers = {'HTTP_IF_MODIFIED_SINCE':''}
+        r = self.get_object(self.account,
+                    self.containers[1],
+                    self.objects[0]['name'],
+                    **headers)
+        
+        #assert get success
+        self.assertEqual(r.status_code, 200)
+        
+        #assert content-type
+        self.assertEqual(r['Content-Type'],
+                         self.objects[0]['meta']['HTTP_CONTENT_TYPE'])
+
+    def test_if_not_modified_since(self):
+        now = datetime.datetime.utcnow()
+        since = now + datetime.timedelta(1)
+        
+        for f in DATE_FORMATS:
+            headers = {'HTTP_IF_MODIFIED_SINCE':'%s' %since.strftime(f)}
+            r = self.get_object(self.account,
+                                self.containers[1],
+                                self.objects[0]['name'],
+                                **headers)
+            
+            #assert not modified
+            self.assertEqual(r.status_code, 304)
+
+    def test_if_unmodified_since(self):
+        now = datetime.datetime.utcnow()
+        since = now + datetime.timedelta(1)
+        
+        for f in DATE_FORMATS:
+            headers = {'HTTP_IF_UNMODIFIED_SINCE':'%s' %since.strftime(f)}
+            r = self.get_object(self.account,
+                                self.containers[1],
+                                self.objects[0]['name'],
+                                **headers)
+            #assert success
+            self.assertEqual(r.status_code, 200)
+            self.assertEqual(self.objects[0]['data'], r.content)
+            
+            #assert content-type
+            self.assertEqual(r['Content-Type'],
+                             self.objects[0]['meta']['HTTP_CONTENT_TYPE'])
+
+    def test_if_unmodified_since_precondition_failed(self):
+        t = datetime.datetime.utcnow()
+        t2 = t - datetime.timedelta(minutes=10)
+        
+        #modify the object
+        self.upload_object(self.account,
+                           self.containers[1],
+                           self.objects[0]['name'],
+                           self.objects[0]['data'][:200])
+        
+        for f in DATE_FORMATS:
+            past = t2.strftime(f)
+            
+            headers = {'HTTP_IF_UNMODIFIED_SINCE':'%s' %past}
+            r = self.get_object(self.account,
+                        self.containers[1],
+                        self.objects[0]['name'],
+                        **headers)
+            #assert get success
+            self.assertEqual(r.status_code, 412)
+
+    def test_hashes(self):
+        l = 8388609
+        fname = 'largefile'
+        o = self.upload_random_data(self.account,
+                                self.containers[1],
+                                fname,
+                                l)
+        if o:
+            r = self.get_object(self.account,
+                                self.containers[1],
+                                fname,
+                                'json')
+            body = json.loads(r.content)
+            hashes = body['hashes']
+            block_size = body['block_size']
+            block_hash = body['block_hash']
+            block_num = l/block_size == 0 and l/block_size or l/block_size + 1
+            self.assertTrue(len(hashes), block_num)
+            i = 0
+            for h in hashes:
+                start = i * block_size
+                end = (i + 1) * block_size
+                hash = compute_block_hash(o['data'][start:end], block_hash)
+                self.assertEqual(h, hash)
+                i += 1
+
+class ObjectPut(BaseTestCase):
+    def setUp(self):
+        BaseTestCase.setUp(self)
+        self.account = 'test'
+        self.container = 'c1'
+        self.create_container(self.account, self.container)
+        
+        self.src = os.path.join('.', 'api', 'tests.py')
+        self.dest = os.path.join('.', 'api', 'chunked_update_test_file')
+        create_chunked_update_test_file(self.src, self.dest)
+
+    def tearDown(self):
+        r = self.list_objects(self.account, self.container)
+        for o in get_content_splitted(r):
+            self.delete_object(self.account, self.container, o)
+        self.delete_container(self.account, self.container)
+        
+        # delete test file
+        os.remove(self.dest)
+
+    def test_upload(self):
+        filename = 'tests.py'
+        fullpath = os.path.join('.', 'api', filename) 
+        f = open(fullpath, 'r')
+        data = f.read()
+        hash = compute_md5_hash(data)
+        meta={'HTTP_ETAG':hash,
+              'HTTP_X_OBJECT_META_TEST':'test1'
+              }
+        type, enc = mimetypes.guess_type(fullpath)
+        meta['HTTP_CONTENT_TYPE'] = type and type or 'plain/text'
+        if enc:
+            meta['HTTP_CONTENT_ENCODING'] = enc
+        r = self.upload_object(self.account,
+                               self.container,
+                               filename,
+                               data,
+                               content_type=meta['HTTP_CONTENT_TYPE'],
+                               **meta)
+        self.assertEqual(r.status_code, 201)
+        r = self.get_object_meta(self.account, self.container, filename)
+        self.assertTrue(r['X-Object-Meta-Test'])
+        self.assertEqual(r['X-Object-Meta-Test'],
+                         meta['HTTP_X_OBJECT_META_TEST'])
+        
+        #assert uploaded content
+        r = self.get_object(self.account, self.container, filename)
+        self.assertEqual(os.path.getsize(fullpath), int(r['Content-Length']))
+        self.assertEqual(data.strip(), r.content.strip())
+
+    def test_upload_unprocessable_entity(self):
+        filename = 'tests.py'
+        fullpath = os.path.join('.', 'api', filename) 
+        f = open(fullpath, 'r')
+        data = f.read()
+        meta={'HTTP_ETAG':'123',
+              'HTTP_X_OBJECT_META_TEST':'test1'
+              }
+        type, enc = mimetypes.guess_type(fullpath)
+        meta['HTTP_CONTENT_TYPE'] = type and type or 'plain/text'
+        if enc:
+            meta['HTTP_CONTENT_ENCODING'] = enc
+        r = self.upload_object(self.account,
+                               self.container,
+                               filename,
+                               data,
+                               content_type = meta['HTTP_CONTENT_TYPE'],
+                               **meta)
+        self.assertEqual(r.status_code, 422)
+
+    def test_chucked_update(self):
+        objname = os.path.split(self.src)[-1:][0]
+        f = open(self.dest, 'r')
+        data = f.read()
+        meta = {}
+        type, enc = mimetypes.guess_type(self.dest)
+        meta['HTTP_CONTENT_TYPE'] = type and type or 'plain/text'
+        if enc:
+            meta['HTTP_CONTENT_ENCODING'] = enc
+        meta.update({'HTTP_TRANSFER_ENCODING':'chunked'})
+        r = self.upload_object(self.account,
+                               self.container,
+                               objname,
+                               data,
+                               content_type = 'plain/text',
+                               **meta)
+        self.assertEqual(r.status_code, 201)
+        
+        r = self.get_object(self.account,
+                            self.container,
+                            objname)
+        uploaded_data = r.content
+        f = open(self.src, 'r')
+        actual_data = f.read()
+        self.assertEqual(actual_data, uploaded_data)
+
+class ObjectCopy(BaseTestCase):
+    def setUp(self):
+        BaseTestCase.setUp(self)
+        self.account = 'test'
+        self.containers = ['c1', 'c2']
+        for c in self.containers:
+            self.create_container(self.account, c)
+        self.obj = self.upload_os_file(self.account,
+                            self.containers[0],
+                            './api/tests.py')
+
+    def tearDown(self):
+        for c in self.containers:
+            for o in get_content_splitted(self.list_objects(self.account, c)):
+                self.delete_object(self.account, c, o)
+            self.delete_container(self.account, c)
+
+    def test_copy(self):
+        with AssertInvariant(self.get_object_meta, self.account,
+                             self.containers[0], self.obj['name']):
+            #perform copy
+            meta = {'HTTP_X_OBJECT_META_TEST':'testcopy'}
+            src_path = os.path.join('/', self.containers[0], self.obj['name'])
+            r = self.copy_object(self.account,
+                             self.containers[0],
+                             'testcopy',
+                             src_path,
+                             **meta)
+            #assert copy success
+            self.assertEqual(r.status_code, 201)
+            
+            #assert access the new object
+            r = self.get_object_meta(self.account,
+                                     self.containers[0],
+                                     'testcopy')
+            self.assertTrue(r['X-Object-Meta-Test'])
+            self.assertTrue(r['X-Object-Meta-Test'], 'testcopy')
+            
+            #assert etag is the same
+            self.assertEqual(r['ETag'], self.obj['hash'])
+            
+            #assert src object still exists
+            r = self.get_object_meta(self.account, self.containers[0],
+                                     self.obj['name'])
+            self.assertEqual(r.status_code, 200)
+
+    def test_copy_from_different_container(self):
+        with AssertInvariant(self.get_object_meta,
+                             self.account,
+                             self.containers[0],
+                             self.obj['name']):
+            meta = {'HTTP_X_OBJECT_META_TEST':'testcopy'}
+            src_path = os.path.join('/', self.containers[0], self.obj['name'])
+            r = self.copy_object(self.account,
+                             self.containers[1],
+                             'testcopy',
+                             src_path,
+                             **meta)
+            self.assertEqual(r.status_code, 201)
+            
+            # assert updated metadata
+            r = self.get_object_meta(self.account,
+                                     self.containers[1],
+                                     'testcopy')
+            self.assertTrue(r['X-Object-Meta-Test'])
+            self.assertTrue(r['X-Object-Meta-Test'], 'testcopy')
+            
+            #assert src object still exists
+            r = self.get_object_meta(self.account, self.containers[0],
+                                     self.obj['name'])
+            self.assertEqual(r.status_code, 200)
+
+    def test_copy_invalid(self):
+        #copy from invalid object
+        meta = {'HTTP_X_OBJECT_META_TEST':'testcopy'}
+        r = self.copy_object(self.account,
+                         self.containers[1],
+                         'testcopy',
+                         os.path.join('/', self.containers[0], 'test.py'),
+                         **meta)
+        self.assertItemNotFound(r)
+        
+        #copy from invalid container
+        meta = {'HTTP_X_OBJECT_META_TEST':'testcopy'}
+        src_path = os.path.join('/', self.containers[1], self.obj['name'])
+        r = self.copy_object(self.account,
+                         self.containers[1],
+                         'testcopy',
+                         src_path,
+                         **meta)
+        self.assertItemNotFound(r)
+
+class ObjectMove(ObjectCopy):
+    def test_move(self):
+        #perform move
+        meta = {'HTTP_X_OBJECT_META_TEST':'testcopy'}
+        src_path = os.path.join('/', self.containers[0], self.obj['name'])
+        r = self.move_object(self.account,
+                         self.containers[0],
+                         'testcopy',
+                         src_path,
+                         **meta)
+        #assert successful move
+        self.assertEqual(r.status_code, 201)
+        
+        #assert updated metadata
+        r = self.get_object_meta(self.account,
+                                 self.containers[0],
+                                 'testcopy')
+        self.assertTrue(r['X-Object-Meta-Test'])
+        self.assertTrue(r['X-Object-Meta-Test'], 'testcopy')
+        
+        #assert src object no more exists
+        r = self.get_object_meta(self.account, self.containers[0],
+                                 self.obj['name'])
+        self.assertItemNotFound(r)
+
+class ObjectPost(BaseTestCase):
+    def setUp(self):
+        BaseTestCase.setUp(self)
+        self.account = 'test'
+        self.containers = ['c1', 'c2']
+        for c in self.containers:
+            self.create_container(self.account, c)
+        self.obj = self.upload_os_file(self.account,
+                                       self.containers[0],
+                                       './api/tests.py')
+
+    def tearDown(self):
+        for c in self.containers:
+            for o in get_content_splitted(self.list_objects(self.account, c)):
+                self.delete_object(self.account, c, o)
+            self.delete_container(self.account, c)
+
+    def test_update_meta(self):
+        #perform update metadata
+        more = {'HTTP_X_OBJECT_META_FOO':'foo',
+                'HTTP_X_OBJECT_META_BAR':'bar'}
+        r = self.update_object(self.account,
+                                self.containers[0],
+                                self.obj['name'],
+                                **more)
+        #assert request accepted
+        self.assertEqual(r.status_code, 202)
+        
+        #assert old metadata are still there
+        r = self.get_object_meta(self.account, self.containers[0],
+                                 self.obj['name'])
+        self.assertTrue('X-Object-Meta-Test' not in r.items())
+        
+        #assert new metadata have been updated
+        for k,v in more.items():
+            key = '-'.join(elem.capitalize() for elem in k.split('_')[1:])
+            self.assertTrue(r[key])
+            self.assertTrue(r[key], v)
+
+    def test_update_object(self,
+                           first_byte_pos=0,
+                           last_byte_pos=499,
+                           instance_length = True,
+                           content_length = 500):
+        l = len(self.obj['data'])
+        length =  instance_length and l or '*'
+        range = 'bytes %d-%d/%s' %(first_byte_pos,
+                                       last_byte_pos,
+                                       length)
+        partial = last_byte_pos - first_byte_pos + 1
+        data = get_random_data(partial)
+        more = {'HTTP_CONTENT_RANGE':range}
+        if content_length:
+            more.update({'CONTENT_LENGTH':'%s' % content_length})
+        
+        r = self.update_object(self.account,
+                                self.containers[0],
+                                self.obj['name'],
+                                data,
+                                'application/octet-stream',
+                                **more)
+        
+        if partial < 0 or (instance_length and l <= last_byte_pos):
+            self.assertEqual(r.status_code, 416)    
+        elif content_length and content_length != partial:
+            self.assertEqual(r.status_code, 400)
+        else:
+            self.assertEqual(r.status_code, 204)
+            
+            #check modified object
+            r = self.get_object(self.account,
+                            self.containers[0],
+                            self.obj['name'])
+            self.assertEqual(r.content[0:partial], data)
+            self.assertEqual(r.content[partial:l], self.obj['data'][partial:l])
+
+    def test_update_object_no_content_length(self):
+        self.test_update_object(content_length = None)
+
+    def test_update_object_invalid_content_length(self):
+        with AssertContentInvariant(self.get_object, self.account, self.containers[0],
+                            self.obj['name']):
+            self.test_update_object(content_length = 1000)
+
+    def test_update_object_with_unknown_instance_length(self):
+        self.test_update_object(instance_length = False)
+
+    def test_update_object_invalid_range(self):
+        with AssertContentInvariant(self.get_object, self.account, self.containers[0],
+                            self.obj['name']):
+            self.test_update_object(499, 0, True)
+    
+    def test_update_object_invalid_range_and_length(self):
+        with AssertContentInvariant(self.get_object, self.account, self.containers[0],
+                            self.obj['name']):
+            self.test_update_object(499, 0, True, -1)
+    
+    def test_update_object_invalid_range_with_no_content_length(self):
+        with AssertContentInvariant(self.get_object, self.account, self.containers[0],
+                            self.obj['name']):
+            self.test_update_object(499, 0, True, content_length = None)
+    
+    def test_update_object_out_of_limits(self):    
+        with AssertContentInvariant(self.get_object, self.account, self.containers[0],
+                            self.obj['name']):
+            l = len(self.obj['data'])
+            self.test_update_object(0, l+1, True)
+
+    def test_append(self):
+        data = get_random_data(500)
+        more = {'CONTENT_LENGTH':'500',
+                'HTTP_CONTENT_RANGE':'bytes */*'}
+        
+        r = self.update_object(self.account,
+                                self.containers[0],
+                                self.obj['name'],
+                                data,
+                                'application/octet-stream',
+                                **more)
+        
+        self.assertEqual(r.status_code, 204)
+        
+        r = self.get_object(self.account,
+                                self.containers[0],
+                                self.obj['name'])
+        self.assertEqual(len(r.content), len(self.obj['data']) + 500)
+        self.assertEqual(r.content[:-500], self.obj['data'])
+
+    def test_update_with_chunked_transfer(self):
+        data, pure = create_random_chunked_data()
+        dl = len(pure)
+        fl = len(self.obj['data'])
+        meta = {'HTTP_TRANSFER_ENCODING':'chunked',
+                'HTTP_CONTENT_RANGE':'bytes 0-/%d' %fl}
+        r = self.update_object(self.account,
+                                self.containers[0],
+                                self.obj['name'],
+                                data,
+                                'application/octet-stream',
+                                **meta)
+        
+        #check modified object
+        r = self.get_object(self.account,
+                        self.containers[0],
+                        self.obj['name'])
+        self.assertEqual(r.content[0:dl], pure)
+        self.assertEqual(r.content[dl:fl], self.obj['data'][dl:fl])
+
+    def test_update_with_chunked_transfer_strict_range(self):
+        data, pure = create_random_chunked_data()
+        dl = len(pure) - 1
+        fl = len(self.obj['data'])
+        meta = {'HTTP_TRANSFER_ENCODING':'chunked',
+                'HTTP_CONTENT_RANGE':'bytes 0-%d/%d' %(dl, fl)}
+        r = self.update_object(self.account,
+                                self.containers[0],
+                                self.obj['name'],
+                                data,
+                                'application/octet-stream',
+                                **meta)
+        
+        #check modified object
+        r = self.get_object(self.account,
+                        self.containers[0],
+                        self.obj['name'])
+        self.assertEqual(r.content[0:dl+1], pure)
+        self.assertEqual(r.content[dl+1:fl], self.obj['data'][dl+1:fl])
+
+class ObjectDelete(BaseTestCase):
+    def setUp(self):
+        BaseTestCase.setUp(self)
+        self.account = 'test'
+        self.containers = ['c1', 'c2']
+        for c in self.containers:
+            self.create_container(self.account, c)
+        self.obj = self.upload_os_file(self.account,
+                            self.containers[0],
+                            './api/tests.py')
+
+    def tearDown(self):
+        for c in self.containers:
+            for o in get_content_splitted(self.list_objects(self.account, c)):
+                self.delete_object(self.account, c, o)
+            self.delete_container(self.account, c)
+
+    def test_delete(self):
+        #perform delete object
+        r = self.delete_object(self.account, self.containers[0],
+                               self.obj['name'])
+        
+        #assert success
+        self.assertEqual(r.status_code, 204)
+
+    def test_delete_invalid(self):
+        #perform delete object
+        r = self.delete_object(self.account, self.containers[1],
+                               self.obj['name'])
+        
+        #assert failure
+        self.assertItemNotFound(r)
+
+class AssertInvariant(object):
+    def __init__(self, callable, *args, **kwargs):
+        self.callable = callable
+        self.args = args
+        self.kwargs = kwargs
+
+    def __enter__(self):
+        self.items = self.callable(*self.args, **self.kwargs).items()
+        return self.items
+
+    def __exit__(self, type, value, tb):
+        items = self.callable(*self.args, **self.kwargs).items()
+        assert self.items == items
+
+class AssertContentInvariant(object):
+    def __init__(self, callable, *args, **kwargs):
+        self.callable = callable
+        self.args = args
+        self.kwargs = kwargs
+
+    def __enter__(self):
+        self.content = self.callable(*self.args, **self.kwargs).content
+        return self.content
+
+    def __exit__(self, type, value, tb):
+        content = self.callable(*self.args, **self.kwargs).content
+        assert self.content == content
+
+def get_content_splitted(response):
+    if response:
+        return response.content.split('\n')
+
+def compute_md5_hash(data):
+    md5 = hashlib.md5()
+    offset = 0
+    md5.update(data)
+    return md5.hexdigest().lower()
+
+def compute_block_hash(data, algorithm):
+    h = hashlib.new(algorithm)
+    h.update(data.rstrip('\x00'))
+    return h.hexdigest()
+
+def create_chunked_update_test_file(src, dest):
+    fr = open(src, 'r')
+    fw = open(dest, 'w')
+    data = fr.readline()
+    while data:
+        fw.write(hex(len(data)))
+        fw.write('\r\n')
+        fw.write(data)
+        data = fr.readline()
+    fw.write(hex(0))
+    fw.write('\r\n')
+
+def create_random_chunked_data(rows=5):
+    i = 0
+    out = []
+    pure= []
+    while i < rows:
+        data = get_random_data(random.randint(1, 100))
+        out.append(hex(len(data)))
+        out.append(data)
+        pure.append(data)
+        i+=1
+    out.append(hex(0))
+    out.append('\r\n')
+    return '\r\n'.join(out), ''.join(pure)
+
+def get_random_data(length=500):
+    char_set = string.ascii_uppercase + string.digits
+    return ''.join(random.choice(char_set) for x in range(length))
+
+o_names = ['kate.jpg',
+           'kate_beckinsale.jpg',
+           'How To Win Friends And Influence People.pdf',
+           'moms_birthday.jpg',
+           'poodle_strut.mov',
+           'Disturbed - Down With The Sickness.mp3',
+           'army_of_darkness.avi',
+           'the_mad.avi',
+           'photos/animals/dogs/poodle.jpg',
+           'photos/animals/dogs/terrier.jpg',
+           'photos/animals/cats/persian.jpg',
+           'photos/animals/cats/siamese.jpg',
+           'photos/plants/fern.jpg',
+           'photos/plants/rose.jpg',
+           'photos/me.jpg']