Joe Gordon | c97f5c7 | 2013-02-14 01:15:57 +0000 | [diff] [blame] | 1 | # Copyright 2013 OpenStack Foundation |
| 2 | # All Rights Reserved. |
| 3 | # |
| 4 | # Licensed under the Apache License, Version 2.0 (the "License"); you may |
| 5 | # not use this file except in compliance with the License. You may obtain |
| 6 | # a copy of the License at |
| 7 | # |
| 8 | # http://www.apache.org/licenses/LICENSE-2.0 |
| 9 | # |
| 10 | # Unless required by applicable law or agreed to in writing, software |
| 11 | # distributed under the License is distributed on an "AS IS" BASIS, WITHOUT |
| 12 | # WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the |
| 13 | # License for the specific language governing permissions and limitations |
| 14 | # under the License. |
| 15 | |
Mikhail S Medvedev | 13168d0 | 2013-06-24 16:13:40 -0500 | [diff] [blame] | 16 | import os |
Pavel Sedlák | a2b757c | 2013-02-25 18:16:04 +0100 | [diff] [blame] | 17 | import shlex |
| 18 | import subprocess |
Joe Gordon | c97f5c7 | 2013-02-14 01:15:57 +0000 | [diff] [blame] | 19 | |
Sean Dague | f682579 | 2013-05-08 13:51:26 -0400 | [diff] [blame] | 20 | import tempest.cli.output_parser |
Matthew Treinish | e2b56b5 | 2014-01-29 19:25:50 +0000 | [diff] [blame] | 21 | from tempest import config |
Matthew Treinish | f4a9b0f | 2013-07-26 16:58:26 -0400 | [diff] [blame] | 22 | from tempest.openstack.common import log as logging |
Pavel Sedlák | a2b757c | 2013-02-25 18:16:04 +0100 | [diff] [blame] | 23 | import tempest.test |
Joe Gordon | c97f5c7 | 2013-02-14 01:15:57 +0000 | [diff] [blame] | 24 | |
Matthew Treinish | 90aedd1 | 2013-02-25 17:56:49 -0500 | [diff] [blame] | 25 | |
Joe Gordon | c97f5c7 | 2013-02-14 01:15:57 +0000 | [diff] [blame] | 26 | LOG = logging.getLogger(__name__) |
| 27 | |
Matthew Treinish | e2b56b5 | 2014-01-29 19:25:50 +0000 | [diff] [blame] | 28 | CONF = config.CONF |
Pavel Sedlák | a2b757c | 2013-02-25 18:16:04 +0100 | [diff] [blame] | 29 | |
| 30 | |
| 31 | class ClientTestBase(tempest.test.BaseTestCase): |
| 32 | @classmethod |
| 33 | def setUpClass(cls): |
| 34 | if not CONF.cli.enabled: |
| 35 | msg = "cli testing disabled" |
| 36 | raise cls.skipException(msg) |
Pavel Sedlák | a2b757c | 2013-02-25 18:16:04 +0100 | [diff] [blame] | 37 | super(ClientTestBase, cls).setUpClass() |
| 38 | |
| 39 | def __init__(self, *args, **kwargs): |
Sean Dague | f682579 | 2013-05-08 13:51:26 -0400 | [diff] [blame] | 40 | self.parser = tempest.cli.output_parser |
Pavel Sedlák | a2b757c | 2013-02-25 18:16:04 +0100 | [diff] [blame] | 41 | super(ClientTestBase, self).__init__(*args, **kwargs) |
| 42 | |
| 43 | def nova(self, action, flags='', params='', admin=True, fail_ok=False): |
| 44 | """Executes nova command for the given action.""" |
JordanP | 71c85f6 | 2014-02-26 15:53:43 +0000 | [diff] [blame] | 45 | flags += ' --endpoint-type %s' % CONF.compute.endpoint_type |
Pavel Sedlák | a2b757c | 2013-02-25 18:16:04 +0100 | [diff] [blame] | 46 | return self.cmd_with_auth( |
| 47 | 'nova', action, flags, params, admin, fail_ok) |
| 48 | |
Joe Gordon | e8b0e15 | 2013-03-25 13:37:15 -0400 | [diff] [blame] | 49 | def nova_manage(self, action, flags='', params='', fail_ok=False, |
Joe Gordon | 0e7cbf8 | 2013-03-25 19:49:12 +0000 | [diff] [blame] | 50 | merge_stderr=False): |
Joe Gordon | 4edb645 | 2013-03-05 21:18:59 +0000 | [diff] [blame] | 51 | """Executes nova-manage command for the given action.""" |
| 52 | return self.cmd( |
Joe Gordon | e8b0e15 | 2013-03-25 13:37:15 -0400 | [diff] [blame] | 53 | 'nova-manage', action, flags, params, fail_ok, merge_stderr) |
Joe Gordon | 4edb645 | 2013-03-05 21:18:59 +0000 | [diff] [blame] | 54 | |
Pavel Sedlák | 5ce5c03 | 2013-02-25 18:41:30 +0100 | [diff] [blame] | 55 | def keystone(self, action, flags='', params='', admin=True, fail_ok=False): |
| 56 | """Executes keystone command for the given action.""" |
| 57 | return self.cmd_with_auth( |
| 58 | 'keystone', action, flags, params, admin, fail_ok) |
| 59 | |
afazekas | f35f940 | 2013-03-25 14:51:13 +0100 | [diff] [blame] | 60 | def glance(self, action, flags='', params='', admin=True, fail_ok=False): |
| 61 | """Executes glance command for the given action.""" |
JordanP | 71c85f6 | 2014-02-26 15:53:43 +0000 | [diff] [blame] | 62 | flags += ' --os-endpoint-type %s' % CONF.image.endpoint_type |
afazekas | f35f940 | 2013-03-25 14:51:13 +0100 | [diff] [blame] | 63 | return self.cmd_with_auth( |
| 64 | 'glance', action, flags, params, admin, fail_ok) |
| 65 | |
Mehdi Abaakouk | 8581c0b | 2013-10-04 10:45:42 +0200 | [diff] [blame] | 66 | def ceilometer(self, action, flags='', params='', admin=True, |
| 67 | fail_ok=False): |
| 68 | """Executes ceilometer command for the given action.""" |
JordanP | 71c85f6 | 2014-02-26 15:53:43 +0000 | [diff] [blame] | 69 | flags += ' --os-endpoint-type %s' % CONF.telemetry.endpoint_type |
Mehdi Abaakouk | 8581c0b | 2013-10-04 10:45:42 +0200 | [diff] [blame] | 70 | return self.cmd_with_auth( |
| 71 | 'ceilometer', action, flags, params, admin, fail_ok) |
| 72 | |
Steven Hardy | 5de54ee | 2013-12-31 15:58:30 +0000 | [diff] [blame] | 73 | def heat(self, action, flags='', params='', admin=True, |
| 74 | fail_ok=False): |
| 75 | """Executes heat command for the given action.""" |
JordanP | 71c85f6 | 2014-02-26 15:53:43 +0000 | [diff] [blame] | 76 | flags += ' --os-endpoint-type %s' % CONF.orchestration.endpoint_type |
Steven Hardy | 5de54ee | 2013-12-31 15:58:30 +0000 | [diff] [blame] | 77 | return self.cmd_with_auth( |
| 78 | 'heat', action, flags, params, admin, fail_ok) |
| 79 | |
saurabh | 467c411 | 2013-07-08 17:08:31 +0530 | [diff] [blame] | 80 | def cinder(self, action, flags='', params='', admin=True, fail_ok=False): |
| 81 | """Executes cinder command for the given action.""" |
JordanP | 71c85f6 | 2014-02-26 15:53:43 +0000 | [diff] [blame] | 82 | flags += ' --endpoint-type %s' % CONF.volume.endpoint_type |
saurabh | 467c411 | 2013-07-08 17:08:31 +0530 | [diff] [blame] | 83 | return self.cmd_with_auth( |
| 84 | 'cinder', action, flags, params, admin, fail_ok) |
| 85 | |
saurabh | 55c29c7 | 2013-07-26 21:15:08 +0530 | [diff] [blame] | 86 | def neutron(self, action, flags='', params='', admin=True, fail_ok=False): |
| 87 | """Executes neutron command for the given action.""" |
JordanP | 71c85f6 | 2014-02-26 15:53:43 +0000 | [diff] [blame] | 88 | flags += ' --endpoint-type %s' % CONF.network.endpoint_type |
saurabh | 55c29c7 | 2013-07-26 21:15:08 +0530 | [diff] [blame] | 89 | return self.cmd_with_auth( |
| 90 | 'neutron', action, flags, params, admin, fail_ok) |
| 91 | |
Sergey Lukjanov | 9c95a25 | 2014-03-13 23:59:22 +0400 | [diff] [blame] | 92 | def sahara(self, action, flags='', params='', admin=True, fail_ok=False): |
| 93 | """Executes sahara command for the given action.""" |
JordanP | 71c85f6 | 2014-02-26 15:53:43 +0000 | [diff] [blame] | 94 | flags += ' --endpoint-type %s' % CONF.data_processing.endpoint_type |
Sergey Lukjanov | 582d1c6 | 2014-01-21 19:51:21 +0400 | [diff] [blame] | 95 | return self.cmd_with_auth( |
Sergey Lukjanov | 9c95a25 | 2014-03-13 23:59:22 +0400 | [diff] [blame] | 96 | # TODO (slukjanov): replace with sahara when new client released |
Sergey Lukjanov | 582d1c6 | 2014-01-21 19:51:21 +0400 | [diff] [blame] | 97 | 'savanna', action, flags, params, admin, fail_ok) |
| 98 | |
Pavel Sedlák | a2b757c | 2013-02-25 18:16:04 +0100 | [diff] [blame] | 99 | def cmd_with_auth(self, cmd, action, flags='', params='', |
| 100 | admin=True, fail_ok=False): |
| 101 | """Executes given command with auth attributes appended.""" |
Attila Fazekas | c3a095b | 2013-08-17 09:15:44 +0200 | [diff] [blame] | 102 | # TODO(jogo) make admin=False work |
Pavel Sedlák | a2b757c | 2013-02-25 18:16:04 +0100 | [diff] [blame] | 103 | creds = ('--os-username %s --os-tenant-name %s --os-password %s ' |
JordanP | 71c85f6 | 2014-02-26 15:53:43 +0000 | [diff] [blame] | 104 | '--os-auth-url %s' % |
Matthew Treinish | e2b56b5 | 2014-01-29 19:25:50 +0000 | [diff] [blame] | 105 | (CONF.identity.admin_username, |
| 106 | CONF.identity.admin_tenant_name, |
| 107 | CONF.identity.admin_password, |
| 108 | CONF.identity.uri)) |
Pavel Sedlák | a2b757c | 2013-02-25 18:16:04 +0100 | [diff] [blame] | 109 | flags = creds + ' ' + flags |
| 110 | return self.cmd(cmd, action, flags, params, fail_ok) |
| 111 | |
Joe Gordon | e8b0e15 | 2013-03-25 13:37:15 -0400 | [diff] [blame] | 112 | def cmd(self, cmd, action, flags='', params='', fail_ok=False, |
Joe Gordon | 0e7cbf8 | 2013-03-25 19:49:12 +0000 | [diff] [blame] | 113 | merge_stderr=False): |
Pavel Sedlák | a2b757c | 2013-02-25 18:16:04 +0100 | [diff] [blame] | 114 | """Executes specified command for the given action.""" |
Mikhail S Medvedev | 13168d0 | 2013-06-24 16:13:40 -0500 | [diff] [blame] | 115 | cmd = ' '.join([os.path.join(CONF.cli.cli_dir, cmd), |
Pavel Sedlák | a2b757c | 2013-02-25 18:16:04 +0100 | [diff] [blame] | 116 | flags, action, params]) |
| 117 | LOG.info("running: '%s'" % cmd) |
Pavel Sedlák | 0d9a84f | 2013-08-27 19:09:26 +0200 | [diff] [blame] | 118 | cmd_str = cmd |
Pavel Sedlák | a2b757c | 2013-02-25 18:16:04 +0100 | [diff] [blame] | 119 | cmd = shlex.split(cmd) |
Pavel Sedlák | 0d9a84f | 2013-08-27 19:09:26 +0200 | [diff] [blame] | 120 | result = '' |
| 121 | result_err = '' |
Pavel Sedlák | a2b757c | 2013-02-25 18:16:04 +0100 | [diff] [blame] | 122 | try: |
Pavel Sedlák | 0d9a84f | 2013-08-27 19:09:26 +0200 | [diff] [blame] | 123 | stdout = subprocess.PIPE |
| 124 | stderr = subprocess.STDOUT if merge_stderr else subprocess.PIPE |
| 125 | proc = subprocess.Popen( |
| 126 | cmd, stdout=stdout, stderr=stderr) |
| 127 | result, result_err = proc.communicate() |
| 128 | if not fail_ok and proc.returncode != 0: |
| 129 | raise CommandFailed(proc.returncode, |
| 130 | cmd, |
Cyril Roelandt | b18d5fb | 2013-10-09 20:23:39 +0000 | [diff] [blame] | 131 | result, |
| 132 | stderr=result_err) |
Pavel Sedlák | 0d9a84f | 2013-08-27 19:09:26 +0200 | [diff] [blame] | 133 | finally: |
| 134 | LOG.debug('output of %s:\n%s' % (cmd_str, result)) |
| 135 | if not merge_stderr and result_err: |
| 136 | LOG.debug('error output of %s:\n%s' % (cmd_str, result_err)) |
Pavel Sedlák | a2b757c | 2013-02-25 18:16:04 +0100 | [diff] [blame] | 137 | return result |
Pavel Sedlák | 5ce5c03 | 2013-02-25 18:41:30 +0100 | [diff] [blame] | 138 | |
| 139 | def assertTableStruct(self, items, field_names): |
| 140 | """Verify that all items has keys listed in field_names.""" |
| 141 | for item in items: |
| 142 | for field in field_names: |
| 143 | self.assertIn(field, item) |
Pavel Sedlák | 1053bd3 | 2013-04-16 16:47:40 +0200 | [diff] [blame] | 144 | |
Pavel Sedlák | 4c18fa1 | 2013-08-22 21:29:45 +0200 | [diff] [blame] | 145 | def assertFirstLineStartsWith(self, lines, beginning): |
| 146 | self.assertTrue(lines[0].startswith(beginning), |
| 147 | msg=('Beginning of first line has invalid content: %s' |
| 148 | % lines[:3])) |
| 149 | |
Pavel Sedlák | 1053bd3 | 2013-04-16 16:47:40 +0200 | [diff] [blame] | 150 | |
| 151 | class CommandFailed(subprocess.CalledProcessError): |
| 152 | # adds output attribute for python2.6 |
Cyril Roelandt | b18d5fb | 2013-10-09 20:23:39 +0000 | [diff] [blame] | 153 | def __init__(self, returncode, cmd, output, stderr=""): |
Pavel Sedlák | 1053bd3 | 2013-04-16 16:47:40 +0200 | [diff] [blame] | 154 | super(CommandFailed, self).__init__(returncode, cmd) |
| 155 | self.output = output |
Cyril Roelandt | b18d5fb | 2013-10-09 20:23:39 +0000 | [diff] [blame] | 156 | self.stderr = stderr |