mirror of
https://github.com/kovidgoyal/kitty.git
synced 2024-11-14 01:20:31 +03:00
302 lines
10 KiB
Python
Executable File
302 lines
10 KiB
Python
Executable File
#!./kitty/launcher/kitty +launch
|
|
# License: GPLv3 Copyright: 2022, Kovid Goyal <kovid at kovidgoyal.net>
|
|
|
|
import io
|
|
import json
|
|
import os
|
|
import sys
|
|
from contextlib import contextmanager, suppress
|
|
from functools import lru_cache
|
|
from typing import Dict, Iterator, List, Set, Tuple, Union
|
|
|
|
import kitty.constants as kc
|
|
from kittens.tui.operations import Mode
|
|
from kitty.cli import (
|
|
GoOption, go_options_for_seq, parse_option_spec, serialize_as_go_string
|
|
)
|
|
from kitty.key_encoding import config_mod_map
|
|
from kitty.key_names import (
|
|
character_key_name_aliases, functional_key_name_aliases
|
|
)
|
|
from kitty.options.types import Options
|
|
from kitty.rc.base import RemoteCommand, all_command_names, command_for_name
|
|
from kitty.rgb import color_names
|
|
|
|
changed: List[str] = []
|
|
|
|
|
|
# Utils {{{
|
|
|
|
def serialize_go_dict(x: Union[Dict[str, int], Dict[int, str], Dict[int, int], Dict[str, str]]) -> str:
|
|
ans = []
|
|
|
|
def s(x: Union[int, str]) -> str:
|
|
if isinstance(x, int):
|
|
return str(x)
|
|
return f'"{serialize_as_go_string(x)}"'
|
|
|
|
for k, v in x.items():
|
|
ans.append(f'{s(k)}: {s(v)}')
|
|
return '{' + ', '.join(ans) + '}'
|
|
|
|
|
|
def replace(template: str, **kw: str) -> str:
|
|
for k, v in kw.items():
|
|
template = template.replace(k, v)
|
|
return template
|
|
# }}}
|
|
|
|
|
|
def generate_completion_for_rc(name: str) -> None:
|
|
cmd = command_for_name(name)
|
|
if cmd.short_desc:
|
|
print(f'{name}.Description = "{serialize_as_go_string(cmd.short_desc)}"')
|
|
|
|
|
|
def generate_completions_for_kitty() -> None:
|
|
print('package completion\n')
|
|
print('func kitty(root *Command) {')
|
|
print('k := root.add_command("kitty", "")')
|
|
print('k.First_arg_may_not_be_subcommand = true')
|
|
print('k.Completion_for_arg = complete_kitty')
|
|
print('at := k.add_command("@", "Remote control")')
|
|
print('at.Description = "Control kitty using commands"')
|
|
for go_name in all_command_names():
|
|
name = go_name.replace('_', '-')
|
|
print(f'{go_name} := at.add_command("{name}", "")')
|
|
generate_completion_for_rc(go_name)
|
|
print(f'k.add_clone("@{name}", "Remote control", {go_name})')
|
|
print('}')
|
|
print('func init() {')
|
|
print('registered_exes["kitty"] = kitty')
|
|
print('}')
|
|
|
|
|
|
# rc command wrappers {{{
|
|
json_field_types: Dict[str, str] = {
|
|
'bool': 'bool', 'str': 'string', 'list.str': '[]string', 'dict.str': 'map[string]string', 'float': 'float64', 'int': 'int',
|
|
'scroll_amount': 'interface{}', 'spacing': 'interface{}', 'colors': 'interface{}',
|
|
}
|
|
|
|
|
|
def go_field_type(json_field_type: str) -> str:
|
|
q = json_field_types.get(json_field_type)
|
|
if q:
|
|
return q
|
|
if json_field_type.startswith('choices.'):
|
|
return 'string'
|
|
if '.' in json_field_type:
|
|
p, r = json_field_type.split('.', 1)
|
|
p = {'list': '[]', 'dict': 'map[string]'}[p]
|
|
return p + go_field_type(r)
|
|
raise TypeError(f'Unknown JSON field type: {json_field_type}')
|
|
|
|
|
|
class JSONField:
|
|
|
|
def __init__(self, line: str) -> None:
|
|
field_def = line.split(':', 1)[0]
|
|
self.required = False
|
|
self.field, self.field_type = field_def.split('/', 1)
|
|
if self.field.endswith('+'):
|
|
self.required = True
|
|
self.field = self.field[:-1]
|
|
self.struct_field_name = self.field[0].upper() + self.field[1:]
|
|
|
|
def go_declaration(self) -> str:
|
|
return self.struct_field_name + ' ' + go_field_type(self.field_type) + f'`json:"{self.field},omitempty"`'
|
|
|
|
|
|
def render_alias_map(alias_map: Dict[str, Tuple[str, ...]]) -> str:
|
|
if not alias_map:
|
|
return ''
|
|
amap = 'switch name {\n'
|
|
for name, aliases in alias_map.items():
|
|
for alias in aliases:
|
|
amap += f'\ncase "{alias}":\nname = "{name}"\n'
|
|
amap += '}'
|
|
return amap
|
|
|
|
|
|
def go_code_for_remote_command(name: str, cmd: RemoteCommand, template: str) -> str:
|
|
template = '\n' + template[len('//go:build exclude'):]
|
|
NO_RESPONSE_BASE = 'false'
|
|
af: List[str] = []
|
|
a = af.append
|
|
alias_map = {}
|
|
od: List[str] = []
|
|
ov: List[str] = []
|
|
option_map: Dict[str, GoOption] = {}
|
|
for o in rc_command_options(name):
|
|
field_dest = o.go_var_name.rstrip('_')
|
|
option_map[field_dest] = o
|
|
if o.aliases:
|
|
alias_map[o.long] = tuple(o.aliases)
|
|
a(o.to_flag_definition())
|
|
if o.dest in ('no_response', 'response_timeout'):
|
|
continue
|
|
od.append(f'{o.go_var_name} {o.go_type}')
|
|
ov.append(o.set_flag_value(f'options_{name}'))
|
|
jd: List[str] = []
|
|
json_fields = []
|
|
field_types: Dict[str, str] = {}
|
|
for line in cmd.protocol_spec.splitlines():
|
|
line = line.strip()
|
|
if ':' not in line:
|
|
continue
|
|
f = JSONField(line)
|
|
json_fields.append(f)
|
|
field_types[f.field] = f.field_type
|
|
jd.append(f.go_declaration())
|
|
jc: List[str] = []
|
|
handled_fields: Set[str] = set()
|
|
jc.extend(cmd.args.as_go_code(name, field_types, handled_fields))
|
|
|
|
unhandled = {}
|
|
used_options = set()
|
|
for field in json_fields:
|
|
oq = (cmd.field_to_option_map or {}).get(field.field, field.field)
|
|
if oq in option_map:
|
|
o = option_map[oq]
|
|
used_options.add(oq)
|
|
jc.append(f'payload.{field.struct_field_name} = options_{name}.{o.go_var_name}')
|
|
elif field.field in handled_fields:
|
|
pass
|
|
else:
|
|
unhandled[field.field] = field
|
|
for x in tuple(unhandled):
|
|
if x == 'match_window' and 'match' in option_map and 'match' not in used_options:
|
|
used_options.add('match')
|
|
o = option_map['match']
|
|
field = unhandled[x]
|
|
jc.append(f'payload.{field.struct_field_name} = options_{name}.{o.go_var_name}')
|
|
del unhandled[x]
|
|
if unhandled:
|
|
raise SystemExit(f'Cant map fields: {", ".join(unhandled)} for cmd: {name}')
|
|
if name != 'send_text':
|
|
unused_options = set(option_map) - used_options - {'no_response', 'response_timeout'}
|
|
if unused_options:
|
|
raise SystemExit(f'Unused options: {", ".join(unused_options)} for command: {name}')
|
|
|
|
argspec = cmd.args.spec
|
|
if argspec:
|
|
argspec = ' ' + argspec
|
|
ans = replace(
|
|
template,
|
|
CMD_NAME=name, __FILE__=__file__, CLI_NAME=name.replace('_', '-'),
|
|
SHORT_DESC=serialize_as_go_string(cmd.short_desc),
|
|
LONG_DESC=serialize_as_go_string(cmd.desc.strip()),
|
|
IS_ASYNC='true' if cmd.is_asynchronous else 'false',
|
|
NO_RESPONSE_BASE=NO_RESPONSE_BASE, ADD_FLAGS_CODE='\n'.join(af),
|
|
WAIT_TIMEOUT=str(cmd.response_timeout),
|
|
ALIAS_NORMALIZE_CODE=render_alias_map(alias_map),
|
|
OPTIONS_DECLARATION_CODE='\n'.join(od),
|
|
SET_OPTION_VALUES_CODE='\n'.join(ov),
|
|
JSON_DECLARATION_CODE='\n'.join(jd),
|
|
JSON_INIT_CODE='\n'.join(jc), ARGSPEC=argspec,
|
|
STRING_RESPONSE_IS_ERROR='true' if cmd.string_return_is_error else 'false',
|
|
STREAM_WANTED='true' if cmd.reads_streaming_data else 'false',
|
|
)
|
|
return ans
|
|
# }}}
|
|
|
|
|
|
# Constants {{{
|
|
def generate_color_names() -> str:
|
|
return 'package style\n\nvar ColorNames = map[string]RGBA{' + '\n'.join(
|
|
f'\t"{name}": RGBA{{ Red:{val.red}, Green:{val.green}, Blue:{val.blue} }},'
|
|
for name, val in color_names.items()
|
|
) + '\n}' + '\n\nvar ColorTable = [256]uint32{' + ', '.join(
|
|
f'{x}' for x in Options.color_table) + '}\n'
|
|
|
|
|
|
def load_ref_map() -> Dict[str, Dict[str, str]]:
|
|
with open('kitty/docs_ref_map_generated.h') as f:
|
|
raw = f.read()
|
|
raw = raw.split('{', 1)[1].split('}', 1)[0]
|
|
data = json.loads(bytes(bytearray(json.loads(f'[{raw}]'))))
|
|
return data # type: ignore
|
|
|
|
|
|
def generate_constants() -> str:
|
|
ref_map = load_ref_map()
|
|
dp = ", ".join(map(lambda x: f'"{serialize_as_go_string(x)}"', kc.default_pager_for_help))
|
|
return f'''\
|
|
package kitty
|
|
|
|
type VersionType struct {{
|
|
Major, Minor, Patch int
|
|
}}
|
|
const VersionString string = "{kc.str_version}"
|
|
const WebsiteBaseURL string = "{kc.website_base_url}"
|
|
const VCSRevision string = ""
|
|
const RC_ENCRYPTION_PROTOCOL_VERSION string = "{kc.RC_ENCRYPTION_PROTOCOL_VERSION}"
|
|
const IsFrozenBuild bool = false
|
|
const HandleTermiosSignals = {Mode.HANDLE_TERMIOS_SIGNALS.value[0]}
|
|
var Version VersionType = VersionType{{Major: {kc.version.major}, Minor: {kc.version.minor}, Patch: {kc.version.patch},}}
|
|
var DefaultPager []string = []string{{ {dp} }}
|
|
var FunctionalKeyNameAliases = map[string]string{serialize_go_dict(functional_key_name_aliases)}
|
|
var CharacterKeyNameAliases = map[string]string{serialize_go_dict(character_key_name_aliases)}
|
|
var ConfigModMap = map[string]uint16{serialize_go_dict(config_mod_map)}
|
|
var RefMap = map[string]string{serialize_go_dict(ref_map['ref'])}
|
|
var DocTitleMap = map[string]string{serialize_go_dict(ref_map['doc'])}
|
|
''' # }}}
|
|
|
|
|
|
# Boilerplate {{{
|
|
|
|
@contextmanager
|
|
def replace_if_needed(path: str) -> Iterator[io.StringIO]:
|
|
buf = io.StringIO()
|
|
yield buf
|
|
orig = ''
|
|
with suppress(FileNotFoundError), open(path, 'r') as f:
|
|
orig = f.read()
|
|
new = buf.getvalue()
|
|
new = f'// Code generated by {os.path.basename(__file__)}; DO NOT EDIT.\n\n' + new
|
|
if orig != new:
|
|
changed.append(path)
|
|
with open(path, 'w') as f:
|
|
f.write(new)
|
|
|
|
|
|
@lru_cache(maxsize=256)
|
|
def rc_command_options(name: str) -> Tuple[GoOption, ...]:
|
|
cmd = command_for_name(name)
|
|
return tuple(go_options_for_seq(parse_option_spec(cmd.options_spec or '\n\n')[0]))
|
|
|
|
|
|
def update_at_commands() -> None:
|
|
with open('tools/cmd/at/template.go') as f:
|
|
template = f.read()
|
|
for name in all_command_names():
|
|
cmd = command_for_name(name)
|
|
code = go_code_for_remote_command(name, cmd, template)
|
|
dest = f'tools/cmd/at/cmd_{name}_generated.go'
|
|
with replace_if_needed(dest) as f:
|
|
f.write(code)
|
|
|
|
|
|
def update_completion() -> None:
|
|
orig = sys.stdout
|
|
try:
|
|
with replace_if_needed('tools/completion/kitty_generated.go') as f:
|
|
sys.stdout = f
|
|
generate_completions_for_kitty()
|
|
finally:
|
|
sys.stdout = orig
|
|
|
|
|
|
def main() -> None:
|
|
with replace_if_needed('constants_generated.go') as f:
|
|
f.write(generate_constants())
|
|
with replace_if_needed('tools/utils/style/color-names_generated.go') as f:
|
|
f.write(generate_color_names())
|
|
update_completion()
|
|
update_at_commands()
|
|
print(json.dumps(changed, indent=2))
|
|
|
|
|
|
if __name__ == '__main__':
|
|
main() # }}}
|