simple tool to display shard relationships

This could be made more general purpose, but for now
`python swift/cli/shard-info.py` will summarise all info
about root and shard containers at end of a probe test.

Change-Id: I876fcb234771d04b1da844ca06cf0077a8bfc049
This commit is contained in:
Alistair Coles 2017-10-23 15:01:44 +01:00
parent 0405471166
commit 8eda9910d5
2 changed files with 154 additions and 1 deletions

146
swift/cli/shard-info.py Normal file
View File

@ -0,0 +1,146 @@
# Copyright (c) 2017 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.
import os
from collections import defaultdict
from swift.common import utils
from swift.common.db_replicator import roundrobin_datadirs
from swift.common.ring import ring
from swift.common.utils import Timestamp
from swift.container.backend import ContainerBroker, DATADIR, DB_STATE
from swift.container.sharder import get_sharding_info
TAB = ' '
def broker_key(broker):
broker.get_info()
return '%s/%s' % (broker.account, broker.container)
def container_type(broker):
return 'ROOT' if broker.is_root_container() else 'SHARD'
def collect_brokers(conf_file, names2nodes):
conf = utils.readconf(conf_file, 'container-replicator')
root = conf.get('devices', '/srv/node')
swift_dir = conf.get('swift_dir', '/etc/swift')
c_ring = ring.Ring(swift_dir, ring_name='container')
dirs = []
brokers = defaultdict(dict)
for node in c_ring.devs:
datadir = os.path.join(root, node['device'], DATADIR)
if os.path.isdir(datadir):
dirs.append((datadir, node['id']))
for part, object_file, node_id in roundrobin_datadirs(dirs):
broker = ContainerBroker(object_file)
names2nodes[broker_key(broker)][node_id] = broker
return brokers
def print_broker_info(node, broker, indent_level=0):
indent = indent_level * TAB
info = broker.get_info()
deleted_at = float(info['delete_timestamp'])
if deleted_at:
deleted_at = Timestamp(info['delete_timestamp']).isoformat
else:
deleted_at = ' - '
print('%s%s, objs: %s, bytes: %s, put: %s, deleted: %s (%s)' %
(indent, DB_STATE[broker.get_db_state()],
info['object_count'], info['bytes_used'],
Timestamp(info['put_timestamp']).isoformat,
deleted_at, node))
def print_db(node, broker, expect_type='ROOT', indent_level=0):
indent = indent_level * TAB
print('%s%s (%s)' % (indent, broker.db_file, node))
actual_type = container_type(broker)
if actual_type != expect_type:
print('%s ERROR expected %s but found %s' %
(indent, expect_type, actual_type))
def print_shard_range(node, sr, indent_level):
indent = indent_level * TAB
print('%s%r-%r, objs: %s, bytes: %s, created: %s, modified: %s, '
'deleted: %s (%s)' %
(indent, sr.lower, sr.upper, sr.object_count, sr.bytes_used,
Timestamp(sr.timestamp).isoformat,
Timestamp(sr.meta_timestamp).isoformat, sr.deleted, node))
def print_shard_range_info(node, shard_ranges, indent_level=0):
for sr in shard_ranges:
print_shard_range(node, sr, indent_level)
def print_sharding_info(node, broker, indent_level=0):
indent = indent_level * TAB
print('%s%s (%s)' % (indent, get_sharding_info(broker), node))
def print_container(name, name2nodes2brokers, expect_type='ROOT',
indent_level=0):
indent = indent_level * TAB
node2broker = name2nodes2brokers[name]
print('%sName: %s' % (indent, name))
print(indent + 'DB files:')
for node, broker in node2broker.items():
print_db(node, broker, expect_type, indent_level=indent_level + 1)
print(indent + 'Info:')
for node, broker in node2broker.items():
print_broker_info(node, broker, indent_level=indent_level + 1)
print(indent + 'Sharding info:')
for node, broker in node2broker.items():
print_sharding_info(node, broker, indent_level=indent_level + 1)
print(indent + 'Shard range info:')
shard_names = set()
for node, broker in node2broker.items():
shard_ranges = broker.get_shard_ranges()
for sr_name in shard_ranges:
shard_names.add(sr_name.name)
print_shard_range_info(node, shard_ranges,
indent_level=indent_level + 1)
print(indent + 'Shards:')
for sr_name in shard_names:
print_container(sr_name, name2nodes2brokers, expect_type='SHARD',
indent_level=indent_level + 1)
print('\n')
def run(conf_files):
name2nodes2brokers = defaultdict(dict)
for conf_file in conf_files:
collect_brokers(conf_file, name2nodes2brokers)
for name, node2broker in name2nodes2brokers.items():
expect_root = False
for node, broker in node2broker.items():
expect_root = broker.is_root_container() or expect_root
if expect_root:
print_container(name, name2nodes2brokers)
if __name__ == '__main__':
conf_dir = '/etc/swift/container-server'
conf_files = [os.path.join(conf_dir, f) for f in os.listdir(conf_dir)
if f.endswith('.conf')]
run(conf_files)

View File

@ -1623,7 +1623,7 @@ class ContainerBroker(DatabaseBroker):
'X-Container-Sysmeta-Shard-Lower', ('', None))
upper, ts = metadata.get(
'X-Container-Sysmeta-Shard-Upper', ('', None))
if created_at is None:
if created_at in (None, ''):
return None
info = self.get_info() # Also ensures self.container is not None
@ -1658,6 +1658,13 @@ class ContainerBroker(DatabaseBroker):
self._root_container = self.container
return
# this is a horrible hack to workaround X-Container-Sysmeta-Shard-Root
# being set to '' when a shard container is deleted. We still want
# is_root_container to be False.
if not path:
self._root_account = self._root_container = ''
return
if path.count('/') != 1 or path.strip('/').count('/') == 0:
raise ValueError('Expected X-Container-Sysmeta-Shard-Root to be '
"of the form 'account/container', got %r" % path)