Make zaqar-bench use credentials from os_client_config

Currently zaqar-bench can't benchmark Zaqar while Zaqar is using
keystone authentication backend.

This patch makes it possible by using os_client_config library. The
library gets authentication parameters from "clouds.yaml". If keystone
authentication environment variables present, they override values from
"clouds.yaml". The aquired parameters then passed to constructors of
each python-zaqarclient's Client object used in zaqar-bench.

The patch also makes benchmark queue names reusable across all
zaqar-bench parts and by this fixes the old DRY princible bug.

To use zaqar-bench with keystone authentication the user must
explicitly set OS_AUTH_STRATEGY=keystone as environment variable
before running the tool. Otherwise the default 'noauth' auth strategy
will be used. This allows the user to run zaqar-bench as usual.

This patch also adds option "--api-version" with it's short version
"api" to zaqar-bench which defaults to Zaqar API v2.

Change-Id: I0a7aaeaeac6da1b2c9f08fbfdddd467de5747a28
Closes-Bug: 1523752
This commit is contained in:
Eva Balycheva 2015-11-28 07:42:09 +03:00
parent 2145c9bf5a
commit 1b8334b35a
7 changed files with 134 additions and 21 deletions

View File

@ -2,3 +2,4 @@ argparse>=1.2.1
gevent>=1.0.1 gevent>=1.0.1
marktime>=0.2.0 marktime>=0.2.0
python-zaqarclient>=0.0.2 python-zaqarclient>=0.0.2
os-client-config!=1.6.2,>=1.4.0

View File

@ -16,6 +16,10 @@ from __future__ import print_function
import json import json
import multiprocessing as mp import multiprocessing as mp
import os
# NOTE(Eva-i): See https://github.com/gevent/gevent/issues/349. Let's keep
# it until the new stable version of gevent(>=1.1) will be released.
os.environ["GEVENT_RESOLVER"] = "ares"
from zaqar.bench import config from zaqar.bench import config
from zaqar.bench import consumer from zaqar.bench import consumer
@ -39,11 +43,8 @@ def _print_verbose_stats(name, stats):
def _reset_queues(): def _reset_queues():
cli = helpers.get_new_client() cli = helpers.get_new_client()
for queue_name in helpers.queue_names:
for i in range(CONF.num_queues): queue = cli.queue(queue_name)
# TODO(kgriffs): DRY up name generation so it is done
# in a helper, vs. being copy-pasted everywhere.
queue = cli.queue(CONF.queue_prefix + '-' + str(i))
queue.delete() queue.delete()

View File

@ -49,11 +49,14 @@ _CLI_OPTIONS = (
default=5, default=5,
help='Number of Observer Workers'), help='Number of Observer Workers'),
cfg.FloatOpt('api_version', short='api', default='2',
help='Zaqar API version to use'),
cfg.IntOpt('messages_per_claim', short='cno', default=5, cfg.IntOpt('messages_per_claim', short='cno', default=5,
help=('Number of messages the consumer will attempt to ' help=('Number of messages the consumer will attempt to '
'claim at a time')), 'claim at a time')),
cfg.IntOpt('messages_per_list', short='lno', default=5, cfg.IntOpt('messages_per_list', short='lno', default=5,
help=('Number of messages the obserer will attempt to ' help=('Number of messages the observer will attempt to '
'list at a time')), 'list at a time')),
cfg.IntOpt('time', short='t', default=5, cfg.IntOpt('time', short='t', default=5,

View File

@ -96,8 +96,9 @@ def load_generator(stats, num_workers, num_queues,
test_duration, url, ttl, grace, limit): test_duration, url, ttl, grace, limit):
cli = helpers.get_new_client() cli = helpers.get_new_client()
queues = [cli.queue(CONF.queue_prefix + '-' + str(i)) queues = []
for i in range(num_queues)] for queue_name in helpers.queue_names:
queues.append(cli.queue(queue_name))
gevent.joinall([ gevent.joinall([
gevent.spawn(claim_delete, gevent.spawn(claim_delete,

View File

@ -11,21 +11,126 @@
# See the License for the specific language governing permissions and # See the License for the specific language governing permissions and
# limitations under the License. # limitations under the License.
# NOTE(Eva-i): Some code was taken from python-zaqarclient.
import os
import sys
import os_client_config
from six.moves import urllib_parse
from zaqarclient.queues import client from zaqarclient.queues import client
from zaqar.bench import config from zaqar.bench import config
CONF = config.conf CONF = config.conf
client_conf = {
'auth_opts': { def _get_credential_args():
'backend': 'noauth', """Retrieves credential arguments for keystone
'options': {
'os_project_id': 'my-lovely-benchmark', Credentials are either read via os-client-config from the environment
}, or from a config file ('clouds.yaml'). Config file variables override those
}, from the environment variables.
}
devstack produces a clouds.yaml with two named clouds - one named
'devstack' which has user privs and one named 'devstack-admin' which
has admin privs. This function will default to getting the credentials from
environment variables. If not all required credentials present in
environment variables, it tries to get credentials for 'devstack-admin'
cloud in clouds.yaml. If no 'devstack-admin' cloud found, it tried to get
credentials for 'devstack' cloud. If no 'devstack' cloud found, throws
an error and stops the application.
"""
os_cfg = os_client_config.OpenStackConfig()
cloud = os_cfg.get_one_cloud()
cred_args = cloud.get_auth_args()
required_options = ['username', 'password', 'auth_url', 'project_name']
if not all(arg in cred_args for arg in required_options):
try:
cloud = os_cfg.get_one_cloud(cloud='devstack-admin')
except Exception:
try:
cloud = os_cfg.get_one_cloud(cloud='devstack')
except Exception:
print("Insufficient amount of credentials found for keystone "
"authentication. Credentials should reside either in "
"environment variables or in 'clouds.yaml' file. If "
"both present, the ones in environment variables will "
"be preferred. Exiting.")
sys.exit()
cred_args = cloud.get_auth_args()
print("Using '{}' credentials".format(cloud.name))
return cred_args
def _generate_client_conf():
auth_strategy = os.environ.get
if auth_strategy == 'keystone':
args = _get_credential_args()
# FIXME(flwang): Now we're hardcode the keystone auth version, since
# there is a 'bug' with the osc-config which is returning the auth_url
# without version. This should be fixed as long as the bug is fixed.
parsed_url = urllib_parse.urlparse(args['auth_url'])
auth_url = args['auth_url']
if not parsed_url.path or parsed_url.path == '/':
auth_url = urllib_parse.urljoin(args['auth_url'], 'v2.0')
conf = {
'auth_opts': {
'backend': 'keystone',
'options': {
'os_username': args['username'],
'os_password': args['password'],
'os_project_name': args['project_name'],
'os_auth_url': auth_url,
'insecure': '',
},
},
}
else:
conf = {
'auth_opts': {
'backend': 'noauth',
'options': {
'os_project_id': 'my-lovely-benchmark',
},
},
}
print("Using '{0}' authentication method".format(conf['auth_opts']
['backend']))
return conf
class LazyAPIVersion(object):
def __init__(self):
self.api_version = None
@property
def get(self):
if self.api_version is None:
conversion_map = {
1.0: 1,
1.1: 1.1,
2.0: 2,
}
try:
self.api_version = conversion_map[CONF.api_version]
except KeyError:
print("Unknown Zaqar API version: '{}'. Exiting...".format(
CONF.api_version))
sys.exit()
print("Benchmarking Zaqar API v{0}...".format(self.api_version))
return self.api_version
client_conf = _generate_client_conf()
client_api = LazyAPIVersion()
queue_names = []
for i in range(CONF.num_queues):
queue_names.append((CONF.queue_prefix + '-' + str(i)))
def get_new_client(): def get_new_client():
return client.Client(CONF.server_url, 1.1, conf=client_conf) return client.Client(CONF.server_url, client_api.get, conf=client_conf)

View File

@ -100,8 +100,9 @@ def load_generator(stats, num_workers, num_queues,
test_duration, limit): test_duration, limit):
cli = helpers.get_new_client() cli = helpers.get_new_client()
queues = [cli.queue(CONF.queue_prefix + '-' + str(i)) queues = []
for i in range(num_queues)] for queue_name in helpers.queue_names:
queues.append(cli.queue(queue_name))
gevent.joinall([ gevent.joinall([
gevent.spawn(observer, gevent.spawn(observer,

View File

@ -105,8 +105,9 @@ def producer(queues, message_pool, stats, test_duration):
def load_generator(stats, num_workers, num_queues, test_duration): def load_generator(stats, num_workers, num_queues, test_duration):
cli = helpers.get_new_client() cli = helpers.get_new_client()
queues = [cli.queue(CONF.queue_prefix + '-' + str(i)) queues = []
for i in range(num_queues)] for queue_name in helpers.queue_names:
queues.append(cli.queue(queue_name))
message_pool = load_messages() message_pool = load_messages()