diff --git a/CHANGES.md b/CHANGES.md index 43b40c86eed..1edbb2b04d6 100644 --- a/CHANGES.md +++ b/CHANGES.md @@ -11,6 +11,16 @@ $ towncrier create ..md --content "Short description" +## __cylc-8.3.1 (Released 2024-07-04)__ + +### 🔧 Fixes + +[#6130](https://github.com/cylc/cylc-flow/pull/6130) - Prevent commands accepting job IDs where it doesn't make sense. + +[#6170](https://github.com/cylc/cylc-flow/pull/6170) - Fix an issue where the Cylc logo could appear in the workflow log. + +[#6176](https://github.com/cylc/cylc-flow/pull/6176) - Fix bug where jobs which fail to submit are not shown in GUI/TUI if submission retries are set. + ## __cylc-8.3.0 (Released 2024-06-18)__ ### ⚠ Breaking Changes diff --git a/changes.d/6130.fix.md b/changes.d/6130.fix.md deleted file mode 100644 index 8423bfdcd40..00000000000 --- a/changes.d/6130.fix.md +++ /dev/null @@ -1 +0,0 @@ -Prevent commands accepting job IDs where it doesn't make sense. \ No newline at end of file diff --git a/changes.d/6170.fix.md b/changes.d/6170.fix.md deleted file mode 100644 index 1675c7dfe7c..00000000000 --- a/changes.d/6170.fix.md +++ /dev/null @@ -1 +0,0 @@ -Fix an issue where the Cylc logo could appear in the workflow log. diff --git a/cylc/flow/task_events_mgr.py b/cylc/flow/task_events_mgr.py index 0f89d2122dd..bf9c2ba3a9b 100644 --- a/cylc/flow/task_events_mgr.py +++ b/cylc/flow/task_events_mgr.py @@ -1535,11 +1535,21 @@ def _insert_task_job( else: job_conf = itask.jobs[-1] + # Job status should be task status unless task is awaiting a + # retry: + if itask.state.status == TASK_STATUS_WAITING and itask.try_timers: + job_status = ( + TASK_STATUS_SUBMITTED if submit_status == 0 + else TASK_STATUS_SUBMIT_FAILED + ) + else: + job_status = itask.state.status + # insert job into data store self.data_store_mgr.insert_job( itask.tdef.name, itask.point, - itask.state.status, + job_status, { **job_conf, # NOTE: the platform name may have changed since task diff --git a/tests/integration/test_task_events_mgr.py b/tests/integration/test_task_events_mgr.py index 62994487624..7ac12274d7b 100644 --- a/tests/integration/test_task_events_mgr.py +++ b/tests/integration/test_task_events_mgr.py @@ -14,12 +14,19 @@ # You should have received a copy of the GNU General Public License # along with this program. If not, see . -from cylc.flow.task_events_mgr import TaskJobLogsRetrieveContext -from cylc.flow.scheduler import Scheduler - +from itertools import product import logging from typing import Any as Fixture +from cylc.flow.task_events_mgr import TaskJobLogsRetrieveContext +from cylc.flow.scheduler import Scheduler +from cylc.flow.data_store_mgr import ( + JOBS, + TASK_STATUSES_ORDERED, + TASK_STATUS_WAITING, + TASK_STATUS_SUBMIT_FAILED, +) + async def test_process_job_logs_retrieval_warns_no_platform( one_conf: Fixture, flow: Fixture, scheduler: Fixture, run: Fixture, @@ -99,3 +106,67 @@ async def test__insert_task_job(flow, one_conf, scheduler, start, validate): i.submit_num for i in schd.data_store_mgr.added['jobs'].values() ] == [1, 2] + + +async def test__always_insert_task_job( + flow, scheduler, mock_glbl_cfg, start, run +): + """Insert Task Job _Always_ inserts a task into the data store. + + Bug https://github.com/cylc/cylc-flow/issues/6172 was caused + by passing task state to data_store_mgr.insert_job: Where + a submission retry was in progress the task state would be + "waiting" which caused the data_store_mgr.insert_job + to return without adding the task to the data store. + This is testing two different cases: + + * Could not select host from platform + * Could not select host from platform group + """ + global_config = """ + [platforms] + [[broken1]] + hosts = no-such-host-1 + [[broken2]] + hosts = no-such-host-2 + [platform groups] + [[broken]] + platforms = broken1 + """ + mock_glbl_cfg('cylc.flow.platforms.glbl_cfg', global_config) + + id_ = flow({ + 'scheduling': {'graph': {'R1': 'broken & broken2'}}, + 'runtime': { + 'root': {'submission retry delays': 'PT10M'}, + 'broken': {'platform': 'broken'}, + 'broken2': {'platform': 'broken2'} + } + }) + + schd = scheduler(id_, run_mode='live') + schd.bad_hosts = {'no-such-host-1', 'no-such-host-2'} + async with start(schd): + schd.task_job_mgr.submit_task_jobs( + schd.workflow, + schd.pool.get_tasks(), + schd.server.curve_auth, + schd.server.client_pub_key_dir, + is_simulation=False + ) + + # Both tasks are in a waiting state: + assert all( + i.state.status == TASK_STATUS_WAITING + for i in schd.pool.get_tasks()) + + # Both tasks have updated the data store with info + # about a failed job: + updates = { + k.split('//')[-1]: v.state + for k, v in schd.data_store_mgr.updated[JOBS].items() + } + assert updates == { + '1/broken/01': 'submit-failed', + '1/broken2/01': 'submit-failed' + }