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:
parent
2145c9bf5a
commit
1b8334b35a
@ -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
|
@ -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()
|
||||||
|
|
||||||
|
|
||||||
|
@ -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,
|
||||||
|
@ -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,
|
||||||
|
@ -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)
|
||||||
|
@ -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,
|
||||||
|
@ -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()
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user