2013-11-11 21:41:33 +00:00
|
|
|
import tempfile
|
|
|
|
|
|
|
|
|
2014-06-17 20:03:43 +00:00
|
|
|
class StoragePaths(object):
|
|
|
|
shared_images = 'sharedimages'
|
|
|
|
|
2014-06-17 20:37:48 +00:00
|
|
|
@staticmethod
|
|
|
|
def temp_store_handler():
|
|
|
|
tmpf = tempfile.TemporaryFile()
|
2014-11-24 21:07:38 +00:00
|
|
|
|
2014-06-17 20:37:48 +00:00
|
|
|
def fn(buf):
|
|
|
|
try:
|
|
|
|
tmpf.write(buf)
|
|
|
|
except IOError:
|
|
|
|
pass
|
2014-11-24 21:07:38 +00:00
|
|
|
|
2014-06-17 20:37:48 +00:00
|
|
|
return tmpf, fn
|
|
|
|
|
2014-06-17 20:03:43 +00:00
|
|
|
def image_path(self, storage_uuid):
|
|
|
|
return '{0}/{1}/'.format(self.shared_images, storage_uuid)
|
|
|
|
|
|
|
|
def image_json_path(self, storage_uuid):
|
|
|
|
base_path = self.image_path(storage_uuid)
|
|
|
|
return '{0}json'.format(base_path)
|
|
|
|
|
|
|
|
def image_layer_path(self, storage_uuid):
|
|
|
|
base_path = self.image_path(storage_uuid)
|
|
|
|
return '{0}layer'.format(base_path)
|
|
|
|
|
|
|
|
def image_ancestry_path(self, storage_uuid):
|
|
|
|
base_path = self.image_path(storage_uuid)
|
|
|
|
return '{0}ancestry'.format(base_path)
|
|
|
|
|
|
|
|
def image_file_trie_path(self, storage_uuid):
|
|
|
|
base_path = self.image_path(storage_uuid)
|
|
|
|
return '{0}files.trie'.format(base_path)
|
|
|
|
|
|
|
|
def image_file_diffs_path(self, storage_uuid):
|
|
|
|
base_path = self.image_path(storage_uuid)
|
|
|
|
return '{0}diffs.json'.format(base_path)
|
|
|
|
|
2013-11-07 04:21:12 +00:00
|
|
|
|
2014-06-17 20:03:43 +00:00
|
|
|
class BaseStorage(StoragePaths):
|
2013-11-07 04:21:12 +00:00
|
|
|
"""Storage is organized as follow:
|
|
|
|
$ROOT/images/<image_id>/json
|
|
|
|
$ROOT/images/<image_id>/layer
|
|
|
|
$ROOT/repositories/<namespace>/<repository_name>/<tag_name>
|
|
|
|
"""
|
|
|
|
|
|
|
|
# Useful if we want to change those locations later without rewriting
|
|
|
|
# the code which uses Storage
|
|
|
|
repositories = 'repositories'
|
|
|
|
images = 'images'
|
|
|
|
# Set the IO buffer to 64kB
|
|
|
|
buffer_size = 64 * 1024
|
|
|
|
|
2014-09-09 19:54:03 +00:00
|
|
|
def get_direct_download_url(self, path, expires_in=60, requires_cors=False):
|
2013-12-04 00:39:07 +00:00
|
|
|
return None
|
|
|
|
|
2014-09-09 19:54:03 +00:00
|
|
|
def get_direct_upload_url(self, path, mime_type, requires_cors=True):
|
|
|
|
return None
|
|
|
|
|
|
|
|
def get_supports_resumable_downloads(self):
|
2014-07-02 04:39:59 +00:00
|
|
|
return False
|
|
|
|
|
2013-11-07 04:21:12 +00:00
|
|
|
def get_content(self, path):
|
|
|
|
raise NotImplementedError
|
|
|
|
|
|
|
|
def put_content(self, path, content):
|
|
|
|
raise NotImplementedError
|
|
|
|
|
|
|
|
def stream_read(self, path):
|
|
|
|
raise NotImplementedError
|
|
|
|
|
|
|
|
def stream_read_file(self, path):
|
|
|
|
raise NotImplementedError
|
|
|
|
|
2014-09-11 19:33:10 +00:00
|
|
|
def stream_write(self, path, fp, content_type=None, content_encoding=None):
|
2013-11-07 04:21:12 +00:00
|
|
|
raise NotImplementedError
|
|
|
|
|
|
|
|
def list_directory(self, path=None):
|
|
|
|
raise NotImplementedError
|
|
|
|
|
|
|
|
def exists(self, path):
|
|
|
|
raise NotImplementedError
|
|
|
|
|
|
|
|
def remove(self, path):
|
|
|
|
raise NotImplementedError
|
2014-09-09 19:54:03 +00:00
|
|
|
|
|
|
|
def get_checksum(self, path):
|
2014-11-24 21:07:38 +00:00
|
|
|
raise NotImplementedError
|