2022-11-05 11:30:09 +01:00
|
|
|
from pathlib import Path
|
|
|
|
|
2023-01-01 20:16:38 +01:00
|
|
|
from qmk.json_schema import merge_ordered_dicts, deep_update, json_load, validate
|
2022-11-05 11:30:09 +01:00
|
|
|
|
2022-12-09 01:54:52 +01:00
|
|
|
CONSTANTS_PATH = Path('data/constants/')
|
|
|
|
KEYCODES_PATH = CONSTANTS_PATH / 'keycodes'
|
|
|
|
EXTRAS_PATH = KEYCODES_PATH / 'extras'
|
|
|
|
|
|
|
|
|
|
|
|
def _find_versions(path, prefix):
|
|
|
|
ret = []
|
|
|
|
for file in path.glob(f'{prefix}_[0-9].[0-9].[0-9].hjson'):
|
|
|
|
ret.append(file.stem.split('_')[-1])
|
|
|
|
|
|
|
|
ret.sort(reverse=True)
|
|
|
|
return ret
|
|
|
|
|
|
|
|
|
2023-01-01 20:16:38 +01:00
|
|
|
def _potential_search_versions(version, lang=None):
|
|
|
|
versions = list_versions(lang)
|
|
|
|
versions.reverse()
|
2022-12-09 01:54:52 +01:00
|
|
|
|
2023-01-01 20:16:38 +01:00
|
|
|
loc = versions.index(version) + 1
|
2022-12-09 01:54:52 +01:00
|
|
|
|
2023-01-01 20:16:38 +01:00
|
|
|
return versions[:loc]
|
2022-12-09 01:54:52 +01:00
|
|
|
|
|
|
|
|
|
|
|
def _search_path(lang=None):
|
|
|
|
return EXTRAS_PATH if lang else KEYCODES_PATH
|
|
|
|
|
|
|
|
|
|
|
|
def _search_prefix(lang=None):
|
|
|
|
return f'keycodes_{lang}' if lang else 'keycodes'
|
2022-11-05 11:30:09 +01:00
|
|
|
|
|
|
|
|
2023-01-01 20:16:38 +01:00
|
|
|
def _locate_files(path, prefix, versions):
|
|
|
|
# collate files by fragment "type"
|
|
|
|
files = {'_': []}
|
|
|
|
for version in versions:
|
|
|
|
files['_'].append(path / f'{prefix}_{version}.hjson')
|
|
|
|
|
|
|
|
for file in path.glob(f'{prefix}_{version}_*.hjson'):
|
|
|
|
fragment = file.stem.replace(f'{prefix}_{version}_', '')
|
|
|
|
if fragment not in files:
|
|
|
|
files[fragment] = []
|
|
|
|
files[fragment].append(file)
|
|
|
|
|
|
|
|
return files
|
|
|
|
|
|
|
|
|
|
|
|
def _process_files(files):
|
|
|
|
# allow override within types of fragments - but not globally
|
|
|
|
spec = {}
|
|
|
|
for category in files.values():
|
|
|
|
specs = []
|
|
|
|
for file in category:
|
|
|
|
specs.append(json_load(file))
|
|
|
|
|
|
|
|
deep_update(spec, merge_ordered_dicts(specs))
|
|
|
|
|
|
|
|
return spec
|
|
|
|
|
|
|
|
|
2022-11-05 11:30:09 +01:00
|
|
|
def _validate(spec):
|
|
|
|
# first throw it to the jsonschema
|
|
|
|
validate(spec, 'qmk.keycodes.v1')
|
|
|
|
|
|
|
|
# no duplicate keycodes
|
|
|
|
keycodes = []
|
|
|
|
for value in spec['keycodes'].values():
|
|
|
|
keycodes.append(value['key'])
|
|
|
|
keycodes.extend(value.get('aliases', []))
|
|
|
|
duplicates = set([x for x in keycodes if keycodes.count(x) > 1])
|
|
|
|
if duplicates:
|
|
|
|
raise ValueError(f'Keycode spec contains duplicate keycodes! ({",".join(duplicates)})')
|
|
|
|
|
|
|
|
|
2022-12-09 01:54:52 +01:00
|
|
|
def load_spec(version, lang=None):
|
2022-11-05 11:30:09 +01:00
|
|
|
"""Build keycode data from the requested spec file
|
|
|
|
"""
|
|
|
|
if version == 'latest':
|
2022-12-09 01:54:52 +01:00
|
|
|
version = list_versions(lang)[0]
|
2022-11-05 11:30:09 +01:00
|
|
|
|
2022-12-09 01:54:52 +01:00
|
|
|
path = _search_path(lang)
|
|
|
|
prefix = _search_prefix(lang)
|
2023-01-01 20:16:38 +01:00
|
|
|
versions = _potential_search_versions(version, lang)
|
2022-11-05 11:30:09 +01:00
|
|
|
|
2023-01-01 20:16:38 +01:00
|
|
|
# Load bases + any fragments
|
|
|
|
spec = _process_files(_locate_files(path, prefix, versions))
|
2022-11-05 11:30:09 +01:00
|
|
|
|
|
|
|
# Sort?
|
2022-12-09 01:54:52 +01:00
|
|
|
spec['keycodes'] = dict(sorted(spec.get('keycodes', {}).items()))
|
2023-02-22 23:50:09 +01:00
|
|
|
spec['ranges'] = dict(sorted(spec.get('ranges', {}).items()))
|
2022-11-05 11:30:09 +01:00
|
|
|
|
|
|
|
# Validate?
|
|
|
|
_validate(spec)
|
|
|
|
|
|
|
|
return spec
|
|
|
|
|
|
|
|
|
2022-12-09 01:54:52 +01:00
|
|
|
def list_versions(lang=None):
|
2022-11-05 11:30:09 +01:00
|
|
|
"""Return available versions - sorted newest first
|
|
|
|
"""
|
2022-12-09 01:54:52 +01:00
|
|
|
path = _search_path(lang)
|
|
|
|
prefix = _search_prefix(lang)
|
|
|
|
|
|
|
|
return _find_versions(path, prefix)
|
|
|
|
|
|
|
|
|
|
|
|
def list_languages():
|
|
|
|
"""Return available languages
|
|
|
|
"""
|
|
|
|
ret = set()
|
|
|
|
for file in EXTRAS_PATH.glob('keycodes_*_[0-9].[0-9].[0-9].hjson'):
|
|
|
|
ret.add(file.stem.split('_')[1])
|
2022-11-05 11:30:09 +01:00
|
|
|
|
|
|
|
return ret
|