api.py 6.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161
  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. from qmk.datetime import current_datetime
  8. from qmk.info import info_json
  9. from qmk.json_encoders import InfoJSONEncoder
  10. from qmk.json_schema import json_load
  11. from qmk.keyboard import find_readme, list_keyboards
  12. from qmk.keycodes import load_spec, list_versions
  13. DATA_PATH = Path('data')
  14. TEMPLATE_PATH = DATA_PATH / 'templates/api/'
  15. BUILD_API_PATH = Path('.build/api_data/')
  16. def _list_constants(output_folder):
  17. """Produce a map of available constants
  18. """
  19. ret = {}
  20. for file in (output_folder / 'constants').glob('**/*_[0-9].[0-9].[0-9].json'):
  21. name, version = file.stem.rsplit('_', 1)
  22. if name not in ret:
  23. ret[name] = []
  24. ret[name].append(version)
  25. # Ensure content is sorted
  26. for name in ret:
  27. ret[name] = sorted(ret[name])
  28. return ret
  29. def _resolve_keycode_specs(output_folder):
  30. """To make it easier for consumers, publish pre-merged spec files
  31. """
  32. for version in list_versions():
  33. overall = load_spec(version)
  34. output_file = output_folder / f'constants/keycodes_{version}.json'
  35. output_file.write_text(json.dumps(overall, indent=4), encoding='utf-8')
  36. # Purge files consumed by 'load_spec'
  37. shutil.rmtree(output_folder / 'constants/keycodes/')
  38. def _filtered_copy(src, dst):
  39. src = Path(src)
  40. dst = Path(dst)
  41. if dst.suffix == '.hjson':
  42. data = json_load(src)
  43. dst = dst.with_suffix('.json')
  44. dst.write_text(json.dumps(data, indent=4), encoding='utf-8')
  45. return dst
  46. return shutil.copy2(src, dst)
  47. def _filtered_keyboard_list():
  48. """Perform basic filtering of list_keyboards
  49. """
  50. keyboard_list = list_keyboards()
  51. if cli.args.filter:
  52. kb_list = []
  53. for keyboard_name in keyboard_list:
  54. if any(i in keyboard_name for i in cli.args.filter):
  55. kb_list.append(keyboard_name)
  56. keyboard_list = kb_list
  57. return keyboard_list
  58. @cli.argument('-n', '--dry-run', arg_only=True, action='store_true', help="Don't write the data to disk.")
  59. @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.")
  60. @cli.subcommand('Generate QMK API data', hidden=False if cli.config.user.developer else True)
  61. def generate_api(cli):
  62. """Generates the QMK API data.
  63. """
  64. v1_dir = BUILD_API_PATH / 'v1'
  65. keyboard_all_file = v1_dir / 'keyboards.json' # A massive JSON containing everything
  66. keyboard_list_file = v1_dir / 'keyboard_list.json' # A simple list of keyboard targets
  67. keyboard_aliases_file = v1_dir / 'keyboard_aliases.json' # A list of historical keyboard names and their new name
  68. keyboard_metadata_file = v1_dir / 'keyboard_metadata.json' # All the data configurator/via needs for initialization
  69. constants_metadata_file = v1_dir / 'constants_metadata.json' # Metadata for available constants
  70. usb_file = v1_dir / 'usb.json' # A mapping of USB VID/PID -> keyboard target
  71. if BUILD_API_PATH.exists():
  72. shutil.rmtree(BUILD_API_PATH)
  73. shutil.copytree(TEMPLATE_PATH, BUILD_API_PATH)
  74. shutil.copytree(DATA_PATH, v1_dir, copy_function=_filtered_copy)
  75. # Filter down when required
  76. keyboard_list = _filtered_keyboard_list()
  77. kb_all = {}
  78. usb_list = {}
  79. # Generate and write keyboard specific JSON files
  80. for keyboard_name in keyboard_list:
  81. kb_all[keyboard_name] = info_json(keyboard_name)
  82. keyboard_dir = v1_dir / 'keyboards' / keyboard_name
  83. keyboard_info = keyboard_dir / 'info.json'
  84. keyboard_readme = keyboard_dir / 'readme.md'
  85. keyboard_readme_src = find_readme(keyboard_name)
  86. keyboard_dir.mkdir(parents=True, exist_ok=True)
  87. keyboard_json = json.dumps({'last_updated': current_datetime(), 'keyboards': {keyboard_name: kb_all[keyboard_name]}})
  88. if not cli.args.dry_run:
  89. keyboard_info.write_text(keyboard_json)
  90. cli.log.debug('Wrote file %s', keyboard_info)
  91. if keyboard_readme_src:
  92. shutil.copyfile(keyboard_readme_src, keyboard_readme)
  93. cli.log.debug('Copied %s -> %s', keyboard_readme_src, keyboard_readme)
  94. if 'usb' in kb_all[keyboard_name]:
  95. usb = kb_all[keyboard_name]['usb']
  96. if 'vid' in usb and usb['vid'] not in usb_list:
  97. usb_list[usb['vid']] = {}
  98. if 'pid' in usb and usb['pid'] not in usb_list[usb['vid']]:
  99. usb_list[usb['vid']][usb['pid']] = {}
  100. if 'vid' in usb and 'pid' in usb:
  101. usb_list[usb['vid']][usb['pid']][keyboard_name] = usb
  102. # Generate data for the global files
  103. keyboard_list = sorted(kb_all)
  104. keyboard_aliases = json_load(Path('data/mappings/keyboard_aliases.hjson'))
  105. keyboard_metadata = {
  106. 'last_updated': current_datetime(),
  107. 'keyboards': keyboard_list,
  108. 'keyboard_aliases': keyboard_aliases,
  109. 'usb': usb_list,
  110. }
  111. # Feature specific handling
  112. _resolve_keycode_specs(v1_dir)
  113. # Write the global JSON files
  114. keyboard_all_json = json.dumps({'last_updated': current_datetime(), 'keyboards': kb_all}, cls=InfoJSONEncoder)
  115. usb_json = json.dumps({'last_updated': current_datetime(), 'usb': usb_list}, cls=InfoJSONEncoder)
  116. keyboard_list_json = json.dumps({'last_updated': current_datetime(), 'keyboards': keyboard_list}, cls=InfoJSONEncoder)
  117. keyboard_aliases_json = json.dumps({'last_updated': current_datetime(), 'keyboard_aliases': keyboard_aliases}, cls=InfoJSONEncoder)
  118. keyboard_metadata_json = json.dumps(keyboard_metadata, cls=InfoJSONEncoder)
  119. constants_metadata_json = json.dumps({'last_updated': current_datetime(), 'constants': _list_constants(v1_dir)})
  120. if not cli.args.dry_run:
  121. keyboard_all_file.write_text(keyboard_all_json)
  122. usb_file.write_text(usb_json)
  123. keyboard_list_file.write_text(keyboard_list_json)
  124. keyboard_aliases_file.write_text(keyboard_aliases_json)
  125. keyboard_metadata_file.write_text(keyboard_metadata_json)
  126. constants_metadata_file.write_text(constants_metadata_json)