From 8ff923b42eadebafacc4a2d0b09a9e7e3c105072 Mon Sep 17 00:00:00 2001 From: ZyX Date: Wed, 17 Sep 2014 08:38:53 +0400 Subject: [PATCH 01/12] Remove direct calls to `sys.stderr.write` from powerline.lint.check --- powerline/lint/__init__.py | 24 ++++++++++++------------ 1 file changed, 12 insertions(+), 12 deletions(-) diff --git a/powerline/lint/__init__.py b/powerline/lint/__init__.py index 6a5d86d5..823927cb 100644 --- a/powerline/lint/__init__.py +++ b/powerline/lint/__init__.py @@ -1465,7 +1465,7 @@ def generate_json_config_loader(lhadproblem): return load_json_config -def check(paths=None, debug=False): +def check(paths=None, debug=False, echoerr=echoerr): search_paths = paths or get_config_paths() find_config_files = generate_config_finder(lambda: search_paths) @@ -1507,7 +1507,7 @@ def check(paths=None, debug=False): paths['top_' + typ].append(extpath) else: hadproblem = True - sys.stderr.write('Path {0} is supposed to be a directory, but it is not\n'.format(d)) + ee(problem='Path {0} is supposed to be a directory, but it is not'.format(d)) hadproblem = False @@ -1522,7 +1522,7 @@ def check(paths=None, debug=False): lists[typ].add(name) if name.startswith('__') or name.endswith('__'): hadproblem = True - sys.stderr.write('File name is not supposed to start or end with “__”: {0}'.format( + ee(problem='File name is not supposed to start or end with “__”: {0}'.format( os.path.join(d, subp) )) configs[typ][ext][name] = os.path.join(d, subp) @@ -1536,7 +1536,7 @@ def check(paths=None, debug=False): for ext in diff: typ = 'colorschemes' if ext in configs['themes'] else 'themes' if not configs['top_' + typ] or typ == 'themes': - sys.stderr.write('{0} extension {1} not present in {2}\n'.format( + ee(problem='{0} extension {1} not present in {2}'.format( ext, 'configuration' if (ext in paths['themes'] and ext in paths['colorschemes']) else 'directory', typ, @@ -1546,11 +1546,11 @@ def check(paths=None, debug=False): main_config = load_config('config', find_config_files, config_loader) except IOError: main_config = {} - sys.stderr.write('\nConfiguration file not found: config.json\n') + ee(problem='Configuration file not found: config.json') hadproblem = True except MarkedError as e: main_config = {} - sys.stderr.write(str(e) + '\n') + ee(problem=str(e)) hadproblem = True else: if main_spec.match( @@ -1567,11 +1567,11 @@ def check(paths=None, debug=False): colors_config = load_config('colors', find_config_files, config_loader) except IOError: colors_config = {} - sys.stderr.write('\nConfiguration file not found: colors.json\n') + ee(problem='Configuration file not found: colors.json') hadproblem = True except MarkedError as e: colors_config = {} - sys.stderr.write(str(e) + '\n') + ee(problem=str(e)) hadproblem = True else: if colors_spec.match(colors_config, context=init_context(colors_config), echoerr=ee)[1]: @@ -1592,7 +1592,7 @@ def check(paths=None, debug=False): try: config, lhadproblem = load(config_file_fp) except MarkedError as e: - sys.stderr.write(str(e) + '\n') + ee(problem=str(e)) hadproblem = True continue if lhadproblem: @@ -1609,7 +1609,7 @@ def check(paths=None, debug=False): try: config, lhadproblem = load(config_file_fp) except MarkedError as e: - sys.stderr.write(str(e) + '\n') + ee(problem=str(e)) hadproblem = True continue if lhadproblem: @@ -1662,7 +1662,7 @@ def check(paths=None, debug=False): try: config, lhadproblem = load(config_file_fp) except MarkedError as e: - sys.stderr.write(str(e) + '\n') + ee(problem=str(e)) hadproblem = True continue if lhadproblem: @@ -1675,7 +1675,7 @@ def check(paths=None, debug=False): try: config, lhadproblem = load(config_file_fp) except MarkedError as e: - sys.stderr.write(str(e) + '\n') + ee(problem=str(e)) hadproblem = True continue if lhadproblem: From afb897ec4d530af0460909f5b2d9ddf593c52160 Mon Sep 17 00:00:00 2001 From: ZyX Date: Fri, 19 Sep 2014 21:23:04 +0400 Subject: [PATCH 02/12] Run powerline-lint from powerline_troubleshoot function --- powerline/bindings/vim/plugin/powerline.vim | 17 ++++++++++++++- powerline/lint/__init__.py | 23 +++++++++++++++++++-- powerline/lint/markedjson/error.py | 5 +++-- 3 files changed, 40 insertions(+), 5 deletions(-) diff --git a/powerline/bindings/vim/plugin/powerline.vim b/powerline/bindings/vim/plugin/powerline.vim index 47a24b16..1871cc38 100644 --- a/powerline/bindings/vim/plugin/powerline.vim +++ b/powerline/bindings/vim/plugin/powerline.vim @@ -71,8 +71,10 @@ try call s:rcmd(" sys.path.append(powerline_appended_path)") call s:rcmd(" ".s:import_cmd."") call s:rcmd(" import vim") - call s:rcmd(" VimPowerline().setup(pyeval=vim.eval('s:pyeval'), pycmd=vim.eval('s:pycmd'), can_replace_pyeval=int(vim.eval('s:can_replace_pyeval')))") + call s:rcmd(" powerline_instance = VimPowerline()") + call s:rcmd(" powerline_instance.setup(pyeval=vim.eval('s:pyeval'), pycmd=vim.eval('s:pycmd'), can_replace_pyeval=int(vim.eval('s:can_replace_pyeval')))") call s:rcmd(" del VimPowerline") + call s:rcmd(" del powerline_instance") call s:rcmd("except Exception:") call s:rcmd(" import traceback, sys") call s:rcmd(" traceback.print_exc(file=sys.stdout)") @@ -135,6 +137,19 @@ finally call s:rcmd(" print('one in ' + real_powerline_dir + ', other in ' + real_this_dir + '.')") call s:rcmd(" print('You should remove one of this. Check out troubleshooting section,')") call s:rcmd(" print('it contains some information about the alternatives.')") + call s:rcmd(" try:") + call s:rcmd(" from powerline.lint import check") + call s:rcmd(" except ImportError:") + call s:rcmd(" print('Failed to import powerline.lint.check, cannot run powerline-lint')") + call s:rcmd(" else:") + call s:rcmd(" try:") + call s:rcmd(" paths = powerline_instance.get_config_paths()") + call s:rcmd(" except NameError:") + call s:rcmd(" pass") + call s:rcmd(" else:") + call s:rcmd(" from powerline.lint.markedjson.error import echoerr") + call s:rcmd(" ee = lambda *args, **kwargs: echoerr(*args, stream=sys.stdout, **kwargs)") + call s:rcmd(" check(paths=paths, echoerr=ee, require_ext='vim')") call s:rcmd("try:") call s:rcmd(" powerline_troubleshoot()") call s:rcmd("finally:") diff --git a/powerline/lint/__init__.py b/powerline/lint/__init__.py index 823927cb..ab7a055e 100644 --- a/powerline/lint/__init__.py +++ b/powerline/lint/__init__.py @@ -383,6 +383,10 @@ class Spec(object): self.isoptional = True return self + def required(self): + self.isoptional = False + return self + def match_checks(self, *args): hadproblem = False for check in self.checks: @@ -451,6 +455,12 @@ class Spec(object): problem_mark=key.mark) return True, hadproblem + def __getitem__(self, key): + return self.specs[self.keys[key]] + + def __setitem__(self, key, value): + return self.update(**{key : value}) + class WithPath(object): def __init__(self, import_paths): @@ -1465,7 +1475,7 @@ def generate_json_config_loader(lhadproblem): return load_json_config -def check(paths=None, debug=False, echoerr=echoerr): +def check(paths=None, debug=False, echoerr=echoerr, require_ext=None): search_paths = paths or get_config_paths() find_config_files = generate_config_finder(lambda: search_paths) @@ -1475,6 +1485,15 @@ def check(paths=None, debug=False, echoerr=echoerr): ee = EchoErr(echoerr, logger) + if require_ext: + used_main_spec = main_spec.copy() + try: + used_main_spec['ext'][require_ext].required() + except KeyError: + used_main_spec['ext'][require_ext] = ext_spec() + else: + used_main_spec = main_spec + lhadproblem = [False] load_json_config = generate_json_config_loader(lhadproblem) @@ -1553,7 +1572,7 @@ def check(paths=None, debug=False, echoerr=echoerr): ee(problem=str(e)) hadproblem = True else: - if main_spec.match( + if used_main_spec.match( main_config, data={'configs': configs, 'lists': lists}, context=init_context(main_config), diff --git a/powerline/lint/markedjson/error.py b/powerline/lint/markedjson/error.py index 643c86b3..6494d10f 100644 --- a/powerline/lint/markedjson/error.py +++ b/powerline/lint/markedjson/error.py @@ -68,8 +68,9 @@ class Mark: def echoerr(*args, **kwargs): - sys.stderr.write('\n') - sys.stderr.write(format_error(*args, **kwargs) + '\n') + stream = kwargs.pop('stream', sys.stderr) + stream.write('\n') + stream.write(format_error(*args, **kwargs) + '\n') def format_error(context=None, context_mark=None, problem=None, problem_mark=None, note=None): From 346c7ab41e73c860da003f995e2767ec86d82699 Mon Sep 17 00:00:00 2001 From: ZyX Date: Fri, 19 Sep 2014 21:52:07 +0400 Subject: [PATCH 03/12] Split huge powerline.lint module into multiple files --- powerline/lint/__init__.py | 1208 +--------------------------- powerline/lint/checks.py | 707 ++++++++++++++++ powerline/lint/context.py | 44 + powerline/lint/imp.py | 52 ++ powerline/lint/markedjson/error.py | 31 + powerline/lint/selfcheck.py | 22 + powerline/lint/spec.py | 375 +++++++++ 7 files changed, 1242 insertions(+), 1197 deletions(-) create mode 100644 powerline/lint/checks.py create mode 100644 powerline/lint/context.py create mode 100644 powerline/lint/imp.py create mode 100644 powerline/lint/selfcheck.py create mode 100644 powerline/lint/spec.py diff --git a/powerline/lint/__init__.py b/powerline/lint/__init__.py index ab7a055e..7eb8dc71 100644 --- a/powerline/lint/__init__.py +++ b/powerline/lint/__init__.py @@ -1,577 +1,33 @@ # vim:fileencoding=utf-8:noet from __future__ import (unicode_literals, division, absolute_import, print_function) -import itertools -import sys import os -import re import logging from collections import defaultdict -from copy import copy from functools import partial -from powerline.lint.markedjson import load from powerline import generate_config_finder, get_config_paths, load_config -from powerline.lib.config import ConfigLoader -from powerline.lint.markedjson.error import echoerr, MarkedError, Mark -from powerline.lint.markedjson.markedvalue import MarkedUnicode from powerline.segments.vim import vim_modes -from powerline.lint.inspect import getconfigargspec -from powerline.lib.threaded import ThreadedSegment from powerline.lib import mergedicts_copy +from powerline.lib.config import ConfigLoader from powerline.lib.unicode import unicode +from powerline.lint.markedjson import load +from powerline.lint.markedjson.error import echoerr, EchoErr, MarkedError +from powerline.lint.checks import (check_matcher_func, check_ext, check_config, check_top_theme, + check_color, check_translated_group_name, check_group, + check_segment_module, check_exinclude_function, type_keys, + check_segment_function, check_args, get_one_segment_function, + check_highlight_groups, check_highlight_group, check_full_segment_data, + get_all_possible_functions, check_segment_data_key) +from powerline.lint.spec import Spec +from powerline.lint.context import init_context def open_file(path): return open(path, 'rb') -EMPTYTUPLE = tuple() - - -class JStr(unicode): - def join(self, iterable): - return super(JStr, self).join((unicode(item) for item in iterable)) - - -key_sep = JStr('/') -list_sep = JStr(', ') - - -def init_context(config): - return ((MarkedUnicode('', config.mark), config),) - - -def context_key(context): - return key_sep.join((c[0] for c in context)) - - -def havemarks(*args, **kwargs): - origin = kwargs.get('origin', '') - for i, v in enumerate(args): - if not hasattr(v, 'mark'): - raise AssertionError('Value #{0}/{1} ({2!r}) has no attribute `mark`'.format(origin, i, v)) - if isinstance(v, dict): - for key, val in v.items(): - havemarks(key, val, origin=(origin + '[' + unicode(i) + ']/' + unicode(key))) - elif isinstance(v, list): - havemarks(*v, origin=(origin + '[' + unicode(i) + ']')) - - -def context_has_marks(context): - for i, v in enumerate(context): - havemarks(v[0], origin='context key') - havemarks(v[1], origin='context val') - - -class EchoErr(object): - __slots__ = ('echoerr', 'logger',) - - def __init__(self, echoerr, logger): - self.echoerr = echoerr - self.logger = logger - - def __call__(self, *args, **kwargs): - self.echoerr(*args, **kwargs) - - -class DelayedEchoErr(EchoErr): - __slots__ = ('echoerr', 'logger', 'errs') - - def __init__(self, echoerr): - super(DelayedEchoErr, self).__init__(echoerr, echoerr.logger) - self.errs = [] - - def __call__(self, *args, **kwargs): - self.errs.append((args, kwargs)) - - def echo_all(self): - for args, kwargs in self.errs: - self.echoerr(*args, **kwargs) - - def __nonzero__(self): - return not not self.errs - - __bool__ = __nonzero__ - - -def new_context_item(key, value): - return ((value.keydict[key], value[key]),) - - -class Spec(object): - def __init__(self, **keys): - self.specs = [] - self.keys = {} - self.checks = [] - self.cmsg = '' - self.isoptional = False - self.uspecs = [] - self.ufailmsg = lambda key: 'found unknown key: {0}'.format(key) - self.did_type = False - self.update(**keys) - - def update(self, **keys): - for k, v in keys.items(): - self.keys[k] = len(self.specs) - self.specs.append(v) - if self.keys and not self.did_type: - self.type(dict) - self.did_type = True - return self - - def copy(self, copied=None): - copied = copied or {} - try: - return copied[id(self)] - except KeyError: - instance = self.__class__() - copied[id(self)] = instance - return self.__class__()._update(self.__dict__, copied) - - def _update(self, d, copied): - self.__dict__.update(d) - self.keys = copy(self.keys) - self.checks = copy(self.checks) - self.uspecs = copy(self.uspecs) - self.specs = [spec.copy(copied) for spec in self.specs] - return self - - def unknown_spec(self, keyfunc, spec): - if isinstance(keyfunc, Spec): - self.specs.append(keyfunc) - keyfunc = len(self.specs) - 1 - self.specs.append(spec) - self.uspecs.append((keyfunc, len(self.specs) - 1)) - return self - - def unknown_msg(self, msgfunc): - self.ufailmsg = msgfunc - return self - - def context_message(self, msg): - self.cmsg = msg - for spec in self.specs: - if not spec.cmsg: - spec.context_message(msg) - return self - - def check_type(self, value, context_mark, data, context, echoerr, types): - havemarks(value) - if type(value.value) not in types: - echoerr( - context=self.cmsg.format(key=context_key(context)), - context_mark=context_mark, - problem='{0!r} must be a {1} instance, not {2}'.format( - value, - list_sep.join((t.__name__ for t in types)), - type(value.value).__name__ - ), - problem_mark=value.mark - ) - return False, True - return True, False - - def check_func(self, value, context_mark, data, context, echoerr, func, msg_func): - havemarks(value) - proceed, echo, hadproblem = func(value, data, context, echoerr) - if echo and hadproblem: - echoerr(context=self.cmsg.format(key=context_key(context)), - context_mark=context_mark, - problem=msg_func(value), - problem_mark=value.mark) - return proceed, hadproblem - - def check_list(self, value, context_mark, data, context, echoerr, item_func, msg_func): - havemarks(value) - i = 0 - hadproblem = False - for item in value: - havemarks(item) - if isinstance(item_func, int): - spec = self.specs[item_func] - proceed, fhadproblem = spec.match( - item, - value.mark, - data, - context + ((MarkedUnicode('list item ' + unicode(i), item.mark), item),), - echoerr - ) - else: - proceed, echo, fhadproblem = item_func(item, data, context, echoerr) - if echo and fhadproblem: - echoerr(context=self.cmsg.format(key=context_key(context) + '/list item ' + unicode(i)), - context_mark=value.mark, - problem=msg_func(item), - problem_mark=item.mark) - if fhadproblem: - hadproblem = True - if not proceed: - return proceed, hadproblem - i += 1 - return True, hadproblem - - def check_either(self, value, context_mark, data, context, echoerr, start, end): - havemarks(value) - new_echoerr = DelayedEchoErr(echoerr) - - hadproblem = False - for spec in self.specs[start:end]: - proceed, hadproblem = spec.match(value, value.mark, data, context, new_echoerr) - if not proceed: - break - if not hadproblem: - return True, False - - new_echoerr.echo_all() - - return False, hadproblem - - def check_tuple(self, value, context_mark, data, context, echoerr, start, end): - havemarks(value) - hadproblem = False - for (i, item, spec) in zip(itertools.count(), value, self.specs[start:end]): - proceed, ihadproblem = spec.match( - item, - value.mark, - data, - context + ((MarkedUnicode('tuple item ' + unicode(i), item.mark), item),), - echoerr - ) - if ihadproblem: - hadproblem = True - if not proceed: - return False, hadproblem - return True, hadproblem - - def type(self, *args): - self.checks.append(('check_type', args)) - return self - - cmp_funcs = { - 'le': lambda x, y: x <= y, - 'lt': lambda x, y: x < y, - 'ge': lambda x, y: x >= y, - 'gt': lambda x, y: x > y, - 'eq': lambda x, y: x == y, - } - - cmp_msgs = { - 'le': 'lesser or equal to', - 'lt': 'lesser then', - 'ge': 'greater or equal to', - 'gt': 'greater then', - 'eq': 'equal to', - } - - def len(self, comparison, cint, msg_func=None): - cmp_func = self.cmp_funcs[comparison] - msg_func = ( - msg_func - or (lambda value: 'length of {0!r} is not {1} {2}'.format( - value, self.cmp_msgs[comparison], cint)) - ) - self.checks.append(( - 'check_func', - (lambda value, *args: (True, True, not cmp_func(len(value), cint))), - msg_func - )) - return self - - def cmp(self, comparison, cint, msg_func=None): - if type(cint) is str: - self.type(unicode) - elif type(cint) is float: - self.type(int, float) - else: - self.type(type(cint)) - cmp_func = self.cmp_funcs[comparison] - msg_func = msg_func or (lambda value: '{0} is not {1} {2}'.format(value, self.cmp_msgs[comparison], cint)) - self.checks.append(( - 'check_func', - (lambda value, *args: (True, True, not cmp_func(value.value, cint))), - msg_func - )) - return self - - def unsigned(self, msg_func=None): - self.type(int) - self.checks.append(( - 'check_func', - (lambda value, *args: (True, True, value < 0)), - (lambda value: '{0} must be greater then zero'.format(value)) - )) - return self - - def list(self, item_func, msg_func=None): - self.type(list) - if isinstance(item_func, Spec): - self.specs.append(item_func) - item_func = len(self.specs) - 1 - self.checks.append(('check_list', item_func, msg_func or (lambda item: 'failed check'))) - return self - - def tuple(self, *specs): - self.type(list) - - max_len = len(specs) - min_len = max_len - for spec in reversed(specs): - if spec.isoptional: - min_len -= 1 - else: - break - if max_len == min_len: - self.len('eq', len(specs)) - else: - self.len('ge', min_len) - self.len('le', max_len) - - start = len(self.specs) - for i, spec in zip(itertools.count(), specs): - self.specs.append(spec) - self.checks.append(('check_tuple', start, len(self.specs))) - return self - - def func(self, func, msg_func=None): - self.checks.append(('check_func', func, msg_func or (lambda value: 'failed check'))) - return self - - def re(self, regex, msg_func=None): - self.type(unicode) - compiled = re.compile(regex) - msg_func = msg_func or (lambda value: 'String "{0}" does not match "{1}"'.format(value, regex)) - self.checks.append(( - 'check_func', - (lambda value, *args: (True, True, not compiled.match(value.value))), - msg_func - )) - return self - - def ident(self, msg_func=None): - msg_func = ( - msg_func - or (lambda value: 'String "{0}" is not an alphanumeric/underscore colon-separated identifier'.format(value)) - ) - return self.re('^\w+(?::\w+)?$', msg_func) - - def oneof(self, collection, msg_func=None): - msg_func = msg_func or (lambda value: '"{0}" must be one of {1!r}'.format(value, list(collection))) - self.checks.append(( - 'check_func', - (lambda value, *args: (True, True, value not in collection)), - msg_func - )) - return self - - def error(self, msg): - self.checks.append(( - 'check_func', - (lambda *args: (True, True, True)), - (lambda value: msg.format(value)) - )) - return self - - def either(self, *specs): - start = len(self.specs) - self.specs.extend(specs) - self.checks.append(('check_either', start, len(self.specs))) - return self - - def optional(self): - self.isoptional = True - return self - - def required(self): - self.isoptional = False - return self - - def match_checks(self, *args): - hadproblem = False - for check in self.checks: - proceed, chadproblem = getattr(self, check[0])(*(args + check[1:])) - if chadproblem: - hadproblem = True - if not proceed: - return False, hadproblem - return True, hadproblem - - def match(self, value, context_mark=None, data=None, context=EMPTYTUPLE, echoerr=echoerr): - havemarks(value) - proceed, hadproblem = self.match_checks(value, context_mark, data, context, echoerr) - if proceed: - if self.keys or self.uspecs: - for key, vali in self.keys.items(): - valspec = self.specs[vali] - if key in value: - proceed, mhadproblem = valspec.match( - value[key], - value.mark, - data, - context + new_context_item(key, value), - echoerr - ) - if mhadproblem: - hadproblem = True - if not proceed: - return False, hadproblem - else: - if not valspec.isoptional: - hadproblem = True - echoerr(context=self.cmsg.format(key=context_key(context)), - context_mark=None, - problem='required key is missing: {0}'.format(key), - problem_mark=value.mark) - for key in value.keys(): - havemarks(key) - if key not in self.keys: - for keyfunc, vali in self.uspecs: - valspec = self.specs[vali] - if isinstance(keyfunc, int): - spec = self.specs[keyfunc] - proceed, khadproblem = spec.match(key, context_mark, data, context, echoerr) - else: - proceed, khadproblem = keyfunc(key, data, context, echoerr) - if khadproblem: - hadproblem = True - if proceed: - proceed, vhadproblem = valspec.match( - value[key], - value.mark, - data, - context + new_context_item(key, value), - echoerr - ) - if vhadproblem: - hadproblem = True - break - else: - hadproblem = True - if self.ufailmsg: - echoerr(context=self.cmsg.format(key=context_key(context)), - context_mark=None, - problem=self.ufailmsg(key), - problem_mark=key.mark) - return True, hadproblem - - def __getitem__(self, key): - return self.specs[self.keys[key]] - - def __setitem__(self, key, value): - return self.update(**{key : value}) - - -class WithPath(object): - def __init__(self, import_paths): - self.import_paths = import_paths - - def __enter__(self): - self.oldpath = sys.path - sys.path = self.import_paths + sys.path - - def __exit__(self, *args): - sys.path = self.oldpath - - -def check_matcher_func(ext, match_name, data, context, echoerr): - havemarks(match_name) - import_paths = [os.path.expanduser(path) for path in context[0][1].get('common', {}).get('paths', [])] - - match_module, separator, match_function = match_name.rpartition('.') - if not separator: - match_module = 'powerline.matchers.{0}'.format(ext) - match_function = match_name - with WithPath(import_paths): - try: - func = getattr(__import__(str(match_module), fromlist=[str(match_function)]), str(match_function)) - except ImportError: - echoerr(context='Error while loading matcher functions', - problem='failed to load module {0}'.format(match_module), - problem_mark=match_name.mark) - return True, False, True - except AttributeError: - echoerr(context='Error while loading matcher functions', - problem='failed to load matcher function {0}'.format(match_function), - problem_mark=match_name.mark) - return True, False, True - - if not callable(func): - echoerr(context='Error while loading matcher functions', - problem='loaded "function" {0} is not callable'.format(match_function), - problem_mark=match_name.mark) - return True, False, True - - if hasattr(func, 'func_code') and hasattr(func.func_code, 'co_argcount'): - if func.func_code.co_argcount != 1: - echoerr( - context='Error while loading matcher functions', - problem=( - 'function {0} accepts {1} arguments instead of 1. ' - 'Are you sure it is the proper function?' - ).format(match_function, func.func_code.co_argcount), - problem_mark=match_name.mark - ) - - return True, False, False - - -def check_ext(ext, data, context, echoerr): - havemarks(ext) - hadsomedirs = False - hadproblem = False - if ext not in data['lists']['exts']: - hadproblem = True - echoerr(context='Error while loading {0} extension configuration'.format(ext), - context_mark=ext.mark, - problem='extension configuration does not exist') - else: - for typ in ('themes', 'colorschemes'): - if ext not in data['configs'][typ] and not data['configs']['top_' + typ]: - hadproblem = True - echoerr(context='Error while loading {0} extension configuration'.format(ext), - context_mark=ext.mark, - problem='{0} configuration does not exist'.format(typ)) - else: - hadsomedirs = True - return hadsomedirs, hadproblem - - -def check_config(d, theme, data, context, echoerr): - context_has_marks(context) - if len(context) == 4: - ext = context[-2][0] - else: - # local_themes - ext = context[-3][0] - if ext not in data['lists']['exts']: - echoerr(context='Error while loading {0} extension configuration'.format(ext), - context_mark=ext.mark, - problem='extension configuration does not exist') - return True, False, True - if ( - (ext not in data['configs'][d] or theme not in data['configs'][d][ext]) - and theme not in data['configs']['top_' + d] - ): - echoerr(context='Error while loading {0} from {1} extension configuration'.format(d[:-1], ext), - problem='failed to find configuration file {0}/{1}/{2}.json'.format(d, ext, theme), - problem_mark=theme.mark) - return True, False, True - return True, False, False - - -def check_top_theme(theme, data, context, echoerr): - context_has_marks(context) - havemarks(theme) - if theme not in data['configs']['top_themes']: - echoerr(context='Error while checking extension configuration (key {key})'.format(key=context_key(context)), - context_mark=context[-2][0].mark, - problem='failed to find top theme {0}'.format(theme), - problem_mark=theme.mark) - return True, False, True - return True, False, False - - function_name_re = '^(\w+\.)*[a-zA-Z_]\w*$' @@ -669,97 +125,6 @@ colors_spec = (Spec( ).context_message('Error while loading colors configuration')) -def check_color(color, data, context, echoerr): - havemarks(color) - if (color not in data['colors_config'].get('colors', {}) - and color not in data['colors_config'].get('gradients', {})): - echoerr( - context='Error while checking highlight group in colorscheme (key {key})'.format( - key=context_key(context)), - problem='found unexistent color or gradient {0}'.format(color), - problem_mark=color.mark - ) - return True, False, True - return True, False, False - - -def check_translated_group_name(group, data, context, echoerr): - return check_group(group, data, context, echoerr) - - -def check_group(group, data, context, echoerr): - havemarks(group) - if not isinstance(group, unicode): - return True, False, False - colorscheme = data['colorscheme'] - ext = data['ext'] - configs = [] - if ext: - if colorscheme == '__main__': - configs.append([config for config in data['ext_colorscheme_configs'][ext].items()]) - configs.append([config for config in data['top_colorscheme_configs'].items()]) - else: - try: - configs.append([data['ext_colorscheme_configs'][ext][colorscheme]]) - except KeyError: - pass - try: - configs.append([data['ext_colorscheme_configs'][ext]['__main__']]) - except KeyError: - pass - try: - configs.append([data['top_colorscheme_configs'][colorscheme]]) - except KeyError: - pass - else: - try: - configs.append([data['top_colorscheme_configs'][colorscheme]]) - except KeyError: - pass - new_echoerr = DelayedEchoErr(echoerr) - hadproblem = False - for config_lst in configs: - tofind = len(config_lst) - not_found = [] - for config in config_lst: - if isinstance(config, tuple): - new_colorscheme, config = config - new_data = data.copy() - new_data['colorscheme'] = new_colorscheme - else: - new_data = data - havemarks(config) - try: - group_data = config['groups'][group] - except KeyError: - not_found.append(config.mark.name) - else: - proceed, echo, chadproblem = check_group( - group_data, - new_data, - context, - echoerr, - ) - if chadproblem: - hadproblem = True - else: - tofind -= 1 - if not tofind: - return proceed, echo, hadproblem - if not proceed: - break - if not_found: - new_echoerr( - context='Error while checking group definition in colorscheme (key {key})'.format( - key=context_key(context)), - problem='name {0} is not present in {1} {2} colorschemes: {3}'.format( - group, tofind, ext, ', '.join(not_found)), - problem_mark=group.mark - ) - new_echoerr.echo_all() - return True, False, hadproblem - - color_spec = Spec().type(unicode).func(check_color).copy name_spec = Spec().type(unicode).len('gt', 0).optional().copy group_name_spec = Spec().ident().copy @@ -814,557 +179,6 @@ shell_colorscheme_spec = (Spec( ).context_message('Error while loading shell colorscheme')) -generic_keys = set(( - 'exclude_modes', 'include_modes', - 'exclude_function', 'include_function', - 'width', 'align', - 'name', - 'draw_soft_divider', 'draw_hard_divider', - 'priority', - 'after', 'before', - 'display' -)) -type_keys = { - 'function': set(('function', 'args', 'draw_inner_divider')), - 'string': set(('contents', 'type', 'highlight_group', 'divider_highlight_group')), - 'segment_list': set(('function', 'segments', 'args', 'type')), -} -required_keys = { - 'function': set(('function',)), - 'string': set(()), - 'segment_list': set(('function', 'segments',)), -} -highlight_keys = set(('highlight_group', 'name')) - - -def check_key_compatibility(segment, data, context, echoerr): - context_has_marks(context) - havemarks(segment) - segment_type = segment.get('type', MarkedUnicode('function', None)) - havemarks(segment_type) - - if segment_type not in type_keys: - echoerr(context='Error while checking segments (key {key})'.format(key=context_key(context)), - problem='found segment with unknown type {0}'.format(segment_type), - problem_mark=segment_type.mark) - return False, False, True - - hadproblem = False - - keys = set(segment) - if not ((keys - generic_keys) < type_keys[segment_type]): - unknown_keys = keys - generic_keys - type_keys[segment_type] - echoerr( - context='Error while checking segments (key {key})'.format(key=context_key(context)), - context_mark=context[-1][1].mark, - problem='found keys not used with the current segment type: {0}'.format( - list_sep.join(unknown_keys)), - problem_mark=list(unknown_keys)[0].mark - ) - hadproblem = True - - if not (keys >= required_keys[segment_type]): - missing_keys = required_keys[segment_type] - keys - echoerr( - context='Error while checking segments (key {key})'.format(key=context_key(context)), - context_mark=context[-1][1].mark, - problem='found missing required keys: {0}'.format( - list_sep.join(missing_keys)) - ) - hadproblem = True - - if not (segment_type == 'function' or (keys & highlight_keys)): - echoerr( - context='Error while checking segments (key {key})'.format(key=context_key(context)), - context_mark=context[-1][1].mark, - problem=( - 'found missing keys required to determine highlight group. ' - 'Either highlight_group or name key must be present' - ) - ) - hadproblem = True - - return True, False, hadproblem - - -def check_segment_module(module, data, context, echoerr): - havemarks(module) - with WithPath(data['import_paths']): - try: - __import__(str(module)) - except ImportError as e: - if echoerr.logger.level >= logging.DEBUG: - echoerr.logger.exception(e) - echoerr(context='Error while checking segments (key {key})'.format(key=context_key(context)), - problem='failed to import module {0}'.format(module), - problem_mark=module.mark) - return True, False, True - return True, False, False - - -def get_function_strings(function_name, context, ext): - if '.' in function_name: - module, function_name = function_name.rpartition('.')[::2] - else: - module = context[0][1].get( - 'default_module', MarkedUnicode('powerline.segments.' + ext, None)) - return module, function_name - - -def check_full_segment_data(segment, data, context, echoerr): - if 'name' not in segment and 'function' not in segment: - return True, False, False - - ext = data['ext'] - theme_segment_data = context[0][1].get('segment_data', {}) - main_theme_name = data['main_config'].get('ext', {}).get(ext, {}).get('theme', None) - if not main_theme_name or data['theme'] == main_theme_name: - top_segment_data = {} - else: - top_segment_data = data['ext_theme_configs'].get(main_theme_name, {}).get('segment_data', {}) - - if segment.get('type', 'function') == 'function': - function_name = segment.get('function') - if function_name: - module, function_name = get_function_strings(function_name, context, ext) - names = [module + '.' + function_name, function_name] - else: - names = [] - elif segment.get('name'): - names = [segment['name']] - else: - return True, False, False - - segment_copy = segment.copy() - - for key in ('before', 'after', 'args', 'contents'): - if key not in segment_copy: - for segment_data in [theme_segment_data, top_segment_data]: - for name in names: - try: - val = segment_data[name][key] - k = segment_data[name].keydict[key] - segment_copy[k] = val - except KeyError: - pass - - return check_key_compatibility(segment_copy, data, context, echoerr) - - -def import_function(function_type, name, data, context, echoerr, module): - context_has_marks(context) - havemarks(name, module) - - with WithPath(data['import_paths']): - try: - func = getattr(__import__(str(module), fromlist=[str(name)]), str(name)) - except ImportError: - echoerr(context='Error while checking segments (key {key})'.format(key=context_key(context)), - context_mark=name.mark, - problem='failed to import module {0}'.format(module), - problem_mark=module.mark) - return None - except AttributeError: - echoerr(context='Error while loading {0} function (key {key})'.format(function_type, key=context_key(context)), - problem='failed to load function {0} from module {1}'.format(name, module), - problem_mark=name.mark) - return None - - if not callable(func): - echoerr(context='Error while checking segments (key {key})'.format(key=context_key(context)), - context_mark=name.mark, - problem='imported "function" {0} from module {1} is not callable'.format(name, module), - problem_mark=module.mark) - return None - - return func - - -def import_segment(*args, **kwargs): - return import_function('segment', *args, **kwargs) - - -def check_segment_function(function_name, data, context, echoerr): - havemarks(function_name) - ext = data['ext'] - module, function_name = get_function_strings(function_name, context, ext) - if context[-2][1].get('type', 'function') == 'function': - func = import_segment(function_name, data, context, echoerr, module=module) - - if not func: - return True, False, True - - hl_groups = [] - divider_hl_group = None - - if func.__doc__: - H_G_USED_STR = 'Highlight groups used: ' - LHGUS = len(H_G_USED_STR) - D_H_G_USED_STR = 'Divider highlight group used: ' - LDHGUS = len(D_H_G_USED_STR) - pointer = 0 - mark_name = '<{0} docstring>'.format(function_name) - for i, line in enumerate(func.__doc__.split('\n')): - if H_G_USED_STR in line: - idx = line.index(H_G_USED_STR) + LHGUS - hl_groups.append(( - line[idx:], - (mark_name, i + 1, idx + 1, func.__doc__), - pointer + idx - )) - elif D_H_G_USED_STR in line: - idx = line.index(D_H_G_USED_STR) + LDHGUS + 2 - mark = Mark(mark_name, i + 1, idx + 1, func.__doc__, pointer + idx) - divider_hl_group = MarkedUnicode(line[idx:-3], mark) - pointer += len(line) + len('\n') - - hadproblem = False - - if divider_hl_group: - r = hl_exists(divider_hl_group, data, context, echoerr, allow_gradients=True) - if r: - echoerr( - context='Error while checking theme (key {key})'.format(key=context_key(context)), - problem=( - 'found highlight group {0} not defined in the following colorschemes: {1}\n' - '(Group name was obtained from function documentation.)' - ).format(divider_hl_group, list_sep.join(r)), - problem_mark=function_name.mark - ) - hadproblem = True - - if hl_groups: - greg = re.compile(r'``([^`]+)``( \(gradient\))?') - parsed_hl_groups = [] - for line, mark_args, pointer in hl_groups: - for s in line.split(', '): - required_pack = [] - sub_pointer = pointer - for subs in s.split(' or '): - match = greg.match(subs) - try: - if not match: - continue - hl_group = MarkedUnicode( - match.group(1), - Mark(*mark_args, pointer=sub_pointer + match.start(1)) - ) - gradient = bool(match.group(2)) - required_pack.append((hl_group, gradient)) - finally: - sub_pointer += len(subs) + len(' or ') - parsed_hl_groups.append(required_pack) - pointer += len(s) + len(', ') - del hl_group, gradient - for required_pack in parsed_hl_groups: - rs = [ - hl_exists(hl_group, data, context, echoerr, allow_gradients=('force' if gradient else False)) - for hl_group, gradient in required_pack - ] - if all(rs): - echoerr( - context='Error while checking theme (key {key})'.format(key=context_key(context)), - problem=( - 'found highlight groups list ({0}) with all groups not defined in some colorschemes\n' - '(Group names were taken from function documentation.)' - ).format(list_sep.join((h[0] for h in required_pack))), - problem_mark=function_name.mark - ) - for r, h in zip(rs, required_pack): - echoerr( - context='Error while checking theme (key {key})'.format(key=context_key(context)), - problem='found highlight group {0} not defined in the following colorschemes: {1}'.format( - h[0], list_sep.join(r)) - ) - hadproblem = True - else: - r = hl_exists(function_name, data, context, echoerr, allow_gradients=True) - if r: - echoerr( - context='Error while checking theme (key {key})'.format(key=context_key(context)), - problem=( - 'found highlight group {0} not defined in the following colorschemes: {1}\n' - '(If not specified otherwise in documentation, ' - 'highlight group for function segments\n' - 'is the same as the function name.)' - ).format(function_name, list_sep.join(r)), - problem_mark=function_name.mark - ) - hadproblem = True - - return True, False, hadproblem - elif context[-2][1].get('type') != 'segment_list': - if function_name not in context[0][1].get('segment_data', {}): - main_theme_name = data['main_config'].get('ext', {}).get(ext, {}).get('theme', None) - if data['theme'] == main_theme_name: - main_theme = {} - else: - main_theme = data['ext_theme_configs'].get(main_theme_name, {}) - if ( - function_name not in main_theme.get('segment_data', {}) - and function_name not in data['ext_theme_configs'].get('__main__', {}).get('segment_data', {}) - and not any(((function_name in theme.get('segment_data', {})) for theme in data['top_themes'].values())) - ): - echoerr(context='Error while checking segments (key {key})'.format(key=context_key(context)), - problem='found useless use of name key (such name is not present in theme/segment_data)', - problem_mark=function_name.mark) - - return True, False, False - - -def hl_exists(hl_group, data, context, echoerr, allow_gradients=False): - havemarks(hl_group) - ext = data['ext'] - if ext not in data['colorscheme_configs']: - # No colorschemes. Error was already reported, no need to report it - # twice - return [] - r = [] - for colorscheme, cconfig in data['colorscheme_configs'][ext].items(): - if hl_group not in cconfig.get('groups', {}): - r.append(colorscheme) - elif not allow_gradients or allow_gradients == 'force': - group_config = cconfig['groups'][hl_group] - havemarks(group_config) - hadgradient = False - for ckey in ('fg', 'bg'): - color = group_config.get(ckey) - if not color: - # No color. Error was already reported. - continue - havemarks(color) - # Gradients are only allowed for function segments. Note that - # whether *either* color or gradient exists should have been - # already checked - hascolor = color in data['colors_config'].get('colors', {}) - hasgradient = color in data['colors_config'].get('gradients', {}) - if hasgradient: - hadgradient = True - if allow_gradients is False and not hascolor and hasgradient: - echoerr( - context='Error while checking highlight group in theme (key {key})'.format( - key=context_key(context)), - context_mark=hl_group.mark, - problem='group {0} is using gradient {1} instead of a color'.format(hl_group, color), - problem_mark=color.mark - ) - r.append(colorscheme) - continue - if allow_gradients == 'force' and not hadgradient: - echoerr( - context='Error while checking highlight group in theme (key {key})'.format( - key=context_key(context)), - context_mark=hl_group.mark, - problem='group {0} should have at least one gradient color, but it has no'.format(hl_group), - problem_mark=group_config.mark - ) - r.append(colorscheme) - return r - - -def check_highlight_group(hl_group, data, context, echoerr): - havemarks(hl_group) - r = hl_exists(hl_group, data, context, echoerr) - if r: - echoerr( - context='Error while checking theme (key {key})'.format(key=context_key(context)), - problem='found highlight group {0} not defined in the following colorschemes: {1}'.format( - hl_group, list_sep.join(r)), - problem_mark=hl_group.mark - ) - return True, False, True - return True, False, False - - -def check_highlight_groups(hl_groups, data, context, echoerr): - havemarks(hl_groups) - rs = [hl_exists(hl_group, data, context, echoerr) for hl_group in hl_groups] - if all(rs): - echoerr( - context='Error while checking theme (key {key})'.format(key=context_key(context)), - problem='found highlight groups list ({0}) with all groups not defined in some colorschemes'.format( - list_sep.join((unicode(h) for h in hl_groups))), - problem_mark=hl_groups.mark - ) - for r, hl_group in zip(rs, hl_groups): - echoerr( - context='Error while checking theme (key {key})'.format(key=context_key(context)), - problem='found highlight group {0} not defined in the following colorschemes: {1}'.format( - hl_group, list_sep.join(r)), - problem_mark=hl_group.mark - ) - return True, False, True - return True, False, False - - -def list_themes(data, context): - theme_type = data['theme_type'] - ext = data['ext'] - main_theme_name = data['main_config'].get('ext', {}).get(ext, {}).get('theme', None) - is_main_theme = (data['theme'] == main_theme_name) - if theme_type == 'top': - return list(itertools.chain(*[ - [(theme_ext, theme) for theme in theme_configs.values()] - for theme_ext, theme_configs in data['theme_configs'].items() - ])) - elif theme_type == 'main' or is_main_theme: - return [(ext, theme) for theme in data['ext_theme_configs'].values()] - else: - return [(ext, context[0][1])] - - -def check_segment_data_key(key, data, context, echoerr): - havemarks(key) - has_module_name = '.' in key - found = False - for ext, theme in list_themes(data, context): - for segments in theme.get('segments', {}).values(): - for segment in segments: - if 'name' in segment: - if key == segment['name']: - found = True - break - else: - function_name = segment.get('function') - if function_name: - module, function_name = get_function_strings(function_name, ((None, theme),), ext) - if has_module_name: - full_name = module + '.' + function_name - if key == full_name: - found = True - break - else: - if key == function_name: - found = True - break - if found: - break - if found: - break - else: - if data['theme_type'] != 'top': - echoerr(context='Error while checking segment data', - problem='found key {0} that cannot be associated with any segment'.format(key), - problem_mark=key.mark) - return True, False, True - - return True, False, False - - -threaded_args_specs = { - 'interval': Spec().cmp('gt', 0.0), - 'update_first': Spec().type(bool), - 'shutdown_event': Spec().error('Shutdown event must be set by powerline'), -} - - -def check_args_variant(func, args, data, context, echoerr): - havemarks(args) - argspec = getconfigargspec(func) - present_args = set(args) - all_args = set(argspec.args) - required_args = set(argspec.args[:-len(argspec.defaults)]) - - hadproblem = False - - if required_args - present_args: - echoerr( - context='Error while checking segment arguments (key {key})'.format(key=context_key(context)), - context_mark=args.mark, - problem='some of the required keys are missing: {0}'.format(list_sep.join(required_args - present_args)) - ) - hadproblem = True - - if not all_args >= present_args: - echoerr(context='Error while checking segment arguments (key {key})'.format(key=context_key(context)), - context_mark=args.mark, - problem='found unknown keys: {0}'.format(list_sep.join(present_args - all_args)), - problem_mark=next(iter(present_args - all_args)).mark) - hadproblem = True - - if isinstance(func, ThreadedSegment): - for key in set(threaded_args_specs) & present_args: - proceed, khadproblem = threaded_args_specs[key].match( - args[key], - args.mark, - data, - context + new_context_item(key, args), - echoerr - ) - if khadproblem: - hadproblem = True - if not proceed: - return hadproblem - - return hadproblem - - -def check_args(get_functions, args, data, context, echoerr): - context_has_marks(context) - new_echoerr = DelayedEchoErr(echoerr) - count = 0 - hadproblem = False - for func in get_functions(data, context, new_echoerr): - count += 1 - shadproblem = check_args_variant(func, args, data, context, echoerr) - if shadproblem: - hadproblem = True - - if not count: - hadproblem = True - if new_echoerr: - new_echoerr.echo_all() - else: - echoerr(context='Error while checking segment arguments (key {key})'.format(key=context_key(context)), - context_mark=context[-2][1].mark, - problem='no suitable segments found') - - return True, False, hadproblem - - -def get_one_segment_function(data, context, echoerr): - ext = data['ext'] - function_name = context[-2][1].get('function') - if function_name: - module, function_name = get_function_strings(function_name, context, ext) - func = import_segment(function_name, data, context, echoerr, module=module) - if func: - yield func - - -def get_all_possible_functions(data, context, echoerr): - name = context[-2][0] - module, name = name.rpartition('.')[::2] - if module: - func = import_segment(name, data, context, echoerr, module=module) - if func: - yield func - else: - for ext, theme_config in list_themes(data, context): - for segments in theme_config.get('segments', {}).values(): - for segment in segments: - if segment.get('type', 'function') == 'function': - function_name = segment.get('function') - current_name = segment.get('name') - if function_name: - module, function_name = get_function_strings(function_name, ((None, theme_config),), ext) - if current_name == name or function_name == name: - func = import_segment(function_name, data, context, echoerr, module=module) - if func: - yield func - - -def check_exinclude_function(name, data, context, echoerr): - ext = data['ext'] - module, name = name.rpartition('.')[::2] - if not module: - module = MarkedUnicode('powerline.selectors.' + ext, None) - func = import_function('selector', name, data, context, echoerr, module=module) - if not func: - return True, False, True - return True, False, False - - args_spec = Spec( pl=Spec().error('pl object must be set by powerline').optional(), segment_info=Spec().error('Segment info dictionary must be set by powerline').optional(), diff --git a/powerline/lint/checks.py b/powerline/lint/checks.py new file mode 100644 index 00000000..6ce70b1b --- /dev/null +++ b/powerline/lint/checks.py @@ -0,0 +1,707 @@ +# vim:fileencoding=utf-8:noet +from __future__ import (unicode_literals, division, absolute_import, print_function) + +import os +import re +import logging + +from powerline.lib.threaded import ThreadedSegment +from powerline.lib.unicode import unicode +from powerline.lint.markedjson.markedvalue import MarkedUnicode +from powerline.lint.markedjson.error import DelayedEchoErr, Mark +from powerline.lint.selfcheck import havemarks, context_has_marks +from powerline.lint.context import context_key, list_sep, list_themes, new_context_item +from powerline.lint.imp import WithPath, import_function, import_segment +from powerline.lint.spec import Spec +from powerline.lint.inspect import getconfigargspec + + +generic_keys = set(( + 'exclude_modes', 'include_modes', + 'exclude_function', 'include_function', + 'width', 'align', + 'name', + 'draw_soft_divider', 'draw_hard_divider', + 'priority', + 'after', 'before', + 'display' +)) +type_keys = { + 'function': set(('function', 'args', 'draw_inner_divider')), + 'string': set(('contents', 'type', 'highlight_group', 'divider_highlight_group')), + 'segment_list': set(('function', 'segments', 'args', 'type')), +} +required_keys = { + 'function': set(('function',)), + 'string': set(()), + 'segment_list': set(('function', 'segments',)), +} +highlight_keys = set(('highlight_group', 'name')) + + +def get_function_strings(function_name, context, ext): + if '.' in function_name: + module, function_name = function_name.rpartition('.')[::2] + else: + module = context[0][1].get( + 'default_module', MarkedUnicode('powerline.segments.' + ext, None)) + return module, function_name + + +def check_matcher_func(ext, match_name, data, context, echoerr): + havemarks(match_name) + import_paths = [os.path.expanduser(path) for path in context[0][1].get('common', {}).get('paths', [])] + + match_module, separator, match_function = match_name.rpartition('.') + if not separator: + match_module = 'powerline.matchers.{0}'.format(ext) + match_function = match_name + with WithPath(import_paths): + try: + func = getattr(__import__(str(match_module), fromlist=[str(match_function)]), str(match_function)) + except ImportError: + echoerr(context='Error while loading matcher functions', + problem='failed to load module {0}'.format(match_module), + problem_mark=match_name.mark) + return True, False, True + except AttributeError: + echoerr(context='Error while loading matcher functions', + problem='failed to load matcher function {0}'.format(match_function), + problem_mark=match_name.mark) + return True, False, True + + if not callable(func): + echoerr(context='Error while loading matcher functions', + problem='loaded "function" {0} is not callable'.format(match_function), + problem_mark=match_name.mark) + return True, False, True + + if hasattr(func, 'func_code') and hasattr(func.func_code, 'co_argcount'): + if func.func_code.co_argcount != 1: + echoerr( + context='Error while loading matcher functions', + problem=( + 'function {0} accepts {1} arguments instead of 1. ' + 'Are you sure it is the proper function?' + ).format(match_function, func.func_code.co_argcount), + problem_mark=match_name.mark + ) + + return True, False, False + + +def check_ext(ext, data, context, echoerr): + havemarks(ext) + hadsomedirs = False + hadproblem = False + if ext not in data['lists']['exts']: + hadproblem = True + echoerr(context='Error while loading {0} extension configuration'.format(ext), + context_mark=ext.mark, + problem='extension configuration does not exist') + else: + for typ in ('themes', 'colorschemes'): + if ext not in data['configs'][typ] and not data['configs']['top_' + typ]: + hadproblem = True + echoerr(context='Error while loading {0} extension configuration'.format(ext), + context_mark=ext.mark, + problem='{0} configuration does not exist'.format(typ)) + else: + hadsomedirs = True + return hadsomedirs, hadproblem + + +def check_config(d, theme, data, context, echoerr): + context_has_marks(context) + if len(context) == 4: + ext = context[-2][0] + else: + # local_themes + ext = context[-3][0] + if ext not in data['lists']['exts']: + echoerr(context='Error while loading {0} extension configuration'.format(ext), + context_mark=ext.mark, + problem='extension configuration does not exist') + return True, False, True + if ( + (ext not in data['configs'][d] or theme not in data['configs'][d][ext]) + and theme not in data['configs']['top_' + d] + ): + echoerr(context='Error while loading {0} from {1} extension configuration'.format(d[:-1], ext), + problem='failed to find configuration file {0}/{1}/{2}.json'.format(d, ext, theme), + problem_mark=theme.mark) + return True, False, True + return True, False, False + + +def check_top_theme(theme, data, context, echoerr): + context_has_marks(context) + havemarks(theme) + if theme not in data['configs']['top_themes']: + echoerr(context='Error while checking extension configuration (key {key})'.format(key=context_key(context)), + context_mark=context[-2][0].mark, + problem='failed to find top theme {0}'.format(theme), + problem_mark=theme.mark) + return True, False, True + return True, False, False + + +def check_color(color, data, context, echoerr): + havemarks(color) + if (color not in data['colors_config'].get('colors', {}) + and color not in data['colors_config'].get('gradients', {})): + echoerr( + context='Error while checking highlight group in colorscheme (key {key})'.format( + key=context_key(context)), + problem='found unexistent color or gradient {0}'.format(color), + problem_mark=color.mark + ) + return True, False, True + return True, False, False + + +def check_translated_group_name(group, data, context, echoerr): + return check_group(group, data, context, echoerr) + + +def check_group(group, data, context, echoerr): + havemarks(group) + if not isinstance(group, unicode): + return True, False, False + colorscheme = data['colorscheme'] + ext = data['ext'] + configs = [] + if ext: + if colorscheme == '__main__': + configs.append([config for config in data['ext_colorscheme_configs'][ext].items()]) + configs.append([config for config in data['top_colorscheme_configs'].items()]) + else: + try: + configs.append([data['ext_colorscheme_configs'][ext][colorscheme]]) + except KeyError: + pass + try: + configs.append([data['ext_colorscheme_configs'][ext]['__main__']]) + except KeyError: + pass + try: + configs.append([data['top_colorscheme_configs'][colorscheme]]) + except KeyError: + pass + else: + try: + configs.append([data['top_colorscheme_configs'][colorscheme]]) + except KeyError: + pass + new_echoerr = DelayedEchoErr(echoerr) + hadproblem = False + for config_lst in configs: + tofind = len(config_lst) + not_found = [] + for config in config_lst: + if isinstance(config, tuple): + new_colorscheme, config = config + new_data = data.copy() + new_data['colorscheme'] = new_colorscheme + else: + new_data = data + havemarks(config) + try: + group_data = config['groups'][group] + except KeyError: + not_found.append(config.mark.name) + else: + proceed, echo, chadproblem = check_group( + group_data, + new_data, + context, + echoerr, + ) + if chadproblem: + hadproblem = True + else: + tofind -= 1 + if not tofind: + return proceed, echo, hadproblem + if not proceed: + break + if not_found: + new_echoerr( + context='Error while checking group definition in colorscheme (key {key})'.format( + key=context_key(context)), + problem='name {0} is not present in {1} {2} colorschemes: {3}'.format( + group, tofind, ext, ', '.join(not_found)), + problem_mark=group.mark + ) + new_echoerr.echo_all() + return True, False, hadproblem + + +def check_key_compatibility(segment, data, context, echoerr): + context_has_marks(context) + havemarks(segment) + segment_type = segment.get('type', MarkedUnicode('function', None)) + havemarks(segment_type) + + if segment_type not in type_keys: + echoerr(context='Error while checking segments (key {key})'.format(key=context_key(context)), + problem='found segment with unknown type {0}'.format(segment_type), + problem_mark=segment_type.mark) + return False, False, True + + hadproblem = False + + keys = set(segment) + if not ((keys - generic_keys) < type_keys[segment_type]): + unknown_keys = keys - generic_keys - type_keys[segment_type] + echoerr( + context='Error while checking segments (key {key})'.format(key=context_key(context)), + context_mark=context[-1][1].mark, + problem='found keys not used with the current segment type: {0}'.format( + list_sep.join(unknown_keys)), + problem_mark=list(unknown_keys)[0].mark + ) + hadproblem = True + + if not (keys >= required_keys[segment_type]): + missing_keys = required_keys[segment_type] - keys + echoerr( + context='Error while checking segments (key {key})'.format(key=context_key(context)), + context_mark=context[-1][1].mark, + problem='found missing required keys: {0}'.format( + list_sep.join(missing_keys)) + ) + hadproblem = True + + if not (segment_type == 'function' or (keys & highlight_keys)): + echoerr( + context='Error while checking segments (key {key})'.format(key=context_key(context)), + context_mark=context[-1][1].mark, + problem=( + 'found missing keys required to determine highlight group. ' + 'Either highlight_group or name key must be present' + ) + ) + hadproblem = True + + return True, False, hadproblem + + +def check_segment_module(module, data, context, echoerr): + havemarks(module) + with WithPath(data['import_paths']): + try: + __import__(str(module)) + except ImportError as e: + if echoerr.logger.level >= logging.DEBUG: + echoerr.logger.exception(e) + echoerr(context='Error while checking segments (key {key})'.format(key=context_key(context)), + problem='failed to import module {0}'.format(module), + problem_mark=module.mark) + return True, False, True + return True, False, False + + +def check_full_segment_data(segment, data, context, echoerr): + if 'name' not in segment and 'function' not in segment: + return True, False, False + + ext = data['ext'] + theme_segment_data = context[0][1].get('segment_data', {}) + main_theme_name = data['main_config'].get('ext', {}).get(ext, {}).get('theme', None) + if not main_theme_name or data['theme'] == main_theme_name: + top_segment_data = {} + else: + top_segment_data = data['ext_theme_configs'].get(main_theme_name, {}).get('segment_data', {}) + + if segment.get('type', 'function') == 'function': + function_name = segment.get('function') + if function_name: + module, function_name = get_function_strings(function_name, context, ext) + names = [module + '.' + function_name, function_name] + else: + names = [] + elif segment.get('name'): + names = [segment['name']] + else: + return True, False, False + + segment_copy = segment.copy() + + for key in ('before', 'after', 'args', 'contents'): + if key not in segment_copy: + for segment_data in [theme_segment_data, top_segment_data]: + for name in names: + try: + val = segment_data[name][key] + k = segment_data[name].keydict[key] + segment_copy[k] = val + except KeyError: + pass + + return check_key_compatibility(segment_copy, data, context, echoerr) + + +def check_segment_function(function_name, data, context, echoerr): + havemarks(function_name) + ext = data['ext'] + module, function_name = get_function_strings(function_name, context, ext) + if context[-2][1].get('type', 'function') == 'function': + func = import_segment(function_name, data, context, echoerr, module=module) + + if not func: + return True, False, True + + hl_groups = [] + divider_hl_group = None + + if func.__doc__: + H_G_USED_STR = 'Highlight groups used: ' + LHGUS = len(H_G_USED_STR) + D_H_G_USED_STR = 'Divider highlight group used: ' + LDHGUS = len(D_H_G_USED_STR) + pointer = 0 + mark_name = '<{0} docstring>'.format(function_name) + for i, line in enumerate(func.__doc__.split('\n')): + if H_G_USED_STR in line: + idx = line.index(H_G_USED_STR) + LHGUS + hl_groups.append(( + line[idx:], + (mark_name, i + 1, idx + 1, func.__doc__), + pointer + idx + )) + elif D_H_G_USED_STR in line: + idx = line.index(D_H_G_USED_STR) + LDHGUS + 2 + mark = Mark(mark_name, i + 1, idx + 1, func.__doc__, pointer + idx) + divider_hl_group = MarkedUnicode(line[idx:-3], mark) + pointer += len(line) + len('\n') + + hadproblem = False + + if divider_hl_group: + r = hl_exists(divider_hl_group, data, context, echoerr, allow_gradients=True) + if r: + echoerr( + context='Error while checking theme (key {key})'.format(key=context_key(context)), + problem=( + 'found highlight group {0} not defined in the following colorschemes: {1}\n' + '(Group name was obtained from function documentation.)' + ).format(divider_hl_group, list_sep.join(r)), + problem_mark=function_name.mark + ) + hadproblem = True + + if hl_groups: + greg = re.compile(r'``([^`]+)``( \(gradient\))?') + parsed_hl_groups = [] + for line, mark_args, pointer in hl_groups: + for s in line.split(', '): + required_pack = [] + sub_pointer = pointer + for subs in s.split(' or '): + match = greg.match(subs) + try: + if not match: + continue + hl_group = MarkedUnicode( + match.group(1), + Mark(*mark_args, pointer=sub_pointer + match.start(1)) + ) + gradient = bool(match.group(2)) + required_pack.append((hl_group, gradient)) + finally: + sub_pointer += len(subs) + len(' or ') + parsed_hl_groups.append(required_pack) + pointer += len(s) + len(', ') + del hl_group, gradient + for required_pack in parsed_hl_groups: + rs = [ + hl_exists(hl_group, data, context, echoerr, allow_gradients=('force' if gradient else False)) + for hl_group, gradient in required_pack + ] + if all(rs): + echoerr( + context='Error while checking theme (key {key})'.format(key=context_key(context)), + problem=( + 'found highlight groups list ({0}) with all groups not defined in some colorschemes\n' + '(Group names were taken from function documentation.)' + ).format(list_sep.join((h[0] for h in required_pack))), + problem_mark=function_name.mark + ) + for r, h in zip(rs, required_pack): + echoerr( + context='Error while checking theme (key {key})'.format(key=context_key(context)), + problem='found highlight group {0} not defined in the following colorschemes: {1}'.format( + h[0], list_sep.join(r)) + ) + hadproblem = True + else: + r = hl_exists(function_name, data, context, echoerr, allow_gradients=True) + if r: + echoerr( + context='Error while checking theme (key {key})'.format(key=context_key(context)), + problem=( + 'found highlight group {0} not defined in the following colorschemes: {1}\n' + '(If not specified otherwise in documentation, ' + 'highlight group for function segments\n' + 'is the same as the function name.)' + ).format(function_name, list_sep.join(r)), + problem_mark=function_name.mark + ) + hadproblem = True + + return True, False, hadproblem + elif context[-2][1].get('type') != 'segment_list': + if function_name not in context[0][1].get('segment_data', {}): + main_theme_name = data['main_config'].get('ext', {}).get(ext, {}).get('theme', None) + if data['theme'] == main_theme_name: + main_theme = {} + else: + main_theme = data['ext_theme_configs'].get(main_theme_name, {}) + if ( + function_name not in main_theme.get('segment_data', {}) + and function_name not in data['ext_theme_configs'].get('__main__', {}).get('segment_data', {}) + and not any(((function_name in theme.get('segment_data', {})) for theme in data['top_themes'].values())) + ): + echoerr(context='Error while checking segments (key {key})'.format(key=context_key(context)), + problem='found useless use of name key (such name is not present in theme/segment_data)', + problem_mark=function_name.mark) + + return True, False, False + + +def hl_exists(hl_group, data, context, echoerr, allow_gradients=False): + havemarks(hl_group) + ext = data['ext'] + if ext not in data['colorscheme_configs']: + # No colorschemes. Error was already reported, no need to report it + # twice + return [] + r = [] + for colorscheme, cconfig in data['colorscheme_configs'][ext].items(): + if hl_group not in cconfig.get('groups', {}): + r.append(colorscheme) + elif not allow_gradients or allow_gradients == 'force': + group_config = cconfig['groups'][hl_group] + havemarks(group_config) + hadgradient = False + for ckey in ('fg', 'bg'): + color = group_config.get(ckey) + if not color: + # No color. Error was already reported. + continue + havemarks(color) + # Gradients are only allowed for function segments. Note that + # whether *either* color or gradient exists should have been + # already checked + hascolor = color in data['colors_config'].get('colors', {}) + hasgradient = color in data['colors_config'].get('gradients', {}) + if hasgradient: + hadgradient = True + if allow_gradients is False and not hascolor and hasgradient: + echoerr( + context='Error while checking highlight group in theme (key {key})'.format( + key=context_key(context)), + context_mark=hl_group.mark, + problem='group {0} is using gradient {1} instead of a color'.format(hl_group, color), + problem_mark=color.mark + ) + r.append(colorscheme) + continue + if allow_gradients == 'force' and not hadgradient: + echoerr( + context='Error while checking highlight group in theme (key {key})'.format( + key=context_key(context)), + context_mark=hl_group.mark, + problem='group {0} should have at least one gradient color, but it has no'.format(hl_group), + problem_mark=group_config.mark + ) + r.append(colorscheme) + return r + + +def check_highlight_group(hl_group, data, context, echoerr): + havemarks(hl_group) + r = hl_exists(hl_group, data, context, echoerr) + if r: + echoerr( + context='Error while checking theme (key {key})'.format(key=context_key(context)), + problem='found highlight group {0} not defined in the following colorschemes: {1}'.format( + hl_group, list_sep.join(r)), + problem_mark=hl_group.mark + ) + return True, False, True + return True, False, False + + +def check_highlight_groups(hl_groups, data, context, echoerr): + havemarks(hl_groups) + rs = [hl_exists(hl_group, data, context, echoerr) for hl_group in hl_groups] + if all(rs): + echoerr( + context='Error while checking theme (key {key})'.format(key=context_key(context)), + problem='found highlight groups list ({0}) with all groups not defined in some colorschemes'.format( + list_sep.join((unicode(h) for h in hl_groups))), + problem_mark=hl_groups.mark + ) + for r, hl_group in zip(rs, hl_groups): + echoerr( + context='Error while checking theme (key {key})'.format(key=context_key(context)), + problem='found highlight group {0} not defined in the following colorschemes: {1}'.format( + hl_group, list_sep.join(r)), + problem_mark=hl_group.mark + ) + return True, False, True + return True, False, False + + +def check_segment_data_key(key, data, context, echoerr): + havemarks(key) + has_module_name = '.' in key + found = False + for ext, theme in list_themes(data, context): + for segments in theme.get('segments', {}).values(): + for segment in segments: + if 'name' in segment: + if key == segment['name']: + found = True + break + else: + function_name = segment.get('function') + if function_name: + module, function_name = get_function_strings(function_name, ((None, theme),), ext) + if has_module_name: + full_name = module + '.' + function_name + if key == full_name: + found = True + break + else: + if key == function_name: + found = True + break + if found: + break + if found: + break + else: + if data['theme_type'] != 'top': + echoerr(context='Error while checking segment data', + problem='found key {0} that cannot be associated with any segment'.format(key), + problem_mark=key.mark) + return True, False, True + + return True, False, False + + +threaded_args_specs = { + 'interval': Spec().cmp('gt', 0.0), + 'update_first': Spec().type(bool), + 'shutdown_event': Spec().error('Shutdown event must be set by powerline'), +} + + +def check_args_variant(func, args, data, context, echoerr): + havemarks(args) + argspec = getconfigargspec(func) + present_args = set(args) + all_args = set(argspec.args) + required_args = set(argspec.args[:-len(argspec.defaults)]) + + hadproblem = False + + if required_args - present_args: + echoerr( + context='Error while checking segment arguments (key {key})'.format(key=context_key(context)), + context_mark=args.mark, + problem='some of the required keys are missing: {0}'.format(list_sep.join(required_args - present_args)) + ) + hadproblem = True + + if not all_args >= present_args: + echoerr(context='Error while checking segment arguments (key {key})'.format(key=context_key(context)), + context_mark=args.mark, + problem='found unknown keys: {0}'.format(list_sep.join(present_args - all_args)), + problem_mark=next(iter(present_args - all_args)).mark) + hadproblem = True + + if isinstance(func, ThreadedSegment): + for key in set(threaded_args_specs) & present_args: + proceed, khadproblem = threaded_args_specs[key].match( + args[key], + args.mark, + data, + context + new_context_item(key, args), + echoerr + ) + if khadproblem: + hadproblem = True + if not proceed: + return hadproblem + + return hadproblem + + +def check_args(get_functions, args, data, context, echoerr): + context_has_marks(context) + new_echoerr = DelayedEchoErr(echoerr) + count = 0 + hadproblem = False + for func in get_functions(data, context, new_echoerr): + count += 1 + shadproblem = check_args_variant(func, args, data, context, echoerr) + if shadproblem: + hadproblem = True + + if not count: + hadproblem = True + if new_echoerr: + new_echoerr.echo_all() + else: + echoerr(context='Error while checking segment arguments (key {key})'.format(key=context_key(context)), + context_mark=context[-2][1].mark, + problem='no suitable segments found') + + return True, False, hadproblem + + +def get_one_segment_function(data, context, echoerr): + ext = data['ext'] + function_name = context[-2][1].get('function') + if function_name: + module, function_name = get_function_strings(function_name, context, ext) + func = import_segment(function_name, data, context, echoerr, module=module) + if func: + yield func + + +def get_all_possible_functions(data, context, echoerr): + name = context[-2][0] + module, name = name.rpartition('.')[::2] + if module: + func = import_segment(name, data, context, echoerr, module=module) + if func: + yield func + else: + for ext, theme_config in list_themes(data, context): + for segments in theme_config.get('segments', {}).values(): + for segment in segments: + if segment.get('type', 'function') == 'function': + function_name = segment.get('function') + current_name = segment.get('name') + if function_name: + module, function_name = get_function_strings(function_name, ((None, theme_config),), ext) + if current_name == name or function_name == name: + func = import_segment(function_name, data, context, echoerr, module=module) + if func: + yield func + + +def check_exinclude_function(name, data, context, echoerr): + ext = data['ext'] + module, name = name.rpartition('.')[::2] + if not module: + module = MarkedUnicode('powerline.selectors.' + ext, None) + func = import_function('selector', name, data, context, echoerr, module=module) + if not func: + return True, False, True + return True, False, False diff --git a/powerline/lint/context.py b/powerline/lint/context.py new file mode 100644 index 00000000..cd1b887e --- /dev/null +++ b/powerline/lint/context.py @@ -0,0 +1,44 @@ +# vim:fileencoding=utf-8:noet +from __future__ import (unicode_literals, division, absolute_import, print_function) + +import itertools + +from powerline.lib.unicode import unicode +from powerline.lint.markedjson.markedvalue import MarkedUnicode + + +class JStr(unicode): + def join(self, iterable): + return super(JStr, self).join((unicode(item) for item in iterable)) + + +key_sep = JStr('/') +list_sep = JStr(', ') + + +def context_key(context): + return key_sep.join((c[0] for c in context)) + + +def init_context(config): + return ((MarkedUnicode('', config.mark), config),) + + +def new_context_item(key, value): + return ((value.keydict[key], value[key]),) + + +def list_themes(data, context): + theme_type = data['theme_type'] + ext = data['ext'] + main_theme_name = data['main_config'].get('ext', {}).get(ext, {}).get('theme', None) + is_main_theme = (data['theme'] == main_theme_name) + if theme_type == 'top': + return list(itertools.chain(*[ + [(theme_ext, theme) for theme in theme_configs.values()] + for theme_ext, theme_configs in data['theme_configs'].items() + ])) + elif theme_type == 'main' or is_main_theme: + return [(ext, theme) for theme in data['ext_theme_configs'].values()] + else: + return [(ext, context[0][1])] diff --git a/powerline/lint/imp.py b/powerline/lint/imp.py new file mode 100644 index 00000000..b6a154dd --- /dev/null +++ b/powerline/lint/imp.py @@ -0,0 +1,52 @@ +# vim:fileencoding=utf-8:noet +from __future__ import (unicode_literals, division, absolute_import, print_function) + +import sys + +from powerline.lint.selfcheck import havemarks, context_has_marks +from powerline.lint.context import context_key + + +class WithPath(object): + def __init__(self, import_paths): + self.import_paths = import_paths + + def __enter__(self): + self.oldpath = sys.path + sys.path = self.import_paths + sys.path + + def __exit__(self, *args): + sys.path = self.oldpath + + +def import_function(function_type, name, data, context, echoerr, module): + context_has_marks(context) + havemarks(name, module) + + with WithPath(data['import_paths']): + try: + func = getattr(__import__(str(module), fromlist=[str(name)]), str(name)) + except ImportError: + echoerr(context='Error while checking segments (key {key})'.format(key=context_key(context)), + context_mark=name.mark, + problem='failed to import module {0}'.format(module), + problem_mark=module.mark) + return None + except AttributeError: + echoerr(context='Error while loading {0} function (key {key})'.format(function_type, key=context_key(context)), + problem='failed to load function {0} from module {1}'.format(name, module), + problem_mark=name.mark) + return None + + if not callable(func): + echoerr(context='Error while checking segments (key {key})'.format(key=context_key(context)), + context_mark=name.mark, + problem='imported "function" {0} from module {1} is not callable'.format(name, module), + problem_mark=module.mark) + return None + + return func + + +def import_segment(*args, **kwargs): + return import_function('segment', *args, **kwargs) diff --git a/powerline/lint/markedjson/error.py b/powerline/lint/markedjson/error.py index 6494d10f..fc09256b 100644 --- a/powerline/lint/markedjson/error.py +++ b/powerline/lint/markedjson/error.py @@ -99,3 +99,34 @@ def format_error(context=None, context_mark=None, problem=None, problem_mark=Non class MarkedError(Exception): def __init__(self, context=None, context_mark=None, problem=None, problem_mark=None, note=None): Exception.__init__(self, format_error(context, context_mark, problem, problem_mark, note)) + + +class EchoErr(object): + __slots__ = ('echoerr', 'logger',) + + def __init__(self, echoerr, logger): + self.echoerr = echoerr + self.logger = logger + + def __call__(self, *args, **kwargs): + self.echoerr(*args, **kwargs) + + +class DelayedEchoErr(EchoErr): + __slots__ = ('echoerr', 'logger', 'errs') + + def __init__(self, echoerr): + super(DelayedEchoErr, self).__init__(echoerr, echoerr.logger) + self.errs = [] + + def __call__(self, *args, **kwargs): + self.errs.append((args, kwargs)) + + def echo_all(self): + for args, kwargs in self.errs: + self.echoerr(*args, **kwargs) + + def __nonzero__(self): + return not not self.errs + + __bool__ = __nonzero__ diff --git a/powerline/lint/selfcheck.py b/powerline/lint/selfcheck.py new file mode 100644 index 00000000..a95f35f6 --- /dev/null +++ b/powerline/lint/selfcheck.py @@ -0,0 +1,22 @@ +# vim:fileencoding=utf-8:noet +from __future__ import (unicode_literals, division, absolute_import, print_function) + +from powerline.lib.unicode import unicode + + +def havemarks(*args, **kwargs): + origin = kwargs.get('origin', '') + for i, v in enumerate(args): + if not hasattr(v, 'mark'): + raise AssertionError('Value #{0}/{1} ({2!r}) has no attribute `mark`'.format(origin, i, v)) + if isinstance(v, dict): + for key, val in v.items(): + havemarks(key, val, origin=(origin + '[' + unicode(i) + ']/' + unicode(key))) + elif isinstance(v, list): + havemarks(*v, origin=(origin + '[' + unicode(i) + ']')) + + +def context_has_marks(context): + for i, v in enumerate(context): + havemarks(v[0], origin='context key') + havemarks(v[1], origin='context val') diff --git a/powerline/lint/spec.py b/powerline/lint/spec.py new file mode 100644 index 00000000..c6fe442a --- /dev/null +++ b/powerline/lint/spec.py @@ -0,0 +1,375 @@ +# vim:fileencoding=utf-8:noet +from __future__ import (unicode_literals, division, absolute_import, print_function) + +import itertools +import re + +from copy import copy + +from powerline.lib.unicode import unicode +from powerline.lint.markedjson.error import echoerr, DelayedEchoErr +from powerline.lint.markedjson.markedvalue import MarkedUnicode +from powerline.lint.selfcheck import havemarks +from powerline.lint.context import context_key, list_sep, new_context_item + + +class Spec(object): + def __init__(self, **keys): + self.specs = [] + self.keys = {} + self.checks = [] + self.cmsg = '' + self.isoptional = False + self.uspecs = [] + self.ufailmsg = lambda key: 'found unknown key: {0}'.format(key) + self.did_type = False + self.update(**keys) + + def update(self, **keys): + for k, v in keys.items(): + self.keys[k] = len(self.specs) + self.specs.append(v) + if self.keys and not self.did_type: + self.type(dict) + self.did_type = True + return self + + def copy(self, copied=None): + copied = copied or {} + try: + return copied[id(self)] + except KeyError: + instance = self.__class__() + copied[id(self)] = instance + return self.__class__()._update(self.__dict__, copied) + + def _update(self, d, copied): + self.__dict__.update(d) + self.keys = copy(self.keys) + self.checks = copy(self.checks) + self.uspecs = copy(self.uspecs) + self.specs = [spec.copy(copied) for spec in self.specs] + return self + + def unknown_spec(self, keyfunc, spec): + if isinstance(keyfunc, Spec): + self.specs.append(keyfunc) + keyfunc = len(self.specs) - 1 + self.specs.append(spec) + self.uspecs.append((keyfunc, len(self.specs) - 1)) + return self + + def unknown_msg(self, msgfunc): + self.ufailmsg = msgfunc + return self + + def context_message(self, msg): + self.cmsg = msg + for spec in self.specs: + if not spec.cmsg: + spec.context_message(msg) + return self + + def check_type(self, value, context_mark, data, context, echoerr, types): + havemarks(value) + if type(value.value) not in types: + echoerr( + context=self.cmsg.format(key=context_key(context)), + context_mark=context_mark, + problem='{0!r} must be a {1} instance, not {2}'.format( + value, + list_sep.join((t.__name__ for t in types)), + type(value.value).__name__ + ), + problem_mark=value.mark + ) + return False, True + return True, False + + def check_func(self, value, context_mark, data, context, echoerr, func, msg_func): + havemarks(value) + proceed, echo, hadproblem = func(value, data, context, echoerr) + if echo and hadproblem: + echoerr(context=self.cmsg.format(key=context_key(context)), + context_mark=context_mark, + problem=msg_func(value), + problem_mark=value.mark) + return proceed, hadproblem + + def check_list(self, value, context_mark, data, context, echoerr, item_func, msg_func): + havemarks(value) + i = 0 + hadproblem = False + for item in value: + havemarks(item) + if isinstance(item_func, int): + spec = self.specs[item_func] + proceed, fhadproblem = spec.match( + item, + value.mark, + data, + context + ((MarkedUnicode('list item ' + unicode(i), item.mark), item),), + echoerr + ) + else: + proceed, echo, fhadproblem = item_func(item, data, context, echoerr) + if echo and fhadproblem: + echoerr(context=self.cmsg.format(key=context_key(context) + '/list item ' + unicode(i)), + context_mark=value.mark, + problem=msg_func(item), + problem_mark=item.mark) + if fhadproblem: + hadproblem = True + if not proceed: + return proceed, hadproblem + i += 1 + return True, hadproblem + + def check_either(self, value, context_mark, data, context, echoerr, start, end): + havemarks(value) + new_echoerr = DelayedEchoErr(echoerr) + + hadproblem = False + for spec in self.specs[start:end]: + proceed, hadproblem = spec.match(value, value.mark, data, context, new_echoerr) + if not proceed: + break + if not hadproblem: + return True, False + + new_echoerr.echo_all() + + return False, hadproblem + + def check_tuple(self, value, context_mark, data, context, echoerr, start, end): + havemarks(value) + hadproblem = False + for (i, item, spec) in zip(itertools.count(), value, self.specs[start:end]): + proceed, ihadproblem = spec.match( + item, + value.mark, + data, + context + ((MarkedUnicode('tuple item ' + unicode(i), item.mark), item),), + echoerr + ) + if ihadproblem: + hadproblem = True + if not proceed: + return False, hadproblem + return True, hadproblem + + def type(self, *args): + self.checks.append(('check_type', args)) + return self + + cmp_funcs = { + 'le': lambda x, y: x <= y, + 'lt': lambda x, y: x < y, + 'ge': lambda x, y: x >= y, + 'gt': lambda x, y: x > y, + 'eq': lambda x, y: x == y, + } + + cmp_msgs = { + 'le': 'lesser or equal to', + 'lt': 'lesser then', + 'ge': 'greater or equal to', + 'gt': 'greater then', + 'eq': 'equal to', + } + + def len(self, comparison, cint, msg_func=None): + cmp_func = self.cmp_funcs[comparison] + msg_func = ( + msg_func + or (lambda value: 'length of {0!r} is not {1} {2}'.format( + value, self.cmp_msgs[comparison], cint)) + ) + self.checks.append(( + 'check_func', + (lambda value, *args: (True, True, not cmp_func(len(value), cint))), + msg_func + )) + return self + + def cmp(self, comparison, cint, msg_func=None): + if type(cint) is str: + self.type(unicode) + elif type(cint) is float: + self.type(int, float) + else: + self.type(type(cint)) + cmp_func = self.cmp_funcs[comparison] + msg_func = msg_func or (lambda value: '{0} is not {1} {2}'.format(value, self.cmp_msgs[comparison], cint)) + self.checks.append(( + 'check_func', + (lambda value, *args: (True, True, not cmp_func(value.value, cint))), + msg_func + )) + return self + + def unsigned(self, msg_func=None): + self.type(int) + self.checks.append(( + 'check_func', + (lambda value, *args: (True, True, value < 0)), + (lambda value: '{0} must be greater then zero'.format(value)) + )) + return self + + def list(self, item_func, msg_func=None): + self.type(list) + if isinstance(item_func, Spec): + self.specs.append(item_func) + item_func = len(self.specs) - 1 + self.checks.append(('check_list', item_func, msg_func or (lambda item: 'failed check'))) + return self + + def tuple(self, *specs): + self.type(list) + + max_len = len(specs) + min_len = max_len + for spec in reversed(specs): + if spec.isoptional: + min_len -= 1 + else: + break + if max_len == min_len: + self.len('eq', len(specs)) + else: + self.len('ge', min_len) + self.len('le', max_len) + + start = len(self.specs) + for i, spec in zip(itertools.count(), specs): + self.specs.append(spec) + self.checks.append(('check_tuple', start, len(self.specs))) + return self + + def func(self, func, msg_func=None): + self.checks.append(('check_func', func, msg_func or (lambda value: 'failed check'))) + return self + + def re(self, regex, msg_func=None): + self.type(unicode) + compiled = re.compile(regex) + msg_func = msg_func or (lambda value: 'String "{0}" does not match "{1}"'.format(value, regex)) + self.checks.append(( + 'check_func', + (lambda value, *args: (True, True, not compiled.match(value.value))), + msg_func + )) + return self + + def ident(self, msg_func=None): + msg_func = ( + msg_func + or (lambda value: 'String "{0}" is not an alphanumeric/underscore colon-separated identifier'.format(value)) + ) + return self.re('^\w+(?::\w+)?$', msg_func) + + def oneof(self, collection, msg_func=None): + msg_func = msg_func or (lambda value: '"{0}" must be one of {1!r}'.format(value, list(collection))) + self.checks.append(( + 'check_func', + (lambda value, *args: (True, True, value not in collection)), + msg_func + )) + return self + + def error(self, msg): + self.checks.append(( + 'check_func', + (lambda *args: (True, True, True)), + (lambda value: msg.format(value)) + )) + return self + + def either(self, *specs): + start = len(self.specs) + self.specs.extend(specs) + self.checks.append(('check_either', start, len(self.specs))) + return self + + def optional(self): + self.isoptional = True + return self + + def required(self): + self.isoptional = False + return self + + def match_checks(self, *args): + hadproblem = False + for check in self.checks: + proceed, chadproblem = getattr(self, check[0])(*(args + check[1:])) + if chadproblem: + hadproblem = True + if not proceed: + return False, hadproblem + return True, hadproblem + + def match(self, value, context_mark=None, data=None, context=(), echoerr=echoerr): + havemarks(value) + proceed, hadproblem = self.match_checks(value, context_mark, data, context, echoerr) + if proceed: + if self.keys or self.uspecs: + for key, vali in self.keys.items(): + valspec = self.specs[vali] + if key in value: + proceed, mhadproblem = valspec.match( + value[key], + value.mark, + data, + context + new_context_item(key, value), + echoerr + ) + if mhadproblem: + hadproblem = True + if not proceed: + return False, hadproblem + else: + if not valspec.isoptional: + hadproblem = True + echoerr(context=self.cmsg.format(key=context_key(context)), + context_mark=None, + problem='required key is missing: {0}'.format(key), + problem_mark=value.mark) + for key in value.keys(): + havemarks(key) + if key not in self.keys: + for keyfunc, vali in self.uspecs: + valspec = self.specs[vali] + if isinstance(keyfunc, int): + spec = self.specs[keyfunc] + proceed, khadproblem = spec.match(key, context_mark, data, context, echoerr) + else: + proceed, khadproblem = keyfunc(key, data, context, echoerr) + if khadproblem: + hadproblem = True + if proceed: + proceed, vhadproblem = valspec.match( + value[key], + value.mark, + data, + context + new_context_item(key, value), + echoerr + ) + if vhadproblem: + hadproblem = True + break + else: + hadproblem = True + if self.ufailmsg: + echoerr(context=self.cmsg.format(key=context_key(context)), + context_mark=None, + problem=self.ufailmsg(key), + problem_mark=key.mark) + return True, hadproblem + + def __getitem__(self, key): + return self.specs[self.keys[key]] + + def __setitem__(self, key, value): + return self.update(**{key: value}) From 479305e815142240c6bffc8381337017a3417e05 Mon Sep 17 00:00:00 2001 From: ZyX Date: Fri, 19 Sep 2014 22:09:32 +0400 Subject: [PATCH 04/12] Remove context_key, init_context and context_has_marks functions --- powerline/lint/__init__.py | 14 +++++------ powerline/lint/checks.py | 50 +++++++++++++++++-------------------- powerline/lint/context.py | 29 +++++++++++++++------ powerline/lint/imp.py | 10 +++----- powerline/lint/selfcheck.py | 6 ----- powerline/lint/spec.py | 12 ++++----- 6 files changed, 61 insertions(+), 60 deletions(-) diff --git a/powerline/lint/__init__.py b/powerline/lint/__init__.py index 7eb8dc71..4470c947 100644 --- a/powerline/lint/__init__.py +++ b/powerline/lint/__init__.py @@ -21,7 +21,7 @@ from powerline.lint.checks import (check_matcher_func, check_ext, check_config, check_highlight_groups, check_highlight_group, check_full_segment_data, get_all_possible_functions, check_segment_data_key) from powerline.lint.spec import Spec -from powerline.lint.context import init_context +from powerline.lint.context import Context def open_file(path): @@ -389,7 +389,7 @@ def check(paths=None, debug=False, echoerr=echoerr, require_ext=None): if used_main_spec.match( main_config, data={'configs': configs, 'lists': lists}, - context=init_context(main_config), + context=Context(main_config), echoerr=ee )[1]: hadproblem = True @@ -407,7 +407,7 @@ def check(paths=None, debug=False, echoerr=echoerr, require_ext=None): ee(problem=str(e)) hadproblem = True else: - if colors_spec.match(colors_config, context=init_context(colors_config), echoerr=ee)[1]: + if colors_spec.match(colors_config, context=Context(colors_config), echoerr=ee)[1]: hadproblem = True if lhadproblem[0]: @@ -432,7 +432,7 @@ def check(paths=None, debug=False, echoerr=echoerr, require_ext=None): hadproblem = True top_colorscheme_configs[colorscheme] = config data['colorscheme'] = colorscheme - if top_colorscheme_spec.match(config, context=init_context(config), data=data, echoerr=ee)[1]: + if top_colorscheme_spec.match(config, context=Context(config), data=data, echoerr=ee)[1]: hadproblem = True ext_colorscheme_configs = defaultdict(lambda: {}) @@ -464,7 +464,7 @@ def check(paths=None, debug=False, echoerr=echoerr, require_ext=None): spec = shell_colorscheme_spec else: spec = colorscheme_spec - if spec.match(config, context=init_context(config), data=data, echoerr=ee)[1]: + if spec.match(config, context=Context(config), data=data, echoerr=ee)[1]: hadproblem = True colorscheme_configs = {} @@ -533,7 +533,7 @@ def check(paths=None, debug=False, echoerr=echoerr, require_ext=None): else: data['theme_type'] = 'regular' spec = theme_spec - if spec.match(config, context=init_context(config), data=data, echoerr=ee)[1]: + if spec.match(config, context=Context(config), data=data, echoerr=ee)[1]: hadproblem = True for top_theme, config in top_theme_configs.items(): @@ -548,7 +548,7 @@ def check(paths=None, debug=False, echoerr=echoerr, require_ext=None): } data['theme_type'] = 'top' data['theme'] = top_theme - if top_theme_spec.match(config, context=init_context(config), data=data, echoerr=ee)[1]: + if top_theme_spec.match(config, context=Context(config), data=data, echoerr=ee)[1]: hadproblem = True return hadproblem diff --git a/powerline/lint/checks.py b/powerline/lint/checks.py index 6ce70b1b..18d2a00e 100644 --- a/powerline/lint/checks.py +++ b/powerline/lint/checks.py @@ -9,8 +9,8 @@ from powerline.lib.threaded import ThreadedSegment from powerline.lib.unicode import unicode from powerline.lint.markedjson.markedvalue import MarkedUnicode from powerline.lint.markedjson.error import DelayedEchoErr, Mark -from powerline.lint.selfcheck import havemarks, context_has_marks -from powerline.lint.context import context_key, list_sep, list_themes, new_context_item +from powerline.lint.selfcheck import havemarks +from powerline.lint.context import list_sep, list_themes, new_context_item from powerline.lint.imp import WithPath, import_function, import_segment from powerline.lint.spec import Spec from powerline.lint.inspect import getconfigargspec @@ -112,7 +112,6 @@ def check_ext(ext, data, context, echoerr): def check_config(d, theme, data, context, echoerr): - context_has_marks(context) if len(context) == 4: ext = context[-2][0] else: @@ -135,10 +134,9 @@ def check_config(d, theme, data, context, echoerr): def check_top_theme(theme, data, context, echoerr): - context_has_marks(context) havemarks(theme) if theme not in data['configs']['top_themes']: - echoerr(context='Error while checking extension configuration (key {key})'.format(key=context_key(context)), + echoerr(context='Error while checking extension configuration (key {key})'.format(key=context.key), context_mark=context[-2][0].mark, problem='failed to find top theme {0}'.format(theme), problem_mark=theme.mark) @@ -152,7 +150,7 @@ def check_color(color, data, context, echoerr): and color not in data['colors_config'].get('gradients', {})): echoerr( context='Error while checking highlight group in colorscheme (key {key})'.format( - key=context_key(context)), + key=context.key), problem='found unexistent color or gradient {0}'.format(color), problem_mark=color.mark ) @@ -228,7 +226,7 @@ def check_group(group, data, context, echoerr): if not_found: new_echoerr( context='Error while checking group definition in colorscheme (key {key})'.format( - key=context_key(context)), + key=context.key), problem='name {0} is not present in {1} {2} colorschemes: {3}'.format( group, tofind, ext, ', '.join(not_found)), problem_mark=group.mark @@ -238,13 +236,12 @@ def check_group(group, data, context, echoerr): def check_key_compatibility(segment, data, context, echoerr): - context_has_marks(context) havemarks(segment) segment_type = segment.get('type', MarkedUnicode('function', None)) havemarks(segment_type) if segment_type not in type_keys: - echoerr(context='Error while checking segments (key {key})'.format(key=context_key(context)), + echoerr(context='Error while checking segments (key {key})'.format(key=context.key), problem='found segment with unknown type {0}'.format(segment_type), problem_mark=segment_type.mark) return False, False, True @@ -255,7 +252,7 @@ def check_key_compatibility(segment, data, context, echoerr): if not ((keys - generic_keys) < type_keys[segment_type]): unknown_keys = keys - generic_keys - type_keys[segment_type] echoerr( - context='Error while checking segments (key {key})'.format(key=context_key(context)), + context='Error while checking segments (key {key})'.format(key=context.key), context_mark=context[-1][1].mark, problem='found keys not used with the current segment type: {0}'.format( list_sep.join(unknown_keys)), @@ -266,7 +263,7 @@ def check_key_compatibility(segment, data, context, echoerr): if not (keys >= required_keys[segment_type]): missing_keys = required_keys[segment_type] - keys echoerr( - context='Error while checking segments (key {key})'.format(key=context_key(context)), + context='Error while checking segments (key {key})'.format(key=context.key), context_mark=context[-1][1].mark, problem='found missing required keys: {0}'.format( list_sep.join(missing_keys)) @@ -275,7 +272,7 @@ def check_key_compatibility(segment, data, context, echoerr): if not (segment_type == 'function' or (keys & highlight_keys)): echoerr( - context='Error while checking segments (key {key})'.format(key=context_key(context)), + context='Error while checking segments (key {key})'.format(key=context.key), context_mark=context[-1][1].mark, problem=( 'found missing keys required to determine highlight group. ' @@ -295,7 +292,7 @@ def check_segment_module(module, data, context, echoerr): except ImportError as e: if echoerr.logger.level >= logging.DEBUG: echoerr.logger.exception(e) - echoerr(context='Error while checking segments (key {key})'.format(key=context_key(context)), + echoerr(context='Error while checking segments (key {key})'.format(key=context.key), problem='failed to import module {0}'.format(module), problem_mark=module.mark) return True, False, True @@ -382,7 +379,7 @@ def check_segment_function(function_name, data, context, echoerr): r = hl_exists(divider_hl_group, data, context, echoerr, allow_gradients=True) if r: echoerr( - context='Error while checking theme (key {key})'.format(key=context_key(context)), + context='Error while checking theme (key {key})'.format(key=context.key), problem=( 'found highlight group {0} not defined in the following colorschemes: {1}\n' '(Group name was obtained from function documentation.)' @@ -421,7 +418,7 @@ def check_segment_function(function_name, data, context, echoerr): ] if all(rs): echoerr( - context='Error while checking theme (key {key})'.format(key=context_key(context)), + context='Error while checking theme (key {key})'.format(key=context.key), problem=( 'found highlight groups list ({0}) with all groups not defined in some colorschemes\n' '(Group names were taken from function documentation.)' @@ -430,7 +427,7 @@ def check_segment_function(function_name, data, context, echoerr): ) for r, h in zip(rs, required_pack): echoerr( - context='Error while checking theme (key {key})'.format(key=context_key(context)), + context='Error while checking theme (key {key})'.format(key=context.key), problem='found highlight group {0} not defined in the following colorschemes: {1}'.format( h[0], list_sep.join(r)) ) @@ -439,7 +436,7 @@ def check_segment_function(function_name, data, context, echoerr): r = hl_exists(function_name, data, context, echoerr, allow_gradients=True) if r: echoerr( - context='Error while checking theme (key {key})'.format(key=context_key(context)), + context='Error while checking theme (key {key})'.format(key=context.key), problem=( 'found highlight group {0} not defined in the following colorschemes: {1}\n' '(If not specified otherwise in documentation, ' @@ -463,7 +460,7 @@ def check_segment_function(function_name, data, context, echoerr): and function_name not in data['ext_theme_configs'].get('__main__', {}).get('segment_data', {}) and not any(((function_name in theme.get('segment_data', {})) for theme in data['top_themes'].values())) ): - echoerr(context='Error while checking segments (key {key})'.format(key=context_key(context)), + echoerr(context='Error while checking segments (key {key})'.format(key=context.key), problem='found useless use of name key (such name is not present in theme/segment_data)', problem_mark=function_name.mark) @@ -501,7 +498,7 @@ def hl_exists(hl_group, data, context, echoerr, allow_gradients=False): if allow_gradients is False and not hascolor and hasgradient: echoerr( context='Error while checking highlight group in theme (key {key})'.format( - key=context_key(context)), + key=context.key), context_mark=hl_group.mark, problem='group {0} is using gradient {1} instead of a color'.format(hl_group, color), problem_mark=color.mark @@ -511,7 +508,7 @@ def hl_exists(hl_group, data, context, echoerr, allow_gradients=False): if allow_gradients == 'force' and not hadgradient: echoerr( context='Error while checking highlight group in theme (key {key})'.format( - key=context_key(context)), + key=context.key), context_mark=hl_group.mark, problem='group {0} should have at least one gradient color, but it has no'.format(hl_group), problem_mark=group_config.mark @@ -525,7 +522,7 @@ def check_highlight_group(hl_group, data, context, echoerr): r = hl_exists(hl_group, data, context, echoerr) if r: echoerr( - context='Error while checking theme (key {key})'.format(key=context_key(context)), + context='Error while checking theme (key {key})'.format(key=context.key), problem='found highlight group {0} not defined in the following colorschemes: {1}'.format( hl_group, list_sep.join(r)), problem_mark=hl_group.mark @@ -539,14 +536,14 @@ def check_highlight_groups(hl_groups, data, context, echoerr): rs = [hl_exists(hl_group, data, context, echoerr) for hl_group in hl_groups] if all(rs): echoerr( - context='Error while checking theme (key {key})'.format(key=context_key(context)), + context='Error while checking theme (key {key})'.format(key=context.key), problem='found highlight groups list ({0}) with all groups not defined in some colorschemes'.format( list_sep.join((unicode(h) for h in hl_groups))), problem_mark=hl_groups.mark ) for r, hl_group in zip(rs, hl_groups): echoerr( - context='Error while checking theme (key {key})'.format(key=context_key(context)), + context='Error while checking theme (key {key})'.format(key=context.key), problem='found highlight group {0} not defined in the following colorschemes: {1}'.format( hl_group, list_sep.join(r)), problem_mark=hl_group.mark @@ -611,14 +608,14 @@ def check_args_variant(func, args, data, context, echoerr): if required_args - present_args: echoerr( - context='Error while checking segment arguments (key {key})'.format(key=context_key(context)), + context='Error while checking segment arguments (key {key})'.format(key=context.key), context_mark=args.mark, problem='some of the required keys are missing: {0}'.format(list_sep.join(required_args - present_args)) ) hadproblem = True if not all_args >= present_args: - echoerr(context='Error while checking segment arguments (key {key})'.format(key=context_key(context)), + echoerr(context='Error while checking segment arguments (key {key})'.format(key=context.key), context_mark=args.mark, problem='found unknown keys: {0}'.format(list_sep.join(present_args - all_args)), problem_mark=next(iter(present_args - all_args)).mark) @@ -642,7 +639,6 @@ def check_args_variant(func, args, data, context, echoerr): def check_args(get_functions, args, data, context, echoerr): - context_has_marks(context) new_echoerr = DelayedEchoErr(echoerr) count = 0 hadproblem = False @@ -657,7 +653,7 @@ def check_args(get_functions, args, data, context, echoerr): if new_echoerr: new_echoerr.echo_all() else: - echoerr(context='Error while checking segment arguments (key {key})'.format(key=context_key(context)), + echoerr(context='Error while checking segment arguments (key {key})'.format(key=context.key), context_mark=context[-2][1].mark, problem='no suitable segments found') diff --git a/powerline/lint/context.py b/powerline/lint/context.py index cd1b887e..f54c31c3 100644 --- a/powerline/lint/context.py +++ b/powerline/lint/context.py @@ -5,6 +5,7 @@ import itertools from powerline.lib.unicode import unicode from powerline.lint.markedjson.markedvalue import MarkedUnicode +from powerline.lint.selfcheck import havemarks class JStr(unicode): @@ -16,14 +17,6 @@ key_sep = JStr('/') list_sep = JStr(', ') -def context_key(context): - return key_sep.join((c[0] for c in context)) - - -def init_context(config): - return ((MarkedUnicode('', config.mark), config),) - - def new_context_item(key, value): return ((value.keydict[key], value[key]),) @@ -42,3 +35,23 @@ def list_themes(data, context): return [(ext, theme) for theme in data['ext_theme_configs'].values()] else: return [(ext, context[0][1])] + + +class Context(tuple): + def __new__(cls, base, other=None): + if other is not None: + return tuple.__new__(cls, tuple.__add__(base, other)) + else: + return tuple.__new__(cls, ((MarkedUnicode('', base.mark), base),)) + + def __add__(self, arg): + assert(len(arg) == 1) + assert(type(arg) is tuple) + assert(len(arg[0]) == 2) + assert(type(arg[0]) is tuple) + havemarks(arg[0][0], arg[0][1]) + return Context.__new__(Context, self, arg) + + @property + def key(self): + return key_sep.join((c[0] for c in self)) diff --git a/powerline/lint/imp.py b/powerline/lint/imp.py index b6a154dd..7fc9a79b 100644 --- a/powerline/lint/imp.py +++ b/powerline/lint/imp.py @@ -3,8 +3,7 @@ from __future__ import (unicode_literals, division, absolute_import, print_funct import sys -from powerline.lint.selfcheck import havemarks, context_has_marks -from powerline.lint.context import context_key +from powerline.lint.selfcheck import havemarks class WithPath(object): @@ -20,26 +19,25 @@ class WithPath(object): def import_function(function_type, name, data, context, echoerr, module): - context_has_marks(context) havemarks(name, module) with WithPath(data['import_paths']): try: func = getattr(__import__(str(module), fromlist=[str(name)]), str(name)) except ImportError: - echoerr(context='Error while checking segments (key {key})'.format(key=context_key(context)), + echoerr(context='Error while checking segments (key {key})'.format(key=context.key), context_mark=name.mark, problem='failed to import module {0}'.format(module), problem_mark=module.mark) return None except AttributeError: - echoerr(context='Error while loading {0} function (key {key})'.format(function_type, key=context_key(context)), + echoerr(context='Error while loading {0} function (key {key})'.format(function_type, key=context.key), problem='failed to load function {0} from module {1}'.format(name, module), problem_mark=name.mark) return None if not callable(func): - echoerr(context='Error while checking segments (key {key})'.format(key=context_key(context)), + echoerr(context='Error while checking segments (key {key})'.format(key=context.key), context_mark=name.mark, problem='imported "function" {0} from module {1} is not callable'.format(name, module), problem_mark=module.mark) diff --git a/powerline/lint/selfcheck.py b/powerline/lint/selfcheck.py index a95f35f6..06d1fbe2 100644 --- a/powerline/lint/selfcheck.py +++ b/powerline/lint/selfcheck.py @@ -14,9 +14,3 @@ def havemarks(*args, **kwargs): havemarks(key, val, origin=(origin + '[' + unicode(i) + ']/' + unicode(key))) elif isinstance(v, list): havemarks(*v, origin=(origin + '[' + unicode(i) + ']')) - - -def context_has_marks(context): - for i, v in enumerate(context): - havemarks(v[0], origin='context key') - havemarks(v[1], origin='context val') diff --git a/powerline/lint/spec.py b/powerline/lint/spec.py index c6fe442a..89a13fa2 100644 --- a/powerline/lint/spec.py +++ b/powerline/lint/spec.py @@ -10,7 +10,7 @@ from powerline.lib.unicode import unicode from powerline.lint.markedjson.error import echoerr, DelayedEchoErr from powerline.lint.markedjson.markedvalue import MarkedUnicode from powerline.lint.selfcheck import havemarks -from powerline.lint.context import context_key, list_sep, new_context_item +from powerline.lint.context import list_sep, new_context_item class Spec(object): @@ -74,7 +74,7 @@ class Spec(object): havemarks(value) if type(value.value) not in types: echoerr( - context=self.cmsg.format(key=context_key(context)), + context=self.cmsg.format(key=context.key), context_mark=context_mark, problem='{0!r} must be a {1} instance, not {2}'.format( value, @@ -90,7 +90,7 @@ class Spec(object): havemarks(value) proceed, echo, hadproblem = func(value, data, context, echoerr) if echo and hadproblem: - echoerr(context=self.cmsg.format(key=context_key(context)), + echoerr(context=self.cmsg.format(key=context.key), context_mark=context_mark, problem=msg_func(value), problem_mark=value.mark) @@ -114,7 +114,7 @@ class Spec(object): else: proceed, echo, fhadproblem = item_func(item, data, context, echoerr) if echo and fhadproblem: - echoerr(context=self.cmsg.format(key=context_key(context) + '/list item ' + unicode(i)), + echoerr(context=self.cmsg.format(key=context.key + '/list item ' + unicode(i)), context_mark=value.mark, problem=msg_func(item), problem_mark=item.mark) @@ -332,7 +332,7 @@ class Spec(object): else: if not valspec.isoptional: hadproblem = True - echoerr(context=self.cmsg.format(key=context_key(context)), + echoerr(context=self.cmsg.format(key=context.key), context_mark=None, problem='required key is missing: {0}'.format(key), problem_mark=value.mark) @@ -362,7 +362,7 @@ class Spec(object): else: hadproblem = True if self.ufailmsg: - echoerr(context=self.cmsg.format(key=context_key(context)), + echoerr(context=self.cmsg.format(key=context.key), context_mark=None, problem=self.ufailmsg(key), problem_mark=key.mark) From 02c913f31556be8ae23d8cfae63aa6b759a806ac Mon Sep 17 00:00:00 2001 From: ZyX Date: Fri, 19 Sep 2014 22:12:20 +0400 Subject: [PATCH 05/12] Replace new_context_item() with Context.enter_key() --- powerline/lint/checks.py | 4 ++-- powerline/lint/context.py | 3 +++ powerline/lint/spec.py | 6 +++--- 3 files changed, 8 insertions(+), 5 deletions(-) diff --git a/powerline/lint/checks.py b/powerline/lint/checks.py index 18d2a00e..6d7a4e10 100644 --- a/powerline/lint/checks.py +++ b/powerline/lint/checks.py @@ -10,7 +10,7 @@ from powerline.lib.unicode import unicode from powerline.lint.markedjson.markedvalue import MarkedUnicode from powerline.lint.markedjson.error import DelayedEchoErr, Mark from powerline.lint.selfcheck import havemarks -from powerline.lint.context import list_sep, list_themes, new_context_item +from powerline.lint.context import list_sep, list_themes from powerline.lint.imp import WithPath, import_function, import_segment from powerline.lint.spec import Spec from powerline.lint.inspect import getconfigargspec @@ -627,7 +627,7 @@ def check_args_variant(func, args, data, context, echoerr): args[key], args.mark, data, - context + new_context_item(key, args), + context.enter_key(args, key), echoerr ) if khadproblem: diff --git a/powerline/lint/context.py b/powerline/lint/context.py index f54c31c3..1c188129 100644 --- a/powerline/lint/context.py +++ b/powerline/lint/context.py @@ -55,3 +55,6 @@ class Context(tuple): @property def key(self): return key_sep.join((c[0] for c in self)) + + def enter_key(self, value, key): + return self + ((value.keydict[key], value[key]),) diff --git a/powerline/lint/spec.py b/powerline/lint/spec.py index 89a13fa2..94c6f96c 100644 --- a/powerline/lint/spec.py +++ b/powerline/lint/spec.py @@ -10,7 +10,7 @@ from powerline.lib.unicode import unicode from powerline.lint.markedjson.error import echoerr, DelayedEchoErr from powerline.lint.markedjson.markedvalue import MarkedUnicode from powerline.lint.selfcheck import havemarks -from powerline.lint.context import list_sep, new_context_item +from powerline.lint.context import list_sep class Spec(object): @@ -322,7 +322,7 @@ class Spec(object): value[key], value.mark, data, - context + new_context_item(key, value), + context.enter_key(value, key), echoerr ) if mhadproblem: @@ -353,7 +353,7 @@ class Spec(object): value[key], value.mark, data, - context + new_context_item(key, value), + context.enter_key(value, key), echoerr ) if vhadproblem: From 891a9cbaa5e76b1e9382e53689aaf5b708268abf Mon Sep 17 00:00:00 2001 From: ZyX Date: Fri, 19 Sep 2014 22:20:01 +0400 Subject: [PATCH 06/12] Add Context.enter_item function for processing marked iterable items --- powerline/lint/context.py | 3 +++ powerline/lint/spec.py | 5 ++--- 2 files changed, 5 insertions(+), 3 deletions(-) diff --git a/powerline/lint/context.py b/powerline/lint/context.py index 1c188129..e9a8cef6 100644 --- a/powerline/lint/context.py +++ b/powerline/lint/context.py @@ -58,3 +58,6 @@ class Context(tuple): def enter_key(self, value, key): return self + ((value.keydict[key], value[key]),) + + def enter_item(self, name, item): + return self + ((MarkedUnicode(name, item.mark), item),) diff --git a/powerline/lint/spec.py b/powerline/lint/spec.py index 94c6f96c..b446337a 100644 --- a/powerline/lint/spec.py +++ b/powerline/lint/spec.py @@ -8,7 +8,6 @@ from copy import copy from powerline.lib.unicode import unicode from powerline.lint.markedjson.error import echoerr, DelayedEchoErr -from powerline.lint.markedjson.markedvalue import MarkedUnicode from powerline.lint.selfcheck import havemarks from powerline.lint.context import list_sep @@ -108,7 +107,7 @@ class Spec(object): item, value.mark, data, - context + ((MarkedUnicode('list item ' + unicode(i), item.mark), item),), + context.enter_item('list item ' + unicode(i), item), echoerr ) else: @@ -149,7 +148,7 @@ class Spec(object): item, value.mark, data, - context + ((MarkedUnicode('tuple item ' + unicode(i), item.mark), item),), + context.enter_item('tuple item ' + unicode(i), item), echoerr ) if ihadproblem: From b333f9ec11b7ae8d3ce874fdc220d0b460601674 Mon Sep 17 00:00:00 2001 From: ZyX Date: Fri, 19 Sep 2014 22:36:16 +0400 Subject: [PATCH 07/12] Forbid most operations for Context instances --- powerline/lint/context.py | 38 ++++++++++++++++++++++++-------------- 1 file changed, 24 insertions(+), 14 deletions(-) diff --git a/powerline/lint/context.py b/powerline/lint/context.py index e9a8cef6..986e618e 100644 --- a/powerline/lint/context.py +++ b/powerline/lint/context.py @@ -38,26 +38,36 @@ def list_themes(data, context): class Context(tuple): - def __new__(cls, base, other=None): - if other is not None: - return tuple.__new__(cls, tuple.__add__(base, other)) - else: - return tuple.__new__(cls, ((MarkedUnicode('', base.mark), base),)) + for func in dir(tuple): + if func in ('__getitem__', '__init__', '__getattribute__', '__len__', '__iter__'): + continue + exec(( + 'def {0}(self, *args, **kwargs):\n' + ' raise TypeError("{0} is not allowed for Context")' + ).format(func)) + del func - def __add__(self, arg): - assert(len(arg) == 1) - assert(type(arg) is tuple) - assert(len(arg[0]) == 2) - assert(type(arg[0]) is tuple) - havemarks(arg[0][0], arg[0][1]) - return Context.__new__(Context, self, arg) + __slots__ = () + + def __new__(cls, base, context_key=None, context_value=None): + if context_key is not None: + assert(context_value is not None) + assert(type(base) is Context) + havemarks(context_key, context_value) + return tuple.__new__(cls, tuple.__add__(base, ((context_key, context_value),))) + else: + havemarks(base) + return tuple.__new__(cls, ((MarkedUnicode('', base.mark), base),)) @property def key(self): return key_sep.join((c[0] for c in self)) def enter_key(self, value, key): - return self + ((value.keydict[key], value[key]),) + return self.enter(value.keydict[key], value[key]) def enter_item(self, name, item): - return self + ((MarkedUnicode(name, item.mark), item),) + return self.enter(MarkedUnicode(name, item.mark), item) + + def enter(self, context_key, context_value): + return Context.__new__(Context, self, context_key, context_value) From 4e3b62f54704f51aa21617291fcc4b344edfa7c5 Mon Sep 17 00:00:00 2001 From: ZyX Date: Fri, 19 Sep 2014 22:38:35 +0400 Subject: [PATCH 08/12] Remove unused new_context_item function --- powerline/lint/context.py | 4 ---- 1 file changed, 4 deletions(-) diff --git a/powerline/lint/context.py b/powerline/lint/context.py index 986e618e..f6fe5794 100644 --- a/powerline/lint/context.py +++ b/powerline/lint/context.py @@ -17,10 +17,6 @@ key_sep = JStr('/') list_sep = JStr(', ') -def new_context_item(key, value): - return ((value.keydict[key], value[key]),) - - def list_themes(data, context): theme_type = data['theme_type'] ext = data['ext'] From b781c72348eeb6cdeaec4d2a1ed21e422d0735cc Mon Sep 17 00:00:00 2001 From: ZyX Date: Fri, 19 Sep 2014 22:40:13 +0400 Subject: [PATCH 09/12] Move list_sep to powerline.lint.checks It is the only place where it is needed. --- powerline/lint/checks.py | 5 ++++- powerline/lint/context.py | 1 - powerline/lint/spec.py | 3 +-- 3 files changed, 5 insertions(+), 4 deletions(-) diff --git a/powerline/lint/checks.py b/powerline/lint/checks.py index 6d7a4e10..4443c0e5 100644 --- a/powerline/lint/checks.py +++ b/powerline/lint/checks.py @@ -10,12 +10,15 @@ from powerline.lib.unicode import unicode from powerline.lint.markedjson.markedvalue import MarkedUnicode from powerline.lint.markedjson.error import DelayedEchoErr, Mark from powerline.lint.selfcheck import havemarks -from powerline.lint.context import list_sep, list_themes +from powerline.lint.context import JStr, list_themes from powerline.lint.imp import WithPath, import_function, import_segment from powerline.lint.spec import Spec from powerline.lint.inspect import getconfigargspec +list_sep = JStr(', ') + + generic_keys = set(( 'exclude_modes', 'include_modes', 'exclude_function', 'include_function', diff --git a/powerline/lint/context.py b/powerline/lint/context.py index f6fe5794..a48a2832 100644 --- a/powerline/lint/context.py +++ b/powerline/lint/context.py @@ -14,7 +14,6 @@ class JStr(unicode): key_sep = JStr('/') -list_sep = JStr(', ') def list_themes(data, context): diff --git a/powerline/lint/spec.py b/powerline/lint/spec.py index b446337a..505a0efb 100644 --- a/powerline/lint/spec.py +++ b/powerline/lint/spec.py @@ -9,7 +9,6 @@ from copy import copy from powerline.lib.unicode import unicode from powerline.lint.markedjson.error import echoerr, DelayedEchoErr from powerline.lint.selfcheck import havemarks -from powerline.lint.context import list_sep class Spec(object): @@ -77,7 +76,7 @@ class Spec(object): context_mark=context_mark, problem='{0!r} must be a {1} instance, not {2}'.format( value, - list_sep.join((t.__name__ for t in types)), + ', '.join((t.__name__ for t in types)), type(value.value).__name__ ), problem_mark=value.mark From e9c9b6519fe7f36e442698cbd04dc193d2bba179 Mon Sep 17 00:00:00 2001 From: ZyX Date: Sat, 20 Sep 2014 00:12:00 +0400 Subject: [PATCH 10/12] Document powerline.lint.spec.Spec object --- powerline/lint/spec.py | 353 ++++++++++++++++++++++++++++++++++++++++- 1 file changed, 352 insertions(+), 1 deletion(-) diff --git a/powerline/lint/spec.py b/powerline/lint/spec.py index 505a0efb..32e88970 100644 --- a/powerline/lint/spec.py +++ b/powerline/lint/spec.py @@ -12,6 +12,51 @@ from powerline.lint.selfcheck import havemarks class Spec(object): + '''Class that describes some JSON value + + In powerline it is only used to describe JSON values stored in powerline + configuration. + + :param dict keys: + Dictionary that maps keys that may be present in the given JSON + dictionary to their descriptions. If this parameter is not empty it + implies that described value has dictionary type. Non-dictionary types + must be described using ``Spec()``: without arguments. + + .. note:: + Methods that create the specifications return ``self``, so calls to them + may be chained: ``Spec().type(unicode).re('^\w+$')``. This does not + apply to functions that *apply* specification like :py:meth`Spec.match`. + + .. note:: + Methods starting with ``check_`` return two values: first determines + whether caller should proceed on running other checks, second + determines whether there were any problems (i.e. whether error was + reported). One should not call these methods directly: there is + :py:meth:`Spec.match` method for checking values. + + .. note:: + In ``check_`` and ``match`` methods specifications are identified by + their indexes for the purpose of simplyfying :py:meth:`Spec.copy` + method. + + Some common parameters: + + ``data``: + Whatever data supplied by the first caller for checker functions. Is not + processed by :py:class:`Spec` methods in any fashion. + ``context``: + :py:class:`powerline.lint.context.Context` instance, describes context + of the value. :py:class:`Spec` methods only use its ``.key`` methods for + error messages. + ``echoerr``: + Callable that should be used to echo errors. Is supposed to take four + optional keyword arguments: ``problem``, ``problem_mark``, ``context``, + ``context_mark``. + ``value``: + Checked value. + ''' + def __init__(self, **keys): self.specs = [] self.keys = {} @@ -24,6 +69,13 @@ class Spec(object): self.update(**keys) def update(self, **keys): + '''Describe additional keys that may be present in given JSON value + + If called with some keyword arguments implies that described value is + a dictionary. If called without keyword parameters it is no-op. + + :return: self. + ''' for k, v in keys.items(): self.keys[k] = len(self.specs) self.specs.append(v) @@ -33,6 +85,14 @@ class Spec(object): return self def copy(self, copied=None): + '''Deep copy the spec + + :param dict copied: + Internal dictionary used for storing already copied values. This + parameter should not be used. + + :return: New :py:class:`Spec` object that is a deep copy of ``self``. + ''' copied = copied or {} try: return copied[id(self)] @@ -42,6 +102,15 @@ class Spec(object): return self.__class__()._update(self.__dict__, copied) def _update(self, d, copied): + '''Helper for the :py:meth:`Spec.copy` function + + Populates new instance with values taken from the old one. + + :param dict d: + ``__dict__`` of the old instance. + :param dict copied: + Storage for already copied values. + ''' self.__dict__.update(d) self.keys = copy(self.keys) self.checks = copy(self.checks) @@ -50,6 +119,30 @@ class Spec(object): return self def unknown_spec(self, keyfunc, spec): + '''Define specification for non-static keys + + This method should be used if key names cannot be determined at runtime + or if a number of keys share identical spec (in order to not repeat it). + :py:meth:`Spec.match` method processes dictionary in the given order: + + * First it tries to use specifications provided at the initialization or + by the :py:meth:`Spec.update` method. + * If no specification for given key was provided it processes + specifications from ``keyfunc`` argument in order they were supplied. + Once some key matches specification supplied second ``spec`` argument + is used to determine correctness of the value. + + :param Spec keyfunc: + :py:class:`Spec` instance or a regular function that returns two + values (the same :py:meth:`Spec.match` returns). This argument is + used to match keys that were not provided at initialization or via + :py:meth:`Spec.update`. + :param Spec spec: + :py:class:`Spec` instance that will be used to check keys matched by + ``keyfunc``. + + :return: self. + ''' if isinstance(keyfunc, Spec): self.specs.append(keyfunc) keyfunc = len(self.specs) - 1 @@ -58,10 +151,31 @@ class Spec(object): return self def unknown_msg(self, msgfunc): + '''Define message which will be used when unknown key was found + + “Unknown” is a key that was not provided at the initialization and via + :py:meth:`Spec.update` and did not match any ``keyfunc`` proided via + :py:meth:`Spec.unknown_spec`. + + :param msgfunc: + Function that takes that unknown key as an argument and returns the + message text. Text will appear at the top (start of the sentence). + + :return: self. + ''' self.ufailmsg = msgfunc return self def context_message(self, msg): + '''Define message that describes context + + :param str msg: + Message that describes context. Is written using the + :py:meth:`str.format` syntax and is expected to display keyword + parameter ``key``. + + :return: self. + ''' self.cmsg = msg for spec in self.specs: if not spec.cmsg: @@ -69,6 +183,15 @@ class Spec(object): return self def check_type(self, value, context_mark, data, context, echoerr, types): + '''Check that given value matches given type(s) + + :param tuple types: + List of accepted types. Since :py:class:`Spec` is supposed to + describe JSON values only ``dict``, ``list``, ``unicode``, ``bool``, + ``float`` and ``NoneType`` types make any sense. + + :return: proceed, hadproblem. + ''' havemarks(value) if type(value.value) not in types: echoerr( @@ -85,6 +208,34 @@ class Spec(object): return True, False def check_func(self, value, context_mark, data, context, echoerr, func, msg_func): + '''Check value using given function + + :param function func: + Callable that should accept four positional parameters: + + #. checked value, + #. ``data`` parameter with arbitrary data (supplied by top-level + caller), + #. current context and + #. function used for echoing errors. + + This callable should return three values: + + #. determines whether ``check_func`` caller should proceed + calling other checks, + #. determines whether ``check_func`` should echo error on its own + (it should be set to False if ``func`` echoes error itself) and + #. determines whether function has found some errors in the checked + value. + + :param function msg_func: + Callable that takes checked value as the only positional parameter + and returns a string that describes the problem. Only useful for + small checker functions since it is ignored when second returned + value is false. + + :return: proceed, hadproblem. + ''' havemarks(value) proceed, echo, hadproblem = func(value, data, context, echoerr) if echo and hadproblem: @@ -95,6 +246,19 @@ class Spec(object): return proceed, hadproblem def check_list(self, value, context_mark, data, context, echoerr, item_func, msg_func): + '''Check that each value in the list matches given specification + + :param function item_func: + Callable like ``func`` from :py:meth:`Spec.check_func`. Unlike + ``func`` this callable is called for each value in the list and may + be a :py:class:`Spec` object index. + :param func msg_func: + Callable like ``msg_func`` from :py:meth:`Spec.check_func`. Should + accept one problematic item and is not used for :py:class:`Spec` + object indicies in ``item_func`` method. + + :return: proceed, hadproblem. + ''' havemarks(value) i = 0 hadproblem = False @@ -124,6 +288,17 @@ class Spec(object): return True, hadproblem def check_either(self, value, context_mark, data, context, echoerr, start, end): + '''Check that given value matches one of the given specifications + + :param int start: + First specification index. + :param int end: + Specification index that is greater by 1 then last specification + index. + + This method does not give an error if any specification from + ``self.specs[start:end]`` is matched by the given value. + ''' havemarks(value) new_echoerr = DelayedEchoErr(echoerr) @@ -140,6 +315,17 @@ class Spec(object): return False, hadproblem def check_tuple(self, value, context_mark, data, context, echoerr, start, end): + '''Check that given value is a list with items matching specifications + + :param int start: + First specification index. + :param int end: + Specification index that is greater by 1 then last specification + index. + + This method checks that each item in the value list matches + specification with index ``start + item_number``. + ''' havemarks(value) hadproblem = False for (i, item, spec) in zip(itertools.count(), value, self.specs[start:end]): @@ -157,6 +343,15 @@ class Spec(object): return True, hadproblem def type(self, *args): + '''Describe value that has one of the types given in arguments + + :param args: + List of accepted types. Since :py:class:`Spec` is supposed to + describe JSON values only ``dict``, ``list``, ``unicode``, ``bool``, + ``float`` and ``NoneType`` types make any sense. + + :return: self. + ''' self.checks.append(('check_type', args)) return self @@ -177,6 +372,20 @@ class Spec(object): } def len(self, comparison, cint, msg_func=None): + '''Describe value that has given length + + :param str comparison: + Type of the comparison. Valid values: ``le``, ``lt``, ``ge``, + ``gt``, ``eq``. + :param int cint: + Integer with which length is compared. + :param function msg_func: + Function that should accept checked value and return message that + describes the problem with this value. Default value will emit + something like “length of ["foo", "bar"] is not greater then 10”. + + :return: self. + ''' cmp_func = self.cmp_funcs[comparison] msg_func = ( msg_func @@ -191,6 +400,25 @@ class Spec(object): return self def cmp(self, comparison, cint, msg_func=None): + '''Describe value that is a number or string that has given property + + :param str comparison: + Type of the comparison. Valid values: ``le``, ``lt``, ``ge``, + ``gt``, ``eq``. This argument will restrict the number or string to + emit True on the given comparison. + :param cint: + Number or string with which value is compared. Type of this + parameter affects required type of the checked value: ``str`` and + ``unicode`` types imply ``unicode`` values, ``float`` type implies + that value can be either ``int`` or ``float``, ``int`` type implies + ``int`` value and for any other type the behavior is undefined. + :param function msg_func: + Function that should accept checked value and return message that + describes the problem with this value. Default value will emit + something like “10 is not greater then 10”. + + :return: self. + ''' if type(cint) is str: self.type(unicode) elif type(cint) is float: @@ -207,6 +435,14 @@ class Spec(object): return self def unsigned(self, msg_func=None): + '''Describe unsigned integer value + + :param function msg_func: + Function that should accept checked value and return message that + describes the problem with this value. + + :return: self. + ''' self.type(int) self.checks.append(( 'check_func', @@ -216,6 +452,20 @@ class Spec(object): return self def list(self, item_func, msg_func=None): + '''Describe list with any number of elements, each matching given spec + + :param item_func: + :py:class:`Spec` instance or a callable. Check out + :py:meth:`Spec.check_list` documentation for more details. Note that + in :py:meth:`Spec.check_list` description :py:class:`Spec` instance + is replaced with its index in ``self.specs``. + :param function msg_func: + Function that should accept checked value and return message that + describes the problem with this value. Default value will emit just + “failed check”, which is rather indescriptive. + + :return: self. + ''' self.type(list) if isinstance(item_func, Spec): self.specs.append(item_func) @@ -224,6 +474,17 @@ class Spec(object): return self def tuple(self, *specs): + '''Describe list with the given number of elements, each matching corresponding spec + + :param (Spec,) specs: + List of specifications. Last element(s) in this list may be + optional. Each element in this list describes element with the same + index in the checked value. Check out :py:meth:`Spec.check_tuple` + for more details, but note that there list of specifications is + replaced with start and end indicies in ``self.specs``. + + :return: self. + ''' self.type(list) max_len = len(specs) @@ -246,10 +507,25 @@ class Spec(object): return self def func(self, func, msg_func=None): + '''Describe value that is checked by the given function + + Check out :py:meth:`Spec.check_func` documentation for more details. + ''' self.checks.append(('check_func', func, msg_func or (lambda value: 'failed check'))) return self def re(self, regex, msg_func=None): + '''Describe value that is a string that matches given regular expression + + :param str regex: + Regular expression that should be matched by the value. + :param function msg_func: + Function that should accept checked value and return message that + describes the problem with this value. Default value will emit + something like “String "xyz" does not match "[a-f]+"”. + + :return: self. + ''' self.type(unicode) compiled = re.compile(regex) msg_func = msg_func or (lambda value: 'String "{0}" does not match "{1}"'.format(value, regex)) @@ -261,6 +537,15 @@ class Spec(object): return self def ident(self, msg_func=None): + '''Describe value that is an identifier like ``foo:bar`` or ``foo`` + + :param function msg_func: + Function that should accept checked value and return message that + describes the problem with this value. Default value will emit + something like “String "xyz" is not an … identifier”. + + :return: self. + ''' msg_func = ( msg_func or (lambda value: 'String "{0}" is not an alphanumeric/underscore colon-separated identifier'.format(value)) @@ -268,6 +553,17 @@ class Spec(object): return self.re('^\w+(?::\w+)?$', msg_func) def oneof(self, collection, msg_func=None): + '''Describe value that is equal to one of the value in the collection + + :param set collection: + A collection of possible values. + :param function msg_func: + Function that should accept checked value and return message that + describes the problem with this value. Default value will emit + something like “"xyz" must be one of {"abc", "def", "ghi"}”. + + :return: self. + ''' msg_func = msg_func or (lambda value: '"{0}" must be one of {1!r}'.format(value, list(collection))) self.checks.append(( 'check_func', @@ -277,6 +573,19 @@ class Spec(object): return self def error(self, msg): + '''Describe value that must not be there + + Useful for giving more descriptive errors for some specific keys then + just “found unknown key: shutdown_event” or for forbidding certain + values when :py:meth:`Spec.unknown_spec` was used. + + :param str msg: + Message given for the offending value. It is formatted using + :py:meth:`str.format` with the only positional parameter which is + the value itself. + + :return: self. + ''' self.checks.append(( 'check_func', (lambda *args: (True, True, True)), @@ -285,20 +594,54 @@ class Spec(object): return self def either(self, *specs): + '''Describes value that matches one of the given specs + + Check out :py:meth:`Spec.check_either` method documentation for more + details, but note that there a list of specs was replaced by start and + end indicies in ``self.specs``. + + :return: self. + ''' start = len(self.specs) self.specs.extend(specs) self.checks.append(('check_either', start, len(self.specs))) return self def optional(self): + '''Mark value as optional + + Only useful for key specs in :py:meth:`Spec.__init__` and + :py:meth:`Spec.update` and some last supplied to :py:meth:`Spec.tuple`. + + :return: self. + ''' self.isoptional = True return self def required(self): + '''Mark value as required + + Only useful for key specs in :py:meth:`Spec.__init__` and + :py:meth:`Spec.update` and some last supplied to :py:meth:`Spec.tuple`. + + .. note:: + Value is required by default. This method is only useful for + altering existing specification (or rather its copy). + + :return: self. + ''' self.isoptional = False return self def match_checks(self, *args): + '''Process checks registered for the given value + + Processes only “top-level” checks: key specifications given using at the + initialization or via :py:meth:`Spec.unknown_spec` are processed by + :py:meth:`Spec.match`. + + :return: proceed, hadproblem. + ''' hadproblem = False for check in self.checks: proceed, chadproblem = getattr(self, check[0])(*(args + check[1:])) @@ -309,6 +652,10 @@ class Spec(object): return True, hadproblem def match(self, value, context_mark=None, data=None, context=(), echoerr=echoerr): + '''Check that given value matches this specification + + :return: proceed, hadproblem. + ''' havemarks(value) proceed, hadproblem = self.match_checks(value, context_mark, data, context, echoerr) if proceed: @@ -367,7 +714,11 @@ class Spec(object): return True, hadproblem def __getitem__(self, key): + '''Get specification for the given key + ''' return self.specs[self.keys[key]] def __setitem__(self, key, value): - return self.update(**{key: value}) + '''Set specification for the given key + ''' + self.update(**{key: value}) From e9e53aa85ff351e79f3f9d19716a4fea0bda3371 Mon Sep 17 00:00:00 2001 From: ZyX Date: Sat, 20 Sep 2014 00:13:27 +0400 Subject: [PATCH 11/12] Move auxilary functions before specifications --- powerline/lint/__init__.py | 20 ++++++++++---------- 1 file changed, 10 insertions(+), 10 deletions(-) diff --git a/powerline/lint/__init__.py b/powerline/lint/__init__.py index 4470c947..55f7a414 100644 --- a/powerline/lint/__init__.py +++ b/powerline/lint/__init__.py @@ -28,6 +28,16 @@ def open_file(path): return open(path, 'rb') +def generate_json_config_loader(lhadproblem): + def load_json_config(config_file_path, load=load, open_file=open_file): + with open_file(config_file_path) as config_file_fp: + r, hadproblem = load(config_file_fp) + if hadproblem: + lhadproblem[0] = True + return r + return load_json_config + + function_name_re = '^(\w+\.)*[a-zA-Z_]\w*$' @@ -279,16 +289,6 @@ theme_spec = common_theme_spec().update( ) -def generate_json_config_loader(lhadproblem): - def load_json_config(config_file_path, load=load, open_file=open_file): - with open_file(config_file_path) as config_file_fp: - r, hadproblem = load(config_file_fp) - if hadproblem: - lhadproblem[0] = True - return r - return load_json_config - - def check(paths=None, debug=False, echoerr=echoerr, require_ext=None): search_paths = paths or get_config_paths() find_config_files = generate_config_finder(lambda: search_paths) From aeb1a32a33a5638934b1679e8ca977cce61fff4b Mon Sep 17 00:00:00 2001 From: ZyX Date: Sat, 20 Sep 2014 00:17:59 +0400 Subject: [PATCH 12/12] Document powerline.lint.check function --- powerline/lint/__init__.py | 18 ++++++++++++++++++ 1 file changed, 18 insertions(+) diff --git a/powerline/lint/__init__.py b/powerline/lint/__init__.py index 55f7a414..49cb511a 100644 --- a/powerline/lint/__init__.py +++ b/powerline/lint/__init__.py @@ -290,6 +290,24 @@ theme_spec = common_theme_spec().update( def check(paths=None, debug=False, echoerr=echoerr, require_ext=None): + '''Check configuration sanity + + :param list paths: + Paths from which configuration should be loaded. + :param bool debug: + Determines whether some information useful for debugging linter should + be output. + :param function echoerr: + Function that will be used to echo the error(s). Should accept four + optional keyword parameters: ``problem`` and ``problem_mark``, and + ``context`` and ``context_mark``. + :param str require_ext: + Require configuration for some extension to be present. + + :return: + ``False`` if user configuration seems to be completely sane and ``True`` + if some problems were found. + ''' search_paths = paths or get_config_paths() find_config_files = generate_config_finder(lambda: search_paths)