
The SsyncSender encodes object file timestamps in a compact form and the SsyncReceiver decodes the timestamps and compares them to its object file set. The encoding represents the meta file timestamp as a delta from the data file timestamp, NOT INCLUDING the data file timestamp offset. Previously, the decoding was erroneously calculating the meta file timestamp as the sum of the delta plus the data file timestamp INCLUDING the offset. For example, if the SssyncSender has object file timestamps: ts_data = t0_1.data ts_meta = t1.data then the receiver would erroneously perceive that the sender has: ts_data = t0_1.data ts_meta = t1_1.data As described in the referenced bug report, this erroneous decoding could cause the SsyncReceiver to request that the SsyncSender sync an object that is already in sync, which results in a 409 Conflict at the receiver. The 409 causes the ssync session to terminate, and the same process repeats on the next attempt. Closes-Bug: #2007643 Co-Authored-By: Clay Gerrard <clay.gerrard@gmail.com> Change-Id: I74a0aac0ac29577026743f87f4b654d85e8fcc80
331 lines
14 KiB
Python
331 lines
14 KiB
Python
#!/usr/bin/python -u
|
|
# Copyright (c) 2010-2012 OpenStack Foundation
|
|
#
|
|
# Licensed under the Apache License, Version 2.0 (the "License");
|
|
# you may not use this file except in compliance with the License.
|
|
# You may obtain a copy of the License at
|
|
#
|
|
# http://www.apache.org/licenses/LICENSE-2.0
|
|
#
|
|
# Unless required by applicable law or agreed to in writing, software
|
|
# distributed under the License is distributed on an "AS IS" BASIS,
|
|
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or
|
|
# implied.
|
|
# See the License for the specific language governing permissions and
|
|
# limitations under the License.
|
|
|
|
from unittest import main
|
|
import random
|
|
|
|
from swiftclient import client
|
|
|
|
from swift.common import direct_client
|
|
from swift.common.request_helpers import get_reserved_name
|
|
from swift.obj import reconstructor
|
|
|
|
from test.probe.common import ReplProbeTest, ECProbeTest
|
|
|
|
|
|
class TestObjectVersioning(ReplProbeTest):
|
|
|
|
def _assert_account_level(self, container_name, hdr_cont_count,
|
|
hdr_obj_count, hdr_bytes, cont_count,
|
|
cont_bytes):
|
|
|
|
headers, containers = client.get_account(self.url, self.token)
|
|
self.assertEqual(hdr_cont_count, headers['x-account-container-count'])
|
|
self.assertEqual(hdr_obj_count, headers['x-account-object-count'])
|
|
self.assertEqual(hdr_bytes, headers['x-account-bytes-used'])
|
|
self.assertEqual(len(containers), 1)
|
|
container = containers[0]
|
|
self.assertEqual(container_name, container['name'])
|
|
self.assertEqual(cont_count, container['count'])
|
|
self.assertEqual(cont_bytes, container['bytes'])
|
|
|
|
def test_account_listing(self):
|
|
versions_header_key = 'X-Versions-Enabled'
|
|
|
|
# Create container1
|
|
container_name = 'container1'
|
|
obj_name = 'object1'
|
|
client.put_container(self.url, self.token, container_name)
|
|
|
|
# Assert account level sees it
|
|
self._assert_account_level(
|
|
container_name,
|
|
hdr_cont_count='1',
|
|
hdr_obj_count='0',
|
|
hdr_bytes='0',
|
|
cont_count=0,
|
|
cont_bytes=0)
|
|
|
|
# Enable versioning
|
|
hdrs = {versions_header_key: 'True'}
|
|
client.post_container(self.url, self.token, container_name, hdrs)
|
|
|
|
# write multiple versions of same obj
|
|
client.put_object(self.url, self.token, container_name, obj_name,
|
|
'version1')
|
|
client.put_object(self.url, self.token, container_name, obj_name,
|
|
'version2')
|
|
|
|
# Assert account level doesn't see object data yet, but it
|
|
# does see the update for the hidden container
|
|
self._assert_account_level(
|
|
container_name,
|
|
hdr_cont_count='2',
|
|
hdr_obj_count='0',
|
|
hdr_bytes='0',
|
|
cont_count=0,
|
|
cont_bytes=0)
|
|
|
|
# Get to final state
|
|
self.get_to_final_state()
|
|
|
|
# Assert account level now sees updated values
|
|
# N.B: Note difference in values between header and container listing
|
|
# header object count is counting both symlink + object versions
|
|
# listing count is counting only symlink (in primary container)
|
|
self._assert_account_level(
|
|
container_name,
|
|
hdr_cont_count='2',
|
|
hdr_obj_count='3',
|
|
hdr_bytes='16',
|
|
cont_count=1,
|
|
cont_bytes=16)
|
|
|
|
client.delete_object(self.url, self.token, container_name, obj_name)
|
|
_headers, current_versions = client.get_container(
|
|
self.url, self.token, container_name)
|
|
self.assertEqual(len(current_versions), 0)
|
|
_headers, all_versions = client.get_container(
|
|
self.url, self.token, container_name, query_string='versions')
|
|
self.assertEqual(len(all_versions), 3)
|
|
|
|
# directly delete primary container to leave an orphan hidden
|
|
# container
|
|
self.direct_delete_container(container=container_name)
|
|
|
|
# Get to final state
|
|
self.get_to_final_state()
|
|
|
|
# The container count decreases, as well as object count. But bytes
|
|
# do not. The discrepancy between header object count, container
|
|
# object count and bytes should indicate orphan hidden container is
|
|
# still around consuming storage
|
|
self._assert_account_level(
|
|
container_name,
|
|
hdr_cont_count='1',
|
|
hdr_obj_count='3',
|
|
hdr_bytes='16',
|
|
cont_count=0,
|
|
cont_bytes=16)
|
|
|
|
# Can't HEAD or list anything, though
|
|
with self.assertRaises(client.ClientException) as caught:
|
|
client.head_container(self.url, self.token, container_name)
|
|
self.assertEqual(caught.exception.http_status, 404)
|
|
with self.assertRaises(client.ClientException) as caught:
|
|
client.get_container(self.url, self.token, container_name)
|
|
self.assertEqual(caught.exception.http_status, 404)
|
|
with self.assertRaises(client.ClientException) as caught:
|
|
client.get_container(self.url, self.token, container_name,
|
|
query_string='versions')
|
|
self.assertEqual(caught.exception.http_status, 404)
|
|
with self.assertRaises(client.ClientException) as caught:
|
|
client.get_object(
|
|
self.url, self.token, container_name, all_versions[1]['name'],
|
|
query_string='version-id=%s' % all_versions[1]['version_id'])
|
|
# A little funny -- maybe this should 404 instead?
|
|
self.assertEqual(caught.exception.http_status, 400)
|
|
|
|
# Fix isn't too bad -- just make the container again!
|
|
client.put_container(self.url, self.token, container_name)
|
|
_headers, current_versions = client.get_container(
|
|
self.url, self.token, container_name)
|
|
self.assertEqual(len(current_versions), 0)
|
|
_headers, all_versions = client.get_container(
|
|
self.url, self.token, container_name, query_string='versions')
|
|
self.assertEqual(len(all_versions), 3)
|
|
|
|
# ... but to actually *access* the versions, you have to enable
|
|
# versioning again
|
|
with self.assertRaises(client.ClientException) as caught:
|
|
client.get_object(
|
|
self.url, self.token, container_name, all_versions[1]['name'],
|
|
query_string='version-id=%s' % all_versions[1]['version_id'])
|
|
self.assertEqual(caught.exception.http_status, 400)
|
|
self.assertIn(b'version-aware operations require',
|
|
caught.exception.http_response_content)
|
|
client.post_container(self.url, self.token, container_name,
|
|
headers={'X-Versions-Enabled': 'true'})
|
|
client.get_object(
|
|
self.url, self.token, container_name, all_versions[1]['name'],
|
|
query_string='version-id=%s' % all_versions[1]['version_id'])
|
|
|
|
def test_missing_versions_container(self):
|
|
versions_header_key = 'X-Versions-Enabled'
|
|
|
|
# Create container1
|
|
container_name = 'container1'
|
|
obj_name = 'object1'
|
|
client.put_container(self.url, self.token, container_name)
|
|
|
|
# Write some data
|
|
client.put_object(self.url, self.token, container_name, obj_name,
|
|
b'null version')
|
|
|
|
# Enable versioning
|
|
hdrs = {versions_header_key: 'True'}
|
|
client.post_container(self.url, self.token, container_name, hdrs)
|
|
|
|
# But directly delete hidden container to leave an orphan primary
|
|
# container
|
|
self.direct_delete_container(container=get_reserved_name(
|
|
'versions', container_name))
|
|
|
|
# Could be worse; we can still list versions and GET data
|
|
_headers, all_versions = client.get_container(
|
|
self.url, self.token, container_name, query_string='versions')
|
|
self.assertEqual(len(all_versions), 1)
|
|
self.assertEqual(all_versions[0]['name'], obj_name)
|
|
self.assertEqual(all_versions[0]['version_id'], 'null')
|
|
|
|
_headers, data = client.get_object(
|
|
self.url, self.token, container_name, obj_name)
|
|
self.assertEqual(data, b'null version')
|
|
|
|
_headers, data = client.get_object(
|
|
self.url, self.token, container_name, obj_name,
|
|
query_string='version-id=null')
|
|
self.assertEqual(data, b'null version')
|
|
|
|
# But most any write is going to fail
|
|
with self.assertRaises(client.ClientException) as caught:
|
|
client.put_object(self.url, self.token, container_name, obj_name,
|
|
b'new version')
|
|
self.assertEqual(caught.exception.http_status, 500)
|
|
with self.assertRaises(client.ClientException) as caught:
|
|
client.delete_object(self.url, self.token, container_name,
|
|
obj_name)
|
|
self.assertEqual(caught.exception.http_status, 500)
|
|
|
|
# Version-aware delete can work, though!
|
|
client.delete_object(self.url, self.token, container_name, obj_name,
|
|
query_string='version-id=null')
|
|
|
|
# Re-enabling versioning should square us
|
|
hdrs = {versions_header_key: 'True'}
|
|
client.post_container(self.url, self.token, container_name, hdrs)
|
|
|
|
client.put_object(self.url, self.token, container_name, obj_name,
|
|
b'new version')
|
|
|
|
_headers, all_versions = client.get_container(
|
|
self.url, self.token, container_name, query_string='versions')
|
|
self.assertEqual(len(all_versions), 1)
|
|
self.assertEqual(all_versions[0]['name'], obj_name)
|
|
self.assertNotEqual(all_versions[0]['version_id'], 'null')
|
|
|
|
_headers, data = client.get_object(
|
|
self.url, self.token, container_name, obj_name)
|
|
self.assertEqual(data, b'new version')
|
|
|
|
|
|
class TestECObjectVersioning(ECProbeTest):
|
|
|
|
def setUp(self):
|
|
super(TestECObjectVersioning, self).setUp()
|
|
self.part, self.nodes = self.object_ring.get_nodes(
|
|
self.account, self.container_name, self.object_name)
|
|
|
|
def test_versioning_with_metadata_replication(self):
|
|
# Enable versioning
|
|
client.put_container(self.url, self.token, self.container_name,
|
|
headers={
|
|
'X-Storage-Policy': self.policy.name,
|
|
'X-Versions-Enabled': 'True',
|
|
})
|
|
# create version with metadata in a handoff location
|
|
failed_primary = random.choice(self.nodes)
|
|
failed_primary_device_path = self.device_dir(failed_primary)
|
|
self.kill_drive(failed_primary_device_path)
|
|
headers = {'x-object-meta-foo': 'meta-foo'}
|
|
client.put_object(self.url, self.token, self.container_name,
|
|
self.object_name, contents='some data',
|
|
headers=headers)
|
|
headers_post = {'x-object-meta-bar': 'meta-bar'}
|
|
client.post_object(self.url, self.token, self.container_name,
|
|
self.object_name, headers=headers_post)
|
|
# find the handoff
|
|
primary_ids = [n['id'] for n in self.nodes]
|
|
for handoff in self.object_ring.devs:
|
|
if handoff['id'] in primary_ids:
|
|
continue
|
|
try:
|
|
headers, etag = self.direct_get(handoff, self.part)
|
|
except direct_client.DirectClientException as err:
|
|
if err.http_status != 404:
|
|
raise
|
|
else:
|
|
break
|
|
else:
|
|
self.fail('unable to find object on handoffs')
|
|
# we want to repair the fault, but avoid doing the handoff revert
|
|
self.revive_drive(failed_primary_device_path)
|
|
handoff_config = (handoff['id'] + 1) % 4
|
|
failed_config = (failed_primary['id'] + 1) % 4
|
|
partner_nodes = reconstructor._get_partners(
|
|
failed_primary['index'], self.nodes)
|
|
random.shuffle(partner_nodes)
|
|
for partner in partner_nodes:
|
|
fix_config = (partner['id'] + 1) % 4
|
|
if fix_config not in (handoff_config, failed_config):
|
|
break
|
|
else:
|
|
self.fail('unable to find fix_config in %r excluding %r & %r' % (
|
|
[(d['device'], (d['id'] + 1) % 4) for d in partner_nodes],
|
|
handoff_config, failed_config))
|
|
|
|
self.reconstructor.once(number=fix_config)
|
|
# validate object in all locations
|
|
missing = []
|
|
etags = set()
|
|
metadata = []
|
|
for node in self.nodes:
|
|
try:
|
|
headers, etag = self.direct_get(node, self.part)
|
|
except direct_client.DirectClientException as err:
|
|
if err.http_status != 404:
|
|
raise
|
|
missing.append(node)
|
|
continue
|
|
etags.add(headers['X-Object-Sysmeta-Ec-Etag'])
|
|
metadata.append(headers['X-Object-Meta-Bar'])
|
|
if missing:
|
|
self.fail('Ran reconstructor config #%s to repair %r but '
|
|
'found 404 on primary: %r' % (
|
|
fix_config, failed_primary['device'],
|
|
[d['device'] for d in missing]))
|
|
self.assertEqual(1, len(etags))
|
|
self.assertEqual(['meta-bar'] * len(self.nodes), metadata)
|
|
# process revert
|
|
self.reconstructor.once(number=handoff_config)
|
|
# validate object (still?) in primary locations
|
|
etags = set()
|
|
metadata = []
|
|
for node in self.nodes:
|
|
headers, etag = self.direct_get(node, self.part)
|
|
etags.add(headers['X-Object-Sysmeta-Ec-Etag'])
|
|
metadata.append(headers['X-Object-Meta-Bar'])
|
|
self.assertEqual(1, len(etags))
|
|
self.assertEqual(['meta-bar'] * len(self.nodes), metadata)
|
|
# and removed form handoff
|
|
with self.assertRaises(direct_client.DirectClientException) as ctx:
|
|
headers, etag = self.direct_get(handoff, self.part)
|
|
self.assertEqual(ctx.exception.http_status, 404)
|
|
|
|
|
|
if __name__ == '__main__':
|
|
main()
|