mirror of
https://github.com/ceph/ceph
synced 2025-01-23 19:46:56 +00:00
b65046e37f
Adds a test case to reproduce a scenario where concurrent versioned object deletes can cause leftover OLH entries to be left behind. Signed-off-by: Cory Snyder <csnyder@1111systems.com>
138 lines
5.3 KiB
Python
Executable File
138 lines
5.3 KiB
Python
Executable File
#!/usr/bin/env python3
|
|
|
|
import logging as log
|
|
import json
|
|
import uuid
|
|
import botocore
|
|
import time
|
|
import threading
|
|
from common import exec_cmd, create_user, boto_connect
|
|
from botocore.config import Config
|
|
|
|
"""
|
|
Tests behavior of bucket versioning.
|
|
"""
|
|
# The test cases in this file have been annotated for inventory.
|
|
# To extract the inventory (in csv format) use the command:
|
|
#
|
|
# grep '^ *# TESTCASE' | sed 's/^ *# TESTCASE //'
|
|
#
|
|
#
|
|
|
|
""" Constants """
|
|
USER = 'versioning-tester'
|
|
DISPLAY_NAME = 'Versioning Testing'
|
|
ACCESS_KEY = 'LTA662PVVDTDWX6M2AB0'
|
|
SECRET_KEY = 'pvtchqajgzqx5581t6qbddbkj0bgf3a69qdkjcea'
|
|
BUCKET_NAME = 'versioning-bucket'
|
|
DATA_POOL = 'default.rgw.buckets.data'
|
|
|
|
def main():
|
|
"""
|
|
execute versioning tests
|
|
"""
|
|
create_user(USER, DISPLAY_NAME, ACCESS_KEY, SECRET_KEY)
|
|
|
|
connection = boto_connect(ACCESS_KEY, SECRET_KEY, Config(retries = {
|
|
'total_max_attempts': 1,
|
|
}))
|
|
|
|
# pre-test cleanup
|
|
try:
|
|
bucket = connection.Bucket(BUCKET_NAME)
|
|
bucket.objects.all().delete()
|
|
bucket.object_versions.all().delete()
|
|
bucket.delete()
|
|
except botocore.exceptions.ClientError as e:
|
|
if not e.response['Error']['Code'] == 'NoSuchBucket':
|
|
raise
|
|
|
|
bucket = connection.create_bucket(Bucket=BUCKET_NAME)
|
|
connection.BucketVersioning(BUCKET_NAME).enable()
|
|
|
|
# reproducer for bug from https://tracker.ceph.com/issues/59663
|
|
# TESTCASE 'verify that index entries and OLH objects are cleaned up after redundant deletes'
|
|
log.debug('TEST: verify that index entries and OLH objects are cleaned up after redundant deletes\n')
|
|
key = str(uuid.uuid4())
|
|
resp = bucket.Object(key).delete()
|
|
assert 'DeleteMarker' in resp, 'DeleteMarker key not present in response'
|
|
assert resp['DeleteMarker'], 'DeleteMarker value not True in response'
|
|
assert 'VersionId' in resp, 'VersionId key not present in response'
|
|
version_id = resp['VersionId']
|
|
bucket.Object(key).delete()
|
|
connection.ObjectVersion(bucket.name, key, version_id).delete()
|
|
# bucket index should now be empty
|
|
out = exec_cmd(f'radosgw-admin bi list --bucket {BUCKET_NAME}')
|
|
json_out = json.loads(out.replace(b'\x80', b'0x80'))
|
|
assert len(json_out) == 0, 'bucket index was not empty after all objects were deleted'
|
|
|
|
(_out, ret) = exec_cmd(f'rados -p {DATA_POOL} ls | grep {key}', check_retcode=False)
|
|
assert ret != 0, 'olh object was not cleaned up'
|
|
|
|
# TESTCASE 'verify that index entries and OLH objects are cleaned up after index linking error'
|
|
log.debug('TEST: verify that index entries and OLH objects are cleaned up after index linking error\n')
|
|
key = str(uuid.uuid4())
|
|
try:
|
|
exec_cmd('ceph config set client rgw_debug_inject_set_olh_err 2')
|
|
time.sleep(1)
|
|
bucket.Object(key).delete()
|
|
finally:
|
|
exec_cmd('ceph config rm client rgw_debug_inject_set_olh_err')
|
|
out = exec_cmd(f'radosgw-admin bi list --bucket {BUCKET_NAME}')
|
|
json_out = json.loads(out.replace(b'\x80', b'0x80'))
|
|
assert len(json_out) == 0, 'bucket index was not empty after op failed'
|
|
(_out, ret) = exec_cmd(f'rados -p {DATA_POOL} ls | grep {key}', check_retcode=False)
|
|
assert ret != 0, 'olh object was not cleaned up'
|
|
|
|
# TESTCASE 'verify that original null object version is intact after failed olh upgrade'
|
|
log.debug('TEST: verify that original null object version is intact after failed olh upgrade\n')
|
|
connection.BucketVersioning(BUCKET_NAME).suspend()
|
|
key = str(uuid.uuid4())
|
|
put_resp = bucket.put_object(Key=key, Body=b"data")
|
|
connection.BucketVersioning(BUCKET_NAME).enable()
|
|
try:
|
|
exec_cmd('ceph config set client rgw_debug_inject_set_olh_err 2')
|
|
time.sleep(1)
|
|
# expected to fail due to the above error injection
|
|
bucket.put_object(Key=key, Body=b"new data")
|
|
except Exception as e:
|
|
log.debug(e)
|
|
finally:
|
|
exec_cmd('ceph config rm client rgw_debug_inject_set_olh_err')
|
|
get_resp = bucket.Object(key).get()
|
|
assert put_resp.e_tag == get_resp['ETag'], 'get did not return null version with correct etag'
|
|
|
|
# TESTCASE 'verify that concurrent delete requests do not leave behind olh entries'
|
|
log.debug('TEST: verify that concurrent delete requests do not leave behind olh entries\n')
|
|
bucket.object_versions.all().delete()
|
|
|
|
key = 'concurrent-delete'
|
|
# create a delete marker
|
|
resp = bucket.Object(key).delete()
|
|
version_id = resp['ResponseMetadata']['HTTPHeaders']['x-amz-version-id']
|
|
try:
|
|
exec_cmd('ceph config set client rgw_debug_inject_latency_bi_unlink 2')
|
|
time.sleep(1)
|
|
|
|
def do_delete():
|
|
connection.ObjectVersion(bucket.name, key, version_id).delete()
|
|
|
|
t2 = threading.Thread(target=do_delete)
|
|
t2.start()
|
|
do_delete()
|
|
t2.join()
|
|
finally:
|
|
exec_cmd('ceph config rm client rgw_debug_inject_latency_bi_unlink')
|
|
out = exec_cmd(f'radosgw-admin bucket check olh --bucket {bucket.name} --dump-keys')
|
|
num_leftover_olh_entries = len(json.loads(out))
|
|
assert num_leftover_olh_entries == 0, \
|
|
'Found leftover olh entries after concurrent deletes'
|
|
|
|
# Clean up
|
|
log.debug("Deleting bucket {}".format(BUCKET_NAME))
|
|
bucket.object_versions.all().delete()
|
|
bucket.delete()
|
|
|
|
main()
|
|
log.info("Completed bucket versioning tests")
|