2013-07-03 16:47:40 -05:00
|
|
|
# Copyright 2012-2013 OpenStack Foundation
|
|
|
|
# Copyright 2013 Nebula Inc.
|
2013-01-24 12:00:30 -06:00
|
|
|
#
|
|
|
|
# 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.
|
|
|
|
|
parseactions: Use ArgumentError, not ArgumentTypeError
If you use the former, you get a pretty error message when there's a
failure. If you use the latter, you get an ugly traceback when used with
the '--debug' flag.
Without this change:
$ openstack flavor create ... --property '' foo
...
Traceback (most recent call last):
File "/tmp/venv/lib/python3.11/site-packages/cliff/app.py", line 402, in run_subcommand
parsed_args = cmd_parser.parse_args(sub_argv)
^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
File "/usr/lib64/python3.11/argparse.py", line 1862, in parse_args
args, argv = self.parse_known_args(args, namespace)
^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
File "/usr/lib64/python3.11/argparse.py", line 1895, in parse_known_args
namespace, args = self._parse_known_args(args, namespace)
^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
File "/usr/lib64/python3.11/argparse.py", line 2107, in _parse_known_args
start_index = consume_optional(start_index)
^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
File "/usr/lib64/python3.11/argparse.py", line 2047, in consume_optional
take_action(action, args, option_string)
File "/usr/lib64/python3.11/argparse.py", line 1971, in take_action
action(self, namespace, argument_values, option_string)
File "/tmp/venv/lib/python3.11/site-packages/osc_lib/cli/parseractions.py", line 45, in __call__
raise argparse.ArgumentTypeError(msg % str(values))
argparse.ArgumentTypeError: Expected 'key=value' type, but got:
clean_up CreateFlavor: Expected 'key=value' type, but got:
With this change:
$ openstack flavor create ... --property '' foo
...
usage: openstack flavor create [-h] [-f {json,shell,table,value,yaml}] [-c COLUMN]
[--noindent] [--prefix PREFIX] [--max-width <integer>]
[--fit-width] [--print-empty] [--id <id>]
[--ram <size-mb>] [--disk <size-gb>]
[--ephemeral <size-gb>] [--swap <size-mb>]
[--vcpus <vcpus>] [--rxtx-factor <factor>]
[--public | --private] [--property <key=value>]
[--project <project>] [--description <description>]
[--project-domain <project-domain>]
<flavor-name>
openstack flavor create: error: argument --property: Expected 'key=value' type, but got:
clean_up CreateFlavor:
Change-Id: I9e78b35ad9d016d7a33655141ec579397c5344c0
Signed-off-by: Stephen Finucane <stephenfin@redhat.com>
2023-11-15 10:21:36 +00:00
|
|
|
import argparse
|
2023-09-06 11:22:49 +01:00
|
|
|
import io
|
2016-07-21 12:14:45 +08:00
|
|
|
import os
|
2017-07-15 16:58:08 +00:00
|
|
|
|
|
|
|
import fixtures
|
2012-12-27 11:50:53 -06:00
|
|
|
import testtools
|
2012-06-20 10:05:56 -05:00
|
|
|
|
2016-09-05 22:14:33 -07:00
|
|
|
from openstackclient.tests.unit import fakes
|
2013-07-03 16:47:40 -05:00
|
|
|
|
2012-06-20 10:05:56 -05:00
|
|
|
|
2014-10-13 16:30:38 -05:00
|
|
|
class ParserException(Exception):
|
|
|
|
pass
|
|
|
|
|
|
|
|
|
2017-04-29 00:32:32 +00:00
|
|
|
class CompareBySet(list):
|
|
|
|
"""Class to compare value using set."""
|
2023-05-08 11:37:42 +01:00
|
|
|
|
2017-04-29 00:32:32 +00:00
|
|
|
def __eq__(self, other):
|
|
|
|
return set(self) == set(other)
|
|
|
|
|
|
|
|
|
2012-12-27 11:50:53 -06:00
|
|
|
class TestCase(testtools.TestCase):
|
2023-03-30 18:58:33 +01:00
|
|
|
# provide additional context for failures
|
|
|
|
maxDiff = None
|
|
|
|
|
2012-06-20 10:05:56 -05:00
|
|
|
def setUp(self):
|
2013-06-10 14:06:41 -05:00
|
|
|
testtools.TestCase.setUp(self)
|
2013-01-24 12:33:17 -06:00
|
|
|
|
2023-05-08 11:37:42 +01:00
|
|
|
if (
|
|
|
|
os.environ.get("OS_STDOUT_CAPTURE") == "True"
|
|
|
|
or os.environ.get("OS_STDOUT_CAPTURE") == "1"
|
|
|
|
):
|
2013-01-22 11:09:11 -06:00
|
|
|
stdout = self.useFixture(fixtures.StringStream("stdout")).stream
|
|
|
|
self.useFixture(fixtures.MonkeyPatch("sys.stdout", stdout))
|
2013-01-24 12:33:17 -06:00
|
|
|
|
2023-05-08 11:37:42 +01:00
|
|
|
if (
|
|
|
|
os.environ.get("OS_STDERR_CAPTURE") == "True"
|
|
|
|
or os.environ.get("OS_STDERR_CAPTURE") == "1"
|
|
|
|
):
|
2013-01-22 11:09:11 -06:00
|
|
|
stderr = self.useFixture(fixtures.StringStream("stderr")).stream
|
|
|
|
self.useFixture(fixtures.MonkeyPatch("sys.stderr", stderr))
|
2013-06-10 14:06:41 -05:00
|
|
|
|
2022-10-05 17:48:32 +01:00
|
|
|
self.log = self.useFixture(fixtures.LoggerFixture())
|
|
|
|
|
2014-07-07 11:42:37 -05:00
|
|
|
def assertNotCalled(self, m, msg=None):
|
|
|
|
"""Assert a function was not called"""
|
|
|
|
|
|
|
|
if m.called:
|
|
|
|
if not msg:
|
|
|
|
msg = 'method %s should not have been called' % m
|
|
|
|
self.fail(msg)
|
|
|
|
|
2013-07-03 16:47:40 -05:00
|
|
|
|
|
|
|
class TestCommand(TestCase):
|
|
|
|
"""Test cliff command classes"""
|
|
|
|
|
|
|
|
def setUp(self):
|
|
|
|
super().setUp()
|
|
|
|
# Build up a fake app
|
|
|
|
self.fake_stdout = fakes.FakeStdout()
|
2015-02-05 23:40:11 -08:00
|
|
|
self.fake_log = fakes.FakeLog()
|
|
|
|
self.app = fakes.FakeApp(self.fake_stdout, self.fake_log)
|
2013-09-09 14:52:45 -05:00
|
|
|
self.app.client_manager = fakes.FakeClientManager()
|
2016-03-08 15:18:16 -06:00
|
|
|
self.app.options = fakes.FakeOptions()
|
2013-07-03 16:47:40 -05:00
|
|
|
|
|
|
|
def check_parser(self, cmd, args, verify_args):
|
|
|
|
cmd_parser = cmd.get_parser('check_parser')
|
2023-09-06 11:22:49 +01:00
|
|
|
stderr = io.StringIO()
|
2019-09-18 11:58:12 -04:00
|
|
|
with fixtures.MonkeyPatch('sys.stderr', stderr):
|
|
|
|
try:
|
|
|
|
parsed_args = cmd_parser.parse_args(args)
|
parseactions: Use ArgumentError, not ArgumentTypeError
If you use the former, you get a pretty error message when there's a
failure. If you use the latter, you get an ugly traceback when used with
the '--debug' flag.
Without this change:
$ openstack flavor create ... --property '' foo
...
Traceback (most recent call last):
File "/tmp/venv/lib/python3.11/site-packages/cliff/app.py", line 402, in run_subcommand
parsed_args = cmd_parser.parse_args(sub_argv)
^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
File "/usr/lib64/python3.11/argparse.py", line 1862, in parse_args
args, argv = self.parse_known_args(args, namespace)
^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
File "/usr/lib64/python3.11/argparse.py", line 1895, in parse_known_args
namespace, args = self._parse_known_args(args, namespace)
^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
File "/usr/lib64/python3.11/argparse.py", line 2107, in _parse_known_args
start_index = consume_optional(start_index)
^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
File "/usr/lib64/python3.11/argparse.py", line 2047, in consume_optional
take_action(action, args, option_string)
File "/usr/lib64/python3.11/argparse.py", line 1971, in take_action
action(self, namespace, argument_values, option_string)
File "/tmp/venv/lib/python3.11/site-packages/osc_lib/cli/parseractions.py", line 45, in __call__
raise argparse.ArgumentTypeError(msg % str(values))
argparse.ArgumentTypeError: Expected 'key=value' type, but got:
clean_up CreateFlavor: Expected 'key=value' type, but got:
With this change:
$ openstack flavor create ... --property '' foo
...
usage: openstack flavor create [-h] [-f {json,shell,table,value,yaml}] [-c COLUMN]
[--noindent] [--prefix PREFIX] [--max-width <integer>]
[--fit-width] [--print-empty] [--id <id>]
[--ram <size-mb>] [--disk <size-gb>]
[--ephemeral <size-gb>] [--swap <size-mb>]
[--vcpus <vcpus>] [--rxtx-factor <factor>]
[--public | --private] [--property <key=value>]
[--project <project>] [--description <description>]
[--project-domain <project-domain>]
<flavor-name>
openstack flavor create: error: argument --property: Expected 'key=value' type, but got:
clean_up CreateFlavor:
Change-Id: I9e78b35ad9d016d7a33655141ec579397c5344c0
Signed-off-by: Stephen Finucane <stephenfin@redhat.com>
2023-11-15 10:21:36 +00:00
|
|
|
except (
|
|
|
|
SystemExit,
|
|
|
|
argparse.ArgumentTypeError,
|
|
|
|
argparse.ArgumentError,
|
|
|
|
):
|
2023-05-08 11:37:42 +01:00
|
|
|
raise ParserException(
|
|
|
|
"Argument parse failed: %s" % stderr.getvalue()
|
|
|
|
)
|
2013-07-03 16:47:40 -05:00
|
|
|
for av in verify_args:
|
|
|
|
attr, value = av
|
|
|
|
if attr:
|
|
|
|
self.assertIn(attr, parsed_args)
|
2015-01-27 17:02:41 +08:00
|
|
|
self.assertEqual(value, getattr(parsed_args, attr))
|
2013-07-03 16:47:40 -05:00
|
|
|
return parsed_args
|