2020-02-24 05:57:48 -05:00
|
|
|
#!/usr/bin/env python
|
|
|
|
|
|
|
|
import os
|
|
|
|
import re
|
|
|
|
import subprocess
|
|
|
|
import sys
|
|
|
|
import tempfile
|
|
|
|
import textwrap
|
|
|
|
import unittest
|
|
|
|
|
2021-01-25 21:49:01 -05:00
|
|
|
from future.utils import iteritems
|
|
|
|
|
2020-02-24 05:57:48 -05:00
|
|
|
|
|
|
|
class ConfgenBaseTestCase(unittest.TestCase):
|
|
|
|
@classmethod
|
|
|
|
def setUpClass(self):
|
|
|
|
self.args = dict()
|
|
|
|
self.functions = {'in': self.assertIn,
|
|
|
|
'not in': self.assertNotIn,
|
|
|
|
'equal': self.assertEqual,
|
|
|
|
'not equal': self.assertNotEqual}
|
|
|
|
|
|
|
|
try:
|
|
|
|
regex_func = self.assertRegex
|
|
|
|
except AttributeError:
|
|
|
|
# Python 2 fallback
|
|
|
|
regex_func = self.assertRegexpMatches
|
|
|
|
finally:
|
2020-04-15 23:59:16 -04:00
|
|
|
self.functions['regex'] = lambda instance, s, expr: regex_func(instance, expr, s) # reverse args order
|
2020-02-24 05:57:48 -05:00
|
|
|
|
|
|
|
def setUp(self):
|
|
|
|
with tempfile.NamedTemporaryFile(prefix='test_confgen_', delete=False) as f:
|
|
|
|
self.output_file = f.name
|
|
|
|
self.addCleanup(os.remove, self.output_file)
|
|
|
|
|
|
|
|
def invoke_confgen(self, args):
|
|
|
|
call_args = [sys.executable, '../../confgen.py']
|
|
|
|
|
|
|
|
for (k, v) in iteritems(args):
|
|
|
|
if k != 'output':
|
|
|
|
if isinstance(v, type('')): # easy Python 2/3 compatible str/unicode
|
|
|
|
call_args += ['--{}'.format(k), v]
|
|
|
|
else:
|
|
|
|
for i in v:
|
|
|
|
call_args += ['--{}'.format(k), i]
|
|
|
|
call_args += ['--output', args['output'], self.output_file] # these arguments belong together
|
|
|
|
|
|
|
|
subprocess.check_call(call_args)
|
|
|
|
|
|
|
|
def invoke_and_test(self, in_text, out_text, test='in'):
|
2020-04-15 23:59:16 -04:00
|
|
|
"""
|
|
|
|
Main utility function for testing confgen:
|
|
|
|
|
|
|
|
- Runs confgen via invoke_confgen(), using output method pre-set in test class setup
|
|
|
|
- in_text is the Kconfig file input content
|
|
|
|
- out_text is some expected output from confgen
|
|
|
|
- 'test' can be any function key from self.functions dict (see above). Default is 'in' to test if
|
|
|
|
out_text is a substring of the full confgen output.
|
|
|
|
"""
|
|
|
|
|
2020-02-24 05:57:48 -05:00
|
|
|
with tempfile.NamedTemporaryFile(mode='w+', prefix='test_confgen_', delete=False) as f:
|
|
|
|
self.addCleanup(os.remove, f.name)
|
|
|
|
f.write(textwrap.dedent(in_text))
|
|
|
|
|
|
|
|
self.args['kconfig'] = f.name
|
|
|
|
|
|
|
|
self.invoke_confgen(self.args)
|
|
|
|
|
|
|
|
with open(self.output_file) as f_result:
|
|
|
|
result = f_result.read()
|
|
|
|
|
2020-04-15 23:59:16 -04:00
|
|
|
try:
|
|
|
|
out_text = textwrap.dedent(out_text)
|
|
|
|
except TypeError:
|
|
|
|
pass # probably a regex
|
|
|
|
|
|
|
|
self.functions[test](self, out_text, result)
|
2020-02-24 05:57:48 -05:00
|
|
|
|
|
|
|
|
|
|
|
class CmakeTestCase(ConfgenBaseTestCase):
|
|
|
|
@classmethod
|
|
|
|
def setUpClass(self):
|
|
|
|
super(CmakeTestCase, self).setUpClass()
|
|
|
|
self.args.update({'output': 'cmake'})
|
|
|
|
|
|
|
|
def testStringEscape(self):
|
|
|
|
self.invoke_and_test("""
|
|
|
|
config PASSWORD
|
|
|
|
string "password"
|
|
|
|
default "\\\\~!@#$%^&*()\\\""
|
|
|
|
""", 'set(CONFIG_PASSWORD "\\\\~!@#$%^&*()\\\"")')
|
|
|
|
|
2020-04-15 23:59:16 -04:00
|
|
|
def testHexPrefix(self):
|
|
|
|
self.invoke_and_test(HEXPREFIX_KCONFIG, 'set(CONFIG_HEX_NOPREFIX "0x33")')
|
|
|
|
self.invoke_and_test(HEXPREFIX_KCONFIG, 'set(CONFIG_HEX_PREFIX "0x77")')
|
|
|
|
|
2020-02-24 05:57:48 -05:00
|
|
|
|
|
|
|
class JsonTestCase(ConfgenBaseTestCase):
|
|
|
|
@classmethod
|
|
|
|
def setUpClass(self):
|
|
|
|
super(JsonTestCase, self).setUpClass()
|
|
|
|
self.args.update({'output': 'json'})
|
|
|
|
|
|
|
|
def testStringEscape(self):
|
|
|
|
self.invoke_and_test("""
|
|
|
|
config PASSWORD
|
|
|
|
string "password"
|
|
|
|
default "\\\\~!@#$%^&*()\\\""
|
|
|
|
""", '"PASSWORD": "\\\\~!@#$%^&*()\\\""')
|
|
|
|
|
2020-04-15 23:59:16 -04:00
|
|
|
def testHexPrefix(self):
|
|
|
|
# hex values come out as integers in JSON, due to no hex type
|
|
|
|
self.invoke_and_test(HEXPREFIX_KCONFIG, '"HEX_NOPREFIX": %d' % 0x33)
|
|
|
|
self.invoke_and_test(HEXPREFIX_KCONFIG, '"HEX_PREFIX": %d' % 0x77)
|
|
|
|
|
2020-02-24 05:57:48 -05:00
|
|
|
|
|
|
|
class JsonMenuTestCase(ConfgenBaseTestCase):
|
|
|
|
@classmethod
|
|
|
|
def setUpClass(self):
|
|
|
|
super(JsonMenuTestCase, self).setUpClass()
|
|
|
|
self.args.update({'output': 'json_menus'})
|
|
|
|
|
|
|
|
def testMultipleRanges(self):
|
|
|
|
self.invoke_and_test("""
|
|
|
|
config IDF_TARGET
|
|
|
|
string "IDF target"
|
|
|
|
default "esp32"
|
|
|
|
|
|
|
|
config SOME_SETTING
|
|
|
|
int "setting for the chip"
|
|
|
|
range 0 100 if IDF_TARGET="esp32s0"
|
|
|
|
range 0 10 if IDF_TARGET="esp32"
|
|
|
|
range -10 1 if IDF_TARGET="esp32s2"
|
|
|
|
""", re.compile(r'"range":\s+\[\s+0,\s+10\s+\]'), 'regex')
|
|
|
|
|
|
|
|
def testHexRanges(self):
|
|
|
|
self.invoke_and_test("""
|
|
|
|
config SOME_SETTING
|
|
|
|
hex "setting for the chip"
|
|
|
|
range 0x0 0xaf if UNDEFINED
|
|
|
|
range 0x10 0xaf
|
|
|
|
""", r'"range":\s+\[\s+16,\s+175\s+\]', 'regex')
|
|
|
|
|
|
|
|
|
|
|
|
class ConfigTestCase(ConfgenBaseTestCase):
|
|
|
|
@classmethod
|
|
|
|
def setUpClass(self):
|
|
|
|
super(ConfigTestCase, self).setUpClass()
|
|
|
|
self.args.update({'output': 'config'})
|
|
|
|
self.input = """
|
|
|
|
config TEST
|
|
|
|
bool "test"
|
|
|
|
default "n"
|
|
|
|
"""
|
|
|
|
|
|
|
|
def setUp(self):
|
|
|
|
super(ConfigTestCase, self).setUp()
|
|
|
|
with tempfile.NamedTemporaryFile(mode='w+', prefix='test_confgen_', delete=False) as f:
|
|
|
|
self.addCleanup(os.remove, f.name)
|
|
|
|
self.args.update({'config': f.name}) # this is input in contrast with {'output': 'config'}
|
|
|
|
f.write(textwrap.dedent("""
|
|
|
|
CONFIG_TEST=y
|
|
|
|
CONFIG_UNKNOWN=y
|
|
|
|
"""))
|
|
|
|
|
|
|
|
def testKeepSavedOption(self):
|
|
|
|
self.invoke_and_test(self.input, 'CONFIG_TEST=y')
|
|
|
|
|
|
|
|
def testDiscardUnknownOption(self):
|
|
|
|
self.invoke_and_test(self.input, 'CONFIG_UNKNOWN', 'not in')
|
|
|
|
|
|
|
|
|
2022-08-24 04:47:03 -04:00
|
|
|
class RenameConfigTestCase(ConfgenBaseTestCase):
|
|
|
|
@classmethod
|
|
|
|
def setUpClass(self):
|
|
|
|
super(RenameConfigTestCase, self).setUpClass()
|
|
|
|
# `args` attribute is a dictionary containing the parameters to pass to `confgen.py`.
|
|
|
|
# Specify the name of the output file, this will generate the argument `--output config`.
|
|
|
|
self.args.update({'output': 'config'})
|
|
|
|
# Setup the KConfig file content in the `input` attribute.
|
|
|
|
# Let's define an option that is enabled by default, this is very important.
|
|
|
|
# Indeed, as we explicitly disables it by its former name below, rename will be considered as functional
|
|
|
|
# if the new name, `(CONFIG_)RENAMED_OPTION` is also disabled in the final configuration file.
|
|
|
|
self.input = """
|
|
|
|
config RENAMED_OPTION
|
|
|
|
bool "Renamed option"
|
|
|
|
default y
|
|
|
|
"""
|
|
|
|
|
|
|
|
def setUp(self):
|
|
|
|
super(RenameConfigTestCase, self).setUp()
|
|
|
|
# Setup the actual test. What we want to do is to have a configuration file containing which
|
|
|
|
# option should be enabled or not, this is the equivalent of the `sdkconfig` that we can find
|
|
|
|
# in the examples.
|
|
|
|
with tempfile.NamedTemporaryFile(mode='w+', prefix='test_confgen_', delete=False) as f:
|
|
|
|
self.addCleanup(os.remove, f.name)
|
|
|
|
# The current file name will be given to `confgen.py` after `--config` argument.
|
|
|
|
self.args.update({'config': f.name})
|
|
|
|
# Specify the content of that configuration file, in our case, we want to explicitely
|
|
|
|
# have an option, which needs to be renamed, disabled/not set.
|
|
|
|
f.write(textwrap.dedent("""
|
|
|
|
# CONFIG_NAMED_OPTION is not set
|
|
|
|
"""))
|
|
|
|
# The configuration file is ready, we need to prepare a `rename` configuration file which will
|
|
|
|
# provide the new name for `CONFIG_NAMED_OPTION` we defined above
|
|
|
|
with tempfile.NamedTemporaryFile(mode='w+', prefix='test_confgen_', delete=False) as f:
|
|
|
|
self.addCleanup(os.remove, f.name)
|
|
|
|
# Same as above, the following entry will result in the generation of `--sdkconfig-rename`
|
|
|
|
# parameter followed by the current temporary file name.
|
|
|
|
self.args.update({'sdkconfig-rename': f.name})
|
|
|
|
# The content of our `rename` file is simple: replace `CONFIG_NAMED_OPTION` by `CONFIG_RENAMED_OPTION`
|
|
|
|
f.write(textwrap.dedent("""
|
|
|
|
CONFIG_NAMED_OPTION CONFIG_RENAMED_OPTION
|
|
|
|
"""))
|
|
|
|
|
|
|
|
def testRenamedOptionDisabled(self):
|
|
|
|
# Invoke the unit test, specify that the final `sdkconfig` generated must contain the string:
|
|
|
|
# "# CONFIG_RENAMED_OPTION is not set"
|
|
|
|
self.invoke_and_test(self.input, '# CONFIG_RENAMED_OPTION is not set')
|
|
|
|
|
|
|
|
|
2020-02-24 05:57:48 -05:00
|
|
|
class HeaderTestCase(ConfgenBaseTestCase):
|
|
|
|
@classmethod
|
|
|
|
def setUpClass(self):
|
|
|
|
super(HeaderTestCase, self).setUpClass()
|
|
|
|
self.args.update({'output': 'header'})
|
|
|
|
|
|
|
|
def testStringEscape(self):
|
|
|
|
self.invoke_and_test("""
|
|
|
|
config PASSWORD
|
|
|
|
string "password"
|
|
|
|
default "\\\\~!@#$%^&*()\\\""
|
|
|
|
""", '#define CONFIG_PASSWORD "\\\\~!@#$%^&*()\\\""')
|
|
|
|
|
2020-04-15 23:59:16 -04:00
|
|
|
def testHexPrefix(self):
|
|
|
|
self.invoke_and_test(HEXPREFIX_KCONFIG, '#define CONFIG_HEX_NOPREFIX 0x33')
|
|
|
|
self.invoke_and_test(HEXPREFIX_KCONFIG, '#define CONFIG_HEX_PREFIX 0x77')
|
|
|
|
|
2020-02-24 05:57:48 -05:00
|
|
|
|
|
|
|
class DocsTestCase(ConfgenBaseTestCase):
|
|
|
|
@classmethod
|
|
|
|
def setUpClass(self):
|
|
|
|
super(DocsTestCase, self).setUpClass()
|
|
|
|
self.args.update({'output': 'docs',
|
|
|
|
'env': 'IDF_TARGET=esp32'})
|
|
|
|
|
|
|
|
def testChoice(self):
|
|
|
|
self.invoke_and_test("""
|
|
|
|
menu "TEST"
|
|
|
|
choice TYPES
|
|
|
|
prompt "types"
|
|
|
|
default TYPES_OP2
|
|
|
|
help
|
|
|
|
Description of TYPES
|
|
|
|
|
|
|
|
config TYPES_OP1
|
|
|
|
bool "option 1"
|
|
|
|
config TYPES_OP2
|
|
|
|
bool "option 2"
|
|
|
|
endchoice
|
|
|
|
endmenu
|
|
|
|
""", """
|
|
|
|
TEST
|
|
|
|
----
|
|
|
|
|
|
|
|
Contains:
|
|
|
|
|
|
|
|
- :ref:`CONFIG_TYPES`
|
|
|
|
|
|
|
|
.. _CONFIG_TYPES:
|
|
|
|
|
|
|
|
CONFIG_TYPES
|
|
|
|
^^^^^^^^^^^^
|
|
|
|
|
|
|
|
types
|
|
|
|
|
|
|
|
:emphasis:`Found in:` :ref:`test`
|
|
|
|
|
|
|
|
Description of TYPES
|
|
|
|
|
|
|
|
Available options:
|
|
|
|
- option 1 (TYPES_OP1)
|
|
|
|
- option 2 (TYPES_OP2)
|
|
|
|
""") # this is more readable than regex
|
|
|
|
|
|
|
|
|
2020-04-15 23:59:16 -04:00
|
|
|
# Used by multiple testHexPrefix() test cases to verify correct hex output for each format
|
|
|
|
HEXPREFIX_KCONFIG = """
|
|
|
|
config HEX_NOPREFIX
|
|
|
|
hex "Hex Item default no prefix"
|
|
|
|
default 33
|
|
|
|
|
|
|
|
config HEX_PREFIX
|
|
|
|
hex "Hex Item default prefix"
|
|
|
|
default 0x77
|
|
|
|
"""
|
|
|
|
|
2021-01-25 21:49:01 -05:00
|
|
|
if __name__ == '__main__':
|
2020-02-24 05:57:48 -05:00
|
|
|
unittest.main()
|