From 90cfc636fee385785cfbe74061c40b4d506386a9 Mon Sep 17 00:00:00 2001 From: Werner Van Geit Date: Mon, 24 Jun 2024 10:21:48 +0200 Subject: [PATCH 01/10] Working version with conf file --- .../osparc-meta-parallelrunner/metadata.yml | 6 + Dockerfile | 3 +- docker_scripts/map.py | 321 +++++++++++------- .../osparc-0.6.5.post0-py3-none-any.whl | Bin 18388 -> 0 bytes ...osparc_client-0.6.5.post0-py3-none-any.whl | Bin 101182 -> 0 bytes validation-client/client.py | 10 +- 6 files changed, 208 insertions(+), 132 deletions(-) delete mode 100755 docker_scripts/osparc-0.6.5.post0-py3-none-any.whl delete mode 100755 docker_scripts/osparc_client-0.6.5.post0-py3-none-any.whl diff --git a/.osparc/osparc-meta-parallelrunner/metadata.yml b/.osparc/osparc-meta-parallelrunner/metadata.yml index e2ffa4f..f6a7741 100755 --- a/.osparc/osparc-meta-parallelrunner/metadata.yml +++ b/.osparc/osparc-meta-parallelrunner/metadata.yml @@ -28,6 +28,12 @@ inputs: description: File with the parameter sets to evaluate type: data:*/* + input_3: + displayOrder: 2.0 + label: Settings + description: + JSON file with settings for the parallel runner + type: data:*/* outputs: output_1: displayOrder: 1.0 diff --git a/Dockerfile b/Dockerfile index e3e795b..d89671b 100755 --- a/Dockerfile +++ b/Dockerfile @@ -15,8 +15,7 @@ RUN apt-get install --yes --no-install-recommends python3 python-is-python3 pyth # Copying boot scripts COPY docker_scripts /docker -RUN pip3 install pathos --upgrade -RUN pip3 install /docker/*.whl +RUN pip3 install pathos osparc --upgrade USER osparcuser diff --git a/docker_scripts/map.py b/docker_scripts/map.py index fd32466..fa59c9d 100755 --- a/docker_scripts/map.py +++ b/docker_scripts/map.py @@ -100,9 +100,13 @@ def create_study_job(template_id, job_inputs, studies_api): class MapRunner: - def __init__(self, input_path, output_path, polling_interval=1): + def __init__( + self, input_path, output_path, polling_interval=1, batch_mode=False + ): """Constructor""" + self.batch_mode = batch_mode + self.input_path = input_path # path where osparc write all our input self.output_path = output_path # path where osparc write all our input self.key_values_path = self.input_path / "key_values.json" @@ -236,9 +240,22 @@ def start(self): waiter_wrong_uuid += 1 else: input_tasks = input_dict["tasks"] - output_tasks = self.run_tasks( - tasks_uuid, input_tasks, n_of_workers + + if self.batch_mode: + n_of_batches = n_of_workers + else: + n_of_batches = len(input_tasks) + + input_batches = self.batch_input_tasks( + input_tasks, n_of_batches ) + + output_batches = self.run_batches( + tasks_uuid, input_batches, n_of_workers + ) + + output_tasks = self.unbatch_output_tasks(output_batches) + output_tasks_content = json.dumps( {"uuid": tasks_uuid, "tasks": output_tasks} ) @@ -253,155 +270,203 @@ def start(self): time.sleep(self.polling_interval) - def run_tasks(self, tasks_uuid, input_tasks, n_of_workers): - logger.info(f"Evaluating: {input_tasks}") + def batch_input_tasks(self, input_tasks, n_of_batches): + batches = [[] for _ in range(n_of_batches)] - self.n_of_finished_tasks = 0 + for task_i, input_task in enumerate(input_tasks): + batch_id = task_i % n_of_batches + batches[batch_id].append(input_task) + return batches - def map_func(task, trial_number=1): - try: - logger.info(f"Running worker for task: {task}") - - input = task["input"] - output = task["output"] - - job_inputs = {"values": {}} - - for param_name, param_input in input.items(): - param_type = param_input["type"] - param_value = param_input["value"] - if param_type == "FileJSON": - param_filename = param_input["filename"] - tmp_dir = tempfile.TemporaryDirectory() - tmp_dir_path = pl.Path(tmp_dir.name) - tmp_input_file_path = tmp_dir_path / param_filename - tmp_input_file_path.write_text(json.dumps(param_value)) - - input_data_file = osparc.FilesApi( - self.api_client - ).upload_file(file=tmp_input_file_path) - job_inputs["values"][param_name] = input_data_file - elif param_type == "file": - file_info = json.loads(param_value) - input_data_file = osparc_client.models.file.File( - id=file_info["id"], - filename=file_info["filename"], - content_type=file_info["content_type"], - checksum=file_info["checksum"], - e_tag=file_info["e_tag"], - ) - job_inputs["values"][param_name] = input_data_file - elif param_type == "integer": - job_inputs["values"][param_name] = int(param_value) - elif param_type == "float": - job_inputs["values"][param_name] = float(param_value) - else: - job_inputs["values"][param_name] = param_value + def unbatch_output_tasks(self, batches): + output_tasks = [] + n_of_tasks = sum(len(batch) for batch in batches) - logger.debug(f"Sending inputs: {job_inputs}") + for task_i in range(n_of_tasks): + batch_id = task_i % len(batches) + output_tasks.append(batches[batch_id].pop(0)) + return output_tasks - with create_study_job( - self.template_id, job_inputs, self.studies_api - ) as job: - job_status = self.studies_api.start_study_job( - study_id=self.template_id, job_id=job.id + def create_job_inputs(self, batch): + """Create job inputs""" + + job_inputs = {"values": {}} + + for task in batch: + input = task["input"] + for param_name, param_input in input.items(): + param_type = param_input["type"] + param_value = param_input["value"] + if param_type == "FileJSON": + param_filename = param_input["filename"] + tmp_dir = tempfile.TemporaryDirectory() + tmp_dir_path = pl.Path(tmp_dir.name) + tmp_input_file_path = tmp_dir_path / param_filename + tmp_input_file_path.write_text(json.dumps(param_value)) + + input_data_file = osparc.FilesApi( + self.api_client + ).upload_file(file=tmp_input_file_path) + processed_param_value = input_data_file + elif param_type == "file": + file_info = json.loads(param_value) + input_data_file = osparc_client.models.file.File( + id=file_info["id"], + filename=file_info["filename"], + content_type=file_info["content_type"], + checksum=file_info["checksum"], + e_tag=file_info["e_tag"], ) + processed_param_value = input_data_file + elif param_type == "integer": + processed_param_value = int(param_value) + elif param_type == "float": + processed_param_value = float(param_value) + else: + processed_param_value = param_value - while ( - job_status.state != "SUCCESS" - and job_status.state != "FAILED" - ): - job_status = self.studies_api.inspect_study_job( - study_id=self.template_id, job_id=job.id - ) - time.sleep(1) + if self.batch_mode: + if param_name not in job_inputs: + job_inputs["values"][param_name] = [] + job_inputs["values"][param_name].append( + processed_param_value + ) + else: + assert len(batch) == 1 + job_inputs["values"][param_name] = processed_param_value - task["status"] = job_status.state + return job_inputs - if job_status.state == "FAILED": - logger.error(f"Task failed: {task}") - raise Exception("Job returned a failed status") - else: - results = self.studies_api.get_study_job_outputs( - study_id=self.template_id, job_id=job.id - ).results - - for probe_name, probe_output in results.items(): - if probe_name not in output: - raise ValueError( - f"Unknown probe in output: {probe_name}" - ) - probe_type = output[probe_name]["type"] - - if probe_type == "FileJSON": - output_file = pl.Path( - osparc.FilesApi( - self.api_client - ).download_file(probe_output.id) - ) - with zipfile.ZipFile( - output_file, "r" - ) as zip_file: - file_results_path = zipfile.Path( - zip_file, - at=output[probe_name]["filename"], - ) - file_results = json.loads( - file_results_path.read_text() - ) - - output[probe_name]["value"] = file_results - elif probe_type == "file": - tmp_output_data_file = osparc.FilesApi( - self.api_client - ).download_file(probe_output.id) - output_data_file = osparc.FilesApi( - self.api_client - ).upload_file(tmp_output_data_file) - - output[probe_name]["value"] = json.dumps( - output_data_file.to_dict() - ) - elif probe_type == "integer": - output[probe_name]["value"] = int(probe_output) - elif probe_type == "float": - output[probe_name]["value"] = float( - probe_output - ) - else: - output[probe_name]["value"] = probe_output - - self.n_of_finished_tasks += 1 - - logger.info( - "Worker has finished task " - f"{self.n_of_finished_tasks} of {len(input_tasks)}" + def run_job(self, job_inputs): + """Run a job with given inputs""" + + logger.debug(f"Sending inputs: {job_inputs}") + + if self.template_id == "TEST_UUID": + logger.info("Map in test mode, just returning input") + self.n_of_finished_batches += 1 + + return job_inputs, "SUCCESS" + + with create_study_job( + self.template_id, job_inputs, self.studies_api + ) as job: + job_status = self.studies_api.start_study_job( + study_id=self.template_id, job_id=job.id + ) + + while ( + job_status.state != "SUCCESS" and job_status.state != "FAILED" + ): + job_status = self.studies_api.inspect_study_job( + study_id=self.template_id, job_id=job.id + ) + time.sleep(1) + + status = job_status.state + + if job_status.state == "FAILED": + logger.error(f"Batch failed: {job_inputs}") + raise Exception("Job returned a failed status") + else: + job_outputs = self.studies_api.get_study_job_outputs( + study_id=self.template_id, job_id=job.id + ).results + + self.n_of_finished_batches += 1 + + return job_outputs, status + + def process_job_outputs(self, results, batch): + if self.template_id == "TEST_UUID": + logger.info("Map in test mode, just returning input") + + return batch + + for task_i, task in enumerate(batch): + output = task["output"] + for probe_name, probe_outputs in results.items(): + probe_output = probe_outputs[task_i] + if probe_name not in output: + raise ValueError(f"Unknown probe in output: {probe_name}") + probe_type = output[probe_name]["type"] + + if probe_type == "FileJSON": + output_file = pl.Path( + osparc.FilesApi(self.api_client).download_file( + probe_output.id + ) + ) + with zipfile.ZipFile(output_file, "r") as zip_file: + file_results_path = zipfile.Path( + zip_file, + at=output[probe_name]["filename"], + ) + file_results = json.loads( + file_results_path.read_text() ) + + output[probe_name]["value"] = file_results + elif probe_type == "file": + tmp_output_data_file = osparc.FilesApi( + self.api_client + ).download_file(probe_output.id) + output_data_file = osparc.FilesApi( + self.api_client + ).upload_file(tmp_output_data_file) + + output[probe_name]["value"] = json.dumps( + output_data_file.to_dict() + ) + elif probe_type == "integer": + output[probe_name]["value"] = int(probe_output) + elif probe_type == "float": + output[probe_name]["value"] = float(probe_output) + else: + output[probe_name]["value"] = probe_output + + def run_batches(self, tasks_uuid, input_batches, n_of_workers): + """Run the tasks""" + + logger.info(f"Evaluating: {input_batches}") + + self.n_of_finished_batches = 0 + + def map_func(batch, trial_number=1): + try: + logger.info(f"Running worker for batch: {batch}") + + job_inputs = self.create_job_inputs(batch) + + job_outputs, status = self.run_job(job_inputs) + + batch = self.process_job_outputs(job_outputs, batch) + + logger.info( + "Worker has finished batch " + f"{self.n_of_finished_batches} of {len(input_batches)}" + ) except Exception as error: if trial_number >= MAX_TRIALS: logger.info( - f"Task {task} failed with error {error} in " + f"Batch {batch} failed with error {error} in " f"trial {trial_number}, not retrying, raising error" ) raise error else: logger.info( - f"Task {task} failed with error {error} in " + f"Batch {batch} failed with error {error} in " f"trial {trial_number}, retrying " ) - task = map_func(task, trial_number=trial_number + 1) + batch = map_func(batch, trial_number=trial_number + 1) - return task - - if self.template_id == "TEST_UUID": - logger.info("Map in test mode, just returning input") - return input_tasks + return batch logger.info( - f"Starting {len(input_tasks)} tasks on {n_of_workers} workers" + f"Starting {len(input_batches)} batches on {n_of_workers} workers" ) with pathos.pools.ThreadPool(nodes=n_of_workers) as pool: - output_tasks = list(pool.map(map_func, input_tasks)) + output_tasks = list(pool.map(map_func, input_batches)) pool.close() pool.join() pool.clear() # Pool is singleton, need to clear old pool diff --git a/docker_scripts/osparc-0.6.5.post0-py3-none-any.whl b/docker_scripts/osparc-0.6.5.post0-py3-none-any.whl deleted file mode 100755 index e76939edc2ac3c60cd6dac7bc38d2bb74622ca45..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 18388 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 GgLnXr5I*k! diff --git a/docker_scripts/osparc_client-0.6.5.post0-py3-none-any.whl b/docker_scripts/osparc_client-0.6.5.post0-py3-none-any.whl deleted file mode 100755 index 721744167202ed62958c8c432b378fc6c5a57c5a..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 101182 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 gxKu2ah>DfN^Z%OI_hC?J?t7nJTc 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∨8@d45`+T# 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 y{)uI?_;=V}E|Nj Date: Mon, 24 Jun 2024 10:28:08 +0200 Subject: [PATCH 02/10] Refactor files --- docker_scripts/main.py | 96 +++++++++++++++++++ docker_scripts/map.bash | 2 +- docker_scripts/{map.py => parallelrunner.py} | 98 ++++++-------------- 3 files changed, 126 insertions(+), 70 deletions(-) create mode 100755 docker_scripts/main.py rename docker_scripts/{map.py => parallelrunner.py} (93%) diff --git a/docker_scripts/main.py b/docker_scripts/main.py new file mode 100755 index 0000000..99a4106 --- /dev/null +++ b/docker_scripts/main.py @@ -0,0 +1,96 @@ +import contextlib +import http.server +import logging +import os +import pathlib as pl +import socketserver +import threading + +import osparc_client +import osparc_client.models.file + +import parallelrunner + +logging.basicConfig( + level=logging.INFO, format="[%(filename)s:%(lineno)d] %(message)s" +) +logger = logging.getLogger(__name__) + +MAX_JOB_CREATE_ATTEMPTS = 5 +MAX_TRIALS = 5 + +HTTP_PORT = 8888 + +MAX_N_OF_WORKERS = 10 + +POLLING_INTERVAL = 1 # second +TEMPLATE_ID_KEY = "input_0" +N_OF_WORKERS_KEY = "input_1" +INPUT_PARAMETERS_KEY = "input_2" + + +def main(): + """Main""" + + input_path = pl.Path(os.environ["DY_SIDECAR_PATH_INPUTS"]) + output_path = pl.Path(os.environ["DY_SIDECAR_PATH_OUTPUTS"]) + + http_dir_path = pl.Path(__file__).parent / "http" + + class HTTPHandler(http.server.SimpleHTTPRequestHandler): + def __init__(self, *args, **kwargs): + super().__init__( + *args, **kwargs, directory=http_dir_path.resolve() + ) + + maprunner = parallelrunner.ParallelRunner( + input_path, output_path, polling_interval=POLLING_INTERVAL + ) + + try: + logger.info( + f"Starting http server at port {HTTP_PORT} and serving path {http_dir_path}" + ) + with socketserver.TCPServer(("", HTTP_PORT), HTTPHandler) as httpd: + httpd_thread = threading.Thread(target=httpd.serve_forever) + httpd_thread.start() + maprunner.setup() + maprunner.start() + maprunner.teardown() + httpd.shutdown() + except Exception as err: # pylint: disable=broad-except + logger.error(f"{err} . Stopping %s", exc_info=True) + + +@contextlib.contextmanager +def create_study_job(template_id, job_inputs, studies_api): + n_of_create_attempts = 0 + while True: + try: + n_of_create_attempts += 1 + job = studies_api.create_study_job( + study_id=template_id, + job_inputs=job_inputs, + ) + break + except osparc_client.exceptions.ApiException as api_exception: + if n_of_create_attempts >= MAX_JOB_CREATE_ATTEMPTS: + raise Exception( + f"Tried {n_of_create_attempts} times to create a job from " + "the study, but failed" + ) + else: + logger.exception(api_exception) + logger.info( + "Received an API Exception from server " + "when creating job, retrying..." + ) + + try: + yield job + finally: + studies_api.delete_study_job(template_id, job.id) + + +if __name__ == "__main__": + main() diff --git a/docker_scripts/map.bash b/docker_scripts/map.bash index f47a654..c9aa4a3 100755 --- a/docker_scripts/map.bash +++ b/docker_scripts/map.bash @@ -2,5 +2,5 @@ pip install -r /docker/requirements.txt echo "Starting map python code" -python3 /docker/map.py +python3 /docker/main.py echo "Closing map python code" diff --git a/docker_scripts/map.py b/docker_scripts/parallelrunner.py similarity index 93% rename from docker_scripts/map.py rename to docker_scripts/parallelrunner.py index fa59c9d..754cf83 100755 --- a/docker_scripts/map.py +++ b/docker_scripts/parallelrunner.py @@ -1,13 +1,10 @@ import contextlib import getpass -import http.server import json import logging import os import pathlib as pl -import socketserver import tempfile -import threading import time import uuid import zipfile @@ -36,70 +33,7 @@ INPUT_PARAMETERS_KEY = "input_2" -def main(): - """Main""" - - input_path = pl.Path(os.environ["DY_SIDECAR_PATH_INPUTS"]) - output_path = pl.Path(os.environ["DY_SIDECAR_PATH_OUTPUTS"]) - - http_dir_path = pl.Path(__file__).parent / "http" - - class HTTPHandler(http.server.SimpleHTTPRequestHandler): - def __init__(self, *args, **kwargs): - super().__init__( - *args, **kwargs, directory=http_dir_path.resolve() - ) - - maprunner = MapRunner( - input_path, output_path, polling_interval=POLLING_INTERVAL - ) - - try: - logger.info( - f"Starting http server at port {HTTP_PORT} and serving path {http_dir_path}" - ) - with socketserver.TCPServer(("", HTTP_PORT), HTTPHandler) as httpd: - httpd_thread = threading.Thread(target=httpd.serve_forever) - httpd_thread.start() - maprunner.setup() - maprunner.start() - maprunner.teardown() - httpd.shutdown() - except Exception as err: # pylint: disable=broad-except - logger.error(f"{err} . Stopping %s", exc_info=True) - - -@contextlib.contextmanager -def create_study_job(template_id, job_inputs, studies_api): - n_of_create_attempts = 0 - while True: - try: - n_of_create_attempts += 1 - job = studies_api.create_study_job( - study_id=template_id, - job_inputs=job_inputs, - ) - break - except osparc_client.exceptions.ApiException as api_exception: - if n_of_create_attempts >= MAX_JOB_CREATE_ATTEMPTS: - raise Exception( - f"Tried {n_of_create_attempts} times to create a job from " - "the study, but failed" - ) - else: - logger.exception(api_exception) - logger.info( - "Received an API Exception from server " - "when creating job, retrying..." - ) - - try: - yield job - finally: - studies_api.delete_study_job(template_id, job.id) - - -class MapRunner: +class ParallelRunner: def __init__( self, input_path, output_path, polling_interval=1, batch_mode=False ): @@ -491,5 +425,31 @@ def read_keyvalues(self): return keyvalues -if __name__ == "__main__": - main() +@contextlib.contextmanager +def create_study_job(template_id, job_inputs, studies_api): + n_of_create_attempts = 0 + while True: + try: + n_of_create_attempts += 1 + job = studies_api.create_study_job( + study_id=template_id, + job_inputs=job_inputs, + ) + break + except osparc_client.exceptions.ApiException as api_exception: + if n_of_create_attempts >= MAX_JOB_CREATE_ATTEMPTS: + raise Exception( + f"Tried {n_of_create_attempts} times to create a job from " + "the study, but failed" + ) + else: + logger.exception(api_exception) + logger.info( + "Received an API Exception from server " + "when creating job, retrying..." + ) + + try: + yield job + finally: + studies_api.delete_study_job(template_id, job.id) From 9efe6a34c2be4958af48efedea3a163b6024dba3 Mon Sep 17 00:00:00 2001 From: Werner Van Geit Date: Mon, 24 Jun 2024 10:31:34 +0200 Subject: [PATCH 03/10] More renaming of files --- docker_scripts/entrypoint.bash | 55 ++++++++++++-------------- docker_scripts/{map.bash => main.bash} | 0 docker_scripts/main.sh | 16 -------- 3 files changed, 26 insertions(+), 45 deletions(-) rename docker_scripts/{map.bash => main.bash} (100%) delete mode 100755 docker_scripts/main.sh diff --git a/docker_scripts/entrypoint.bash b/docker_scripts/entrypoint.bash index c8714a7..163b215 100755 --- a/docker_scripts/entrypoint.bash +++ b/docker_scripts/entrypoint.bash @@ -10,34 +10,31 @@ HOST_USERID=$(stat -c %u "${DY_SIDECAR_PATH_INPUTS}") HOST_GROUPID=$(stat -c %g "${DY_SIDECAR_PATH_INPUTS}") CONTAINER_GROUPNAME=$(getent group | grep "${HOST_GROUPID}" | cut --delimiter=: --fields=1 || echo "") - OSPARC_USER='osparcuser' -if [ "$HOST_USERID" -eq 0 ] -then - echo "Warning: Folder mounted owned by root user... adding $OSPARC_USER to root..." - addgroup "$OSPARC_USER" root -else - echo "Folder mounted owned by user $HOST_USERID:$HOST_GROUPID-'$CONTAINER_GROUPNAME'..." - # take host's credentials in $OSPARC_USER - if [ -z "$CONTAINER_GROUPNAME" ] - then - echo "Creating new group my$OSPARC_USER" - CONTAINER_GROUPNAME=my$OSPARC_USER - addgroup --gid "$HOST_GROUPID" "$CONTAINER_GROUPNAME" - else - echo "group already exists" - fi - - echo "adding $OSPARC_USER to group $CONTAINER_GROUPNAME..." - usermod --append --groups "$CONTAINER_GROUPNAME" "$OSPARC_USER" - - echo "changing owner ship of state directory /home/${OSPARC_USER}/work/workspace" - chown --recursive "$OSPARC_USER" "/home/${OSPARC_USER}/work/workspace" - echo "changing owner ship of state directory ${DY_SIDECAR_PATH_INPUTS}" - chown --recursive "$OSPARC_USER" "${DY_SIDECAR_PATH_INPUTS}" - echo "changing owner ship of state directory ${DY_SIDECAR_PATH_OUTPUTS}" - chown --recursive "$OSPARC_USER" "${DY_SIDECAR_PATH_OUTPUTS}" -fi - -exec gosu "$OSPARC_USER" /docker/map.bash +if [ "$HOST_USERID" -eq 0 ]; then + echo "Warning: Folder mounted owned by root user... adding $OSPARC_USER to root..." + addgroup "$OSPARC_USER" root +else + echo "Folder mounted owned by user $HOST_USERID:$HOST_GROUPID-'$CONTAINER_GROUPNAME'..." + # take host's credentials in $OSPARC_USER + if [ -z "$CONTAINER_GROUPNAME" ]; then + echo "Creating new group my$OSPARC_USER" + CONTAINER_GROUPNAME=my$OSPARC_USER + addgroup --gid "$HOST_GROUPID" "$CONTAINER_GROUPNAME" + else + echo "group already exists" + fi + + echo "adding $OSPARC_USER to group $CONTAINER_GROUPNAME..." + usermod --append --groups "$CONTAINER_GROUPNAME" "$OSPARC_USER" + + echo "changing owner ship of state directory /home/${OSPARC_USER}/work/workspace" + chown --recursive "$OSPARC_USER" "/home/${OSPARC_USER}/work/workspace" + echo "changing owner ship of state directory ${DY_SIDECAR_PATH_INPUTS}" + chown --recursive "$OSPARC_USER" "${DY_SIDECAR_PATH_INPUTS}" + echo "changing owner ship of state directory ${DY_SIDECAR_PATH_OUTPUTS}" + chown --recursive "$OSPARC_USER" "${DY_SIDECAR_PATH_OUTPUTS}" +fi + +exec gosu "$OSPARC_USER" /docker/main.bash diff --git a/docker_scripts/map.bash b/docker_scripts/main.bash similarity index 100% rename from docker_scripts/map.bash rename to docker_scripts/main.bash diff --git a/docker_scripts/main.sh b/docker_scripts/main.sh deleted file mode 100755 index 31a1924..0000000 --- a/docker_scripts/main.sh +++ /dev/null @@ -1,16 +0,0 @@ -#!/bin/sh -set -o errexit -set -o nounset - -echo "Starting main.sh" -echo "Creating virtual environment" -python3 -m venv --system-site-packages --symlinks --upgrade ${OSPARC_VENV_DIR} -${OSPARC_VENV_DIR}/bin/pip install -qU pip wheel setuptools - - -[ ! -z "${OSPARC_REQUIREMENTS_TXT}" ] && ${OSPARC_VENV_DIR}/bin/pip install -qr ${OSPARC_REQUIREMENTS_TXT} - -echo "Executing code ${OSPARC_USER_ENTRYPOINT_PATH}" -cd ${OSPARC_USER_ENTRYPOINT_DIR} -${OSPARC_VENV_DIR}/bin/python3 ${OSPARC_USER_ENTRYPOINT_PATH} -echo "Stopping main.sh" From 63347bcf0ae274725f853bbf04c20e13744a24b7 Mon Sep 17 00:00:00 2001 From: Werner Van Geit Date: Mon, 24 Jun 2024 10:38:46 +0200 Subject: [PATCH 04/10] Clean up constants --- docker_scripts/main.py | 47 ++------------------------------ docker_scripts/parallelrunner.py | 28 ++++++++++++------- 2 files changed, 20 insertions(+), 55 deletions(-) diff --git a/docker_scripts/main.py b/docker_scripts/main.py index 99a4106..2939741 100755 --- a/docker_scripts/main.py +++ b/docker_scripts/main.py @@ -1,4 +1,3 @@ -import contextlib import http.server import logging import os @@ -6,9 +5,6 @@ import socketserver import threading -import osparc_client -import osparc_client.models.file - import parallelrunner logging.basicConfig( @@ -16,18 +12,8 @@ ) logger = logging.getLogger(__name__) -MAX_JOB_CREATE_ATTEMPTS = 5 -MAX_TRIALS = 5 - HTTP_PORT = 8888 -MAX_N_OF_WORKERS = 10 - -POLLING_INTERVAL = 1 # second -TEMPLATE_ID_KEY = "input_0" -N_OF_WORKERS_KEY = "input_1" -INPUT_PARAMETERS_KEY = "input_2" - def main(): """Main""" @@ -44,7 +30,8 @@ def __init__(self, *args, **kwargs): ) maprunner = parallelrunner.ParallelRunner( - input_path, output_path, polling_interval=POLLING_INTERVAL + input_path, + output_path, ) try: @@ -62,35 +49,5 @@ def __init__(self, *args, **kwargs): logger.error(f"{err} . Stopping %s", exc_info=True) -@contextlib.contextmanager -def create_study_job(template_id, job_inputs, studies_api): - n_of_create_attempts = 0 - while True: - try: - n_of_create_attempts += 1 - job = studies_api.create_study_job( - study_id=template_id, - job_inputs=job_inputs, - ) - break - except osparc_client.exceptions.ApiException as api_exception: - if n_of_create_attempts >= MAX_JOB_CREATE_ATTEMPTS: - raise Exception( - f"Tried {n_of_create_attempts} times to create a job from " - "the study, but failed" - ) - else: - logger.exception(api_exception) - logger.info( - "Received an API Exception from server " - "when creating job, retrying..." - ) - - try: - yield job - finally: - studies_api.delete_study_job(template_id, job.id) - - if __name__ == "__main__": main() diff --git a/docker_scripts/parallelrunner.py b/docker_scripts/parallelrunner.py index 754cf83..d31b8da 100755 --- a/docker_scripts/parallelrunner.py +++ b/docker_scripts/parallelrunner.py @@ -20,14 +20,12 @@ ) logger = logging.getLogger(__name__) +POLLING_INTERVAL = 1 # second + MAX_JOB_CREATE_ATTEMPTS = 5 MAX_TRIALS = 5 - -HTTP_PORT = 8888 - MAX_N_OF_WORKERS = 10 -POLLING_INTERVAL = 1 # second TEMPLATE_ID_KEY = "input_0" N_OF_WORKERS_KEY = "input_1" INPUT_PARAMETERS_KEY = "input_2" @@ -35,11 +33,21 @@ class ParallelRunner: def __init__( - self, input_path, output_path, polling_interval=1, batch_mode=False + self, + input_path, + output_path, + batch_mode=False, + polling_interval=POLLING_INTERVAL, + max_n_of_workers=MAX_N_OF_WORKERS, + max_trials=MAX_TRIALS, + max_job_create_attempts=MAX_JOB_CREATE_ATTEMPTS, ): """Constructor""" self.batch_mode = batch_mode + self.max_n_of_workers = max_n_of_workers + self.max_trials = max_trials + self.max_job_create_attempts = max_job_create_attempts self.input_path = input_path # path where osparc write all our input self.output_path = output_path # path where osparc write all our input @@ -127,13 +135,13 @@ def start(self): n_of_workers = key_values[N_OF_WORKERS_KEY]["value"] if n_of_workers is None: raise ValueError("Number of workers can't be None") - elif n_of_workers > MAX_N_OF_WORKERS: + elif n_of_workers > self.max_n_of_workers: logger.warning( "Attempt to set number of workers to more than " - f"is allowed ({MAX_N_OF_WORKERS}), limiting value " + f"is allowed ({self.max_n_of_workers}), limiting value " "to maximum amount" ) - n_of_workers = MAX_N_OF_WORKERS + n_of_workers = self.max_n_of_workers last_tasks_uuid = "" waiter_wrong_uuid = 0 @@ -381,7 +389,7 @@ def map_func(batch, trial_number=1): f"{self.n_of_finished_batches} of {len(input_batches)}" ) except Exception as error: - if trial_number >= MAX_TRIALS: + if trial_number >= self.max_trials: logger.info( f"Batch {batch} failed with error {error} in " f"trial {trial_number}, not retrying, raising error" @@ -437,7 +445,7 @@ def create_study_job(template_id, job_inputs, studies_api): ) break except osparc_client.exceptions.ApiException as api_exception: - if n_of_create_attempts >= MAX_JOB_CREATE_ATTEMPTS: + if n_of_create_attempts >= self.max_job_create_attempts: raise Exception( f"Tried {n_of_create_attempts} times to create a job from " "the study, but failed" From fd96adcc47b2284cfd181a74d0c1ea39cad688da Mon Sep 17 00:00:00 2001 From: Werner Van Geit Date: Mon, 24 Jun 2024 13:08:06 +0200 Subject: [PATCH 05/10] Add pydantic code for configuration file --- docker_scripts/main.py | 20 +++++++++++++------- docker_scripts/requirements.txt | 1 + validation/inputs/parallelrunner.json | 3 +++ 3 files changed, 17 insertions(+), 7 deletions(-) create mode 100644 validation/inputs/parallelrunner.json diff --git a/docker_scripts/main.py b/docker_scripts/main.py index 2939741..87cee43 100755 --- a/docker_scripts/main.py +++ b/docker_scripts/main.py @@ -1,10 +1,12 @@ import http.server import logging -import os import pathlib as pl import socketserver import threading +import pydantic as pyda +import pydantic_settings + import parallelrunner logging.basicConfig( @@ -18,8 +20,9 @@ def main(): """Main""" - input_path = pl.Path(os.environ["DY_SIDECAR_PATH_INPUTS"]) - output_path = pl.Path(os.environ["DY_SIDECAR_PATH_OUTPUTS"]) + settings = MainSettings() + settings = settings.parse_file(settings.input_path / "parallelrunner.json") + logging.info(f"Received the following settings: {settings}") http_dir_path = pl.Path(__file__).parent / "http" @@ -29,10 +32,7 @@ def __init__(self, *args, **kwargs): *args, **kwargs, directory=http_dir_path.resolve() ) - maprunner = parallelrunner.ParallelRunner( - input_path, - output_path, - ) + maprunner = parallelrunner.ParallelRunner(**settings.dict()) try: logger.info( @@ -49,5 +49,11 @@ def __init__(self, *args, **kwargs): logger.error(f"{err} . Stopping %s", exc_info=True) +class MainSettings(pydantic_settings.BaseSettings): + batch_mode: bool = False + input_path: pyda.DirectoryPath = pyda.Field(alias="DY_SIDECAR_PATH_INPUTS") + output_path: pl.Path = pyda.Field(alias="DY_SIDECAR_PATH_OUTPUTS") + + if __name__ == "__main__": main() diff --git a/docker_scripts/requirements.txt b/docker_scripts/requirements.txt index 69cb810..4be555f 100755 --- a/docker_scripts/requirements.txt +++ b/docker_scripts/requirements.txt @@ -1 +1,2 @@ osparc_filecomms +pydantic-settings diff --git a/validation/inputs/parallelrunner.json b/validation/inputs/parallelrunner.json new file mode 100644 index 0000000..619408a --- /dev/null +++ b/validation/inputs/parallelrunner.json @@ -0,0 +1,3 @@ +{ + "batch_mode": true +} From ef988e10099145558a14df5ee7bc42b030e5883c Mon Sep 17 00:00:00 2001 From: Werner Van Geit Date: Mon, 24 Jun 2024 13:27:48 +0200 Subject: [PATCH 06/10] Moved parallelrunner.json validation file --- docker_scripts/main.py | 14 +++- docker_scripts/parallelrunner.py | 73 +++++++++---------- .../inputs/{ => input_3}/parallelrunner.json | 0 3 files changed, 49 insertions(+), 38 deletions(-) rename validation/inputs/{ => input_3}/parallelrunner.json (100%) diff --git a/docker_scripts/main.py b/docker_scripts/main.py index 87cee43..4c02eee 100755 --- a/docker_scripts/main.py +++ b/docker_scripts/main.py @@ -3,6 +3,7 @@ import pathlib as pl import socketserver import threading +import time import pydantic as pyda import pydantic_settings @@ -15,13 +16,23 @@ logger = logging.getLogger(__name__) HTTP_PORT = 8888 +INPUT_CONF_KEY = "input_3" def main(): """Main""" settings = MainSettings() - settings = settings.parse_file(settings.input_path / "parallelrunner.json") + config_path = settings.input_path / INPUT_CONF_KEY / "parallelrunner.json" + + waiter = 0 + while not config_path.exists(): + if waiter % 10 == 0: + logger.info("Waiting for parallelrunner.json to exist ...") + time.sleep(settings.file_polling_interval) + waiter += 1 + + settings = settings.parse_file(config_path) logging.info(f"Received the following settings: {settings}") http_dir_path = pl.Path(__file__).parent / "http" @@ -51,6 +62,7 @@ def __init__(self, *args, **kwargs): class MainSettings(pydantic_settings.BaseSettings): batch_mode: bool = False + file_polling_interval: int = 1 input_path: pyda.DirectoryPath = pyda.Field(alias="DY_SIDECAR_PATH_INPUTS") output_path: pl.Path = pyda.Field(alias="DY_SIDECAR_PATH_OUTPUTS") diff --git a/docker_scripts/parallelrunner.py b/docker_scripts/parallelrunner.py index d31b8da..e9dc0a2 100755 --- a/docker_scripts/parallelrunner.py +++ b/docker_scripts/parallelrunner.py @@ -20,7 +20,7 @@ ) logger = logging.getLogger(__name__) -POLLING_INTERVAL = 1 # second +FILE_POLLING_INTERVAL = 1 # second MAX_JOB_CREATE_ATTEMPTS = 5 MAX_TRIALS = 5 @@ -37,7 +37,7 @@ def __init__( input_path, output_path, batch_mode=False, - polling_interval=POLLING_INTERVAL, + file_polling_interval=FILE_POLLING_INTERVAL, max_n_of_workers=MAX_N_OF_WORKERS, max_trials=MAX_TRIALS, max_job_create_attempts=MAX_JOB_CREATE_ATTEMPTS, @@ -64,7 +64,7 @@ def __init__( if self.output_tasks_path.exists(): self.output_tasks_path.unlink() - self.polling_interval = polling_interval + self.file_polling_interval = file_polling_interval self.caller_uuid = None self.uuid = str(uuid.uuid4()) @@ -102,7 +102,7 @@ def start(self): while not self.key_values_path.exists(): if waiter % 10 == 0: logger.info("Waiting for key_values.json to exist ...") - time.sleep(self.polling_interval) + time.sleep(self.file_polling_interval) waiter += 1 key_values = json.loads(self.key_values_path.read_text()) @@ -125,7 +125,7 @@ def start(self): f"exist in key_values, current content: {key_values}..." ) key_values = json.loads(self.key_values_path.read_text()) - time.sleep(self.polling_interval) + time.sleep(self.file_polling_interval) waiter += 1 self.template_id = key_values[TEMPLATE_ID_KEY]["value"] @@ -153,7 +153,7 @@ def start(self): f"Waiting for input file at {self.input_tasks_path}..." ) self.handshaker.retry_last_write() - time.sleep(self.polling_interval) + time.sleep(self.file_polling_interval) waiter_input_exists += 1 input_dict = json.loads(self.input_tasks_path.read_text()) @@ -166,7 +166,7 @@ def start(self): "Received command with wrong caller uuid: " f"{caller_uuid} or map uuid: {map_uuid}" ) - time.sleep(self.polling_interval) + time.sleep(self.file_polling_interval) waiter_wrong_uuid += 1 continue @@ -178,7 +178,7 @@ def start(self): if tasks_uuid == last_tasks_uuid: if waiter_wrong_uuid % 10 == 0: logger.info("Waiting for new tasks uuid") - time.sleep(self.polling_interval) + time.sleep(self.file_polling_interval) waiter_wrong_uuid += 1 else: input_tasks = input_dict["tasks"] @@ -210,7 +210,7 @@ def start(self): else: raise ValueError("Command unknown: {command}") - time.sleep(self.polling_interval) + time.sleep(self.file_polling_interval) def batch_input_tasks(self, input_tasks, n_of_batches): batches = [[] for _ in range(n_of_batches)] @@ -290,7 +290,7 @@ def run_job(self, job_inputs): return job_inputs, "SUCCESS" - with create_study_job( + with self.create_study_job( self.template_id, job_inputs, self.studies_api ) as job: job_status = self.studies_api.start_study_job( @@ -432,32 +432,31 @@ def read_keyvalues(self): return keyvalues - -@contextlib.contextmanager -def create_study_job(template_id, job_inputs, studies_api): - n_of_create_attempts = 0 - while True: - try: - n_of_create_attempts += 1 - job = studies_api.create_study_job( - study_id=template_id, - job_inputs=job_inputs, - ) - break - except osparc_client.exceptions.ApiException as api_exception: - if n_of_create_attempts >= self.max_job_create_attempts: - raise Exception( - f"Tried {n_of_create_attempts} times to create a job from " - "the study, but failed" - ) - else: - logger.exception(api_exception) - logger.info( - "Received an API Exception from server " - "when creating job, retrying..." + @contextlib.contextmanager + def create_study_job(self, template_id, job_inputs, studies_api): + n_of_create_attempts = 0 + while True: + try: + n_of_create_attempts += 1 + job = studies_api.create_study_job( + study_id=template_id, + job_inputs=job_inputs, ) + break + except osparc_client.exceptions.ApiException as api_exception: + if n_of_create_attempts >= self.max_job_create_attempts: + raise Exception( + f"Tried {n_of_create_attempts} times to create a job from " + "the study, but failed" + ) + else: + logger.exception(api_exception) + logger.info( + "Received an API Exception from server " + "when creating job, retrying..." + ) - try: - yield job - finally: - studies_api.delete_study_job(template_id, job.id) + try: + yield job + finally: + studies_api.delete_study_job(template_id, job.id) diff --git a/validation/inputs/parallelrunner.json b/validation/inputs/input_3/parallelrunner.json similarity index 100% rename from validation/inputs/parallelrunner.json rename to validation/inputs/input_3/parallelrunner.json From 0343fa29aaa90d124a321dcef73590bcda12e55c Mon Sep 17 00:00:00 2001 From: Werner Van Geit Date: Mon, 24 Jun 2024 13:28:13 +0200 Subject: [PATCH 07/10] Bump version --- .bumpversion.cfg | 2 +- .osparc/osparc-meta-parallelrunner/metadata.yml | 2 +- Makefile | 2 +- docker-compose-local.yml | 2 +- 4 files changed, 4 insertions(+), 4 deletions(-) diff --git a/.bumpversion.cfg b/.bumpversion.cfg index 307e31c..5855b3e 100755 --- a/.bumpversion.cfg +++ b/.bumpversion.cfg @@ -1,5 +1,5 @@ [bumpversion] -current_version = 0.0.12 +current_version = 0.0.13 commit = False message = service version: {current_version} → {new_version} tag = False diff --git a/.osparc/osparc-meta-parallelrunner/metadata.yml b/.osparc/osparc-meta-parallelrunner/metadata.yml index f6a7741..5699dc5 100755 --- a/.osparc/osparc-meta-parallelrunner/metadata.yml +++ b/.osparc/osparc-meta-parallelrunner/metadata.yml @@ -1,7 +1,7 @@ name: Parallel Runner description: "ParallelRunnerService" key: simcore/services/dynamic/osparc-meta-parallelrunner -version: 0.0.12 +version: 0.0.13 integration-version: 2.0.0 type: dynamic authors: diff --git a/Makefile b/Makefile index 600ae47..b847236 100755 --- a/Makefile +++ b/Makefile @@ -5,7 +5,7 @@ SHELL = /bin/sh MAKEFLAGS += -j2 export DOCKER_IMAGE_NAME ?= osparc-map -export DOCKER_IMAGE_TAG ?= 0.0.12 +export DOCKER_IMAGE_TAG ?= 0.0.13 export MASTER_AWS_REGISTRY ?= registry.osparc-master-zmt.click export MASTER_REGISTRY ?= registry.osparc-master.speag.com diff --git a/docker-compose-local.yml b/docker-compose-local.yml index eefd4fa..05a697c 100755 --- a/docker-compose-local.yml +++ b/docker-compose-local.yml @@ -1,7 +1,7 @@ version: '3.7' services: osparc-meta-parallelrunner: - image: simcore/services/dynamic/osparc-meta-parallelrunner:0.0.12 + image: simcore/services/dynamic/osparc-meta-parallelrunner:0.0.13 ports: - "8888:8888" environment: From 7a3b3dec4b12102e40fc77fc4bcbc45819051f11 Mon Sep 17 00:00:00 2001 From: Werner Van Geit Date: Mon, 24 Jun 2024 13:28:34 +0200 Subject: [PATCH 08/10] Bump version --- .bumpversion.cfg | 2 +- .osparc/osparc-meta-parallelrunner/metadata.yml | 2 +- Makefile | 2 +- docker-compose-local.yml | 2 +- 4 files changed, 4 insertions(+), 4 deletions(-) diff --git a/.bumpversion.cfg b/.bumpversion.cfg index 5855b3e..9bf794c 100755 --- a/.bumpversion.cfg +++ b/.bumpversion.cfg @@ -1,5 +1,5 @@ [bumpversion] -current_version = 0.0.13 +current_version = 0.0.14 commit = False message = service version: {current_version} → {new_version} tag = False diff --git a/.osparc/osparc-meta-parallelrunner/metadata.yml b/.osparc/osparc-meta-parallelrunner/metadata.yml index 5699dc5..418a14b 100755 --- a/.osparc/osparc-meta-parallelrunner/metadata.yml +++ b/.osparc/osparc-meta-parallelrunner/metadata.yml @@ -1,7 +1,7 @@ name: Parallel Runner description: "ParallelRunnerService" key: simcore/services/dynamic/osparc-meta-parallelrunner -version: 0.0.13 +version: 0.0.14 integration-version: 2.0.0 type: dynamic authors: diff --git a/Makefile b/Makefile index b847236..9bb411c 100755 --- a/Makefile +++ b/Makefile @@ -5,7 +5,7 @@ SHELL = /bin/sh MAKEFLAGS += -j2 export DOCKER_IMAGE_NAME ?= osparc-map -export DOCKER_IMAGE_TAG ?= 0.0.13 +export DOCKER_IMAGE_TAG ?= 0.0.14 export MASTER_AWS_REGISTRY ?= registry.osparc-master-zmt.click export MASTER_REGISTRY ?= registry.osparc-master.speag.com diff --git a/docker-compose-local.yml b/docker-compose-local.yml index 05a697c..e66faaa 100755 --- a/docker-compose-local.yml +++ b/docker-compose-local.yml @@ -1,7 +1,7 @@ version: '3.7' services: osparc-meta-parallelrunner: - image: simcore/services/dynamic/osparc-meta-parallelrunner:0.0.13 + image: simcore/services/dynamic/osparc-meta-parallelrunner:0.0.14 ports: - "8888:8888" environment: From ac0a6ca02b149e9d99fcb3c92330672a7fcdb2b7 Mon Sep 17 00:00:00 2001 From: Werner Van Geit Date: Mon, 24 Jun 2024 16:06:06 +0200 Subject: [PATCH 09/10] Check if template_id exists before running job --- docker_scripts/main.py | 5 ++++- docker_scripts/parallelrunner.py | 21 ++++++++++++++++++--- 2 files changed, 22 insertions(+), 4 deletions(-) diff --git a/docker_scripts/main.py b/docker_scripts/main.py index 4c02eee..78e0604 100755 --- a/docker_scripts/main.py +++ b/docker_scripts/main.py @@ -4,6 +4,7 @@ import socketserver import threading import time +import typing import pydantic as pyda import pydantic_settings @@ -64,7 +65,9 @@ class MainSettings(pydantic_settings.BaseSettings): batch_mode: bool = False file_polling_interval: int = 1 input_path: pyda.DirectoryPath = pyda.Field(alias="DY_SIDECAR_PATH_INPUTS") - output_path: pl.Path = pyda.Field(alias="DY_SIDECAR_PATH_OUTPUTS") + output_path: pyda.DirectoryPath = pyda.Field( + alias="DY_SIDECAR_PATH_OUTPUTS" + ) if __name__ == "__main__": diff --git a/docker_scripts/parallelrunner.py b/docker_scripts/parallelrunner.py index e9dc0a2..bf7e076 100755 --- a/docker_scripts/parallelrunner.py +++ b/docker_scripts/parallelrunner.py @@ -23,6 +23,7 @@ FILE_POLLING_INTERVAL = 1 # second MAX_JOB_CREATE_ATTEMPTS = 5 +JOB_CREATE_ATTEMPTS_DELAY = 5 MAX_TRIALS = 5 MAX_N_OF_WORKERS = 10 @@ -388,16 +389,22 @@ def map_func(batch, trial_number=1): "Worker has finished batch " f"{self.n_of_finished_batches} of {len(input_batches)}" ) + except ParallelRunner.FatalException as error: + logger.info( + f"Batch {batch} failed with fatal error ({error}) in " + f"trial {trial_number}, not retrying, raising error" + ) + raise error except Exception as error: if trial_number >= self.max_trials: logger.info( - f"Batch {batch} failed with error {error} in " + f"Batch {batch} failed with error ({error}) in " f"trial {trial_number}, not retrying, raising error" ) raise error else: logger.info( - f"Batch {batch} failed with error {error} in " + f"Batch {batch} failed with error ({error}) in " f"trial {trial_number}, retrying " ) batch = map_func(batch, trial_number=trial_number + 1) @@ -449,14 +456,22 @@ def create_study_job(self, template_id, job_inputs, studies_api): f"Tried {n_of_create_attempts} times to create a job from " "the study, but failed" ) + elif api_exception.reason.upper() == "NOT FOUND": + raise ParallelRunner.FatalException( + f"Study template not found: {template_id}" + ) else: logger.exception(api_exception) logger.info( - "Received an API Exception from server " + "Received an unhandled API Exception from server " "when creating job, retrying..." ) + time.sleep(JOB_CREATE_ATTEMPTS_DELAY) try: yield job finally: studies_api.delete_study_job(template_id, job.id) + + class FatalException(Exception): + pass From 64adcbc8d9563a48e15f6a4e10638b8ea817337d Mon Sep 17 00:00:00 2001 From: Werner Van Geit Date: Mon, 24 Jun 2024 16:25:53 +0200 Subject: [PATCH 10/10] Disable apiclient in test mode --- Dockerfile | 2 +- Makefile | 2 +- docker_scripts/entrypoint.bash | 2 +- docker_scripts/parallelrunner.py | 51 +++++++++++++++++++++----------- docker_scripts/requirements.txt | 2 -- 5 files changed, 37 insertions(+), 22 deletions(-) diff --git a/Dockerfile b/Dockerfile index d89671b..25a4eb9 100755 --- a/Dockerfile +++ b/Dockerfile @@ -15,7 +15,7 @@ RUN apt-get install --yes --no-install-recommends python3 python-is-python3 pyth # Copying boot scripts COPY docker_scripts /docker -RUN pip3 install pathos osparc --upgrade +RUN pip3 install pathos osparc pydantic-settings osparc-filecomms --upgrade USER osparcuser diff --git a/Makefile b/Makefile index 9bb411c..c6d1a47 100755 --- a/Makefile +++ b/Makefile @@ -4,7 +4,7 @@ SHELL = /bin/sh .DEFAULT_GOAL := help MAKEFLAGS += -j2 -export DOCKER_IMAGE_NAME ?= osparc-map +export DOCKER_IMAGE_NAME ?= osparc-meta-parallelrunner export DOCKER_IMAGE_TAG ?= 0.0.14 export MASTER_AWS_REGISTRY ?= registry.osparc-master-zmt.click diff --git a/docker_scripts/entrypoint.bash b/docker_scripts/entrypoint.bash index 163b215..7a95a27 100755 --- a/docker_scripts/entrypoint.bash +++ b/docker_scripts/entrypoint.bash @@ -4,7 +4,7 @@ set -euo pipefail IFS=$'\n\t' INFO="INFO: [$(basename "$0")] " -echo "$INFO" "Starting container for map ..." +echo "$INFO" "Starting container for parallelrunner ..." HOST_USERID=$(stat -c %u "${DY_SIDECAR_PATH_INPUTS}") HOST_GROUPID=$(stat -c %g "${DY_SIDECAR_PATH_INPUTS}") diff --git a/docker_scripts/parallelrunner.py b/docker_scripts/parallelrunner.py index bf7e076..7c9e856 100755 --- a/docker_scripts/parallelrunner.py +++ b/docker_scripts/parallelrunner.py @@ -44,6 +44,7 @@ def __init__( max_job_create_attempts=MAX_JOB_CREATE_ATTEMPTS, ): """Constructor""" + self.test_mode = False self.batch_mode = batch_mode self.max_n_of_workers = max_n_of_workers @@ -89,8 +90,12 @@ def setup(self): username=os.environ["OSPARC_API_KEY"], password=os.environ["OSPARC_API_SECRET"], ) - self.api_client = osparc.ApiClient(self.osparc_cfg) - self.studies_api = osparc_client.StudiesApi(self.api_client) + if self.test_mode: + self.api_client = None + self.studies_api = None + else: + self.api_client = osparc.ApiClient(self.osparc_cfg) + self.studies_api = osparc_client.StudiesApi(self.api_client) def start(self): """Start the Python Runner""" @@ -133,6 +138,11 @@ def start(self): if self.template_id is None: raise ValueError("Template ID can't be None") + if self.template_id == "TEST_UUID": + self.test_mode = True + self.api_client = None + self.studies_api = None + n_of_workers = key_values[N_OF_WORKERS_KEY]["value"] if n_of_workers is None: raise ValueError("Number of workers can't be None") @@ -247,20 +257,26 @@ def create_job_inputs(self, batch): tmp_input_file_path = tmp_dir_path / param_filename tmp_input_file_path.write_text(json.dumps(param_value)) - input_data_file = osparc.FilesApi( - self.api_client - ).upload_file(file=tmp_input_file_path) - processed_param_value = input_data_file + if self.test_mode: + processed_param_value = None + else: + input_data_file = osparc.FilesApi( + self.api_client + ).upload_file(file=tmp_input_file_path) + processed_param_value = input_data_file elif param_type == "file": file_info = json.loads(param_value) - input_data_file = osparc_client.models.file.File( - id=file_info["id"], - filename=file_info["filename"], - content_type=file_info["content_type"], - checksum=file_info["checksum"], - e_tag=file_info["e_tag"], - ) - processed_param_value = input_data_file + if self.test_mode: + processed_param_value = None + else: + input_data_file = osparc_client.models.file.File( + id=file_info["id"], + filename=file_info["filename"], + content_type=file_info["content_type"], + checksum=file_info["checksum"], + e_tag=file_info["e_tag"], + ) + processed_param_value = input_data_file elif param_type == "integer": processed_param_value = int(param_value) elif param_type == "float": @@ -285,7 +301,7 @@ def run_job(self, job_inputs): logger.debug(f"Sending inputs: {job_inputs}") - if self.template_id == "TEST_UUID": + if self.test_mode: logger.info("Map in test mode, just returning input") self.n_of_finished_batches += 1 @@ -423,8 +439,9 @@ def map_func(batch, trial_number=1): return output_tasks def teardown(self): - logger.info("Closing map ...") - self.api_client.close() + logger.info("Closing parallelrunner ...") + if self.api_client: + self.api_client.close() def read_keyvalues(self): """Read keyvalues file""" diff --git a/docker_scripts/requirements.txt b/docker_scripts/requirements.txt index 4be555f..e69de29 100755 --- a/docker_scripts/requirements.txt +++ b/docker_scripts/requirements.txt @@ -1,2 +0,0 @@ -osparc_filecomms -pydantic-settings