2018-05-16 06:54:22 +00:00
|
|
|
#!/usr/bin/env python
|
|
|
|
#
|
|
|
|
# Long-running server process uses stdin & stdout to communicate JSON
|
|
|
|
# with a caller
|
|
|
|
#
|
|
|
|
from __future__ import print_function
|
2021-01-26 02:49:01 +00:00
|
|
|
|
2018-05-16 06:54:22 +00:00
|
|
|
import argparse
|
|
|
|
import json
|
|
|
|
import os
|
|
|
|
import sys
|
2019-03-14 08:54:04 +00:00
|
|
|
import tempfile
|
2018-05-16 06:54:22 +00:00
|
|
|
|
2021-01-26 02:49:01 +00:00
|
|
|
import confgen
|
|
|
|
import kconfiglib
|
2020-09-21 09:37:23 +00:00
|
|
|
from confgen import FatalError, __version__
|
2019-09-11 09:18:18 +00:00
|
|
|
|
2019-01-25 04:16:51 +00:00
|
|
|
# Min/Max supported protocol versions
|
|
|
|
MIN_PROTOCOL_VERSION = 1
|
|
|
|
MAX_PROTOCOL_VERSION = 2
|
|
|
|
|
2018-12-04 12:46:48 +00:00
|
|
|
|
2018-05-16 06:54:22 +00:00
|
|
|
def main():
|
|
|
|
parser = argparse.ArgumentParser(description='confserver.py v%s - Config Generation Tool' % __version__, prog=os.path.basename(sys.argv[0]))
|
|
|
|
|
|
|
|
parser.add_argument('--config',
|
2018-12-04 12:46:48 +00:00
|
|
|
help='Project configuration settings',
|
|
|
|
required=True)
|
2018-05-16 06:54:22 +00:00
|
|
|
|
|
|
|
parser.add_argument('--kconfig',
|
|
|
|
help='KConfig file with config item definitions',
|
|
|
|
required=True)
|
|
|
|
|
2019-07-16 14:39:12 +00:00
|
|
|
parser.add_argument('--sdkconfig-rename',
|
|
|
|
help='File with deprecated Kconfig options',
|
|
|
|
required=False)
|
|
|
|
|
2018-05-16 06:54:22 +00:00
|
|
|
parser.add_argument('--env', action='append', default=[],
|
|
|
|
help='Environment to set when evaluating the config file', metavar='NAME=VAL')
|
|
|
|
|
2019-06-27 10:30:42 +00:00
|
|
|
parser.add_argument('--env-file', type=argparse.FileType('r'),
|
|
|
|
help='Optional file to load environment variables from. Contents '
|
|
|
|
'should be a JSON object where each key/value pair is a variable.')
|
|
|
|
|
2019-01-25 04:16:51 +00:00
|
|
|
parser.add_argument('--version', help='Set protocol version to use on initial status',
|
|
|
|
type=int, default=MAX_PROTOCOL_VERSION)
|
|
|
|
|
2018-05-16 06:54:22 +00:00
|
|
|
args = parser.parse_args()
|
|
|
|
|
2019-01-25 04:16:51 +00:00
|
|
|
if args.version < MIN_PROTOCOL_VERSION:
|
2021-01-26 02:49:01 +00:00
|
|
|
print('Version %d is older than minimum supported protocol version %d. Client is much older than ESP-IDF version?' %
|
2019-01-25 04:16:51 +00:00
|
|
|
(args.version, MIN_PROTOCOL_VERSION))
|
|
|
|
|
|
|
|
if args.version > MAX_PROTOCOL_VERSION:
|
2021-01-26 02:49:01 +00:00
|
|
|
print('Version %d is newer than maximum supported protocol version %d. Client is newer than ESP-IDF version?' %
|
2019-01-25 04:16:51 +00:00
|
|
|
(args.version, MAX_PROTOCOL_VERSION))
|
|
|
|
|
2018-05-16 06:54:22 +00:00
|
|
|
try:
|
2021-01-26 02:49:01 +00:00
|
|
|
args.env = [(name,value) for (name,value) in (e.split('=',1) for e in args.env)]
|
2018-05-16 06:54:22 +00:00
|
|
|
except ValueError:
|
2018-12-04 12:46:48 +00:00
|
|
|
print("--env arguments must each contain =. To unset an environment variable, use 'ENV='")
|
|
|
|
sys.exit(1)
|
2018-05-16 06:54:22 +00:00
|
|
|
|
|
|
|
for name, value in args.env:
|
|
|
|
os.environ[name] = value
|
|
|
|
|
2019-06-27 10:30:42 +00:00
|
|
|
if args.env_file is not None:
|
|
|
|
env = json.load(args.env_file)
|
2019-09-10 07:58:52 +00:00
|
|
|
os.environ.update(confgen.dict_enc_for_env(env))
|
2019-06-27 10:30:42 +00:00
|
|
|
|
2019-07-16 14:39:12 +00:00
|
|
|
run_server(args.kconfig, args.config, args.sdkconfig_rename)
|
2018-05-16 06:54:22 +00:00
|
|
|
|
|
|
|
|
2019-07-16 14:39:12 +00:00
|
|
|
def run_server(kconfig, sdkconfig, sdkconfig_rename, default_version=MAX_PROTOCOL_VERSION):
|
2018-05-16 06:54:22 +00:00
|
|
|
config = kconfiglib.Kconfig(kconfig)
|
2019-07-16 14:39:12 +00:00
|
|
|
sdkconfig_renames = [sdkconfig_rename] if sdkconfig_rename else []
|
2021-01-26 02:49:01 +00:00
|
|
|
sdkconfig_renames += os.environ.get('COMPONENT_SDKCONFIG_RENAMES', '').split()
|
2019-07-16 14:39:12 +00:00
|
|
|
deprecated_options = confgen.DeprecatedOptions(config.config_prefix, path_rename_files=sdkconfig_renames)
|
2019-08-16 09:24:20 +00:00
|
|
|
f_o = tempfile.NamedTemporaryFile(mode='w+b', delete=False)
|
|
|
|
try:
|
2019-03-14 08:54:04 +00:00
|
|
|
with open(sdkconfig, mode='rb') as f_i:
|
|
|
|
f_o.write(f_i.read())
|
2019-08-16 09:24:20 +00:00
|
|
|
f_o.close() # need to close as DeprecatedOptions will reopen, and Windows only allows one open file
|
2019-03-14 08:54:04 +00:00
|
|
|
deprecated_options.replace(sdkconfig_in=f_o.name, sdkconfig_out=sdkconfig)
|
2019-08-16 09:24:20 +00:00
|
|
|
finally:
|
|
|
|
os.unlink(f_o.name)
|
2018-05-16 06:54:22 +00:00
|
|
|
config.load_config(sdkconfig)
|
|
|
|
|
2021-01-26 02:49:01 +00:00
|
|
|
print('Server running, waiting for requests on stdin...', file=sys.stderr)
|
2019-01-25 04:16:51 +00:00
|
|
|
|
2018-05-16 06:54:22 +00:00
|
|
|
config_dict = confgen.get_json_values(config)
|
|
|
|
ranges_dict = get_ranges(config)
|
2019-01-25 04:16:51 +00:00
|
|
|
visible_dict = get_visible(config)
|
|
|
|
|
|
|
|
if default_version == 1:
|
|
|
|
# V1: no 'visibility' key, send value None for any invisible item
|
|
|
|
values_dict = dict((k, v if visible_dict[k] else False) for (k,v) in config_dict.items())
|
2021-01-26 02:49:01 +00:00
|
|
|
json.dump({'version': 1, 'values': values_dict, 'ranges': ranges_dict}, sys.stdout)
|
2019-01-25 04:16:51 +00:00
|
|
|
else:
|
|
|
|
# V2 onwards: separate visibility from version
|
2021-01-26 02:49:01 +00:00
|
|
|
json.dump({'version': default_version, 'values': config_dict, 'ranges': ranges_dict, 'visible': visible_dict}, sys.stdout)
|
|
|
|
print('\n')
|
2019-04-23 06:24:24 +00:00
|
|
|
sys.stdout.flush()
|
2018-05-16 06:54:22 +00:00
|
|
|
|
|
|
|
while True:
|
|
|
|
line = sys.stdin.readline()
|
|
|
|
if not line:
|
|
|
|
break
|
2019-03-15 01:17:14 +00:00
|
|
|
try:
|
|
|
|
req = json.loads(line)
|
|
|
|
except ValueError as e: # json module throws JSONDecodeError (sublcass of ValueError) on Py3 but ValueError on Py2
|
2021-01-26 02:49:01 +00:00
|
|
|
response = {'version': default_version, 'error': ['JSON formatting error: %s' % e]}
|
2019-03-15 01:17:14 +00:00
|
|
|
json.dump(response, sys.stdout)
|
2021-01-26 02:49:01 +00:00
|
|
|
print('\n')
|
2019-04-23 06:24:24 +00:00
|
|
|
sys.stdout.flush()
|
2019-03-15 01:17:14 +00:00
|
|
|
continue
|
2018-05-16 06:54:22 +00:00
|
|
|
before = confgen.get_json_values(config)
|
|
|
|
before_ranges = get_ranges(config)
|
2019-01-25 04:16:51 +00:00
|
|
|
before_visible = get_visible(config)
|
2018-05-16 06:54:22 +00:00
|
|
|
|
2021-01-26 02:49:01 +00:00
|
|
|
if 'load' in req: # load a new sdkconfig
|
2019-03-14 06:47:08 +00:00
|
|
|
|
2021-01-26 02:49:01 +00:00
|
|
|
if req.get('version', default_version) == 1:
|
2019-03-14 06:47:08 +00:00
|
|
|
# for V1 protocol, send all items when loading new sdkconfig.
|
|
|
|
# (V2+ will only send changes, same as when setting an item)
|
|
|
|
before = {}
|
|
|
|
before_ranges = {}
|
|
|
|
before_visible = {}
|
2018-05-16 06:54:22 +00:00
|
|
|
|
|
|
|
# if no new filename is supplied, use existing sdkconfig path, otherwise update the path
|
2021-01-26 02:49:01 +00:00
|
|
|
if req['load'] is None:
|
|
|
|
req['load'] = sdkconfig
|
2018-05-16 06:54:22 +00:00
|
|
|
else:
|
2021-01-26 02:49:01 +00:00
|
|
|
sdkconfig = req['load']
|
2018-05-16 06:54:22 +00:00
|
|
|
|
2021-01-26 02:49:01 +00:00
|
|
|
if 'save' in req:
|
|
|
|
if req['save'] is None:
|
|
|
|
req['save'] = sdkconfig
|
2018-05-16 06:54:22 +00:00
|
|
|
else:
|
2021-01-26 02:49:01 +00:00
|
|
|
sdkconfig = req['save']
|
2018-05-16 06:54:22 +00:00
|
|
|
|
2019-03-14 08:54:04 +00:00
|
|
|
error = handle_request(deprecated_options, config, req)
|
2018-05-16 06:54:22 +00:00
|
|
|
|
|
|
|
after = confgen.get_json_values(config)
|
|
|
|
after_ranges = get_ranges(config)
|
2019-01-25 04:16:51 +00:00
|
|
|
after_visible = get_visible(config)
|
2018-05-16 06:54:22 +00:00
|
|
|
|
|
|
|
values_diff = diff(before, after)
|
|
|
|
ranges_diff = diff(before_ranges, after_ranges)
|
2019-01-25 04:16:51 +00:00
|
|
|
visible_diff = diff(before_visible, after_visible)
|
2021-01-26 02:49:01 +00:00
|
|
|
if req['version'] == 1:
|
2019-01-25 04:16:51 +00:00
|
|
|
# V1 response, invisible items have value None
|
|
|
|
for k in (k for (k,v) in visible_diff.items() if not v):
|
|
|
|
values_diff[k] = None
|
2021-01-26 02:49:01 +00:00
|
|
|
response = {'version': 1, 'values': values_diff, 'ranges': ranges_diff}
|
2019-01-25 04:16:51 +00:00
|
|
|
else:
|
|
|
|
# V2+ response, separate visibility values
|
2021-01-26 02:49:01 +00:00
|
|
|
response = {'version': req['version'], 'values': values_diff, 'ranges': ranges_diff, 'visible': visible_diff}
|
2018-05-16 06:54:22 +00:00
|
|
|
if error:
|
|
|
|
for e in error:
|
2021-01-26 02:49:01 +00:00
|
|
|
print('Error: %s' % e, file=sys.stderr)
|
|
|
|
response['error'] = error
|
2018-05-16 06:54:22 +00:00
|
|
|
json.dump(response, sys.stdout)
|
2021-01-26 02:49:01 +00:00
|
|
|
print('\n')
|
2019-04-23 06:24:24 +00:00
|
|
|
sys.stdout.flush()
|
2018-05-16 06:54:22 +00:00
|
|
|
|
|
|
|
|
2019-03-14 08:54:04 +00:00
|
|
|
def handle_request(deprecated_options, config, req):
|
2021-01-26 02:49:01 +00:00
|
|
|
if 'version' not in req:
|
2018-12-04 12:46:48 +00:00
|
|
|
return ["All requests must have a 'version'"]
|
2019-01-25 04:16:51 +00:00
|
|
|
|
2021-01-26 02:49:01 +00:00
|
|
|
if req['version'] < MIN_PROTOCOL_VERSION or req['version'] > MAX_PROTOCOL_VERSION:
|
|
|
|
return ['Unsupported request version %d. Server supports versions %d-%d' % (
|
|
|
|
req['version'],
|
2019-01-25 04:16:51 +00:00
|
|
|
MIN_PROTOCOL_VERSION,
|
|
|
|
MAX_PROTOCOL_VERSION)]
|
2018-05-16 06:54:22 +00:00
|
|
|
|
|
|
|
error = []
|
|
|
|
|
2021-01-26 02:49:01 +00:00
|
|
|
if 'load' in req:
|
|
|
|
print('Loading config from %s...' % req['load'], file=sys.stderr)
|
2018-05-16 06:54:22 +00:00
|
|
|
try:
|
2021-01-26 02:49:01 +00:00
|
|
|
config.load_config(req['load'])
|
2018-05-16 06:54:22 +00:00
|
|
|
except Exception as e:
|
2021-01-26 02:49:01 +00:00
|
|
|
error += ['Failed to load from %s: %s' % (req['load'], e)]
|
2018-05-16 06:54:22 +00:00
|
|
|
|
2021-01-26 02:49:01 +00:00
|
|
|
if 'set' in req:
|
|
|
|
handle_set(config, error, req['set'])
|
2018-05-16 06:54:22 +00:00
|
|
|
|
2021-01-26 02:49:01 +00:00
|
|
|
if 'save' in req:
|
2018-05-16 06:54:22 +00:00
|
|
|
try:
|
2021-01-26 02:49:01 +00:00
|
|
|
print('Saving config to %s...' % req['save'], file=sys.stderr)
|
|
|
|
confgen.write_config(deprecated_options, config, req['save'])
|
2018-05-16 06:54:22 +00:00
|
|
|
except Exception as e:
|
2021-01-26 02:49:01 +00:00
|
|
|
error += ['Failed to save to %s: %s' % (req['save'], e)]
|
2018-05-16 06:54:22 +00:00
|
|
|
|
|
|
|
return error
|
|
|
|
|
2018-12-04 12:46:48 +00:00
|
|
|
|
2018-05-16 06:54:22 +00:00
|
|
|
def handle_set(config, error, to_set):
|
2018-12-04 12:46:48 +00:00
|
|
|
missing = [k for k in to_set if k not in config.syms]
|
2018-05-16 06:54:22 +00:00
|
|
|
if missing:
|
2021-01-26 02:49:01 +00:00
|
|
|
error.append('The following config symbol(s) were not found: %s' % (', '.join(missing)))
|
2018-05-16 06:54:22 +00:00
|
|
|
# replace name keys with the full config symbol for each key:
|
2018-12-04 12:46:48 +00:00
|
|
|
to_set = dict((config.syms[k],v) for (k,v) in to_set.items() if k not in missing)
|
2018-05-16 06:54:22 +00:00
|
|
|
|
|
|
|
# Work through the list of values to set, noting that
|
|
|
|
# some may not be immediately applicable (maybe they depend
|
|
|
|
# on another value which is being set). Therefore, defer
|
|
|
|
# knowing if any value is unsettable until then end
|
|
|
|
|
|
|
|
while len(to_set):
|
2018-12-04 12:46:48 +00:00
|
|
|
set_pass = [(k,v) for (k,v) in to_set.items() if k.visibility]
|
2018-05-16 06:54:22 +00:00
|
|
|
if not set_pass:
|
|
|
|
break # no visible keys left
|
|
|
|
for (sym,val) in set_pass:
|
|
|
|
if sym.type in (kconfiglib.BOOL, kconfiglib.TRISTATE):
|
2018-12-04 12:46:48 +00:00
|
|
|
if val is True:
|
2018-05-16 06:54:22 +00:00
|
|
|
sym.set_value(2)
|
2018-12-04 12:46:48 +00:00
|
|
|
elif val is False:
|
2018-05-16 06:54:22 +00:00
|
|
|
sym.set_value(0)
|
|
|
|
else:
|
2021-01-26 02:49:01 +00:00
|
|
|
error.append('Boolean symbol %s only accepts true/false values' % sym.name)
|
2020-01-22 06:43:40 +00:00
|
|
|
elif sym.type == kconfiglib.HEX:
|
|
|
|
try:
|
|
|
|
if not isinstance(val, int):
|
|
|
|
val = int(val, 16) # input can be a decimal JSON value or a string of hex digits
|
2020-04-16 04:04:54 +00:00
|
|
|
sym.set_value(hex(val))
|
2020-01-22 06:43:40 +00:00
|
|
|
except ValueError:
|
2021-01-26 02:49:01 +00:00
|
|
|
error.append('Hex symbol %s can accept a decimal integer or a string of hex digits, only')
|
2018-05-16 06:54:22 +00:00
|
|
|
else:
|
|
|
|
sym.set_value(str(val))
|
2021-01-26 02:49:01 +00:00
|
|
|
print('Set %s' % sym.name)
|
2018-05-16 06:54:22 +00:00
|
|
|
del to_set[sym]
|
|
|
|
|
|
|
|
if len(to_set):
|
2021-01-26 02:49:01 +00:00
|
|
|
error.append('The following config symbol(s) were not visible so were not updated: %s' % (', '.join(s.name for s in to_set)))
|
2018-05-16 06:54:22 +00:00
|
|
|
|
|
|
|
|
|
|
|
def diff(before, after):
|
|
|
|
"""
|
2019-01-25 04:16:51 +00:00
|
|
|
Return a dictionary with the difference between 'before' and 'after',
|
|
|
|
for items which are present in 'after' dictionary
|
2018-05-16 06:54:22 +00:00
|
|
|
"""
|
|
|
|
diff = dict((k,v) for (k,v) in after.items() if before.get(k, None) != v)
|
|
|
|
return diff
|
|
|
|
|
|
|
|
|
|
|
|
def get_ranges(config):
|
|
|
|
ranges_dict = {}
|
2018-12-04 12:46:48 +00:00
|
|
|
|
2019-09-10 07:58:52 +00:00
|
|
|
def is_base_n(i, n):
|
|
|
|
try:
|
|
|
|
int(i, n)
|
|
|
|
return True
|
|
|
|
except ValueError:
|
|
|
|
return False
|
|
|
|
|
|
|
|
def get_active_range(sym):
|
|
|
|
"""
|
|
|
|
Returns a tuple of (low, high) integer values if a range
|
|
|
|
limit is active for this symbol, or (None, None) if no range
|
|
|
|
limit exists.
|
|
|
|
"""
|
|
|
|
base = kconfiglib._TYPE_TO_BASE[sym.orig_type] if sym.orig_type in kconfiglib._TYPE_TO_BASE else 0
|
|
|
|
|
|
|
|
try:
|
|
|
|
for low_expr, high_expr, cond in sym.ranges:
|
|
|
|
if kconfiglib.expr_value(cond):
|
|
|
|
low = int(low_expr.str_value, base) if is_base_n(low_expr.str_value, base) else 0
|
|
|
|
high = int(high_expr.str_value, base) if is_base_n(high_expr.str_value, base) else 0
|
|
|
|
return (low, high)
|
|
|
|
except ValueError:
|
|
|
|
pass
|
|
|
|
return (None, None)
|
|
|
|
|
2018-05-16 06:54:22 +00:00
|
|
|
def handle_node(node):
|
|
|
|
sym = node.item
|
|
|
|
if not isinstance(sym, kconfiglib.Symbol):
|
|
|
|
return
|
2019-09-10 07:58:52 +00:00
|
|
|
active_range = get_active_range(sym)
|
2018-05-16 06:54:22 +00:00
|
|
|
if active_range[0] is not None:
|
|
|
|
ranges_dict[sym.name] = active_range
|
|
|
|
|
2019-09-10 07:58:52 +00:00
|
|
|
for n in config.node_iter():
|
|
|
|
handle_node(n)
|
2018-05-16 06:54:22 +00:00
|
|
|
return ranges_dict
|
|
|
|
|
|
|
|
|
2019-01-25 04:16:51 +00:00
|
|
|
def get_visible(config):
|
|
|
|
"""
|
|
|
|
Return a dict mapping node IDs (config names or menu node IDs) to True/False for their visibility
|
|
|
|
"""
|
|
|
|
result = {}
|
|
|
|
menus = []
|
|
|
|
|
|
|
|
# when walking the menu the first time, only
|
|
|
|
# record whether the config symbols are visible
|
|
|
|
# and make a list of menu nodes (that are not symbols)
|
|
|
|
def handle_node(node):
|
|
|
|
sym = node.item
|
|
|
|
try:
|
|
|
|
visible = (sym.visibility != 0)
|
|
|
|
result[node] = visible
|
|
|
|
except AttributeError:
|
|
|
|
menus.append(node)
|
2019-09-10 07:58:52 +00:00
|
|
|
for n in config.node_iter():
|
|
|
|
handle_node(n)
|
2019-01-25 04:16:51 +00:00
|
|
|
|
|
|
|
# now, figure out visibility for each menu. A menu is visible if any of its children are visible
|
|
|
|
for m in reversed(menus): # reverse to start at leaf nodes
|
|
|
|
result[m] = any(v for (n,v) in result.items() if n.parent == m)
|
|
|
|
|
|
|
|
# return a dict mapping the node ID to its visibility.
|
|
|
|
result = dict((confgen.get_menu_node_id(n),v) for (n,v) in result.items())
|
|
|
|
|
|
|
|
return result
|
|
|
|
|
|
|
|
|
2018-05-16 06:54:22 +00:00
|
|
|
if __name__ == '__main__':
|
|
|
|
try:
|
|
|
|
main()
|
|
|
|
except FatalError as e:
|
2021-01-26 02:49:01 +00:00
|
|
|
print('A fatal error occurred: %s' % e, file=sys.stderr)
|
2018-05-16 06:54:22 +00:00
|
|
|
sys.exit(2)
|