object-server can 409 in response to x-if-delete-at

Previously, if the expirer had a stale work item (because the object
was overwritten or deleted, or some other process handled the delete),
then it would keep retrying for reclaim_age, but every time it'd get
back a 412.

Now, have the object-server be smart enough to say, "I have more recent
information than you" and let the expirer accept that as success.

Change-Id: I0a94482ed16cb30ce79074e053e6177fe97bcaa9
This commit is contained in:
Tim Burke 2018-01-17 22:07:26 +00:00
parent cd2c73fd95
commit f4cff9fbba
4 changed files with 37 additions and 10 deletions

View File

@ -306,10 +306,13 @@ class ObjectExpirer(Daemon):
'<account>/<container>/<object>' '<account>/<container>/<object>'
:param timestamp: The timestamp the X-Delete-At value must match to :param timestamp: The timestamp the X-Delete-At value must match to
perform the actual delete. perform the actual delete.
:raises UnexpectedResponse: if the delete was unsuccessful and
should be retried later
""" """
path = '/v1/' + urllib.parse.quote(actual_obj.lstrip('/')) path = '/v1/' + urllib.parse.quote(actual_obj.lstrip('/'))
self.swift.make_request( self.swift.make_request(
'DELETE', path, 'DELETE', path,
{'X-If-Delete-At': str(timestamp), 'X-Timestamp': str(timestamp), {'X-If-Delete-At': str(timestamp),
'X-Timestamp': str(timestamp),
'X-Backend-Clean-Expiring-Object-Queue': 'no'}, 'X-Backend-Clean-Expiring-Object-Queue': 'no'},
(2,)) (2, HTTP_CONFLICT))

View File

@ -1076,6 +1076,9 @@ class ObjectController(BaseStorageServer):
if not orig_timestamp: if not orig_timestamp:
# no object found at all # no object found at all
return HTTPNotFound() return HTTPNotFound()
if orig_timestamp >= req_timestamp:
# Found a newer object -- return 409 as work item is stale
return HTTPConflict()
if orig_delete_at != req_if_delete_at: if orig_delete_at != req_if_delete_at:
return HTTPPreconditionFailed( return HTTPPreconditionFailed(
request=request, request=request,

View File

@ -742,7 +742,7 @@ class TestObjectExpirer(TestCase):
self.assertEqual(got_env[0]['PATH_INFO'], '/v1/path/to/object name') self.assertEqual(got_env[0]['PATH_INFO'], '/v1/path/to/object name')
def test_delete_actual_object_returns_expected_error(self): def test_delete_actual_object_returns_expected_error(self):
def do_test(test_status): def do_test(test_status, should_raise):
calls = [0] calls = [0]
def fake_app(env, start_response): def fake_app(env, start_response):
@ -753,18 +753,21 @@ class TestObjectExpirer(TestCase):
internal_client.loadapp = lambda *a, **kw: fake_app internal_client.loadapp = lambda *a, **kw: fake_app
x = expirer.ObjectExpirer({}) x = expirer.ObjectExpirer({})
self.assertRaises(internal_client.UnexpectedResponse, if should_raise:
x.delete_actual_object, '/path/to/object', with self.assertRaises(internal_client.UnexpectedResponse):
'1234') x.delete_actual_object('/path/to/object', '1234')
else:
x.delete_actual_object('/path/to/object', '1234')
self.assertEqual(calls[0], 1) self.assertEqual(calls[0], 1)
# object was deleted and tombstone reaped # object was deleted and tombstone reaped
do_test('404 Not Found') do_test('404 Not Found', True)
# object was overwritten *after* the original expiration, or # object was overwritten *after* the original expiration, or
do_test('409 Conflict', False)
# object was deleted but tombstone still exists, or # object was deleted but tombstone still exists, or
# object was overwritten ahead of the original expiration, or # object was overwritten ahead of the original expiration, or
# object was POSTed to with a new (or no) expiration, or ... # object was POSTed to with a new (or no) expiration, or ...
do_test('412 Precondition Failed') do_test('412 Precondition Failed', True)
def test_delete_actual_object_does_not_handle_odd_stuff(self): def test_delete_actual_object_does_not_handle_odd_stuff(self):

View File

@ -6146,14 +6146,32 @@ class TestObjectController(unittest.TestCase):
self.assertFalse(os.path.isfile(objfile)) self.assertFalse(os.path.isfile(objfile))
# make the x-if-delete-at with all the right bits (again) # make the x-if-delete-at with all the right bits (again)
req = Request.blank(
'/sda1/p/a/c/o', environ={'REQUEST_METHOD': 'DELETE'},
headers={'X-Timestamp': delete_at_timestamp,
'X-If-Delete-At': delete_at_timestamp})
resp = req.get_response(self.object_controller)
self.assertEqual(resp.status_int, 409)
self.assertFalse(os.path.isfile(objfile))
# overwrite with new content
req = Request.blank(
'/sda1/p/a/c/o', environ={'REQUEST_METHOD': 'PUT'},
headers={
'X-Timestamp': str(test_time + 100),
'Content-Length': '0',
'Content-Type': 'application/octet-stream'})
resp = req.get_response(self.object_controller)
self.assertEqual(resp.status_int, 201, resp.body)
# simulate processing a stale expirer queue entry
req = Request.blank( req = Request.blank(
'/sda1/p/a/c/o', '/sda1/p/a/c/o',
environ={'REQUEST_METHOD': 'DELETE'}, environ={'REQUEST_METHOD': 'DELETE'},
headers={'X-Timestamp': delete_at_timestamp, headers={'X-Timestamp': delete_at_timestamp,
'X-If-Delete-At': delete_at_timestamp}) 'X-If-Delete-At': delete_at_timestamp})
resp = req.get_response(self.object_controller) resp = req.get_response(self.object_controller)
self.assertEqual(resp.status_int, 412) self.assertEqual(resp.status_int, 409)
self.assertFalse(os.path.isfile(objfile))
# make the x-if-delete-at for some not found # make the x-if-delete-at for some not found
req = Request.blank( req = Request.blank(