diff --git a/changes.d/6130.fix.md b/changes.d/6130.fix.md new file mode 100644 index 00000000000..8423bfdcd40 --- /dev/null +++ b/changes.d/6130.fix.md @@ -0,0 +1 @@ +Prevent commands accepting job IDs where it doesn't make sense. \ No newline at end of file diff --git a/cylc/flow/command_validation.py b/cylc/flow/command_validation.py index c7a9b2762dc..1c57d452c43 100644 --- a/cylc/flow/command_validation.py +++ b/cylc/flow/command_validation.py @@ -18,12 +18,13 @@ from typing import ( + Iterable, List, Optional, ) from cylc.flow.exceptions import InputError -from cylc.flow.id import Tokens +from cylc.flow.id import IDTokens, Tokens from cylc.flow.task_outputs import TASK_OUTPUT_SUCCEEDED from cylc.flow.flow_mgr import FLOW_ALL, FLOW_NEW, FLOW_NONE @@ -228,3 +229,35 @@ def consistency( """ if outputs and prereqs: raise InputError("Use --prerequisite or --output, not both.") + + +def is_tasks(tasks: Iterable[str]): + """All tasks in a list of tasks are task ID's without trailing job ID. + + Examples: + # All legal + >>> is_tasks(['1/foo', '1/bar', '*/baz', '*/*']) + + # Some legal + >>> is_tasks(['1/foo/NN', '1/bar', '*/baz', '*/*/42']) + Traceback (most recent call last): + ... + cylc.flow.exceptions.InputError: This command does not take job ids: + * 1/foo/NN + * */*/42 + + # None legal + >>> is_tasks(['*/baz/12']) + Traceback (most recent call last): + ... + cylc.flow.exceptions.InputError: This command does not take job ids: + * */baz/12 + """ + bad_tasks: List[str] = [] + for task in tasks: + tokens = Tokens('//' + task) + if tokens.lowest_token == IDTokens.Job.value: + bad_tasks.append(task) + if bad_tasks: + msg = 'This command does not take job ids:\n * ' + raise InputError(msg + '\n * '.join(bad_tasks)) diff --git a/cylc/flow/commands.py b/cylc/flow/commands.py index 28de0d16ed5..a4ea43df5cf 100644 --- a/cylc/flow/commands.py +++ b/cylc/flow/commands.py @@ -145,6 +145,7 @@ async def set_prereqs_and_outputs( outputs = validate.outputs(outputs) prerequisites = validate.prereqs(prerequisites) validate.flow_opts(flow, flow_wait) + validate.is_tasks(tasks) yield @@ -172,6 +173,8 @@ async def stop( task: Optional[str] = None, flow_num: Optional[int] = None, ): + if task: + validate.is_tasks([task]) yield if flow_num: schd.pool.stop_flow(flow_num) @@ -214,6 +217,7 @@ async def stop( @_command('release') async def release(schd: 'Scheduler', tasks: Iterable[str]): """Release held tasks.""" + validate.is_tasks(tasks) yield yield schd.pool.release_held_tasks(tasks) @@ -237,6 +241,7 @@ async def resume(schd: 'Scheduler'): @_command('poll_tasks') async def poll_tasks(schd: 'Scheduler', tasks: Iterable[str]): """Poll pollable tasks or a task or family if options are provided.""" + validate.is_tasks(tasks) yield if schd.get_run_mode() == RunMode.SIMULATION: yield 0 @@ -248,6 +253,7 @@ async def poll_tasks(schd: 'Scheduler', tasks: Iterable[str]): @_command('kill_tasks') async def kill_tasks(schd: 'Scheduler', tasks: Iterable[str]): """Kill all tasks or a task/family if options are provided.""" + validate.is_tasks(tasks) yield itasks, _, bad_items = schd.pool.filter_task_proxies(tasks) if schd.get_run_mode() == RunMode.SIMULATION: @@ -264,6 +270,7 @@ async def kill_tasks(schd: 'Scheduler', tasks: Iterable[str]): @_command('hold') async def hold(schd: 'Scheduler', tasks: Iterable[str]): """Hold specified tasks.""" + validate.is_tasks(tasks) yield yield schd.pool.hold_tasks(tasks) @@ -304,6 +311,7 @@ async def set_verbosity(schd: 'Scheduler', level: Union[int, str]): @_command('remove_tasks') async def remove_tasks(schd: 'Scheduler', tasks: Iterable[str]): """Remove tasks.""" + validate.is_tasks(tasks) yield yield schd.pool.remove_tasks(tasks) @@ -430,5 +438,6 @@ async def force_trigger_tasks( flow_descr: Optional[str] = None, ): """Manual task trigger.""" + validate.is_tasks(tasks) yield yield schd.pool.force_trigger_tasks(tasks, flow, flow_wait, flow_descr)