diff --git a/driver/aliases.py b/driver/aliases.py index 88dfb09ce0..28b162d206 100644 --- a/driver/aliases.py +++ b/driver/aliases.py @@ -1,9 +1,7 @@ -import os - from .util import DRIVER_DIR -PORTFOLIO_DIR = os.path.join(DRIVER_DIR, "portfolios") +PORTFOLIO_DIR = DRIVER_DIR / "portfolios" ALIASES = {} @@ -143,12 +141,11 @@ def _get_lama(pref): PORTFOLIOS = {} -for portfolio in os.listdir(PORTFOLIO_DIR): - if portfolio == "__pycache__": +for portfolio in PORTFOLIO_DIR.iterdir(): + if portfolio.name == "__pycache__": continue - name, ext = os.path.splitext(portfolio) - assert ext == ".py", portfolio - PORTFOLIOS[name.replace("_", "-")] = os.path.join(PORTFOLIO_DIR, portfolio) + assert portfolio.suffix == ".py", str(portfolio) + PORTFOLIOS[portfolio.stem.replace("_", "-")] = PORTFOLIO_DIR / portfolio def show_aliases(): diff --git a/driver/arguments.py b/driver/arguments.py index cb61f8f942..9800d1e795 100644 --- a/driver/arguments.py +++ b/driver/arguments.py @@ -1,5 +1,5 @@ import argparse -import os.path +from pathlib import Path import re import sys @@ -47,8 +47,7 @@ that exceed their time or memory limit are aborted, and the next configuration is run.""" -EXAMPLE_PORTFOLIO = os.path.relpath( - aliases.PORTFOLIOS["seq-opt-fdss-1"], start=util.REPO_ROOT_DIR) +EXAMPLE_PORTFOLIO = aliases.PORTFOLIOS["seq-opt-fdss-1"].relative_to(util.REPO_ROOT_DIR) EXAMPLES = [ ("Translate and find a plan with A* + LM-Cut:", @@ -60,7 +59,7 @@ ("Run predefined configuration (LAMA-2011) on translated task:", ["--alias", "seq-sat-lama-2011", "output.sas"]), ("Run a portfolio on a translated task:", - ["--portfolio", EXAMPLE_PORTFOLIO, + ["--portfolio", str(EXAMPLE_PORTFOLIO), "--search-time-limit", "30m", "output.sas"]), ("Run the search component in debug mode (with assertions enabled) " "and validate the resulting plan:", @@ -81,7 +80,17 @@ "--evaluator", '"hff=ff()"', "--search", '"eager_greedy([hff], preferred=[hff])"']), ] -EPILOG = """component options: + +def _format_example(description, parameters): + call_string = " ".join([Path(sys.argv[0]).name] + parameters) + return f"{description}\n{call_string}" + + +def _format_examples(examples): + return "\n\n".join(_format_example(*example) for example in examples) + + +EPILOG = f"""component options: --translate-options OPTION1 OPTION2 ... --search-options OPTION1 OPTION2 ... pass OPTION1 OPTION2 ... to specified planner component @@ -89,11 +98,11 @@ Examples: -%s -""" % "\n\n".join("%s\n%s" % (desc, " ".join([os.path.basename(sys.argv[0])] + parameters)) for desc, parameters in EXAMPLES) +{_format_examples(EXAMPLES)} +""" COMPONENTS_PLUS_OVERALL = ["translate", "search", "validate", "overall"] -DEFAULT_SAS_FILE = "output.sas" +DEFAULT_SAS_FILE = Path("output.sas") """ @@ -102,7 +111,7 @@ """ def print_usage_and_exit_with_driver_input_error(parser, msg): parser.print_usage() - returncodes.exit_with_driver_input_error("{}: error: {}".format(os.path.basename(sys.argv[0]), msg)) + returncodes.exit_with_driver_input_error(f"{Path(sys.argv[0]).name}: error: {msg}") class RawHelpFormatter(argparse.HelpFormatter): @@ -211,8 +220,8 @@ def _set_components_automatically(parser, args): def _set_components_and_inputs(parser, args): """Set args.components to the planner components to be run and set - args.translate_inputs and args.search_input to the correct input - filenames. + args.translate_inputs, args.search_input, and args.validate_inputs + to the correct input filenames. Rules: 1. If any --run-xxx option is specified, then the union @@ -239,7 +248,6 @@ def _set_components_and_inputs(parser, args): assert args.components first = args.components[0] - num_files = len(args.filenames) # When passing --help to any of the components (or -h to the # translator), we don't require input filenames and silently # swallow any that are provided. This is undocumented to avoid @@ -247,26 +255,36 @@ def _set_components_and_inputs(parser, args): if first == "translate": if "--help" in args.translate_options or "-h" in args.translate_options: args.translate_inputs = [] - elif num_files == 1: - task_file, = args.filenames - domain_file = util.find_domain_filename(task_file) - args.translate_inputs = [domain_file, task_file] - elif num_files == 2: - args.translate_inputs = args.filenames else: - print_usage_and_exit_with_driver_input_error( - parser, "translator needs one or two input files") + args.translate_inputs = _get_pddl_input_files(args, parser, "translator") elif first == "search": if "--help" in args.search_options: args.search_input = None - elif num_files == 1: - args.search_input, = args.filenames + elif len(args.filenames) == 1: + args.search_input = Path(args.filenames[0]) else: print_usage_and_exit_with_driver_input_error( parser, "search needs exactly one input file") else: assert False, first + if "validate" in args.components: + args.validate_inputs = _get_pddl_input_files(args, parser, "validate") + + +def _get_pddl_input_files(args, parser, component): + num_files = len(args.filenames) + if num_files == 1: + task = Path(args.filenames[0]) + domain = util.find_domain_path(task) + elif num_files == 2: + domain = Path(args.filenames[0]) + task = Path(args.filenames[1]) + else: + print_usage_and_exit_with_driver_input_error( + parser, f"{component} needs one or two PDDL input files") + return [domain, task] + def _set_translator_output_options(parser, args): if any("--sas-file" in opt for opt in args.translate_options): @@ -397,20 +415,20 @@ def parse_args(): help="set log level (most verbose: debug; least verbose: warning; default: %(default)s)") driver_other.add_argument( - "--plan-file", metavar="FILE", default="sas_plan", + "--plan-file", metavar="FILE", default="sas_plan", type=Path, help="write plan(s) to FILE (default: %(default)s; anytime configurations append .1, .2, ...)") driver_other.add_argument( - "--sas-file", metavar="FILE", + "--sas-file", metavar="FILE", type=Path, help="intermediate file for storing the translator output " - "(implies --keep-sas-file, default: {})".format(DEFAULT_SAS_FILE)) + f"(implies --keep-sas-file, default: {DEFAULT_SAS_FILE})") driver_other.add_argument( "--keep-sas-file", action="store_true", help="keep translator output file (implied by --sas-file, default: " "delete file if translator and search component are active)") driver_other.add_argument( - "--portfolio", metavar="FILE", + "--portfolio", metavar="FILE", type=Path, help="run a portfolio specified in FILE") driver_other.add_argument( "--portfolio-bound", metavar="VALUE", default=None, type=int, diff --git a/driver/call.py b/driver/call.py index 4b583ee088..5509bf37d0 100644 --- a/driver/call.py +++ b/driver/call.py @@ -10,9 +10,14 @@ import sys +def _replace_paths_with_strings(cmd): + return [str(x) for x in cmd] + + def print_call_settings(nick, cmd, stdin, time_limit, memory_limit): + cmd = _replace_paths_with_strings(cmd) if stdin is not None: - stdin = shlex.quote(stdin) + stdin = shlex.quote(str(stdin)) logging.info("{} stdin: {}".format(nick, stdin)) limits.print_limits(nick, time_limit, memory_limit) @@ -47,6 +52,7 @@ def fail(exception, exitcode): def check_call(nick, cmd, stdin=None, time_limit=None, memory_limit=None): + cmd = _replace_paths_with_strings(cmd) print_call_settings(nick, cmd, stdin, time_limit, memory_limit) kwargs = {"preexec_fn": _get_preexec_function(time_limit, memory_limit)} @@ -60,6 +66,7 @@ def check_call(nick, cmd, stdin=None, time_limit=None, memory_limit=None): def get_error_output_and_returncode(nick, cmd, time_limit=None, memory_limit=None): + cmd = _replace_paths_with_strings(cmd) print_call_settings(nick, cmd, None, time_limit, memory_limit) preexec_fn = _get_preexec_function(time_limit, memory_limit) diff --git a/driver/cleanup.py b/driver/cleanup.py index 7046ad529b..a9f6163661 100644 --- a/driver/cleanup.py +++ b/driver/cleanup.py @@ -1,17 +1,5 @@ -from itertools import count -import os - -def _try_remove(f): - try: - os.remove(f) - except OSError: - return False - return True +from .plan_manager import PlanManager def cleanup_temporary_files(args): - _try_remove(args.sas_file) - _try_remove(args.plan_file) - - for i in count(1): - if not _try_remove("%s.%s" % (args.plan_file, i)): - break + args.sas_file.unlink(missing_ok=True) + PlanManager(args.plan_file).delete_existing_plans() diff --git a/driver/main.py b/driver/main.py index 7c41781b8c..e9ff38a51b 100644 --- a/driver/main.py +++ b/driver/main.py @@ -1,5 +1,4 @@ import logging -import os import sys from . import aliases @@ -16,7 +15,7 @@ def main(): logging.basicConfig(level=getattr(logging, args.log_level.upper()), format="%(levelname)-8s %(message)s", stream=sys.stdout) - logging.debug("processed args: %s" % args) + logging.debug(f"processed args: {args}") if args.version: print(__version__) @@ -40,16 +39,16 @@ def main(): elif component == "search": (exitcode, continue_execution) = run_components.run_search(args) if not args.keep_sas_file: - print("Remove intermediate file {}".format(args.sas_file)) - os.remove(args.sas_file) + print(f"Remove intermediate file {args.sas_file}") + args.sas_file.unlink() elif component == "validate": (exitcode, continue_execution) = run_components.run_validate(args) else: - assert False, "Error: unhandled component: {}".format(component) - print("{component} exit code: {exitcode}".format(**locals())) + assert False, f"Error: unhandled component: {component}" + print(f"{component} exit code: {exitcode}") print() if not continue_execution: - print("Driver aborting after {}".format(component)) + print(f"Driver aborting after {component}") break try: diff --git a/driver/plan_manager.py b/driver/plan_manager.py index bca0277625..58937eacab 100644 --- a/driver/plan_manager.py +++ b/driver/plan_manager.py @@ -1,6 +1,5 @@ import itertools -import os -import os.path +from pathlib import Path import re from . import returncodes @@ -9,20 +8,19 @@ _PLAN_INFO_REGEX = re.compile(r"; cost = (\d+) \((unit cost|general cost)\)\n") -def _read_last_line(filename): +def _read_last_line(path: Path): line = None - with open(filename) as input_file: + with path.open() as input_file: for line in input_file: pass return line -def _parse_plan(plan_filename): +def _parse_plan(plan_path: Path): """Parse a plan file and return a pair (cost, problem_type) summarizing the salient information. Return (None, None) for incomplete plans.""" - - last_line = _read_last_line(plan_filename) or "" + last_line = _read_last_line(plan_path) or "" match = _PLAN_INFO_REGEX.match(last_line) if match: return int(match.group(1)), match.group(2) @@ -31,7 +29,7 @@ def _parse_plan(plan_filename): class PlanManager: - def __init__(self, plan_prefix, portfolio_bound=None, single_plan=False): + def __init__(self, plan_prefix: Path, portfolio_bound=None, single_plan=False): self._plan_prefix = plan_prefix self._plan_costs = [] self._problem_type = None @@ -73,23 +71,22 @@ def process_new_plans(self): Read newly generated plans and store the relevant information. If the last plan file is incomplete, delete it. """ - had_incomplete_plan = False for counter in itertools.count(self.get_plan_counter() + 1): - plan_filename = self._get_plan_file(counter) + plan_path = self._get_plan_path(counter) def bogus_plan(msg): - returncodes.exit_with_driver_critical_error("%s: %s" % (plan_filename, msg)) - if not os.path.exists(plan_filename): + returncodes.exit_with_driver_critical_error(f"{str(plan_path)}: {msg}") + if not plan_path.exists(): break if had_incomplete_plan: bogus_plan("plan found after incomplete plan") - cost, problem_type = _parse_plan(plan_filename) + cost, problem_type = _parse_plan(plan_path) if cost is None: had_incomplete_plan = True - print("%s is incomplete. Deleted the file." % plan_filename) - os.remove(plan_filename) + print(f"{plan_path} is incomplete. Deleted the file.") + plan_path.unlink() else: - print("plan manager: found new plan with cost %d" % cost) + print(f"plan manager: found new plan with cost {cost}") if self._problem_type is None: # This is the first plan we found. self._problem_type = problem_type @@ -103,20 +100,20 @@ def bogus_plan(msg): def get_existing_plans(self): """Yield all plans that match the given plan prefix.""" - if os.path.exists(self._plan_prefix): + if self._plan_prefix.exists(): yield self._plan_prefix for counter in itertools.count(start=1): - plan_filename = self._get_plan_file(counter) - if os.path.exists(plan_filename): - yield plan_filename + plan_path = self._get_plan_path(counter) + if plan_path.exists(): + yield plan_path else: break def delete_existing_plans(self): """Delete all plans that match the given plan prefix.""" for plan in self.get_existing_plans(): - os.remove(plan) + plan.unlink() - def _get_plan_file(self, number): - return "%s.%d" % (self._plan_prefix, number) + def _get_plan_path(self, number): + return Path(f"{(self._plan_prefix)}.{number}") diff --git a/driver/portfolio_runner.py b/driver/portfolio_runner.py index 011ecff050..746020596c 100644 --- a/driver/portfolio_runner.py +++ b/driver/portfolio_runner.py @@ -15,6 +15,7 @@ __all__ = ["run"] +from pathlib import Path import subprocess import sys @@ -184,17 +185,16 @@ def can_change_cost_type(args): return any("S_COST_TYPE" in part or "H_COST_TRANSFORM" in part for part in args) -def get_portfolio_attributes(portfolio): +def get_portfolio_attributes(portfolio: Path): attributes = {} - with open(portfolio, "rb") as portfolio_file: - content = portfolio_file.read() - try: - exec(content, attributes) - except Exception: - returncodes.exit_with_driver_critical_error( - "The portfolio %s could not be loaded. Maybe it still " - "uses the old portfolio syntax? See the FDSS portfolios " - "for examples using the new syntax." % portfolio) + content = portfolio.read_bytes() + try: + exec(content, attributes) + except Exception: + returncodes.exit_with_driver_critical_error( + f"The portfolio {portfolio} could not be loaded. Maybe it still " + "uses the old portfolio syntax? See the FDSS portfolios " + "for examples using the new syntax.") if "CONFIGS" not in attributes: returncodes.exit_with_driver_critical_error("portfolios must define CONFIGS") if "OPTIMAL" not in attributes: @@ -202,7 +202,7 @@ def get_portfolio_attributes(portfolio): return attributes -def run(portfolio, executable, sas_file, plan_manager, time, memory): +def run(portfolio: Path, executable, sas_file, plan_manager, time, memory): """ Run the configs in the given portfolio file. diff --git a/driver/run_components.py b/driver/run_components.py index 6043ce333c..bdb5df127d 100644 --- a/driver/run_components.py +++ b/driver/run_components.py @@ -1,6 +1,6 @@ -import errno import logging -import os.path +import os +from pathlib import Path import shutil import subprocess import sys @@ -20,35 +20,36 @@ returncodes.exit_with_driver_unsupported_error("Unsupported OS: " + os.name) # TODO: We might want to turn translate into a module and call it with "python3 -m translate". -REL_TRANSLATE_PATH = os.path.join("translate", "translate.py") -REL_SEARCH_PATH = f"downward{BINARY_EXT}" +REL_TRANSLATE_PATH = Path("translate") / "translate.py" +REL_SEARCH_PATH = Path(f"downward{BINARY_EXT}") # Older versions of VAL use lower case, newer versions upper case. We prefer the # older version because this is what our build instructions recommend. -VALIDATE = (shutil.which(f"validate{BINARY_EXT}") or - shutil.which(f"Validate{BINARY_EXT}")) +_VALIDATE_NAME = (shutil.which(f"validate{BINARY_EXT}") or + shutil.which(f"Validate{BINARY_EXT}")) +VALIDATE = Path(_VALIDATE_NAME) if _VALIDATE_NAME else None -def get_executable(build, rel_path): +def get_executable(build: str, rel_path: Path): # First, consider 'build' to be a path directly to the binaries. # The path can be absolute or relative to the current working # directory. - build_dir = build - if not os.path.exists(build_dir): + build_dir = Path(build) + if not build_dir.exists(): # If build is not a full path to the binaries, it might be the # name of a build in our standard directory structure. # in this case, the binaries are in # '/builds//bin'. - build_dir = os.path.join(util.BUILDS_DIR, build, "bin") - if not os.path.exists(build_dir): + build_dir = util.BUILDS_DIR / build / "bin" + if not build_dir.exists(): returncodes.exit_with_driver_input_error( - "Could not find build '{build}' at {build_dir}. " - "Please run './build.py {build}'.".format(**locals())) + f"Could not find build '{build}' at {build_dir}. " + f"Please run './build.py {build}'.") - abs_path = os.path.join(build_dir, rel_path) - if not os.path.exists(abs_path): + abs_path = build_dir / rel_path + if not abs_path.exists(): returncodes.exit_with_driver_input_error( - "Could not find '{rel_path}' in build '{build}'. " - "Please run './build.py {build}'.".format(**locals())) + f"Could not find '{rel_path}' in build '{build}'. " + f"Please run './build.py {build}'.") return abs_path @@ -115,7 +116,7 @@ def run_search(args): if args.portfolio: assert not args.search_options - logging.info("search portfolio: %s" % args.portfolio) + logging.info(f"search portfolio: {args.portfolio}") return portfolio_runner.run( args.portfolio, executable, args.search_input, plan_manager, time_limit, memory_limit) @@ -150,25 +151,15 @@ def run_validate(args): "Error: Trying to run validate but it was not found on the PATH.") logging.info("Running validate.") - num_files = len(args.filenames) - if num_files == 1: - task, = args.filenames - domain = util.find_domain_filename(task) - elif num_files == 2: - domain, task = args.filenames - else: - returncodes.exit_with_driver_input_error("validate needs one or two PDDL input files.") - plan_files = list(PlanManager(args.plan_file).get_existing_plans()) if not plan_files: print("Not running validate since no plans found.") return (0, True) - validate_inputs = [domain, task] + plan_files try: call.check_call( "validate", - [VALIDATE] + validate_inputs, + [VALIDATE] + args.validate_inputs + plan_files, time_limit=args.validate_time_limit, memory_limit=args.validate_memory_limit) except OSError as err: diff --git a/driver/tests.py b/driver/tests.py index 7b060dca8b..33f6c61d93 100644 --- a/driver/tests.py +++ b/driver/tests.py @@ -14,11 +14,11 @@ from .aliases import ALIASES, PORTFOLIOS from .arguments import EXAMPLES -from .call import check_call +from .call import check_call, _replace_paths_with_strings from . import limits from . import returncodes from .run_components import get_executable, REL_SEARCH_PATH -from .util import REPO_ROOT_DIR, find_domain_filename +from .util import REPO_ROOT_DIR, find_domain_path def cleanup(): @@ -32,6 +32,7 @@ def teardown_module(module): def run_driver(parameters): cmd = [sys.executable, "fast-downward.py", "--keep"] + parameters + cmd = _replace_paths_with_strings(cmd) return subprocess.check_call(cmd, cwd=REPO_ROOT_DIR) @@ -59,7 +60,7 @@ def test_portfolios(): def _get_portfolio_configs(portfolio: Path): - content = portfolio.read_text() + content = portfolio.read_bytes() attributes = {} try: exec(content, attributes) @@ -95,7 +96,7 @@ def _run_search(config): def _get_all_portfolio_configs(): all_configs = set() for portfolio in PORTFOLIOS.values(): - configs = _get_portfolio_configs(Path(portfolio)) + configs = _get_portfolio_configs(portfolio) all_configs |= set(tuple(_convert_to_standalone_config(config)) for config in configs) return all_configs @@ -125,8 +126,8 @@ def preexec_fn(): def test_automatic_domain_file_name_computation(): - benchmarks_dir = os.path.join(REPO_ROOT_DIR, "benchmarks") + benchmarks_dir = REPO_ROOT_DIR / "benchmarks" for dirpath, dirnames, filenames in os.walk(benchmarks_dir): for filename in filenames: if "domain" not in filename: - assert find_domain_filename(os.path.join(dirpath, filename)) + assert find_domain_path(dirpath / filename) diff --git a/driver/util.py b/driver/util.py index 2bc571097e..7016a2b806 100644 --- a/driver/util.py +++ b/driver/util.py @@ -1,11 +1,12 @@ import os +from pathlib import Path from . import returncodes -DRIVER_DIR = os.path.abspath(os.path.dirname(__file__)) -REPO_ROOT_DIR = os.path.dirname(DRIVER_DIR) -BUILDS_DIR = os.path.join(REPO_ROOT_DIR, "builds") +DRIVER_DIR = Path(__file__).parent.resolve() +REPO_ROOT_DIR = DRIVER_DIR.parent +BUILDS_DIR = REPO_ROOT_DIR / "builds" def get_elapsed_time(): @@ -19,25 +20,22 @@ def get_elapsed_time(): return sum(os.times()[:4]) -def find_domain_filename(task_filename): +def find_domain_path(task_path: Path): """ - Find domain filename for the given task using automatic naming rules. + Find domain path for the given task using automatic naming rules. """ - dirname, basename = os.path.split(task_filename) - basename_root, ext = os.path.splitext(basename) - domain_basenames = [ "domain.pddl", - basename_root + "-domain" + ext, - basename[:3] + "-domain.pddl", # for airport - "domain_" + basename, - "domain-" + basename, + task_path.stem + "-domain" + task_path.suffix, + task_path.name[:3] + "-domain.pddl", # for airport + "domain_" + task_path.name, + "domain-" + task_path.name, ] for domain_basename in domain_basenames: - domain_filename = os.path.join(dirname, domain_basename) - if os.path.exists(domain_filename): - return domain_filename + domain_path = task_path.parent / domain_basename + if domain_path.exists(): + return domain_path returncodes.exit_with_driver_input_error( "Error: Could not find domain file using automatic naming rules.") diff --git a/src/translate/pddl_parser/parsing_functions.py b/src/translate/pddl_parser/parsing_functions.py index e21e81b9d3..bbfce44ae6 100644 --- a/src/translate/pddl_parser/parsing_functions.py +++ b/src/translate/pddl_parser/parsing_functions.py @@ -297,7 +297,7 @@ def parse_literal(context, alist, type_dict, predicate_dict, negated=False): if arity != len(alist) - 1: context.error(f"Predicate '{predicate_name}' of arity {arity} used" - f" with {len(alist) -1} arguments.", alist) + f" with {len(alist) - 1} arguments.", alist) if negated: return pddl.NegatedAtom(pred_id, alist[1:])