summaryrefslogtreecommitdiff
path: root/functional/common
diff options
context:
space:
mode:
authorSteve Martinelli <stevemar@ca.ibm.com>2014-09-19 02:42:55 +0000
committerSteve Martinelli <stevemar@ca.ibm.com>2014-10-01 19:46:07 -0400
commit742982af4bb94b73a78c06688732acf1c8127f8a (patch)
tree0c84f0dbcb181e044c440bf7411811b014d72495 /functional/common
parent02320a5a2437acdc501a003cca53310a444e4b4e (diff)
downloadpython-openstackclient-742982af4bb94b73a78c06688732acf1c8127f8a.tar.gz
Add functional tests to osc
Create a script that kicks off function tests that exercise openstackclient commands against a cloud. If no keystone/openstack process is detected, a devstack instance is spun up and the tests are run against that. There is also a hook added to tox.ini so that we can run these tests easily from a gate job. Change-Id: I3cc8b2b800de7ca74af506d2c7e8ee481fa985f0
Diffstat (limited to 'functional/common')
-rw-r--r--functional/common/__init__.py0
-rw-r--r--functional/common/exceptions.py26
-rw-r--r--functional/common/test.py129
3 files changed, 155 insertions, 0 deletions
diff --git a/functional/common/__init__.py b/functional/common/__init__.py
new file mode 100644
index 00000000..e69de29b
--- /dev/null
+++ b/functional/common/__init__.py
diff --git a/functional/common/exceptions.py b/functional/common/exceptions.py
new file mode 100644
index 00000000..47c6071e
--- /dev/null
+++ b/functional/common/exceptions.py
@@ -0,0 +1,26 @@
+# 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.
+
+
+class CommandFailed(Exception):
+ def __init__(self, returncode, cmd, output, stderr):
+ super(CommandFailed, self).__init__()
+ self.returncode = returncode
+ self.cmd = cmd
+ self.stdout = output
+ self.stderr = stderr
+
+ def __str__(self):
+ return ("Command '%s' returned non-zero exit status %d.\n"
+ "stdout:\n%s\n"
+ "stderr:\n%s" % (self.cmd, self.returncode,
+ self.stdout, self.stderr))
diff --git a/functional/common/test.py b/functional/common/test.py
new file mode 100644
index 00000000..c1bb0b10
--- /dev/null
+++ b/functional/common/test.py
@@ -0,0 +1,129 @@
+# 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 re
+import shlex
+import subprocess
+import testtools
+
+import six
+
+from functional.common import exceptions
+
+
+def execute(cmd, action, flags='', params='', fail_ok=False,
+ merge_stderr=False):
+ """Executes specified command for the given action."""
+ cmd = ' '.join([cmd, flags, action, params])
+ cmd = shlex.split(cmd.encode('utf-8'))
+ result = ''
+ result_err = ''
+ stdout = subprocess.PIPE
+ stderr = subprocess.STDOUT if merge_stderr else subprocess.PIPE
+ proc = subprocess.Popen(cmd, stdout=stdout, stderr=stderr)
+ result, result_err = proc.communicate()
+ if not fail_ok and proc.returncode != 0:
+ raise exceptions.CommandFailed(proc.returncode, cmd, result,
+ result_err)
+ return result
+
+
+class TestCase(testtools.TestCase):
+
+ delimiter_line = re.compile('^\+\-[\+\-]+\-\+$')
+
+ def openstack(self, action, flags='', params='', fail_ok=False):
+ """Executes openstackclient command for the given action."""
+ return execute('openstack', action, flags, params, fail_ok)
+
+ def assert_table_structure(self, items, field_names):
+ """Verify that all items have keys listed in field_names."""
+ for item in items:
+ for field in field_names:
+ self.assertIn(field, item)
+
+ def assert_show_fields(self, items, field_names):
+ """Verify that all items have keys listed in field_names."""
+ for item in items:
+ for key in six.iterkeys(item):
+ self.assertIn(key, field_names)
+
+ def parse_show(self, raw_output):
+ """Return list of dicts with item values parsed from cli output."""
+
+ items = []
+ table_ = self.table(raw_output)
+ for row in table_['values']:
+ item = {}
+ item[row[0]] = row[1]
+ items.append(item)
+ return items
+
+ def parse_listing(self, raw_output):
+ """Return list of dicts with basic item parsed from cli output."""
+
+ items = []
+ table_ = self.table(raw_output)
+ for row in table_['values']:
+ item = {}
+ for col_idx, col_key in enumerate(table_['headers']):
+ item[col_key] = row[col_idx]
+ items.append(item)
+ return items
+
+ def table(self, output_lines):
+ """Parse single table from cli output.
+
+ Return dict with list of column names in 'headers' key and
+ rows in 'values' key.
+ """
+ table_ = {'headers': [], 'values': []}
+ columns = None
+
+ if not isinstance(output_lines, list):
+ output_lines = output_lines.split('\n')
+
+ if not output_lines[-1]:
+ # skip last line if empty (just newline at the end)
+ output_lines = output_lines[:-1]
+
+ for line in output_lines:
+ if self.delimiter_line.match(line):
+ columns = self._table_columns(line)
+ continue
+ if '|' not in line:
+ continue
+ row = []
+ for col in columns:
+ row.append(line[col[0]:col[1]].strip())
+ if table_['headers']:
+ table_['values'].append(row)
+ else:
+ table_['headers'] = row
+
+ return table_
+
+ def _table_columns(self, first_table_row):
+ """Find column ranges in output line.
+
+ Return list of tuples (start,end) for each column
+ detected by plus (+) characters in delimiter line.
+ """
+ positions = []
+ start = 1 # there is '+' at 0
+ while start < len(first_table_row):
+ end = first_table_row.find('+', start)
+ if end == -1:
+ break
+ positions.append((start, end))
+ start = end + 1
+ return positions