Merge latest trunk
This commit is contained in:
commit
38cc809bb8
@ -76,6 +76,8 @@ s3_store_create_bucket_on_put = False
|
|||||||
|
|
||||||
# ============ Image Cache Options ========================
|
# ============ Image Cache Options ========================
|
||||||
|
|
||||||
|
image_cache_enabled = False
|
||||||
|
|
||||||
# Directory that the Image Cache writes data to
|
# Directory that the Image Cache writes data to
|
||||||
# Make sure this is also set in glance-pruner.conf
|
# Make sure this is also set in glance-pruner.conf
|
||||||
image_cache_datadir = /var/lib/glance/image-cache/
|
image_cache_datadir = /var/lib/glance/image-cache/
|
||||||
|
@ -214,26 +214,30 @@ class Controller(api.BaseController):
|
|||||||
if cache.enabled:
|
if cache.enabled:
|
||||||
if cache.hit(id):
|
if cache.hit(id):
|
||||||
# hit
|
# hit
|
||||||
logger.debug("image cache HIT, retrieving image '%s'"
|
logger.debug("image '%s' is a cache HIT", id)
|
||||||
" from cache", id)
|
|
||||||
image_iterator = get_from_cache(image, cache)
|
image_iterator = get_from_cache(image, cache)
|
||||||
else:
|
else:
|
||||||
# miss
|
# miss
|
||||||
logger.debug("image cache MISS, retrieving image '%s'"
|
logger.debug("image '%s' is a cache MISS", id)
|
||||||
" from store and tee'ing into cache", id)
|
|
||||||
|
|
||||||
# We only want to tee-into the cache if we're not currently
|
# Make sure we're not already prefetching or caching the image
|
||||||
# prefetching an image
|
# that just generated the miss
|
||||||
image_id = image['id']
|
if cache.is_image_currently_prefetching(id):
|
||||||
if cache.is_image_currently_prefetching(image_id):
|
logger.debug("image '%s' is already being prefetched,"
|
||||||
|
" not tee'ing into the cache", id)
|
||||||
|
image_iterator = get_from_store(image)
|
||||||
|
elif cache.is_image_currently_being_written(id):
|
||||||
|
logger.debug("image '%s' is already being cached,"
|
||||||
|
" not tee'ing into the cache", id)
|
||||||
image_iterator = get_from_store(image)
|
image_iterator = get_from_store(image)
|
||||||
else:
|
else:
|
||||||
# NOTE(sirp): If we're about to download and cache an
|
# NOTE(sirp): If we're about to download and cache an
|
||||||
# image which is currently in the prefetch queue, just
|
# image which is currently in the prefetch queue, just
|
||||||
# delete the queue items since we're caching it anyway
|
# delete the queue items since we're caching it anyway
|
||||||
if cache.is_image_queued_for_prefetch(image_id):
|
if cache.is_image_queued_for_prefetch(id):
|
||||||
cache.delete_queued_prefetch_image(image_id)
|
cache.delete_queued_prefetch_image(id)
|
||||||
|
|
||||||
|
logger.debug("tee'ing image '%s' into cache", id)
|
||||||
image_iterator = get_from_store_tee_into_cache(
|
image_iterator = get_from_store_tee_into_cache(
|
||||||
image, cache)
|
image, cache)
|
||||||
else:
|
else:
|
||||||
|
@ -59,17 +59,16 @@ def get_backend_class(backend):
|
|||||||
:param backend: Name of backend to create
|
:param backend: Name of backend to create
|
||||||
"""
|
"""
|
||||||
# NOTE(sirp): avoiding circular import
|
# NOTE(sirp): avoiding circular import
|
||||||
from glance.store.http import HTTPBackend
|
import glance.store.http
|
||||||
from glance.store.s3 import S3Backend
|
import glance.store.s3
|
||||||
from glance.store.swift import SwiftBackend
|
import glance.store.swift
|
||||||
from glance.store.filesystem import FilesystemBackend
|
import glance.store.filesystem
|
||||||
|
|
||||||
BACKENDS = {
|
BACKENDS = {
|
||||||
"file": FilesystemBackend,
|
"filesystem": glance.store.filesystem.FilesystemBackend,
|
||||||
"http": HTTPBackend,
|
"http": glance.store.http.HTTPBackend,
|
||||||
"https": HTTPBackend,
|
"swift": glance.store.swift.SwiftBackend,
|
||||||
"swift": SwiftBackend,
|
"s3": glance.store.s3.S3Backend}
|
||||||
"s3": S3Backend}
|
|
||||||
|
|
||||||
try:
|
try:
|
||||||
return BACKENDS[backend]
|
return BACKENDS[backend]
|
||||||
|
@ -30,7 +30,8 @@ import glance.store.location
|
|||||||
|
|
||||||
logger = logging.getLogger('glance.store.filesystem')
|
logger = logging.getLogger('glance.store.filesystem')
|
||||||
|
|
||||||
glance.store.location.add_scheme_map({'file': 'filesystem'})
|
glance.store.location.add_scheme_map({'file': 'filesystem',
|
||||||
|
'filesystem': 'filesystem'})
|
||||||
|
|
||||||
|
|
||||||
class StoreLocation(glance.store.location.StoreLocation):
|
class StoreLocation(glance.store.location.StoreLocation):
|
||||||
@ -51,7 +52,7 @@ class StoreLocation(glance.store.location.StoreLocation):
|
|||||||
versions of Python.
|
versions of Python.
|
||||||
"""
|
"""
|
||||||
pieces = urlparse.urlparse(uri)
|
pieces = urlparse.urlparse(uri)
|
||||||
assert pieces.scheme == 'file'
|
assert pieces.scheme in ('file', 'filesystem')
|
||||||
self.scheme = pieces.scheme
|
self.scheme = pieces.scheme
|
||||||
path = (pieces.netloc + pieces.path).strip()
|
path = (pieces.netloc + pieces.path).strip()
|
||||||
if path == '':
|
if path == '':
|
||||||
|
@ -84,6 +84,18 @@ class StoreLocation(glance.store.location.StoreLocation):
|
|||||||
which is entirely retarded, and breaks urlparse miserably.
|
which is entirely retarded, and breaks urlparse miserably.
|
||||||
This function works around that issue.
|
This function works around that issue.
|
||||||
"""
|
"""
|
||||||
|
# Make sure that URIs that contain multiple schemes, such as:
|
||||||
|
# swift://user:pass@http://authurl.com/v1/container/obj
|
||||||
|
# are immediately rejected.
|
||||||
|
if uri.count('://') != 1:
|
||||||
|
reason = ("URI Cannot contain more than one occurrence of a "
|
||||||
|
"scheme. If you have specified a "
|
||||||
|
"URI like s3://user:pass@https://s3.amazonaws.com/"
|
||||||
|
"bucket/key, you need to change it to use the "
|
||||||
|
"s3+https:// scheme, like so: "
|
||||||
|
"s3+https://user:pass@s3.amazonaws.com/bucket/key")
|
||||||
|
raise exception.BadStoreUri(uri, reason)
|
||||||
|
|
||||||
pieces = urlparse.urlparse(uri)
|
pieces = urlparse.urlparse(uri)
|
||||||
assert pieces.scheme in ('s3', 's3+http', 's3+https')
|
assert pieces.scheme in ('s3', 's3+http', 's3+https')
|
||||||
self.scheme = pieces.scheme
|
self.scheme = pieces.scheme
|
||||||
|
@ -44,10 +44,13 @@ class StoreLocation(glance.store.location.StoreLocation):
|
|||||||
the following:
|
the following:
|
||||||
|
|
||||||
swift://user:pass@authurl.com/container/obj-id
|
swift://user:pass@authurl.com/container/obj-id
|
||||||
|
swift://account:user:pass@authurl.com/container/obj-id
|
||||||
swift+http://user:pass@authurl.com/container/obj-id
|
swift+http://user:pass@authurl.com/container/obj-id
|
||||||
swift+https://user:pass@authurl.com/container/obj-id
|
swift+https://user:pass@authurl.com/container/obj-id
|
||||||
|
|
||||||
The swift+https:// URIs indicate there is an HTTPS authentication URL
|
The swift+http:// URIs indicate there is an HTTP authentication URL.
|
||||||
|
The default for Swift is an HTTPS authentication URL, so swift:// and
|
||||||
|
swift+https:// are the same...
|
||||||
"""
|
"""
|
||||||
|
|
||||||
def process_specs(self):
|
def process_specs(self):
|
||||||
@ -85,6 +88,18 @@ class StoreLocation(glance.store.location.StoreLocation):
|
|||||||
|
|
||||||
swift://account:user:pass@authurl.com/container/obj
|
swift://account:user:pass@authurl.com/container/obj
|
||||||
"""
|
"""
|
||||||
|
# Make sure that URIs that contain multiple schemes, such as:
|
||||||
|
# swift://user:pass@http://authurl.com/v1/container/obj
|
||||||
|
# are immediately rejected.
|
||||||
|
if uri.count('://') != 1:
|
||||||
|
reason = ("URI Cannot contain more than one occurrence of a "
|
||||||
|
"scheme. If you have specified a "
|
||||||
|
"URI like swift://user:pass@http://authurl.com/v1/"
|
||||||
|
"container/obj, you need to change it to use the "
|
||||||
|
"swift+http:// scheme, like so: "
|
||||||
|
"swift+http://user:pass@authurl.com/v1/container/obj")
|
||||||
|
raise exception.BadStoreUri(uri, reason)
|
||||||
|
|
||||||
pieces = urlparse.urlparse(uri)
|
pieces = urlparse.urlparse(uri)
|
||||||
assert pieces.scheme in ('swift', 'swift+http', 'swift+https')
|
assert pieces.scheme in ('swift', 'swift+http', 'swift+https')
|
||||||
self.scheme = pieces.scheme
|
self.scheme = pieces.scheme
|
||||||
@ -127,9 +142,10 @@ class StoreLocation(glance.store.location.StoreLocation):
|
|||||||
try:
|
try:
|
||||||
self.obj = path_parts.pop()
|
self.obj = path_parts.pop()
|
||||||
self.container = path_parts.pop()
|
self.container = path_parts.pop()
|
||||||
self.authurl = netloc
|
if not netloc.startswith('http'):
|
||||||
if len(path_parts) > 0:
|
# push hostname back into the remaining to build full authurl
|
||||||
self.authurl = netloc + '/' + '/'.join(path_parts).strip('/')
|
path_parts.insert(0, netloc)
|
||||||
|
self.authurl = '/'.join(path_parts)
|
||||||
except IndexError:
|
except IndexError:
|
||||||
reason = "Badly formed Swift URI"
|
reason = "Badly formed Swift URI"
|
||||||
raise exception.BadStoreUri(uri, reason)
|
raise exception.BadStoreUri(uri, reason)
|
||||||
@ -143,16 +159,10 @@ class StoreLocation(glance.store.location.StoreLocation):
|
|||||||
|
|
||||||
HTTPS is assumed, unless 'swift+http' is specified.
|
HTTPS is assumed, unless 'swift+http' is specified.
|
||||||
"""
|
"""
|
||||||
if self.scheme.startswith('swift+http'):
|
if self.scheme in ('swift+https', 'swift'):
|
||||||
auth_scheme = 'http://'
|
|
||||||
elif self.scheme.startswith('swift+https'):
|
|
||||||
auth_scheme = 'https://'
|
|
||||||
elif self.scheme.startswith('swift'):
|
|
||||||
auth_scheme = 'https://'
|
auth_scheme = 'https://'
|
||||||
else:
|
else:
|
||||||
logger.warn("Unrecognized scheme '%s', defaulting auth url"
|
auth_scheme = 'http://'
|
||||||
" to https", self.scheme)
|
|
||||||
auth_scheme = 'https://'
|
|
||||||
|
|
||||||
full_url = ''.join([auth_scheme, self.authurl])
|
full_url = ''.join([auth_scheme, self.authurl])
|
||||||
return full_url
|
return full_url
|
||||||
|
@ -138,6 +138,7 @@ class TestStoreLocation(unittest.TestCase):
|
|||||||
|
|
||||||
self.assertEqual("swift", loc.scheme)
|
self.assertEqual("swift", loc.scheme)
|
||||||
self.assertEqual("example.com", loc.authurl)
|
self.assertEqual("example.com", loc.authurl)
|
||||||
|
self.assertEqual("https://example.com", loc.swift_auth_url)
|
||||||
self.assertEqual("images", loc.container)
|
self.assertEqual("images", loc.container)
|
||||||
self.assertEqual("1", loc.obj)
|
self.assertEqual("1", loc.obj)
|
||||||
self.assertEqual(None, loc.user)
|
self.assertEqual(None, loc.user)
|
||||||
@ -148,6 +149,7 @@ class TestStoreLocation(unittest.TestCase):
|
|||||||
|
|
||||||
self.assertEqual("swift+https", loc.scheme)
|
self.assertEqual("swift+https", loc.scheme)
|
||||||
self.assertEqual("authurl.com", loc.authurl)
|
self.assertEqual("authurl.com", loc.authurl)
|
||||||
|
self.assertEqual("https://authurl.com", loc.swift_auth_url)
|
||||||
self.assertEqual("images", loc.container)
|
self.assertEqual("images", loc.container)
|
||||||
self.assertEqual("1", loc.obj)
|
self.assertEqual("1", loc.obj)
|
||||||
self.assertEqual("user", loc.user)
|
self.assertEqual("user", loc.user)
|
||||||
@ -159,17 +161,19 @@ class TestStoreLocation(unittest.TestCase):
|
|||||||
|
|
||||||
self.assertEqual("swift+https", loc.scheme)
|
self.assertEqual("swift+https", loc.scheme)
|
||||||
self.assertEqual("authurl.com/v1", loc.authurl)
|
self.assertEqual("authurl.com/v1", loc.authurl)
|
||||||
|
self.assertEqual("https://authurl.com/v1", loc.swift_auth_url)
|
||||||
self.assertEqual("container", loc.container)
|
self.assertEqual("container", loc.container)
|
||||||
self.assertEqual("12345", loc.obj)
|
self.assertEqual("12345", loc.obj)
|
||||||
self.assertEqual("user", loc.user)
|
self.assertEqual("user", loc.user)
|
||||||
self.assertEqual("pass", loc.key)
|
self.assertEqual("pass", loc.key)
|
||||||
self.assertEqual(uri, loc.get_uri())
|
self.assertEqual(uri, loc.get_uri())
|
||||||
|
|
||||||
uri = 'swift://account:user:pass@authurl.com/v1/container/12345'
|
uri = 'swift+http://account:user:pass@authurl.com/v1/container/12345'
|
||||||
loc.parse_uri(uri)
|
loc.parse_uri(uri)
|
||||||
|
|
||||||
self.assertEqual("swift", loc.scheme)
|
self.assertEqual("swift+http", loc.scheme)
|
||||||
self.assertEqual("authurl.com/v1", loc.authurl)
|
self.assertEqual("authurl.com/v1", loc.authurl)
|
||||||
|
self.assertEqual("http://authurl.com/v1", loc.swift_auth_url)
|
||||||
self.assertEqual("container", loc.container)
|
self.assertEqual("container", loc.container)
|
||||||
self.assertEqual("12345", loc.obj)
|
self.assertEqual("12345", loc.obj)
|
||||||
self.assertEqual("account:user", loc.user)
|
self.assertEqual("account:user", loc.user)
|
||||||
@ -185,6 +189,9 @@ class TestStoreLocation(unittest.TestCase):
|
|||||||
bad_uri = 'swift://user@example.com:8080/images/1'
|
bad_uri = 'swift://user@example.com:8080/images/1'
|
||||||
self.assertRaises(exception.BadStoreUri, loc.parse_uri, bad_uri)
|
self.assertRaises(exception.BadStoreUri, loc.parse_uri, bad_uri)
|
||||||
|
|
||||||
|
bad_uri = 'swift://user:pass@http://example.com:8080/images/1'
|
||||||
|
self.assertRaises(exception.BadStoreUri, loc.parse_uri, bad_uri)
|
||||||
|
|
||||||
def test_s3_store_location(self):
|
def test_s3_store_location(self):
|
||||||
"""
|
"""
|
||||||
Test the specific StoreLocation for the S3 store
|
Test the specific StoreLocation for the S3 store
|
||||||
@ -233,7 +240,7 @@ class TestStoreLocation(unittest.TestCase):
|
|||||||
self.assertEqual("pass/withslash", loc.secretkey)
|
self.assertEqual("pass/withslash", loc.secretkey)
|
||||||
self.assertEqual(uri, loc.get_uri())
|
self.assertEqual(uri, loc.get_uri())
|
||||||
|
|
||||||
bad_uri = 'swif://'
|
bad_uri = 's://'
|
||||||
self.assertRaises(Exception, loc.parse_uri, bad_uri)
|
self.assertRaises(Exception, loc.parse_uri, bad_uri)
|
||||||
|
|
||||||
bad_uri = 's3://'
|
bad_uri = 's3://'
|
||||||
@ -241,3 +248,33 @@ class TestStoreLocation(unittest.TestCase):
|
|||||||
|
|
||||||
bad_uri = 's3://accesskey@example.com:8080/images/1'
|
bad_uri = 's3://accesskey@example.com:8080/images/1'
|
||||||
self.assertRaises(exception.BadStoreUri, loc.parse_uri, bad_uri)
|
self.assertRaises(exception.BadStoreUri, loc.parse_uri, bad_uri)
|
||||||
|
|
||||||
|
bad_uri = 's3://user:pass@http://example.com:8080/images/1'
|
||||||
|
self.assertRaises(exception.BadStoreUri, loc.parse_uri, bad_uri)
|
||||||
|
|
||||||
|
def test_get_backend_class(self):
|
||||||
|
"""
|
||||||
|
Test that the backend returned by glance.store.get_backend_class
|
||||||
|
is correct or raises an appropriate error.
|
||||||
|
"""
|
||||||
|
good_results = {
|
||||||
|
'swift': glance.store.swift.SwiftBackend,
|
||||||
|
'swift+http': glance.store.swift.SwiftBackend,
|
||||||
|
'swift+https': glance.store.swift.SwiftBackend,
|
||||||
|
's3': glance.store.s3.S3Backend,
|
||||||
|
's3+http': glance.store.s3.S3Backend,
|
||||||
|
's3+https': glance.store.s3.S3Backend,
|
||||||
|
'file': glance.store.filesystem.FilesystemBackend,
|
||||||
|
'filesystem': glance.store.filesystem.FilesystemBackend,
|
||||||
|
'http': glance.store.http.HTTPBackend,
|
||||||
|
'https': glance.store.http.HTTPBackend}
|
||||||
|
|
||||||
|
for scheme, store in good_results.items():
|
||||||
|
self.assertEqual(glance.store.get_backend_class(scheme), store)
|
||||||
|
|
||||||
|
bad_results = ['fil', 'swift+h', 'unknown']
|
||||||
|
|
||||||
|
for store in bad_results:
|
||||||
|
self.assertRaises(glance.store.UnsupportedBackend,
|
||||||
|
glance.store.get_backend_class,
|
||||||
|
store)
|
||||||
|
@ -124,7 +124,9 @@ def stub_out_swift_common_client(stubs):
|
|||||||
def fake_http_connection(*args, **kwargs):
|
def fake_http_connection(*args, **kwargs):
|
||||||
return None
|
return None
|
||||||
|
|
||||||
def fake_get_auth(*args, **kwargs):
|
def fake_get_auth(url, *args, **kwargs):
|
||||||
|
if 'http' in url and '://' not in url:
|
||||||
|
raise ValueError('Invalid url %s' % url)
|
||||||
return None, None
|
return None, None
|
||||||
|
|
||||||
stubs.Set(swift.common.client,
|
stubs.Set(swift.common.client,
|
||||||
|
Loading…
Reference in New Issue
Block a user