Add config type docstrings to settings.asciidoc

This commit is contained in:
Florian Bruhin 2017-09-15 15:36:19 +02:00
parent d7d8d191c0
commit 413c7ec1ac
4 changed files with 863 additions and 259 deletions

File diff suppressed because it is too large Load Diff

View File

@ -42,8 +42,6 @@ Config types can do different conversations:
This also validates whether the object is actually correct (type/value). This also validates whether the object is actually correct (type/value).
""" """
# FIXME:conf show the type docstrings in the documentation
import re import re
import html import html
import codecs import codecs
@ -311,7 +309,9 @@ class MappingType(BaseType):
class String(BaseType): class String(BaseType):
"""Base class for a string setting (case-insensitive). """A string value.
See the setting's valid values for more information on allowed values.
Attributes: Attributes:
minlen: Minimum length (inclusive). minlen: Minimum length (inclusive).
@ -404,7 +404,10 @@ class UniqueCharString(String):
class List(BaseType): class List(BaseType):
"""Base class for a (string-)list setting.""" """A list of values.
When setting from a string, pass a json-like list, e.g. `["one", "two"]`.
"""
_show_valtype = True _show_valtype = True
@ -474,10 +477,10 @@ class List(BaseType):
class FlagList(List): class FlagList(List):
"""Base class for a list setting that contains one or more flags. """A list of flags.
Lists with duplicate flags are invalid and each item is checked against Lists with duplicate flags are invalid. Each item is checked against
self.valid_values (if not empty). the valid values of the setting.
""" """
combinable_values = None combinable_values = None
@ -521,7 +524,11 @@ class FlagList(List):
class Bool(BaseType): class Bool(BaseType):
"""Base class for a boolean setting.""" """A boolean setting, either `true` or `false`.
When setting from a string, `1`, `yes`, `on` and `true` count as true, while
`0`, `no`, `off` and `false` count as false (case-insensitive).
"""
def __init__(self, none_ok=False): def __init__(self, none_ok=False):
super().__init__(none_ok) super().__init__(none_ok)
@ -552,7 +559,7 @@ class Bool(BaseType):
class BoolAsk(Bool): class BoolAsk(Bool):
"""A yes/no/ask question.""" """Like `Bool`, but `ask` is allowed as additional value."""
def __init__(self, none_ok=False): def __init__(self, none_ok=False):
super().__init__(none_ok) super().__init__(none_ok)
@ -950,6 +957,8 @@ class QtFont(Font):
"""A Font which gets converted to a QFont.""" """A Font which gets converted to a QFont."""
__doc__ = Font.__doc__ # for src2asciidoc.py
def to_py(self, value): def to_py(self, value):
self._basic_py_validation(value, str) self._basic_py_validation(value, str)
if not value: if not value:
@ -1011,6 +1020,9 @@ class Regex(BaseType):
"""A regular expression. """A regular expression.
When setting from `config.py`, both a string or a `re.compile(...)` object
are valid.
Attributes: Attributes:
flags: The flags to be used when a string is passed. flags: The flags to be used when a string is passed.
_regex_type: The Python type of a regex object. _regex_type: The Python type of a regex object.
@ -1077,7 +1089,10 @@ class Regex(BaseType):
class Dict(BaseType): class Dict(BaseType):
"""A dictionary of values.""" """A dictionary of values.
When setting from a string, pass a json-like dict, e.g. `{"key", "value"}`.
"""
def __init__(self, keytype, valtype, *, fixed_keys=None, def __init__(self, keytype, valtype, *, fixed_keys=None,
required_keys=None, none_ok=False): required_keys=None, none_ok=False):
@ -1214,7 +1229,7 @@ class Directory(BaseType):
class FormatString(BaseType): class FormatString(BaseType):
"""A string with '{foo}'-placeholders.""" """A string with placeholders."""
def __init__(self, fields, none_ok=False): def __init__(self, fields, none_ok=False):
super().__init__(none_ok) super().__init__(none_ok)
@ -1240,6 +1255,8 @@ class ShellCommand(List):
"""A shell command as a list. """A shell command as a list.
See the documentation for `List`.
Attributes: Attributes:
placeholder: If there should be a placeholder. placeholder: If there should be a placeholder.
""" """
@ -1261,7 +1278,7 @@ class ShellCommand(List):
class Proxy(BaseType): class Proxy(BaseType):
"""A proxy URL or special value.""" """A proxy URL, or `system`/`none`."""
def __init__(self, none_ok=False): def __init__(self, none_ok=False):
super().__init__(none_ok) super().__init__(none_ok)
@ -1332,7 +1349,7 @@ class SearchEngineUrl(BaseType):
class FuzzyUrl(BaseType): class FuzzyUrl(BaseType):
"""A single URL.""" """A URL which gets interpreted as search if needed."""
def to_py(self, value): def to_py(self, value):
from qutebrowser.utils import urlutils from qutebrowser.utils import urlutils
@ -1426,7 +1443,7 @@ class VerticalPosition(String):
class Url(BaseType): class Url(BaseType):
"""A URL.""" """A URL as a string."""
def to_py(self, value): def to_py(self, value):
self._basic_py_validation(value, str) self._basic_py_validation(value, str)

View File

@ -142,7 +142,7 @@ class DocstringParser:
"""Parse the long description in the docstring.""" """Parse the long description in the docstring."""
if line.startswith('Args:'): if line.startswith('Args:'):
self._state = self.State.arg_start self._state = self.State.arg_start
elif line.strip() == '//': elif line.strip() == '//' or line.startswith('Attributes:'):
self._state = self.State.desc_hidden self._state = self.State.desc_hidden
elif line.strip(): elif line.strip():
self._long_desc_parts.append(line.strip()) self._long_desc_parts.append(line.strip())

View File

@ -37,7 +37,7 @@ import qutebrowser.app
from scripts import asciidoc2html, utils from scripts import asciidoc2html, utils
from qutebrowser import qutebrowser, commands from qutebrowser import qutebrowser, commands
from qutebrowser.commands import cmdutils, argparser from qutebrowser.commands import cmdutils, argparser
from qutebrowser.config import configdata from qutebrowser.config import configdata, configtypes
from qutebrowser.utils import docutils, usertypes from qutebrowser.utils import docutils, usertypes
FILE_HEADER = """ FILE_HEADER = """
@ -153,6 +153,36 @@ def _get_setting_quickref():
return '\n'.join(out) return '\n'.join(out)
def _get_configtypes():
"""Get configtypes classes to document."""
predicate = lambda e: (inspect.isclass(e) and
e not in [configtypes.BaseType,
configtypes.MappingType,
# pylint: disable=protected-access
configtypes._Numeric] and
issubclass(e, configtypes.BaseType))
yield from inspect.getmembers(configtypes, predicate)
def _get_setting_types_quickref():
"""Generate the setting types quick reference."""
out = []
out.append('[[types]]')
out.append('[options="header",width="75%",cols="25%,75%"]')
out.append('|==============')
out.append('|Type|Description')
for name, typ in _get_configtypes():
parser = docutils.DocstringParser(typ)
desc = parser.short_desc
if parser.long_desc:
desc += '\n\n' + parser.long_desc
out.append('|{}|{}'.format(name, desc))
out.append('|==============')
return '\n'.join(out)
def _get_command_doc(name, cmd): def _get_command_doc(name, cmd):
"""Generate the documentation for a command.""" """Generate the documentation for a command."""
output = ['[[{}]]'.format(name)] output = ['[[{}]]'.format(name)]
@ -383,9 +413,12 @@ def _generate_setting_option(f, opt):
"""Generate documentation for a single section.""" """Generate documentation for a single section."""
f.write("\n") f.write("\n")
f.write('[[{}]]'.format(opt.name) + "\n") f.write('[[{}]]'.format(opt.name) + "\n")
f.write("== {}".format(opt.name) + "\n") f.write("=== {}".format(opt.name) + "\n")
f.write(opt.description + "\n") f.write(opt.description + "\n")
f.write("\n") f.write("\n")
f.write('Type: <<types,{typ}>>\n'.format(
typ=opt.typ.__class__.__name__))
f.write("\n")
valid_values = opt.typ.get_valid_values() valid_values = opt.typ.get_valid_values()
if valid_values is not None: if valid_values is not None:
@ -408,10 +441,13 @@ def generate_settings(filename):
configdata.init() configdata.init()
with _open_file(filename) as f: with _open_file(filename) as f:
f.write(FILE_HEADER) f.write(FILE_HEADER)
f.write("= Settings\n") f.write("= Setting reference\n\n")
f.write("== All settings\n")
f.write(_get_setting_quickref() + "\n") f.write(_get_setting_quickref() + "\n")
for opt in sorted(configdata.DATA.values()): for opt in sorted(configdata.DATA.values()):
_generate_setting_option(f, opt) _generate_setting_option(f, opt)
f.write("\n== Setting types\n")
f.write(_get_setting_types_quickref() + "\n")
def _format_block(filename, what, data): def _format_block(filename, what, data):