From 204adc511c923584988995f00a8c0a65a66e0e76 Mon Sep 17 00:00:00 2001 From: Florian Mirus Date: Fri, 11 Oct 2024 15:12:54 +0200 Subject: [PATCH 1/7] scenario_execution library for docker interaction (#205) --- docs/dictionary.txt | 6 +- docs/libraries.rst | 161 +++++++++++++++++- libs/scenario_execution_docker/MANIFEST.in | 1 + libs/scenario_execution_docker/README.md | 8 + libs/scenario_execution_docker/package.xml | 23 +++ .../resource/scenario_execution_docker | 0 .../scenario_execution_docker/__init__.py | 15 ++ .../actions/__init__.py | 15 ++ .../actions/docker_copy.py | 99 +++++++++++ .../actions/docker_exec.py | 105 ++++++++++++ .../actions/docker_put.py | 95 +++++++++++ .../actions/docker_run.py | 115 +++++++++++++ .../get_osc_library.py | 19 +++ .../lib_osc/docker.osc | 32 ++++ .../scenarios/test_docker_copy.osc | 13 ++ .../scenarios/test_docker_exec.osc | 10 ++ .../scenarios/test_docker_put.osc | 12 ++ .../scenarios/test_docker_run.osc | 9 + libs/scenario_execution_docker/setup.cfg | 4 + libs/scenario_execution_docker/setup.py | 51 ++++++ test/scenario_execution_docker_test/README.md | 3 + .../package.xml | 23 +++ .../resource/scenario_execution_docker_test | 0 test/scenario_execution_docker_test/setup.cfg | 4 + test/scenario_execution_docker_test/setup.py | 42 +++++ .../test/test_docker_copy.py | 81 +++++++++ .../test/test_docker_exec.py | 84 +++++++++ .../test/test_docker_put.py | 73 ++++++++ .../test/test_docker_run.py | 100 +++++++++++ 29 files changed, 1201 insertions(+), 2 deletions(-) create mode 100644 libs/scenario_execution_docker/MANIFEST.in create mode 100644 libs/scenario_execution_docker/README.md create mode 100644 libs/scenario_execution_docker/package.xml create mode 100644 libs/scenario_execution_docker/resource/scenario_execution_docker create mode 100644 libs/scenario_execution_docker/scenario_execution_docker/__init__.py create mode 100644 libs/scenario_execution_docker/scenario_execution_docker/actions/__init__.py create mode 100644 libs/scenario_execution_docker/scenario_execution_docker/actions/docker_copy.py create mode 100644 libs/scenario_execution_docker/scenario_execution_docker/actions/docker_exec.py create mode 100644 libs/scenario_execution_docker/scenario_execution_docker/actions/docker_put.py create mode 100644 libs/scenario_execution_docker/scenario_execution_docker/actions/docker_run.py create mode 100644 libs/scenario_execution_docker/scenario_execution_docker/get_osc_library.py create mode 100644 libs/scenario_execution_docker/scenario_execution_docker/lib_osc/docker.osc create mode 100644 libs/scenario_execution_docker/scenarios/test_docker_copy.osc create mode 100644 libs/scenario_execution_docker/scenarios/test_docker_exec.osc create mode 100644 libs/scenario_execution_docker/scenarios/test_docker_put.osc create mode 100644 libs/scenario_execution_docker/scenarios/test_docker_run.osc create mode 100644 libs/scenario_execution_docker/setup.cfg create mode 100644 libs/scenario_execution_docker/setup.py create mode 100644 test/scenario_execution_docker_test/README.md create mode 100644 test/scenario_execution_docker_test/package.xml create mode 100644 test/scenario_execution_docker_test/resource/scenario_execution_docker_test create mode 100644 test/scenario_execution_docker_test/setup.cfg create mode 100644 test/scenario_execution_docker_test/setup.py create mode 100644 test/scenario_execution_docker_test/test/test_docker_copy.py create mode 100644 test/scenario_execution_docker_test/test/test_docker_exec.py create mode 100644 test/scenario_execution_docker_test/test/test_docker_put.py create mode 100644 test/scenario_execution_docker_test/test/test_docker_run.py diff --git a/docs/dictionary.txt b/docs/dictionary.txt index f46eb76c..4a822588 100644 --- a/docs/dictionary.txt +++ b/docs/dictionary.txt @@ -33,4 +33,8 @@ png svg Kubernetes yaml -absolutized \ No newline at end of file +absolutized +moveit +replan +effector +mnt diff --git a/docs/libraries.rst b/docs/libraries.rst index 84c77a0d..b875dec7 100644 --- a/docs/libraries.rst +++ b/docs/libraries.rst @@ -10,6 +10,8 @@ Beside ``osc.standard`` provided by OpenSCENARIO 2 (which we divide into ``osc.s * - Name - Description + * - ``osc.docker`` + - Docker Library (provided with :repo_link:`libs/scenario_execution_docker`) * - ``osc.gazebo`` - Gazebo Library (provided with :repo_link:`libs/scenario_execution_gazebo`) * - ``osc.helpers`` @@ -32,6 +34,163 @@ Beside ``osc.standard`` provided by OpenSCENARIO 2 (which we divide into ``osc.s Additional features can be implemented by defining your own library. +Docker +------ + +The library contains actions to interact with `Docker `_. Import it with ``import osc.docker``. It's provided by the package :repo_link:`libs/scenario_execution_docker`. + +``docker_run()`` +^^^^^^^^^^^^^^^^ + +Runs a Docker container + +.. list-table:: + :widths: 15 15 5 65 + :header-rows: 1 + :class: tight-table + + * - Parameter + - Type + - Default + - Description + * - ``image`` + - ``string`` + - + - The image to run + * - ``command`` + - ``string`` + - + - The command to run in the container + * - ``container_name`` + - ``string`` + - + - The name for this container + * - ``detach`` + - ``bool`` + - false + - Whether to run container in the background + * - ``environment`` + - ``list of string`` + - + - Environment variables to set inside the container, i.e., a list of strings in the format ["SOMEVARIABLE=xxx"]. + * - ``network`` + - ``string`` + - + - Name of the network this container will be connected to at creation time + * - ``privileged`` + - ``bool`` + - false + - Give extended privileges to this container + * - ``remove`` + - ``bool`` + - true + - Remove the container when it as finished running + * - ``stream`` + - ``bool`` + - true + - If true and detach is false, return a log generator instead of a string. Ignored if detach is true. + * - ``volumes`` + - ``list of string`` + - + - A list of strings which each one of its elements specifies a mount volume: ['/home/user1/:/mount/vol2','/home/user2/:/mount/vol1'] + +``docker_exec()`` +^^^^^^^^^^^^^^^^^ + +Runs a command inside a given Docker container + +.. list-table:: + :widths: 15 15 5 65 + :header-rows: 1 + :class: tight-table + + * - Parameter + - Type + - Default + - Description + * - ``container`` + - ``string`` + - + - The name or id of the container to run the command in + * - ``container`` + - ``string`` + - + - The name or id of the container to run the command in + * - ``command`` + - ``string`` + - + - The command to run inside the container + * - ``environment`` + - ``list of string`` + - + - Environment variables to set inside the container, i.e., a list of strings in the format ["SOMEVARIABLE=xxx"]. + * - ``privileged`` + - ``bool`` + - false + - Give extended privileges to this container + * - ``user`` + - ``string`` + - root + - User to execute command as + * - ``workdir`` + - ``string`` + - + - Path to working directory for this exec session + +``docker_copy()`` +^^^^^^^^^^^^^^^^^ + +Copy a file or folder from the container. +Note that this actions potentially blocks other action calls if the copied content is large. +In case large files or folders need to be copied, consider mounting a volume to the container instead of this action. + +.. list-table:: + :widths: 15 15 5 65 + :header-rows: 1 + :class: tight-table + + * - Parameter + - Type + - Default + - Description + * - ``container`` + - ``string`` + - + - The name or id of the container to run the command in + * - ``file_path`` + - ``string`` + - + - Path to the file or folder inside the container to retrieve + +``docker_put()`` +^^^^^^^^^^^^^^^^^ + +Copy a file or folder from the local system into a running container. +Note that this actions potentially blocks other action calls if the copied content is large. +In case large files or folders need to be copied, consider mounting a volume to the container instead of this action. + +.. list-table:: + :widths: 15 15 5 65 + :header-rows: 1 + :class: tight-table + + * - Parameter + - Type + - Default + - Description + * - ``container`` + - ``string`` + - + - The name or id of the container to put the file or folder into + * - ``source_path`` + - ``string`` + - + - Path to the file or folder in the local system to copy + * - ``target_path`` + - ``string`` + - + - Target path inside the container to put the file or folder + Gazebo ------ @@ -1604,4 +1763,4 @@ Capture the screen content within a video. * - ``frame_rate`` - ``float`` - ``25.0`` - - Frame-rate of the resulting video \ No newline at end of file + - Frame-rate of the resulting video diff --git a/libs/scenario_execution_docker/MANIFEST.in b/libs/scenario_execution_docker/MANIFEST.in new file mode 100644 index 00000000..15f85d4a --- /dev/null +++ b/libs/scenario_execution_docker/MANIFEST.in @@ -0,0 +1 @@ +include scenario_execution_docker/lib_osc/*.osc diff --git a/libs/scenario_execution_docker/README.md b/libs/scenario_execution_docker/README.md new file mode 100644 index 00000000..432b872d --- /dev/null +++ b/libs/scenario_execution_docker/README.md @@ -0,0 +1,8 @@ +# Scenario Execution Library for Docker interaction + +The `scenario_execution_docker` package provides actions to interact with docker. + +It provides the following scenario execution library: + +- `docker.osc`: Actions to interact with docker + diff --git a/libs/scenario_execution_docker/package.xml b/libs/scenario_execution_docker/package.xml new file mode 100644 index 00000000..92bafcc9 --- /dev/null +++ b/libs/scenario_execution_docker/package.xml @@ -0,0 +1,23 @@ + + + + scenario_execution_docker + 1.2.0 + Scenario Execution library for docker interactions + Intel Labs + Intel Labs + Apache-2.0 + + scenario_execution + + python3-docker + + ament_copyright + ament_flake8 + ament_pep257 + python3-pytest + + + ament_python + + diff --git a/libs/scenario_execution_docker/resource/scenario_execution_docker b/libs/scenario_execution_docker/resource/scenario_execution_docker new file mode 100644 index 00000000..e69de29b diff --git a/libs/scenario_execution_docker/scenario_execution_docker/__init__.py b/libs/scenario_execution_docker/scenario_execution_docker/__init__.py new file mode 100644 index 00000000..3ba13780 --- /dev/null +++ b/libs/scenario_execution_docker/scenario_execution_docker/__init__.py @@ -0,0 +1,15 @@ +# Copyright (C) 2024 Intel Corporation +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, +# software distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions +# and limitations under the License. +# +# SPDX-License-Identifier: Apache-2.0 diff --git a/libs/scenario_execution_docker/scenario_execution_docker/actions/__init__.py b/libs/scenario_execution_docker/scenario_execution_docker/actions/__init__.py new file mode 100644 index 00000000..3ba13780 --- /dev/null +++ b/libs/scenario_execution_docker/scenario_execution_docker/actions/__init__.py @@ -0,0 +1,15 @@ +# Copyright (C) 2024 Intel Corporation +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, +# software distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions +# and limitations under the License. +# +# SPDX-License-Identifier: Apache-2.0 diff --git a/libs/scenario_execution_docker/scenario_execution_docker/actions/docker_copy.py b/libs/scenario_execution_docker/scenario_execution_docker/actions/docker_copy.py new file mode 100644 index 00000000..11b2c4d5 --- /dev/null +++ b/libs/scenario_execution_docker/scenario_execution_docker/actions/docker_copy.py @@ -0,0 +1,99 @@ +# Copyright (C) 2024 Intel Corporation +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, +# software distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions +# and limitations under the License. +# +# SPDX-License-Identifier: Apache-2.0 + +from enum import Enum + +import os +import docker +import tempfile +import tarfile +import py_trees +from scenario_execution.actions.base_action import BaseAction, ActionError + + +class CopyStatus(Enum): + IDLE = 1 + FOUND_CONTAINER = 2 + COPYING = 3 + DONE = 4 + + +class DockerCopy(BaseAction): + """ + Copy a file or folder from a running container + """ + + def __init__(self, container: str, file_path: str): + super().__init__() + self.container = container + self.file_path = file_path + + self.container_object = None + self.current_state = CopyStatus.IDLE + self.output_dir = None + self.client = None + self.result_data = None + + def setup(self, **kwargs): + # create docker client + self.client = docker.from_env() + + # check output_dir + if "output_dir" not in kwargs: + raise ActionError("output_dir not defined.", action=self) + + if kwargs['output_dir']: + if not os.path.exists(kwargs['output_dir']): + raise ActionError(f"Specified destination dir '{kwargs['output_dir']}' does not exist", action=self) + self.output_dir = kwargs['output_dir'] + + def update(self) -> py_trees.common.Status: + if self.current_state == CopyStatus.IDLE: + try: + self.container_object = self.client.containers.get(self.container) + self.current_state = CopyStatus.FOUND_CONTAINER + except docker.errors.APIError as e: + self.feedback_message = f"Docker container {self.container} not yet running {e}" # pylint: disable= attribute-defined-outside-init + return py_trees.common.Status.RUNNING + + if self.current_state == CopyStatus.FOUND_CONTAINER: + try: + self.result_data, _ = self.container_object.get_archive( + path=self.file_path) + self.current_state = CopyStatus.COPYING + self.feedback_message = f"Copying data from path {self.file_path} in container {self.container} to {self.output_dir}" # pylint: disable= attribute-defined-outside-init + except docker.errors.APIError as e: + self.feedback_message = f"Copying of data from path {self.file_path} failed: {e}" # pylint: disable= attribute-defined-outside-init + return py_trees.common.Status.FAILURE + + if self.current_state == CopyStatus.COPYING: + output_tar = tempfile.NamedTemporaryFile(suffix=".tar") + try: + with open(output_tar.name, 'wb') as f: + for chunk in self.result_data: + f.write(chunk) + with tarfile.open(output_tar.name, 'r') as tar: + tar.extractall(self.output_dir) + self.current_state = CopyStatus.DONE + except tarfile.ReadError as e: + self.feedback_message = f"Copying of data from path {self.file_path} failed: {e}" # pylint: disable= attribute-defined-outside-init + return py_trees.common.Status.FAILURE + + if self.current_state == CopyStatus.DONE: + self.feedback_message = f"Finished copying of data from path {self.file_path} to {self.output_dir}" # pylint: disable= attribute-defined-outside-init + return py_trees.common.Status.SUCCESS + + return py_trees.common.Status.RUNNING diff --git a/libs/scenario_execution_docker/scenario_execution_docker/actions/docker_exec.py b/libs/scenario_execution_docker/scenario_execution_docker/actions/docker_exec.py new file mode 100644 index 00000000..9d9e257c --- /dev/null +++ b/libs/scenario_execution_docker/scenario_execution_docker/actions/docker_exec.py @@ -0,0 +1,105 @@ +# Copyright (C) 2024 Intel Corporation +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, +# software distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions +# and limitations under the License. +# +# SPDX-License-Identifier: Apache-2.0 + +from enum import Enum + +import docker +import py_trees +from scenario_execution.actions.base_action import BaseAction + + +class ExecutionStatus(Enum): + IDLE = 1 + FOUND_CONTAINER = 2 + EXECUTING = 3 + DONE = 4 + + +class DockerExec(BaseAction): + """ + Run a command inside a container + """ + + def __init__(self, container: str, command: str, + environment: list, privileged: bool, + user: str, workdir: str): + super().__init__() + self.container = container + self.command = command + self.environment = environment + self.privileged = privileged + self.user = user + self.workdir = workdir + + self.client = None + self.container_object = None + self.execution_instance = None + self.execution_output = None + self.current_state = ExecutionStatus.IDLE + + def setup(self, **kwargs): + # create docker client + self.client = docker.from_env() + + def update(self) -> py_trees.common.Status: + if self.current_state == ExecutionStatus.IDLE: + try: + self.container_object = self.client.containers.get(self.container) + self.current_state = ExecutionStatus.FOUND_CONTAINER + except docker.errors.APIError as e: + self.feedback_message = f"Docker container {self.container} not yet running {e}" # pylint: disable= attribute-defined-outside-init + return py_trees.common.Status.RUNNING + + if self.current_state == ExecutionStatus.FOUND_CONTAINER: + try: + self.execution_instance = self.client.api.exec_create( + self.container_object.id, + self.command, + environment=self.environment, + privileged=self.privileged, + user=self.user, + workdir=self.workdir) + + self.execution_output = self.client.api.exec_start( + self.execution_instance['Id'], + tty=False, + stream=True + ) + self.current_state = ExecutionStatus.EXECUTING + self.feedback_message = f"Executing '{self.command}' in container {self.container}" # pylint: disable= attribute-defined-outside-init + except docker.errors.APIError as e: + self.feedback_message = f"Docker exec of command '{self.command}' failed: {e}" # pylint: disable= attribute-defined-outside-init + return py_trees.common.Status.FAILURE + + if self.current_state == ExecutionStatus.EXECUTING: + try: + log = next(self.execution_output) + self.feedback_message = f"Executing '{self.command}' in container {self.container} with output: {log.decode()}" # pylint: disable= attribute-defined-outside-init + except StopIteration: + self.current_state = ExecutionStatus.DONE + + if self.current_state == ExecutionStatus.DONE: + exit_metadata = self.client.api.exec_inspect(self.execution_instance['Id']) + assert not exit_metadata['Running'] + exit_code = exit_metadata['ExitCode'] + if exit_code == 0: + self.feedback_message = f"Finished execution of '{self.command}' in container {self.container}" # pylint: disable= attribute-defined-outside-init + return py_trees.common.Status.SUCCESS + else: + self.feedback_message = f"Execution of '{self.command}' in container {self.container} failed" # pylint: disable= attribute-defined-outside-init + return py_trees.common.Status.FAILURE + + return py_trees.common.Status.RUNNING diff --git a/libs/scenario_execution_docker/scenario_execution_docker/actions/docker_put.py b/libs/scenario_execution_docker/scenario_execution_docker/actions/docker_put.py new file mode 100644 index 00000000..8ea96dd9 --- /dev/null +++ b/libs/scenario_execution_docker/scenario_execution_docker/actions/docker_put.py @@ -0,0 +1,95 @@ +# Copyright (C) 2024 Intel Corporation +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, +# software distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions +# and limitations under the License. +# +# SPDX-License-Identifier: Apache-2.0 + +from enum import Enum + +import os +import docker +import tempfile +import tarfile +import py_trees +from scenario_execution.actions.base_action import BaseAction, ActionError + + +class CopyStatus(Enum): + IDLE = 1 + FOUND_CONTAINER = 2 + COPYING = 3 + DONE = 4 + + +class DockerPut(BaseAction): + """ + Copy a file or folder from the local filesystem into a running container + """ + + def __init__(self, container: str, source_path: str, target_path: str): + super().__init__() + self.container = container + self.source_path = source_path + self.target_path = target_path + + self.container_object = None + self.current_state = CopyStatus.IDLE + self.client = None + self.tar = None + + def setup(self, **kwargs): + # create docker client + self.client = docker.from_env() + + # check if source path exists + if not os.path.exists(self.source_path): + raise ActionError(f"The given source path {self.source_path} does not exist", action=self) + + def update(self) -> py_trees.common.Status: + if self.current_state == CopyStatus.IDLE: + try: + self.container_object = self.client.containers.get(self.container) + self.current_state = CopyStatus.FOUND_CONTAINER + except docker.errors.APIError as e: + self.feedback_message = f"Docker container {self.container} not yet running {e}" # pylint: disable= attribute-defined-outside-init + return py_trees.common.Status.RUNNING + + if self.current_state == CopyStatus.FOUND_CONTAINER: + self.tar = tempfile.NamedTemporaryFile(suffix=".tar") + try: + with tarfile.open(self.tar.name, 'w:') as tar: + tar.add( + self.source_path, + arcname=os.path.basename(self.source_path)) + self.current_state = CopyStatus.COPYING + except tarfile.ReadError as e: + self.feedback_message = f"Compressing data to a tar file from path {self.source_path} failed: {e}" # pylint: disable= attribute-defined-outside-init + return py_trees.common.Status.FAILURE + + if self.current_state == CopyStatus.COPYING: + success = self.container_object.put_archive( + path=self.target_path, + data=self.tar + ) + if success: + self.current_state = CopyStatus.DONE + self.feedback_message = f"Copying data from path {self.source_path} to {self.target_path} inside container {self.container}" # pylint: disable= attribute-defined-outside-init + else: + self.feedback_message = f"Copying data from path {self.source_path} to {self.target_path} inside container {self.container} failed: {e}" # pylint: disable= attribute-defined-outside-init + return py_trees.common.Status.FAILURE + + if self.current_state == CopyStatus.DONE: + self.feedback_message = f"Finished copying data from path {self.source_path} to {self.target_path} inside container {self.container}" # pylint: disable= attribute-defined-outside-init + return py_trees.common.Status.SUCCESS + + return py_trees.common.Status.RUNNING diff --git a/libs/scenario_execution_docker/scenario_execution_docker/actions/docker_run.py b/libs/scenario_execution_docker/scenario_execution_docker/actions/docker_run.py new file mode 100644 index 00000000..9350c780 --- /dev/null +++ b/libs/scenario_execution_docker/scenario_execution_docker/actions/docker_run.py @@ -0,0 +1,115 @@ +# Copyright (C) 2024 Intel Corporation +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, +# software distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions +# and limitations under the License. +# +# SPDX-License-Identifier: Apache-2.0 + +import os +from enum import Enum + +import docker +import py_trees +from scenario_execution.actions.base_action import BaseAction, ActionError + + +class ContainerStatus(Enum): + IDLE = 1 + RUNNING = 2 + DONE = 3 + + +class DockerRun(BaseAction): + """ + Run a container + """ + + def __init__(self, image: str, command: str, container_name: str, + detach: bool, environment: list, network: str, + privileged: bool, remove: bool, stream: bool, volumes: list): + super().__init__() + self.image = image + self.command = command + self.container_name = container_name + self.detach = detach + self.environment = environment + self.network = network + self.privileged = privileged + self.remove = remove + self.stream = stream + self.volumes = volumes + + self.client = None + self.container = None + self.current_state = ContainerStatus.IDLE + + def setup(self, **kwargs): + # create docker client + self.client = docker.from_env() + # check docker image + filterred_images = self.client.images.list(filters={'reference': self.image}) + if len(filterred_images) == 0: + raise ActionError(f"Required docker image '{self.image}' does not exist.", action=self) + + def update(self) -> py_trees.common.Status: + if self.current_state == ContainerStatus.IDLE: + try: + self.container = self.client.containers.run( + self.image, + command=self.command, + detach=self.detach, + environment=self.environment, + name=self.container_name, + network=self.network, + privileged=self.privileged, + stream=self.stream, + remove=self.remove, + user=os.getuid(), + group_add=[os.getgid()], + volumes=self.volumes) + except docker.errors.APIError as e: + self.feedback_message = f"Docker run failed: {e}" # pylint: disable= attribute-defined-outside-init + return py_trees.common.Status.FAILURE + self.current_state = ContainerStatus.RUNNING + self.feedback_message = f"Running docker container {self.image}" # pylint: disable= attribute-defined-outside-init + return py_trees.common.Status.RUNNING + + if self.current_state == ContainerStatus.RUNNING: + if self.stream and not self.detach: + try: + log = next(self.container) + self.feedback_message = f"Running container {self.image} with output: {log.decode()}" # pylint: disable= attribute-defined-outside-init + except StopIteration: + self.current_state = ContainerStatus.DONE + self.feedback_message = f"Docker container {self.image} finished cleanly" # pylint: disable= attribute-defined-outside-init + return py_trees.common.Status.SUCCESS + elif self.detach: + self.container.reload() + if self.container: + res = self.container.status + self.feedback_message = f"Running container {self.image} with status {res}" # pylint: disable= attribute-defined-outside-init + if res in ["removing", "exited"]: + self.current_state = ContainerStatus.DONE + self.feedback_message = f"Docker container {self.image} finished cleanly" # pylint: disable= attribute-defined-outside-init + return py_trees.common.Status.SUCCESS + + if self.current_state == ContainerStatus.DONE: + self.feedback_message = f"Docker container {self.image} finished cleanly" # pylint: disable= attribute-defined-outside-init + return py_trees.common.Status.SUCCESS + + return py_trees.common.Status.RUNNING + + def shutdown(self): + if self.container is None: + return + + self.container.stop(timeout=0) diff --git a/libs/scenario_execution_docker/scenario_execution_docker/get_osc_library.py b/libs/scenario_execution_docker/scenario_execution_docker/get_osc_library.py new file mode 100644 index 00000000..959aa95e --- /dev/null +++ b/libs/scenario_execution_docker/scenario_execution_docker/get_osc_library.py @@ -0,0 +1,19 @@ +# Copyright (C) 2024 Intel Corporation +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, +# software distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions +# and limitations under the License. +# +# SPDX-License-Identifier: Apache-2.0 + + +def get_osc_library(): + return 'scenario_execution_docker', 'docker.osc' diff --git a/libs/scenario_execution_docker/scenario_execution_docker/lib_osc/docker.osc b/libs/scenario_execution_docker/scenario_execution_docker/lib_osc/docker.osc new file mode 100644 index 00000000..85ac1007 --- /dev/null +++ b/libs/scenario_execution_docker/scenario_execution_docker/lib_osc/docker.osc @@ -0,0 +1,32 @@ +action docker_run: + # run a docker container + image: string # the image to run + command: string # the command to run in the container + container_name: string # the name for this container + detach: bool = false # Run container in the background + environment: list of string # Environment variables to set inside the container, i.e., a list of strings in the format ["SOMEVARIABLE=xxx"]. + network: string # Name of the network this container will be connected to at creation time + privileged: bool = false # Give extended privileges to this container + remove: bool = true # Remove the container when it as finished running + stream: bool = true # If true and detach is false, return a log generator instead of a string. Ignored if detach is true. + volumes: list of string # A list of strings which each one of its elements specifies a mount volume: ['/home/user1/:/mnt/vol2','/home/user2:/mnt/vol1'] + +action docker_exec: + # Run a command inside a given container + container: string # the name or id of the container to run the command in + command: string # the command to run inside the container + environment: list of string # Environment variables to set inside the container, i.e., a list of strings in the format ["SOMEVARIABLE=xxx"]. + privileged: bool = false # Give extended privileges to this container + user: string = 'root' # User to execute command as. Default: root + workdir: string # Path to working directory for this exec session + +action docker_copy: + # Copy a file or folder from the container + container: string # the name or id of the container to get the file or folder from + file_path: string # Path to the file or folder to retrieve + +action docker_put: + # Copy a file or folder from the local system into a running container + container: string # the name or id of the container to put the file or folder into + source_path: string # Path to the file or folder in the local system to copy + target_path: string # Target path inside the container to put the file or folder diff --git a/libs/scenario_execution_docker/scenarios/test_docker_copy.osc b/libs/scenario_execution_docker/scenarios/test_docker_copy.osc new file mode 100644 index 00000000..3fb0d7ff --- /dev/null +++ b/libs/scenario_execution_docker/scenarios/test_docker_copy.osc @@ -0,0 +1,13 @@ +import osc.docker +import osc.helpers + +scenario test_docker_copy: + timeout(25s) + do parallel: + docker_run(image: 'ubuntu', command: 'sleep 20', detach: true, container_name: 'sleeping_beauty', remove: true) + serial: + docker_exec(container: 'sleeping_beauty', command: 'mkdir -p /tmp/test_dir/') + docker_exec(container: 'sleeping_beauty', command: 'touch /tmp/test_dir/test.txt') + docker_exec(container: 'sleeping_beauty', command: 'touch /tmp/test_dir/test_1.txt') + docker_copy(container: 'sleeping_beauty', file_path: '/tmp/test_dir/') + emit end \ No newline at end of file diff --git a/libs/scenario_execution_docker/scenarios/test_docker_exec.osc b/libs/scenario_execution_docker/scenarios/test_docker_exec.osc new file mode 100644 index 00000000..2386eecb --- /dev/null +++ b/libs/scenario_execution_docker/scenarios/test_docker_exec.osc @@ -0,0 +1,10 @@ +import osc.docker +import osc.helpers + +scenario test_docker_exec: + timeout(15s) + do parallel: + docker_run(image: 'ubuntu', command: 'sleep 10', detach: true, container_name: 'sleeping_beauty', remove: true) + serial: + docker_exec(container: 'sleeping_beauty', command: 'echo hello world') + emit end \ No newline at end of file diff --git a/libs/scenario_execution_docker/scenarios/test_docker_put.osc b/libs/scenario_execution_docker/scenarios/test_docker_put.osc new file mode 100644 index 00000000..a4bc50ce --- /dev/null +++ b/libs/scenario_execution_docker/scenarios/test_docker_put.osc @@ -0,0 +1,12 @@ +import osc.docker +import osc.helpers + +# for this scenario to succeed, create a folder 'test_dir' in your /tmp folder with some dummy files in it +scenario test_docker_put: + timeout(10s) + do parallel: + docker_run(image: 'ubuntu', command: 'sleep 5', detach: true, container_name: 'sleeping_beauty', remove: true) + serial: + docker_put(container: 'sleeping_beauty', source_path: '/tmp/test_dir/', target_path: '/tmp/') + docker_exec(container: 'sleeping_beauty', command: 'ls /tmp/ | grep test_dir') + emit end \ No newline at end of file diff --git a/libs/scenario_execution_docker/scenarios/test_docker_run.osc b/libs/scenario_execution_docker/scenarios/test_docker_run.osc new file mode 100644 index 00000000..1f10ac2e --- /dev/null +++ b/libs/scenario_execution_docker/scenarios/test_docker_run.osc @@ -0,0 +1,9 @@ +import osc.docker +import osc.helpers + +scenario test_docker_run: + timeout(15s) + do serial: + docker_run(image: 'ubuntu', command: 'sleep 5', detach: true, remove: true, container_name: 'sleeping_beauty') + docker_run(image: 'ubuntu', command: 'echo hello world') + emit end \ No newline at end of file diff --git a/libs/scenario_execution_docker/setup.cfg b/libs/scenario_execution_docker/setup.cfg new file mode 100644 index 00000000..66ac754b --- /dev/null +++ b/libs/scenario_execution_docker/setup.cfg @@ -0,0 +1,4 @@ +[develop] +script_dir=$base/lib/scenario_execution_docker +[install] +install_scripts=$base/lib/scenario_execution_docker diff --git a/libs/scenario_execution_docker/setup.py b/libs/scenario_execution_docker/setup.py new file mode 100644 index 00000000..b46c4e63 --- /dev/null +++ b/libs/scenario_execution_docker/setup.py @@ -0,0 +1,51 @@ +# Copyright (C) 2024 Intel Corporation +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, +# software distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions +# and limitations under the License. +# +# SPDX-License-Identifier: Apache-2.0 + +""" Setup python package """ +from setuptools import find_namespace_packages, setup + +PACKAGE_NAME = 'scenario_execution_docker' + +setup( + name=PACKAGE_NAME, + version='1.2.0', + packages=find_namespace_packages(), + data_files=[ + ('share/ament_index/resource_index/packages', + ['resource/' + PACKAGE_NAME]), + ('share/' + PACKAGE_NAME, ['package.xml']) + ], + install_requires=['setuptools'], + zip_safe=True, + maintainer='Intel Labs', + maintainer_email='scenario-execution@intel.com', + description='Scenario Execution library for Docker interactions', + license='Apache License 2.0', + tests_require=['pytest'], + include_package_data=True, + entry_points={ + 'scenario_execution.actions': [ + 'docker_run = scenario_execution_docker.actions.docker_run:DockerRun', + 'docker_exec = scenario_execution_docker.actions.docker_exec:DockerExec', + 'docker_copy = scenario_execution_docker.actions.docker_copy:DockerCopy', + 'docker_put = scenario_execution_docker.actions.docker_put:DockerPut', + ], + 'scenario_execution.osc_libraries': [ + 'docker = ' + 'scenario_execution_docker.get_osc_library:get_osc_library', + ] + }, +) diff --git a/test/scenario_execution_docker_test/README.md b/test/scenario_execution_docker_test/README.md new file mode 100644 index 00000000..024938d4 --- /dev/null +++ b/test/scenario_execution_docker_test/README.md @@ -0,0 +1,3 @@ +# Tests of Scenario Execution Library for Docker + +The `scenario_execution_docker_test` package tests functionality of `scenario_execution_docker`. diff --git a/test/scenario_execution_docker_test/package.xml b/test/scenario_execution_docker_test/package.xml new file mode 100644 index 00000000..e29c3e3a --- /dev/null +++ b/test/scenario_execution_docker_test/package.xml @@ -0,0 +1,23 @@ + + + + scenario_execution_docker_test + 1.2.0 + Tests for Scenario Execution library for Docker + Intel Labs + Intel Labs + Apache-2.0 + + scenario_execution_docker + scenario_execution_os + geometry_msgs + + ament_copyright + ament_flake8 + ament_pep257 + python3-pytest + + + ament_python + + diff --git a/test/scenario_execution_docker_test/resource/scenario_execution_docker_test b/test/scenario_execution_docker_test/resource/scenario_execution_docker_test new file mode 100644 index 00000000..e69de29b diff --git a/test/scenario_execution_docker_test/setup.cfg b/test/scenario_execution_docker_test/setup.cfg new file mode 100644 index 00000000..84c2fa13 --- /dev/null +++ b/test/scenario_execution_docker_test/setup.cfg @@ -0,0 +1,4 @@ +[develop] +script_dir=$base/lib/scenario_execution_docker_test +[install] +install_scripts=$base/lib/scenario_execution_docker_test diff --git a/test/scenario_execution_docker_test/setup.py b/test/scenario_execution_docker_test/setup.py new file mode 100644 index 00000000..967571ce --- /dev/null +++ b/test/scenario_execution_docker_test/setup.py @@ -0,0 +1,42 @@ +# Copyright (C) 2024 Intel Corporation +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, +# software distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions +# and limitations under the License. +# +# SPDX-License-Identifier: Apache-2.0 + +""" Setup python package """ +from glob import glob +import os +from setuptools import find_namespace_packages, setup + +PACKAGE_NAME = 'scenario_execution_docker_test' + +setup( + name=PACKAGE_NAME, + version='1.2.0', + packages=find_namespace_packages(), + data_files=[ + ('share/ament_index/resource_index/packages', + ['resource/' + PACKAGE_NAME]), + ('share/' + PACKAGE_NAME, ['package.xml']), + (os.path.join('share', PACKAGE_NAME, 'launch'), glob('launch/*launch.py')), + ], + install_requires=['setuptools'], + zip_safe=True, + maintainer='Intel Labs', + maintainer_email='scenario-execution@intel.com', + description='Tests for Scenario Execution library for Docker', + license='Apache License 2.0', + tests_require=['pytest'], + entry_points={} +) diff --git a/test/scenario_execution_docker_test/test/test_docker_copy.py b/test/scenario_execution_docker_test/test/test_docker_copy.py new file mode 100644 index 00000000..e7d9b304 --- /dev/null +++ b/test/scenario_execution_docker_test/test/test_docker_copy.py @@ -0,0 +1,81 @@ +# Copyright (C) 2024 Intel Corporation +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, +# software distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions +# and limitations under the License. +# +# SPDX-License-Identifier: Apache-2.0 + +import py_trees +import unittest +import tempfile +from scenario_execution import ScenarioExecution +from scenario_execution.model.osc2_parser import OpenScenario2Parser +from scenario_execution.model.model_to_py_tree import create_py_tree +from scenario_execution.utils.logging import Logger + +from antlr4.InputStream import InputStream + + +class TestDockerCopy(unittest.TestCase): + # pylint: disable=missing-function-docstring,missing-class-docstring + + def setUp(self) -> None: + self.parser = OpenScenario2Parser(Logger('test', False)) + self.tmp_dir = tempfile.TemporaryDirectory() + self.scenario_execution = ScenarioExecution(debug=False, log_model=False, live_tree=False, + scenario_file="test.osc", output_dir=self.tmp_dir.name) + self.tree = py_trees.composites.Sequence(name="", memory=True) + + def tearDown(self): + self.tmp_dir.cleanup() + + def parse(self, scenario_content): + parsed_tree = self.parser.parse_input_stream(InputStream(scenario_content)) + model = self.parser.create_internal_model(parsed_tree, self.tree, "test.osc", False) + self.tree = create_py_tree(model, self.tree, self.parser.logger, False) + self.scenario_execution.tree = self.tree + self.scenario_execution.run() + + def test_success(self): + self.parse(""" +import osc.docker +import osc.helpers +import osc.os + +scenario test_success: + timeout(10) + do parallel: + docker_run(image: 'ubuntu', command: 'sleep 5', detach: true, container_name: 'sleeping_beauty_copy_success', remove: true) + serial: + docker_exec(container: 'sleeping_beauty_copy_success', command: 'mkdir -p /tmp/test_dir/') + docker_exec(container: 'sleeping_beauty_copy_success', command: 'touch /tmp/test_dir/test.txt') + docker_exec(container: 'sleeping_beauty_copy_success', command: 'touch /tmp/test_dir/test_1.txt') + docker_copy(container: 'sleeping_beauty_copy_success', file_path: '/tmp/test_dir/') + check_file_exists(file_name: '""" + self.tmp_dir.name + '/test_dir/test.txt' + """') + check_file_exists(file_name: '""" + self.tmp_dir.name + '/test_dir/test_1.txt' + """') + emit end +""") + self.assertTrue(self.scenario_execution.process_results()) + + def test_failure_missing_file(self): + self.parse(""" +import osc.docker +import osc.helpers + +scenario test_fail: + timeout(10s) + do parallel: + docker_run(image: 'ubuntu', command: 'sleep 5', detach: true, container_name: 'sleeping_beauty_copy_fail', remove: true) + serial: + docker_copy(container: 'sleeping_beauty_copy_fail', file_path: '/tmp/test_dir/') +""") + self.assertFalse(self.scenario_execution.process_results()) diff --git a/test/scenario_execution_docker_test/test/test_docker_exec.py b/test/scenario_execution_docker_test/test/test_docker_exec.py new file mode 100644 index 00000000..30675dc8 --- /dev/null +++ b/test/scenario_execution_docker_test/test/test_docker_exec.py @@ -0,0 +1,84 @@ +# Copyright (C) 2024 Intel Corporation +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, +# software distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions +# and limitations under the License. +# +# SPDX-License-Identifier: Apache-2.0 + +import py_trees +import unittest +import tempfile +from scenario_execution import ScenarioExecution +from scenario_execution.model.osc2_parser import OpenScenario2Parser +from scenario_execution.model.model_to_py_tree import create_py_tree +from scenario_execution.utils.logging import Logger + +from antlr4.InputStream import InputStream + + +class TestDockerExec(unittest.TestCase): + # pylint: disable=missing-function-docstring,missing-class-docstring + + def setUp(self) -> None: + self.parser = OpenScenario2Parser(Logger('test', False)) + self.scenario_execution = ScenarioExecution(debug=False, log_model=False, live_tree=False, + scenario_file="test.osc", output_dir=None) + self.tree = py_trees.composites.Sequence(name="", memory=True) + self.tmp_file = tempfile.NamedTemporaryFile() + + def parse(self, scenario_content): + parsed_tree = self.parser.parse_input_stream(InputStream(scenario_content)) + model = self.parser.create_internal_model(parsed_tree, self.tree, "test.osc", False) + self.tree = create_py_tree(model, self.tree, self.parser.logger, False) + self.scenario_execution.tree = self.tree + self.scenario_execution.run() + + def test_success(self): + self.parse(""" +import osc.docker +import osc.helpers + +scenario test_success: + timeout(30s) + do parallel: + docker_run(image: 'ubuntu', command: 'sleep 10', detach: true, container_name: 'sleeping_beauty_exec_success', remove: true) + serial: + docker_exec(container: 'sleeping_beauty_exec_success', command: 'echo hello world') + emit end +""") + self.assertTrue(self.scenario_execution.process_results()) + + def test_failure(self): + self.parse(""" +import osc.docker +import osc.helpers + +scenario test_failure: + timeout(15s) + do parallel: + docker_run(image: 'ubuntu', command: 'sleep 10', detach: true, container_name: 'sleeping_beauty_exec_fail', remove: true) + serial: + docker_exec(container: 'sleeping_beauty_exec_fail', command: 'ls UKNOWN_DIR') +""") + self.assertFalse(self.scenario_execution.process_results()) + + def test_failure_container_not_running(self): + self.parse(""" +import osc.docker +import osc.helpers + +scenario test_failure_container_not_running: + timeout(3s) + do serial: + docker_exec(container: 'sleeping_beauty', command: 'echo hello world') +""") + self.assertFalse(self.scenario_execution.process_results()) diff --git a/test/scenario_execution_docker_test/test/test_docker_put.py b/test/scenario_execution_docker_test/test/test_docker_put.py new file mode 100644 index 00000000..93872120 --- /dev/null +++ b/test/scenario_execution_docker_test/test/test_docker_put.py @@ -0,0 +1,73 @@ +# Copyright (C) 2024 Intel Corporation +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, +# software distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions +# and limitations under the License. +# +# SPDX-License-Identifier: Apache-2.0 + +import py_trees +import unittest +import tempfile +from scenario_execution import ScenarioExecution +from scenario_execution.model.osc2_parser import OpenScenario2Parser +from scenario_execution.model.model_to_py_tree import create_py_tree +from scenario_execution.utils.logging import Logger + +from antlr4.InputStream import InputStream + + +class TestDockerPut(unittest.TestCase): + # pylint: disable=missing-function-docstring,missing-class-docstring + + def setUp(self) -> None: + self.parser = OpenScenario2Parser(Logger('test', False)) + self.scenario_execution = ScenarioExecution(debug=False, log_model=False, live_tree=False, + scenario_file="test.osc", output_dir=None) + self.tree = py_trees.composites.Sequence(name="", memory=True) + self.tmp_file = tempfile.NamedTemporaryFile() + + def parse(self, scenario_content): + parsed_tree = self.parser.parse_input_stream(InputStream(scenario_content)) + model = self.parser.create_internal_model(parsed_tree, self.tree, "test.osc", False) + self.tree = create_py_tree(model, self.tree, self.parser.logger, False) + self.scenario_execution.tree = self.tree + self.scenario_execution.run() + + def test_success(self): + self.parse(""" +import osc.docker +import osc.helpers + +scenario test_success: + timeout(10s) + do parallel: + docker_run(image: 'ubuntu', command: 'sleep 5', detach: true, container_name: 'sleeping_beauty_put', remove: true) + serial: + docker_put(container: 'sleeping_beauty_put', source_path: '""" + self.tmp_file.name + """', target_path: '/tmp/') + docker_exec(container: 'sleeping_beauty_put', command: 'test -f """ + self.tmp_file.name + """') + emit end +""") + self.assertTrue(self.scenario_execution.process_results()) + + def test_failure_missing_file(self): + self.parse(""" +import osc.docker +import osc.helpers + +scenario test_failure_unknown_file: + timeout(10s) + do parallel: + docker_run(image: 'ubuntu', command: 'sleep 5', detach: true, container_name: 'sleeping_beauty_put_fail', remove: true) + serial: + docker_put(container: 'sleeping_beauty_put_fail', source_path: 'UNKNOWN', target_path: '/tmp/') +""") + self.assertFalse(self.scenario_execution.process_results()) diff --git a/test/scenario_execution_docker_test/test/test_docker_run.py b/test/scenario_execution_docker_test/test/test_docker_run.py new file mode 100644 index 00000000..182818c3 --- /dev/null +++ b/test/scenario_execution_docker_test/test/test_docker_run.py @@ -0,0 +1,100 @@ +# Copyright (C) 2024 Intel Corporation +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, +# software distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions +# and limitations under the License. +# +# SPDX-License-Identifier: Apache-2.0 + +import py_trees +import unittest +import tempfile +from scenario_execution import ScenarioExecution +from scenario_execution.model.osc2_parser import OpenScenario2Parser +from scenario_execution.model.model_to_py_tree import create_py_tree +from scenario_execution.utils.logging import Logger + +from antlr4.InputStream import InputStream + + +class TestDockerRun(unittest.TestCase): + # pylint: disable=missing-function-docstring,missing-class-docstring + + def setUp(self) -> None: + self.parser = OpenScenario2Parser(Logger('test', False)) + self.scenario_execution = ScenarioExecution(debug=False, log_model=False, live_tree=False, + scenario_file="test.osc", output_dir=None) + self.tree = py_trees.composites.Sequence(name="", memory=True) + self.tmp_dir = tempfile.TemporaryDirectory() + + def tearDown(self): + self.tmp_dir.cleanup() + + def parse(self, scenario_content): + parsed_tree = self.parser.parse_input_stream(InputStream(scenario_content)) + model = self.parser.create_internal_model(parsed_tree, self.tree, "test.osc", False) + self.tree = create_py_tree(model, self.tree, self.parser.logger, False) + self.scenario_execution.tree = self.tree + self.scenario_execution.run() + + def test_success_stream(self): + self.parse(""" +import osc.docker +import osc.helpers + +scenario test: + timeout(10s) + do serial: + docker_run(image: 'ubuntu', command: 'echo hello world') + emit end +""") + self.assertTrue(self.scenario_execution.process_results()) + + def test_success_detach(self): + self.parse(""" +import osc.docker +import osc.helpers + +scenario test: + timeout(10s) + do serial: + docker_run(image: 'ubuntu', command: 'sleep 5', detach: true) + emit end +""") + self.assertTrue(self.scenario_execution.process_results()) + + def test_fail_unknown_command(self): + self.parse(""" +import osc.docker +import osc.helpers + +scenario test: + timeout(10s) + do serial: + docker_run(image: 'ubuntu', command: 'UKNOWN', detach: true) + emit end +""") + self.assertFalse(self.scenario_execution.process_results()) + + def test_success_volume_mount(self): + self.parse(""" +import osc.docker +import osc.helpers + +scenario test: + timeout(10s) + do parallel: + docker_run(image: 'ubuntu', command: 'sleep 5', detach: true, volumes: ['""" + self.tmp_dir.name + """:/data'], container_name: 'sleeping_beauty_run_volume') + serial: + docker_exec(container: 'sleeping_beauty_run_volume', command: 'ls /data') + emit end +""") + self.assertTrue(self.scenario_execution.process_results()) From 705e7e9d5f9b92ff892da883fd81a79b4d382493 Mon Sep 17 00:00:00 2001 From: Nikhil Date: Sun, 20 Oct 2024 18:27:13 +0200 Subject: [PATCH 2/7] fix run_process action after retriggering (#215) --- scenario_execution/scenario_execution/actions/run_process.py | 1 + 1 file changed, 1 insertion(+) diff --git a/scenario_execution/scenario_execution/actions/run_process.py b/scenario_execution/scenario_execution/actions/run_process.py index 2f9f3a1b..df893249 100644 --- a/scenario_execution/scenario_execution/actions/run_process.py +++ b/scenario_execution/scenario_execution/actions/run_process.py @@ -44,6 +44,7 @@ def execute(self, command=None, wait_for_shutdown=True, shutdown_timeout=10, shu self.wait_for_shutdown = wait_for_shutdown self.shutdown_timeout = shutdown_timeout self.shutdown_signal = shutdown_signal[1] + self.executed = False def update(self) -> py_trees.common.Status: """ From cf5582dfc00af0f61b68e7a49aa74de0c955eb6e Mon Sep 17 00:00:00 2001 From: Florian Mirus Date: Mon, 21 Oct 2024 17:04:37 +0200 Subject: [PATCH 3/7] Documentation update with apt installation instructions (#220) --- docs/setup.rst | 40 +++++++++++++++++++++++++++++----------- 1 file changed, 29 insertions(+), 11 deletions(-) diff --git a/docs/setup.rst b/docs/setup.rst index 0119ede0..1f530da7 100644 --- a/docs/setup.rst +++ b/docs/setup.rst @@ -2,42 +2,60 @@ Setup ===== -Installation from source as ROS 2 workspace -------------------------------------------- +Installation with ROS 2 +----------------------- Prerequisites ^^^^^^^^^^^^^ -Install ROS2 humble following the `installation instructions `_. +Install ROS2 following the `installation instructions `_ for your distribution `$ROS_DISTRO`. -Installation -^^^^^^^^^^^^^ +Scenario execution currently supports the ROS 2 distributions `Humble `_ and `Jazzy `_. -Clone the scenario execution repository +Installation as Debian package (recommended) +^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ + +To install scenario execution together with all its libraries, run .. code-block:: bash - git clone https://github.com/IntelLabs/scenario_execution.git + sudo apt update && sudo apt install -y ros-$ROS_DISTRO-scenario_execution* + +To install just the core packages of scenario execution, run + +.. code-block:: bash + + sudo apt update && sudo apt install -y ros-$ROS_DISTRO-scenario_execution ros-$ROS_DISTRO-scenario_execution_ros ros-$ROS_DISTRO-scenario_execution_rviz -and update its submodules by running the following command in the root folder of the cloned repository + +Developer Installation (from source as ROS 2 workspace) +^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ + +Clone the scenario execution repository .. code-block:: bash - git submodule update --init + git clone https://github.com/IntelLabs/scenario_execution.git -install the necessary dependencies +and install the necessary dependencies .. code-block:: bash rosdep install --from-paths . --ignore-src pip3 install -r requirements.txt -and build it +Now, build your workspace by running .. code-block:: bash colcon build +and source your installation by running + +.. code-block:: bash + + source /opt/ros/$ROS_DISTRO/setup.bash && source install/setup.bash + .. _install_with_pip: Installation with pip as standalone Python package From e8dd4964a0ecb40bff0ce21b9359ac393ccc71e2 Mon Sep 17 00:00:00 2001 From: Florian Mirus Date: Thu, 14 Nov 2024 07:31:30 +0100 Subject: [PATCH 4/7] update from OpenSCENARIO 2.0 to OpenSCENARIO DSL V2.1.0 (#221) --- README.md | 2 +- docs/architecture.rst | 14 +- docs/index.rst | 4 +- docs/libraries.rst | 2 +- ...{openscenario2.rst => openscenarioDSL.rst} | 13 +- docs/tutorials.rst | 16 +- .../scenarios/example_external_method.osc | 4 +- examples/example_scenario/hello_world.osc | 2 +- .../lib_osc/floorplan_dsl.osc | 2 +- .../lib_osc/gazebo.osc | 2 +- .../lib_osc/kubernetes.osc | 2 +- .../test_kubernetes_create_delete.osc | 2 +- .../scenarios/test_kubernetes_pod_exec.osc | 2 +- .../lib_osc/moveit2.osc | 4 +- .../scenario_execution_nav2/lib_osc/nav2.osc | 4 +- .../lib_osc/pybullet.osc | 4 +- .../scenario_execution_x11/lib_osc/x11.osc | 2 +- scenario_execution/README.md | 4 +- .../scenario_execution/get_osc_library.py | 4 +- .../scenario_execution/lib_osc/helpers.osc | 2 +- .../scenario_execution/lib_osc/robotics.osc | 2 +- .../scenario_execution/lib_osc/standard.osc | 1013 +---------------- .../lib_osc/{standard_base.osc => types.osc} | 117 +- scenario_execution/setup.py | 2 +- .../test/test_osc2_parser_not_supported.py | 2 +- .../test/test_osc2_standard_osc.py | 2 +- scenario_execution/test/test_run_process.py | 10 +- 27 files changed, 116 insertions(+), 1123 deletions(-) rename docs/{openscenario2.rst => openscenarioDSL.rst} (89%) rename scenario_execution/scenario_execution/lib_osc/{standard_base.osc => types.osc} (69%) diff --git a/README.md b/README.md index d2761879..79cfb171 100644 --- a/README.md +++ b/README.md @@ -3,7 +3,7 @@ [![Super-Linter](https://github.com/IntelLabs/Scenario_Execution/actions/workflows/scan.yml/badge.svg)](https://github.com/marketplace/actions/super-linter) [![OpenSSF Scorecard](https://api.scorecard.dev/projects/github.com/IntelLabs/scenario_execution/badge)](https://scorecard.dev/viewer/?uri=github.com/IntelLabs/scenario_execution) -Scenario execution is a backend- and middleware-agnostic library written in Python based on the generic scenario description language [OpenSCENARIO 2](https://www.asam.net/static_downloads/public/asam-openscenario/2.0.0/welcome.html) and [pytrees](https://py-trees.readthedocs.io/en/devel/). +Scenario execution is a backend- and middleware-agnostic library written in Python based on the generic scenario description language [OpenSCENARIO DSL](https://www.asam.net/standards/detail/openscenario-dsl/) and [pytrees](https://py-trees.readthedocs.io/en/devel/). It reads a scenario definition from a file and then executes it, reusing available checks and actions. It is easily extendable through a library mechanism. This separation of the scenario definition from implementation massively reduces the manual efforts of scenario creation. diff --git a/docs/architecture.rst b/docs/architecture.rst index c80a9cce..bf687e7a 100644 --- a/docs/architecture.rst +++ b/docs/architecture.rst @@ -7,8 +7,8 @@ Architecture Overview of Scenario Execution -Scenario execution is built as a Python library on top of two open-source components: the generic scenario description language `OpenSCENARIO 2 `_ and `PyTrees `_. -In general, the user defines a scenario in the OpenSCENARIO 2 language, scenario execution parses the scenario, translates it to a behavior tree, executes it and finally gathers the test results. +Scenario execution is built as a Python library on top of two open-source components: the generic scenario description language `OpenSCENARIO DSL `_ and `PyTrees `_. +In general, the user defines a scenario in the OpenSCENARIO DSL language, scenario execution parses the scenario, translates it to a behavior tree, executes it and finally gathers the test results. .. figure:: images/scenario_execution_arch.png @@ -18,7 +18,7 @@ In general, the user defines a scenario in the OpenSCENARIO 2 language, scenario Our implementation is highly modular separating the core components from simulation- and/or middleware-specific modules realized through a plugin-based approach. In principle, any additional feature that is required by a specific scenario and that can be implemented in Python could be realized as additional library. -A library typically provides an OpenSCENARIO 2 file with additional definitions and may provide code implementing additional functionality such as conditions or actions. +A library typically provides an OpenSCENARIO DSL file with additional definitions and may provide code implementing additional functionality such as conditions or actions. Currently, the following sub-packages and libraries are available: @@ -63,11 +63,11 @@ The Internal Model Builder, implemented as a Model Listener does an initial chec Modules ------- -- ``scenario_execution``: The base package for scenario execution. It provides the parsing of OpenSCENARIO 2 files and the conversion to py-trees. It's middleware agnostic and can therefore be used as a basis for more specific implementations (e.g. ROS). It also provides basic OpenSCENARIO 2 libraries and actions. -- ``scenario_execution_ros``: This package uses ``scenario_execution`` as a basis and implements a ROS2 version of scenario execution. It provides a OpenSCENARIO 2 library with basic ROS2-related actions like publishing on a topic or calling a service. +- ``scenario_execution``: The base package for scenario execution. It provides the parsing of OpenSCENARIO DSL files and the conversion to py-trees. It's middleware agnostic and can therefore be used as a basis for more specific implementations (e.g. ROS). It also provides basic OpenSCENARIO DSL libraries and actions. +- ``scenario_execution_ros``: This package uses ``scenario_execution`` as a basis and implements a ROS2 version of scenario execution. It provides a OpenSCENARIO DSL library with basic ROS2-related actions like publishing on a topic or calling a service. - ``scenario_execution_control``: Provides code to control scenario execution (in ROS2) from another application such as RViz. -- ``scenario_execution_coverage``: Provides tools to generate concrete scenarios from abstract OpenSCENARIO 2 scenario definition and execute them. -- ``scenario_execution_gazebo``: Provides a `Gazebo `_-specific OpenSCENARIO 2 library with actions. +- ``scenario_execution_coverage``: Provides tools to generate concrete scenarios from abstract OpenSCENARIO DSL scenario definition and execute them. +- ``scenario_execution_gazebo``: Provides a `Gazebo `_-specific OpenSCENARIO DSL library with actions. - ``scenario_execution_interfaces``: Provides ROS2 `interfaces `__, more specifically, messages and services, which are used to interface ROS2 with the ``scenario_execution_control`` package. - ``scenario_execution_rviz``: Contains several `rviz `__ plugins for visualizing and controlling scenarios when working with ROS2. - ``simulation/gazebo_tf_publisher``: Publish ground truth transforms from simulation within TF. diff --git a/docs/index.rst b/docs/index.rst index be716594..7a5b1f8f 100644 --- a/docs/index.rst +++ b/docs/index.rst @@ -3,7 +3,7 @@ Scenario Execution ================== Scenario Execution for Robotics is a backend- and middleware-agnostic library, that enables the robotics community to perform reproducible experiments at scale and allows a seamless transition from simulation to real-world experiments. -Scenario Execution is written in Python and builds upon the generic scenario description language `OpenScenario2 `__ and `pytrees `__. +Scenario Execution is written in Python and builds upon the generic scenario description language `OpenSCENARIO DSL `__ and `pytrees `__. Scenario Execution reads a scenario definition from a file, translates it to a py-trees behavior tree and then executes it. This separation of the scenario definition from the implementation massively reduces the manual efforts of (robotics) scenario creation. Although Scenario Execution can be used as a pure Python library, it is mainly targeted to be used with the `Robot Operating System (ROS2) `__. The backend-agnostic implementation allows Scenario Execution to be used with both, robotics simulators such as `Gazebo `__ and physical robots, with minimal adaptations necessary in the scenario description file.  @@ -37,5 +37,5 @@ If you use Scenario Execution for Robotics in your scientific work, please cite architecture libraries development - openscenario2 + openscenarioDSL how_to_cite diff --git a/docs/libraries.rst b/docs/libraries.rst index b875dec7..0b9c930c 100644 --- a/docs/libraries.rst +++ b/docs/libraries.rst @@ -1,7 +1,7 @@ Libraries ========= -Beside ``osc.standard`` provided by OpenSCENARIO 2 (which we divide into ``osc.standard`` and ``osc.standard_base``), multiple libraries are provided with scenario execution. +Beside ``osc.standard`` and ``osc.types`` provided by OpenSCENARIO DSL, multiple libraries are provided with scenario execution. .. list-table:: :widths: 40 60 diff --git a/docs/openscenario2.rst b/docs/openscenarioDSL.rst similarity index 89% rename from docs/openscenario2.rst rename to docs/openscenarioDSL.rst index 310b4862..cb55a734 100644 --- a/docs/openscenario2.rst +++ b/docs/openscenarioDSL.rst @@ -1,17 +1,16 @@ -OpenSCENARIO 2 -============== +OpenSCENARIO DSL +================ General ------- -This tool supports a subset of the `OpenSCENARIO -2 `__ standard. +This tool supports a subset of the `OpenSCENARIO DSL `__ standard. The official documentation is available -`here `__. +`here `__. The `standard library of -OSC2 `__ +OSC2 `__ was adapted to be usable by the current parsing support of scenario execution. @@ -46,7 +45,7 @@ Mapping to py-trees Supported features ------------------ -In the following the OpenSCENARIO 2 keywords are listed with their current support status. +In the following the OpenSCENARIO DSL keywords are listed with their current support status. ======================= ==================== ============================= diff --git a/docs/tutorials.rst b/docs/tutorials.rst index 958d21a4..1685d705 100644 --- a/docs/tutorials.rst +++ b/docs/tutorials.rst @@ -8,13 +8,13 @@ Code for all tutorials is available in :repo_link:`examples`. Define and Execute Scenario --------------------------- -To create a scenario in OpenSCENARIO 2 syntax, first create a file +To create a scenario in OpenSCENARIO DSL syntax, first create a file with the extension ``.osc``. Input the following code in the file. .. code-block:: # import the libraries with import expression - import osc.standard.base + import osc.types import osc.helpers # declare the scenario by the syntax: "scenario scenario_name:" @@ -25,17 +25,17 @@ with the extension ``.osc``. Input the following code in the file. wait elapsed(3s) # wait three seconds log("Good Bye!") # log another message -The first two lines ``import osc.standard.base`` and ``import osc.helpers`` will import the named libraries that provide required definitions. In this example ``helpers`` library provides the ``log`` action and ``standard.base`` provides the definition of the `s` unit to specify seconds. +The first two lines ``import osc.types`` and ``import osc.helpers`` will import the named libraries that provide required definitions. In this example ``helpers`` library provides the ``log`` action and ``types`` provides the definition of the `s` unit to specify seconds. .. note:: - Comments in OpenSCENARIO 2 always start with ``#``. + Comments in OpenSCENARIO DSL always start with ``#``. Then, a scenario with the name ``hello_world`` get declared. The following colon states that all following and indented lines are part of it. The single top-level action of the scenario is defined in the ``do`` directive. The term ``serial`` states that the included actions will be executed in sequence. .. note:: - OpenSCENARIO 2 supports the following compositions: + OpenSCENARIO DSL supports the following compositions: * ``parallel``: execute actions in parallel, continue afterwards * ``serial``: execute actions, one after the other @@ -70,11 +70,11 @@ Create Scenario Library ----------------------- To add new features to scenario execution, extensions libraries can be created. An extension library typically provides one or more -OpenSCENARIO 2 definition files and might additionally provide action implementations. +OpenSCENARIO DSL definition files and might additionally provide action implementations. To show how to create such a library for scenario execution, we will add a ``custom_action`` action as an example. -First, we need to define the ``custom_action`` in a OpenSCENARIO 2 file. +First, we need to define the ``custom_action`` in a OpenSCENARIO DSL file. .. code-block:: @@ -417,7 +417,7 @@ It is possible to call external python methods and use their return value within .. code-block:: - import osc.standard.base + import osc.types import osc.helpers struct lib: diff --git a/examples/example_external_method/scenarios/example_external_method.osc b/examples/example_external_method/scenarios/example_external_method.osc index b43c5323..398f497a 100644 --- a/examples/example_external_method/scenarios/example_external_method.osc +++ b/examples/example_external_method/scenarios/example_external_method.osc @@ -1,4 +1,4 @@ -import osc.standard.base +import osc.types import osc.helpers struct lib: @@ -7,4 +7,4 @@ struct lib: scenario example_external_method: do serial: - log(lib.factorial(4)) \ No newline at end of file + log(lib.factorial(4)) diff --git a/examples/example_scenario/hello_world.osc b/examples/example_scenario/hello_world.osc index f2c178a5..6884b315 100644 --- a/examples/example_scenario/hello_world.osc +++ b/examples/example_scenario/hello_world.osc @@ -1,4 +1,4 @@ -import osc.standard.base +import osc.types import osc.helpers scenario hello_world: diff --git a/libs/scenario_execution_floorplan_dsl/scenario_execution_floorplan_dsl/lib_osc/floorplan_dsl.osc b/libs/scenario_execution_floorplan_dsl/scenario_execution_floorplan_dsl/lib_osc/floorplan_dsl.osc index 43b196d6..e406b93e 100644 --- a/libs/scenario_execution_floorplan_dsl/scenario_execution_floorplan_dsl/lib_osc/floorplan_dsl.osc +++ b/libs/scenario_execution_floorplan_dsl/scenario_execution_floorplan_dsl/lib_osc/floorplan_dsl.osc @@ -1,4 +1,4 @@ -import osc.standard.base +import osc.types actor floorplan_generator: result: string diff --git a/libs/scenario_execution_gazebo/scenario_execution_gazebo/lib_osc/gazebo.osc b/libs/scenario_execution_gazebo/scenario_execution_gazebo/lib_osc/gazebo.osc index bbe47e75..6134f3df 100644 --- a/libs/scenario_execution_gazebo/scenario_execution_gazebo/lib_osc/gazebo.osc +++ b/libs/scenario_execution_gazebo/scenario_execution_gazebo/lib_osc/gazebo.osc @@ -1,4 +1,4 @@ -import osc.standard.base +import osc.types import osc.robotics action actor_exists: diff --git a/libs/scenario_execution_kubernetes/scenario_execution_kubernetes/lib_osc/kubernetes.osc b/libs/scenario_execution_kubernetes/scenario_execution_kubernetes/lib_osc/kubernetes.osc index 76224aed..f830c6a3 100644 --- a/libs/scenario_execution_kubernetes/scenario_execution_kubernetes/lib_osc/kubernetes.osc +++ b/libs/scenario_execution_kubernetes/scenario_execution_kubernetes/lib_osc/kubernetes.osc @@ -1,4 +1,4 @@ -import osc.standard.base +import osc.types enum kubernetes_element_type: [ pod, diff --git a/libs/scenario_execution_kubernetes/scenarios/test_kubernetes_create_delete.osc b/libs/scenario_execution_kubernetes/scenarios/test_kubernetes_create_delete.osc index f13f3448..e6ddfa0d 100644 --- a/libs/scenario_execution_kubernetes/scenarios/test_kubernetes_create_delete.osc +++ b/libs/scenario_execution_kubernetes/scenarios/test_kubernetes_create_delete.osc @@ -1,4 +1,4 @@ -import osc.standard.base +import osc.types import osc.kubernetes import osc.helpers diff --git a/libs/scenario_execution_kubernetes/scenarios/test_kubernetes_pod_exec.osc b/libs/scenario_execution_kubernetes/scenarios/test_kubernetes_pod_exec.osc index 786b7236..b06c8e30 100644 --- a/libs/scenario_execution_kubernetes/scenarios/test_kubernetes_pod_exec.osc +++ b/libs/scenario_execution_kubernetes/scenarios/test_kubernetes_pod_exec.osc @@ -1,4 +1,4 @@ -import osc.standard.base +import osc.types import osc.kubernetes import osc.helpers diff --git a/libs/scenario_execution_moveit2/scenario_execution_moveit2/lib_osc/moveit2.osc b/libs/scenario_execution_moveit2/scenario_execution_moveit2/lib_osc/moveit2.osc index 457bde61..ff27bc90 100644 --- a/libs/scenario_execution_moveit2/scenario_execution_moveit2/lib_osc/moveit2.osc +++ b/libs/scenario_execution_moveit2/scenario_execution_moveit2/lib_osc/moveit2.osc @@ -1,4 +1,4 @@ -import osc.standard.base +import osc.types import osc.robotics enum move_group_type: [ @@ -46,4 +46,4 @@ action arm.move_to_pose: max_velocity_scaling_factor: float = 0.1 # Scaling factors for optionally reducing the maximum joint velocities namespace_override: string = '' # if set, it's used as namespace (instead of the associated actor's name) action_topic: string = 'move_action' # Name of action - success_on_acceptance: bool = false # succeed on goal acceptance \ No newline at end of file + success_on_acceptance: bool = false # succeed on goal acceptance diff --git a/libs/scenario_execution_nav2/scenario_execution_nav2/lib_osc/nav2.osc b/libs/scenario_execution_nav2/scenario_execution_nav2/lib_osc/nav2.osc index 439c9e27..47ec230f 100644 --- a/libs/scenario_execution_nav2/scenario_execution_nav2/lib_osc/nav2.osc +++ b/libs/scenario_execution_nav2/scenario_execution_nav2/lib_osc/nav2.osc @@ -1,4 +1,4 @@ -import osc.standard.base +import osc.types import osc.robotics action differential_drive_robot.init_nav2: @@ -21,4 +21,4 @@ action differential_drive_robot.nav_to_pose: goal_pose: pose_3d # goal pose to navigate to namespace_override: string = '' # if set, it's used as namespace (instead of the associated actor's name) action_topic: string = 'navigate_to_pose' # Name of action - success_on_acceptance: bool = false # succeed on goal acceptance \ No newline at end of file + success_on_acceptance: bool = false # succeed on goal acceptance diff --git a/libs/scenario_execution_pybullet/scenario_execution_pybullet/lib_osc/pybullet.osc b/libs/scenario_execution_pybullet/scenario_execution_pybullet/lib_osc/pybullet.osc index 7be517e5..fc87185e 100644 --- a/libs/scenario_execution_pybullet/scenario_execution_pybullet/lib_osc/pybullet.osc +++ b/libs/scenario_execution_pybullet/scenario_execution_pybullet/lib_osc/pybullet.osc @@ -1,4 +1,4 @@ -import osc.standard.base +import osc.types import osc.robotics actor simulation_pybullet @@ -20,4 +20,4 @@ action actor_pybullet.set_joint_motor_control: action actor_pybullet.distance_traveled: # Wait until a defined distance was traveled, based on simulation ground truth - distance: length # traveled distance at which the action succeeds. \ No newline at end of file + distance: length # traveled distance at which the action succeeds. diff --git a/libs/scenario_execution_x11/scenario_execution_x11/lib_osc/x11.osc b/libs/scenario_execution_x11/scenario_execution_x11/lib_osc/x11.osc index de8faaf7..0a6ebe62 100644 --- a/libs/scenario_execution_x11/scenario_execution_x11/lib_osc/x11.osc +++ b/libs/scenario_execution_x11/scenario_execution_x11/lib_osc/x11.osc @@ -1,4 +1,4 @@ -import osc.standard.base +import osc.types action capture_screen: # Capture the screen content within a video diff --git a/scenario_execution/README.md b/scenario_execution/README.md index 856eb1e3..3fac806f 100644 --- a/scenario_execution/README.md +++ b/scenario_execution/README.md @@ -4,8 +4,8 @@ The `scenario_execution` package is the core package of Scenario Execution for R Furthermore, it provides the following scenario execution libraries: -- `standard.osc`: The OpenSCENARIO 2 standard library. It is slightly modified to be in sync with the feature set of scenario execution and imports `standard_base.osc`. -- `standard_base.osc`: The base parts of the OpenSCENARIO 2 standard library such as units and basic structs. For convenience, numerical struct members are initialized with 0. +- `standard.osc`: The OpenSCENARIO DSL standard library. It is slightly modified to be in sync with the feature set of scenario execution and imports `standard_base.osc`. +- `standard_base.osc`: The base parts of the OpenSCENARIO DSL standard library such as units and basic structs. For convenience, numerical struct members are initialized with 0. - `robotics.osc`: robotic-specific specifications. - `helper.osc`: helper actions such as logging or running external processes. diff --git a/scenario_execution/scenario_execution/get_osc_library.py b/scenario_execution/scenario_execution/get_osc_library.py index 96acc85e..3c981c83 100644 --- a/scenario_execution/scenario_execution/get_osc_library.py +++ b/scenario_execution/scenario_execution/get_osc_library.py @@ -26,5 +26,5 @@ def get_standard_library(): return 'scenario_execution', 'standard.osc' -def get_standard_base_library(): - return 'scenario_execution', 'standard_base.osc' +def get_types_library(): + return 'scenario_execution', 'types.osc' diff --git a/scenario_execution/scenario_execution/lib_osc/helpers.osc b/scenario_execution/scenario_execution/lib_osc/helpers.osc index a632aa74..7f90f873 100644 --- a/scenario_execution/scenario_execution/lib_osc/helpers.osc +++ b/scenario_execution/scenario_execution/lib_osc/helpers.osc @@ -1,4 +1,4 @@ -import osc.standard.base +import osc.types enum signal: [ sighup = 1, diff --git a/scenario_execution/scenario_execution/lib_osc/robotics.osc b/scenario_execution/scenario_execution/lib_osc/robotics.osc index be271e58..4c5dfd75 100644 --- a/scenario_execution/scenario_execution/lib_osc/robotics.osc +++ b/scenario_execution/scenario_execution/lib_osc/robotics.osc @@ -1,3 +1,3 @@ -import osc.standard.base +import osc.types actor robot inherits osc_actor diff --git a/scenario_execution/scenario_execution/lib_osc/standard.osc b/scenario_execution/scenario_execution/lib_osc/standard.osc index af1d0d55..14d3859c 100644 --- a/scenario_execution/scenario_execution/lib_osc/standard.osc +++ b/scenario_execution/scenario_execution/lib_osc/standard.osc @@ -1,990 +1,23 @@ -########### -# Standard: OpenSCENARIO 2.0 domain model -# Copyright: ASAM 2021-2022 -# Version: -# Standard Publication Date: -# Publication Reference Document: -# Reference Document Date: -########### - -# ATTENTION: This file is modified to be compatible with -# the feature set of scenario execution. - -import osc.standard.base - -struct axle: - max_steering: angle # Mandatory: Maximum steering angle for the wheels on the axle. - wheel_diameter: length # Mandatory: Diameter for the wheels on this axle - track_width: length # Mandatory: Distance between the centerline of the outer wheels on opposing sides of the axle - position_x: length # Mandatory: Longitudinal position of the axle in the x-axis of the vehicle. For a 2-axle vehicle, the rear axle must have position_x = 0m - position_z: length # Mandatory: Vertical position of the axle in the vehicle's z-axis - number_of_wheels: uint # Mandatory: Number of wheels on the axle. - -struct crossing_type: - marking: crossing_marking # Optional: Define the type of markings on the crossing - use: crossing_use # Optional: Define the type of use for the crossing - elevation: crossing_elevation # Optional: Define the type of elevation for the crossing - - -######################### -# MODIFICATION: type definition must be done before usage -######################### - -enum path_interpolation: [ - straight_line, # Join the points with straight lines - smooth] # Join the points with a smooth line - -struct route: # Object of map - length: length # Optional: Nominal length of the route, measured along the s-axis of the route. Does not apply to route_point - directionality: directionality # Mandatory: Directionality for movement of traffic_participant actors on the route - min_lanes: uint # Optional: Minimum number of drivable lanes along this route. Applies only to these children: road, lane_section - max_lanes: uint # Optional: Maximum number of drivable lanes along this route. Applies only to these children: road, lane_section - anchors: list of string # Optional: The strings in here can be matched to unique items in the map files specified in file_name -# def start_point() -> route_point is undefined -# def end_point() -> route_point is undefined - -struct route_point inherits route_element: - route: route # Mandatory: route in which this point is located - #s: length = 0.0m # Optional: Coordinate along the s-axis of the corresponding route #TODO: s gets misinterpreted - t: length = 0.0m # Optional: Coordinate along the t-axis of the corresponding route - -struct xyz_point inherits route_element: - position: position_3d # Optional: Position in Cartesian (xyz) coordinates - -struct odr_point inherits route_element: - road_id: string # Mandatory: ASAM OpenDRIVE identifier for the road - lane_id: string # Optional: ASAM OpenDRIVE identifier for the lane. If specified, the t-coordinate is measured from the lane centerline. If not specified, the t-coordinate is measured from the ASAM OpenDRIVE reference line -# s: length = 0.0m # Optional: Coordinate along the ASAM OpenDRIVE s-axis - t: length = 0.0m # Optional: Coordinate along the ASAM OpenDRIVE t-axis - -# tag::library-path[] -struct path inherits route: - points: list of pose_3d # Mandatory: List of points in world x-y-z-coordinates. The individual pose_3d elements can have unconstrained coordinates. - interpolation: path_interpolation # Mandatory: Choose how to join the points of the path. - -struct relative_path: - interpolation: path_interpolation # Mandatory: Choose how to join the points of the path. - -struct relative_path_pose_3d inherits relative_path: - points: list of pose_3d # Mandatory: List of points in world x-y-z-coordinates. The individual pose_3d elements can have unconstrained coordinates. - -struct relative_path_st inherits relative_path: - points: list of route_point # Mandatory: Sequence of route_point that form the relative path - -struct relative_path_odr inherits relative_path: - points: list of odr_point # Mandatory: Sequence of odr_point that form the relative path -# end::library-path[] - -# tag::library-trajectory[] -struct trajectory: - points: list of pose_3d # Mandatory: List of points in world x-y-z-coordinates. The individual pose_3d elements can have unconstrained coordinates. - time_stamps: list of time # Mandatory: Time stamps for each element in points. The lists time_stamps and points must have the same length - interpolation: path_interpolation # Mandatory: Choose how to join the points of the trajectory. - -struct relative_trajectory: - time_stamps: list of time # Mandatory: Time stamps for each element in points. The lists time_stamps and points must have the same length - interpolation: path_interpolation # Mandatory: Choose how to join the points of the trajectory. - -struct relative_trajectory_pose_3d inherits relative_trajectory: - points: list of pose_3d # Mandatory: List of points in world x-y-z-coordinates. The individual pose_3d elements can have unconstrained coordinates. - -struct relative_trajectory_st inherits relative_trajectory: - points: list of route_point # Mandatory: Sequence of route_point that form the relative trajectory - -struct relative_trajectory_odr inherits relative_trajectory: - points: list of odr_point # Mandatory: Sequence of odr_point that form the relative trajectory -# end::library-trajectory[] - -# tag::library-shape-any[] -struct any_shape -# def duration() -> time is undefined - -struct any_acceleration_shape inherits any_shape -# def compute(time: time) -> acceleration is undefined - -struct any_speed_shape inherits any_shape -# def compute(time: time) -> speed is undefined - -struct any_position_shape inherits any_shape -# def compute(time: time) -> length is undefined - -struct any_lateral_shape inherits any_shape -# def compute(time: time) -> length is undefined -# end::library-shape-any[] - -# tag::library-shape-common[] -struct common_acceleration_shape inherits any_acceleration_shape: - rate_profile: dynamic_profile - rate_peak: jerk = 0.0mpspsps - target: acceleration = 0.0mpsps - -struct common_speed_shape inherits any_speed_shape: - rate_profile: dynamic_profile - rate_peak: acceleration = 0.0mpsps - target: speed = 0.0mps - -struct common_position_shape inherits any_position_shape: - rate_profile: dynamic_profile - rate_peak: speed = 0.0mps - target: length = 0.0m - -struct common_lateral_shape inherits any_lateral_shape: - rate_profile: dynamic_profile - rate_peak: speed = 0.0mps - target: length = 0.0m -# end::library-shape-common[] - -struct bm_engine # Reference to an object representing the bm_engine, whose content is implementation-dependent. - # Fields to be defined by the user or implementation - -struct behavioral_model: - bm_engine: bm_engine # Reference to the "behavioral model engine" - -########### -# Enums -########### - -enum color: [ - white, # RGB(255,255,255) - silver, # RGB(192,192,192) - gray, # RGB(128,128,128) - black, # RGB(0,0,0) - red, # RGB(255,0,0) - maroon, # RGB(128,0,0) - yellow, # RGB(255,255,0) - olive, # RGB(128,128,0) - lime, # RGB(0,255,0) - green, # RGB(0,128,0) - aqua, # RGB(0,255,255) - teal, # RGB(0,128,128) - blue, # RGB(0,0,255) - navy, # RGB(0,0,128) - fuchsia, # RGB(255,255,0) - purple] # RGB(255,0,255) - -enum intended_infrastructure: [ - driving, # OpenDRIVE: "normal" drivable road, which is not one of the other types - sidewalk, # OpenDRIVE: Lane on which pedestrians can walk safely - biking, # OpenDRIVE: Lane reserved for Cyclists - rail, # OpenDRIVE: Lane reserved for trains - tram, # OpenDRIVE: Lane reserved for trams - bus, # OpenDRIVE: Lane reserved for bus - taxi, # OpenDRIVE: Lane reserved for taxi - hov] # OpenDRIVE: Lane reserved for High Occupancy Vehicles (HOV) - -enum vehicle_category: [ - car, # Power-driven vehicle with maximum mass not exceeding 3.5 t having at least four wheels comprising not more than eight seats in addition to the driver seat. (UN category L7,M1,N1) - bus, # Power-driven vehicle having at least four wheels comprising more than eight seats in addition to the driver seat.(UN category M2,M3) - truck, # Power-driven vehicle with maximum mass exceeding 3.5t having at least four wheels. Designed for the carriage of goods. (UN category N2,N3) - trailer, # Vehicle designed to be towed by another vehicle. Non-permanently connected to towing vehicle, meaning it can be separated by an operation without involving facilities normally only found in workshop. - vru_vehicle, # Vehicle without a crash-resistant passenger cell intended for one to few passengers or small goods transport. With its occupant it results in a vulnerable road user. (UN category L1-L6 plus bicycles, pedelecs, e-bicycles, personal mobility devices, wheelchairs, mobility scooters and so on.) - other] # Unspecified but known type of vehicle (for example, stroller, shopping cart, etc) - -enum driving_rule: [ - left_hand_traffic, # Traffic drives on the left side of the road - right_hand_traffic] # Traffic drives on the right side of the road - -enum directionality: [ - uni_direction, # A traffic_participant can move legally in only one direction along the s-axis. - longitudinal, # Usually applies to lane_type driving and vru_vehicles - bi_direction, # A traffic_participant can move legally in both directions along the longitudinal s-axis. Usually applies to lane_type driving and vru_vehicles - split, # Applies for multi-lane elements: there are lanes with opposing uni_direction traffic flow within the route - free, # A traffic_participant can legally move in any direction (longitudinal or lateral). Usually applies to lane_type pedestrian or lane_use mix_traffic_vru - none, # No expected traffic flow. Usually applies to lane_type non_driving - other] # Other type of directionality - -enum lane_type: [ - driving, # Driving lane for road vehicles. See the driving_lane_use subtype - non_driving, # Non-driving lanes in road vehicles infrastructure. See the non_driving_lane_use subtype - vru_vehicles, # Lanes designated for VRU vehicles. See the vru_vehicles_lane_use subtype - pedestrian, # Lanes for pedestrians. See the pedestrian_lane_use subtype - other] # If the lane has another type - -enum lane_use: [ - normal, # A normal driving lane for road vehicles (OSI). Should be used in combination with lane_type == driving. - exit, # A deceleration lane in parallel to the main road (OSI). Should be used in combination with lane_type == driving. - entry, # An acceleration lane in parallel to the main road (OSI). Should be used in combination with lane_type == driving. - on_ramp, # A ramp from rural or urban roads joining a motorway (OSI). Should be used in combination with lane_type == driving. - off_ramp, # A ramp leading off a motorway onto rural or urban roads (OSI). Should be used in combination with lane_type == driving. - connecting_ramp, # A ramp that connects two motorways (OSI). Should be used in combination with lane_type == driving. - hov, # A lane for High Occupancy Vehicles (HOV), usually in highways. Should be used in combination with lane_type == driving. - bus, # A lane restricted for use only by busses. Should be used in combination with lane_type == driving. - mixed_traffic_vru, # A lane for mixed car and vru (vehicle and pedestrian) traffic, normally in urban areas. Should be used in combination with lane_type == driving or vru_vehicles. - parking, # A lane with parking spaces (OSI). Should be used in combination with lane_type == non_driving. - stop, # A hard shoulder on motorways for emergency stops (OSI). Should be used in combination with lane_type == non_driving. - restricted, # A lane on which road vehicles should not drive (OSI). Should be used in combination with lane_type == non_driving. - border, # A hard border on the edge of a road (OSI). Should be used in combination with lane_type == non_driving. - shoulder, # A soft border on the edge of a road (OSI). Should be used in combination with lane_type == non_driving. - curb, # An elevated surface with different height compared to the drivable lanes. Should be used in combination with lane_type == non_driving. - median, # An inaccessible lane for road vehicles and pedestrians. Typically used to separate the traffic. Should be used in combination with lane_type == non_driving. - bicycle, # A lane that is designated for bicycles (OSI). Should be used in combination with lane_type == vru_vehicles. - motorcycle, # A lane that is designated for motorcycles. Should be used in combination with lane_type == vru_vehicles. - sidewalk, # A lane that is designated for pedestrians (OSI). Should be used in combination with lane_type == pedestrian. - protected_sidewalk, # A lane for pedestrians with a barrier to separate it from road traffic. Should be used in combination with lane_type == pedestrian. - none, # The lane has no use. - other] # The lane has another use. - -enum side_left_right: [ - left, - right] - -enum crossing_marking: [ # Has connection to crossing_type - unmarked, # No crossing-markings on the road - marked, # The road or walking surface has markings that indicate a crossing - zebra, # Common type of marked crossing with thick zebra stripes - other] # Other type of markings for the crossing - -enum crossing_use: [ # Has connection to crossing_type - pedestrian, # Crossing is used by pedestrians (person, animal) and/or vehicles that usually move on sidewalks (wheelchair, stroller) - animal, # Animal crossing. For example, on a rural road or highway - bicycle, # Crossing for bicycles - rail_road, # Crossing for rail vehicles (train, subway, tram, ...) - other] # Other use for crossing - -enum crossing_elevation: [ - road_level, # Crossing is at same level as driving surface - curb_level, # Crossing is elevated from driving surface, often at the same level as a walking surface (sidewalk) or curb - refuge_island, # Along the crossing, the elevation may change between road and curb levels. For example, with refuge island(s) in the middle - other] # Another elevation type - -enum junction_direction: [ - straight, # The out_road is 0deg relative to the in_road - right, # The out_road is 90deg relative to the in_road - u_turn, # The out_road is 180deg relative to the in_road - left, # The out_road is 270deg relative to the in_road - other] # If none of the above apply - -enum route_overlap_kind: [ - equal, # Both routes have the same length, and coincide at the start and end points - start, # Both routes coincide at their start points - end, # Both routes coincide at their end points - inside, # The first route is fully inside the second route. Their start and end points do not have to coincide - any, # Any part of the first route overlaps with any part of the second route - other] # If none of the above apply - -enum lateral_overlap_kind: [ - never, # The two routes never overlap laterally. They never share a common lane. - sometimes, # In some segments of the route, the two routes can share a common lane. - always ] # The always routes share a common lane. - -enum dynamic_profile: [ - none, # No specific dynamic profile - constant, # Use constant first derivative - smooth, # Use smooth first derivative - asap] # Reach value as soon as possible - -enum lane_change_side: [ - left, # Lane to the left of the reference entity - right, # Lane to the right of the reference entity - inside, # Lane to the inside of the reference entity - outside, # Lane to the outside of the reference entity - same] # Same lane as the reference entity - -enum gap_direction: [ - ahead, # Gap in the positive direction of the s-axis, with respect to the reference entity - behind, # Gap in the negative direction of the s-axis, with respect to the reference entity - left, # Gap in the positive direction of the t-axis, with respect to the reference entity - right, # Gap in the negative direction of the t-axis, with respect to the reference entity - inside, # Gap in the direction pointing towards opposing traffic - outside] # Gap in the direction pointing away from opposing traffic - -enum headway_direction: [ - ahead, # Headway in the positive direction of the s-axis, with respect to the reference entity - behind] # Headway in the negative direction of the s-axis, with respect to the reference entity - -enum lat_measure_by: [ - left_to_left, - left_to_center, - left_to_right, - center_to_left, - center_to_right, - right_to_left, - right_to_center, - right_to_right, - closest] - -enum yaw_measure_by: [ - length_to_length, - length_to_width, - width_to_length, - width_to_width, - relative_to_north, - relative_to_road] - -enum orientation_measured_by: [ - absolute, - relative_to_reference, - relative_to_road] - -enum movement_options: [ - prefer_physical, # Perform the movement physical if possible - prefer_non_physical, # Perform the non physical way if the implementation allows that. For example, a test track may ignore this request. - must_be_physical] # An error message is issued, if this action cannot be physically performed for any reason. - -enum connect_route_points: [ - road, # Use the road element that contains this point - lane_section, # Use the lane_section element that contains this point - lane, # Use the lane element that contains this point - crossing, # Use the crossing element that contains this point - waypoint] # Use the point itself. The route must pass exactly through this point - -enum at: [ - start, - end, - all] - -enum movement_mode: [ - monotonous, # This movement mode adheres to the laws of physics. On top of that it limits the level of surprise that a movement may have. - other] # Not necessarily monotonous. This is the default. - -enum track: [ - actual, # Actual or projected. The default is actual, meaning that the vehicle reacts to the behavior of the referenced vehicle. - projected] # - -enum distance_direction: [ - longitudinal, # Measure distance in the x-coordinate. Positive means that the `reference` is in front of the `physical_object` that calls the method. - lateral] # Measure distance in the y-coordinate. Positive means that the `reference` is to the left of the `physical_object` that calls the method. - -enum distance_mode: [ - reference_points, # Measures the distance between the reference points. - bounding_boxes] # Measures the distance between the bounding boxes. - -enum relative_transform: [ - world_relative, - object_relative, - road_relative, - lane_relative -] - -enum on_route_type: [ on_road, on_lane_section, on_lane, on_crossing ] - -enum route_distance_enum: [ from_start, from_end ] - -########### -# Actor -########### - -actor physical_object inherits osc_actor: - bounding_box: bounding_box # Mandatory: See bounding_box struct - color: color # Optional: See color enum - geometry_reference: string # Optional: Opaque reference of an associated 3D geometry model of the physical object. It is implementation-specific how model references are resolved to 3D models. - center_of_gravity: position_3d # Mandatory: Center of gravity of the object. If unknown, the center of the bounding box may be used instead. - var pose: pose_3d # Position and orientation measured in world coordinates with world system as reference. -# def object_distance(reference: physical_object, direction: distance_direction, mode: distance_mode = reference_points) -> length is undefined -# def get_s_coord(route_type: on_route_type = on_road) -> length is undefined -# def get_t_coord(route_type: on_route_type= on_road) -> length is undefined -# def get_route_point(route_type: on_route_type = on_road) -> route_point is undefined - -actor stationary_object inherits physical_object - -actor movable_object inherits physical_object: -# def distance_along_route(route: route, from: route_distance_enum = from_start) -> length is undefined - var velocity: velocity_6d # Mandatory: Translational and rotational velocity measured in object coordinates with world system as reference. - var acceleration: acceleration_6d # Mandatory: Translational and rotational acceleration measured in object coordinates with world system as reference. - var speed: speed # Mandatory: Speed in center_of_gravity defined as sqrt(velocity.translational.x^2 + velocity.translational.y^2) * sign(velocity.translational.x) - -actor traffic_participant inherits movable_object: - intended_infrastructure: list of intended_infrastructure # Mandatory: See intended_infrastructure for definition. Intended usage is for further specification of an entity. For example, together with vehicle_category or to provide hints for implemenations where to spawn and / or auto-route entities. Note that multiple types of infrastructure can be assigned because of the list character of this type. -# def time_to_collision(reference: physical_object) -> time is undefined -# def time_headway(reference: physical_object) -> time is undefined -# def space_gap(reference: physical_object, direction: distance_direction) -> length is undefined - -actor vehicle inherits traffic_participant: - vehicle_category: vehicle_category # Mandatory: See vehicle_category - axles: list of axle # Mandatory: See axle - rear_overhang: length # Mandatory: Rear overhang of the vehicle or more explicitly the horizontal distance between the end of the bounding_box and the center of the rear axle. - -actor person inherits traffic_participant - -actor animal inherits traffic_participant - -########### -# environment (an Actor) -########### -struct weather: - air: air # Optional: See struct air - rain: precipitation # Optional: Liquid water in form of droplets falling under gravity. - snow: precipitation # Optional: Frozen water in delicately-crystalline flakes falling under gravity. - wind: wind # Optional: See struct wind - fog: fog # Optional: See struct fog - clouds: clouds # Optional: See struct clouds - -actor environment inherits osc_actor: - geodetic_position: geodetic_position_2d # Optional: Geodetic position of world coordinate frame regarding WGS84. Regarding usage for determination of angle of celestial light sources see remark above. - datetime: time # Optional: Date and time at start of scenario as Unix time, i.e. Number of seconds that have elapsed since January 1, 1970 (midnight UTC/GMT), not counting leap seconds. Regarding usage for determination of angle of celestial light sources see remark above. - var weather: weather # Optional: See struct weather - sun: celestial_light_source # Optional: Sun as instance of celestial_light_source. - moon: celestial_light_source # Optional: Moon as instance of celestial_light_source. -# def local_to_unix_time(year: uint, month: uint, day: uint, hour: uint, minute: uint, second: uint, time_zone: float) -> time is undefined - -struct air: - temperature: temperature = 25.0celsius # Optional: Temperature on ground level. - atmospheric_pressure: pressure = 101325pascal # Optional: Atmospheric pressure on ground level. - relative_humidity: float = 0.0 # Optional: Relative humidity on ground level. - -struct precipitation: - intensity: speed = 0.0mps # Optional: Global intensity of precipitation given as volumetric flux. In case of (partially) solid precipitation, the equivalent melted volume shall be considered. Note that volumetric flux is describing a volume flow across an area, but after reduction the unit results in the same unit as for speeds. As of now it is not possible in - -struct wind: - speed: speed # Mandatory: The expected value of wind speed. To estimate the expected value, rolling mean value over a specific short interval (for example, 3s) can be used. - direction: angle # Mandatory: The origin direction of the wind (not target direction) in the ground/x-y-plane with clockwise increasing values to match common definitions. This results in 0 deg for a wind blowing from the North 90 deg for a wind blowing from the East 90 deg, if x-axis and y-axis are mapped to East and North. - -struct fog: - visual_range: length # Mandatory: Value of optical range of visible light in the standard setting, which corresponds to a certain density of fog. - -struct clouds: - cloudiness: uint # Mandatory: Using okta scale to define which portion of the sky is covered with clouds. Ranging from 0 for completely clear sky to 8 for a completely overcast sky. Values above 8 shall not be used. - -struct celestial_light_source: - var position: celestial_position_2d # Mandatory: Position of the light source, see definition of physical type celestial_position_2d. - -########### -# Map (an Actor) -########### - -actor map inherits osc_actor: - map_file: string - routes: list of route - junctions: list of junction - driving_rule: driving_rule -# def odr_to_route_point(road_id: string, lane_id: string, s: length, t: length) -> route_point is undefined -# def xyz_to_route_point(x: length, y: length, z: length) -> route_point is undefined -# def route_point_to_xyz(route_point: route_point) -> xyz_point is undefined -# def outer_side() -> side_left_right is undefined -# def inner_side() -> side_left_right is undefined -# def create_route(routes: list of route, connect_points_by: connect_route_points, legal_route: bool) -> compound_route is undefined -# def create_route_point(route: route, s: length, t: length) -> route_point is undefined -# def create_xyz_point(x: length, y: length, z: length) -> xyz_point is undefined -# def create_odr_point(road_id: string, lane_id: string, s: length, t: length) -> odr_point is undefined -# def create_path(points: list of pose_3d, interpolation: path_interpolation) -> path is undefined -# def create_path_odr_points(points: list of odr_point, interpolation: path_interpolation, on_road_network: bool) -> path is undefined -# def create_path_route_points(points: list of route_point, interpolation: path_interpolation, on_road_network: bool) -> path is undefined -# def create_trajectory(points: list of pose_3d, time_stamps: list of time,interpolation: path_interpolation) -> trajectory is undefined -# def create_trajectory_odr_points(points: list of odr_point, time_stamps: list of time, interpolation: path_interpolation, on_road_network: bool) -> trajectory is undefined -# def create_trajectory_route_points(points: list of route_point, time_stamps: list of time, interpolation: path_interpolation, on_road_network: bool) -> trajectory is undefined -# def resolve_relative_path(relative_path: relative_path, reference: physical_object, transform: relative_transform) -> path is undefined -# def resolve_relative_trajectory(relative_trajectory: relative_trajectory, reference: physical_object, transform: relative_transform) -> trajectory is undefined -# def get_map_file() -> string is undefined #from map.map_file - -struct route: # Object of map - length: length # Optional: Nominal length of the route, measured along the s-axis of the route. Does not apply to route_point - directionality: directionality # Mandatory: Directionality for movement of traffic_participant actors on the route - min_lanes: uint # Optional: Minimum number of drivable lanes along this route. Applies only to these children: road, lane_section - max_lanes: uint # Optional: Maximum number of drivable lanes along this route. Applies only to these children: road, lane_section - anchors: list of string # Optional: The strings in here can be matched to unique items in the map files specified in file_name -# def start_point() -> route_point is undefined -# def end_point() -> route_point is undefined - -struct route_element inherits route # Interface Class (of route) - -struct road inherits route_element: - s_positive: list of lane_section # Mandatory: List of lane_section elements that flow in the positive direction of the road s-axis - s_negative: list of lane_section # Optional: List of lane_section elements that flow in the negative direction of the road s-axis - -struct lane_section inherits route_element: - road: road # Mandatory: Where the lane_section resides - lanes: list of lane # Mandatory: List of lanes that compose the lane_section - s_axis: lane # Mandatory: Choose, which lane is used to determine the s-axis of the lane_section. Must be a member of it.lanes - -struct lane inherits route_element: - lane_section: lane_section # Mandatory: Where the lane resides - lane_type: lane_type # Mandatory: Type of lane - lane_use: lane_use # Mandatory: A subtype of the lane_type. Use compatible pairs of lane_type and lane_use - width: length # Optional: Nominal width of the lane - -struct crossing inherits route_element: - start_lane: lane # Mandatory: Crossing starts on this lane - end_lane: lane # Mandatory: Crossing ends on this lane - start_s_coord: length # Mandatory: On the starts_from lane, the crossing connects at this point in the lane s-axis (and zero in the t-axis) - end_s_coord: length # Mandatory: On the ends_on lane, the crossing connects at this point in the lane s-axis (and zero in the t-axis) - width: length # Mandatory: Nominal width of the crossing, measured perpendicular to the crossing s-axis - crossing_type: crossing_type # Mandatory: Type of crossing - -struct junction: - roads: list of road # Mandatory: A list of road - -struct compound_route inherits route: - route_elements: list of route_element # Mandatory: A list of route_element. - -struct compound_lane inherits route: - lanes: list of lane # Mandatory: A list of lane - -########### -# Action - movable_object -########### - -action osc_actor.osc_action - -action movable_object.action_for_movable_object inherits osc_actor.osc_action - -action movable_object.move inherits movable_object.action_for_movable_object - -action movable_object.remain_stationary inherits movable_object.action_for_movable_object - -action movable_object.assign_position inherits movable_object.action_for_movable_object: - position: position_3d # Optional: Desired 3-dimensional position assigned by the user - route_point: route_point # Optional: Desired route_point assigned by the user - odr_point: odr_point # Optional: Desired odr_point assigned by the user - -action movable_object.assign_orientation inherits movable_object.action_for_movable_object: - target: orientation_3d # Mandatory: Desired 3-dimensional orientation assigned by the user - -action movable_object.assign_speed inherits movable_object.action_for_movable_object: - target: speed # Mandatory: Desired (scalar) speed assigned by the user - -action movable_object.assign_acceleration inherits movable_object.action_for_movable_object: - target: acceleration # Mandatory: Desired (scalar) acceleration assigned by the user - -action movable_object.replay_path inherits movable_object.action_for_movable_object: - absolute: path # Mandatory: Absolute path. Includes a list of points - relative: relative_path # Mandatory: Relative path. Includes a list of points -# reference: physical_object with: # Optional: Use with relative paths. Specify the reference entity that defines the origin for the point coordinates. Default: the actor itself -# keep(default it == actor) -# transform: relative_transform with: # Optional: Use with relative paths. Coordinates of the points are relative to the reference entity. Default = object_relative -# keep(default it == object_relative) -# start_offset: length with: # Optional: Offset at which to begin following the path, measured from the path's start. Default = 0m -# keep(default it == 0m) -# end_offset: length with: # Optional: Offset at which to end following the path, measured from the path's end. Default = 0m -# keep(default it == 0m) - -action movable_object.replay_trajectory inherits movable_object.action_for_movable_object: - absolute: trajectory # Mandatory: Absolute trajectory. Includes a list of points and a list of corresponding time stamps - relative: relative_trajectory # Mandatory: Relative trajectory. Includes a list of points and a list of corresponding time stamps -# reference: physical_object with: # Optional: Use with relative trajectories. Specify the reference entity that defines the origin for the point coordinates. Default: the actor itself -# keep(default it == actor) -# transform: relative_transform with: # Optional: Use with relative trajectories. Coordinates of the points are relative to the reference entity. Default = object_relative -# keep(default it == object_relative) -# start_offset: length with: # Optional: Offset at which to begin following the trajectory, measured from the trajectory's start. Default = 0m -# keep(default it == 0m) -# end_offset: length with: # Optional: Offset at which to end following the trajectory, measured from the trajectory's end. Default = 0m -# keep(default it == 0m) - -action movable_object.change_position inherits movable_object.action_for_movable_object: - target: position_3d # Mandatory: Target value for the position at the end of the action - interpolation: path_interpolation # Mandatory: The interpolation method used to join the start and end points - on_road_network: bool # Mandatory: The action takes place completely on the road network of the scenario - -action movable_object.change_speed inherits movable_object.action_for_movable_object: - target: speed # Mandatory: Target value for the speed at the end of the action -# rate_profile: dynamic_profile with: # Optional: Assign a shape for the change of the speed variable. This profile affects the acceleration during action execution -# keep(default it == none) - rate_peak: acceleration = 0.0mpsps # Optional: Target value for the peak acceleration that must be achieved during the action - -# action movable_object.keep_speed inherits movable_object.action_for_movable_object - -action movable_object.change_acceleration inherits movable_object.action_for_movable_object: - target: acceleration # Mandatory: Target value for the scalar acceleration at the end of the action -# rate_profile: dynamic_profile with: # Optional: Assign a shape for the change of the speed variable. This profile affects the jerk during action execution -# keep(default it == none) - rate_peak: jerk = 0.0mpspsps # Optional: Target value for the peak jerk that must be achieved during the action - -action movable_object.keep_acceleration inherits movable_object.action_for_movable_object - -action movable_object.follow_path inherits movable_object.action_for_movable_object: - absolute: path # Mandatory: Absolute path. Includes a list of points - relative: relative_path # Mandatory: Relative path. Includes a list of points -# reference: physical_object with: # Optional: Use with relative paths. Specify the reference entity that defines the origin for the point coordinates. Default: the actor itself -# keep(default it == actor) -# transform: relative_transform with: # Optional: Use with relative paths. Coordinates of the points are relative to the reference entity. Default = object_relative -# keep(default it == object_relative) -# start_offset: length with: # Optional: Offset at which to begin following the path, measured from the path's start. Default = 0m -# keep(default it == 0m) -# end_offset: length with: # Optional: Offset at which to end following the path, measured from the path's end. Default = 0m -# keep(default it == 0m) - -action movable_object.follow_trajectory inherits movable_object.action_for_movable_object: - absolute: trajectory # Mandatory: Absolute trajectory. Includes a list of points and a list of corresponding time stamps - relative: relative_trajectory # Mandatory: Relative trajectory. Includes a list of points and a list of corresponding time stamps -# reference: physical_object with: # Optional: Use with relative trajectories. Specify the reference entity that defines the origin for the point coordinates. Default: the actor itself -# keep(default it == actor) -# transform: relative_transform with: # Optional: Use with relative trajectories. Coordinates of the points are relative to the reference entity. Default = object_relative -# keep(default it == object_relative) -# start_offset: length with: # Optional: Offset at which to begin following the trajectory, measured from the trajectory's start. Default = 0m -# keep(default it == 0m) -# end_offset: length with: # Optional: Offset at which to end following the trajectory, measured from the trajectory's end. Default = 0m -# keep(default it == 0m) - -########### -# Action - vehicle -########### - -action vehicle.action_for_vehicle inherits movable_object.action_for_movable_object - -action vehicle.drive inherits vehicle.action_for_vehicle - -action vehicle.follow_lane inherits vehicle.action_for_vehicle: -# offset: length with: # Optional: Default=0.0. Offset from center of the lane for the actor to follow, using the t-axis of the lane -# keep(default it == 0.0m) -# rate_profile: dynamic_profile with: # Optional: Assign a shape for the change of the lateral position variable (t-axis). This profile affects the lateral velocity during action execution -# keep(default it == none) - rate_peak: speed = 0.0mps # Optional: Target value for the peak lateral velocity that must be achieved during the action - target: lane # Optional: The actor must be in this lane at the start, throughout, and the end of the action. If this argument is ignored, the actor follows the current lane when the action is invoked - -action vehicle.change_lane inherits vehicle.action_for_vehicle: -# num_of_lanes: int with: # Optional: The target lane is "num_of_lanes" to the side of the reference entity. Use in conjunction with "side" -# keep(default it == 1) - side: lane_change_side # Optional: Select on which side of the reference entity -# reference: physical_object with: # Optional: Default=it.actor. Reference to the entity that is used to determine the target lane. If this argument is omitted, the actor itself is used as reference -# keep(default it == actor) -# offset: length with: # Optional: Default=0.0. Target offset from center of the target lane that the actor follows at the end of the action -# keep(default it == 0.0m) -# rate_profile: dynamic_profile with: # Optional: Assign a shape for the change of the lateral position variable (t-axis). This profile affects the lateral velocity during action execution '/ -# keep(default it == none) - rate_peak: speed = 0.0mps # Optional: Target value for the peak lateral velocity that must be achieved during the action - target: lane # Mandatory: The actor starts and finishes the action in the target lane - -action vehicle.change_space_gap inherits vehicle.action_for_vehicle: - target: length # Mandatory: Target distance between the actor and the reference entity. Distance is measured according to the space_gap() method - direction: gap_direction # Mandatory: Placement of the actor with respect to the reference entity. [ahead, behind] means distance is measured in the s-axis. [left, right, inside, outside] means distance is measured in the t-axis - reference: physical_object # Mandatory: The actor reaches the driving distance to this reference entity - -action vehicle.keep_space_gap inherits vehicle.action_for_vehicle: - reference: physical_object # Mandatory: The actor keeps the driving distance to this reference entity - direction: distance_direction # Mandatory: Direction in which the space gap is kept with respect to the reference entity. [longitudinal] to keep distance in the s-axis. [lateral] to keep distance in the t-axis - -action vehicle.change_time_headway inherits vehicle.action_for_vehicle: - target: time # Mandatory: Target time headway between the actor and the reference entity. Time headway is measured according to the time_headway() method - direction: headway_direction # Mandatory: Placement of the actor with respect to the reference entity - reference: physical_object # Mandatory: The actor reaches the time headway to this reference entity - -action vehicle.keep_time_headway inherits vehicle.action_for_vehicle: - reference: physical_object # Mandatory: The actor keeps the driving distance to this reference entity - direction: headway_direction # Mandatory: Direction in which the space gap is kept with respect to the reference entity. [longitudinal] to keep distance in the s-axis. [lateral] to keep distance in the t-axis - -########### -# Action - person -########### - -action person.action_for_person inherits movable_object.action_for_movable_object - -action person.walk inherits person.action_for_person - -########### -# Action - environment -########### - -action environment.action_for_environment inherits osc_actor.osc_action - -action environment.air inherits environment.action_for_environment: - temperature: temperature = 25.0celsius - pressure: pressure = 101325pascal - relative_humidity: float = 0.0 - -action environment.rain inherits environment.action_for_environment: - intensity: speed = 0.0mps - -action environment.snow inherits environment.action_for_environment: - intensity: speed = 0.0mps - -action environment.wind inherits environment.action_for_environment: - speed: speed = 0.0mps - direction: angle = 0.0rad - -action environment.fog inherits environment.action_for_environment: - visual_range: length = 0.0m - -action environment.cloud inherits environment.action_for_environment: - cloudiness: uint = 0 - -action environment.assign_celestial_position inherits environment.action_for_environment: - light_source: celestial_light_source - azimuth: angle - elevation: angle - -##################################### -# Modifier - location based modifiers -##################################### - -# modifier position of movable_object.action_for_movable_object: -# distance: length # Mandatory: A value with a distance unit. -# time: time with: # Mandatory: A value with a time unit. -# keep(default it == 0sec) -# ahead_of, behind: physical_object # Optional: A named instance of the vehicle actor example: example, vehicle2 - -# at: at with: # Optional: uses enum at to support: start | end | all (default: all) -# keep(default it == all) -# movement_mode: movement_mode with: # Optional: uses enum movement_mode: monotonous | other (default: other) -# keep(default it == other) -# track: track with: # Optional: uses enum track: actual | projected (default: actual) -# keep(default it == actual) -# shape: any_shape # Optional: uses struct any_shape - -# modifier distance of movable_object.action_for_movable_object: -# distance: length # Mandatory: The distance the actor should travel in the current movement. - -# at: at with: # Optional: uses enum at to support: start | end | all (default: all) -# keep(default it == all) -# movement_mode: movement_mode with: # Optional: uses enum movement_mode: monotonous | other (default: other) -# keep(default it == other) -# track: track with: # Optional: uses enum track: actual | projected (default: actual) -# keep(default it == actual) -# shape: any_shape # Optional: uses struct any_shape - -# modifier lane of movable_object.action_for_movable_object: -# lane: uint with: # Mandatory: An integer indicating a required lane for a drive -# keep(default it == 1) -# same_as: physical_object -# side_of: physical_object -# side: side_left_right -# from: side_left_right - -# at: at with: # Optional: uses enum at to support: start | end | all (default: all) -# keep(default it == all) -# movement_mode: movement_mode with: # Optional: uses enum movement_mode: monotonous | other (default: other) -# keep(default it == other) -# track: track with: # Optional: uses enum track: actual | projected (default: actual) -# keep(default it == actual) -# shape: any_shape # Optional: uses struct any_shape - -# modifier keep_lane of movable_object.action_for_movable_object: -# at: at with: # Optional: uses enum at to support: start | end | all (default: all) -# keep(default it == all) -# movement_mode: movement_mode with: # Optional: uses enum movement_mode: monotonous | other (default: other) -# keep(default it == other) -# track: track with: # Optional: uses enum track: actual | projected (default: actual) -# keep(default it == actual) -# shape: any_shape # Optional: uses struct any_shape - -# modifier lateral of movable_object.action_for_movable_object: -# distance: length with: # Mandatory: The offset from reference line. The default is [-10.0..10.0]centimeter. -# keep(default it in [-10.0cm..10.0cm]) -# left_of, right_of, same_as, side_of: physical_object # Optional: -# side: side_left_right # Optional: -# measure_by: lat_measure_by with: # Optional: This parameter specifies the measurement start and end points. -# keep(default it == closest) - -# at: at with: # Optional: uses enum at to support: start | end | all (default: all) -# keep(default it == all) -# movement_mode: movement_mode with: # Optional: uses enum movement_mode: monotonous | other (default: other) -# keep(default it == other) -# track: track with: # Optional: uses enum track: actual | projected (default: actual) -# keep(default it == actual) -# shape: any_shape # Optional: uses struct any_shape - -# modifier yaw of movable_object.action_for_movable_object: -# angle: angle # Mandatory: A value with an angle unit. This parameter is mandatory -# relative_to: physical_object # Optional: A named instance of the vehicle actor example: example, vehicle2 -# measure_by: yaw_measure_by with: # Optional: Defines reference lines of the context and referenced objects, where width is an x-axis and long is a y-axis. -# keep(default it == relative_to_road) - -# at: at with: # Optional: uses enum at to support: start | end | all (default: all) -# keep(default it == all) -# movement_mode: movement_mode with: # Optional: uses enum movement_mode: monotonous | other (default: other) -# keep(default it == other) -# track: track with: # Optional: uses enum track: actual | projected (default: actual) -# keep(default it == actual) -# shape: any_shape # Optional: uses struct any_shape - -# modifier orientation of movable_object.action_for_movable_object: -# yaw: angle # Mandatory: A value with an angle unit. This parameter is mandatory for either yaw, pitch and roll -# pitch: angle # Mandatory: A value with an angle unit. This parameter is mandatory for either yaw, pitch and roll -# roll: angle # Mandatory: A value with an angle unit. This parameter is mandatory for either yaw, pitch and roll -# relative_to: physical_object # Optional: A named instance of the vehicle actor example: example, vehicle2 -# measured_by: orientation_measured_by with: # Optional: defines how to measure the desired orientation. Values include absolute, relative_to_reference, relative_to_road. the default is relative_to_road -# keep(default it == relative_to_road) - -# at: at with: # Optional: uses enum at to support: start | end | all (default: all) -# keep(default it == all) -# movement_mode: movement_mode with: # Optional: uses enum movement_mode: monotonous | other (default: other) -# keep(default it == other) -# track: track with: # Optional: uses enum track: actual | projected (default: actual) -# keep(default it == actual) -# shape: any_shape # Optional: uses struct any_shape - -#modifier along of movable_object.action_for_movable_object: -# route: route # Mandatory: The route to move on. -# start_offset: length with: # Optional: Offset at which to begin following the route, measured from the route's start. Default = 0m -# keep(default it == 0m) -# end_offset: length # Optional: Offset at which to end following the route, measured from the route's end. Default = 0m -# keep(default it == 0m) - -# at: at with: # Optional: uses enum at to support: start | end | all (default: all) -# keep(default it == all) -# movement_mode: movement_mode with: # Optional: uses enum movement_mode: monotonous | other (default: other) -# keep(default it == other) -# track: track with: # Optional: uses enum track: actual | projected (default: actual) -# keep(default it == actual) -# shape: any_shape # Optional: uses struct any_shape - -# modifier keep_position of movable_object.action_for_movable_object: -# shape: any_shape # Optional: uses struct any_shape - -# at: at with: # Optional: uses enum at to support: start | end | all (default: all) -# keep(default it == all) -# movement_mode: movement_mode with: # Optional: uses enum movement_mode: monotonous | other (default: other) -# keep(default it == other) -# track: track with: # Optional: uses enum track: actual | projected (default: actual) -# keep(default it == actual) - -#modifier along_trajectory of movable_object.action_for_movable_object: -# trajectory: trajectory # The trajectory to move along. -# start_offset: length with: # Offset at which to begin following the trajectory, measured from the trajectory's start. Default = 0m -# keep(default it == 0m) -# end_offset: length with: # Offset at which to end following the trajectory, measured from the trajectory's end. Default = 0m -# keep(default it == 0m) - -# at: at with: # Optional: uses enum at to support: start | end | all (default: all) -# keep(default it == all) -# movement_mode: movement_mode with: # Optional: uses enum movement_mode: monotonous | other (default: other) -# keep(default it == other) -# track: track with: # Optional: uses enum track: actual | projected (default: actual) -# keep(default it == actual) - -# modifier stationary_object.location: -# pose: pose_3d # Mandatory. Location of the stationary object for the whole scenario. - -########################################### -# Modifier - rate of change based modifiers -########################################### - -# modifier speed of movable_object.action_for_movable_object: -# speed: speed # Mandatory: The vehicle's desired speed. -# faster_than, slower_than, same_as: physical_object # Optional: A named instance of the actor example: example, vehicle2 - -# at: at with: # Optional: uses enum at to support: start | end | all (default: all) -# keep(default it == all) -# movement_mode: movement_mode with: # Optional: uses enum movement_mode: monotonous | other (default: other) -# keep(default it == other) -# track: track with: # Optional: uses enum track: actual | projected (default: actual) -# keep(default it == actual) -# shape: any_shape # Optional: uses struct any_shape - -# modifier acceleration of movable_object.action_for_movable_object: -# acceleration: acceleration # A value appended with an acceleration unit. -# faster_than, slower_than, same_as: physical_object # Optional: A named instance of the actor example: example, vehicle2 - -# at: at with: # Optional: uses enum at to support: start | end | all (default: all) -# keep(default it == all) -# movement_mode: movement_mode with: # Optional: uses enum movement_mode: monotonous | other (default: other) -# keep(default it == other) -# track: track with: # Optional: uses enum track: actual | projected (default: actual) -# keep(default it == actual) -# shape: any_shape # Optional: uses struct any_shape - -# modifier keep_speed of movable_object.action_for_movable_object: -# at: at with: # Optional: uses enum at to support: start | end | all (default: all) -# keep(default it == all) -# movement_mode: movement_mode with: # Optional: uses enum movement_mode: monotonous | other (default: other) -# keep(default it == other) -# track: track with: # Optional: uses enum track: actual | projected (default: actual) -# keep(default it == actual) -# shape: any_shape # Optional: uses struct any_shape - -# modifier change_speed of movable_object.action_for_movable_object: -# speed: speed # Mandatory: The vehicle's desired speed. - -# at: at with: # Optional: uses enum at to support: start | end | all (default: all) -# keep(default it == all) -# movement_mode: movement_mode with: # Optional: uses enum movement_mode: monotonous | other (default: other) -# keep(default it == other) -# track: track with: # Optional: uses enum track: actual | projected (default: actual) -# keep(default it == actual) -# shape: any_shape # Optional: uses struct any_shape - -# modifier physical_movement of movable_object.action_for_movable_object: -# option: movement_options - -# at: at with: # Optional: uses enum at to support: start | end | all (default: all) -# keep(default it == all) -# movement_mode: movement_mode with: # Optional: uses enum movement_mode: monotonous | other (default: other) -# keep(default it == other) -# track: track with: # Optional: uses enum track: actual | projected (default: actual) -# keep(default it == actual) -# shape: any_shape # Optional: uses struct any_shape - -# modifier avoid_collisions of movable_object.action_for_movable_object: -# avoid: bool # Mandatory: Either true or false. - -# at: at with: # Optional: uses enum at to support: start | end | all (default: all) -# keep(default it == all) -# movement_mode: movement_mode with: # Optional: uses enum movement_mode: monotonous | other (default: other) -# keep(default it == other) -# track: track with: # Optional: uses enum track: actual | projected (default: actual) -# keep(default it == actual) -# shape: any_shape # Optional: uses struct any_shape - -# modifier change_lane of movable_object.action_for_movable_object: -# lane: uint with: -# keep(default it == 1) # Mandatory: The number of lanes to change from. The default is 1. -# side: side_left_right # Optional: Left or right. The side is randomized if not specified. - -# at: at with: # Optional: uses enum at to support: start | end | all (default: all) -# keep(default it == all) -# movement_mode: movement_mode with: # Optional: uses enum movement_mode: monotonous | other (default: other) -# keep(default it == other) -# track: track with: # Optional: uses enum track: actual | projected (default: actual) -# keep(default it == actual) -# shape: any_shape # Optional: uses struct any_shape - -# ################################ -# # Modifier - map based modifiers -# ################################ - -# modifier map.number_of_lanes: -# route: route # Mandatory: The route that has these constraints -# num_of_lanes: uint # Mandatory: The desired number of lanes -# lane_type: lane_type with: # Optional: Apply the constraint to the number of lanes with this type. -# keep(default it == driving) -# lane_use: lane_use with: # Optional: Apply the constraint to the number of lanes with this use. -# keep(default it == normal) -# directionality: directionality with: # Optional: Apply the constraint to the number of lanes with this directionality. -# keep(default it == uni_direction) - -# modifier map.routes_are_in_sequence: -# preceding: route # The first route -# succeeding: route # The second route, which follows after the first route. -# road: road # Optional: The road that will contain this sequence of routes. - -# modifier map.roads_follow_in_junction: -# junction: junction # The junction to be used. -# in_road: road # The chosen road that leads into the junction. -# out_road: road # The chosen road that leads away from the junction. -# direction: junction_direction # Indicates the direction of the out_road relative to the in_road. -# clockwise_count: uint # out_road is clockwise_count roads from in_road, counting clockwise. -# number_of_roads: uint # Total number of in_roads connected to the junction. -# in_lane: lane # The chosen lane within in_road. -# out_lane: lane # The chosen lane within out_road. -# junction_route: route # The element(s) that connect the in_lane or in_road to the out_lane or out_road within the junction. -# resulting_route: route # The route going from in_lane or in_road to the out_lane or out_road. - -# modifier map.routes_overlap: -# route1: route # Mandatory: The first of the overlapping routes. -# route2: route # Mandatory: The second of the overlapping routes. -# overlap_kind: route_overlap_kind # Mandatory: The type of expected overlap. Notice route1 is considered the first route to interpret the values of the enum. - -# modifier map.lane_side: -# lane1: lane # Mandatory: The first lane. -# side: side_left_right # Mandatory: Locate lane1 on this side of lane2. -# lane2: lane # Mandatory: The second lane. -# count: uint # Mandatory: How far is lane1 from lane2? -# lane_section: lane_section # Optional: lane_section where the lanes reside. - -# modifier map.compound_lane_side: -# lane1: compound_lane # Mandatory: The first compound_lane -# side: side_left_right # Mandatory: Locate lane1 on this side of lane2 -# lane2: compound_lane # Mandatory: The second compound_lane -# count: uint # Mandatory: number of from lane1 from lane2 -# route: route # Optional: The route where the compound lanes reside - -# modifier map.end_lane: -# lane: lane # Mandatory: This lane ends in its current lane_section - -# modifier map.start_lane: -# lane: lane # Mandatory: This lane starts in its current lane_section - -# modifier map.crossing_connects: -# crossing: crossing # Mandatory: The crossing that is connected to the specified lanes -# start_lane: lane # Mandatory: The lane where crossing starts (starting from the centerline of the lane) -# end_lane: lane # Mandatory: The destination lane where the crossing ends (ending on the centerline of the lane). -# start_s_coord: length # Mandatory: The crossing origin derived from a s-position along the centerline of start_lane. -# start_angle: angle with: # Optional: The angle at which the straight centerline of the crossing originates from the start lane. Default is perpendicular. -# keep(default it == 90deg) - -# modifier map.routes_are_opposite: -# route1: route # Mandatory: The first uni-directional route. -# route2: route # Mandatory: The second uni-directional route. -# containing_road: road # Mandatory: The road to which both routes belong. -# lateral_overlap: lateral_overlap_kind # Mandatory: Specifies if the routes overlap lateral, meaning they become a single two-way lane. - -# modifier map.set_map_file: -# file: string # Mandatory: The path and file name for the map file. +########### +# ASAM OpenSCENARIO(R) DSL V2.1.0 +# +# (C) 2024 ASAM e.V. +# +# Domain model library +# This file is informative. The normative definitions are published in the specification. +# +# Any use is limited to the scope described in the ASAM license terms. +# See http://www.asam.net/license.html for further details. +# In alteration to the regular license terms, ASAM allows unrestricted distribution of this standard. +# paragraph 2 (1) of ASAM's regular license terms is therefore substituted by the following clause: +# "The licensor grants everyone a basic, non-exclusive and unlimited license to use the standard ASAM OpenSCENARIO DSL". +########### + +########### +# ATTENTION: This file is modified to be compatible with +# the feature set of scenario execution. +# - domain include disabled +########### + +import osc.types +#import "domain.osc" diff --git a/scenario_execution/scenario_execution/lib_osc/standard_base.osc b/scenario_execution/scenario_execution/lib_osc/types.osc similarity index 69% rename from scenario_execution/scenario_execution/lib_osc/standard_base.osc rename to scenario_execution/scenario_execution/lib_osc/types.osc index 57c8af11..3ececbc5 100644 --- a/scenario_execution/scenario_execution/lib_osc/standard_base.osc +++ b/scenario_execution/scenario_execution/lib_osc/types.osc @@ -1,17 +1,35 @@ ########### -# Standard: OpenSCENARIO 2.0 domain model -# Copyright: ASAM 2021-2022 -# Version: -# Standard Publication Date: -# Publication Reference Document: -# Reference Document Date: +# ASAM OpenSCENARIO(R) DSL V2.1.0 +# +# (C) 2024 ASAM e.V. +# +# Domain model library +# This file is informative. The normative definitions are published in the specification. +# +# Any use is limited to the scope described in the ASAM license terms. +# See http://www.asam.net/license.html for further details. +# In alteration to the regular license terms, ASAM allows unrestricted distribution of this standard. +# paragraph 2 (1) of ASAM's regular license terms is therefore substituted by the following clause: +# "The licensor grants everyone a basic, non-exclusive and unlimited license to use the standard ASAM OpenSCENARIO DSL". ########### +########### +# ATTENTION: This file is modified to be compatible with +# the feature set of scenario execution. +# - default values added +# - additional types added +# - namespace disabled +########### + +# TODO: enable once namespaces are supported +# namespace stdtypes +# +# export * + ######################## # Scalar types and units ######################## -# tag::library-physical-length[] type length is SI(m: 1) unit nanometer of length is SI(m: 1, factor: 0.000000001) unit nm of length is SI(m: 1, factor: 0.000000001) @@ -28,9 +46,7 @@ unit inch of length is SI(m: 1, factor: 0.0254) unit feet of length is SI(m: 1, factor: 0.3048) unit mile of length is SI(m: 1, factor: 1609.344) unit mi of length is SI(m: 1, factor: 1609.344) -# end::library-physical-length[] -# tag::library-physical-time[] type time is SI(s: 1) unit millisecond of time is SI(s: 1, factor: 0.001) unit ms of time is SI(s: 1, factor: 0.001) @@ -41,9 +57,7 @@ unit minute of time is SI(s: 1, factor: 60) unit min of time is SI(s: 1, factor: 60) unit hour of time is SI(s: 1, factor: 3600) unit h of time is SI(s: 1, factor: 3600) -# end::library-physical-time[] -# tag::library-physical-speed[] type speed is SI(m: 1, s: -1) unit meter_per_second of speed is SI(m: 1, s: -1, factor: 1) unit mps of speed is SI(m: 1, s: -1, factor: 1) @@ -55,9 +69,7 @@ unit mph of speed is SI(m: 1, s: -1, factor: 0.447038889) unit miph of speed is SI(m: 1, s: -1, factor: 0.447038889) unit mmph of speed is SI(m: 1, s: -1, factor: 0.000000278) unit millimeter_per_hour of speed is SI(m: 1, s: -1, factor: 0.000000278) -# end::library-physical-speed[] -# tag::library-physical-acceleration[] type acceleration is SI(m: 1, s: -2) unit meter_per_sec_sqr of acceleration is SI(m: 1, s: -2, factor: 1) unit mpsps of acceleration is SI(m: 1, s: -2, factor: 1) @@ -66,41 +78,31 @@ unit kilometer_per_hour_per_sec of acceleration is SI(m: 1, s: -2, factor: 0.277 unit kmphps of acceleration is SI(m: 1, s: -2, factor: 0.277777778) unit mile_per_hour_per_sec of acceleration is SI(m: 1, s: -2, factor: 0.447038889) unit miphps of acceleration is SI(m: 1, s: -2, factor: 0.447038889) -# end::library-physical-acceleration[] -# tag::library-physical-jerk[] type jerk is SI(m: 1, s: -3) unit meter_per_sec_cubed of jerk is SI(m: 1, s: -3, factor: 1) unit mpspsps of jerk is SI(m: 1, s: -3, factor: 1) unit mile_per_sec_cubed of jerk is SI(m: 1, s: -3, factor: 1609.344) unit mipspsps of jerk is SI(m: 1, s: -3, factor: 1609.344) -# end::library-physical-jerk[] -# tag::library-physical-angle[] type angle is SI(rad: 1) -unit degree of angle is SI(rad: 1, factor: 57.295779513) -unit deg of angle is SI(rad: 1, factor: 57.295779513) +unit degree of angle is SI(rad: 1, factor: 0.01745329252) +unit deg of angle is SI(rad: 1, factor: 0.01745329252) unit radian of angle is SI(rad: 1, factor: 1) unit rad of angle is SI(rad: 1, factor: 1) -# end::library-physical-angle[] -# tag::library-physical-angular_rate[] type angular_rate is SI(rad: 1, s: -1) -unit degree_per_sec of angular_rate is SI(rad: 1, s: -1, factor: 57.295779513) -unit degps of angular_rate is SI(rad: 1, s: -1, factor: 57.295779513) +unit degree_per_sec of angular_rate is SI(rad: 1, s: -1, factor: 0.01745329252) +unit degps of angular_rate is SI(rad: 1, s: -1, factor: 0.01745329252) unit radian_per_sec of angular_rate is SI(rad: 1, s: -1, factor: 1) unit radps of angular_rate is SI(rad: 1, s: -1, factor: 1) -# end::library-physical-angular_rate[] -# tag::library-physical-angular_acceleration[] type angular_acceleration is SI(rad: 1, s: -2) -unit degree_per_sec_sqr of angular_acceleration is SI(rad: 1, s: -2, factor: 57.295779513) -unit degpsps of angular_acceleration is SI(rad: 1, s: -2, factor: 57.295779513) +unit degree_per_sec_sqr of angular_acceleration is SI(rad: 1, s: -2, factor: 0.01745329252) +unit degpsps of angular_acceleration is SI(rad: 1, s: -2, factor: 0.01745329252) unit radian_per_sec_sqr of angular_acceleration is SI(rad: 1, s: -2, factor: 1) unit radpsps of angular_acceleration is SI(rad: 1, s: -2, factor: 1) -# end::library-physical-angular_acceleration[] -# tag::library-physical-mass[] type mass is SI(kg: 1) unit gram of mass is SI(kg: 1, factor: 0.001) unit kilogram of mass is SI(kg: 1, factor: 1) @@ -108,9 +110,7 @@ unit kg of mass is SI(kg: 1, factor: 1) unit ton of mass is SI(kg: 1, factor: 1000) unit pound of mass is SI(kg: 1, factor: 0.45359237) unit lb of mass is SI(kg: 1, factor: 0.45359237) -# end::library-physical-mass[] -# tag::library-physical-temperature[] type temperature is SI(K: 1) unit K of temperature is SI(K: 1, factor: 1) unit kelvin of temperature is SI(K: 1, factor: 1) @@ -118,131 +118,92 @@ unit celsius of temperature is SI(K: 1, factor: 1, offset: 273.15) unit C of temperature is SI(K: 1, factor: 1, offset: 273.15) unit fahrenheit of temperature is SI(K: 1, factor: 0.555555556, offset: 255.372222222) unit F of temperature is SI(K: 1, factor: 0.555555556, offset: 255.372222222) -# end::library-physical-temperature[] -# tag::library-physical-pressure[] type pressure is SI(kg: 1, m: -1, s: -2) unit newton_per_meter_sqr of pressure is SI(kg: 1, m: -1, s: -2, factor: 1) unit Pa of pressure is SI(kg: 1, m: -1, s: -2, factor: 1) unit pascal of pressure is SI(kg: 1, m: -1, s: -2, factor: 1) unit hPa of pressure is SI(kg: 1, m: -1, s: -2, factor: 100) unit atm of pressure is SI(kg: 1, m: -1, s: -2, factor: 101325) -# end::library-physical-pressure[] -# tag::library-physical-luminous_intensity[] type luminous_intensity is SI(cd: 1) unit cd of luminous_intensity is SI(cd: 1, factor: 1) unit candela of luminous_intensity is SI(cd: 1, factor: 1) -# end::library-physical-luminous_intensity[] -# tag::library-physical-luminous_flux[] type luminous_flux is SI(cd: 1, rad: 2) unit lm of luminous_flux is SI(cd: 1, rad: 2, factor: 1) unit lumen of luminous_flux is SI(cd: 1, rad: 2, factor: 1) -# end::library-physical-luminous_flux[] -# tag::library-physical-illuminance[] type illuminance is SI(cd: 1, rad: 2, m: -2) unit lx of illuminance is SI(cd: 1, rad: 2, m: -2, factor: 1) unit lux of illuminance is SI(cd: 1, rad: 2, m: -2, factor: 1) -# end::library-physical-illuminance[] -# tag::library-physical-electrical_current[] type electrical_current is SI(A: 1) unit ampere of electrical_current is SI(A: 1, factor: 1) unit A of electrical_current is SI(A: 1, factor: 1) -# end::library-physical-electrical_current[] -# tag::library-physical-amount_of_substance[] type amount_of_substance is SI(mol: 1) unit mole of amount_of_substance is SI(mol: 1, factor: 1) unit mol of amount_of_substance is SI(mol: 1, factor: 1) -# end::library-physical-amount_of_substance[] ########### # Structs ########### -# tag::library-position_3d[] -struct position_3d: +struct position + +struct position_3d inherits position: x: length = 0.0m y: length = 0.0m z: length = 0.0m - #def norm() -> length is undefined -# end::library-position_3d[] + # def norm() -> length is undefined -# tag::library-celestial_position_2d[] -struct celestial_position_2d: +struct celestial_position_2d inherits position: azimuth: angle = 0.0rad elevation: angle = 0.0rad -# end::library-celestial_position_2d[] -# tag::library-geodetic_position_2d[] -struct geodetic_position_2d: +struct geodetic_position_2d inherits position: latitude: angle = 0.0rad longitude: angle = 0.0rad -# end::library-geodetic_position_2d[] -# tag::library-orientation_3d[] struct orientation_3d: roll: angle = 0.0rad pitch: angle = 0.0rad yaw: angle = 0.0rad -# end::library-orientation_3d[] -# tag::library-pose_3d[] struct pose_3d: position: position_3d orientation: orientation_3d -# end::library-pose_3d[] -# tag::library-translational_velocity_3d[] struct translational_velocity_3d: x: speed = 0.0mps y: speed = 0.0mps z: speed = 0.0mps -# def norm() -> speed is undefined -# end::library-translational_velocity_3d[] + # def norm() -> speed is undefined -# tag::library-orientation_rate_3d[] struct orientation_rate_3d: roll: angular_rate = 0.0radps pitch: angular_rate = 0.0radps yaw: angular_rate = 0.0radps -# end::library-orientation_rate_3d[] -# tag::library-velocity_6d[] struct velocity_6d: translational: translational_velocity_3d angular: orientation_rate_3d -# end::library-velocity_6d[] -# tag::library-translational_acceleration_3d[] struct translational_acceleration_3d: x: acceleration = 0.0mpsps y: acceleration = 0.0mpsps z: acceleration = 0.0mpsps -# def norm() -> acceleration is undefined -# end::library-translational_acceleration_3d[] + # def norm() -> acceleration is undefined -# tag::library-orientation_acceleration_3d[] struct orientation_acceleration_3d: roll: angular_acceleration = 0.0radpsps pitch: angular_acceleration = 0.0radpsps yaw: angular_acceleration = 0.0radpsps -# end::library-orientation_acceleration_3d[] -# tag::library-acceleration_6d[] struct acceleration_6d: translational: translational_acceleration_3d angular: orientation_acceleration_3d -# end::library-acceleration_6d[] - -struct bounding_box: - center: position_3d # Mandatory: Represents the geometrical center of the bounding box expressed in coordinates that refer to the coordinate system of the physical_object - length: length # Mandatory: Dimension in x-direction of the coordinate system of the physical_object - width: length # Mandatory: Dimension in y-direction of the coordinate system of the physical_object - height: length # Mandatory: Dimension in z-direction of the coordinate system of the physical_object # ########### # # Actor diff --git a/scenario_execution/setup.py b/scenario_execution/setup.py index 3a52f044..b4e35020 100644 --- a/scenario_execution/setup.py +++ b/scenario_execution/setup.py @@ -78,7 +78,7 @@ 'scenario_execution.osc_libraries': [ 'helpers = scenario_execution.get_osc_library:get_helpers_library', 'standard = scenario_execution.get_osc_library:get_standard_library', - 'standard.base = scenario_execution.get_osc_library:get_standard_base_library', + 'types = scenario_execution.get_osc_library:get_types_library', 'robotics = scenario_execution.get_osc_library:get_robotics_library', ] }, diff --git a/scenario_execution/test/test_osc2_parser_not_supported.py b/scenario_execution/test/test_osc2_parser_not_supported.py index ce5f7197..5210603e 100644 --- a/scenario_execution/test/test_osc2_parser_not_supported.py +++ b/scenario_execution/test/test_osc2_parser_not_supported.py @@ -116,7 +116,7 @@ def test_cast(self): def test_sample(self): scenario_content = """ -import osc.standard.base +import osc.types scenario simple_drive: environment: environment diff --git a/scenario_execution/test/test_osc2_standard_osc.py b/scenario_execution/test/test_osc2_standard_osc.py index 28197f0c..f200330c 100644 --- a/scenario_execution/test/test_osc2_standard_osc.py +++ b/scenario_execution/test/test_osc2_standard_osc.py @@ -43,6 +43,6 @@ def test_standard_osc(self): def test_standard_common_osc(self): scenario_content = """ -import osc.standard.base +import osc.types """ model = self.parse(scenario_content) diff --git a/scenario_execution/test/test_run_process.py b/scenario_execution/test/test_run_process.py index 5a049fbd..e4716e79 100644 --- a/scenario_execution/test/test_run_process.py +++ b/scenario_execution/test/test_run_process.py @@ -45,7 +45,7 @@ def execute(self, scenario_content): def test_failure(self): scenario_content = """ -import osc.standard.base +import osc.types import osc.helpers scenario test_run_process: @@ -59,7 +59,7 @@ def test_failure(self): def test_success(self): scenario_content = """ -import osc.standard.base +import osc.types import osc.helpers scenario test_run_process: @@ -73,7 +73,7 @@ def test_success(self): def test_multi_element_command(self): scenario_content = """ -import osc.standard.base +import osc.types import osc.helpers scenario test_run_process: @@ -86,7 +86,7 @@ def test_multi_element_command(self): def test_wait_for_shutdown_false(self): scenario_content = """ -import osc.standard.base +import osc.types import osc.helpers scenario test_run_process: @@ -108,7 +108,7 @@ def test_wait_for_shutdown_false(self): def test_signal_parsing(self): scenario_content = """ -import osc.standard.base +import osc.types import osc.helpers scenario test_run_process: From 32a9eeab1948f0293bf20a789ce712bb7ec65274 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=87=BA=E9=97=A8=E4=B8=89=E4=B8=8D=E6=83=B9?= Date: Thu, 14 Nov 2024 14:54:12 +0800 Subject: [PATCH 5/7] chore: add register_signal option to ScenarioExecution for thread (#212) Co-authored-by: pipinet --- .../scenario_execution/scenario_execution_base.py | 8 +++++--- 1 file changed, 5 insertions(+), 3 deletions(-) diff --git a/scenario_execution/scenario_execution/scenario_execution_base.py b/scenario_execution/scenario_execution/scenario_execution_base.py index 1a1e9a30..9f71c22e 100644 --- a/scenario_execution/scenario_execution/scenario_execution_base.py +++ b/scenario_execution/scenario_execution/scenario_execution_base.py @@ -92,13 +92,15 @@ def __init__(self, setup_timeout=py_trees.common.Duration.INFINITE, tick_period: float = 0.1, scenario_parameter_file=None, - logger=None) -> None: + logger=None, + register_signal=True) -> None: def signal_handler(sig, frame): self.on_scenario_shutdown(False, "Aborted") - signal.signal(signal.SIGHUP, signal_handler) - signal.signal(signal.SIGTERM, signal_handler) + if register_signal: + signal.signal(signal.SIGHUP, signal_handler) + signal.signal(signal.SIGTERM, signal_handler) self.current_scenario_start = None self.current_scenario = None From 0089459845226385af5e4bd994ce8408ad343d3b Mon Sep 17 00:00:00 2001 From: Nikhil Date: Wed, 20 Nov 2024 07:55:42 +0100 Subject: [PATCH 6/7] fix/kubernetes-pod-status (#216) * fix-kubernetes-pod-status * format * fix feedback message and repeat * cleanup --- .../kubernetes_wait_for_pod_status.py | 25 ++++++++++++------- 1 file changed, 16 insertions(+), 9 deletions(-) diff --git a/libs/scenario_execution_kubernetes/scenario_execution_kubernetes/kubernetes_wait_for_pod_status.py b/libs/scenario_execution_kubernetes/scenario_execution_kubernetes/kubernetes_wait_for_pod_status.py index 9a51e51c..cb00501c 100644 --- a/libs/scenario_execution_kubernetes/scenario_execution_kubernetes/kubernetes_wait_for_pod_status.py +++ b/libs/scenario_execution_kubernetes/scenario_execution_kubernetes/kubernetes_wait_for_pod_status.py @@ -32,16 +32,17 @@ class KubernetesWaitForPodStatusState(Enum): class KubernetesWaitForPodStatus(BaseAction): - def __init__(self, within_cluster: bool): + def __init__(self, within_cluster: bool, namespace: str): super().__init__() self.target = None - self.namespace = None + self.namespace = namespace self.expected_status = None self.within_cluster = within_cluster self.regex = None self.client = None self.update_queue = queue.Queue() self.current_state = KubernetesWaitForPodStatusState.IDLE + self.last_state = None def setup(self, **kwargs): if self.within_cluster: @@ -53,44 +54,50 @@ def setup(self, **kwargs): self.monitoring_thread = threading.Thread(target=self.watch_pods, daemon=True) self.monitoring_thread.start() - def execute(self, target: str, regex: bool, status: tuple, namespace: str): + def execute(self, target: str, regex: bool, status: tuple, ): self.target = target - self.namespace = namespace if not isinstance(status, tuple) or not isinstance(status[0], str): raise ValueError("Status expected to be enum.") self.expected_status = status[0] self.regex = regex self.current_state = KubernetesWaitForPodStatusState.MONITORING + self.last_state = None def update(self) -> py_trees.common.Status: while not self.update_queue.empty(): item = self.update_queue.get() if len(item) != 2: return py_trees.common.Status.FAILURE - - self.feedback_message = f"waiting for status of pod '{self.target}'." # pylint: disable= attribute-defined-outside-init + if self.last_state is None: + self.feedback_message = f"waiting for status of pod '{self.target}'." # pylint: disable= attribute-defined-outside-init if not self.regex: if item[0] != self.target: continue else: if not re.search(self.target, item[0]): continue - if item[1].lower() == self.expected_status: + if item[1].lower() == self.expected_status and self.last_state is not None: self.feedback_message = f"Pod '{item[0]}' changed to expected status '{item[1].lower()}'." # pylint: disable= attribute-defined-outside-init + self.current_state = KubernetesWaitForPodStatusState.IDLE return py_trees.common.Status.SUCCESS else: self.feedback_message = f"Pod '{item[0]}' changed to status '{item[1].lower()}', expected '{self.expected_status}'." # pylint: disable= attribute-defined-outside-init + self.last_state = item[1].lower() return py_trees.common.Status.RUNNING def watch_pods(self): w = watch.Watch() try: - # TODO: make use of send_initial_events=false in the future + initial_pods = self.client.list_namespaced_pod(namespace=self.namespace).items + for pod in initial_pods: + pod_name = pod.metadata.name + pod_status = pod.status.phase + self.update_queue.put((pod_name, pod_status)) for event in w.stream(self.client.list_namespaced_pod, namespace=self.namespace): pod_name = event['object'].metadata.name pod_status = event['object'].status.phase if self.current_state == KubernetesWaitForPodStatusState.MONITORING: self.update_queue.put((pod_name, pod_status)) except ApiException as e: - self.logger.error(f"Error accessing kubernetes: {e}") + self.logger.error(f"Error accessing Kubernetes: {e}") self.update_queue.put(()) From 99d70058df5903fd71e09c2efd400afefc00b0fe Mon Sep 17 00:00:00 2001 From: Nikhil Date: Wed, 20 Nov 2024 09:01:38 +0100 Subject: [PATCH 7/7] feature/random-functions (#217) * add random int and random list * format * add unittest * generic method * format --- .../external_methods/random.py | 10 +++ .../scenario_execution/lib_osc/helpers.osc | 4 +- .../test/test_external_methods_random.py | 85 +++++++++++++++++++ 3 files changed, 97 insertions(+), 2 deletions(-) create mode 100644 scenario_execution/test/test_external_methods_random.py diff --git a/scenario_execution/scenario_execution/external_methods/random.py b/scenario_execution/scenario_execution/external_methods/random.py index 93e633c9..bd36ac94 100644 --- a/scenario_execution/scenario_execution/external_methods/random.py +++ b/scenario_execution/scenario_execution/external_methods/random.py @@ -22,3 +22,13 @@ def seed(seed_value: int = 0): def get_float(min_val: dict, max_val: float): return rd.uniform(min_val, max_val) # nosec B311 + + +def get_int(min_val: int, max_val: int): + return rd.randint(min_val, max_val) # nosec B311 + + +def get_random_list_element(elements_list: list): + if not elements_list: + return None # Return None if the list is empty + return rd.choice(elements_list) # nosec B311 diff --git a/scenario_execution/scenario_execution/lib_osc/helpers.osc b/scenario_execution/scenario_execution/lib_osc/helpers.osc index 7f90f873..217efef8 100644 --- a/scenario_execution/scenario_execution/lib_osc/helpers.osc +++ b/scenario_execution/scenario_execution/lib_osc/helpers.osc @@ -31,8 +31,8 @@ action run_process: struct random: def seed(seed_value: int = 0) is external scenario_execution.external_methods.random.seed() def get_float(min_val: float, max_val: float) -> float is external scenario_execution.external_methods.random.get_float() - - + def get_int(min_val: int, max_val: int) -> int is external scenario_execution.external_methods.random.get_int() + def get_random_string(elements_list: list of string) -> string is external scenario_execution.external_methods.random.get_random_list_element() ######################### # Common modifiers diff --git a/scenario_execution/test/test_external_methods_random.py b/scenario_execution/test/test_external_methods_random.py new file mode 100644 index 00000000..210aadfc --- /dev/null +++ b/scenario_execution/test/test_external_methods_random.py @@ -0,0 +1,85 @@ +# Copyright (C) 2024 Intel Corporation +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, +# software distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions +# and limitations under the License. +# +# SPDX-License-Identifier: Apache-2.0 + +import unittest +import py_trees +from datetime import datetime +from scenario_execution.scenario_execution_base import ScenarioExecution +from scenario_execution.model.osc2_parser import OpenScenario2Parser +from scenario_execution.model.model_to_py_tree import create_py_tree +from .common import DebugLogger +from antlr4.InputStream import InputStream + + +class TestExternalMethodsRandom(unittest.TestCase): + # pylint: disable=missing-function-docstring + + def setUp(self) -> None: + self.logger = DebugLogger("") + self.parser = OpenScenario2Parser(self.logger) + self.tree = py_trees.composites.Sequence(name="", memory=True) + self.scenario_execution = ScenarioExecution(debug=False, + log_model=False, + live_tree=False, + scenario_file='test', + output_dir='', logger=self.logger) + self.tree = py_trees.composites.Sequence(name="", memory=True) + + def execute(self, scenario_content): + parsed_tree = self.parser.parse_input_stream(InputStream(scenario_content)) + model = self.parser.create_internal_model(parsed_tree, self.tree, "test.osc", False) + self.tree = create_py_tree(model, self.tree, self.parser.logger, False) + self.scenario_execution.tree = self.tree + self.scenario_execution.run() + + def test_get_random_int(self): + scenario_content = """ +import osc.helpers + +scenario test_success: + do serial: + wait elapsed(random.get_int(0, 5)) + emit end +""" + parsed_tree = self.parser.parse_input_stream(InputStream(scenario_content)) + model = self.parser.create_internal_model(parsed_tree, self.tree, "test.osc", False) + self.tree = create_py_tree(model, self.tree, self.parser.logger, False) + self.scenario_execution.tree = self.tree + + start_time = datetime.now() + self.scenario_execution.run() + end_time = datetime.now() + self.assertTrue(self.scenario_execution.process_results()) + + delta = end_time - start_time + self.assertLess(delta.total_seconds(), 5.) + + def test_get_random_string(self): + scenario_content = """ +import osc.helpers + +scenario test_success: + do serial: + log(random.get_random_string(["test", "test-scenario", "scenario-test"])) + emit end +""" + self.execute(scenario_content) + self.assertTrue(self.scenario_execution.process_results()) + valid_strings = ["test", "test-scenario", "scenario-test"] + log_messages = self.logger.logs_info + for log_message in log_messages: + if log_message in valid_strings: + self.assertIn(log_message, valid_strings)