2014-09-22 20:21:00 +02:00
|
|
|
#!/usr/bin/env python3
|
2014-08-06 23:51:44 +02:00
|
|
|
# vim: ft=python fileencoding=utf-8 sts=4 sw=4 et:
|
2014-08-02 21:13:14 +02:00
|
|
|
|
2016-01-04 07:12:39 +01:00
|
|
|
# Copyright 2014-2016 Florian Bruhin (The Compiler) <mail@qutebrowser.org>
|
2014-08-06 23:51:44 +02:00
|
|
|
|
2014-05-28 16:48:19 +02:00
|
|
|
# This file is part of qutebrowser.
|
|
|
|
#
|
|
|
|
# qutebrowser is free software: you can redistribute it and/or modify
|
|
|
|
# it under the terms of the GNU General Public License as published by
|
|
|
|
# the Free Software Foundation, either version 3 of the License, or
|
|
|
|
# (at your option) any later version.
|
|
|
|
#
|
|
|
|
# qutebrowser is distributed in the hope that it will be useful,
|
|
|
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
# GNU General Public License for more details.
|
|
|
|
#
|
|
|
|
# You should have received a copy of the GNU General Public License
|
|
|
|
# along with qutebrowser. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
|
|
|
|
"""Generate asciidoc source for qutebrowser based on docstrings."""
|
|
|
|
|
|
|
|
import os
|
|
|
|
import sys
|
2014-08-05 23:20:26 +02:00
|
|
|
import html
|
2014-07-16 22:26:00 +02:00
|
|
|
import shutil
|
2014-09-07 22:57:05 +02:00
|
|
|
import os.path
|
2014-05-28 16:48:19 +02:00
|
|
|
import inspect
|
2014-07-16 22:26:00 +02:00
|
|
|
import subprocess
|
2014-08-26 19:10:14 +02:00
|
|
|
import collections
|
|
|
|
import tempfile
|
2014-09-07 20:11:38 +02:00
|
|
|
import argparse
|
2014-05-28 16:48:19 +02:00
|
|
|
|
2015-06-28 22:31:30 +02:00
|
|
|
sys.path.insert(0, os.path.join(os.path.dirname(__file__), os.pardir,
|
2016-04-27 18:30:54 +02:00
|
|
|
os.pardir))
|
2014-05-28 16:48:19 +02:00
|
|
|
|
2014-07-17 21:35:27 +02:00
|
|
|
# We import qutebrowser.app so all @cmdutils-register decorators are run.
|
2014-08-04 03:47:09 +02:00
|
|
|
import qutebrowser.app
|
2014-10-26 19:42:01 +01:00
|
|
|
from scripts import asciidoc2html, utils
|
2014-09-08 07:44:32 +02:00
|
|
|
from qutebrowser import qutebrowser
|
2015-06-12 11:30:46 +02:00
|
|
|
from qutebrowser.commands import cmdutils, command
|
2014-08-26 19:10:14 +02:00
|
|
|
from qutebrowser.config import configdata
|
2014-09-23 04:22:51 +02:00
|
|
|
from qutebrowser.utils import docutils
|
2014-05-28 16:48:19 +02:00
|
|
|
|
2014-07-17 21:35:27 +02:00
|
|
|
|
2014-09-07 20:11:38 +02:00
|
|
|
class UsageFormatter(argparse.HelpFormatter):
|
|
|
|
|
|
|
|
"""Patched HelpFormatter to include some asciidoc markup in the usage.
|
|
|
|
|
|
|
|
This does some horrible things, but the alternative would be to reimplement
|
|
|
|
argparse.HelpFormatter while copying 99% of the code :-/
|
|
|
|
"""
|
|
|
|
|
|
|
|
def _format_usage(self, usage, actions, groups, _prefix):
|
|
|
|
"""Override _format_usage to not add the 'usage:' prefix."""
|
|
|
|
return super()._format_usage(usage, actions, groups, '')
|
|
|
|
|
2015-06-12 11:30:46 +02:00
|
|
|
def _get_default_metavar_for_optional(self, action):
|
|
|
|
"""Do name transforming when getting metavar."""
|
|
|
|
return command.arg_name(action.dest.upper())
|
|
|
|
|
|
|
|
def _get_default_metavar_for_positional(self, action):
|
|
|
|
"""Do name transforming when getting metavar."""
|
|
|
|
return command.arg_name(action.dest)
|
|
|
|
|
2014-09-07 20:11:38 +02:00
|
|
|
def _metavar_formatter(self, action, default_metavar):
|
|
|
|
"""Override _metavar_formatter to add asciidoc markup to metavars.
|
|
|
|
|
|
|
|
Most code here is copied from Python 3.4's argparse.py.
|
|
|
|
"""
|
|
|
|
if action.metavar is not None:
|
|
|
|
result = "'{}'".format(action.metavar)
|
|
|
|
elif action.choices is not None:
|
|
|
|
choice_strs = [str(choice) for choice in action.choices]
|
2016-01-22 19:40:10 +01:00
|
|
|
result = ('{' + ','.join('*{}*'.format(e) for e in choice_strs) +
|
|
|
|
'}')
|
2014-09-07 20:11:38 +02:00
|
|
|
else:
|
|
|
|
result = "'{}'".format(default_metavar)
|
|
|
|
|
|
|
|
def fmt(tuple_size):
|
2014-09-08 07:44:32 +02:00
|
|
|
"""Format the result according to the tuple size."""
|
2014-09-07 20:11:38 +02:00
|
|
|
if isinstance(result, tuple):
|
|
|
|
return result
|
|
|
|
else:
|
|
|
|
return (result, ) * tuple_size
|
|
|
|
return fmt
|
|
|
|
|
|
|
|
def _format_actions_usage(self, actions, groups):
|
|
|
|
"""Override _format_actions_usage to add asciidoc markup to flags.
|
|
|
|
|
|
|
|
Because argparse.py's _format_actions_usage is very complex, we first
|
|
|
|
monkey-patch the option strings to include the asciidoc markup, then
|
|
|
|
run the original method, then undo the patching.
|
|
|
|
"""
|
|
|
|
old_option_strings = {}
|
|
|
|
for action in actions:
|
|
|
|
old_option_strings[action] = action.option_strings[:]
|
|
|
|
action.option_strings = ['*{}*'.format(s)
|
|
|
|
for s in action.option_strings]
|
|
|
|
ret = super()._format_actions_usage(actions, groups)
|
|
|
|
for action in actions:
|
|
|
|
action.option_strings = old_option_strings[action]
|
|
|
|
return ret
|
|
|
|
|
|
|
|
|
2014-07-16 22:26:00 +02:00
|
|
|
def _open_file(name, mode='w'):
|
|
|
|
"""Open a file with a preset newline/encoding mode."""
|
|
|
|
return open(name, mode, newline='\n', encoding='utf-8')
|
2014-05-28 16:48:19 +02:00
|
|
|
|
2014-07-17 21:35:27 +02:00
|
|
|
|
2014-09-08 07:44:32 +02:00
|
|
|
def _get_cmd_syntax(_name, cmd):
|
2014-09-07 20:11:38 +02:00
|
|
|
"""Get the command syntax for a command.
|
|
|
|
|
|
|
|
We monkey-patch the parser's formatter_class here to use our UsageFormatter
|
|
|
|
which adds some asciidoc markup.
|
|
|
|
"""
|
|
|
|
old_fmt_class = cmd.parser.formatter_class
|
|
|
|
cmd.parser.formatter_class = UsageFormatter
|
|
|
|
usage = cmd.parser.format_usage().rstrip()
|
|
|
|
cmd.parser.formatter_class = old_fmt_class
|
2014-09-05 07:12:35 +02:00
|
|
|
return usage
|
2014-05-28 17:47:11 +02:00
|
|
|
|
2014-05-29 00:54:30 +02:00
|
|
|
|
2014-07-16 22:26:00 +02:00
|
|
|
def _get_command_quickref(cmds):
|
2014-07-17 21:35:27 +02:00
|
|
|
"""Generate the command quick reference."""
|
2014-05-29 00:54:30 +02:00
|
|
|
out = []
|
2014-05-29 21:38:06 +02:00
|
|
|
out.append('[options="header",width="75%",cols="25%,75%"]')
|
2014-05-29 00:54:30 +02:00
|
|
|
out.append('|==============')
|
|
|
|
out.append('|Command|Description')
|
|
|
|
for name, cmd in cmds:
|
|
|
|
desc = inspect.getdoc(cmd.handler).splitlines()[0]
|
2014-09-07 20:11:38 +02:00
|
|
|
out.append('|<<{},{}>>|{}'.format(name, name, desc))
|
2014-05-29 00:54:30 +02:00
|
|
|
out.append('|==============')
|
|
|
|
return '\n'.join(out)
|
|
|
|
|
|
|
|
|
2014-07-16 22:26:00 +02:00
|
|
|
def _get_setting_quickref():
|
2014-07-17 21:35:27 +02:00
|
|
|
"""Generate the settings quick reference."""
|
2014-05-29 00:54:30 +02:00
|
|
|
out = []
|
|
|
|
for sectname, sect in configdata.DATA.items():
|
|
|
|
if not getattr(sect, 'descriptions'):
|
|
|
|
continue
|
2014-09-07 23:11:22 +02:00
|
|
|
out.append("")
|
2014-05-29 00:54:30 +02:00
|
|
|
out.append(".Quick reference for section ``{}''".format(sectname))
|
2014-05-29 21:38:06 +02:00
|
|
|
out.append('[options="header",width="75%",cols="25%,75%"]')
|
2014-05-29 00:54:30 +02:00
|
|
|
out.append('|==============')
|
|
|
|
out.append('|Setting|Description')
|
2014-07-17 21:35:27 +02:00
|
|
|
for optname, _option in sect.items():
|
2014-08-02 22:53:57 +02:00
|
|
|
desc = sect.descriptions[optname].splitlines()[0]
|
2014-09-21 21:23:16 +02:00
|
|
|
out.append('|<<{}-{},{}>>|{}'.format(
|
2014-05-29 21:08:22 +02:00
|
|
|
sectname, optname, optname, desc))
|
2014-05-29 00:54:30 +02:00
|
|
|
out.append('|==============')
|
|
|
|
return '\n'.join(out)
|
|
|
|
|
|
|
|
|
2014-07-16 22:26:00 +02:00
|
|
|
def _get_command_doc(name, cmd):
|
2014-07-17 21:35:27 +02:00
|
|
|
"""Generate the documentation for a command."""
|
2014-09-07 20:11:38 +02:00
|
|
|
output = ['[[{}]]'.format(name)]
|
2014-09-07 21:09:21 +02:00
|
|
|
output += ['=== {}'.format(name)]
|
2014-09-05 07:12:35 +02:00
|
|
|
syntax = _get_cmd_syntax(name, cmd)
|
2014-08-03 01:00:25 +02:00
|
|
|
if syntax != name:
|
|
|
|
output.append('Syntax: +:{}+'.format(syntax))
|
2014-09-07 20:11:38 +02:00
|
|
|
output.append("")
|
2014-09-23 04:22:51 +02:00
|
|
|
parser = docutils.DocstringParser(cmd.handler)
|
2014-09-05 06:38:57 +02:00
|
|
|
output.append(parser.short_desc)
|
2014-09-07 20:11:38 +02:00
|
|
|
if parser.long_desc:
|
|
|
|
output.append("")
|
|
|
|
output.append(parser.long_desc)
|
2014-09-07 21:04:39 +02:00
|
|
|
|
2015-04-06 19:48:36 +02:00
|
|
|
output += list(_get_command_doc_args(cmd, parser))
|
|
|
|
output += list(_get_command_doc_count(cmd, parser))
|
|
|
|
output += list(_get_command_doc_notes(cmd))
|
|
|
|
|
|
|
|
output.append("")
|
|
|
|
output.append("")
|
|
|
|
return '\n'.join(output)
|
|
|
|
|
|
|
|
|
|
|
|
def _get_command_doc_args(cmd, parser):
|
|
|
|
"""Get docs for the arguments of a command.
|
|
|
|
|
|
|
|
Args:
|
|
|
|
cmd: The Command to get the docs for.
|
|
|
|
parser: The DocstringParser to use.
|
|
|
|
|
|
|
|
Yield:
|
|
|
|
Strings which should be added to the docs.
|
|
|
|
"""
|
2014-09-07 21:04:39 +02:00
|
|
|
if cmd.pos_args:
|
2015-04-06 19:48:36 +02:00
|
|
|
yield ""
|
|
|
|
yield "==== positional arguments"
|
2014-09-15 00:03:39 +02:00
|
|
|
for arg, name in cmd.pos_args:
|
2014-09-07 21:04:39 +02:00
|
|
|
try:
|
2015-04-06 19:48:36 +02:00
|
|
|
yield "* +'{}'+: {}".format(name, parser.arg_descs[arg])
|
2014-09-07 21:04:39 +02:00
|
|
|
except KeyError as e:
|
|
|
|
raise KeyError("No description for arg {} of command "
|
2015-05-30 19:29:37 +02:00
|
|
|
"'{}'!".format(e, cmd.name)) from e
|
2014-09-07 21:04:39 +02:00
|
|
|
|
|
|
|
if cmd.opt_args:
|
2015-04-06 19:48:36 +02:00
|
|
|
yield ""
|
|
|
|
yield "==== optional arguments"
|
2014-09-07 21:04:39 +02:00
|
|
|
for arg, (long_flag, short_flag) in cmd.opt_args.items():
|
|
|
|
try:
|
2015-04-06 19:48:36 +02:00
|
|
|
yield '* +*{}*+, +*{}*+: {}'.format(short_flag, long_flag,
|
|
|
|
parser.arg_descs[arg])
|
2015-05-30 19:29:37 +02:00
|
|
|
except KeyError as e:
|
2014-09-07 21:04:39 +02:00
|
|
|
raise KeyError("No description for arg {} of command "
|
2015-05-30 19:29:37 +02:00
|
|
|
"'{}'!".format(e, cmd.name)) from e
|
2014-09-07 21:04:39 +02:00
|
|
|
|
2015-04-06 19:48:36 +02:00
|
|
|
|
|
|
|
def _get_command_doc_count(cmd, parser):
|
|
|
|
"""Get docs for the count of a command.
|
|
|
|
|
|
|
|
Args:
|
|
|
|
cmd: The Command to get the docs for.
|
|
|
|
parser: The DocstringParser to use.
|
|
|
|
|
|
|
|
Yield:
|
|
|
|
Strings which should be added to the docs.
|
|
|
|
"""
|
2015-04-20 19:29:29 +02:00
|
|
|
if cmd.count_arg is not None:
|
2015-04-06 19:48:36 +02:00
|
|
|
yield ""
|
|
|
|
yield "==== count"
|
2015-05-18 22:23:39 +02:00
|
|
|
try:
|
|
|
|
yield parser.arg_descs[cmd.count_arg]
|
2015-05-30 19:29:37 +02:00
|
|
|
except KeyError as e:
|
2015-05-18 22:23:39 +02:00
|
|
|
raise KeyError("No description for count arg {!r} of command "
|
2015-05-30 19:29:37 +02:00
|
|
|
"{!r}!".format(cmd.count_arg, cmd.name)) from e
|
2015-04-06 19:48:36 +02:00
|
|
|
|
|
|
|
|
|
|
|
def _get_command_doc_notes(cmd):
|
|
|
|
"""Get docs for the notes of a command.
|
|
|
|
|
|
|
|
Args:
|
|
|
|
cmd: The Command to get the docs for.
|
|
|
|
parser: The DocstringParser to use.
|
2014-09-13 00:33:15 +02:00
|
|
|
|
2015-04-06 19:48:36 +02:00
|
|
|
Yield:
|
|
|
|
Strings which should be added to the docs.
|
|
|
|
"""
|
2015-04-06 17:32:33 +02:00
|
|
|
if cmd.maxsplit is not None or cmd.no_cmd_split:
|
2015-04-06 19:48:36 +02:00
|
|
|
yield ""
|
|
|
|
yield "==== note"
|
2015-04-06 17:32:33 +02:00
|
|
|
if cmd.maxsplit is not None:
|
2015-04-06 19:48:36 +02:00
|
|
|
yield ("* This command does not split arguments after the last "
|
|
|
|
"argument and handles quotes literally.")
|
2015-04-06 17:32:33 +02:00
|
|
|
if cmd.no_cmd_split is not None:
|
2015-04-06 19:48:36 +02:00
|
|
|
yield ("* With this command, +;;+ is interpreted literally "
|
|
|
|
"instead of splitting off a second command.")
|
2014-05-28 16:48:19 +02:00
|
|
|
|
|
|
|
|
2015-02-05 08:22:50 +01:00
|
|
|
def _get_action_metavar(action, nargs=1):
|
|
|
|
"""Get the metavar to display for an argparse action.
|
|
|
|
|
|
|
|
Args:
|
|
|
|
action: The argparse action to get the metavar for.
|
|
|
|
nargs: The nargs setting for the related argument.
|
|
|
|
"""
|
2014-07-24 00:38:23 +02:00
|
|
|
if action.metavar is not None:
|
2015-02-05 08:22:50 +01:00
|
|
|
if isinstance(action.metavar, str):
|
|
|
|
elems = [action.metavar] * nargs
|
|
|
|
else:
|
|
|
|
elems = action.metavar
|
|
|
|
return ' '.join("'{}'".format(e) for e in elems)
|
2014-07-24 00:38:23 +02:00
|
|
|
elif action.choices is not None:
|
2015-12-01 07:16:32 +01:00
|
|
|
choices = ','.join(str(e) for e in action.choices)
|
2014-07-24 00:38:23 +02:00
|
|
|
return "'{{{}}}'".format(choices)
|
|
|
|
else:
|
|
|
|
return "'{}'".format(action.dest.upper())
|
|
|
|
|
|
|
|
|
|
|
|
def _format_action_args(action):
|
|
|
|
"""Get an argument string based on an argparse action."""
|
|
|
|
if action.nargs is None:
|
|
|
|
return _get_action_metavar(action)
|
|
|
|
elif action.nargs == '?':
|
|
|
|
return '[{}]'.format(_get_action_metavar(action))
|
|
|
|
elif action.nargs == '*':
|
|
|
|
return '[{mv} [{mv} ...]]'.format(mv=_get_action_metavar(action))
|
|
|
|
elif action.nargs == '+':
|
|
|
|
return '{mv} [{mv} ...]'.format(mv=_get_action_metavar(action))
|
|
|
|
elif action.nargs == '...':
|
|
|
|
return '...'
|
|
|
|
else:
|
2015-02-05 08:22:50 +01:00
|
|
|
return _get_action_metavar(action, nargs=action.nargs)
|
2014-07-24 00:38:23 +02:00
|
|
|
|
|
|
|
|
|
|
|
def _format_action(action):
|
|
|
|
"""Get an invocation string/help from an argparse action."""
|
2015-04-01 21:36:56 +02:00
|
|
|
if action.help == argparse.SUPPRESS:
|
|
|
|
return None
|
2014-07-24 00:38:23 +02:00
|
|
|
if not action.option_strings:
|
|
|
|
invocation = '*{}*::'.format(_get_action_metavar(action))
|
|
|
|
else:
|
|
|
|
parts = []
|
|
|
|
if action.nargs == 0:
|
|
|
|
# Doesn't take a value, so the syntax is -s, --long
|
|
|
|
parts += ['*{}*'.format(s) for s in action.option_strings]
|
|
|
|
else:
|
|
|
|
# Takes a value, so the syntax is -s ARGS or --long ARGS.
|
|
|
|
args_string = _format_action_args(action)
|
|
|
|
for opt in action.option_strings:
|
|
|
|
parts.append('*{}* {}'.format(opt, args_string))
|
|
|
|
invocation = ', '.join(parts) + '::'
|
2014-09-07 19:04:09 +02:00
|
|
|
return '{}\n {}\n'.format(invocation, action.help)
|
2014-07-24 00:38:23 +02:00
|
|
|
|
|
|
|
|
2014-09-07 20:11:38 +02:00
|
|
|
def generate_commands(filename):
|
2014-07-17 21:35:27 +02:00
|
|
|
"""Generate the complete commands section."""
|
2014-09-07 20:11:38 +02:00
|
|
|
with _open_file(filename) as f:
|
2014-09-07 21:09:21 +02:00
|
|
|
f.write("= Commands\n")
|
2014-09-07 20:11:38 +02:00
|
|
|
normal_cmds = []
|
|
|
|
hidden_cmds = []
|
|
|
|
debug_cmds = []
|
|
|
|
for name, cmd in cmdutils.cmd_dict.items():
|
2014-09-13 00:22:27 +02:00
|
|
|
if name in cmdutils.aliases:
|
|
|
|
continue
|
2014-09-07 20:11:38 +02:00
|
|
|
if cmd.hide:
|
|
|
|
hidden_cmds.append((name, cmd))
|
|
|
|
elif cmd.debug:
|
|
|
|
debug_cmds.append((name, cmd))
|
2015-01-15 21:29:40 +01:00
|
|
|
elif not cmd.deprecated:
|
2014-09-07 20:11:38 +02:00
|
|
|
normal_cmds.append((name, cmd))
|
|
|
|
normal_cmds.sort()
|
|
|
|
hidden_cmds.sort()
|
|
|
|
debug_cmds.sort()
|
|
|
|
f.write("\n")
|
2014-09-07 21:09:21 +02:00
|
|
|
f.write("== Normal commands\n")
|
2014-09-07 20:11:38 +02:00
|
|
|
f.write(".Quick reference\n")
|
2014-09-07 21:19:04 +02:00
|
|
|
f.write(_get_command_quickref(normal_cmds) + '\n')
|
2014-09-07 20:11:38 +02:00
|
|
|
for name, cmd in normal_cmds:
|
|
|
|
f.write(_get_command_doc(name, cmd))
|
|
|
|
f.write("\n")
|
2014-09-07 21:09:21 +02:00
|
|
|
f.write("== Hidden commands\n")
|
2014-09-07 20:11:38 +02:00
|
|
|
f.write(".Quick reference\n")
|
2014-09-07 21:19:04 +02:00
|
|
|
f.write(_get_command_quickref(hidden_cmds) + '\n')
|
2014-09-07 20:11:38 +02:00
|
|
|
for name, cmd in hidden_cmds:
|
|
|
|
f.write(_get_command_doc(name, cmd))
|
|
|
|
f.write("\n")
|
2014-09-07 21:09:21 +02:00
|
|
|
f.write("== Debugging commands\n")
|
2014-09-07 20:11:38 +02:00
|
|
|
f.write("These commands are mainly intended for debugging. They are "
|
|
|
|
"hidden if qutebrowser was started without the "
|
|
|
|
"`--debug`-flag.\n")
|
|
|
|
f.write("\n")
|
|
|
|
f.write(".Quick reference\n")
|
2014-09-07 21:19:04 +02:00
|
|
|
f.write(_get_command_quickref(debug_cmds) + '\n')
|
2014-09-07 20:11:38 +02:00
|
|
|
for name, cmd in debug_cmds:
|
|
|
|
f.write(_get_command_doc(name, cmd))
|
|
|
|
|
|
|
|
|
2015-11-30 07:11:03 +01:00
|
|
|
def _generate_setting_section(f, sectname, sect):
|
|
|
|
"""Generate documentation for a single section."""
|
|
|
|
for optname, option in sect.items():
|
|
|
|
f.write("\n")
|
|
|
|
f.write('[[{}-{}]]'.format(sectname, optname) + "\n")
|
|
|
|
f.write("=== {}".format(optname) + "\n")
|
|
|
|
f.write(sect.descriptions[optname] + "\n")
|
|
|
|
f.write("\n")
|
|
|
|
valid_values = option.typ.valid_values
|
|
|
|
if valid_values is not None:
|
|
|
|
f.write("Valid values:\n")
|
|
|
|
f.write("\n")
|
|
|
|
for val in valid_values:
|
|
|
|
try:
|
|
|
|
desc = valid_values.descriptions[val]
|
|
|
|
f.write(" * +{}+: {}".format(val, desc) + "\n")
|
|
|
|
except KeyError:
|
|
|
|
f.write(" * +{}+".format(val) + "\n")
|
|
|
|
f.write("\n")
|
|
|
|
if option.default():
|
|
|
|
f.write("Default: +pass:[{}]+\n".format(html.escape(
|
|
|
|
option.default())))
|
|
|
|
else:
|
|
|
|
f.write("Default: empty\n")
|
|
|
|
|
|
|
|
|
2014-09-07 20:11:38 +02:00
|
|
|
def generate_settings(filename):
|
2014-07-17 21:35:27 +02:00
|
|
|
"""Generate the complete settings section."""
|
2014-09-07 20:11:38 +02:00
|
|
|
with _open_file(filename) as f:
|
2014-09-07 21:09:21 +02:00
|
|
|
f.write("= Settings\n")
|
2014-09-07 20:11:38 +02:00
|
|
|
f.write(_get_setting_quickref() + "\n")
|
|
|
|
for sectname, sect in configdata.DATA.items():
|
|
|
|
f.write("\n")
|
2014-09-07 21:09:21 +02:00
|
|
|
f.write("== {}".format(sectname) + "\n")
|
2014-09-07 20:11:38 +02:00
|
|
|
f.write(configdata.SECTION_DESC[sectname] + "\n")
|
|
|
|
if not getattr(sect, 'descriptions'):
|
|
|
|
pass
|
|
|
|
else:
|
2015-11-30 07:11:03 +01:00
|
|
|
_generate_setting_section(f, sectname, sect)
|
2014-05-28 19:12:12 +02:00
|
|
|
|
|
|
|
|
2014-07-24 01:51:23 +02:00
|
|
|
def _get_authors():
|
|
|
|
"""Get a list of authors based on git commit logs."""
|
2015-10-01 18:25:46 +02:00
|
|
|
corrections = {
|
|
|
|
'binix': 'sbinix',
|
|
|
|
'Averrin': 'Alexey "Averrin" Nabrodov',
|
|
|
|
'Alexey Nabrodov': 'Alexey "Averrin" Nabrodov',
|
2015-11-17 19:30:51 +01:00
|
|
|
'Michael': 'Halfwit',
|
2015-11-18 08:12:02 +01:00
|
|
|
'Error 800': 'error800',
|
2015-11-23 11:24:40 +01:00
|
|
|
'larryhynes': 'Larry Hynes',
|
2015-11-23 13:26:32 +01:00
|
|
|
'Daniel': 'Daniel Schadt',
|
2016-05-07 21:39:19 +02:00
|
|
|
'Alexey Glushko': 'haitaka',
|
2015-10-01 18:25:46 +02:00
|
|
|
}
|
2014-07-16 22:26:00 +02:00
|
|
|
commits = subprocess.check_output(['git', 'log', '--format=%aN'])
|
2015-03-18 20:24:47 +01:00
|
|
|
authors = [corrections.get(author, author)
|
|
|
|
for author in commits.decode('utf-8').splitlines()]
|
|
|
|
cnt = collections.Counter(authors)
|
2014-12-05 22:36:35 +01:00
|
|
|
return sorted(cnt, key=lambda k: (cnt[k], k), reverse=True)
|
2014-07-24 01:51:23 +02:00
|
|
|
|
|
|
|
|
2014-09-05 07:45:47 +02:00
|
|
|
def _format_block(filename, what, data):
|
|
|
|
"""Format a block in a file.
|
2014-07-24 01:51:23 +02:00
|
|
|
|
2014-09-05 07:45:47 +02:00
|
|
|
The block is delimited by markers like these:
|
|
|
|
// QUTE_*_START
|
|
|
|
...
|
|
|
|
// QUTE_*_END
|
2014-07-24 01:51:23 +02:00
|
|
|
|
2014-09-05 07:45:47 +02:00
|
|
|
The * part is the part which should be given as 'what'.
|
|
|
|
|
|
|
|
Args:
|
|
|
|
filename: The file to change.
|
|
|
|
what: What to change (authors, options, etc.)
|
|
|
|
data; A list of strings which is the new data.
|
|
|
|
"""
|
|
|
|
what = what.upper()
|
|
|
|
oshandle, tmpname = tempfile.mkstemp()
|
|
|
|
try:
|
|
|
|
with _open_file(filename, mode='r') as infile, \
|
|
|
|
_open_file(oshandle, mode='w') as temp:
|
|
|
|
found_start = False
|
|
|
|
found_end = False
|
|
|
|
for line in infile:
|
|
|
|
if line.strip() == '// QUTE_{}_START'.format(what):
|
|
|
|
temp.write(line)
|
|
|
|
temp.write(''.join(data))
|
|
|
|
found_start = True
|
|
|
|
elif line.strip() == '// QUTE_{}_END'.format(what.upper()):
|
|
|
|
temp.write(line)
|
|
|
|
found_end = True
|
|
|
|
elif (not found_start) or found_end:
|
|
|
|
temp.write(line)
|
|
|
|
if not found_start:
|
|
|
|
raise Exception("Marker '// QUTE_{}_START' not found in "
|
|
|
|
"'{}'!".format(what, filename))
|
|
|
|
elif not found_end:
|
|
|
|
raise Exception("Marker '// QUTE_{}_END' not found in "
|
|
|
|
"'{}'!".format(what, filename))
|
2015-03-11 20:14:39 +01:00
|
|
|
except:
|
2014-09-05 07:45:47 +02:00
|
|
|
os.remove(tmpname)
|
|
|
|
raise
|
|
|
|
else:
|
|
|
|
os.remove(filename)
|
|
|
|
shutil.move(tmpname, filename)
|
2014-07-24 01:51:23 +02:00
|
|
|
|
|
|
|
|
|
|
|
def regenerate_authors(filename):
|
|
|
|
"""Re-generate the authors inside README based on the commits made."""
|
2014-09-05 07:45:47 +02:00
|
|
|
data = ['* {}\n'.format(author) for author in _get_authors()]
|
|
|
|
_format_block(filename, 'authors', data)
|
|
|
|
|
|
|
|
|
|
|
|
def regenerate_manpage(filename):
|
|
|
|
"""Update manpage OPTIONS using an argparse parser."""
|
|
|
|
# pylint: disable=protected-access
|
2014-09-08 07:44:32 +02:00
|
|
|
parser = qutebrowser.get_argparser()
|
2014-09-07 19:04:09 +02:00
|
|
|
groups = []
|
2014-09-05 07:45:47 +02:00
|
|
|
# positionals, optionals and user-defined groups
|
|
|
|
for group in parser._action_groups:
|
2014-09-07 19:04:09 +02:00
|
|
|
groupdata = []
|
|
|
|
groupdata.append('=== {}'.format(group.title))
|
2014-09-05 07:45:47 +02:00
|
|
|
if group.description is not None:
|
2014-09-07 19:04:09 +02:00
|
|
|
groupdata.append(group.description)
|
2014-09-05 07:45:47 +02:00
|
|
|
for action in group._group_actions:
|
2015-04-01 21:36:56 +02:00
|
|
|
action_data = _format_action(action)
|
|
|
|
if action_data is not None:
|
|
|
|
groupdata.append(action_data)
|
2014-09-07 19:04:09 +02:00
|
|
|
groups.append('\n'.join(groupdata))
|
|
|
|
options = '\n'.join(groups)
|
2014-09-05 07:45:47 +02:00
|
|
|
# epilog
|
|
|
|
if parser.epilog is not None:
|
2015-01-20 00:02:16 +01:00
|
|
|
options += parser.epilog
|
2014-09-05 07:45:47 +02:00
|
|
|
_format_block(filename, 'options', options)
|
2014-07-16 22:26:00 +02:00
|
|
|
|
|
|
|
|
2016-04-01 18:18:41 +02:00
|
|
|
def regenerate_cheatsheet():
|
|
|
|
"""Generate cheatsheet PNGs based on the SVG."""
|
|
|
|
files = [
|
|
|
|
('doc/img/cheatsheet-small.png', 300, 185),
|
|
|
|
('doc/img/cheatsheet-big.png', 3342, 2060),
|
|
|
|
]
|
|
|
|
|
|
|
|
for filename, x, y in files:
|
|
|
|
subprocess.check_call(['inkscape', '-e', filename, '-b', 'white',
|
|
|
|
'-w', str(x), '-h', str(y),
|
|
|
|
'misc/cheatsheet.svg'])
|
|
|
|
|
|
|
|
|
2014-09-08 07:44:32 +02:00
|
|
|
def main():
|
|
|
|
"""Regenerate all documentation."""
|
2014-10-26 19:42:01 +01:00
|
|
|
utils.change_cwd()
|
2014-10-08 22:24:05 +02:00
|
|
|
print("Generating manpage...")
|
2014-09-05 07:45:47 +02:00
|
|
|
regenerate_manpage('doc/qutebrowser.1.asciidoc')
|
2014-10-08 22:24:05 +02:00
|
|
|
print("Generating settings help...")
|
2014-09-08 12:18:54 +02:00
|
|
|
generate_settings('doc/help/settings.asciidoc')
|
2014-10-08 22:24:05 +02:00
|
|
|
print("Generating command help...")
|
2014-09-08 12:18:54 +02:00
|
|
|
generate_commands('doc/help/commands.asciidoc')
|
2016-03-16 06:22:36 +01:00
|
|
|
if '--no-authors' not in sys.argv:
|
|
|
|
print("Generating authors in README...")
|
|
|
|
regenerate_authors('README.asciidoc')
|
2016-04-01 18:18:41 +02:00
|
|
|
if '--cheatsheet' in sys.argv:
|
|
|
|
print("Regenerating cheatsheet .pngs")
|
|
|
|
regenerate_cheatsheet()
|
2014-09-16 20:08:08 +02:00
|
|
|
if '--html' in sys.argv:
|
|
|
|
asciidoc2html.main()
|
|
|
|
|
2014-09-08 07:44:32 +02:00
|
|
|
|
|
|
|
if __name__ == '__main__':
|
|
|
|
main()
|