#!/usr/bin/env python2 # # Copyright (c) 2014 Microsoft Corporation. All rights reserved. # Released under Apache 2.0 license as described in the file LICENSE. # # Author: Soonho Kong import os.path import sys import subprocess import string import argparse def lean_exe_name(): """ Return a lean executable name """ import platform if platform.system() == 'Windows': return "lean.exe" else: return "lean" def find_makefile(path, makefile_names): """ Find makefile in a given directory. Args: path: a string of path to look up makefile_names: a list of strings to search Return: When found, return the full path of a makefile Otherwise, return False. """ for makefile in makefile_names: makefile_pathname = os.path.join(path, makefile) if os.path.isfile(makefile_pathname): return makefile_pathname return False def find_makefile_upward(path, makefile_names): """ Strating from a given directory, search upward to find a makefile Args: path: a string of path to start the search Return: When found, return the full path of a makefile Otherwise, return False. """ makefile = find_makefile(path, makefile_names) if makefile: return makefile up = os.path.dirname(path) if up != path: return find_makefile_upward(up, makefile_names) return False def which(program): """ Lookup program in a path """ import os def is_exe(fpath): return os.path.isfile(fpath) and os.access(fpath, os.X_OK) fpath, fname = os.path.split(program) if fpath: if is_exe(program): return program else: for path in os.environ["PATH"].split(os.pathsep): path = path.strip('"') exe_file = os.path.join(path, program) if is_exe(exe_file): return exe_file return None def find_lean_exe(): """ Find a fullpath of Lean executable """ # First Look up environment variable lean=os.environ.get('LEAN') if lean: return lean # Otherwise, Look up in PATH import platform lean = which(lean_exe_name()) # Otherwise, look up the directory where lmake is at script_dir = os.path.dirname(os.path.realpath(__file__)) lean = os.path.join(script_dir, lean_exe_name()) if os.path.isfile(lean): return lean return False def find_lean_opt(): """ Return a list of lean options from env_var LEAN_OPTIONS """ lean_opt = os.environ.get('LEAN_OPTIONS') return string.split(lean_opt) if lean_opt is not None else [] def call_lean(leanfile, options): """ Call lean with options """ from collections import OrderedDict lean_exe = find_lean_exe() lean_opt = list(OrderedDict.fromkeys(find_lean_opt() + options)) subprocess.call([lean_exe] + lean_opt + [leanfile], stderr=subprocess.STDOUT) def get_lean(s): """ Given a string s, return corresponding .lean file if exists """ # xyz.olean => realpath(xyz.lean) if len(s) > 6 and s[-6:] == ".olean": leanfile = os.path.realpath(s[:-6] + ".lean") return leanfile if os.path.isfile(leanfile) else None # xyz.lean => realpath(xyz.lean) if os.path.isfile(s) and len(s) > 5 and s[-5:] == ".lean": return os.path.realpath(s) # xyz => realpath(xyz.lean) if os.path.isfile(s + ".lean"): return os.path.realpath(s + ".lean") return None def get_olean(s): """ Given a string s, return corresponding .olean file if exists """ # xyz.olean => realpath(xyz.olean) if len(s) > 6 and s[-6:] == ".olean": leanfile = s[:-6] + ".lean" return os.path.realpath(s) if os.path.isfile(leanfile) else None # xyz.lean => realpath(xyz.olean) if os.path.isfile(s) and len(s) > 5 and s[-5:] == ".lean": leanfile = os.path.realpath(s) return leanfile[:-5] + ".olean" # xyz => realpath(xyz.olean) if os.path.isfile(s + ".lean"): return os.path.realpath(s + ".olean") return None def get_target(s): """ Extract a target from an argument if we have "xyz.lean", return "xyz.olean" Otherwise, return s as it is. (it might be phony target such as 'clean') """ oleanfile = get_olean(s) return oleanfile if oleanfile is not None else s def call_makefile(directory, makefile, args, lean_options): """ Call makefile with a target generated from a given arg """ env_copy = os.environ.copy() env_copy['LEAN'] = find_lean_exe() lean_options_str = ' '.join(lean_options) if env_copy.get("LEAN_OPTIONS"): env_copy['LEAN_OPTIONS'] = env_copy['LEAN_OPTIONS'] + lean_options_str else: env_copy['LEAN_OPTIONS'] = lean_options_str cmd = ["make", "-j"] if makefile: cmd = cmd + ["--makefile", makefile] if directory: cmd = cmd + ["-C", directory] for arg in args: target = get_target(arg) cmd.append(target) subprocess.call(cmd, stderr=subprocess.STDOUT, env=env_copy) def parse_arg(argv): """ Parse arguments """ parser = argparse.ArgumentParser(description='Process arguments.') parser.add_argument('--flycheck', '-F', action='store_true', default=False, help="Use --flycheck option for Lean") parser.add_argument('--flyinfo', '-I', action='store_true', default=False, help="Use --flyinfo option for Lean") parser.add_argument('--directory', '-C', action='store', help="Change to directory dir before reading the makefiles or doing anything else.") parser.add_argument('--makefile', '-f', '--file', action='store', help="Use file as a makefile.") parser.add_argument('targets', nargs='*') args = parser.parse_args(argv) lean_options = [] if args.flycheck: lean_options.append("--flycheck") if args.flyinfo: lean_options.append("--flyinfo") return (args.directory, args.makefile, lean_options, args.targets) def main(argv=None): if argv is None: argv = sys.argv[1:] (directory, makefile, lean_options, args) = parse_arg(argv) working_dir = os.getcwd() if makefile is None and directory is None: makefile_names = ["GNUmakefile", "makefile", "Makefile"] makefile = find_makefile_upward(working_dir, makefile_names) if makefile: directory = os.path.dirname(makefile) if directory or makefile: call_makefile(directory, makefile, args, lean_options) else: for arg in args: leanfile = get_lean(arg) if leanfile and os.path.isfile(leanfile): call_lean(leanfile, lean_options) if __name__ == "__main__": sys.exit(main())