2019-08-25 11:58:24 -07:00
""" Compile a QMK Firmware.
You can compile a keymap already in the repo or using a QMK Configurator export .
"""
2021-05-19 15:24:46 -07:00
from subprocess import DEVNULL
2021-04-14 19:00:22 -07:00
from argcomplete . completers import FilesCompleter
2019-08-25 11:58:24 -07:00
from milc import cli
2021-02-07 21:02:51 +00:00
import qmk . path
2020-03-13 15:47:04 -07:00
from qmk . decorators import automagic_keyboard , automagic_keymap
from qmk . commands import compile_configurator_json , create_make_command , parse_configurator_json
2021-04-14 19:00:22 -07:00
from qmk . keyboard import keyboard_completer , keyboard_folder
from qmk . keymap import keymap_completer
2019-08-25 11:58:24 -07:00
2021-04-14 19:00:22 -07:00
@cli.argument ( ' filename ' , nargs = ' ? ' , arg_only = True , type = qmk . path . FileType ( ' r ' ) , completer = FilesCompleter ( ' .json ' ) , help = ' The configurator export to compile ' )
@cli.argument ( ' -kb ' , ' --keyboard ' , type = keyboard_folder , completer = keyboard_completer , help = ' The keyboard to build a firmware for. Ignored when a configurator export is supplied. ' )
@cli.argument ( ' -km ' , ' --keymap ' , completer = keymap_completer , help = ' The keymap to build a firmware for. Ignored when a configurator export is supplied. ' )
2020-03-13 15:47:04 -07:00
@cli.argument ( ' -n ' , ' --dry-run ' , arg_only = True , action = ' store_true ' , help = " Don ' t actually build, just show the make command to be run. " )
2021-08-18 01:46:59 +03:00
@cli.argument ( ' -j ' , ' --parallel ' , type = int , default = 1 , help = " Set the number of parallel make jobs; 0 means unlimited. " )
2021-01-16 15:13:04 -08:00
@cli.argument ( ' -e ' , ' --env ' , arg_only = True , action = ' append ' , default = [ ] , help = " Set a variable to be passed to make. May be passed multiple times. " )
@cli.argument ( ' -c ' , ' --clean ' , arg_only = True , action = ' store_true ' , help = " Remove object files before compiling. " )
2019-09-22 13:25:33 -07:00
@cli.subcommand ( ' Compile a QMK Firmware. ' )
2020-03-13 15:47:04 -07:00
@automagic_keyboard
@automagic_keymap
2019-09-22 13:25:33 -07:00
def compile ( cli ) :
2019-08-25 11:58:24 -07:00
""" Compile a QMK Firmware.
If a Configurator export is supplied this command will create a new keymap , overwriting an existing keymap if one exists .
2020-02-17 11:42:11 -08:00
If a keyboard and keymap are provided this command will build a firmware based on that .
2019-08-25 11:58:24 -07:00
"""
2021-01-16 15:13:04 -08:00
if cli . args . clean and not cli . args . filename and not cli . args . dry_run :
2022-06-15 22:43:54 +01:00
if cli . config . compile . keyboard and cli . config . compile . keymap :
command = create_make_command ( cli . config . compile . keyboard , cli . config . compile . keymap , ' clean ' )
cli . run ( command , capture_output = False , stdin = DEVNULL )
2021-01-16 15:13:04 -08:00
# Build the environment vars
envs = { }
for env in cli . args . env :
if ' = ' in env :
key , value = env . split ( ' = ' , 1 )
envs [ key ] = value
else :
cli . log . warning ( ' Invalid environment variable: %s ' , env )
# Determine the compile command
2020-03-13 15:47:04 -07:00
command = None
2019-08-25 11:58:24 -07:00
if cli . args . filename :
2020-03-13 15:47:04 -07:00
# If a configurator JSON was provided generate a keymap and compile it
2019-10-04 23:38:34 -07:00
user_keymap = parse_configurator_json ( cli . args . filename )
2021-01-16 15:13:04 -08:00
command = compile_configurator_json ( user_keymap , parallel = cli . config . compile . parallel , * * envs )
2019-10-04 23:38:34 -07:00
2020-02-17 11:42:11 -08:00
else :
2020-03-13 15:47:04 -07:00
if cli . config . compile . keyboard and cli . config . compile . keymap :
2020-02-17 11:42:11 -08:00
# Generate the make command for a specific keyboard/keymap.
2021-01-16 15:13:04 -08:00
command = create_make_command ( cli . config . compile . keyboard , cli . config . compile . keymap , parallel = cli . config . compile . parallel , * * envs )
2020-03-13 15:47:04 -07:00
elif not cli . config . compile . keyboard :
cli . log . error ( ' Could not determine keyboard! ' )
elif not cli . config . compile . keymap :
cli . log . error ( ' Could not determine keymap! ' )
2020-02-07 13:48:37 -07:00
2020-03-13 15:47:04 -07:00
# Compile the firmware, if we're able to
if command :
cli . log . info ( ' Compiling keymap with {fg_cyan} %s ' , ' ' . join ( command ) )
if not cli . args . dry_run :
cli . echo ( ' \n ' )
2021-01-16 15:13:04 -08:00
# FIXME(skullydazed/anyone): Remove text=False once milc 1.0.11 has had enough time to be installed everywhere.
compile = cli . run ( command , capture_output = False , text = False )
2021-01-02 09:27:35 -08:00
return compile . returncode
2019-08-25 11:58:24 -07:00
2020-03-13 15:47:04 -07:00
else :
cli . log . error ( ' You must supply a configurator export, both `--keyboard` and `--keymap`, or be in a directory for a keyboard or keymap. ' )
cli . echo ( ' usage: qmk compile [-h] [-b] [-kb KEYBOARD] [-km KEYMAP] [filename] ' )
return False