Added Sahara Clusters scenario

The scenario creates and deletes a Hadoop cluster.

Change-Id: I1280f5f3f4cd7415788a3e474dd1852c68a3c35c
This commit is contained in:
Nikita Konovalov 2014-07-25 14:03:01 +04:00
parent 24218d9ff8
commit e2a3e1c941
7 changed files with 387 additions and 8 deletions

View File

@ -0,0 +1,31 @@
{
"SaharaClusters.create_and_delete_cluster": [
{
"args": {
"flavor": {
"name": "m1.small"
},
"node_count": 2,
"plugin_name": "vanilla",
"hadoop_version": "2.3.0"
},
"runner": {
"type": "constant",
"times": 4,
"concurrency": 2
},
"context": {
"users": {
"tenants": 1,
"users_per_tenant": 1
},
"sahara_image": {
"image_url": "http://sahara-files.mirantis.com/sahara-icehouse-vanilla-2.3.0-ubuntu-13.10.qcow2",
"username": "ubuntu",
"plugin_name": "vanilla",
"hadoop_version": "2.3.0"
}
}
}
]
}

View File

@ -0,0 +1,22 @@
---
SaharaClusters.create_and_delete_cluster:
-
args:
flavor:
name: "m1.small"
node_count: 2
plugin_name: "vanilla"
hadoop_version: "2.3.0"
runner:
type: "constant"
times: 4
concurrency: 2
context:
users:
tenants: 1
users_per_tenant: 1
sahara_image:
image_url: "http://sahara-files.mirantis.com/sahara-icehouse-vanilla-2.3.0-ubuntu-13.10.qcow2"
username: "ubuntu"
plugin_name: "vanilla"
hadoop_version: "2.3.0"

View File

@ -331,6 +331,17 @@
#nova_server_image_delete_poll_interval=2.0 #nova_server_image_delete_poll_interval=2.0
#
# Options defined in rally.benchmark.scenarios.sahara.utils
#
# A timeout in seconds for a cluster create operation
#cluster_create_timeout=600
# Cluster status polling interval in seconds
#cluster_check_interval=5
[database] [database]
# #

View File

@ -0,0 +1,62 @@
# Copyright 2014: Mirantis Inc.
# All Rights Reserved.
#
# 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 rally.benchmark.scenarios import base
from rally.benchmark.scenarios.sahara import utils
from rally.benchmark import types
from rally.benchmark import validation
from rally import consts
from rally.openstack.common import log as logging
LOG = logging.getLogger(__name__)
class SaharaClusters(utils.SaharaScenario):
@types.set(flavor=types.FlavorResourceType)
@validation.add(validation.flavor_exists('flavor'))
@validation.required_services(consts.Service.SAHARA)
@validation.required_contexts("users", "sahara_image")
@validation.add(validation.number("node_count", minval=2,
integer_only=True))
@base.scenario(context={"cleanup": ["sahara"]})
def create_and_delete_cluster(self, flavor, node_count,
plugin_name="vanilla",
hadoop_version="2.3.0"):
"""Test the Sahara Cluster launch and delete commands.
This scenario launches a Hadoop cluster, waits until it becomes
'Active' and deletes it.
:param flavor: The Nova flavor that will be for nodes in the
created node groups
:param node_count: The total number of instances in a cluster (>= 2)
:param plugin_name: The name of a provisioning plugin
:param hadoop_version: The version of Hadoop distribution supported by
the specified plugin.
"""
tenant_id = self.clients("keystone").tenant_id
image_id = self.context()["sahara_images"][tenant_id]
LOG.debug("Using Image: %s" % image_id)
cluster = self._launch_cluster(flavor_id=flavor,
image_id=image_id,
node_count=node_count,
plugin_name=plugin_name,
hadoop_version=hadoop_version)
self._delete_cluster(cluster)

View File

@ -13,10 +13,28 @@
# License for the specific language governing permissions and limitations # License for the specific language governing permissions and limitations
# under the License. # under the License.
from rally.benchmark.scenarios import base as scenario_base from oslo.config import cfg
from saharaclient.api import base as sahara_base
from rally.benchmark.scenarios import base
from rally.benchmark import utils as bench_utils
from rally.openstack.common import log as logging
LOG = logging.getLogger(__name__)
CONF = cfg.CONF
CREATE_CLUSTER_OPTS = [
cfg.IntOpt("cluster_create_timeout", default=600,
help="A timeout in seconds for a cluster create operation"),
cfg.IntOpt("cluster_check_interval", default=5,
help="Cluster status polling interval in seconds")
]
benchmark_group = cfg.OptGroup(name='benchmark', title='benchmark options')
CONF.register_opts(CREATE_CLUSTER_OPTS, group=benchmark_group)
class SaharaScenario(scenario_base.Scenario): class SaharaScenario(base.Scenario):
RESOURCE_NAME_LENGTH = 20 RESOURCE_NAME_LENGTH = 20
@ -34,14 +52,26 @@ class SaharaScenario(scenario_base.Scenario):
} }
} }
@scenario_base.atomic_action_timer('sahara.list_node_group_templates') REPLICATION_CONFIGS = {
"vanilla": {
"1.2.1": {
"target": "HDFS",
"config_name": "dfs.replication"
},
"2.3.0": {
"target": "HDFS",
"config_name": "dfs.replication"
}
}
}
@base.atomic_action_timer('sahara.list_node_group_templates')
def _list_node_group_templates(self): def _list_node_group_templates(self):
"""Returns user Node Group Templates list.""" """Returns user Node Group Templates list."""
return self.clients("sahara").node_group_templates.list() return self.clients("sahara").node_group_templates.list()
@scenario_base.atomic_action_timer( @base.atomic_action_timer('sahara.create_master_node_group_template')
'sahara.create_master_node_group_template')
def _create_master_node_group_template(self, flavor_id, plugin_name, def _create_master_node_group_template(self, flavor_id, plugin_name,
hadoop_version): hadoop_version):
"""Creates a master Node Group Template with a random name. """Creates a master Node Group Template with a random name.
@ -63,8 +93,7 @@ class SaharaScenario(scenario_base.Scenario):
node_processes=self.NODE_PROCESSES[plugin_name][hadoop_version] node_processes=self.NODE_PROCESSES[plugin_name][hadoop_version]
["master"]) ["master"])
@scenario_base.atomic_action_timer( @base.atomic_action_timer('sahara.create_worker_node_group_template')
'sahara.create_worker_node_group_template')
def _create_worker_node_group_template(self, flavor_id, plugin_name, def _create_worker_node_group_template(self, flavor_id, plugin_name,
hadoop_version): hadoop_version):
"""Creates a worker Node Group Template with a random name. """Creates a worker Node Group Template with a random name.
@ -86,7 +115,7 @@ class SaharaScenario(scenario_base.Scenario):
node_processes=self.NODE_PROCESSES[plugin_name][hadoop_version] node_processes=self.NODE_PROCESSES[plugin_name][hadoop_version]
["worker"]) ["worker"])
@scenario_base.atomic_action_timer('sahara.delete_node_group_template') @base.atomic_action_timer('sahara.delete_node_group_template')
def _delete_node_group_template(self, node_group): def _delete_node_group_template(self, node_group):
"""Deletes a Node Group Template by id. """Deletes a Node Group Template by id.
@ -95,3 +124,86 @@ class SaharaScenario(scenario_base.Scenario):
""" """
self.clients("sahara").node_group_templates.delete(node_group.id) self.clients("sahara").node_group_templates.delete(node_group.id)
@base.atomic_action_timer('sahara.launch_cluster')
def _launch_cluster(self, plugin_name, hadoop_version, flavor_id,
image_id, node_count):
"""Creates a cluster and wait until it becomes Active.
The cluster is created with two node groups. The master Node Group is
created with one instance. The worker node group contains
node_count - 1 instances.
:param plugin_name: The provisioning plugin name
:param hadoop_version: Hadoop version supported by the plugin
:param flavor_id: The flavor which will be used to create instances
:param image_id: The image id that will be used to boot instances
:param node_count: The total number of instances. 1 master node, others
for the workers
:return: The created cluster
"""
node_groups = [
{
"name": "master-ng",
"flavor_id": flavor_id,
"node_processes": self.NODE_PROCESSES[plugin_name]
[hadoop_version]["master"],
"count": 1
}, {
"name": "worker-ng",
"flavor_id": flavor_id,
"node_processes": self.NODE_PROCESSES[plugin_name]
[hadoop_version]["worker"],
"count": node_count - 1
}
]
name = self._generate_random_name(prefix="sahara-cluster-")
replication_value = min(node_count - 1, 3)
# 3 is a default Hadoop replication
conf = self.REPLICATION_CONFIGS[plugin_name][hadoop_version]
LOG.debug("Using replication factor: %s" % replication_value)
cluster_object = self.clients("sahara").clusters.create(
name=name,
plugin_name=plugin_name,
hadoop_version=hadoop_version,
node_groups=node_groups,
default_image_id=image_id,
cluster_configs={conf["target"]: {
conf["config_name"]: replication_value}
}
)
def is_active(cluster_id):
return self.clients("sahara").clusters.get(
cluster_id).status.lower() == "active"
bench_utils.wait_for(
resource=cluster_object.id, is_ready=is_active,
timeout=CONF.benchmark.cluster_create_timeout,
check_interval=CONF.benchmark.cluster_check_interval)
return self.clients("sahara").clusters.get(cluster_object.id)
@base.atomic_action_timer('sahara.delete_cluster')
def _delete_cluster(self, cluster):
"""Calls a Cluster delete by id and waits for complete deletion.
:param cluster: The Cluster to be deleted
:return:
"""
self.clients("sahara").clusters.delete(cluster.id)
def is_deleted(cl_id):
try:
self.clients("sahara").clusters.get(cl_id)
return False
except sahara_base.APIException:
return True
bench_utils.wait_for(resource=cluster.id, is_ready=is_deleted)

View File

@ -0,0 +1,52 @@
# Copyright 2014: Mirantis Inc.
# All Rights Reserved.
#
# 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 mock
from rally.benchmark.scenarios.sahara import clusters
from tests import test
SAHARA_CLUSTERS = "rally.benchmark.scenarios.sahara.clusters.SaharaClusters"
SAHARA_UTILS = 'rally.benchmark.scenarios.sahara.utils'
class SaharaNodeGroupTemplatesTestCase(test.TestCase):
@mock.patch(SAHARA_CLUSTERS + "._delete_cluster")
@mock.patch(SAHARA_CLUSTERS + "._launch_cluster",
return_value=mock.MagicMock(id=42))
@mock.patch(SAHARA_UTILS + '.SaharaScenario.clients')
def test_create_and_delete_cluster(self, mock_clients, mock_launch_cluster,
mock_delete_cluster):
clusters_scenario = clusters.SaharaClusters()
clusters_scenario.clients("keystone").tenant_id = "test_tenant"
clusters_scenario.context = mock.MagicMock(return_value={
"sahara_images": {"test_tenant": "test_image"}}
)
clusters_scenario.create_and_delete_cluster("test_flavor", 5,
"test_plugin",
"test_version")
mock_launch_cluster.assert_called_once_with(
flavor_id="test_flavor",
image_id="test_image",
node_count=5,
plugin_name="test_plugin",
hadoop_version="test_version")
mock_delete_cluster.assert_called_once_with(
mock_launch_cluster.return_value)

View File

@ -14,6 +14,7 @@
# under the License. # under the License.
import mock import mock
from saharaclient.api import base as sahara_base
from rally.benchmark.scenarios.sahara import utils from rally.benchmark.scenarios.sahara import utils
from tests.benchmark.scenarios import test_base from tests.benchmark.scenarios import test_base
@ -108,3 +109,91 @@ class SaharaNodeGroupTemplatesScenarioTestCase(test.TestCase):
self._test_atomic_action_timer(scenario.atomic_actions(), self._test_atomic_action_timer(scenario.atomic_actions(),
'sahara.delete_node_group_template') 'sahara.delete_node_group_template')
@mock.patch(SAHARA_UTILS + '.SaharaScenario._generate_random_name',
return_value="random_name")
@mock.patch(SAHARA_UTILS + '.SaharaScenario.clients')
def test_launch_cluster(self, mock_clients, mock_random_name):
scenario = utils.SaharaScenario()
mock_processes = {
"test_plugin": {
"test_version": {
"master": ["p1"],
"worker": ["p2"]
}
}
}
mock_configs = {
"test_plugin": {
"test_version": {
"target": "HDFS",
"config_name": "dfs.replication"
}
}
}
node_groups = [
{
"name": "master-ng",
"flavor_id": "test_flavor",
"node_processes": ["p1"],
"count": 1
}, {
"name": "worker-ng",
"flavor_id": "test_flavor",
"node_processes": ["p2"],
"count": 41
}
]
scenario.NODE_PROCESSES = mock_processes
scenario.REPLICATION_CONFIGS = mock_configs
mock_clients("sahara").clusters.create.return_value = mock.MagicMock(
id="test_cluster_id")
mock_clients("sahara").clusters.get.return_value = mock.MagicMock(
status="active")
scenario._launch_cluster(
plugin_name="test_plugin",
hadoop_version="test_version",
flavor_id="test_flavor",
image_id="test_image",
node_count=42
)
mock_clients("sahara").clusters.create.assert_called_once_with(
name="random_name",
plugin_name="test_plugin",
hadoop_version="test_version",
node_groups=node_groups,
default_image_id="test_image",
cluster_configs={"HDFS": {"dfs.replication": 3}}
)
self._test_atomic_action_timer(scenario.atomic_actions(),
'sahara.launch_cluster')
@mock.patch(SAHARA_UTILS + '.SaharaScenario.clients')
def test_delete_cluster(self, mock_clients):
scenario = utils.SaharaScenario()
cluster = mock.MagicMock(id=42)
mock_clients("sahara").clusters.get.side_effect = [
cluster, sahara_base.APIException()
]
scenario._delete_cluster(cluster)
delete_mock = mock_clients("sahara").clusters.delete
delete_mock.assert_called_once_with(42)
mock_clients("sahara").clusters.get.assert_has_calls([
mock.call(42),
mock.call(42)])
self._test_atomic_action_timer(scenario.atomic_actions(),
'sahara.delete_cluster')