Merge pull request #1638 from coreos-inc/swift-retry-seek

Add retry support to Swift
This commit is contained in:
josephschorr 2016-08-01 14:04:54 -04:00 committed by GitHub
commit b0bffe56ca
3 changed files with 18 additions and 5 deletions

View file

@ -7,6 +7,7 @@ import copy
import hmac import hmac
import string import string
import logging import logging
from _pyio import BufferedReader
from uuid import uuid4 from uuid import uuid4
from swiftclient.client import Connection, ClientException from swiftclient.client import Connection, ClientException
@ -26,7 +27,7 @@ logger = logging.getLogger(__name__)
_PartUploadMetadata = namedtuple('_PartUploadMetadata', ['path', 'offset', 'length']) _PartUploadMetadata = namedtuple('_PartUploadMetadata', ['path', 'offset', 'length'])
_SEGMENTS_KEY = 'segments' _SEGMENTS_KEY = 'segments'
_SEGMENT_DIRECTORY = 'segments' _SEGMENT_DIRECTORY = 'segments'
_MAXIMUM_SEGMENT_SIZE = 5000000000 # 5 GB _MAXIMUM_SEGMENT_SIZE = 200000000 # ~200 MB
_DEFAULT_SWIFT_CONNECT_TIMEOUT = 5 # seconds _DEFAULT_SWIFT_CONNECT_TIMEOUT = 5 # seconds
class SwiftStorage(BaseStorage): class SwiftStorage(BaseStorage):
@ -240,6 +241,7 @@ class SwiftStorage(BaseStorage):
# are finished hitting the data limit. # are finished hitting the data limit.
total_bytes_written = 0 total_bytes_written = 0
upload_error = None upload_error = None
read_until_end = length == filelike.READ_UNTIL_END
while True: while True:
try: try:
@ -253,12 +255,12 @@ class SwiftStorage(BaseStorage):
upload_error = ex upload_error = ex
break break
if length != filelike.READ_UNTIL_END: if not read_until_end:
length = length - bytes_written length = length - bytes_written
offset = offset + bytes_written offset = offset + bytes_written
total_bytes_written = total_bytes_written + bytes_written total_bytes_written = total_bytes_written + bytes_written
if bytes_written == 0 or length <= 0: if bytes_written == 0 or (not read_until_end and length <= 0):
return total_bytes_written, storage_metadata, upload_error return total_bytes_written, storage_metadata, upload_error
return total_bytes_written, storage_metadata, upload_error return total_bytes_written, storage_metadata, upload_error
@ -277,6 +279,11 @@ class SwiftStorage(BaseStorage):
limiting_fp = filelike.LimitingStream(in_fp, length) limiting_fp = filelike.LimitingStream(in_fp, length)
# If retries are requested, then we need to use a buffered reader to allow for calls to
# seek() on retries from within the Swift client.
if self._retry_count > 0:
limiting_fp = BufferedReader(limiting_fp, buffer_size=length)
# Write the segment to Swift. # Write the segment to Swift.
self.stream_write(segment_path, limiting_fp, content_type) self.stream_write(segment_path, limiting_fp, content_type)

View file

@ -88,6 +88,14 @@ class TestLimitingStream(unittest.TestCase):
self.assertEquals('', stream.read(1)) self.assertEquals('', stream.read(1))
self.assertEquals(stream.tell(), 3) self.assertEquals(stream.tell(), 3)
def test_seek_to_tell(self):
fileobj = StringIO('this is a cool test')
stream = LimitingStream(fileobj, 3)
stream.seek(stream.tell())
self.assertEquals('thi', stream.read(4))
self.assertEquals(stream.tell(), 3)
class TestStreamSlice(unittest.TestCase): class TestStreamSlice(unittest.TestCase):
def test_none_read(self): def test_none_read(self):

View file

@ -1,8 +1,6 @@
import unittest import unittest
import os import os
from StringIO import StringIO
from util.registry.queueprocess import QueueResult from util.registry.queueprocess import QueueResult
from util.registry.queuefile import QueueFile from util.registry.queuefile import QueueFile