logo

qmk_firmware

custom branch of QMK firmware git clone https://anongit.hacktivis.me/git/qmk_firmware.git

api.py (8326B)


  1. """This script automates the generation of the QMK API data.
  2. """
  3. from pathlib import Path
  4. import shutil
  5. import json
  6. from milc import cli
  7. import qmk.path
  8. from qmk.datetime import current_datetime
  9. from qmk.info import info_json
  10. from qmk.json_schema import json_load
  11. from qmk.keymap import list_keymaps
  12. from qmk.keyboard import find_readme, list_keyboards, keyboard_alias_definitions
  13. from qmk.keycodes import load_spec, list_versions, list_languages
  14. DATA_PATH = Path('data')
  15. TEMPLATE_PATH = DATA_PATH / 'templates/api/'
  16. BUILD_API_PATH = Path('.build/api_data/')
  17. def _list_constants(output_folder):
  18. """Produce a map of available constants
  19. """
  20. ret = {}
  21. for file in (output_folder / 'constants').glob('**/*_[0-9].[0-9].[0-9].json'):
  22. name, version = file.stem.rsplit('_', 1)
  23. if name not in ret:
  24. ret[name] = []
  25. ret[name].append(version)
  26. # Ensure content is sorted
  27. for name in ret:
  28. ret[name] = sorted(ret[name])
  29. return ret
  30. def _resolve_keycode_specs(output_folder):
  31. """To make it easier for consumers, publish pre-merged spec files
  32. """
  33. for version in list_versions():
  34. overall = load_spec(version)
  35. output_file = output_folder / f'constants/keycodes_{version}.json'
  36. output_file.write_text(json.dumps(overall, separators=(',', ':')), encoding='utf-8')
  37. for lang in list_languages():
  38. for version in list_versions(lang):
  39. overall = load_spec(version, lang)
  40. output_file = output_folder / f'constants/keycodes_{lang}_{version}.json'
  41. output_file.write_text(json.dumps(overall, separators=(',', ':')), encoding='utf-8')
  42. # Purge files consumed by 'load_spec'
  43. shutil.rmtree(output_folder / 'constants/keycodes/')
  44. def _filtered_copy(src, dst):
  45. src = Path(src)
  46. dst = Path(dst)
  47. if dst.suffix == '.hjson':
  48. data = json_load(src)
  49. dst = dst.with_suffix('.json')
  50. dst.write_text(json.dumps(data, separators=(',', ':')), encoding='utf-8')
  51. return dst
  52. if dst.suffix == '.jsonschema':
  53. data = json_load(src)
  54. dst.write_text(json.dumps(data), encoding='utf-8')
  55. return dst
  56. return shutil.copy2(src, dst)
  57. def _filtered_keyboard_list():
  58. """Perform basic filtering of list_keyboards
  59. """
  60. keyboard_list = list_keyboards()
  61. if cli.args.filter:
  62. kb_list = []
  63. for keyboard_name in keyboard_list:
  64. if any(i in keyboard_name for i in cli.args.filter):
  65. kb_list.append(keyboard_name)
  66. keyboard_list = kb_list
  67. return keyboard_list
  68. @cli.argument('-n', '--dry-run', arg_only=True, action='store_true', help="Don't write the data to disk.")
  69. @cli.argument('-f', '--filter', arg_only=True, action='append', default=[], help="Filter the list of keyboards based on partial name matches the supplied value. May be passed multiple times.")
  70. @cli.subcommand('Generate QMK API data', hidden=False if cli.config.user.developer else True)
  71. def generate_api(cli):
  72. """Generates the QMK API data.
  73. """
  74. v1_dir = BUILD_API_PATH / 'v1'
  75. keyboard_all_file = v1_dir / 'keyboards.json' # A massive JSON containing everything
  76. keyboard_list_file = v1_dir / 'keyboard_list.json' # A simple list of keyboard targets
  77. keyboard_aliases_file = v1_dir / 'keyboard_aliases.json' # A list of historical keyboard names and their new name
  78. keyboard_metadata_file = v1_dir / 'keyboard_metadata.json' # All the data configurator/via needs for initialization
  79. constants_metadata_file = v1_dir / 'constants_metadata.json' # Metadata for available constants
  80. usb_file = v1_dir / 'usb.json' # A mapping of USB VID/PID -> keyboard target
  81. if BUILD_API_PATH.exists():
  82. shutil.rmtree(BUILD_API_PATH)
  83. shutil.copytree(TEMPLATE_PATH, BUILD_API_PATH)
  84. shutil.copytree(DATA_PATH, v1_dir, copy_function=_filtered_copy)
  85. # Filter down when required
  86. keyboard_list = _filtered_keyboard_list()
  87. kb_all = {}
  88. usb_list = {}
  89. # Generate and write keyboard specific JSON files
  90. for keyboard_name in keyboard_list:
  91. kb_json = info_json(keyboard_name)
  92. kb_all[keyboard_name] = kb_json
  93. keyboard_dir = v1_dir / 'keyboards' / keyboard_name
  94. keyboard_info = keyboard_dir / 'info.json'
  95. keyboard_readme = keyboard_dir / 'readme.md'
  96. keyboard_readme_src = find_readme(keyboard_name)
  97. # Populate the list of JSON keymaps
  98. for keymap in list_keymaps(keyboard_name, c=False, fullpath=True):
  99. keymap_rel = qmk.path.under_qmk_firmware(keymap)
  100. if keymap_rel is None:
  101. cli.log.debug('Skipping keymap %s (not in qmk_firmware)', keymap)
  102. continue
  103. if (keymap_rel / 'keymap.c').exists():
  104. cli.log.debug('Skipping keymap %s (not pure dd keymap)', keymap)
  105. continue
  106. kb_json['keymaps'][keymap.name] = {
  107. # TODO: deprecate 'url' as consumer needs to know its potentially hjson
  108. 'url': f'https://raw.githubusercontent.com/qmk/qmk_firmware/master/{keymap_rel}/keymap.json',
  109. # Instead consumer should grab from API and not repo directly
  110. 'path': (keymap_rel / 'keymap.json').as_posix(),
  111. }
  112. keyboard_dir.mkdir(parents=True, exist_ok=True)
  113. keyboard_json = json.dumps({'last_updated': current_datetime(), 'keyboards': {keyboard_name: kb_json}}, separators=(',', ':'))
  114. if not cli.args.dry_run:
  115. keyboard_info.write_text(keyboard_json, encoding='utf-8')
  116. cli.log.debug('Wrote file %s', keyboard_info)
  117. if keyboard_readme_src:
  118. shutil.copyfile(keyboard_readme_src, keyboard_readme)
  119. cli.log.debug('Copied %s -> %s', keyboard_readme_src, keyboard_readme)
  120. # resolve keymaps as json
  121. for keymap in kb_json['keymaps']:
  122. keymap_hjson = kb_json['keymaps'][keymap]['path']
  123. keymap_json = v1_dir / keymap_hjson
  124. keymap_json.parent.mkdir(parents=True, exist_ok=True)
  125. keymap_json.write_text(json.dumps(json_load(Path(keymap_hjson)), separators=(',', ':')), encoding='utf-8')
  126. cli.log.debug('Wrote keymap %s', keymap_json)
  127. if 'usb' in kb_json:
  128. usb = kb_json['usb']
  129. if 'vid' in usb and usb['vid'] not in usb_list:
  130. usb_list[usb['vid']] = {}
  131. if 'pid' in usb and usb['pid'] not in usb_list[usb['vid']]:
  132. usb_list[usb['vid']][usb['pid']] = {}
  133. if 'vid' in usb and 'pid' in usb:
  134. usb_list[usb['vid']][usb['pid']][keyboard_name] = usb
  135. # Generate data for the global files
  136. keyboard_list = sorted(kb_all)
  137. keyboard_aliases = keyboard_alias_definitions()
  138. keyboard_metadata = {
  139. 'last_updated': current_datetime(),
  140. 'keyboards': keyboard_list,
  141. 'keyboard_aliases': keyboard_aliases,
  142. 'usb': usb_list,
  143. }
  144. # Feature specific handling
  145. _resolve_keycode_specs(v1_dir)
  146. # Write the global JSON files
  147. keyboard_all_json = json.dumps({'last_updated': current_datetime(), 'keyboards': kb_all}, separators=(',', ':'))
  148. usb_json = json.dumps({'last_updated': current_datetime(), 'usb': usb_list}, separators=(',', ':'))
  149. keyboard_list_json = json.dumps({'last_updated': current_datetime(), 'keyboards': keyboard_list}, separators=(',', ':'))
  150. keyboard_aliases_json = json.dumps({'last_updated': current_datetime(), 'keyboard_aliases': keyboard_aliases}, separators=(',', ':'))
  151. keyboard_metadata_json = json.dumps(keyboard_metadata, separators=(',', ':'))
  152. constants_metadata_json = json.dumps({'last_updated': current_datetime(), 'constants': _list_constants(v1_dir)}, separators=(',', ':'))
  153. if not cli.args.dry_run:
  154. keyboard_all_file.write_text(keyboard_all_json, encoding='utf-8')
  155. usb_file.write_text(usb_json, encoding='utf-8')
  156. keyboard_list_file.write_text(keyboard_list_json, encoding='utf-8')
  157. keyboard_aliases_file.write_text(keyboard_aliases_json, encoding='utf-8')
  158. keyboard_metadata_file.write_text(keyboard_metadata_json, encoding='utf-8')
  159. constants_metadata_file.write_text(constants_metadata_json, encoding='utf-8')