From e1aa722a9efcb59977825d0a9f3b580cfb8f20f2 Mon Sep 17 00:00:00 2001 From: samuelmyoussef Date: Fri, 8 Dec 2023 10:15:43 +0100 Subject: [PATCH 01/85] change abstract class to no server --- sofagym/AbstractEnv.py | 242 +++++++++++++++++++++++++++-------------- 1 file changed, 160 insertions(+), 82 deletions(-) diff --git a/sofagym/AbstractEnv.py b/sofagym/AbstractEnv.py index 20b42c8..9c6160b 100644 --- a/sofagym/AbstractEnv.py +++ b/sofagym/AbstractEnv.py @@ -20,7 +20,11 @@ import splib3 from sofagym.viewer import Viewer -from sofagym.rpc_server import start_server, add_new_step, get_result, clean_registry, close_scene + +import importlib + +import Sofa +import SofaRuntime class AbstractEnv(gym.Env): @@ -39,7 +43,6 @@ class AbstractEnv(gym.Env): while an action is still ongoing. close: Terminate the simulation. configure: Add element in the configuration. - clean: clean the registery. _formataction.. : transforme the type of action to use server. Arguments: @@ -112,7 +115,7 @@ class AbstractEnv(gym.Env): """ - def __init__(self, config=None, render_mode: Optional[str] = None): + def __init__(self, config=None, render_mode: Optional[str]=None, root=None): """ Classic initialization of a class in python. @@ -132,43 +135,56 @@ def __init__(self, config=None, render_mode: Optional[str] = None): if config is not None: self.config.update(config) - self.initialization() - - self.render_mode = render_mode - - def initialization(self): - """Initialization of all parameters. + self.scene = self.config['scene'] - Parameters: - ---------- - None. + self._getState = importlib.import_module("sofagym.envs."+self.scene+"."+self.scene+"Toolbox").getState + self._getReward = importlib.import_module("sofagym.envs."+self.scene+"."+self.scene+"Toolbox").getReward + self._startCmd = importlib.import_module("sofagym.envs."+self.scene+"."+self.scene+"Toolbox").startCmd + self._getPos = importlib.import_module("sofagym.envs."+self.scene+"."+self.scene+"Toolbox").getPos + + try: + self.create_scene = importlib.import_module("sofagym.envs."+self.scene+"." + self.scene + "Scene").createScene + except Exception as exc: + print("sofagym.envs."+self.scene+"." + self.scene + "Scene") + raise NotImplementedError("Importing your SOFA Scene Failed") from exc + + if isinstance(root, type(None)): + self.root = self.init_simulation() + else: + self.root = root - Returns: - ------- - None. - """ + self.viewer = None + self.render_mode = render_mode self.goalList = None self.goal = None self.past_actions = [] - + self.pos = [] + self.past_pos = [] + self.num_envs = 40 self.np_random = None self.seed(self.config['seed']) - self.viewer = None - self.automatic_rendering_callback = None - - self.timer = 0 self.timeout = self.config["timeout"] - # Start the server which distributes the calculations to its clients - start_server(self.config) + self.init_save_paths() + + def init_save_paths(self): + """Create directories to save results and images. + + Parameters: + ---------- + None. + Returns: + ------- + None. + """ if 'save_data' in self.config and self.config['save_data']: save_path_results = self.config['save_path']+"/data" os.makedirs(save_path_results, exist_ok=True) @@ -258,23 +274,6 @@ def _formatactionDict(self, action): return action - def clean(self): - """Function to clean the registery . - - Close clients who are processing unused sequences of actions (for - planning) - - Parameters: - ---------- - None. - - Returns: - ------- - None. - """ - - clean_registry(self.past_actions) - def step(self, action): """Executes one action in the environment. @@ -303,27 +302,22 @@ def step(self, action): action = self._formataction(action) - # Pass the actions to the server to launch the simulation. - result_id = add_new_step(self.past_actions, action) - self.past_actions.append(action) + self.pos = self.step_simulation(action) - # Request results from the server. - # print("[INFO] >>> Result id:", result_id) - results = get_result(result_id, timeout=self.timeout) + self.past_actions.append(action) + self.past_pos.append(self.pos) - obs = np.array(results["observation"]) # to work with baseline - reward = results["reward"] - done = results["done"] + obs = np.array(self._getState(self.root), dtype=np.float32) + done, reward = self._getReward(self.root) # Avoid long explorations by using a timer. self.timer += 1 if self.timer >= self.config["timer_limit"]: # reward = -150 truncated = True - info={}#(not use here) + + info = {} #(not use here) - if self.config["planning"]: - self.clean() return obs, reward, done, info def async_step(self, action): @@ -376,7 +370,6 @@ def reset(self): obs, info """ - self.clean() self.viewer = None splib3.animation.animate.manager = None @@ -390,8 +383,13 @@ def reset(self): self.timer = 0 self.past_actions = [] + self.past_pos = [] + + self.root = self.init_simulation() + + obs = np.array(self._getState(self.root), dtype=np.float32) - return + return obs def render(self, mode): """See the current state of the environment. @@ -411,40 +409,16 @@ def render(self, mode): self.render_mode = mode # Define the viewer at the first run of render. - if not self.viewer: + if self.viewer is None: display_size = self.config["display_size"] # Sim display if 'zFar' in self.config: zFar = self.config['zFar'] else: zFar = 0 - self.viewer = Viewer(self, display_size, zFar=zFar, save_path=self.config["save_path_image"]) + self.viewer = Viewer(self, self.root, display_size, zFar=zFar, save_path=self.config["save_path_image"]) # Use the viewer to display the environment. return self.viewer.render() - def _automatic_rendering(self): - """Automatically render the intermediate frames while an action is still ongoing. - - This allows to render the whole video and not only single steps - corresponding to agent decision-making. - If a callback has been set, use it to perform the rendering. This is - useful for the environment wrappers such as video-recording monitor that - need to access these intermediate renderings. - - Parameters: - ---------- - None. - - Returns: - ------- - None. - - """ - if self.viewer is not None: - if self.automatic_rendering_callback: - self.automatic_rendering_callback() - else: - self.render() - def close(self): """Terminate simulation. @@ -460,8 +434,7 @@ def close(self): """ if self.viewer is not None: self.viewer.close() - - close_scene() + print("All clients are closed. Bye Bye.") def configure(self, config): @@ -478,3 +451,108 @@ def configure(self, config): """ self.config.update(config) + + def init_simulation(self, mode='simu_and_visu'): + """Function to create scene and initialize all variables. + + Parameters: + ---------- + config: Dictionary. + Configuration of the environment. + _startCmd: function + Initialize the command. + mode: string, default = 'simu_and_visu' + Init a scene with or without visu and computations. + In ['simu', 'visu', 'simu_and_visu'] + + Returns: + ------- + root: + The loaded and initialized scene. + + """ + # Load the scene + root = Sofa.Core.Node("root") + + SofaRuntime.importPlugin("Sofa.Component") + self.create_scene(root, self.config, mode = mode) + Sofa.Simulation.init(root) + + # Realise action from history + if self.config['start_from_history'] is not None and self._startCmd is not None: + print(">> Start from history ...") + render = self.config['render'] + self.config.update({'render': 0}) + + for action in self.config['start_from_history']: + self.step_simulation(action) + + self.config.update({'render': render}) + print(">> ... Done.") + + # Init Reward and GoalSetter + root.GoalSetter.update(self.config["goalList"]) + root.Reward.update(self.config["goalList"]) + + try: + root.StateInitializer.init_state(self.config["init_states"]) + except AttributeError as error: + print(error) + + if 'time_before_start' in self.config: + print(">> Time before start:", self.config["time_before_start"], "steps. Initialization ...") + for _ in range(self.config["time_before_start"]): + Sofa.Simulation.animate(root, self.config["dt"]) + print(">> ... Done.") + # Update Reward and GoalSetter + root.GoalSetter.update(self.config["goalList"]) + root.Reward.update(self.config["goalList"]) + + return root + + def step_simulation(self, action): + """Realise one step in the simulation. + + Apply action and execute 5 times scale_factor simulation steps of dt s. + + Parameters: + ---------- + root: + The scene. + config: Dictionary + The configuration of the environment. + action: int + The action to apply in the environment. + _startCmd: function + Initialize the command. + _getPos: function + Get the position of the object in the scene. + + Returns: + -------- + position(s): list + The positions of object(s) in the scene. + + """ + goal = self.config['goalPos'] + render = self.config['render'] + surface_size = self.config['display_size'] + + self.root.GoalSetter.set_mo_pos(goal) + + # Create the command from action + self._startCmd(self.root, action, self.config["dt"]*(self.config["scale_factor"]-1)) + pos = [] + # Realise scale_factor simulation steps of 0.01 s + for _ in range(self.config["scale_factor"]): + Sofa.Simulation.animate(self.root, self.config["dt"]) + + #if render == 2: + # pos.append(self._getPos(self.root)) + # if self.viewer is not None: + # self.viewer.render_simulation(self.root) + + if render == 1: + pos.append(self._getPos(self.root)) + + return pos From 19280da34d3ccf210ec41078472f0c27210f9f31 Mon Sep 17 00:00:00 2001 From: samuelmyoussef Date: Fri, 8 Dec 2023 10:33:22 +0100 Subject: [PATCH 02/85] add server class --- sofagym/AbstractEnv.py | 124 ++++++++++++++++++++++++++++++++++++++++- 1 file changed, 122 insertions(+), 2 deletions(-) diff --git a/sofagym/AbstractEnv.py b/sofagym/AbstractEnv.py index 9c6160b..6f5b192 100644 --- a/sofagym/AbstractEnv.py +++ b/sofagym/AbstractEnv.py @@ -19,6 +19,7 @@ import splib3 +from sofagym.rpc_server import start_server, add_new_step, get_result, clean_registry, close_scene from sofagym.viewer import Viewer import importlib @@ -115,7 +116,7 @@ class AbstractEnv(gym.Env): """ - def __init__(self, config=None, render_mode: Optional[str]=None, root=None): + def __init__(self, default_config, config=None, render_mode: Optional[str]=None, root=None): """ Classic initialization of a class in python. @@ -130,7 +131,7 @@ def __init__(self, config=None, render_mode: Optional[str]=None, root=None): """ # Define a DEFAULT_CONFIG in sub-class. - self.config = copy.deepcopy(self.DEFAULT_CONFIG) + self.config = copy.deepcopy(default_config) self.config["dt"] = self.config.get('dt', 0.01) if config is not None: self.config.update(config) @@ -556,3 +557,122 @@ def step_simulation(self, action): pos.append(self._getPos(self.root)) return pos + + + + +class ServerEnv(AbstractEnv): + def __init__(self, default_config, config=None, render_mode: Optional[str]=None, root=None): + super().__init__(default_config, config, root=root) + + # Start the server which distributes the calculations to its clients + start_server(self.config) + + def step(self, action): + """Executes one action in the environment. + + Apply action and execute scale_factor simulation steps of 0.01 s. + + Parameters: + ---------- + action: int + Action applied in the environment. + + Returns: + ------- + obs(ObsType): + The new state of the agent. + reward(float): + The reward obtain after applying the action in the current state. + done(bool): + Whether the agent reaches the terminal state + info(dict): + additional information (not used here) + """ + + # assert self.action_space.contains(action), "%r (%s) invalid" % (action, type(action)) + + action = self._formataction(action) + + # Pass the actions to the server to launch the simulation. + result_id = add_new_step(self.past_actions, action) + self.past_actions.append(action) + + # Request results from the server. + # print("[INFO] >>> Result id:", result_id) + results = get_result(result_id, timeout=self.timeout) + + obs = np.array(results["observation"]) # to work with baseline + reward = results["reward"] + done = results["done"] + + # Avoid long explorations by using a timer. + self.timer += 1 + if self.timer >= self.config["timer_limit"]: + # reward = -150 + truncated = True + info={}#(not use here) + + if self.config["planning"]: + self.clean() + return obs, reward, done, info + + def reset(self): + """Reset simulation. + + Parameters: + ---------- + None. + + Returns: + ------- + obs, info + """ + self.clean() + self.viewer = None + + splib3.animation.animate.manager = None + if not self.goalList: + self.goalList = self.config["goalList"] + + # Set a new random goal from the list + id_goal = self.np_random.choice(range(len(self.goalList))) + self.config.update({'goal_node': id_goal}) + self.goal = self.goalList[id_goal] + + self.timer = 0 + self.past_actions = [] + + return + + def clean(self): + """Function to clean the registery . + + Close clients who are processing unused sequences of actions (for + planning) + + Parameters: + ---------- + None. + + Returns: + ------- + None. + """ + clean_registry(self.past_actions) + + def close(self): + """Terminate simulation. + + Close the viewer and the scene. + + Parametres: + ---------- + None. + + Returns: + ------- + None. + """ + super().close() + close_scene() From c9ffbf3c12ef7a16fcbb9ce05193c535a36e6019 Mon Sep 17 00:00:00 2001 From: samuelmyoussef Date: Fri, 8 Dec 2023 11:06:00 +0100 Subject: [PATCH 03/85] modify viewer --- sofagym/viewer.py | 20 +++++++++++++++----- 1 file changed, 15 insertions(+), 5 deletions(-) diff --git a/sofagym/viewer.py b/sofagym/viewer.py index 7241f68..19c5866 100644 --- a/sofagym/viewer.py +++ b/sofagym/viewer.py @@ -62,7 +62,7 @@ class Viewer: """ - def __init__(self, env, surface_size, zFar=100, save_path=None, create_video=None, fps=10): + def __init__(self, env, root, surface_size, zFar=100, save_path=None, create_video=None, fps=10): """ Classic initialization of a class in python. @@ -97,7 +97,11 @@ def __init__(self, env, surface_size, zFar=100, save_path=None, create_video=Non self.agent_display = None self.frame = 0 - self.root = init_simulation(self.env.config, mode = 'visu') + if self.env.config["use_server"]: + self.root = init_simulation(self.env.config, mode = 'visu') + else: + self.root = root + scene = self.env.config['scene'] self._setPos = importlib.import_module("sofagym.envs."+scene+"."+scene+"Toolbox").setPos @@ -132,15 +136,21 @@ def render(self, pos=None): # Recovering an image and handling error cases try: if pos is None: - pos = get_position(self.env.past_actions)['position'] + if self.env.config["use_server"]: + pos = get_position(self.env.past_actions)['position'] + else: + pos = self.env.pos + self.frame += 1 + if pos == []: image = np.zeros((self.surface_size[0], self.surface_size[1], 3)) else: num_im = 0 for p in pos: - self._setPos(self.root, p) - Sofa.Simulation.animate(self.root, self.root.getDt()) + if self.env.config["use_server"]: + self._setPos(self.root, p) + Sofa.Simulation.animate(self.root, self.root.getDt()) glViewport(0, 0, self.surface_size[0], self.surface_size[1]) From 56c468b4528669c6f2958b0a30b4ef6b18c43dca Mon Sep 17 00:00:00 2001 From: samuelmyoussef Date: Fri, 8 Dec 2023 11:17:45 +0100 Subject: [PATCH 04/85] modify simulate & goal initializtion --- sofagym/AbstractEnv.py | 64 +++++++++++++++++++++++------------------- sofagym/simulate.py | 22 ++++++++++----- 2 files changed, 50 insertions(+), 36 deletions(-) diff --git a/sofagym/AbstractEnv.py b/sofagym/AbstractEnv.py index 6f5b192..715be19 100644 --- a/sofagym/AbstractEnv.py +++ b/sofagym/AbstractEnv.py @@ -149,16 +149,9 @@ def __init__(self, default_config, config=None, render_mode: Optional[str]=None, print("sofagym.envs."+self.scene+"." + self.scene + "Scene") raise NotImplementedError("Importing your SOFA Scene Failed") from exc - if isinstance(root, type(None)): - self.root = self.init_simulation() - else: - self.root = root - self.viewer = None self.render_mode = render_mode - self.goalList = None - self.goal = None self.past_actions = [] self.pos = [] @@ -175,6 +168,23 @@ def __init__(self, default_config, config=None, render_mode: Optional[str]=None, self.init_save_paths() + self.goal = None + if self.config["goal"]: + self.goalList = self.config["goalList"] + self.init_goal() + + if isinstance(root, type(None)): + self.root = self.init_simulation() + else: + self.root = root + + def init_goal(self): + # Set a new random goal from the list + id_goal = self.np_random.choice(range(len(self.goalList))) + self.config.update({'goal_node': id_goal}) + self.goal = self.goalList[id_goal] + self.config.update({'goalPos': self.goal}) + def init_save_paths(self): """Create directories to save results and images. @@ -298,7 +308,6 @@ def step(self, action): """ - # assert self.action_space.contains(action), "%r (%s) invalid" % (action, type(action)) action = self._formataction(action) @@ -374,13 +383,9 @@ def reset(self): self.viewer = None splib3.animation.animate.manager = None - if not self.goalList: - self.goalList = self.config["goalList"] - # Set a new random goal from the list - id_goal = self.np_random.choice(range(len(self.goalList))) - self.config.update({'goal_node': id_goal}) - self.goal = self.goalList[id_goal] + if self.config["goal"]: + self.init_goal() self.timer = 0 self.past_actions = [] @@ -492,8 +497,10 @@ def init_simulation(self, mode='simu_and_visu'): print(">> ... Done.") # Init Reward and GoalSetter - root.GoalSetter.update(self.config["goalList"]) - root.Reward.update(self.config["goalList"]) + if self.config["goal"]: + root.GoalSetter.update(self.goal) + + root.Reward.update(0) try: root.StateInitializer.init_state(self.config["init_states"]) @@ -505,9 +512,12 @@ def init_simulation(self, mode='simu_and_visu'): for _ in range(self.config["time_before_start"]): Sofa.Simulation.animate(root, self.config["dt"]) print(">> ... Done.") + # Update Reward and GoalSetter - root.GoalSetter.update(self.config["goalList"]) - root.Reward.update(self.config["goalList"]) + if self.config["goal"]: + root.GoalSetter.update(self.goal) + + root.Reward.update(self.goal) return root @@ -535,12 +545,13 @@ def step_simulation(self, action): The positions of object(s) in the scene. """ - goal = self.config['goalPos'] + if self.config["goal"]: + goal = self.config['goalPos'] + self.root.GoalSetter.set_mo_pos(goal) + render = self.config['render'] surface_size = self.config['display_size'] - self.root.GoalSetter.set_mo_pos(goal) - # Create the command from action self._startCmd(self.root, action, self.config["dt"]*(self.config["scale_factor"]-1)) pos = [] @@ -589,7 +600,6 @@ def step(self, action): info(dict): additional information (not used here) """ - # assert self.action_space.contains(action), "%r (%s) invalid" % (action, type(action)) action = self._formataction(action) @@ -632,13 +642,9 @@ def reset(self): self.viewer = None splib3.animation.animate.manager = None - if not self.goalList: - self.goalList = self.config["goalList"] - - # Set a new random goal from the list - id_goal = self.np_random.choice(range(len(self.goalList))) - self.config.update({'goal_node': id_goal}) - self.goal = self.goalList[id_goal] + + if self.config["goal"]: + self.init_goal() self.timer = 0 self.past_actions = [] diff --git a/sofagym/simulate.py b/sofagym/simulate.py index 4ac2dab..7d0f649 100644 --- a/sofagym/simulate.py +++ b/sofagym/simulate.py @@ -62,11 +62,13 @@ def init_simulation(config, _startCmd=None, mode="simu_and_visu"): print(">> ... Done.") # Init Reward and GoalSetter - root.GoalSetter.update() - root.Reward.update() + if config["goal"]: + root.GoalSetter.update(config['goalPos']) + + root.Reward.update(0) try: - root.StateInitializer.init_state() + root.StateInitializer.init_state(config["init_states"]) except AttributeError as error: print(error) @@ -75,9 +77,13 @@ def init_simulation(config, _startCmd=None, mode="simu_and_visu"): for i in range(config["time_before_start"]): Sofa.Simulation.animate(root, config["dt"]) print(">> ... Done.") + # Update Reward and GoalSetter - root.GoalSetter.update() - root.Reward.update() + if config["goal"]: + root.GoalSetter.update(config['goalPos']) + root.Reward.update(config['goalPos']) + else: + root.Reward.update() return root @@ -106,11 +112,13 @@ def step_simulation(root, config, action, _startCmd, _getPos, viewer=None): The positions of object(s) in the scene. """ - goal = config['goalPos'] + if config["goal"]: + goal = config['goalPos'] + root.GoalSetter.set_mo_pos(goal) + render = config['render'] surface_size = config['display_size'] - root.GoalSetter.set_mo_pos(goal) # Create the command from action _startCmd(root, action, config["dt"]*(config["scale_factor"]-1)) From 7e3d0f0178784a1200a11c15a5833716d4ae120c Mon Sep 17 00:00:00 2001 From: samuelmyoussef Date: Mon, 23 Sep 2024 16:07:28 +0200 Subject: [PATCH 05/85] simulation init and reset --- sofagym/AbstractEnv.py | 37 ++++++++++++++++++------------------- sofagym/simulate.py | 4 +--- 2 files changed, 19 insertions(+), 22 deletions(-) diff --git a/sofagym/AbstractEnv.py b/sofagym/AbstractEnv.py index 715be19..a138f9f 100644 --- a/sofagym/AbstractEnv.py +++ b/sofagym/AbstractEnv.py @@ -173,10 +173,7 @@ def __init__(self, default_config, config=None, render_mode: Optional[str]=None, self.goalList = self.config["goalList"] self.init_goal() - if isinstance(root, type(None)): - self.root = self.init_simulation() - else: - self.root = root + self.root = root def init_goal(self): # Set a new random goal from the list @@ -210,6 +207,9 @@ def init_save_paths(self): self.configure({"save_path_image": save_path_image, "save_path_results": save_path_results}) + def init_root(self): + self.init_simulation() + def seed(self, seed=None): """ Computes the random generators of the environment. @@ -389,9 +389,12 @@ def reset(self): self.timer = 0 self.past_actions = [] + self.pos = [] self.past_pos = [] - self.root = self.init_simulation() + Sofa.Simulation.reset(self.root) + self.root = None + self.init_simulation() obs = np.array(self._getState(self.root), dtype=np.float32) @@ -478,11 +481,11 @@ def init_simulation(self, mode='simu_and_visu'): """ # Load the scene - root = Sofa.Core.Node("root") + self.root = Sofa.Core.Node("root") SofaRuntime.importPlugin("Sofa.Component") - self.create_scene(root, self.config, mode = mode) - Sofa.Simulation.init(root) + self.create_scene(self.root, self.config, mode=mode) + Sofa.Simulation.init(self.root) # Realise action from history if self.config['start_from_history'] is not None and self._startCmd is not None: @@ -498,28 +501,24 @@ def init_simulation(self, mode='simu_and_visu'): # Init Reward and GoalSetter if self.config["goal"]: - root.GoalSetter.update(self.goal) + self.root.GoalSetter.update(self.goal) - root.Reward.update(0) + self.root.Reward.update(0) - try: - root.StateInitializer.init_state(self.config["init_states"]) - except AttributeError as error: - print(error) + if self.config["randomize_states"]: + self.root.StateInitializer.init_state(self.config["init_states"]) if 'time_before_start' in self.config: print(">> Time before start:", self.config["time_before_start"], "steps. Initialization ...") for _ in range(self.config["time_before_start"]): - Sofa.Simulation.animate(root, self.config["dt"]) + Sofa.Simulation.animate(self.root, self.config["dt"]) print(">> ... Done.") # Update Reward and GoalSetter if self.config["goal"]: - root.GoalSetter.update(self.goal) + self.root.GoalSetter.update(self.goal) - root.Reward.update(self.goal) - - return root + self.root.Reward.update(self.goal) def step_simulation(self, action): """Realise one step in the simulation. diff --git a/sofagym/simulate.py b/sofagym/simulate.py index 7d0f649..a529bc2 100644 --- a/sofagym/simulate.py +++ b/sofagym/simulate.py @@ -67,10 +67,8 @@ def init_simulation(config, _startCmd=None, mode="simu_and_visu"): root.Reward.update(0) - try: + if config["randomize_states"]: root.StateInitializer.init_state(config["init_states"]) - except AttributeError as error: - print(error) if 'time_before_start' in config: print(">> Time before start:", config["time_before_start"], "steps. Initialization ...") From a8ace0e748ae9418c5bacb407f3e00853f78e05b Mon Sep 17 00:00:00 2001 From: samuelmyoussef Date: Mon, 23 Sep 2024 16:17:49 +0200 Subject: [PATCH 06/85] remove unneeded goal and reward update and unused reward update param --- sofagym/AbstractEnv.py | 16 +++++----------- sofagym/simulate.py | 18 ++++++------------ 2 files changed, 11 insertions(+), 23 deletions(-) diff --git a/sofagym/AbstractEnv.py b/sofagym/AbstractEnv.py index a138f9f..0d83103 100644 --- a/sofagym/AbstractEnv.py +++ b/sofagym/AbstractEnv.py @@ -499,12 +499,6 @@ def init_simulation(self, mode='simu_and_visu'): self.config.update({'render': render}) print(">> ... Done.") - # Init Reward and GoalSetter - if self.config["goal"]: - self.root.GoalSetter.update(self.goal) - - self.root.Reward.update(0) - if self.config["randomize_states"]: self.root.StateInitializer.init_state(self.config["init_states"]) @@ -514,11 +508,11 @@ def init_simulation(self, mode='simu_and_visu'): Sofa.Simulation.animate(self.root, self.config["dt"]) print(">> ... Done.") - # Update Reward and GoalSetter - if self.config["goal"]: - self.root.GoalSetter.update(self.goal) - - self.root.Reward.update(self.goal) + # Update Reward and GoalSetter + if self.config["goal"]: + self.root.GoalSetter.update(self.goal) + + self.root.Reward.update(self.goal) def step_simulation(self, action): """Realise one step in the simulation. diff --git a/sofagym/simulate.py b/sofagym/simulate.py index a529bc2..b06b52f 100644 --- a/sofagym/simulate.py +++ b/sofagym/simulate.py @@ -61,12 +61,6 @@ def init_simulation(config, _startCmd=None, mode="simu_and_visu"): config.update({'render': render}) print(">> ... Done.") - # Init Reward and GoalSetter - if config["goal"]: - root.GoalSetter.update(config['goalPos']) - - root.Reward.update(0) - if config["randomize_states"]: root.StateInitializer.init_state(config["init_states"]) @@ -76,12 +70,12 @@ def init_simulation(config, _startCmd=None, mode="simu_and_visu"): Sofa.Simulation.animate(root, config["dt"]) print(">> ... Done.") - # Update Reward and GoalSetter - if config["goal"]: - root.GoalSetter.update(config['goalPos']) - root.Reward.update(config['goalPos']) - else: - root.Reward.update() + # Update Reward and GoalSetter + if config["goal"]: + root.GoalSetter.update(config['goalPos']) + root.Reward.update(config['goalPos']) + else: + root.Reward.update() return root From 55cd226d2ed53ef3b7abe1f9d15c43d6f8e2a58e Mon Sep 17 00:00:00 2001 From: samuelmyoussef Date: Wed, 25 Sep 2024 11:39:04 +0200 Subject: [PATCH 07/85] add general methods to init and randomize states and goal --- sofagym/AbstractEnv.py | 65 +++++++++++++++++++++++++++++++----------- 1 file changed, 48 insertions(+), 17 deletions(-) diff --git a/sofagym/AbstractEnv.py b/sofagym/AbstractEnv.py index 0d83103..64f7e06 100644 --- a/sofagym/AbstractEnv.py +++ b/sofagym/AbstractEnv.py @@ -168,19 +168,14 @@ def __init__(self, default_config, config=None, render_mode: Optional[str]=None, self.init_save_paths() - self.goal = None - if self.config["goal"]: - self.goalList = self.config["goalList"] - self.init_goal() - self.root = root - def init_goal(self): - # Set a new random goal from the list - id_goal = self.np_random.choice(range(len(self.goalList))) - self.config.update({'goal_node': id_goal}) - self.goal = self.goalList[id_goal] - self.config.update({'goalPos': self.goal}) + self.init_states = None + + self.goal = None + + self.nb_actions = self.config["nb_actions"] + self.dim_state = self.config["dim_state"] def init_save_paths(self): """Create directories to save results and images. @@ -210,6 +205,35 @@ def init_save_paths(self): def init_root(self): self.init_simulation() + def initialize_states(self): + if self.config["randomize_states"]: + self.init_states = self.randomize_init_states() + self.config.update({'init_states': list(self.init_states)}) + else: + self.init_states = self.config["init_states"] + + def randomize_init_states(self): + """Randomize initial states. + + Returns: + ------- + init_states: list + List of random initial states for the environment. + + Note: + ---- + This method should be implemented according to needed random initialization. + """ + return self.config["init_states"] + + def init_goal(self): + # Set a new random goal from the list + goalList = self.config["goalList"] + id_goal = self.np_random.choice(range(len(goalList))) + self.config.update({'goal_node': id_goal}) + self.goal = goalList[id_goal] + self.config.update({'goalPos': self.goal}) + def seed(self, seed=None): """ Computes the random generators of the environment. @@ -227,6 +251,19 @@ def seed(self, seed=None): self.np_random, seed = seeding.np_random(seed) return [seed] + def get_available_actions(self): + """Gives the actions available in the environment. + + Parameters: + ---------- + None. + + Returns: + ------- + list of the action available in the environment. + """ + return self.action_space + def _formataction(self, action): """Change the type of action to be in [list, float, int]. @@ -384,9 +421,6 @@ def reset(self): splib3.animation.animate.manager = None - if self.config["goal"]: - self.init_goal() - self.timer = 0 self.past_actions = [] self.pos = [] @@ -635,9 +669,6 @@ def reset(self): self.viewer = None splib3.animation.animate.manager = None - - if self.config["goal"]: - self.init_goal() self.timer = 0 self.past_actions = [] From 8215ff22e5bfbb4e68f640cf5a01cf0e099205e4 Mon Sep 17 00:00:00 2001 From: samuelmyoussef Date: Tue, 24 Sep 2024 14:42:02 +0200 Subject: [PATCH 08/85] class instantiation --- sofagym/envs/BubbleMotion/BubbleMotionEnv.py | 56 +++++++++++--------- 1 file changed, 30 insertions(+), 26 deletions(-) diff --git a/sofagym/envs/BubbleMotion/BubbleMotionEnv.py b/sofagym/envs/BubbleMotion/BubbleMotionEnv.py index 5f0d33f..ac1e73b 100644 --- a/sofagym/envs/BubbleMotion/BubbleMotionEnv.py +++ b/sofagym/envs/BubbleMotion/BubbleMotionEnv.py @@ -8,14 +8,16 @@ __copyright__ = "(c) 2021, Inria" __date__ = "Feb 3 2021" -from sofagym.AbstractEnv import AbstractEnv +from sofagym.AbstractEnv import AbstractEnv, ServerEnv from sofagym.rpc_server import start_scene from gym import spaces import os, sys import numpy as np -class BubbleMotionEnv(AbstractEnv): +from typing import Optional + +class BubbleMotionEnv: """Sub-class of AbstractEnv, dedicated to the gripper scene. See the class AbstractEnv for arguments and methods. @@ -27,6 +29,7 @@ class BubbleMotionEnv(AbstractEnv): "deterministic": True, "source": [5, -5, 20], "target": [5, 5, 0], + "goal": True, "goalList": [[7, 0, 20]], "start_node": None, "scale_factor": 20, @@ -46,43 +49,47 @@ class BubbleMotionEnv(AbstractEnv): "seed": None, "dt": 0.01, "max_pressure": 40, - "board_dim": 8 + "board_dim": 8, + "randomize_states": False, + "use_server": False } + + def __init__(self, config = None, root=None, use_server: Optional[bool]=False): + self.use_server = self.DEFAULT_CONFIG["use_server"] + self.env = ServerEnv(self.DEFAULT_CONFIG, config, root=root) if self.use_server else AbstractEnv(self.DEFAULT_CONFIG, config, root=root) - def __init__(self, config = None): - super().__init__(config) nb_actions = -1 low = np.array([-1]*9) high = np.array([1]*9) - self.action_space = spaces.Box(low=low, high=high, shape=(9,), dtype='float32') + self.env.action_space = spaces.Box(low=low, high=high, shape=(9,), dtype=np.float32) self.nb_actions = str(nb_actions) dim_state = 15 low_coordinates = np.array([0]*dim_state) high_coordinates = np.array([80]*dim_state) - self.observation_space = spaces.Box(low_coordinates, high_coordinates, - dtype='float32') - + self.env.observation_space = spaces.Box(low_coordinates, high_coordinates, dtype=np.float32) - def step(self, action): - return super().step(action) + if self.env.root is None and not self.use_server: + self.env.init_root() + # called when an attribute is not found: + def __getattr__(self, name): + # assume it is implemented by self.instance + return self.env.__getattribute__(name) + def reset(self): """Reset simulation. - - Note: - ---- - We launch a client to create the scene. The scene of the program is - client_Env.py. - """ - super().reset() + self.env.reset() - self.config.update({'goalPos': self.goal}) - obs = start_scene(self.config, self.nb_actions) + if self.use_server: + obs = start_scene(self.env.config, self.nb_actions) + state = np.array(obs['observation'], dtype=np.float32) + else: + state = np.array(self.env._getState(self.env.root), dtype=np.float32) - return np.array(obs['observation']) - + return state + def get_available_actions(self): """Gives the actions available in the environment. @@ -94,7 +101,4 @@ def get_available_actions(self): ------- list of the action available in the environment. """ - return self.action_space - - - + return self.env.action_space From 8d26c3235124317edd4fd4da432f7fc82774a075 Mon Sep 17 00:00:00 2001 From: samuelmyoussef Date: Tue, 24 Sep 2024 15:02:03 +0200 Subject: [PATCH 09/85] add some default configs --- sofagym/envs/BubbleMotion/BubbleMotionEnv.py | 7 +++++-- 1 file changed, 5 insertions(+), 2 deletions(-) diff --git a/sofagym/envs/BubbleMotion/BubbleMotionEnv.py b/sofagym/envs/BubbleMotion/BubbleMotionEnv.py index ac1e73b..a79fb67 100644 --- a/sofagym/envs/BubbleMotion/BubbleMotionEnv.py +++ b/sofagym/envs/BubbleMotion/BubbleMotionEnv.py @@ -25,6 +25,7 @@ class BubbleMotionEnv: #Setting a default configuration path = os.path.dirname(os.path.abspath(__file__)) metadata = {'render.modes': ['human', 'rgb_array']} + dim_state = 15 DEFAULT_CONFIG = {"scene": "BubbleMotion", "deterministic": True, "source": [5, -5, 20], @@ -50,6 +51,8 @@ class BubbleMotionEnv: "dt": 0.01, "max_pressure": 40, "board_dim": 8, + "nb_actions": -1, + "dim_state": dim_state, "randomize_states": False, "use_server": False } @@ -58,13 +61,13 @@ def __init__(self, config = None, root=None, use_server: Optional[bool]=False): self.use_server = self.DEFAULT_CONFIG["use_server"] self.env = ServerEnv(self.DEFAULT_CONFIG, config, root=root) if self.use_server else AbstractEnv(self.DEFAULT_CONFIG, config, root=root) - nb_actions = -1 + nb_actions = self.env.config["nb_actions"] low = np.array([-1]*9) high = np.array([1]*9) self.env.action_space = spaces.Box(low=low, high=high, shape=(9,), dtype=np.float32) self.nb_actions = str(nb_actions) - dim_state = 15 + dim_state = self.env.config["dim_state"] low_coordinates = np.array([0]*dim_state) high_coordinates = np.array([80]*dim_state) self.env.observation_space = spaces.Box(low_coordinates, high_coordinates, dtype=np.float32) From 234e3a411a5ada731f22b11877051b9e612d2805 Mon Sep 17 00:00:00 2001 From: samuelmyoussef Date: Wed, 25 Sep 2024 11:45:06 +0200 Subject: [PATCH 10/85] initialize states --- sofagym/envs/BubbleMotion/BubbleMotionEnv.py | 30 +++++++++++++++++++ .../envs/BubbleMotion/BubbleMotionScene.py | 3 +- 2 files changed, 32 insertions(+), 1 deletion(-) diff --git a/sofagym/envs/BubbleMotion/BubbleMotionEnv.py b/sofagym/envs/BubbleMotion/BubbleMotionEnv.py index a79fb67..84bfa69 100644 --- a/sofagym/envs/BubbleMotion/BubbleMotionEnv.py +++ b/sofagym/envs/BubbleMotion/BubbleMotionEnv.py @@ -54,6 +54,7 @@ class BubbleMotionEnv: "nb_actions": -1, "dim_state": dim_state, "randomize_states": False, + "init_states": [0] * dim_state, "use_server": False } @@ -61,6 +62,8 @@ def __init__(self, config = None, root=None, use_server: Optional[bool]=False): self.use_server = self.DEFAULT_CONFIG["use_server"] self.env = ServerEnv(self.DEFAULT_CONFIG, config, root=root) if self.use_server else AbstractEnv(self.DEFAULT_CONFIG, config, root=root) + self.initialize_states() + nb_actions = self.env.config["nb_actions"] low = np.array([-1]*9) high = np.array([1]*9) @@ -79,10 +82,37 @@ def __init__(self, config = None, root=None, use_server: Optional[bool]=False): def __getattr__(self, name): # assume it is implemented by self.instance return self.env.__getattribute__(name) + + def initialize_states(self): + if self.env.config["randomize_states"]: + self.init_states = self.randomize_init_states() + self.env.config.update({'init_states': list(self.init_states)}) + else: + self.init_states = self.env.config["init_states"] + + bd = self.env.config["board_dim"] + init_pos = [4+(bd-4)*self.env.np_random.random(), 4+(bd-4)*self.env.np_random.random(), 5] + self.env.config.update({'init_pos': init_pos}) + + def randomize_init_states(self): + """Randomize initial states. + + Returns: + ------- + init_states: list + List of random initial states for the environment. + + Note: + ---- + This method should be implemented according to needed random initialization. + """ + return self.env.config["init_states"] def reset(self): """Reset simulation. """ + self.initialize_states() + self.env.reset() if self.use_server: diff --git a/sofagym/envs/BubbleMotion/BubbleMotionScene.py b/sofagym/envs/BubbleMotion/BubbleMotionScene.py index 6d953d2..d6feb6a 100644 --- a/sofagym/envs/BubbleMotion/BubbleMotionScene.py +++ b/sofagym/envs/BubbleMotion/BubbleMotionScene.py @@ -45,6 +45,7 @@ def createScene(rootNode, config={"source": [5, -5, 20], "zFar":4000, "dt": 0.01, "max_pressure": 40, + "init_pos": [5, 5, 5], "board_dim": 8}, mode='simu_and_visu'): @@ -59,7 +60,7 @@ def createScene(rootNode, config={"source": [5, -5, 20], pos_goal = [1+bd*np.random.random(), 1+bd*np.random.random(), 2] add_goal_node(rootNode, pos_goal) - init_pos = [4+(bd-4)*np.random.random(), 4+(bd-4)*np.random.random(), 5] + init_pos = config["init_pos"] bubblemotion_config = {'init_pos': init_pos, "dt": config["dt"], "max_pressure": config["max_pressure"]} bubblemotion = BubbleMotion(bubblemotion_config=bubblemotion_config) From f96c776af88af854ff348d25faa62e788958db86 Mon Sep 17 00:00:00 2001 From: samuelmyoussef Date: Wed, 25 Sep 2024 12:41:31 +0200 Subject: [PATCH 11/85] goal init and update --- sofagym/envs/BubbleMotion/BubbleMotionEnv.py | 12 ++++++++++++ sofagym/envs/BubbleMotion/BubbleMotionScene.py | 2 +- sofagym/envs/BubbleMotion/BubbleMotionToolbox.py | 7 ++++--- 3 files changed, 17 insertions(+), 4 deletions(-) diff --git a/sofagym/envs/BubbleMotion/BubbleMotionEnv.py b/sofagym/envs/BubbleMotion/BubbleMotionEnv.py index 84bfa69..6756c44 100644 --- a/sofagym/envs/BubbleMotion/BubbleMotionEnv.py +++ b/sofagym/envs/BubbleMotion/BubbleMotionEnv.py @@ -64,6 +64,9 @@ def __init__(self, config = None, root=None, use_server: Optional[bool]=False): self.initialize_states() + if self.env.config["goal"]: + self.init_goal() + nb_actions = self.env.config["nb_actions"] low = np.array([-1]*9) high = np.array([1]*9) @@ -107,12 +110,21 @@ def randomize_init_states(self): This method should be implemented according to needed random initialization. """ return self.env.config["init_states"] + + def init_goal(self): + bd = self.env.config["board_dim"] + pos_goal = [1+bd*self.env.np_random.random(), 1+bd*self.env.np_random.random(), 2] + self.env.goal = pos_goal + self.env.config.update({'goalPos': self.env.goal}) def reset(self): """Reset simulation. """ self.initialize_states() + if self.env.config["goal"]: + self.init_goal() + self.env.reset() if self.use_server: diff --git a/sofagym/envs/BubbleMotion/BubbleMotionScene.py b/sofagym/envs/BubbleMotion/BubbleMotionScene.py index d6feb6a..97dee83 100644 --- a/sofagym/envs/BubbleMotion/BubbleMotionScene.py +++ b/sofagym/envs/BubbleMotion/BubbleMotionScene.py @@ -57,7 +57,7 @@ def createScene(rootNode, config={"source": [5, -5, 20], visu(rootNode, config, position_spot, direction_spot, cutoff=250) bd = config["board_dim"] - pos_goal = [1+bd*np.random.random(), 1+bd*np.random.random(), 2] + pos_goal = config["goalPos"] add_goal_node(rootNode, pos_goal) init_pos = config["init_pos"] diff --git a/sofagym/envs/BubbleMotion/BubbleMotionToolbox.py b/sofagym/envs/BubbleMotion/BubbleMotionToolbox.py index 9d4dc71..8f4e5db 100644 --- a/sofagym/envs/BubbleMotion/BubbleMotionToolbox.py +++ b/sofagym/envs/BubbleMotion/BubbleMotionToolbox.py @@ -83,7 +83,7 @@ def getReward(self): return r, dist - def update(self): + def update(self, goal=None): """Update function. This function is used as an initialization function. @@ -97,6 +97,7 @@ def update(self): None. """ + self.goal = goal[:2] current_sphere_pos = self._getSpherePos() self.init_goal_dist = float(np.linalg.norm(current_sphere_pos-self.goal)) @@ -158,8 +159,8 @@ def __init__(self, *args, **kwargs): if 'goalPos' in kwargs: self.goalPos = kwargs["goalPos"] - def update(self): - pass + def update(self, goal): + self.goalPos = goal def set_mo_pos(self, goal): pass From 57b790f34fb25f5eee2cd6df555ba89ce6d15fe6 Mon Sep 17 00:00:00 2001 From: samuelmyoussef Date: Thu, 26 Sep 2024 02:38:52 +0200 Subject: [PATCH 12/85] class instantiation --- sofagym/envs/Trunk/TrunkEnv.py | 58 +++++++++++++++++++--------------- 1 file changed, 33 insertions(+), 25 deletions(-) diff --git a/sofagym/envs/Trunk/TrunkEnv.py b/sofagym/envs/Trunk/TrunkEnv.py index 5988b9a..9e87c40 100644 --- a/sofagym/envs/Trunk/TrunkEnv.py +++ b/sofagym/envs/Trunk/TrunkEnv.py @@ -12,14 +12,15 @@ import numpy as np import sys -from sofagym.AbstractEnv import AbstractEnv +from sofagym.AbstractEnv import AbstractEnv, ServerEnv from sofagym.rpc_server import start_scene from gym import spaces +from typing import Optional -class TrunkEnv(AbstractEnv): +class TrunkEnv: """Sub-class of AbstractEnv, dedicated to the trunk scene. See the class AbstractEnv for arguments and methods. @@ -31,6 +32,7 @@ class TrunkEnv(AbstractEnv): "deterministic": True, "source": [300, 0, 80], "target": [0, 0, 80], + "goal": True, "goalList": [[40, 40, 100], [-10, 20, 80]], "start_node": None, "scale_factor": 5, @@ -45,39 +47,47 @@ class TrunkEnv(AbstractEnv): "discrete": True, "seed": None, "start_from_history": None, - "python_version": "python3", - "dt": 0.01 + "python_version": sys.version, + "dt": 0.01, + "randomize_states": False, + "use_server": False } - def __init__(self, config = None): - super().__init__(config) + def __init__(self, config = None, root=None, use_server: Optional[bool]=False): + self.use_server = self.DEFAULT_CONFIG["use_server"] + self.env = ServerEnv(self.DEFAULT_CONFIG, config, root=root) if self.use_server else AbstractEnv(self.DEFAULT_CONFIG, config, root=root) + nb_actions = 16 - self.action_space = spaces.Discrete(nb_actions) + self.env.action_space = spaces.Discrete(nb_actions) self.nb_actions = str(nb_actions) dim_state = 66 low_coordinates = np.array([-1]*dim_state) high_coordinates = np.array([1]*dim_state) - self.observation_space = spaces.Box(low_coordinates, high_coordinates, dtype='float32') + self.env.observation_space = spaces.Box(low_coordinates, high_coordinates, dtype=np.float32) + + if self.env.root is None and not self.use_server: + self.env.init_root() - def step(self, action): - return super().step(action) + # called when an attribute is not found: + def __getattr__(self, name): + # assume it is implemented by self.instance + return self.env.__getattribute__(name) def reset(self): """Reset simulation. - - Note: - ---- - We launch a client to create the scene. The scene of the program is - client_Env.py. - """ - super().reset() - - self.config.update({'goalPos': self.goal}) - obs = start_scene(self.config, self.nb_actions) - - return np.array(obs['observation']) + self.env.reset() + + self.env.config.update({'goalPos': self.env.goal}) + + if self.use_server: + obs = start_scene(self.env.config, self.nb_actions) + state = np.array(obs['observation'], dtype=np.float32) + else: + state = np.array(self.env._getState(self.env.root), dtype=np.float32) + + return state def get_available_actions(self): """Gives the actions available in the environment. @@ -90,6 +100,4 @@ def get_available_actions(self): ------- list of the action available in the environment. """ - return list(range(int(self.nb_actions))) - - + return self.env.action_space From 5c8a2d54cb69d08bb61f87a02f4626c0d69be68d Mon Sep 17 00:00:00 2001 From: samuelmyoussef Date: Thu, 26 Sep 2024 02:39:46 +0200 Subject: [PATCH 13/85] add some default configs --- sofagym/envs/Trunk/TrunkEnv.py | 7 +++++-- 1 file changed, 5 insertions(+), 2 deletions(-) diff --git a/sofagym/envs/Trunk/TrunkEnv.py b/sofagym/envs/Trunk/TrunkEnv.py index 9e87c40..a7ad195 100644 --- a/sofagym/envs/Trunk/TrunkEnv.py +++ b/sofagym/envs/Trunk/TrunkEnv.py @@ -28,6 +28,7 @@ class TrunkEnv: # Setting a default configuration path = os.path.dirname(os.path.abspath(__file__)) metadata = {'render.modes': ['human', 'rgb_array']} + dim_state = 66 DEFAULT_CONFIG = {"scene": "Trunk", "deterministic": True, "source": [300, 0, 80], @@ -49,6 +50,8 @@ class TrunkEnv: "start_from_history": None, "python_version": sys.version, "dt": 0.01, + "nb_actions": 16, + "dim_state": dim_state, "randomize_states": False, "use_server": False } @@ -57,11 +60,11 @@ def __init__(self, config = None, root=None, use_server: Optional[bool]=False): self.use_server = self.DEFAULT_CONFIG["use_server"] self.env = ServerEnv(self.DEFAULT_CONFIG, config, root=root) if self.use_server else AbstractEnv(self.DEFAULT_CONFIG, config, root=root) - nb_actions = 16 + nb_actions = self.env.config["nb_actions"] self.env.action_space = spaces.Discrete(nb_actions) self.nb_actions = str(nb_actions) - dim_state = 66 + dim_state = self.env.config["dim_state"] low_coordinates = np.array([-1]*dim_state) high_coordinates = np.array([1]*dim_state) self.env.observation_space = spaces.Box(low_coordinates, high_coordinates, dtype=np.float32) From 75778c2ee6409a06f46fa6441f76346cd0d0467b Mon Sep 17 00:00:00 2001 From: samuelmyoussef Date: Thu, 26 Sep 2024 02:42:03 +0200 Subject: [PATCH 14/85] initialize states --- sofagym/envs/Trunk/TrunkEnv.py | 26 ++++++++++++++++++++++++++ 1 file changed, 26 insertions(+) diff --git a/sofagym/envs/Trunk/TrunkEnv.py b/sofagym/envs/Trunk/TrunkEnv.py index a7ad195..1db308b 100644 --- a/sofagym/envs/Trunk/TrunkEnv.py +++ b/sofagym/envs/Trunk/TrunkEnv.py @@ -53,6 +53,7 @@ class TrunkEnv: "nb_actions": 16, "dim_state": dim_state, "randomize_states": False, + "init_states": [0] * dim_state, "use_server": False } @@ -60,6 +61,8 @@ def __init__(self, config = None, root=None, use_server: Optional[bool]=False): self.use_server = self.DEFAULT_CONFIG["use_server"] self.env = ServerEnv(self.DEFAULT_CONFIG, config, root=root) if self.use_server else AbstractEnv(self.DEFAULT_CONFIG, config, root=root) + self.initialize_states() + nb_actions = self.env.config["nb_actions"] self.env.action_space = spaces.Discrete(nb_actions) self.nb_actions = str(nb_actions) @@ -77,9 +80,32 @@ def __getattr__(self, name): # assume it is implemented by self.instance return self.env.__getattribute__(name) + def initialize_states(self): + if self.env.config["randomize_states"]: + self.init_states = self.randomize_init_states() + self.env.config.update({'init_states': list(self.init_states)}) + else: + self.init_states = self.env.config["init_states"] + + def randomize_init_states(self): + """Randomize initial states. + + Returns: + ------- + init_states: list + List of random initial states for the environment. + + Note: + ---- + This method should be implemented according to needed random initialization. + """ + return self.env.config["init_states"] + def reset(self): """Reset simulation. """ + self.initialize_states() + self.env.reset() self.env.config.update({'goalPos': self.env.goal}) From 5ad58837e6fb62d8808cf6c8748810e9b5dea7ae Mon Sep 17 00:00:00 2001 From: samuelmyoussef Date: Thu, 26 Sep 2024 02:45:39 +0200 Subject: [PATCH 15/85] goal init and update --- sofagym/envs/Trunk/TrunkEnv.py | 18 +++++++++++++++--- sofagym/envs/Trunk/TrunkToolbox.py | 7 ++++--- 2 files changed, 19 insertions(+), 6 deletions(-) diff --git a/sofagym/envs/Trunk/TrunkEnv.py b/sofagym/envs/Trunk/TrunkEnv.py index 1db308b..c688a66 100644 --- a/sofagym/envs/Trunk/TrunkEnv.py +++ b/sofagym/envs/Trunk/TrunkEnv.py @@ -63,6 +63,9 @@ def __init__(self, config = None, root=None, use_server: Optional[bool]=False): self.initialize_states() + if self.env.config["goal"]: + self.init_goal() + nb_actions = self.env.config["nb_actions"] self.env.action_space = spaces.Discrete(nb_actions) self.nb_actions = str(nb_actions) @@ -101,14 +104,23 @@ def randomize_init_states(self): """ return self.env.config["init_states"] + def init_goal(self): + # Set a new random goal from the list + goalList = self.env.config["goalList"] + id_goal = self.env.np_random.choice(range(len(goalList))) + self.env.config.update({'goal_node': id_goal}) + self.env.goal = goalList[id_goal] + self.env.config.update({'goalPos': self.env.goal}) + def reset(self): """Reset simulation. """ self.initialize_states() - - self.env.reset() - self.env.config.update({'goalPos': self.env.goal}) + if self.env.config["goal"]: + self.init_goal() + + self.env.reset() if self.use_server: obs = start_scene(self.env.config, self.nb_actions) diff --git a/sofagym/envs/Trunk/TrunkToolbox.py b/sofagym/envs/Trunk/TrunkToolbox.py index 38af0dc..ce09fd6 100644 --- a/sofagym/envs/Trunk/TrunkToolbox.py +++ b/sofagym/envs/Trunk/TrunkToolbox.py @@ -81,7 +81,7 @@ def getReward(self): return min(reward**(1/2), 1.0), current_dist - def update(self): + def update(self, goal=None): """Update function. This function is used as an initialization function. @@ -95,7 +95,7 @@ def update(self): None. """ - + self.goal_pos = goal trunkTips = self._computeTips() self.init_dist = np.linalg.norm(np.array(trunkTips)-np.array(self.goal_pos)) self.prev_dist = self.init_dist @@ -160,7 +160,7 @@ def __init__(self, *args, **kwargs): if kwargs["goalPos"]: self.goalPos = kwargs["goalPos"] - def update(self): + def update(self, goal): """Set the position of the goal. This function is used as an initialization function. @@ -174,6 +174,7 @@ def update(self): None. """ + self.goalPos = goal with self.goalMO.position.writeable() as position: position += self.goalPos From 8cfbc8280f10a5158291ca8d342ab6f815f4ea56 Mon Sep 17 00:00:00 2001 From: samuelmyoussef Date: Thu, 26 Sep 2024 02:54:38 +0200 Subject: [PATCH 16/85] class instantiation --- sofagym/envs/TrunkCup/TrunkCupEnv.py | 63 +++++++++++++++------------- 1 file changed, 35 insertions(+), 28 deletions(-) diff --git a/sofagym/envs/TrunkCup/TrunkCupEnv.py b/sofagym/envs/TrunkCup/TrunkCupEnv.py index edd049e..199aca6 100644 --- a/sofagym/envs/TrunkCup/TrunkCupEnv.py +++ b/sofagym/envs/TrunkCup/TrunkCupEnv.py @@ -8,15 +8,17 @@ __copyright__ = "(c) 2020, Inria" __date__ = "Oct 7 2020" -import os +import os, sys import numpy as np -from sofagym.AbstractEnv import AbstractEnv +from sofagym.AbstractEnv import AbstractEnv, ServerEnv from sofagym.rpc_server import start_scene from gym import spaces -class TrunkCupEnv(AbstractEnv): +from typing import Optional + +class TrunkCupEnv: """Sub-class of AbstractEnv, dedicated to the trunk scene. See the class AbstractEnv for arguments and methods. @@ -28,6 +30,7 @@ class TrunkCupEnv(AbstractEnv): "deterministic": True, "source": [500.0, 0, 100], "target": [0, 0, 100], + "goal": True, "goalList": [[40, 40, 100], [-10, 20, 80]], "start_node": None, "scale_factor": 5, @@ -42,40 +45,47 @@ class TrunkCupEnv(AbstractEnv): "discrete": True, "seed": None, "start_from_history": None, - "python_version": "python3.8", - "dt": 0.01 + "python_version": sys.version, + "dt": 0.01, + "randomize_states": False, + "use_server": False } - def __init__(self, config=None): - super().__init__(config) + def __init__(self, config = None, root=None, use_server: Optional[bool]=False): + self.use_server = self.DEFAULT_CONFIG["use_server"] + self.env = ServerEnv(self.DEFAULT_CONFIG, config, root=root) if self.use_server else AbstractEnv(self.DEFAULT_CONFIG, config, root=root) + nb_actions = 16 - self.action_space = spaces.Discrete(nb_actions) + self.env.action_space = spaces.Discrete(nb_actions) self.nb_actions = str(nb_actions) dim_state = 66 low_coordinates = np.array([-1]*dim_state) high_coordinates = np.array([1]*dim_state) - self.observation_space = spaces.Box(low_coordinates, high_coordinates, - dtype='float32') + self.env.observation_space = spaces.Box(low_coordinates, high_coordinates, dtype=np.float32) + + if self.env.root is None and not self.use_server: + self.env.init_root() - def step(self, action): - return super().step(action) + # called when an attribute is not found: + def __getattr__(self, name): + # assume it is implemented by self.instance + return self.env.__getattribute__(name) def reset(self): """Reset simulation. - - Note: - ---- - We launch a client to create the scene. The scene of the program is - client_Env.py. - """ - super().reset() - - self.config.update({'goalPos': self.goal}) - obs = start_scene(self.config, self.nb_actions) - - return np.array(obs['observation']) + self.env.reset() + + self.env.config.update({'goalPos': self.env.goal}) + + if self.use_server: + obs = start_scene(self.env.config, self.nb_actions) + state = np.array(obs['observation'], dtype=np.float32) + else: + state = np.array(self.env._getState(self.env.root), dtype=np.float32) + + return state def get_available_actions(self): """Gives the actions available in the environment. @@ -88,7 +98,4 @@ def get_available_actions(self): ------- list of the action available in the environment. """ - return list(range(int(self.nb_actions))) - - - + return self.env.action_space From 07978557850f9604e93377e9fdba9fea28f4438a Mon Sep 17 00:00:00 2001 From: samuelmyoussef Date: Thu, 26 Sep 2024 02:55:47 +0200 Subject: [PATCH 17/85] add some default configs --- sofagym/envs/TrunkCup/TrunkCupEnv.py | 7 +++++-- 1 file changed, 5 insertions(+), 2 deletions(-) diff --git a/sofagym/envs/TrunkCup/TrunkCupEnv.py b/sofagym/envs/TrunkCup/TrunkCupEnv.py index 199aca6..296234b 100644 --- a/sofagym/envs/TrunkCup/TrunkCupEnv.py +++ b/sofagym/envs/TrunkCup/TrunkCupEnv.py @@ -26,6 +26,7 @@ class TrunkCupEnv: # Setting a default configuration path = os.path.dirname(os.path.abspath(__file__)) metadata = {'render.modes': ['human', 'rgb_array']} + dim_state = 66 DEFAULT_CONFIG = {"scene": "TrunkCup", "deterministic": True, "source": [500.0, 0, 100], @@ -47,6 +48,8 @@ class TrunkCupEnv: "start_from_history": None, "python_version": sys.version, "dt": 0.01, + "nb_actions": 16, + "dim_state": dim_state, "randomize_states": False, "use_server": False } @@ -55,11 +58,11 @@ def __init__(self, config = None, root=None, use_server: Optional[bool]=False): self.use_server = self.DEFAULT_CONFIG["use_server"] self.env = ServerEnv(self.DEFAULT_CONFIG, config, root=root) if self.use_server else AbstractEnv(self.DEFAULT_CONFIG, config, root=root) - nb_actions = 16 + nb_actions = self.env.config["nb_actions"] self.env.action_space = spaces.Discrete(nb_actions) self.nb_actions = str(nb_actions) - dim_state = 66 + dim_state = self.env.config["dim_state"] low_coordinates = np.array([-1]*dim_state) high_coordinates = np.array([1]*dim_state) self.env.observation_space = spaces.Box(low_coordinates, high_coordinates, dtype=np.float32) From fa15f2ab32438a158c2e5b80ae4b0124c8c81226 Mon Sep 17 00:00:00 2001 From: samuelmyoussef Date: Thu, 26 Sep 2024 02:59:18 +0200 Subject: [PATCH 18/85] initialize states --- sofagym/envs/TrunkCup/TrunkCupEnv.py | 26 ++++++++++++++++++++++++++ 1 file changed, 26 insertions(+) diff --git a/sofagym/envs/TrunkCup/TrunkCupEnv.py b/sofagym/envs/TrunkCup/TrunkCupEnv.py index 296234b..707b429 100644 --- a/sofagym/envs/TrunkCup/TrunkCupEnv.py +++ b/sofagym/envs/TrunkCup/TrunkCupEnv.py @@ -51,6 +51,7 @@ class TrunkCupEnv: "nb_actions": 16, "dim_state": dim_state, "randomize_states": False, + "init_states": [0] * dim_state, "use_server": False } @@ -58,6 +59,8 @@ def __init__(self, config = None, root=None, use_server: Optional[bool]=False): self.use_server = self.DEFAULT_CONFIG["use_server"] self.env = ServerEnv(self.DEFAULT_CONFIG, config, root=root) if self.use_server else AbstractEnv(self.DEFAULT_CONFIG, config, root=root) + self.initialize_states() + nb_actions = self.env.config["nb_actions"] self.env.action_space = spaces.Discrete(nb_actions) self.nb_actions = str(nb_actions) @@ -75,9 +78,32 @@ def __getattr__(self, name): # assume it is implemented by self.instance return self.env.__getattribute__(name) + def initialize_states(self): + if self.env.config["randomize_states"]: + self.init_states = self.randomize_init_states() + self.env.config.update({'init_states': list(self.init_states)}) + else: + self.init_states = self.env.config["init_states"] + + def randomize_init_states(self): + """Randomize initial states. + + Returns: + ------- + init_states: list + List of random initial states for the environment. + + Note: + ---- + This method should be implemented according to needed random initialization. + """ + return self.env.config["init_states"] + def reset(self): """Reset simulation. """ + self.initialize_states() + self.env.reset() self.env.config.update({'goalPos': self.env.goal}) From d5983ac659252928545ac4a02cd6a44eec876b70 Mon Sep 17 00:00:00 2001 From: samuelmyoussef Date: Thu, 26 Sep 2024 03:06:26 +0200 Subject: [PATCH 19/85] goal init and update --- sofagym/envs/TrunkCup/TrunkCupEnv.py | 18 +++++++++++++++--- sofagym/envs/TrunkCup/TrunkCupToolbox.py | 7 ++++--- 2 files changed, 19 insertions(+), 6 deletions(-) diff --git a/sofagym/envs/TrunkCup/TrunkCupEnv.py b/sofagym/envs/TrunkCup/TrunkCupEnv.py index 707b429..600818f 100644 --- a/sofagym/envs/TrunkCup/TrunkCupEnv.py +++ b/sofagym/envs/TrunkCup/TrunkCupEnv.py @@ -61,6 +61,9 @@ def __init__(self, config = None, root=None, use_server: Optional[bool]=False): self.initialize_states() + if self.env.config["goal"]: + self.init_goal() + nb_actions = self.env.config["nb_actions"] self.env.action_space = spaces.Discrete(nb_actions) self.nb_actions = str(nb_actions) @@ -99,14 +102,23 @@ def randomize_init_states(self): """ return self.env.config["init_states"] + def init_goal(self): + # Set a new random goal from the list + goalList = self.env.config["goalList"] + id_goal = self.env.np_random.choice(range(len(goalList))) + self.env.config.update({'goal_node': id_goal}) + self.env.goal = goalList[id_goal] + self.env.config.update({'goalPos': self.env.goal}) + def reset(self): """Reset simulation. """ self.initialize_states() - - self.env.reset() - self.env.config.update({'goalPos': self.env.goal}) + if self.env.config["goal"]: + self.init_goal() + + self.env.reset() if self.use_server: obs = start_scene(self.env.config, self.nb_actions) diff --git a/sofagym/envs/TrunkCup/TrunkCupToolbox.py b/sofagym/envs/TrunkCup/TrunkCupToolbox.py index 11f0fb4..27a730a 100644 --- a/sofagym/envs/TrunkCup/TrunkCupToolbox.py +++ b/sofagym/envs/TrunkCup/TrunkCupToolbox.py @@ -79,7 +79,7 @@ def getReward(self): return self.init_dist - current_dist, current_dist - def update(self): + def update(self, goal=None): """Update function. This function is used as an initialization function. @@ -93,7 +93,7 @@ def update(self): None. """ - + self.goal_pos = goal bary = self._computeCupBary() self.init_dist = np.sqrt(np.sum((bary-self.goal_pos)**2)) @@ -156,7 +156,7 @@ def __init__(self, *args, **kwargs): if kwargs["goalPos"]: self.goalPos = kwargs["goalPos"] - def update(self): + def update(self, goal): """Set the position of the goal. This function is used as an initialization function. @@ -170,6 +170,7 @@ def update(self): None. """ + self.goalPos = goal with self.goalMO.position.writeable() as position: position += self.goalPos From 06a64796de300bc0b7e6db358129ec1d703b1360 Mon Sep 17 00:00:00 2001 From: samuelmyoussef Date: Thu, 26 Sep 2024 02:13:36 +0200 Subject: [PATCH 20/85] class instantiation --- sofagym/envs/StemPendulum/StemPendulumEnv.py | 66 +++++++++++--------- 1 file changed, 35 insertions(+), 31 deletions(-) diff --git a/sofagym/envs/StemPendulum/StemPendulumEnv.py b/sofagym/envs/StemPendulum/StemPendulumEnv.py index 99b3899..997df95 100644 --- a/sofagym/envs/StemPendulum/StemPendulumEnv.py +++ b/sofagym/envs/StemPendulum/StemPendulumEnv.py @@ -8,26 +8,29 @@ __copyright__ = "(c) 2021, Inria" __date__ = "Feb 3 2021" -import os +import os, sys import numpy as np -from sofagym.AbstractEnv import AbstractEnv +from sofagym.AbstractEnv import AbstractEnv, ServerEnv from sofagym.rpc_server import start_scene from gym import spaces -class StemPendulumEnv(AbstractEnv): +from typing import Optional + +class StemPendulumEnv: """Sub-class of AbstractEnv, dedicated to the gripper scene. See the class AbstractEnv for arguments and methods. """ #Setting a default configuration - path = path = os.path.dirname(os.path.abspath(__file__)) + path = os.path.dirname(os.path.abspath(__file__)) metadata = {'render.modes': ['human', 'rgb_array']} DEFAULT_CONFIG = {"scene": "StemPendulum", "deterministic": True, "source": [0, 0, 30], "target": [0, 0, 0], + "goal": False, "goalList": [[7, 0, 20]], "start_node": None, "scale_factor": 10, @@ -42,48 +45,52 @@ class StemPendulumEnv(AbstractEnv): "planning": False, "discrete": False, "start_from_history": None, - "python_version": "python3.8", + "python_version": sys.version, "zFar": 4000, "time_before_start": 0, "seed": None, "max_torque": 500, + "randomize_states": False, + "use_server": False } - def __init__(self, config = None): - super().__init__(config) + def __init__(self, config = None, root=None, use_server: Optional[bool]=False): + self.use_server = self.DEFAULT_CONFIG["use_server"] + self.env = ServerEnv(self.DEFAULT_CONFIG, config, root=root) if self.use_server else AbstractEnv(self.DEFAULT_CONFIG, config, root=root) + nb_actions = -1 low = np.array([-1]*1) high = np.array([1]*1) - self.action_space = spaces.Box(low=low, high=high, shape=(1,), dtype='float32') + self.env.action_space = spaces.Box(low=low, high=high, shape=(1,), dtype=np.float32) self.nb_actions = str(nb_actions) dim_state = 5 low_coordinates = np.array([-2]*dim_state) high_coordinates = np.array([2]*dim_state) - self.observation_space = spaces.Box(low_coordinates, high_coordinates, - dtype='float32') + self.env.observation_space = spaces.Box(low_coordinates, high_coordinates, dtype=np.float32) + + if self.env.root is None and not self.use_server: + self.env.init_root() - def step(self, action): - return super().step(action) + # called when an attribute is not found: + def __getattr__(self, name): + # assume it is implemented by self.instance + return self.env.__getattribute__(name) def reset(self): """Reset simulation. - - Note: - ---- - We launch a client to create the scene. The scene of the program is - client_Env.py. - """ - super().reset() - - self.count = 0 - self.config.update({'goalPos': self.goal}) - # obs = super().reset() - # return np.array(obs) - obs = start_scene(self.config, self.nb_actions) - - return np.array(obs['observation']) + self.env.reset() + + self.env.config.update({'goalPos': self.env.goal}) + + if self.use_server: + obs = start_scene(self.env.config, self.nb_actions) + state = np.array(obs['observation'], dtype=np.float32) + else: + state = np.array(self.env._getState(self.env.root), dtype=np.float32) + + return state def get_available_actions(self): """Gives the actions available in the environment. @@ -96,7 +103,4 @@ def get_available_actions(self): ------- list of the action available in the environment. """ - return self.action_space - - - + return self.env.action_space From 3e3e0b78faab29820391f2e4c8fda41770a94a7c Mon Sep 17 00:00:00 2001 From: samuelmyoussef Date: Thu, 26 Sep 2024 02:17:05 +0200 Subject: [PATCH 21/85] add some default configs --- sofagym/envs/StemPendulum/StemPendulumEnv.py | 7 +++++-- 1 file changed, 5 insertions(+), 2 deletions(-) diff --git a/sofagym/envs/StemPendulum/StemPendulumEnv.py b/sofagym/envs/StemPendulum/StemPendulumEnv.py index 997df95..39bc7d2 100644 --- a/sofagym/envs/StemPendulum/StemPendulumEnv.py +++ b/sofagym/envs/StemPendulum/StemPendulumEnv.py @@ -26,6 +26,7 @@ class StemPendulumEnv: #Setting a default configuration path = os.path.dirname(os.path.abspath(__file__)) metadata = {'render.modes': ['human', 'rgb_array']} + dim_state = 5 DEFAULT_CONFIG = {"scene": "StemPendulum", "deterministic": True, "source": [0, 0, 30], @@ -50,6 +51,8 @@ class StemPendulumEnv: "time_before_start": 0, "seed": None, "max_torque": 500, + "nb_actions": -1, + "dim_state": dim_state, "randomize_states": False, "use_server": False } @@ -58,13 +61,13 @@ def __init__(self, config = None, root=None, use_server: Optional[bool]=False): self.use_server = self.DEFAULT_CONFIG["use_server"] self.env = ServerEnv(self.DEFAULT_CONFIG, config, root=root) if self.use_server else AbstractEnv(self.DEFAULT_CONFIG, config, root=root) - nb_actions = -1 + nb_actions = self.env.config["nb_actions"] low = np.array([-1]*1) high = np.array([1]*1) self.env.action_space = spaces.Box(low=low, high=high, shape=(1,), dtype=np.float32) self.nb_actions = str(nb_actions) - dim_state = 5 + dim_state = self.env.config["dim_state"] low_coordinates = np.array([-2]*dim_state) high_coordinates = np.array([2]*dim_state) self.env.observation_space = spaces.Box(low_coordinates, high_coordinates, dtype=np.float32) From ce2f4314ad9bb56c7dc98c1bfa494179eabb8f61 Mon Sep 17 00:00:00 2001 From: samuelmyoussef Date: Thu, 26 Sep 2024 02:17:58 +0200 Subject: [PATCH 22/85] initialize states --- sofagym/envs/StemPendulum/StemPendulumEnv.py | 26 ++++++++++++++++++++ 1 file changed, 26 insertions(+) diff --git a/sofagym/envs/StemPendulum/StemPendulumEnv.py b/sofagym/envs/StemPendulum/StemPendulumEnv.py index 39bc7d2..ce6ac3c 100644 --- a/sofagym/envs/StemPendulum/StemPendulumEnv.py +++ b/sofagym/envs/StemPendulum/StemPendulumEnv.py @@ -54,6 +54,7 @@ class StemPendulumEnv: "nb_actions": -1, "dim_state": dim_state, "randomize_states": False, + "init_states": [0] * dim_state, "use_server": False } @@ -61,6 +62,8 @@ def __init__(self, config = None, root=None, use_server: Optional[bool]=False): self.use_server = self.DEFAULT_CONFIG["use_server"] self.env = ServerEnv(self.DEFAULT_CONFIG, config, root=root) if self.use_server else AbstractEnv(self.DEFAULT_CONFIG, config, root=root) + self.initialize_states() + nb_actions = self.env.config["nb_actions"] low = np.array([-1]*1) high = np.array([1]*1) @@ -80,9 +83,32 @@ def __getattr__(self, name): # assume it is implemented by self.instance return self.env.__getattribute__(name) + def initialize_states(self): + if self.env.config["randomize_states"]: + self.init_states = self.randomize_init_states() + self.env.config.update({'init_states': list(self.init_states)}) + else: + self.init_states = self.env.config["init_states"] + + def randomize_init_states(self): + """Randomize initial states. + + Returns: + ------- + init_states: list + List of random initial states for the environment. + + Note: + ---- + This method should be implemented according to needed random initialization. + """ + return self.env.config["init_states"] + def reset(self): """Reset simulation. """ + self.initialize_states() + self.env.reset() self.env.config.update({'goalPos': self.env.goal}) From eacada16d6251221e45429e2711963ccae78ff8b Mon Sep 17 00:00:00 2001 From: samuelmyoussef Date: Thu, 26 Sep 2024 02:25:03 +0200 Subject: [PATCH 23/85] goal remove --- sofagym/envs/StemPendulum/StemPendulumEnv.py | 11 +++++++---- sofagym/envs/StemPendulum/StemPendulumScene.py | 11 +---------- sofagym/envs/StemPendulum/StemPendulumToolbox.py | 13 +------------ 3 files changed, 9 insertions(+), 26 deletions(-) diff --git a/sofagym/envs/StemPendulum/StemPendulumEnv.py b/sofagym/envs/StemPendulum/StemPendulumEnv.py index ce6ac3c..6edb51a 100644 --- a/sofagym/envs/StemPendulum/StemPendulumEnv.py +++ b/sofagym/envs/StemPendulum/StemPendulumEnv.py @@ -32,7 +32,6 @@ class StemPendulumEnv: "source": [0, 0, 30], "target": [0, 0, 0], "goal": False, - "goalList": [[7, 0, 20]], "start_node": None, "scale_factor": 10, "dt": 0.01, @@ -64,6 +63,9 @@ def __init__(self, config = None, root=None, use_server: Optional[bool]=False): self.initialize_states() + if self.env.config["goal"]: + self.init_goal() + nb_actions = self.env.config["nb_actions"] low = np.array([-1]*1) high = np.array([1]*1) @@ -108,10 +110,11 @@ def reset(self): """Reset simulation. """ self.initialize_states() - - self.env.reset() - self.env.config.update({'goalPos': self.env.goal}) + if self.env.config["goal"]: + self.init_goal() + + self.env.reset() if self.use_server: obs = start_scene(self.env.config, self.nb_actions) diff --git a/sofagym/envs/StemPendulum/StemPendulumScene.py b/sofagym/envs/StemPendulum/StemPendulumScene.py index bda7037..ac19a1e 100644 --- a/sofagym/envs/StemPendulum/StemPendulumScene.py +++ b/sofagym/envs/StemPendulum/StemPendulumScene.py @@ -26,20 +26,12 @@ from sofagym.utils import addRigidObject from StemPendulum import StemPendulum -from StemPendulumToolbox import rewardShaper, sceneModerator, applyAction, goalSetter +from StemPendulumToolbox import rewardShaper, sceneModerator, applyAction from Controller import ControllerStemPendulum -def add_goal_node(root, pos): - goal = root.addChild("Goal") - goal.addObject('VisualStyle', displayFlags="showCollisionModels") - goal_mo = goal.addObject('MechanicalObject', name='GoalMO', showObject=True, drawMode="1", showObjectScale=0.5, - showColor=[1, 0, 0, 0.5], position= pos) - return goal_mo - def createScene(rootNode, config = {"source": [0, 0, 30], "target": [0, 0, 0], - "goalPos": [7, 0, 20], "seed": None, "zFar":4000, "max_torque": 500, @@ -60,7 +52,6 @@ def createScene(rootNode, config = {"source": [0, 0, 30], stempendulum = StemPendulum(stempendulum_config = stempendulum_config) stempendulum.onEnd(rootNode) - rootNode.addObject(goalSetter(name="GoalSetter")) rootNode.addObject(sceneModerator(name="sceneModerator", stempendulum = stempendulum)) rootNode.addObject(rewardShaper(name="Reward", rootNode=rootNode)) rootNode.addObject(applyAction(name="applyAction", root= rootNode, stempendulum=stempendulum)) diff --git a/sofagym/envs/StemPendulum/StemPendulumToolbox.py b/sofagym/envs/StemPendulum/StemPendulumToolbox.py index 4ced153..96a1a89 100644 --- a/sofagym/envs/StemPendulum/StemPendulumToolbox.py +++ b/sofagym/envs/StemPendulum/StemPendulumToolbox.py @@ -88,7 +88,7 @@ def getReward(self): r = -float(dist)/(2*self.beam_len) return r - def update(self): + def update(self, goal=None): """Update function. This function is used as an initialization function. @@ -152,17 +152,6 @@ def getState(rootNode): return state -class goalSetter(Sofa.Core.Controller): - def __init__(self, *args, **kwargs): - Sofa.Core.Controller.__init__(self, *args, **kwargs) - - def update(self): - pass - - def set_mo_pos(self, goal): - pass - - def getReward(rootNode): r = rootNode.Reward.getReward() return False, r From 4a399fe108607c25bf1970bd7fb1084c4b8e8ec9 Mon Sep 17 00:00:00 2001 From: samuelmyoussef Date: Thu, 26 Sep 2024 01:47:47 +0200 Subject: [PATCH 24/85] class instantiation --- sofagym/envs/SimpleMaze/SimpleMazeEnv.py | 66 +++++++++++++----------- 1 file changed, 37 insertions(+), 29 deletions(-) diff --git a/sofagym/envs/SimpleMaze/SimpleMazeEnv.py b/sofagym/envs/SimpleMaze/SimpleMazeEnv.py index 9d2d461..de09688 100644 --- a/sofagym/envs/SimpleMaze/SimpleMazeEnv.py +++ b/sofagym/envs/SimpleMaze/SimpleMazeEnv.py @@ -8,15 +8,17 @@ __copyright__ = "(c) 2021, Robocath, CNRS, Inria" __date__ = "Mar 23 2021" -import os +import os, sys import numpy as np -from sofagym.AbstractEnv import AbstractEnv +from sofagym.AbstractEnv import AbstractEnv, ServerEnv from sofagym.rpc_server import start_scene from gym import spaces -class SimpleMazeEnv(AbstractEnv): +from typing import Optional + +class SimpleMazeEnv: """Sub-class of AbstractEnv, dedicated to the trunk scene. See the class AbstractEnv for arguments and methods. @@ -28,6 +30,7 @@ class SimpleMazeEnv(AbstractEnv): "deterministic": True, "source": [0, 200, 0], "target": [0, 0, 0], + "goal": True, "goalList": [301, 334, 317, 312], "goal_node": 334, "start_node": 269, @@ -42,41 +45,48 @@ class SimpleMazeEnv(AbstractEnv): "discrete": True, "seed": 0, "start_from_history": None, - "python_version": "python3.9", + "python_version": sys.version, "zFar": 5000, - "dt": 0.01 + "dt": 0.01, + "randomize_states": False, + "use_server": False } - def __init__(self, config=None): - super().__init__(config) + def __init__(self, config = None, root=None, use_server: Optional[bool]=False): + self.use_server = self.DEFAULT_CONFIG["use_server"] + self.env = ServerEnv(self.DEFAULT_CONFIG, config, root=root) if self.use_server else AbstractEnv(self.DEFAULT_CONFIG, config, root=root) + nb_actions = 4 - self.action_space = spaces.Discrete(nb_actions) + self.env.action_space = spaces.Discrete(nb_actions) self.nb_actions = str(nb_actions) dim_state = 13 low_coordinates = np.array([-1]*dim_state) high_coordinates = np.array([1]*dim_state) - self.observation_space = spaces.Box(low_coordinates, high_coordinates, - dtype='float32') + self.env.observation_space = spaces.Box(low_coordinates, high_coordinates, dtype=np.float32) + + if self.env.root is None and not self.use_server: + self.env.init_root() - def step(self, action): - return super().step(action) + # called when an attribute is not found: + def __getattr__(self, name): + # assume it is implemented by self.instance + return self.env.__getattribute__(name) def reset(self): """Reset simulation. - - Note: - ---- - We launch a client to create the scene. The scene of the program is - client_Env.py. - """ - super().reset() - - self.config.update({'goalPos': self.goal}) - obs = start_scene(self.config, self.nb_actions) - - return np.array(obs['observation']) + self.env.reset() + + self.env.config.update({'goalPos': self.env.goal}) + + if self.use_server: + obs = start_scene(self.env.config, self.nb_actions) + state = np.array(obs['observation'], dtype=np.float32) + else: + state = np.array(self.env._getState(self.env.root), dtype=np.float32) + + return state def get_available_actions(self): """Gives the actions available in the environment. @@ -90,15 +100,13 @@ def get_available_actions(self): list of the action available in the environment. """ if self.nb_actions == "4": - if not self.past_actions: + if not self.env.past_actions: return [0, 1, 2, 3] - last_action = self.past_actions[-1] + last_action = self.env.past_actions[-1] print(last_action) available_actions = [[0, 1, 2], [0, 1, 3], [0, 2, 3], [1, 2, 3]] return available_actions[last_action] - return list(range(int(self.nb_actions))) - - + return self.env.action_space From cb2eba3a0113622fbe14544e009f5c7641750e04 Mon Sep 17 00:00:00 2001 From: samuelmyoussef Date: Thu, 26 Sep 2024 01:49:54 +0200 Subject: [PATCH 25/85] add some default configs --- sofagym/envs/SimpleMaze/SimpleMazeEnv.py | 9 ++++++--- 1 file changed, 6 insertions(+), 3 deletions(-) diff --git a/sofagym/envs/SimpleMaze/SimpleMazeEnv.py b/sofagym/envs/SimpleMaze/SimpleMazeEnv.py index de09688..01ca248 100644 --- a/sofagym/envs/SimpleMaze/SimpleMazeEnv.py +++ b/sofagym/envs/SimpleMaze/SimpleMazeEnv.py @@ -26,6 +26,7 @@ class SimpleMazeEnv: # Setting a default configuration path = os.path.dirname(os.path.abspath(__file__)) metadata = {'render.modes': ['human', 'rgb_array']} + dim_state = 13 DEFAULT_CONFIG = {"scene": "SimpleMaze", "deterministic": True, "source": [0, 200, 0], @@ -34,7 +35,7 @@ class SimpleMazeEnv: "goalList": [301, 334, 317, 312], "goal_node": 334, "start_node": 269, - "scale_factor": 200, + "scale_factor": 10, "timer_limit": 50, "timeout": 30, "display_size": (1600, 800), @@ -48,6 +49,8 @@ class SimpleMazeEnv: "python_version": sys.version, "zFar": 5000, "dt": 0.01, + "nb_actions": 4, + "dim_state": dim_state, "randomize_states": False, "use_server": False } @@ -56,11 +59,11 @@ def __init__(self, config = None, root=None, use_server: Optional[bool]=False): self.use_server = self.DEFAULT_CONFIG["use_server"] self.env = ServerEnv(self.DEFAULT_CONFIG, config, root=root) if self.use_server else AbstractEnv(self.DEFAULT_CONFIG, config, root=root) - nb_actions = 4 + nb_actions = self.env.config["nb_actions"] self.env.action_space = spaces.Discrete(nb_actions) self.nb_actions = str(nb_actions) - dim_state = 13 + dim_state = self.env.config["dim_state"] low_coordinates = np.array([-1]*dim_state) high_coordinates = np.array([1]*dim_state) self.env.observation_space = spaces.Box(low_coordinates, high_coordinates, dtype=np.float32) From 96a40fb5150ca932685bfca96b95119b10d59be9 Mon Sep 17 00:00:00 2001 From: samuelmyoussef Date: Thu, 26 Sep 2024 01:51:35 +0200 Subject: [PATCH 26/85] initialize states --- sofagym/envs/SimpleMaze/SimpleMazeEnv.py | 26 ++++++++++++++++++++++++ 1 file changed, 26 insertions(+) diff --git a/sofagym/envs/SimpleMaze/SimpleMazeEnv.py b/sofagym/envs/SimpleMaze/SimpleMazeEnv.py index 01ca248..33fa65d 100644 --- a/sofagym/envs/SimpleMaze/SimpleMazeEnv.py +++ b/sofagym/envs/SimpleMaze/SimpleMazeEnv.py @@ -52,6 +52,7 @@ class SimpleMazeEnv: "nb_actions": 4, "dim_state": dim_state, "randomize_states": False, + "init_states": [0] * dim_state, "use_server": False } @@ -59,6 +60,8 @@ def __init__(self, config = None, root=None, use_server: Optional[bool]=False): self.use_server = self.DEFAULT_CONFIG["use_server"] self.env = ServerEnv(self.DEFAULT_CONFIG, config, root=root) if self.use_server else AbstractEnv(self.DEFAULT_CONFIG, config, root=root) + self.initialize_states() + nb_actions = self.env.config["nb_actions"] self.env.action_space = spaces.Discrete(nb_actions) self.nb_actions = str(nb_actions) @@ -76,9 +79,32 @@ def __getattr__(self, name): # assume it is implemented by self.instance return self.env.__getattribute__(name) + def initialize_states(self): + if self.env.config["randomize_states"]: + self.init_states = self.randomize_init_states() + self.env.config.update({'init_states': list(self.init_states)}) + else: + self.init_states = self.env.config["init_states"] + + def randomize_init_states(self): + """Randomize initial states. + + Returns: + ------- + init_states: list + List of random initial states for the environment. + + Note: + ---- + This method should be implemented according to needed random initialization. + """ + return self.env.config["init_states"] + def reset(self): """Reset simulation. """ + self.initialize_states() + self.env.reset() self.env.config.update({'goalPos': self.env.goal}) From 258b2bd9a27913563d76204a2719f4d6f0fc1859 Mon Sep 17 00:00:00 2001 From: samuelmyoussef Date: Thu, 26 Sep 2024 01:56:44 +0200 Subject: [PATCH 27/85] goal init and update --- sofagym/envs/SimpleMaze/SimpleMazeEnv.py | 18 +++++++++++++++--- sofagym/envs/SimpleMaze/SimpleMazeScene.py | 4 ++-- sofagym/envs/SimpleMaze/SimpleMazeToolbox.py | 10 ++++------ 3 files changed, 21 insertions(+), 11 deletions(-) diff --git a/sofagym/envs/SimpleMaze/SimpleMazeEnv.py b/sofagym/envs/SimpleMaze/SimpleMazeEnv.py index 33fa65d..3613c76 100644 --- a/sofagym/envs/SimpleMaze/SimpleMazeEnv.py +++ b/sofagym/envs/SimpleMaze/SimpleMazeEnv.py @@ -62,6 +62,9 @@ def __init__(self, config = None, root=None, use_server: Optional[bool]=False): self.initialize_states() + if self.env.config["goal"]: + self.init_goal() + nb_actions = self.env.config["nb_actions"] self.env.action_space = spaces.Discrete(nb_actions) self.nb_actions = str(nb_actions) @@ -99,15 +102,24 @@ def randomize_init_states(self): This method should be implemented according to needed random initialization. """ return self.env.config["init_states"] + + def init_goal(self): + # Set a new random goal from the list + goalList = self.env.config["goalList"] + id_goal = self.env.np_random.choice(range(len(goalList))) + self.env.config.update({'goal_node': id_goal}) + self.env.goal = goalList[id_goal] + self.env.config.update({'goalPos': self.env.goal}) def reset(self): """Reset simulation. """ self.initialize_states() - - self.env.reset() - self.env.config.update({'goalPos': self.env.goal}) + if self.env.config["goal"]: + self.init_goal() + + self.env.reset() if self.use_server: obs = start_scene(self.env.config, self.nb_actions) diff --git a/sofagym/envs/SimpleMaze/SimpleMazeScene.py b/sofagym/envs/SimpleMaze/SimpleMazeScene.py index 537c354..50a2b97 100644 --- a/sofagym/envs/SimpleMaze/SimpleMazeScene.py +++ b/sofagym/envs/SimpleMaze/SimpleMazeScene.py @@ -37,7 +37,7 @@ def add_goal_node(root): def createScene(root, config={"source": [0, 1000, 0], "target": [0, 0, 0], "goal_node": 0, - "goalPos": [0.0, 0.0, 0.0], + "goalPos": 0, "dt": 0.01}, mode='simu_and_visu'): # Chose the mode: visualization or computations (or both) @@ -92,7 +92,7 @@ def createScene(root, config={"source": [0, 1000, 0], goal = add_goal_node(root) - root.addObject(rewardShaper(name="Reward", rootNode=root, goal_node=config['goalList'][config['goal_node']], + root.addObject(rewardShaper(name="Reward", rootNode=root, goal_node=config['goalPos'], path_mesh=p_mesh, path_mo=p_mo, ball_mo=ball_mo)) root.addObject(goalSetter(name="GoalSetter", rootNode=root, goal=goal, goalPos=config['goalPos'])) diff --git a/sofagym/envs/SimpleMaze/SimpleMazeToolbox.py b/sofagym/envs/SimpleMaze/SimpleMazeToolbox.py index 4b0d233..8f4ee40 100644 --- a/sofagym/envs/SimpleMaze/SimpleMazeToolbox.py +++ b/sofagym/envs/SimpleMaze/SimpleMazeToolbox.py @@ -156,7 +156,7 @@ def getReward(self): else: return 0.0, False - def update(self): + def update(self, goal=None): """Update function. This function is used as an initialization function. @@ -170,7 +170,7 @@ def update(self): None. """ - + self.goal_node = goal edges = [] with self.path_mesh.edges.writeable() as Topoedges: for edge in Topoedges: @@ -233,7 +233,7 @@ def __init__(self, *args, **kwargs): if kwargs["goalPos"]: self.goalPos = kwargs["goalPos"] - def update(self): + def update(self, goal): """Set the position of the goal. This function is used as an initialization function. @@ -247,6 +247,7 @@ def update(self): None. """ + self.goalPos = goal new_position = self.rootNode.model.Maze.Path.dofs.position.value[self.goalPos][:3] with self.goal.GoalMO.position.writeable() as position: position[0] = new_position @@ -447,9 +448,6 @@ def getPos(root): _: list The position(s) of the object(s) of the scene. """ - - root.GoalSetter.update() - maze = root.model.Maze.maze_mesh_mo.position.value.tolist() spheres = root.Sphere.sphere_mo.position.value.tolist() From cefa2335ada3afca04be88b9a2eea7d42d5b1db4 Mon Sep 17 00:00:00 2001 From: samuelmyoussef Date: Thu, 26 Sep 2024 00:59:14 +0200 Subject: [PATCH 28/85] class instantiation --- sofagym/envs/Maze/MazeEnv.py | 57 +++++++++++++++++++++--------------- 1 file changed, 34 insertions(+), 23 deletions(-) diff --git a/sofagym/envs/Maze/MazeEnv.py b/sofagym/envs/Maze/MazeEnv.py index 0c804a6..667052e 100644 --- a/sofagym/envs/Maze/MazeEnv.py +++ b/sofagym/envs/Maze/MazeEnv.py @@ -8,15 +8,17 @@ __copyright__ = "(c) 2020, Inria" __date__ = "Oct 7 2020" -import os +import os, sys import numpy as np -from sofagym.AbstractEnv import AbstractEnv +from sofagym.AbstractEnv import AbstractEnv, ServerEnv from sofagym.rpc_server import start_scene from gym import spaces -class MazeEnv(AbstractEnv): +from typing import Optional + +class MazeEnv: """Sub-class of AbstractEnv, dedicated to the trunk scene. See the class AbstractEnv for arguments and methods. @@ -28,6 +30,7 @@ class MazeEnv(AbstractEnv): "deterministic": True, "source": [-82.0819, 186.518, 135.963], "target": [-2.09447, 5.75347, -4.34572], + "goal": True, "goalList": [334, 317, 312, 301], "goal_node": 270, "start_node": 269, @@ -43,41 +46,49 @@ class MazeEnv(AbstractEnv): "discrete": True, "seed": 0, "start_from_history": None, - "python_version": "python3.8", + "python_version": sys.version, "zFar": 1000, "dt": 0.01, "time_before_start": 20, + "randomize_states": False, + "use_server": False } - def __init__(self, config=None): - super().__init__(config) + def __init__(self, config = None, root=None, use_server: Optional[bool]=False): + self.use_server = self.DEFAULT_CONFIG["use_server"] + self.env = ServerEnv(self.DEFAULT_CONFIG, config, root=root) if self.use_server else AbstractEnv(self.DEFAULT_CONFIG, config, root=root) + nb_actions = 6 - self.action_space = spaces.Discrete(nb_actions) + self.env.action_space = spaces.Discrete(nb_actions) self.nb_actions = str(nb_actions) dim_state = 9 low_coordinates = np.array([-1]*dim_state) high_coordinates = np.array([1]*dim_state) - self.observation_space = spaces.Box(low_coordinates, high_coordinates, dtype='float32') + self.env.observation_space = spaces.Box(low_coordinates, high_coordinates, dtype=np.float32) - def step(self, action): - return super().step(action) + if self.env.root is None and not self.use_server: + self.env.init_root() + + # called when an attribute is not found: + def __getattr__(self, name): + # assume it is implemented by self.instance + return self.env.__getattribute__(name) def reset(self): """Reset simulation. - - Note: - ---- - We launch a client to create the scene. The scene of the program is - client_Env.py. - """ - super().reset() - - self.config.update({'goalPos': self.goal}) - obs = start_scene(self.config, self.nb_actions) - - return np.array(obs['observation']) + self.env.reset() + + self.env.config.update({'goalPos': self.env.goal}) + + if self.use_server: + obs = start_scene(self.env.config, self.nb_actions) + state = np.array(obs['observation'], dtype=np.float32) + else: + state = np.array(self.env._getState(self.env.root), dtype=np.float32) + + return state def get_available_actions(self): """Gives the actions available in the environment. @@ -90,4 +101,4 @@ def get_available_actions(self): ------- list of the action available in the environment. """ - return list(range(int(self.nb_actions))) \ No newline at end of file + return self.env.action_space From 896f74e022244b04f553a3d606de25963fddbb11 Mon Sep 17 00:00:00 2001 From: samuelmyoussef Date: Thu, 26 Sep 2024 01:01:11 +0200 Subject: [PATCH 29/85] add some default configs --- sofagym/envs/Maze/MazeEnv.py | 7 +++++-- 1 file changed, 5 insertions(+), 2 deletions(-) diff --git a/sofagym/envs/Maze/MazeEnv.py b/sofagym/envs/Maze/MazeEnv.py index 667052e..3efa874 100644 --- a/sofagym/envs/Maze/MazeEnv.py +++ b/sofagym/envs/Maze/MazeEnv.py @@ -26,6 +26,7 @@ class MazeEnv: # Setting a default configuration path = os.path.dirname(os.path.abspath(__file__)) metadata = {'render.modes': ['human', 'rgb_array']} + dim_state = 9 DEFAULT_CONFIG = {"scene": "Maze", "deterministic": True, "source": [-82.0819, 186.518, 135.963], @@ -50,6 +51,8 @@ class MazeEnv: "zFar": 1000, "dt": 0.01, "time_before_start": 20, + "nb_actions": 6, + "dim_state": dim_state, "randomize_states": False, "use_server": False } @@ -58,11 +61,11 @@ def __init__(self, config = None, root=None, use_server: Optional[bool]=False): self.use_server = self.DEFAULT_CONFIG["use_server"] self.env = ServerEnv(self.DEFAULT_CONFIG, config, root=root) if self.use_server else AbstractEnv(self.DEFAULT_CONFIG, config, root=root) - nb_actions = 6 + nb_actions = self.env.config["nb_actions"] self.env.action_space = spaces.Discrete(nb_actions) self.nb_actions = str(nb_actions) - dim_state = 9 + dim_state = self.env.config["dim_state"] low_coordinates = np.array([-1]*dim_state) high_coordinates = np.array([1]*dim_state) self.env.observation_space = spaces.Box(low_coordinates, high_coordinates, dtype=np.float32) From b186e1241f2b701bb7dfc77d1b5fc3adabb1db69 Mon Sep 17 00:00:00 2001 From: samuelmyoussef Date: Thu, 26 Sep 2024 01:02:29 +0200 Subject: [PATCH 30/85] initialize states --- sofagym/envs/Maze/MazeEnv.py | 26 ++++++++++++++++++++++++++ 1 file changed, 26 insertions(+) diff --git a/sofagym/envs/Maze/MazeEnv.py b/sofagym/envs/Maze/MazeEnv.py index 3efa874..3cb8ed0 100644 --- a/sofagym/envs/Maze/MazeEnv.py +++ b/sofagym/envs/Maze/MazeEnv.py @@ -54,6 +54,7 @@ class MazeEnv: "nb_actions": 6, "dim_state": dim_state, "randomize_states": False, + "init_states": [0] * dim_state, "use_server": False } @@ -61,6 +62,8 @@ def __init__(self, config = None, root=None, use_server: Optional[bool]=False): self.use_server = self.DEFAULT_CONFIG["use_server"] self.env = ServerEnv(self.DEFAULT_CONFIG, config, root=root) if self.use_server else AbstractEnv(self.DEFAULT_CONFIG, config, root=root) + self.initialize_states() + nb_actions = self.env.config["nb_actions"] self.env.action_space = spaces.Discrete(nb_actions) self.nb_actions = str(nb_actions) @@ -78,9 +81,32 @@ def __getattr__(self, name): # assume it is implemented by self.instance return self.env.__getattribute__(name) + def initialize_states(self): + if self.env.config["randomize_states"]: + self.init_states = self.randomize_init_states() + self.env.config.update({'init_states': list(self.init_states)}) + else: + self.init_states = self.env.config["init_states"] + + def randomize_init_states(self): + """Randomize initial states. + + Returns: + ------- + init_states: list + List of random initial states for the environment. + + Note: + ---- + This method should be implemented according to needed random initialization. + """ + return self.env.config["init_states"] + def reset(self): """Reset simulation. """ + self.initialize_states() + self.env.reset() self.env.config.update({'goalPos': self.env.goal}) From a945bd29b9a7780d436c1422dcf78885d69c373c Mon Sep 17 00:00:00 2001 From: samuelmyoussef Date: Thu, 26 Sep 2024 01:10:41 +0200 Subject: [PATCH 31/85] goal init and update --- sofagym/envs/Maze/MazeEnv.py | 18 +++++++++++++++--- sofagym/envs/Maze/MazeScene.py | 2 +- sofagym/envs/Maze/MazeToolbox.py | 7 ++++--- 3 files changed, 20 insertions(+), 7 deletions(-) diff --git a/sofagym/envs/Maze/MazeEnv.py b/sofagym/envs/Maze/MazeEnv.py index 3cb8ed0..dcb7bb6 100644 --- a/sofagym/envs/Maze/MazeEnv.py +++ b/sofagym/envs/Maze/MazeEnv.py @@ -64,6 +64,9 @@ def __init__(self, config = None, root=None, use_server: Optional[bool]=False): self.initialize_states() + if self.env.config["goal"]: + self.init_goal() + nb_actions = self.env.config["nb_actions"] self.env.action_space = spaces.Discrete(nb_actions) self.nb_actions = str(nb_actions) @@ -102,14 +105,23 @@ def randomize_init_states(self): """ return self.env.config["init_states"] + def init_goal(self): + # Set a new random goal from the list + goalList = self.env.config["goalList"] + id_goal = self.env.np_random.choice(range(len(goalList))) + self.env.config.update({'goal_node': id_goal}) + self.env.goal = goalList[id_goal] + self.env.config.update({'goalPos': self.env.goal}) + def reset(self): """Reset simulation. """ self.initialize_states() - - self.env.reset() - self.env.config.update({'goalPos': self.env.goal}) + if self.env.config["goal"]: + self.init_goal() + + self.env.reset() if self.use_server: obs = start_scene(self.env.config, self.nb_actions) diff --git a/sofagym/envs/Maze/MazeScene.py b/sofagym/envs/Maze/MazeScene.py index d26b20e..55bb1d6 100644 --- a/sofagym/envs/Maze/MazeScene.py +++ b/sofagym/envs/Maze/MazeScene.py @@ -52,7 +52,7 @@ def createScene(rootNode, config={"source": [0, 300, 0], "target": [0, 0, 0], "goalList": [0, 0, 0], "goal_node": 0, - "goalPos": [0.0, 0.0, 0.0], + "goalPos": 0, "dt": 0.01}, mode='simu_and_visu'): pluginList = ["ArticulatedSystemPlugin", diff --git a/sofagym/envs/Maze/MazeToolbox.py b/sofagym/envs/Maze/MazeToolbox.py index e4e7656..bfa0ac3 100644 --- a/sofagym/envs/Maze/MazeToolbox.py +++ b/sofagym/envs/Maze/MazeToolbox.py @@ -110,7 +110,7 @@ def getReward(self): else: return new_ratio, None - def update(self): + def update(self, goal=None): """Update function. This function is used as an initialization function. @@ -124,7 +124,7 @@ def update(self): None. """ - + self.goal_node = goal edges = [] with self.path_mesh.edges.writeable() as Topoedges: for edge in Topoedges: @@ -188,7 +188,7 @@ def __init__(self, *args, **kwargs): if kwargs["goalPos"]: self.goalPos = kwargs["goalPos"] - def update(self): + def update(self, goal): """Set the position of the goal. This function is used as an initialization function. @@ -202,6 +202,7 @@ def update(self): None. """ + self.goalPos = goal new_position = self.rootNode.Modelling.Tripod.RigidifiedStructure.FreeCenter.Maze.Path.dofs.position.value[self.goalPos][:3] with self.goal.GoalMO.position.writeable() as position: position[0] = new_position From d75f9d9421d850aa4390c6e54afdc2ce271297c7 Mon Sep 17 00:00:00 2001 From: samuelmyoussef Date: Thu, 26 Sep 2024 01:42:32 +0200 Subject: [PATCH 32/85] add AnimationManager --- sofagym/envs/Maze/MazeScene.py | 3 +++ 1 file changed, 3 insertions(+) diff --git a/sofagym/envs/Maze/MazeScene.py b/sofagym/envs/Maze/MazeScene.py index 55bb1d6..ca44c8b 100644 --- a/sofagym/envs/Maze/MazeScene.py +++ b/sofagym/envs/Maze/MazeScene.py @@ -22,6 +22,7 @@ from MazeToolbox import goalSetter, rewardShaper from Sphere import Sphere from splib3.animation import animate +from splib3.animation import AnimationManagerController from splib3.numerics import RigidDof from stlib3.scene import ContactHeader, Scene from tripod import Tripod @@ -115,6 +116,8 @@ def createScene(rootNode, config={"source": [0, 300, 0], scene.Settings.mouseButton.stiffness = 10 scene.Simulation.TimeIntegrationSchema.rayleighStiffness = 0.05 ContactHeader(rootNode, alarmDistance=0.5, contactDistance=0.2, frictionCoef=0.2) + + rootNode.addObject(AnimationManagerController(rootNode, name="AnimationManager")) # Visu if visu: From d0984197834f6ea9a04dcfe2ccb544f838627f65 Mon Sep 17 00:00:00 2001 From: samuelmyoussef Date: Thu, 26 Sep 2024 00:34:33 +0200 Subject: [PATCH 33/85] class instantiation --- sofagym/envs/Gripper/GripperEnv.py | 60 ++++++++++++++++-------------- 1 file changed, 33 insertions(+), 27 deletions(-) diff --git a/sofagym/envs/Gripper/GripperEnv.py b/sofagym/envs/Gripper/GripperEnv.py index 10e7c41..cbae99b 100644 --- a/sofagym/envs/Gripper/GripperEnv.py +++ b/sofagym/envs/Gripper/GripperEnv.py @@ -8,15 +8,17 @@ __copyright__ = "(c) 2020, Inria" __date__ = "Oct 7 2020" -import os +import os, sys import numpy as np -from sofagym.AbstractEnv import AbstractEnv +from sofagym.AbstractEnv import AbstractEnv, ServerEnv from sofagym.rpc_server import start_scene from gym import spaces -class GripperEnv(AbstractEnv): +from typing import Optional + +class GripperEnv: """Sub-class of AbstractEnv, dedicated to the gripper scene. See the class AbstractEnv for arguments and methods. @@ -28,6 +30,7 @@ class GripperEnv(AbstractEnv): "deterministic": True, "source": [0, -80, 350], "target": [0, -80, 0], + "goal": True, "goalList": [[0, 10, 0], [0, 20, 0], [0, 30, 0]], "start_node": None, "scale_factor": 5, @@ -42,42 +45,45 @@ class GripperEnv(AbstractEnv): "discrete": True, "seed": None, "start_from_history": None, - "python_version": "python3", - "dt": 0.01 + "python_version": sys.version, + "dt": 0.01, + "randomize_states": False, + "use_server": False } - def __init__(self, config=None): - super().__init__(config) + def __init__(self, config = None, root=None, use_server: Optional[bool]=False): + self.use_server = self.DEFAULT_CONFIG["use_server"] + self.env = ServerEnv(self.DEFAULT_CONFIG, config, root=root) if self.use_server else AbstractEnv(self.DEFAULT_CONFIG, config, root=root) + nb_actions = 8 - self.action_space = spaces.Discrete(nb_actions) + self.env.action_space = spaces.Discrete(nb_actions) self.nb_actions = str(nb_actions) dim_state = 31 low_coordinates = np.array([-1]*dim_state) high_coordinates = np.array([1]*dim_state) - self.observation_space = spaces.Box(low_coordinates, high_coordinates, - dtype='float32') - - + self.env.observation_space = spaces.Box(low_coordinates, high_coordinates, dtype=np.float32) - def step(self, action): - return super().step(action) + if self.env.root is None and not self.use_server: + self.env.init_root() + + # called when an attribute is not found: + def __getattr__(self, name): + # assume it is implemented by self.instance + return self.env.__getattribute__(name) def reset(self): """Reset simulation. - - Note: - ---- - We launch a client to create the scene. The scene of the program is - client_Env.py. - """ - super().reset() - - self.config.update({'goalPos': self.goal}) - obs = start_scene(self.config, self.nb_actions) - - return (np.array(obs['observation'])) + self.env.reset() + + if self.use_server: + obs = start_scene(self.env.config, self.nb_actions) + state = np.array(obs['observation'], dtype=np.float32) + else: + state = np.array(self.env._getState(self.env.root), dtype=np.float32) + + return state def get_available_actions(self): """Gives the actions available in the environment. @@ -90,4 +96,4 @@ def get_available_actions(self): ------- list of the action available in the environment. """ - return list(range(int(self.nb_actions))) + return self.env.action_space From afd0ec5172c556892cfb79c521ab6575464f442f Mon Sep 17 00:00:00 2001 From: samuelmyoussef Date: Thu, 26 Sep 2024 00:35:42 +0200 Subject: [PATCH 34/85] add some default configs --- sofagym/envs/Gripper/GripperEnv.py | 7 +++++-- 1 file changed, 5 insertions(+), 2 deletions(-) diff --git a/sofagym/envs/Gripper/GripperEnv.py b/sofagym/envs/Gripper/GripperEnv.py index cbae99b..0c67438 100644 --- a/sofagym/envs/Gripper/GripperEnv.py +++ b/sofagym/envs/Gripper/GripperEnv.py @@ -26,6 +26,7 @@ class GripperEnv: # Setting a default configuration path = os.path.dirname(os.path.abspath(__file__)) metadata = {'render.modes': ['human', 'rgb_array']} + dim_state = 31 DEFAULT_CONFIG = {"scene": "Gripper", "deterministic": True, "source": [0, -80, 350], @@ -47,6 +48,8 @@ class GripperEnv: "start_from_history": None, "python_version": sys.version, "dt": 0.01, + "nb_actions": 8, + "dim_state": dim_state, "randomize_states": False, "use_server": False } @@ -55,11 +58,11 @@ def __init__(self, config = None, root=None, use_server: Optional[bool]=False): self.use_server = self.DEFAULT_CONFIG["use_server"] self.env = ServerEnv(self.DEFAULT_CONFIG, config, root=root) if self.use_server else AbstractEnv(self.DEFAULT_CONFIG, config, root=root) - nb_actions = 8 + nb_actions = self.env.config["nb_actions"] self.env.action_space = spaces.Discrete(nb_actions) self.nb_actions = str(nb_actions) - dim_state = 31 + dim_state = self.env.config["dim_state"] low_coordinates = np.array([-1]*dim_state) high_coordinates = np.array([1]*dim_state) self.env.observation_space = spaces.Box(low_coordinates, high_coordinates, dtype=np.float32) From f498f98de2c87ebd367f924bbdf47f66124cc625 Mon Sep 17 00:00:00 2001 From: samuelmyoussef Date: Thu, 26 Sep 2024 00:38:15 +0200 Subject: [PATCH 35/85] initialize states --- sofagym/envs/Gripper/GripperEnv.py | 26 ++++++++++++++++++++++++++ 1 file changed, 26 insertions(+) diff --git a/sofagym/envs/Gripper/GripperEnv.py b/sofagym/envs/Gripper/GripperEnv.py index 0c67438..b040e21 100644 --- a/sofagym/envs/Gripper/GripperEnv.py +++ b/sofagym/envs/Gripper/GripperEnv.py @@ -51,12 +51,15 @@ class GripperEnv: "nb_actions": 8, "dim_state": dim_state, "randomize_states": False, + "init_states": [0] * dim_state, "use_server": False } def __init__(self, config = None, root=None, use_server: Optional[bool]=False): self.use_server = self.DEFAULT_CONFIG["use_server"] self.env = ServerEnv(self.DEFAULT_CONFIG, config, root=root) if self.use_server else AbstractEnv(self.DEFAULT_CONFIG, config, root=root) + + self.initialize_states() nb_actions = self.env.config["nb_actions"] self.env.action_space = spaces.Discrete(nb_actions) @@ -74,10 +77,33 @@ def __init__(self, config = None, root=None, use_server: Optional[bool]=False): def __getattr__(self, name): # assume it is implemented by self.instance return self.env.__getattribute__(name) + + def initialize_states(self): + if self.env.config["randomize_states"]: + self.init_states = self.randomize_init_states() + self.env.config.update({'init_states': list(self.init_states)}) + else: + self.init_states = self.env.config["init_states"] + + def randomize_init_states(self): + """Randomize initial states. + + Returns: + ------- + init_states: list + List of random initial states for the environment. + + Note: + ---- + This method should be implemented according to needed random initialization. + """ + return self.env.config["init_states"] def reset(self): """Reset simulation. """ + self.initialize_states() + self.env.reset() if self.use_server: From 00f32909ba36d541fcc82b8cd7cb9d0ae263dc95 Mon Sep 17 00:00:00 2001 From: samuelmyoussef Date: Thu, 26 Sep 2024 00:43:02 +0200 Subject: [PATCH 36/85] goal init and update --- sofagym/envs/Gripper/GripperEnv.py | 16 +++++++++++++++- sofagym/envs/Gripper/GripperTools.py | 6 ++++-- 2 files changed, 19 insertions(+), 3 deletions(-) diff --git a/sofagym/envs/Gripper/GripperEnv.py b/sofagym/envs/Gripper/GripperEnv.py index b040e21..35e0d56 100644 --- a/sofagym/envs/Gripper/GripperEnv.py +++ b/sofagym/envs/Gripper/GripperEnv.py @@ -61,6 +61,9 @@ def __init__(self, config = None, root=None, use_server: Optional[bool]=False): self.initialize_states() + if self.env.config["goal"]: + self.init_goal() + nb_actions = self.env.config["nb_actions"] self.env.action_space = spaces.Discrete(nb_actions) self.nb_actions = str(nb_actions) @@ -99,11 +102,22 @@ def randomize_init_states(self): """ return self.env.config["init_states"] + def init_goal(self): + # Set a new random goal from the list + goalList = self.env.config["goalList"] + id_goal = self.env.np_random.choice(range(len(goalList))) + self.env.config.update({'goal_node': id_goal}) + self.env.goal = goalList[id_goal] + self.env.config.update({'goalPos': self.env.goal}) + def reset(self): """Reset simulation. """ self.initialize_states() - + + if self.env.config["goal"]: + self.init_goal() + self.env.reset() if self.use_server: diff --git a/sofagym/envs/Gripper/GripperTools.py b/sofagym/envs/Gripper/GripperTools.py index a1a6aa9..fec68e5 100644 --- a/sofagym/envs/Gripper/GripperTools.py +++ b/sofagym/envs/Gripper/GripperTools.py @@ -89,7 +89,7 @@ def getReward(self): self.cost = current_cost return round(reward, 3)/20, self.cost - def update(self): + def update(self, goal=None): """Compute the distance between object and goal. This function is used as an initialization function. @@ -103,6 +103,7 @@ def update(self): None. """ + self.goal_pos = goal self.cost = abs(self.effMO.position[0][1] - self.goal_pos[1]) @@ -145,7 +146,7 @@ def __init__(self, *args, **kwargs): if kwargs["goalPos"]: self.goalPos = kwargs["goalPos"] - def update(self): + def update(self, goal): """Set the position of the goal. This function is used as an initialization function. @@ -159,6 +160,7 @@ def update(self): None. """ + self.goalPos = goal with self.goalMO.position.writeable() as position: position += self.goalPos From 24ef8cb5b8839a156f8cb31ecd12937d59a227c7 Mon Sep 17 00:00:00 2001 From: samuelmyoussef Date: Thu, 26 Sep 2024 03:24:55 +0200 Subject: [PATCH 37/85] class instantiation --- sofagym/envs/Diamond/DiamondEnv.py | 83 ++++++++++++++++++------------ 1 file changed, 49 insertions(+), 34 deletions(-) diff --git a/sofagym/envs/Diamond/DiamondEnv.py b/sofagym/envs/Diamond/DiamondEnv.py index c50adc9..7174edd 100644 --- a/sofagym/envs/Diamond/DiamondEnv.py +++ b/sofagym/envs/Diamond/DiamondEnv.py @@ -8,9 +8,9 @@ __copyright__ = "(c) 2021, Robocath, CNRS, Inria" __date__ = "Dec 01 2021" -import os +import os, sys -from sofagym.AbstractEnv import AbstractEnv +from sofagym.AbstractEnv import AbstractEnv, ServerEnv from sofagym.rpc_server import start_scene from sofagym.viewer import LegacyViewer from sofagym.envs.Diamond.DiamondToolbox import startCmd @@ -19,7 +19,9 @@ import numpy as np -class DiamondRobotEnv(AbstractEnv): +from typing import Optional + +class DiamondRobotEnv: """Sub-class of AbstractEnv, dedicated to the trunk scene. See the class AbstractEnv for arguments and methods. @@ -31,6 +33,7 @@ class DiamondRobotEnv(AbstractEnv): "deterministic": True, "source": [-288, -81, 147], "target": [4, -6, 52], + "goal": True, "goalList": [[30.0, 0.0, 150.0], [-30.0, 0.0, 150.0], [0.0, 30.0, 150.0], [0.0, -30.0, 150.0]], "scale_factor": 5, "timer_limit": 50, @@ -43,49 +46,59 @@ class DiamondRobotEnv(AbstractEnv): "discrete": True, "seed": 0, "start_from_history": None, - "python_version": "python3.8", + "python_version": sys.version, "zFar": 5000, - "dt": 0.01 + "dt": 0.01, + "randomize_states": False, + "use_server": False } - def __init__(self, config=None): - super().__init__(config) + def __init__(self, config = None, root=None, use_server: Optional[bool]=False): + self.use_server = self.DEFAULT_CONFIG["use_server"] + self.env = ServerEnv(self.DEFAULT_CONFIG, config, root=root) if self.use_server else AbstractEnv(self.DEFAULT_CONFIG, config, root=root) + nb_actions = 8 - self.action_space = spaces.Discrete(nb_actions) + self.env.action_space = spaces.Discrete(nb_actions) self.nb_actions = str(nb_actions) dim_state = (6, 3) low_coordinates = np.ones(shape=dim_state)*-1 high_coordinates = np.ones(shape=dim_state) - self.observation_space = spaces.Box(low_coordinates, high_coordinates, - dtype='float32') + self.env.observation_space = spaces.Box(low_coordinates, high_coordinates, dtype=np.float32) + + if self.env.root is None and not self.use_server: + self.env.init_root() + + # called when an attribute is not found: + def __getattr__(self, name): + # assume it is implemented by self.instance + return self.env.__getattribute__(name) def step(self, action): - if self.viewer: - self.viewer.step(action) + if self.use_server: + if self.env.viewer: + self.env.viewer.step(action) - return super().step(action) + return self.env.step(action) def reset(self): """Reset simulation. - - Note: - ---- - We launch a client to create the scene. The scene of the program is - client_Env.py. - """ - super().reset() + self.env.reset() - self.goal = [-30 + 60 * np.random.random(), -30 + 60 * np.random.random(), 125 + 20 * np.random.random()] + self.env.goal = [-30 + 60 * self.env.np_random.random(), -30 + 60 * self.env.np_random.random(), 125 + 20 * self.env.np_random.random()] - self.config.update({'goalPos': self.goal}) - obs = start_scene(self.config, self.nb_actions) - if self.viewer: - self.viewer.reset() - - return np.array(obs['observation']) + self.env.config.update({'goalPos': self.env.goal}) + if self.use_server: + obs = start_scene(self.env.config, self.nb_actions) + if self.env.viewer: + self.env.viewer.reset() + state = np.array(obs['observation'], dtype=np.float32) + else: + state = np.array(self.env._getState(self.env.root), dtype=np.float32) + + return state def render(self, mode='rgb_array'): """See the current state of the environment. @@ -102,13 +115,15 @@ def render(self, mode='rgb_array'): ------- None. """ - if not self.viewer: - display_size = self.config["display_size"] # Sim display - self.viewer = LegacyViewer(self, display_size, startCmd=startCmd) + if self.use_server: + if not self.env.viewer: + display_size = self.env.config["display_size"] # Sim display + self.env.viewer = LegacyViewer(self, display_size, startCmd=startCmd) - # Use the viewer to display the environment. - self.viewer.render() - + # Use the viewer to display the environment. + self.env.viewer.render() + else: + self.env.render(mode) def get_available_actions(self): """Gives the actions available in the environment. @@ -121,4 +136,4 @@ def get_available_actions(self): ------- list of the action available in the environment. """ - return list(range(int(self.nb_actions))) + return self.env.action_space From 05e21894ca896cddba433a212565beb614f9f1f8 Mon Sep 17 00:00:00 2001 From: samuelmyoussef Date: Thu, 26 Sep 2024 03:26:26 +0200 Subject: [PATCH 38/85] add some default configs --- sofagym/envs/Diamond/DiamondEnv.py | 7 +++++-- 1 file changed, 5 insertions(+), 2 deletions(-) diff --git a/sofagym/envs/Diamond/DiamondEnv.py b/sofagym/envs/Diamond/DiamondEnv.py index 7174edd..9c8c4c9 100644 --- a/sofagym/envs/Diamond/DiamondEnv.py +++ b/sofagym/envs/Diamond/DiamondEnv.py @@ -29,6 +29,7 @@ class DiamondRobotEnv: # Setting a default configuration path = os.path.dirname(os.path.abspath(__file__)) metadata = {'render.modes': ['human', 'rgb_array']} + dim_state = (6, 3) DEFAULT_CONFIG = {"scene": "Diamond", "deterministic": True, "source": [-288, -81, 147], @@ -49,6 +50,8 @@ class DiamondRobotEnv: "python_version": sys.version, "zFar": 5000, "dt": 0.01, + "nb_actions": 8, + "dim_state": dim_state, "randomize_states": False, "use_server": False } @@ -57,11 +60,11 @@ def __init__(self, config = None, root=None, use_server: Optional[bool]=False): self.use_server = self.DEFAULT_CONFIG["use_server"] self.env = ServerEnv(self.DEFAULT_CONFIG, config, root=root) if self.use_server else AbstractEnv(self.DEFAULT_CONFIG, config, root=root) - nb_actions = 8 + nb_actions = self.env.config["nb_actions"] self.env.action_space = spaces.Discrete(nb_actions) self.nb_actions = str(nb_actions) - dim_state = (6, 3) + dim_state = self.env.config["dim_state"] low_coordinates = np.ones(shape=dim_state)*-1 high_coordinates = np.ones(shape=dim_state) self.env.observation_space = spaces.Box(low_coordinates, high_coordinates, dtype=np.float32) From 1cff4312aa33df6297186277f9d7ff08e035ba5b Mon Sep 17 00:00:00 2001 From: samuelmyoussef Date: Thu, 26 Sep 2024 03:28:01 +0200 Subject: [PATCH 39/85] initialize states --- sofagym/envs/Diamond/DiamondEnv.py | 26 ++++++++++++++++++++++++++ 1 file changed, 26 insertions(+) diff --git a/sofagym/envs/Diamond/DiamondEnv.py b/sofagym/envs/Diamond/DiamondEnv.py index 9c8c4c9..c248a2e 100644 --- a/sofagym/envs/Diamond/DiamondEnv.py +++ b/sofagym/envs/Diamond/DiamondEnv.py @@ -53,6 +53,7 @@ class DiamondRobotEnv: "nb_actions": 8, "dim_state": dim_state, "randomize_states": False, + "init_states": 0, "use_server": False } @@ -60,6 +61,8 @@ def __init__(self, config = None, root=None, use_server: Optional[bool]=False): self.use_server = self.DEFAULT_CONFIG["use_server"] self.env = ServerEnv(self.DEFAULT_CONFIG, config, root=root) if self.use_server else AbstractEnv(self.DEFAULT_CONFIG, config, root=root) + self.initialize_states() + nb_actions = self.env.config["nb_actions"] self.env.action_space = spaces.Discrete(nb_actions) self.nb_actions = str(nb_actions) @@ -76,6 +79,27 @@ def __init__(self, config = None, root=None, use_server: Optional[bool]=False): def __getattr__(self, name): # assume it is implemented by self.instance return self.env.__getattribute__(name) + + def initialize_states(self): + if self.env.config["randomize_states"]: + self.init_states = self.randomize_init_states() + self.env.config.update({'init_states': list(self.init_states)}) + else: + self.init_states = self.env.config["init_states"] + + def randomize_init_states(self): + """Randomize initial states. + + Returns: + ------- + init_states: list + List of random initial states for the environment. + + Note: + ---- + This method should be implemented according to needed random initialization. + """ + return self.env.config["init_states"] def step(self, action): if self.use_server: @@ -87,6 +111,8 @@ def step(self, action): def reset(self): """Reset simulation. """ + self.initialize_states() + self.env.reset() self.env.goal = [-30 + 60 * self.env.np_random.random(), -30 + 60 * self.env.np_random.random(), 125 + 20 * self.env.np_random.random()] From 00ad68a318beca4c379754ee50ac92e38045a550 Mon Sep 17 00:00:00 2001 From: samuelmyoussef Date: Thu, 26 Sep 2024 03:31:00 +0200 Subject: [PATCH 40/85] goal init and update --- sofagym/envs/Diamond/DiamondEnv.py | 15 +++++++++++---- sofagym/envs/Diamond/DiamondToolbox.py | 7 ++++--- 2 files changed, 15 insertions(+), 7 deletions(-) diff --git a/sofagym/envs/Diamond/DiamondEnv.py b/sofagym/envs/Diamond/DiamondEnv.py index c248a2e..835a5a5 100644 --- a/sofagym/envs/Diamond/DiamondEnv.py +++ b/sofagym/envs/Diamond/DiamondEnv.py @@ -63,6 +63,9 @@ def __init__(self, config = None, root=None, use_server: Optional[bool]=False): self.initialize_states() + if self.env.config["goal"]: + self.init_goal() + nb_actions = self.env.config["nb_actions"] self.env.action_space = spaces.Discrete(nb_actions) self.nb_actions = str(nb_actions) @@ -101,6 +104,11 @@ def randomize_init_states(self): """ return self.env.config["init_states"] + def init_goal(self): + # Set a new random goal from the list + self.env.goal = [-30 + 60 * self.env.np_random.random(), -30 + 60 * self.env.np_random.random(), 125 + 20 * self.env.np_random.random()] + self.env.config.update({'goalPos': self.env.goal}) + def step(self, action): if self.use_server: if self.env.viewer: @@ -112,12 +120,11 @@ def reset(self): """Reset simulation. """ self.initialize_states() - - self.env.reset() - self.env.goal = [-30 + 60 * self.env.np_random.random(), -30 + 60 * self.env.np_random.random(), 125 + 20 * self.env.np_random.random()] + if self.env.config["goal"]: + self.init_goal() - self.env.config.update({'goalPos': self.env.goal}) + self.env.reset() if self.use_server: obs = start_scene(self.env.config, self.nb_actions) diff --git a/sofagym/envs/Diamond/DiamondToolbox.py b/sofagym/envs/Diamond/DiamondToolbox.py index 8710078..aeb3921 100644 --- a/sofagym/envs/Diamond/DiamondToolbox.py +++ b/sofagym/envs/Diamond/DiamondToolbox.py @@ -83,7 +83,7 @@ def getReward(self): return min(3*reward**(1/2), 1.0), current_dist - def update(self): + def update(self, goal=None): """Update function. This function is used as an initialization function. @@ -97,7 +97,7 @@ def update(self): None. """ - + self.goal_pos = goal tip = self.rootNode.Robot.Actuators.actuatedPoints.position[0] self.init_dist = np.linalg.norm(np.array(tip)-np.array(self.goal_pos)) self.prev_dist = self.init_dist @@ -145,7 +145,7 @@ def __init__(self, *args, **kwargs): if kwargs["goalPos"]: self.goalPos = kwargs["goalPos"] - def update(self): + def update(self, goal): """Set the position of the goal. This function is used as an initialization function. @@ -159,6 +159,7 @@ def update(self): None. """ + self.goalPos = goal with self.goalMO.position.writeable() as position: position[0] = self.goalPos From 21479f7913abf7f8929fe8fd655cc745ae9d9533 Mon Sep 17 00:00:00 2001 From: samuelmyoussef Date: Wed, 25 Sep 2024 23:41:22 +0200 Subject: [PATCH 41/85] class instantiation --- sofagym/envs/CTR/CTREnv.py | 89 +++++++++++++++++++++++--------------- 1 file changed, 53 insertions(+), 36 deletions(-) diff --git a/sofagym/envs/CTR/CTREnv.py b/sofagym/envs/CTR/CTREnv.py index 31153c2..32d32e6 100644 --- a/sofagym/envs/CTR/CTREnv.py +++ b/sofagym/envs/CTR/CTREnv.py @@ -8,17 +8,19 @@ __copyright__ = "(c) 2021, Robocath, CNRS, Inria" __date__ = "Dec 01 2021" -from sofagym.AbstractEnv import AbstractEnv +from sofagym.AbstractEnv import AbstractEnv, ServerEnv from sofagym.rpc_server import start_scene from sofagym.viewer import LegacyViewer from sofagym.envs.CTR.CTRToolbox import startCmd from gym import spaces -import os +import os, sys import numpy as np +from typing import Optional -class ConcentricTubeRobotEnv(AbstractEnv): + +class ConcentricTubeRobotEnv: """Sub-class of AbstractEnv, dedicated to the trunk scene. See the class AbstractEnv for arguments and methods. @@ -34,6 +36,7 @@ class ConcentricTubeRobotEnv(AbstractEnv): "scale": 30, "rotation": [140.0, 0.0, 0.0], "translation": [0.0, 0.0, 0.0], + "goal": True, "goalList": [[0.0, 0.0, 0.0]], "goalPos": [0.0, 0.0, 0.0], "scale_factor": 10, @@ -47,56 +50,67 @@ class ConcentricTubeRobotEnv(AbstractEnv): "discrete": True, "seed": 0, "start_from_history": None, - "python_version": "python3.8", + "python_version": sys.version, "zFar": 5000, - "dt": 0.01 + "dt": 0.01, + "randomize_states": False, + "use_server": False } - def __init__(self, config=None): - super().__init__(config) + def __init__(self, config = None, root=None, use_server: Optional[bool]=False): + self.use_server = self.DEFAULT_CONFIG["use_server"] + self.env = ServerEnv(self.DEFAULT_CONFIG, config, root=root) if self.use_server else AbstractEnv(self.DEFAULT_CONFIG, config, root=root) + nb_actions = 12 - self.action_space = spaces.Discrete(nb_actions) + self.env.action_space = spaces.Discrete(nb_actions) self.nb_actions = str(nb_actions) dim_state = 12 low_coordinates = np.array([-1]*dim_state) high_coordinates = np.array([1]*dim_state) - self.observation_space = spaces.Box(low_coordinates, high_coordinates, - dtype='float32') + self.env.observation_space = spaces.Box(low_coordinates, high_coordinates, dtype=np.float32) self.default_action = 3 + if self.env.root is None and not self.use_server: + self.env.init_root() + + # called when an attribute is not found: + def __getattr__(self, name): + # assume it is implemented by self.instance + return self.env.__getattribute__(name) + def step(self, action): - if self.viewer: - self.viewer.step(action) + if self.use_server: + if self.env.viewer: + self.env.viewer.step(action) - return super().step(action) + return self.env.step(action) def reset(self): """Reset simulation. - - Note: - ---- - We launch a client to create the scene. The scene of the program is - client_Env.py. - """ - super().reset() + self.env.reset() - y = -20 + 50 * np.random.random() + y = -20 + 50 * self.env.np_random.random() - self.goal = [0.0, y, abs(y) + 30 * np.random.random()] + self.env.goal = [0.0, y, abs(y) + 30 * self.env.np_random.random()] - self.config.update({'goalPos': self.goal}) - obs = start_scene(self.config, self.nb_actions) - if self.viewer: - self.viewer.reset() + self.env.config.update({'goalPos': self.env.goal}) - self.step(0) - self.step(4) - self.step(8) + if self.use_server: + obs = start_scene(self.env.config, self.nb_actions) + if self.env.viewer: + self.env.viewer.reset() - return np.array(obs['observation']) + self.env.step(0) + self.env.step(4) + self.env.step(8) + state = np.array(obs['observation'], dtype=np.float32) + else: + state = np.array(self.env._getState(self.env.root), dtype=np.float32) + + return state def render(self, mode='rgb_array'): """See the current state of the environment. @@ -113,12 +127,15 @@ def render(self, mode='rgb_array'): ------- None. """ - if not self.viewer: - display_size = self.config["display_size"] # Sim display - self.viewer = LegacyViewer(self, display_size, startCmd=startCmd) + if self.use_server: + if not self.env.viewer: + display_size = self.env.config["display_size"] # Sim display + self.env.viewer = LegacyViewer(self, display_size, startCmd=startCmd) - # Use the viewer to display the environment. - self.viewer.render() + # Use the viewer to display the environment. + self.env.viewer.render() + else: + self.env.render(mode) def get_available_actions(self): """Gives the actions available in the environment. @@ -131,4 +148,4 @@ def get_available_actions(self): ------- list of the action available in the environment. """ - return list(range(int(self.nb_actions))) + return self.env.action_space From e925b8651d4af400a56fc1835921bf17333d28fe Mon Sep 17 00:00:00 2001 From: samuelmyoussef Date: Wed, 25 Sep 2024 23:42:59 +0200 Subject: [PATCH 42/85] add some default configs --- sofagym/envs/CTR/CTREnv.py | 7 +++++-- 1 file changed, 5 insertions(+), 2 deletions(-) diff --git a/sofagym/envs/CTR/CTREnv.py b/sofagym/envs/CTR/CTREnv.py index 32d32e6..8b36fc1 100644 --- a/sofagym/envs/CTR/CTREnv.py +++ b/sofagym/envs/CTR/CTREnv.py @@ -28,6 +28,7 @@ class ConcentricTubeRobotEnv: # Setting a default configuration path = os.path.dirname(os.path.abspath(__file__)) metadata = {'render.modes': ['human', 'rgb_array']} + dim_state = 12 DEFAULT_CONFIG = {"scene": "CTR", "deterministic": True, "source": [-150, 0, 30], @@ -53,6 +54,8 @@ class ConcentricTubeRobotEnv: "python_version": sys.version, "zFar": 5000, "dt": 0.01, + "nb_actions": 12, + "dim_state": dim_state, "randomize_states": False, "use_server": False } @@ -61,11 +64,11 @@ def __init__(self, config = None, root=None, use_server: Optional[bool]=False): self.use_server = self.DEFAULT_CONFIG["use_server"] self.env = ServerEnv(self.DEFAULT_CONFIG, config, root=root) if self.use_server else AbstractEnv(self.DEFAULT_CONFIG, config, root=root) - nb_actions = 12 + nb_actions = self.env.config["nb_actions"] self.env.action_space = spaces.Discrete(nb_actions) self.nb_actions = str(nb_actions) - dim_state = 12 + dim_state = self.env.config["dim_state"] low_coordinates = np.array([-1]*dim_state) high_coordinates = np.array([1]*dim_state) self.env.observation_space = spaces.Box(low_coordinates, high_coordinates, dtype=np.float32) From 497234adfb0479e4d228a8ada97e6e74d4b836ec Mon Sep 17 00:00:00 2001 From: samuelmyoussef Date: Wed, 25 Sep 2024 23:44:51 +0200 Subject: [PATCH 43/85] initialize states --- sofagym/envs/CTR/CTREnv.py | 28 +++++++++++++++++++++++++++- 1 file changed, 27 insertions(+), 1 deletion(-) diff --git a/sofagym/envs/CTR/CTREnv.py b/sofagym/envs/CTR/CTREnv.py index 8b36fc1..d18581e 100644 --- a/sofagym/envs/CTR/CTREnv.py +++ b/sofagym/envs/CTR/CTREnv.py @@ -57,6 +57,7 @@ class ConcentricTubeRobotEnv: "nb_actions": 12, "dim_state": dim_state, "randomize_states": False, + "init_states": [0] * dim_state, "use_server": False } @@ -64,6 +65,8 @@ def __init__(self, config = None, root=None, use_server: Optional[bool]=False): self.use_server = self.DEFAULT_CONFIG["use_server"] self.env = ServerEnv(self.DEFAULT_CONFIG, config, root=root) if self.use_server else AbstractEnv(self.DEFAULT_CONFIG, config, root=root) + self.initialize_states() + nb_actions = self.env.config["nb_actions"] self.env.action_space = spaces.Discrete(nb_actions) self.nb_actions = str(nb_actions) @@ -82,17 +85,40 @@ def __init__(self, config = None, root=None, use_server: Optional[bool]=False): def __getattr__(self, name): # assume it is implemented by self.instance return self.env.__getattribute__(name) + + def initialize_states(self): + if self.env.config["randomize_states"]: + self.init_states = self.randomize_init_states() + self.env.config.update({'init_states': list(self.init_states)}) + else: + self.init_states = self.env.config["init_states"] + + def randomize_init_states(self): + """Randomize initial states. + + Returns: + ------- + init_states: list + List of random initial states for the environment. + + Note: + ---- + This method should be implemented according to needed random initialization. + """ + return self.env.config["init_states"] def step(self, action): if self.use_server: if self.env.viewer: self.env.viewer.step(action) - + return self.env.step(action) def reset(self): """Reset simulation. """ + self.initialize_states() + self.env.reset() y = -20 + 50 * self.env.np_random.random() From 841836cd2700692661a946c35d8a5a05ecb40b55 Mon Sep 17 00:00:00 2001 From: samuelmyoussef Date: Wed, 25 Sep 2024 23:51:59 +0200 Subject: [PATCH 44/85] goal init and update --- sofagym/envs/CTR/CTREnv.py | 18 ++++++++++++------ sofagym/envs/CTR/CTRScene.py | 1 + sofagym/envs/CTR/CTRToolbox.py | 7 ++++--- 3 files changed, 17 insertions(+), 9 deletions(-) diff --git a/sofagym/envs/CTR/CTREnv.py b/sofagym/envs/CTR/CTREnv.py index d18581e..703078e 100644 --- a/sofagym/envs/CTR/CTREnv.py +++ b/sofagym/envs/CTR/CTREnv.py @@ -67,6 +67,9 @@ def __init__(self, config = None, root=None, use_server: Optional[bool]=False): self.initialize_states() + if self.env.config["goal"]: + self.init_goal() + nb_actions = self.env.config["nb_actions"] self.env.action_space = spaces.Discrete(nb_actions) self.nb_actions = str(nb_actions) @@ -107,6 +110,12 @@ def randomize_init_states(self): """ return self.env.config["init_states"] + def init_goal(self): + # Set a new random goal from the list + y = -20 + 50 * self.env.np_random.random() + self.env.goal = [0.0, y, abs(y) + 30 * self.env.np_random.random()] + self.env.config.update({'goalPos': self.env.goal}) + def step(self, action): if self.use_server: if self.env.viewer: @@ -118,14 +127,11 @@ def reset(self): """Reset simulation. """ self.initialize_states() - - self.env.reset() - - y = -20 + 50 * self.env.np_random.random() - self.env.goal = [0.0, y, abs(y) + 30 * self.env.np_random.random()] + if self.env.config["goal"]: + self.init_goal() - self.env.config.update({'goalPos': self.env.goal}) + self.env.reset() if self.use_server: obs = start_scene(self.env.config, self.nb_actions) diff --git a/sofagym/envs/CTR/CTRScene.py b/sofagym/envs/CTR/CTRScene.py index ac607be..08fd0d7 100644 --- a/sofagym/envs/CTR/CTRScene.py +++ b/sofagym/envs/CTR/CTRScene.py @@ -186,6 +186,7 @@ def add_geometry(root, config, mode_simu=True, mode_visu=True): "rotation": [140.0, 0.0, 0.0], "translation": [0.0, 0.0, 0.0], "goalList": [[0.0, 0.0, 10.0]], + "goalPos": [0.0, 0.0, 0.0], "scale_factor": 10, "timer_limit": 50, "timeout": 30, diff --git a/sofagym/envs/CTR/CTRToolbox.py b/sofagym/envs/CTR/CTRToolbox.py index 1ef487b..6f988a4 100644 --- a/sofagym/envs/CTR/CTRToolbox.py +++ b/sofagym/envs/CTR/CTRToolbox.py @@ -86,7 +86,7 @@ def getReward(self): return min(3*reward**(1/2), 1.0), current_dist - def update(self): + def update(self, goal=None): """Update function. This function is used as an initialization function. @@ -100,7 +100,7 @@ def update(self): None. """ - + self.goal_pos = goal tip = self.root.InstrumentCombined.DOFs.position[-1][:3] self.init_dist = np.linalg.norm(np.array(tip)-np.array(self.goal_pos)) self.prev_dist = self.init_dist @@ -148,7 +148,7 @@ def __init__(self, *args, **kwargs): if kwargs["goalPos"]: self.goalPos = kwargs["goalPos"] - def update(self): + def update(self, goal): """Set the position of the goal. This function is used as an initialization function. @@ -162,6 +162,7 @@ def update(self): None. """ + self.goalPos = goal with self.goalMO.position.writeable() as position: print("update", self.goalPos) position[0] = self.goalPos From 014f91ecd7c37d9d2009543395916eead1e741e6 Mon Sep 17 00:00:00 2001 From: samuelmyoussef Date: Thu, 8 Feb 2024 15:24:21 +0100 Subject: [PATCH 45/85] class instantiation for no server or server env --- sofagym/envs/CatheterBeam/CatheterBeamEnv.py | 48 ++++++++++++-------- 1 file changed, 29 insertions(+), 19 deletions(-) diff --git a/sofagym/envs/CatheterBeam/CatheterBeamEnv.py b/sofagym/envs/CatheterBeam/CatheterBeamEnv.py index 54729e9..f344b42 100644 --- a/sofagym/envs/CatheterBeam/CatheterBeamEnv.py +++ b/sofagym/envs/CatheterBeam/CatheterBeamEnv.py @@ -1,14 +1,15 @@ import os import sys +from typing import Optional import numpy as np from gym import spaces -from sofagym.AbstractEnv import AbstractEnv +from sofagym.AbstractEnv import AbstractEnv, ServerEnv from sofagym.rpc_server import start_scene -class CatheterBeamEnv(AbstractEnv): +class CatheterBeamEnv: """Sub-class of AbstractEnv, dedicated to the catheter beam scene. See the class AbstractEnv for arguments and methods. @@ -40,36 +41,45 @@ class CatheterBeamEnv(AbstractEnv): "scale": 30, "rotation": [140.0, 0.0, 0.0], "translation": [0.0, 0.0, 0.0], - "goalList": [1226, 1663, 1797, 1544, 2233, 2580, 3214] + "goal": True, + "goalList": [1226, 1663, 1797, 1544, 2233, 2580, 3214], + "randomize_states": False, + "use_server": False } - def __init__(self, config = None): - super().__init__(config) + def __init__(self, config = None, root=None, use_server: Optional[bool]=False): + self.use_server = self.DEFAULT_CONFIG["use_server"] + self.env = ServerEnv(self.DEFAULT_CONFIG, config, root=root) if self.use_server else AbstractEnv(self.DEFAULT_CONFIG, config, root=root) + nb_actions = 12 - self.action_space = spaces.Discrete(nb_actions) + self.env.action_space = spaces.Discrete(nb_actions) self.nb_actions = str(nb_actions) dim_state = 12 low_coordinates = np.array([-1]*dim_state) high_coordinates = np.array([1]*dim_state) - self.observation_space = spaces.Box(low_coordinates, high_coordinates, dtype=np.float32) + self.env.observation_space = spaces.Box(low_coordinates, high_coordinates, dtype=np.float32) - def reset(self): - """Reset simulation. + if self.env.root is None and not self.use_server: + self.env.init_root() - Note: - ---- - We launch a client to create the scene. The scene of the program is - client_Env.py. + # called when an attribute is not found: + def __getattr__(self, name): + # assume it is implemented by self.instance + return self.env.__getattribute__(name) + def reset(self): + """Reset simulation. """ - super().reset() - - self.config.update({'goalPos': self.goal}) + self.env.reset() - obs = start_scene(self.config, self.nb_actions) + if self.use_server: + obs = start_scene(self.env.config, self.nb_actions) + state = np.array(obs['observation'], dtype=np.float32) + else: + state = np.array(self.env._getState(self.env.root), dtype=np.float32) - return np.array(obs['observation']) + return state def get_available_actions(self): """Gives the actions available in the environment. @@ -82,4 +92,4 @@ def get_available_actions(self): ------- list of the action available in the environment. """ - return self.action_space + return self.env.action_space From 092824d30c8c0e92a1c4d5cd39d1a883b6ba93ce Mon Sep 17 00:00:00 2001 From: samuelmyoussef Date: Thu, 8 Feb 2024 16:26:36 +0100 Subject: [PATCH 46/85] add some default configs --- sofagym/envs/CatheterBeam/CatheterBeamEnv.py | 9 ++++++--- 1 file changed, 6 insertions(+), 3 deletions(-) diff --git a/sofagym/envs/CatheterBeam/CatheterBeamEnv.py b/sofagym/envs/CatheterBeam/CatheterBeamEnv.py index f344b42..42d7a7d 100644 --- a/sofagym/envs/CatheterBeam/CatheterBeamEnv.py +++ b/sofagym/envs/CatheterBeam/CatheterBeamEnv.py @@ -15,8 +15,9 @@ class CatheterBeamEnv: See the class AbstractEnv for arguments and methods. """ #Setting a default configuration - path = path = os.path.dirname(os.path.abspath(__file__)) + path = os.path.dirname(os.path.abspath(__file__)) metadata = {'render.modes': ['human', 'rgb_array']} + dim_state = 12 DEFAULT_CONFIG = {"scene": "CatheterBeam", "deterministic": True, "source": [-1169.51, 298.574, 257.631], @@ -43,6 +44,8 @@ class CatheterBeamEnv: "translation": [0.0, 0.0, 0.0], "goal": True, "goalList": [1226, 1663, 1797, 1544, 2233, 2580, 3214], + "nb_actions": 12, + "dim_state": dim_state, "randomize_states": False, "use_server": False } @@ -51,11 +54,11 @@ def __init__(self, config = None, root=None, use_server: Optional[bool]=False): self.use_server = self.DEFAULT_CONFIG["use_server"] self.env = ServerEnv(self.DEFAULT_CONFIG, config, root=root) if self.use_server else AbstractEnv(self.DEFAULT_CONFIG, config, root=root) - nb_actions = 12 + nb_actions = self.env.config["nb_actions"] self.env.action_space = spaces.Discrete(nb_actions) self.nb_actions = str(nb_actions) - dim_state = 12 + dim_state = self.env.config["dim_state"] low_coordinates = np.array([-1]*dim_state) high_coordinates = np.array([1]*dim_state) self.env.observation_space = spaces.Box(low_coordinates, high_coordinates, dtype=np.float32) From 549445ebfdf9405d5e2292036f6556513312a691 Mon Sep 17 00:00:00 2001 From: samuelmyoussef Date: Thu, 8 Feb 2024 17:41:12 +0100 Subject: [PATCH 47/85] initialize states --- sofagym/envs/CatheterBeam/CatheterBeamEnv.py | 26 ++++++++++++++++++++ 1 file changed, 26 insertions(+) diff --git a/sofagym/envs/CatheterBeam/CatheterBeamEnv.py b/sofagym/envs/CatheterBeam/CatheterBeamEnv.py index 42d7a7d..9bcaa57 100644 --- a/sofagym/envs/CatheterBeam/CatheterBeamEnv.py +++ b/sofagym/envs/CatheterBeam/CatheterBeamEnv.py @@ -47,6 +47,7 @@ class CatheterBeamEnv: "nb_actions": 12, "dim_state": dim_state, "randomize_states": False, + "init_states": [0] * dim_state, "use_server": False } @@ -54,6 +55,8 @@ def __init__(self, config = None, root=None, use_server: Optional[bool]=False): self.use_server = self.DEFAULT_CONFIG["use_server"] self.env = ServerEnv(self.DEFAULT_CONFIG, config, root=root) if self.use_server else AbstractEnv(self.DEFAULT_CONFIG, config, root=root) + self.initialize_states() + nb_actions = self.env.config["nb_actions"] self.env.action_space = spaces.Discrete(nb_actions) self.nb_actions = str(nb_actions) @@ -71,9 +74,32 @@ def __getattr__(self, name): # assume it is implemented by self.instance return self.env.__getattribute__(name) + def initialize_states(self): + if self.env.config["randomize_states"]: + self.init_states = self.randomize_init_states() + self.env.config.update({'init_states': list(self.init_states)}) + else: + self.init_states = self.env.config["init_states"] + + def randomize_init_states(self): + """Randomize initial states. + + Returns: + ------- + init_states: list + List of random initial states for the environment. + + Note: + ---- + This method should be implemented according to needed random initialization. + """ + return self.env.config["init_states"] + def reset(self): """Reset simulation. """ + self.initialize_states() + self.env.reset() if self.use_server: From dbc5f913367b19c1bc884ab438a5fb3a878cb402 Mon Sep 17 00:00:00 2001 From: samuelmyoussef Date: Wed, 25 Sep 2024 10:31:38 +0200 Subject: [PATCH 48/85] goal init and update --- sofagym/envs/CatheterBeam/CatheterBeamEnv.py | 16 +++++++++++++++- sofagym/envs/CatheterBeam/CatheterBeamToolbox.py | 6 ++++-- 2 files changed, 19 insertions(+), 3 deletions(-) diff --git a/sofagym/envs/CatheterBeam/CatheterBeamEnv.py b/sofagym/envs/CatheterBeam/CatheterBeamEnv.py index 9bcaa57..efb1f8b 100644 --- a/sofagym/envs/CatheterBeam/CatheterBeamEnv.py +++ b/sofagym/envs/CatheterBeam/CatheterBeamEnv.py @@ -57,6 +57,9 @@ def __init__(self, config = None, root=None, use_server: Optional[bool]=False): self.initialize_states() + if self.env.config["goal"]: + self.init_goal() + nb_actions = self.env.config["nb_actions"] self.env.action_space = spaces.Discrete(nb_actions) self.nb_actions = str(nb_actions) @@ -95,11 +98,22 @@ def randomize_init_states(self): """ return self.env.config["init_states"] + def init_goal(self): + # Set a new random goal from the list + goalList = self.env.config["goalList"] + id_goal = self.env.np_random.choice(range(len(goalList))) + self.env.config.update({'goal_node': id_goal}) + self.env.goal = goalList[id_goal] + self.env.config.update({'goalPos': self.env.goal}) + def reset(self): """Reset simulation. """ self.initialize_states() - + + if self.env.config["goal"]: + self.init_goal() + self.env.reset() if self.use_server: diff --git a/sofagym/envs/CatheterBeam/CatheterBeamToolbox.py b/sofagym/envs/CatheterBeam/CatheterBeamToolbox.py index d50bd20..d985a28 100644 --- a/sofagym/envs/CatheterBeam/CatheterBeamToolbox.py +++ b/sofagym/envs/CatheterBeam/CatheterBeamToolbox.py @@ -77,7 +77,7 @@ def getReward(self): return -current_dist, current_dist - def update(self): + def update(self, goal=None): """Update function. This function is used as an initialization function. @@ -91,6 +91,7 @@ def update(self): None. """ + self.goal_pos = self.root.CollisionModel.DOFs1.position.value[goal][:3] tip = self.root.InstrumentCombined.DOFs.position[-1][:3] self.init_dist = np.linalg.norm(np.array(tip)-np.array(self.goal_pos)) self.prev_dist = self.init_dist @@ -137,7 +138,7 @@ def __init__(self, *args, **kwargs): if kwargs["goalPos"]: self.goalPos = kwargs["goalPos"] - def update(self): + def update(self, goal): """Set the position of the goal. This function is used as an initialization function. @@ -151,6 +152,7 @@ def update(self): None. """ + self.goalPos = goal new_position = self.rootNode.CollisionModel.DOFs1.position.value[self.goalPos][:3] with self.goal.GoalMO.position.writeable() as position: position[0] = new_position From 1d63f61229187a653f520a6226643d9f42786c66 Mon Sep 17 00:00:00 2001 From: samuelmyoussef Date: Wed, 25 Sep 2024 17:04:04 +0200 Subject: [PATCH 49/85] class instantiation --- .../envs/CatchTheObject/CatchTheObjectEnv.py | 64 ++++++++++--------- 1 file changed, 35 insertions(+), 29 deletions(-) diff --git a/sofagym/envs/CatchTheObject/CatchTheObjectEnv.py b/sofagym/envs/CatchTheObject/CatchTheObjectEnv.py index aecb624..6ceb7d9 100644 --- a/sofagym/envs/CatchTheObject/CatchTheObjectEnv.py +++ b/sofagym/envs/CatchTheObject/CatchTheObjectEnv.py @@ -8,16 +8,18 @@ __copyright__ = "(c) 2021, Inria" __date__ = "Feb 3 2021" -import os +import os, sys -from sofagym.AbstractEnv import AbstractEnv +from sofagym.AbstractEnv import AbstractEnv, ServerEnv from sofagym.rpc_server import start_scene from gym import spaces import numpy as np -class CatchTheObject(AbstractEnv): +from typing import Optional + +class CatchTheObject: """Sub-class of AbstractEnv, dedicated to the gripper scene. See the class AbstractEnv for arguments and methods. @@ -29,6 +31,7 @@ class CatchTheObject(AbstractEnv): "deterministic": True, "source": [0, -70, 10], "target": [0, 0, 10], + "goal": True, "goalList": [[7, 0, 20]], "start_node": None, "scale_factor": 10, @@ -43,50 +46,53 @@ class CatchTheObject(AbstractEnv): "planning": False, "discrete": False, "start_from_history": None, - "python_version": "python3.9", + "python_version": sys.version, "zFar": 4000, "time_before_start": 0, "seed": None, "max_move": 10, - "max_pressure": 15 + "max_pressure": 15, + "randomize_states": False, + "use_server": False } - def __init__(self, config = None): - super().__init__(config) + def __init__(self, config = None, root=None, use_server: Optional[bool]=False): + self.use_server = self.DEFAULT_CONFIG["use_server"] + self.env = ServerEnv(self.DEFAULT_CONFIG, config, root=root) if self.use_server else AbstractEnv(self.DEFAULT_CONFIG, config, root=root) + nb_actions = -1 low = np.array([-1]*1) high = np.array([1]*1) - self.action_space = spaces.Box(low=low, high=high, shape=(1,), dtype='float32') + self.env.action_space = spaces.Box(low=low, high=high, shape=(1,), dtype=np.float32) self.nb_actions = str(nb_actions) dim_state = 5 low_coordinates = np.array([-1]*dim_state) high_coordinates = np.array([1]*dim_state) - self.observation_space = spaces.Box(low_coordinates, high_coordinates, - dtype='float32') + self.env.observation_space = spaces.Box(low_coordinates, high_coordinates, dtype=np.float32) - def step(self, action): - return super().step(action) + if self.env.root is None and not self.use_server: + self.env.init_root() + + # called when an attribute is not found: + def __getattr__(self, name): + # assume it is implemented by self.instance + return self.env.__getattribute__(name) def reset(self): """Reset simulation. - - Note: - ---- - We launch a client to create the scene. The scene of the program is - client_Env.py. - """ - - super().reset() - - self.count = 0 - self.config.update({'goalPos': self.goal}) - # obs = super().reset() - # return np.array(obs) - obs = start_scene(self.config, self.nb_actions) - - return np.array(obs['observation']) + self.env.reset() + + self.env.config.update({'goalPos': self.env.goal}) + + if self.use_server: + obs = start_scene(self.env.config, self.nb_actions) + state = np.array(obs['observation'], dtype=np.float32) + else: + state = np.array(self.env._getState(self.env.root), dtype=np.float32) + + return state def get_available_actions(self): """Gives the actions available in the environment. @@ -99,7 +105,7 @@ def get_available_actions(self): ------- list of the action available in the environment. """ - return self.action_space + return self.env.action_space From f79435adcfa66fd865153f2709d316a534fde0bd Mon Sep 17 00:00:00 2001 From: samuelmyoussef Date: Wed, 25 Sep 2024 17:05:28 +0200 Subject: [PATCH 50/85] add some default configs --- sofagym/envs/CatchTheObject/CatchTheObjectEnv.py | 7 +++++-- 1 file changed, 5 insertions(+), 2 deletions(-) diff --git a/sofagym/envs/CatchTheObject/CatchTheObjectEnv.py b/sofagym/envs/CatchTheObject/CatchTheObjectEnv.py index 6ceb7d9..7782b23 100644 --- a/sofagym/envs/CatchTheObject/CatchTheObjectEnv.py +++ b/sofagym/envs/CatchTheObject/CatchTheObjectEnv.py @@ -27,6 +27,7 @@ class CatchTheObject: #Setting a default configuration path = os.path.dirname(os.path.abspath(__file__)) metadata = {'render.modes': ['human', 'rgb_array']} + dim_state = 5 DEFAULT_CONFIG = {"scene": "CatchTheObject", "deterministic": True, "source": [0, -70, 10], @@ -52,6 +53,8 @@ class CatchTheObject: "seed": None, "max_move": 10, "max_pressure": 15, + "nb_actions": -1, + "dim_state": dim_state, "randomize_states": False, "use_server": False } @@ -60,13 +63,13 @@ def __init__(self, config = None, root=None, use_server: Optional[bool]=False): self.use_server = self.DEFAULT_CONFIG["use_server"] self.env = ServerEnv(self.DEFAULT_CONFIG, config, root=root) if self.use_server else AbstractEnv(self.DEFAULT_CONFIG, config, root=root) - nb_actions = -1 + nb_actions = self.env.config["nb_actions"] low = np.array([-1]*1) high = np.array([1]*1) self.env.action_space = spaces.Box(low=low, high=high, shape=(1,), dtype=np.float32) self.nb_actions = str(nb_actions) - dim_state = 5 + dim_state = self.env.config["dim_state"] low_coordinates = np.array([-1]*dim_state) high_coordinates = np.array([1]*dim_state) self.env.observation_space = spaces.Box(low_coordinates, high_coordinates, dtype=np.float32) From 18580c08a02a2ff865f7e14af9b9863d7a496739 Mon Sep 17 00:00:00 2001 From: samuelmyoussef Date: Wed, 25 Sep 2024 17:11:28 +0200 Subject: [PATCH 51/85] initialize states --- .../envs/CatchTheObject/CatchTheObjectEnv.py | 26 +++++++++++++++++++ 1 file changed, 26 insertions(+) diff --git a/sofagym/envs/CatchTheObject/CatchTheObjectEnv.py b/sofagym/envs/CatchTheObject/CatchTheObjectEnv.py index 7782b23..2c122c6 100644 --- a/sofagym/envs/CatchTheObject/CatchTheObjectEnv.py +++ b/sofagym/envs/CatchTheObject/CatchTheObjectEnv.py @@ -56,6 +56,7 @@ class CatchTheObject: "nb_actions": -1, "dim_state": dim_state, "randomize_states": False, + "init_states": [0] * dim_state, "use_server": False } @@ -63,6 +64,8 @@ def __init__(self, config = None, root=None, use_server: Optional[bool]=False): self.use_server = self.DEFAULT_CONFIG["use_server"] self.env = ServerEnv(self.DEFAULT_CONFIG, config, root=root) if self.use_server else AbstractEnv(self.DEFAULT_CONFIG, config, root=root) + self.initialize_states() + nb_actions = self.env.config["nb_actions"] low = np.array([-1]*1) high = np.array([1]*1) @@ -82,9 +85,32 @@ def __getattr__(self, name): # assume it is implemented by self.instance return self.env.__getattribute__(name) + def initialize_states(self): + if self.env.config["randomize_states"]: + self.init_states = self.randomize_init_states() + self.env.config.update({'init_states': list(self.init_states)}) + else: + self.init_states = self.env.config["init_states"] + + def randomize_init_states(self): + """Randomize initial states. + + Returns: + ------- + init_states: list + List of random initial states for the environment. + + Note: + ---- + This method should be implemented according to needed random initialization. + """ + return self.env.config["init_states"] + def reset(self): """Reset simulation. """ + self.initialize_states() + self.env.reset() self.env.config.update({'goalPos': self.env.goal}) From 1f923ae7575932e06077267ac8bb4b733e78b41a Mon Sep 17 00:00:00 2001 From: samuelmyoussef Date: Wed, 25 Sep 2024 17:16:19 +0200 Subject: [PATCH 52/85] goal init and update --- .../envs/CatchTheObject/CatchTheObjectEnv.py | 18 +++++++++++++++--- .../CatchTheObject/CatchTheObjectToolbox.py | 7 ++++--- 2 files changed, 19 insertions(+), 6 deletions(-) diff --git a/sofagym/envs/CatchTheObject/CatchTheObjectEnv.py b/sofagym/envs/CatchTheObject/CatchTheObjectEnv.py index 2c122c6..8fd2e47 100644 --- a/sofagym/envs/CatchTheObject/CatchTheObjectEnv.py +++ b/sofagym/envs/CatchTheObject/CatchTheObjectEnv.py @@ -66,6 +66,9 @@ def __init__(self, config = None, root=None, use_server: Optional[bool]=False): self.initialize_states() + if self.env.config["goal"]: + self.init_goal() + nb_actions = self.env.config["nb_actions"] low = np.array([-1]*1) high = np.array([1]*1) @@ -106,14 +109,23 @@ def randomize_init_states(self): """ return self.env.config["init_states"] + def init_goal(self): + # Set a new random goal from the list + goalList = self.env.config["goalList"] + id_goal = self.env.np_random.choice(range(len(goalList))) + self.env.config.update({'goal_node': id_goal}) + self.env.goal = goalList[id_goal] + self.env.config.update({'goalPos': self.env.goal}) + def reset(self): """Reset simulation. """ self.initialize_states() - - self.env.reset() - self.env.config.update({'goalPos': self.env.goal}) + if self.env.config["goal"]: + self.init_goal() + + self.env.reset() if self.use_server: obs = start_scene(self.env.config, self.nb_actions) diff --git a/sofagym/envs/CatchTheObject/CatchTheObjectToolbox.py b/sofagym/envs/CatchTheObject/CatchTheObjectToolbox.py index 7087b65..7c0f8b6 100644 --- a/sofagym/envs/CatchTheObject/CatchTheObjectToolbox.py +++ b/sofagym/envs/CatchTheObject/CatchTheObjectToolbox.py @@ -88,7 +88,7 @@ def getReward(self): return r, dist, under - def update(self): + def update(self, goal=None): """Update function. This function is used as an initialization function. @@ -102,6 +102,7 @@ def update(self): None. """ + pos_goal = goal self.max_dist = np.linalg.norm(np.array([0, self.ball.max_high])-np.array([self.cart.max_move, 0])) @@ -113,8 +114,8 @@ def __init__(self, *args, **kwargs): if 'goalPos' in kwargs: self.goalPos = kwargs["goalPos"] - def update(self): - pass + def update(self, goal): + self.goalPos = goal def set_mo_pos(self, goal): pass From 68c266f5e817d70186365b93b9ebf55e7c6b75ba Mon Sep 17 00:00:00 2001 From: samuelmyoussef Date: Wed, 25 Sep 2024 15:52:32 +0200 Subject: [PATCH 53/85] class instantiation --- .../CartStemContact/CartStemContactEnv.py | 79 +++++++++++-------- 1 file changed, 44 insertions(+), 35 deletions(-) diff --git a/sofagym/envs/CartStemContact/CartStemContactEnv.py b/sofagym/envs/CartStemContact/CartStemContactEnv.py index 9c8323b..73486fe 100644 --- a/sofagym/envs/CartStemContact/CartStemContactEnv.py +++ b/sofagym/envs/CartStemContact/CartStemContactEnv.py @@ -8,27 +8,30 @@ __copyright__ = "(c) 2021, Inria" __date__ = "Feb 3 2021" -import os +import os, sys -from sofagym.AbstractEnv import AbstractEnv +from sofagym.AbstractEnv import AbstractEnv, ServerEnv from sofagym.rpc_server import start_scene from gym import spaces import numpy as np -class CartStemContactEnv(AbstractEnv): +from typing import Optional + +class CartStemContactEnv: """Sub-class of AbstractEnv, dedicated to the gripper scene. See the class AbstractEnv for arguments and methods. """ #Setting a default configuration - path = os.path.dirname(os.path.abspath(__file__)) + path = os.path.dirname(os.path.abspath(__file__)) metadata = {'render.modes': ['human', 'rgb_array']} DEFAULT_CONFIG = {"scene": "CartStemContact", "deterministic": True, "source": [0, -50, 10], "target": [0, 0, 10], + "goal": True, "goalList": [[7, 0, 20]], "start_node": None, "scale_factor": 30, @@ -43,60 +46,66 @@ class CartStemContactEnv(AbstractEnv): "planning": False, "discrete": False, "start_from_history": None, - "python_version": "python3.9", + "python_version": sys.version, "zFar": 4000, "time_before_start": 0, "seed": None, "init_x": 5, "cube_x": [-6, 6], "max_move": 7.5, + "randomize_states": False, + "use_server": False } + def __init__(self, config = None, root=None, use_server: Optional[bool]=False): + self.use_server = self.DEFAULT_CONFIG["use_server"] + self.env = ServerEnv(self.DEFAULT_CONFIG, config, root=root) if self.use_server else AbstractEnv(self.DEFAULT_CONFIG, config, root=root) - def __init__(self, config = None): - super().__init__(config) nb_actions = -1 low = np.array([-1]*1) high = np.array([1]*1) - self.action_space = spaces.Box(low=low, high=high, shape=(1,), dtype='float32') + self.env.action_space = spaces.Box(low=low, high=high, shape=(1,), dtype=np.float32) self.nb_actions = str(nb_actions) dim_state = 8 - low_coordinates = np.array([-1]*dim_state) high_coordinates = np.array([1]*dim_state) - self.observation_space = spaces.Box(low_coordinates, high_coordinates, - dtype='float32') + self.env.observation_space = spaces.Box(low_coordinates, high_coordinates, dtype=np.float32) - def step(self, action): - return super().step(action) + if self.env.root is None and not self.use_server: + self.env.init_root() + + # called when an attribute is not found: + def __getattr__(self, name): + # assume it is implemented by self.instance + return self.env.__getattribute__(name) def reset(self): """Reset simulation. - - Note: - ---- - We launch a client to create the scene. The scene of the program is - client_Env.py. - """ - low_cube, high_cube = -6+ 2*np.random.random(), 6 - 2*np.random.random() - self.config.update({'cube_x': [low_cube, high_cube]}) - self.config.update({'init_x': (low_cube + 3) + (high_cube-low_cube-3)*np.random.random()}) + low_cube, high_cube = -6+ 2*self.env.np_random.random(), 6 - 2*self.env.np_random.random() + self.env.config.update({'cube_x': [low_cube, high_cube]}) + self.env.config.update({'init_x': (low_cube + 3) + (high_cube-low_cube-3)*self.env.np_random.random()}) - if np.random.random() > 0.5: - x_goal = low_cube + 3.5*np.random.random() + if self.env.np_random.random() > 0.5: + x_goal = low_cube + 3.5*self.env.np_random.random() else: - x_goal = high_cube - 3.5*np.random.random() - self.config.update({'goalList': [[x_goal, 0, 20]]}) - self.goalList = self.config["goalList"] - super().reset() - self.config.update({'max_move': max(abs(low_cube-1), high_cube+1)}) - self.config.update({'goalPos': self.goal}) - - obs = start_scene(self.config, self.nb_actions) - - return np.array(obs['observation']) + x_goal = high_cube - 3.5*self.env.np_random.random() + self.env.config.update({'goalList': [[x_goal, 0, 20]]}) + self.goalList = self.env.config["goalList"] + + self.env.reset() + + self.env.config.update({'max_move': max(abs(low_cube-1), high_cube+1)}) + self.env.config.update({'goalPos': self.env.goal}) + + if self.use_server: + obs = start_scene(self.env.config, self.nb_actions) + state = np.array(obs['observation'], dtype=np.float32) + else: + state = np.array(self.env._getState(self.env.root), dtype=np.float32) + + return state def get_available_actions(self): """Gives the actions available in the environment. @@ -109,6 +118,6 @@ def get_available_actions(self): ------- list of the action available in the environment. """ - return self.action_space + return self.env.action_space From b5a074e58f13ffe9be953940d5a6115b5e7a8f6e Mon Sep 17 00:00:00 2001 From: samuelmyoussef Date: Wed, 25 Sep 2024 15:55:12 +0200 Subject: [PATCH 54/85] add some default configs --- sofagym/envs/CartStemContact/CartStemContactEnv.py | 7 +++++-- 1 file changed, 5 insertions(+), 2 deletions(-) diff --git a/sofagym/envs/CartStemContact/CartStemContactEnv.py b/sofagym/envs/CartStemContact/CartStemContactEnv.py index 73486fe..578bfef 100644 --- a/sofagym/envs/CartStemContact/CartStemContactEnv.py +++ b/sofagym/envs/CartStemContact/CartStemContactEnv.py @@ -27,6 +27,7 @@ class CartStemContactEnv: #Setting a default configuration path = os.path.dirname(os.path.abspath(__file__)) metadata = {'render.modes': ['human', 'rgb_array']} + dim_state = 8 DEFAULT_CONFIG = {"scene": "CartStemContact", "deterministic": True, "source": [0, -50, 10], @@ -53,6 +54,8 @@ class CartStemContactEnv: "init_x": 5, "cube_x": [-6, 6], "max_move": 7.5, + "nb_actions": -1, + "dim_state": dim_state, "randomize_states": False, "use_server": False } @@ -61,13 +64,13 @@ def __init__(self, config = None, root=None, use_server: Optional[bool]=False): self.use_server = self.DEFAULT_CONFIG["use_server"] self.env = ServerEnv(self.DEFAULT_CONFIG, config, root=root) if self.use_server else AbstractEnv(self.DEFAULT_CONFIG, config, root=root) - nb_actions = -1 + nb_actions = self.env.config["nb_actions"] low = np.array([-1]*1) high = np.array([1]*1) self.env.action_space = spaces.Box(low=low, high=high, shape=(1,), dtype=np.float32) self.nb_actions = str(nb_actions) - dim_state = 8 + dim_state = self.env.config["dim_state"] low_coordinates = np.array([-1]*dim_state) high_coordinates = np.array([1]*dim_state) self.env.observation_space = spaces.Box(low_coordinates, high_coordinates, dtype=np.float32) From 9448604d68ad5b31a659a8e48f3f402c69cf04da Mon Sep 17 00:00:00 2001 From: samuelmyoussef Date: Wed, 25 Sep 2024 16:04:10 +0200 Subject: [PATCH 55/85] initialize states --- .../CartStemContact/CartStemContactEnv.py | 36 ++++++++++++++++--- 1 file changed, 32 insertions(+), 4 deletions(-) diff --git a/sofagym/envs/CartStemContact/CartStemContactEnv.py b/sofagym/envs/CartStemContact/CartStemContactEnv.py index 578bfef..79c665d 100644 --- a/sofagym/envs/CartStemContact/CartStemContactEnv.py +++ b/sofagym/envs/CartStemContact/CartStemContactEnv.py @@ -56,6 +56,7 @@ class CartStemContactEnv: "max_move": 7.5, "nb_actions": -1, "dim_state": dim_state, + "init_states": [0] * dim_state, "randomize_states": False, "use_server": False } @@ -64,6 +65,8 @@ def __init__(self, config = None, root=None, use_server: Optional[bool]=False): self.use_server = self.DEFAULT_CONFIG["use_server"] self.env = ServerEnv(self.DEFAULT_CONFIG, config, root=root) if self.use_server else AbstractEnv(self.DEFAULT_CONFIG, config, root=root) + self.initialize_states() + nb_actions = self.env.config["nb_actions"] low = np.array([-1]*1) high = np.array([1]*1) @@ -82,10 +85,14 @@ def __init__(self, config = None, root=None, use_server: Optional[bool]=False): def __getattr__(self, name): # assume it is implemented by self.instance return self.env.__getattribute__(name) + + def initialize_states(self): + if self.env.config["randomize_states"]: + self.init_states = self.randomize_init_states() + self.env.config.update({'init_states': list(self.init_states)}) + else: + self.init_states = self.env.config["init_states"] - def reset(self): - """Reset simulation. - """ low_cube, high_cube = -6+ 2*self.env.np_random.random(), 6 - 2*self.env.np_random.random() self.env.config.update({'cube_x': [low_cube, high_cube]}) self.env.config.update({'init_x': (low_cube + 3) + (high_cube-low_cube-3)*self.env.np_random.random()}) @@ -94,12 +101,33 @@ def reset(self): x_goal = low_cube + 3.5*self.env.np_random.random() else: x_goal = high_cube - 3.5*self.env.np_random.random() + self.env.config.update({'goalList': [[x_goal, 0, 20]]}) + self.env.config.update({'max_move': max(abs(low_cube-1), high_cube+1)}) + + def randomize_init_states(self): + """Randomize initial states. + + Returns: + ------- + init_states: list + List of random initial states for the environment. + + Note: + ---- + This method should be implemented according to needed random initialization. + """ + return self.env.config["init_states"] + + def reset(self): + """Reset simulation. + """ + self.initialize_states() + self.goalList = self.env.config["goalList"] self.env.reset() - self.env.config.update({'max_move': max(abs(low_cube-1), high_cube+1)}) self.env.config.update({'goalPos': self.env.goal}) if self.use_server: From 2ed982f60a376a3f17a32d0f508cdd1f54204387 Mon Sep 17 00:00:00 2001 From: samuelmyoussef Date: Wed, 25 Sep 2024 16:16:16 +0200 Subject: [PATCH 56/85] goal init and update --- .../CartStemContact/CartStemContactEnv.py | 20 ++++++++++++++----- .../CartStemContact/CartStemContactToolbox.py | 7 ++++--- 2 files changed, 19 insertions(+), 8 deletions(-) diff --git a/sofagym/envs/CartStemContact/CartStemContactEnv.py b/sofagym/envs/CartStemContact/CartStemContactEnv.py index 79c665d..1568867 100644 --- a/sofagym/envs/CartStemContact/CartStemContactEnv.py +++ b/sofagym/envs/CartStemContact/CartStemContactEnv.py @@ -67,6 +67,9 @@ def __init__(self, config = None, root=None, use_server: Optional[bool]=False): self.initialize_states() + if self.env.config["goal"]: + self.init_goal() + nb_actions = self.env.config["nb_actions"] low = np.array([-1]*1) high = np.array([1]*1) @@ -104,7 +107,7 @@ def initialize_states(self): self.env.config.update({'goalList': [[x_goal, 0, 20]]}) self.env.config.update({'max_move': max(abs(low_cube-1), high_cube+1)}) - + def randomize_init_states(self): """Randomize initial states. @@ -119,16 +122,23 @@ def randomize_init_states(self): """ return self.env.config["init_states"] + def init_goal(self): + # Set a new random goal from the list + goalList = self.env.config["goalList"] + id_goal = self.env.np_random.choice(range(len(goalList))) + self.env.config.update({'goal_node': id_goal}) + self.env.goal = goalList[id_goal] + self.env.config.update({'goalPos': self.env.goal}) + def reset(self): """Reset simulation. """ self.initialize_states() - self.goalList = self.env.config["goalList"] - - self.env.reset() + if self.env.config["goal"]: + self.init_goal() - self.env.config.update({'goalPos': self.env.goal}) + self.env.reset() if self.use_server: obs = start_scene(self.env.config, self.nb_actions) diff --git a/sofagym/envs/CartStemContact/CartStemContactToolbox.py b/sofagym/envs/CartStemContact/CartStemContactToolbox.py index 462f8ee..1bd5d49 100644 --- a/sofagym/envs/CartStemContact/CartStemContactToolbox.py +++ b/sofagym/envs/CartStemContact/CartStemContactToolbox.py @@ -89,7 +89,7 @@ def getReward(self): return reward, dist - def update(self): + def update(self, goal=None): """Update function. This function is used as an initialization function. @@ -103,6 +103,7 @@ def update(self): None. """ + self.goal = goal current_sphere_pos = self._getSpherePos() self.init_goal_dist = float(abs(current_sphere_pos[0]-self.goal[0])) @@ -119,8 +120,8 @@ def __init__(self, *args, **kwargs): if 'goalPos' in kwargs: self.goalPos = kwargs["goalPos"] - def update(self): - pass + def update(self, goal): + self.goalPos = goal def set_mo_pos(self, goal): pass From 233b5573b1ae9722b5194224c3f5b95090b5dfb6 Mon Sep 17 00:00:00 2001 From: samuelmyoussef Date: Wed, 25 Sep 2024 14:27:29 +0200 Subject: [PATCH 57/85] class instantiation --- sofagym/envs/CartStem/CartStemEnv.py | 59 +++++++++++++++++----------- 1 file changed, 35 insertions(+), 24 deletions(-) diff --git a/sofagym/envs/CartStem/CartStemEnv.py b/sofagym/envs/CartStem/CartStemEnv.py index e46a089..ec8a576 100644 --- a/sofagym/envs/CartStem/CartStemEnv.py +++ b/sofagym/envs/CartStem/CartStemEnv.py @@ -8,14 +8,16 @@ __copyright__ = "(c) 2021, Inria" __date__ = "Feb 3 2021" -from sofagym.AbstractEnv import AbstractEnv +from sofagym.AbstractEnv import AbstractEnv, ServerEnv from sofagym.rpc_server import start_scene from gym import spaces -import os +import os, sys import numpy as np -class CartStemEnv(AbstractEnv): +from typing import Optional + +class CartStemEnv: """Sub-class of AbstractEnv, dedicated to the gripper scene. See the class AbstractEnv for arguments and methods. @@ -27,6 +29,7 @@ class CartStemEnv(AbstractEnv): "deterministic": True, "source": [0, -70, 10], "target": [0, 0, 10], + "goal": False, "goalList": [[7, 0, 20]], "start_node": None, "scale_factor": 10, @@ -41,51 +44,59 @@ class CartStemEnv(AbstractEnv): "planning": False, "discrete": False, "start_from_history": None, - "python_version": "python3.9", + "python_version": sys.version, "zFar": 4000, "time_before_start": 0, "seed": None, "init_x": 0, "max_move": 40, + "randomize_states": False, + "use_server": False } - def __init__(self, config = None): - super().__init__(config) + def __init__(self, config = None, root=None, use_server: Optional[bool]=False): + self.use_server = self.DEFAULT_CONFIG["use_server"] + self.env = ServerEnv(self.DEFAULT_CONFIG, config, root=root) if self.use_server else AbstractEnv(self.DEFAULT_CONFIG, config, root=root) + nb_actions = 2 - self.action_space = spaces.Discrete(nb_actions) + self.env.action_space = spaces.Discrete(nb_actions) self.nb_actions = str(nb_actions) dim_state = 4 low_coordinates = np.array([-100]*dim_state) high_coordinates = np.array([100]*dim_state) - self.observation_space = spaces.Box(low_coordinates, high_coordinates, - dtype='float32') + self.env.observation_space = spaces.Box(low_coordinates, high_coordinates, dtype=np.float32) + + if self.env.root is None and not self.use_server: + self.env.init_root() + # called when an attribute is not found: + def __getattr__(self, name): + # assume it is implemented by self.instance + return self.env.__getattribute__(name) def step(self, action): - obs, reward, done, info = super().step(action) - if abs(obs[0]) > self.config["max_move"]: + obs, reward, done, info = self.env.step(action) + if abs(obs[0]) > self.env.config["max_move"]: done = True return obs, reward, done, info def reset(self): """Reset simulation. - - Note: - ---- - We launch a client to create the scene. The scene of the program is - client_Env.py. - """ - super().reset() + self.env.reset() + + self.env.config.update({'init_x': -(self.env.config["max_move"]/8) + (self.env.config["max_move"]/4)*self.env.np_random.random()}) + self.env.config.update({'goalPos': self.env.goal}) - self.config.update({'init_x': -(self.config["max_move"]/8) + (self.config["max_move"]/4)*np.random.random()}) - self.config.update({'goalPos': self.goal}) + if self.use_server: + obs = start_scene(self.env.config, self.nb_actions) + state = np.array(obs['observation'], dtype=np.float32) + else: + state = np.array(self.env._getState(self.env.root), dtype=np.float32) - obs = start_scene(self.config, self.nb_actions) - - return np.array(obs['observation']) + return state def get_available_actions(self): """Gives the actions available in the environment. @@ -98,6 +109,6 @@ def get_available_actions(self): ------- list of the action available in the environment. """ - return self.action_space + return self.env.action_space From f4028fa32681706a410386ac1979481197aeb586 Mon Sep 17 00:00:00 2001 From: samuelmyoussef Date: Wed, 25 Sep 2024 14:34:31 +0200 Subject: [PATCH 58/85] add some default configs --- sofagym/envs/CartStem/CartStemEnv.py | 9 ++++++--- 1 file changed, 6 insertions(+), 3 deletions(-) diff --git a/sofagym/envs/CartStem/CartStemEnv.py b/sofagym/envs/CartStem/CartStemEnv.py index ec8a576..decc259 100644 --- a/sofagym/envs/CartStem/CartStemEnv.py +++ b/sofagym/envs/CartStem/CartStemEnv.py @@ -23,8 +23,9 @@ class CartStemEnv: See the class AbstractEnv for arguments and methods. """ #Setting a default configuration - path = path = os.path.dirname(os.path.abspath(__file__)) + path = os.path.dirname(os.path.abspath(__file__)) metadata = {'render.modes': ['human', 'rgb_array']} + dim_state = 4 DEFAULT_CONFIG = {"scene": "CartStem", "deterministic": True, "source": [0, -70, 10], @@ -50,6 +51,8 @@ class CartStemEnv: "seed": None, "init_x": 0, "max_move": 40, + "nb_actions": 2, + "dim_state": dim_state, "randomize_states": False, "use_server": False } @@ -58,11 +61,11 @@ def __init__(self, config = None, root=None, use_server: Optional[bool]=False): self.use_server = self.DEFAULT_CONFIG["use_server"] self.env = ServerEnv(self.DEFAULT_CONFIG, config, root=root) if self.use_server else AbstractEnv(self.DEFAULT_CONFIG, config, root=root) - nb_actions = 2 + nb_actions = self.env.config["nb_actions"] self.env.action_space = spaces.Discrete(nb_actions) self.nb_actions = str(nb_actions) - dim_state = 4 + dim_state = self.env.config["dim_state"] low_coordinates = np.array([-100]*dim_state) high_coordinates = np.array([100]*dim_state) self.env.observation_space = spaces.Box(low_coordinates, high_coordinates, dtype=np.float32) From d804d9b07ddc059fea0d065f7c705edc5298b866 Mon Sep 17 00:00:00 2001 From: samuelmyoussef Date: Wed, 25 Sep 2024 14:47:59 +0200 Subject: [PATCH 59/85] initialize states --- sofagym/envs/CartStem/CartStemEnv.py | 29 +++++++++++++++++++++++++++- 1 file changed, 28 insertions(+), 1 deletion(-) diff --git a/sofagym/envs/CartStem/CartStemEnv.py b/sofagym/envs/CartStem/CartStemEnv.py index decc259..8484e47 100644 --- a/sofagym/envs/CartStem/CartStemEnv.py +++ b/sofagym/envs/CartStem/CartStemEnv.py @@ -54,6 +54,7 @@ class CartStemEnv: "nb_actions": 2, "dim_state": dim_state, "randomize_states": False, + "init_states": [0] * dim_state, "use_server": False } @@ -61,6 +62,8 @@ def __init__(self, config = None, root=None, use_server: Optional[bool]=False): self.use_server = self.DEFAULT_CONFIG["use_server"] self.env = ServerEnv(self.DEFAULT_CONFIG, config, root=root) if self.use_server else AbstractEnv(self.DEFAULT_CONFIG, config, root=root) + self.initialize_states() + nb_actions = self.env.config["nb_actions"] self.env.action_space = spaces.Discrete(nb_actions) self.nb_actions = str(nb_actions) @@ -77,6 +80,29 @@ def __init__(self, config = None, root=None, use_server: Optional[bool]=False): def __getattr__(self, name): # assume it is implemented by self.instance return self.env.__getattribute__(name) + + def initialize_states(self): + if self.env.config["randomize_states"]: + self.init_states = self.randomize_init_states() + self.env.config.update({'init_states': list(self.init_states)}) + else: + self.init_states = self.env.config["init_states"] + + self.env.config.update({'init_x': -(self.env.config["max_move"]/8) + (self.env.config["max_move"]/4)*self.env.np_random.random()}) + + def randomize_init_states(self): + """Randomize initial states. + + Returns: + ------- + init_states: list + List of random initial states for the environment. + + Note: + ---- + This method should be implemented according to needed random initialization. + """ + return self.env.config["init_states"] def step(self, action): obs, reward, done, info = self.env.step(action) @@ -88,9 +114,10 @@ def step(self, action): def reset(self): """Reset simulation. """ + self.initialize_states() + self.env.reset() - self.env.config.update({'init_x': -(self.env.config["max_move"]/8) + (self.env.config["max_move"]/4)*self.env.np_random.random()}) self.env.config.update({'goalPos': self.env.goal}) if self.use_server: From 2c3736d81a0128c44130e6d3a3e8fdbab071d5f6 Mon Sep 17 00:00:00 2001 From: samuelmyoussef Date: Wed, 25 Sep 2024 14:56:02 +0200 Subject: [PATCH 60/85] goal remove --- sofagym/envs/CartStem/CartStemEnv.py | 9 ++++++--- sofagym/envs/CartStem/CartStemScene.py | 12 +----------- sofagym/envs/CartStem/CartStemToolbox.py | 13 +------------ 3 files changed, 8 insertions(+), 26 deletions(-) diff --git a/sofagym/envs/CartStem/CartStemEnv.py b/sofagym/envs/CartStem/CartStemEnv.py index 8484e47..b557e71 100644 --- a/sofagym/envs/CartStem/CartStemEnv.py +++ b/sofagym/envs/CartStem/CartStemEnv.py @@ -31,7 +31,6 @@ class CartStemEnv: "source": [0, -70, 10], "target": [0, 0, 10], "goal": False, - "goalList": [[7, 0, 20]], "start_node": None, "scale_factor": 10, "dt": 0.01, @@ -64,6 +63,9 @@ def __init__(self, config = None, root=None, use_server: Optional[bool]=False): self.initialize_states() + if self.env.config["goal"]: + self.init_goal() + nb_actions = self.env.config["nb_actions"] self.env.action_space = spaces.Discrete(nb_actions) self.nb_actions = str(nb_actions) @@ -116,9 +118,10 @@ def reset(self): """ self.initialize_states() - self.env.reset() + if self.env.config["goal"]: + self.init_goal() - self.env.config.update({'goalPos': self.env.goal}) + self.env.reset() if self.use_server: obs = start_scene(self.env.config, self.nb_actions) diff --git a/sofagym/envs/CartStem/CartStemScene.py b/sofagym/envs/CartStem/CartStemScene.py index a6a9fac..4cb3778 100644 --- a/sofagym/envs/CartStem/CartStemScene.py +++ b/sofagym/envs/CartStem/CartStemScene.py @@ -23,20 +23,11 @@ from sofagym.header import addVisu as visu from CartStem import CartStem -from CartStemToolbox import rewardShaper, sceneModerator, applyAction, goalSetter - - -def add_goal_node(root, pos): - goal = root.addChild("Goal") - goal.addObject('VisualStyle', displayFlags="showCollisionModels") - goal_mo = goal.addObject('MechanicalObject', name='GoalMO', showObject=True, drawMode="1", showObjectScale=0.5, - showColor=[1, 0, 0, 0.5], position= pos) - return goal_mo +from CartStemToolbox import rewardShaper, sceneModerator, applyAction def createScene(rootNode, config={"source": [0, -70, 10], "target": [0, 0, 10], - "goalPos": [7, 0, 20], "seed": None, "zFar":4000, "init_x": 0, @@ -62,7 +53,6 @@ def createScene(rootNode, config={"source": [0, -70, 10], cartstem.onEnd(rootNode) cartstem.cart.addObject('ConstantForceField', totalForce=[0, 0, 0, 0, 0, 0]) - rootNode.addObject(goalSetter(name="GoalSetter")) rootNode.addObject(rewardShaper(name="Reward", rootNode=rootNode, max_dist= cartstem_config['max_move'])) rootNode.addObject(sceneModerator(name="sceneModerator", cartstem = cartstem)) rootNode.addObject(applyAction(name="applyAction", root= rootNode, cartstem=cartstem)) diff --git a/sofagym/envs/CartStem/CartStemToolbox.py b/sofagym/envs/CartStem/CartStemToolbox.py index 31c947e..01fd3f5 100644 --- a/sofagym/envs/CartStem/CartStemToolbox.py +++ b/sofagym/envs/CartStem/CartStemToolbox.py @@ -86,7 +86,7 @@ def getReward(self): dist = abs(sphere_pos-cart_pos) return 1, dist - def update(self): + def update(self, goal=None): """Update function. This function is used as an initialization function. @@ -151,17 +151,6 @@ def getState(rootNode): return state -class goalSetter(Sofa.Core.Controller): - def __init__(self, *args, **kwargs): - Sofa.Core.Controller.__init__(self, *args, **kwargs) - - def update(self): - pass - - def set_mo_pos(self, goal): - pass - - def getReward(rootNode): r, dist = rootNode.Reward.getReward() done = dist > 5 From d1d5cec57a76fdd3deb1ff697dfe268f5ff03c4b Mon Sep 17 00:00:00 2001 From: samuelmyoussef Date: Fri, 8 Dec 2023 10:53:51 +0100 Subject: [PATCH 61/85] no server --- sofagym/envs/CartPole/CartPoleEnv.py | 18 ++++++------------ sofagym/envs/CartPole/CartPoleScene.py | 5 +++-- sofagym/envs/CartPole/CartPoleToolbox.py | 19 +++++++++---------- 3 files changed, 18 insertions(+), 24 deletions(-) diff --git a/sofagym/envs/CartPole/CartPoleEnv.py b/sofagym/envs/CartPole/CartPoleEnv.py index 75280b9..9b660e7 100644 --- a/sofagym/envs/CartPole/CartPoleEnv.py +++ b/sofagym/envs/CartPole/CartPoleEnv.py @@ -5,7 +5,6 @@ import numpy as np from gym import spaces from sofagym.AbstractEnv import AbstractEnv -from sofagym.rpc_server import start_scene class CartPoleEnv(AbstractEnv): @@ -22,7 +21,7 @@ class CartPoleEnv(AbstractEnv): "target": [0, 0, 0], "goalList": [[0]], "start_node": None, - "scale_factor": 10, + "scale_factor": 1, "dt": 0.001, "timer_limit": 80, "timeout": 50, @@ -40,10 +39,12 @@ class CartPoleEnv(AbstractEnv): "seed": None, "init_x": 0, "max_move": 24, + "max_angle": 0.418, + "init_states": [0]*4 } - def __init__(self, config=None): - super().__init__(config) + def __init__(self, config=None, root=None): + super().__init__(config, root=root) self.x_threshold = 100 self.theta_threshold_radians = self.config["max_move"] * math.pi / 180 @@ -67,20 +68,13 @@ def __init__(self, config=None): def reset(self): """Reset simulation. - - Note: - ---- - We launch a client to create the scene. The scene of the program is - client_Env.py. - """ super().reset() self.config.update({'goalPos': self.goal}) init_states = self.np_random.uniform(low=-0.05, high=0.05, size=(4,)) self.config.update({'init_states': list(init_states)}) - obs = start_scene(self.config, self.nb_actions) - return np.array(obs['observation']) + return np.array(init_states) def get_available_actions(self): """Gives the actions available in the environment. diff --git a/sofagym/envs/CartPole/CartPoleScene.py b/sofagym/envs/CartPole/CartPoleScene.py index 7d5a58b..ebe358d 100644 --- a/sofagym/envs/CartPole/CartPoleScene.py +++ b/sofagym/envs/CartPole/CartPoleScene.py @@ -48,7 +48,7 @@ def addRigidObject(node, filename, collisionFilename=None, position=[0, 0, 0, 0, return object -def createScene(root, +def createScene(root, config={"source": [0, 0, 160], "target": [0, 0, 0], "goalPos": None, @@ -57,7 +57,8 @@ def createScene(root, "init_x": 0, "max_move": 24, "max_angle": 0.418, - "dt": 0.01}, + "dt": 0.01, + "init_states": [0]*4}, mode='simu_and_visu'): # Choose the mode: visualization or computations (or both) diff --git a/sofagym/envs/CartPole/CartPoleToolbox.py b/sofagym/envs/CartPole/CartPoleToolbox.py index c323dc6..70e8460 100644 --- a/sofagym/envs/CartPole/CartPoleToolbox.py +++ b/sofagym/envs/CartPole/CartPoleToolbox.py @@ -52,13 +52,13 @@ def __init__(self, *args, **kwargs): if kwargs["init_states"] is not None: self.init_states = kwargs["init_states"] else: - print(">> ERROR: no inital states given.") + print(">> ERROR: no initial states given.") exit(1) self.cart = self.rootNode.Modeling.Cart self.pole = self.rootNode.Modeling.Pole - def init_state(self): + def init_state(self, init_states): """Randomly initialize the environment state. Parameters: @@ -70,6 +70,7 @@ def init_state(self): None. """ + self.init_states = init_states cart_pos, cart_vel, pole_theta, pole_theta_dot = self.init_states with self.cart.MechanicalObject.position.writeable() as position: @@ -124,14 +125,11 @@ def __init__(self, *args, **kwargs): self.rootNode = None if kwargs["rootNode"]: self.rootNode = kwargs["rootNode"] - if kwargs["max_angle"]: + if kwargs["max_angle"] is not None: self.max_angle = kwargs["max_angle"] - else: - print(">> ERROR: give a max angle for the normalization of the reward.") - exit(1) - if kwargs["pole_length"]: + if kwargs["pole_length"] is not None: self.pole_length = kwargs["pole_length"] - + self.cart = self.rootNode.Modeling.Cart self.pole = self.rootNode.Modeling.Pole @@ -152,7 +150,7 @@ def getReward(self): return 1, pole_theta, self.max_angle - def update(self): + def update(self, goal): """Update function. This function is used as an initialization function. @@ -218,7 +216,7 @@ class GoalSetter(Sofa.Core.Controller): def __init__(self, *args, **kwargs): Sofa.Core.Controller.__init__(self, *args, **kwargs) - def update(self): + def update(self, goal): pass def set_mo_pos(self, goal): @@ -330,6 +328,7 @@ def startCmd_CartPole(rootNode, incr, duration): """ # Definition of the elements of the animation + def executeAnimation(rootNode, incr, factor): rootNode.ApplyAction.apply_action(incr) From 941b1f0415ed905124175773b7ed83c5d1e0c87e Mon Sep 17 00:00:00 2001 From: samuelmyoussef Date: Mon, 23 Sep 2024 16:37:24 +0200 Subject: [PATCH 62/85] class instantiation --- sofagym/envs/CartPole/CartPoleEnv.py | 46 +++++++++++++++++++--------- 1 file changed, 31 insertions(+), 15 deletions(-) diff --git a/sofagym/envs/CartPole/CartPoleEnv.py b/sofagym/envs/CartPole/CartPoleEnv.py index 9b660e7..ac64418 100644 --- a/sofagym/envs/CartPole/CartPoleEnv.py +++ b/sofagym/envs/CartPole/CartPoleEnv.py @@ -1,13 +1,15 @@ import math import os import sys +from typing import Optional import numpy as np from gym import spaces -from sofagym.AbstractEnv import AbstractEnv +from sofagym.AbstractEnv import AbstractEnv, ServerEnv +from sofagym.rpc_server import start_scene -class CartPoleEnv(AbstractEnv): +class CartPoleEnv: """Sub-class of AbstractEnv, dedicated to the cart pole scene. See the class AbstractEnv for arguments and methods. @@ -19,9 +21,10 @@ class CartPoleEnv(AbstractEnv): "deterministic": True, "source": [0, 0, 160], "target": [0, 0, 0], + "goal": False, "goalList": [[0]], "start_node": None, - "scale_factor": 1, + "scale_factor": 10, "dt": 0.001, "timer_limit": 80, "timeout": 50, @@ -40,15 +43,18 @@ class CartPoleEnv(AbstractEnv): "init_x": 0, "max_move": 24, "max_angle": 0.418, - "init_states": [0]*4 + "randomize_states": True, + "init_states": [0]*4, + "use_server": False } - def __init__(self, config=None, root=None): - super().__init__(config, root=root) + def __init__(self, config=None, root=None, use_server: Optional[bool]=False): + self.use_server = self.DEFAULT_CONFIG["use_server"] + self.env = ServerEnv(self.DEFAULT_CONFIG, config, root=root) if self.use_server else AbstractEnv(self.DEFAULT_CONFIG, config, root=root) self.x_threshold = 100 - self.theta_threshold_radians = self.config["max_move"] * math.pi / 180 - self.config.update({'max_angle': self.theta_threshold_radians}) + self.theta_threshold_radians = self.env.config["max_move"] * math.pi / 180 + self.env.config.update({'max_angle': self.theta_threshold_radians}) high = np.array( [ @@ -61,19 +67,29 @@ def __init__(self, config=None, root=None): ) nb_actions = 2 - self.action_space = spaces.Discrete(nb_actions) + self.env.action_space = spaces.Discrete(nb_actions) self.nb_actions = str(nb_actions) - self.observation_space = spaces.Box(-high, high, dtype=np.float32) + self.env.observation_space = spaces.Box(-high, high, dtype=np.float32) + + # called when an attribute is not found: + def __getattr__(self, name): + # assume it is implemented by self.instance + return self.env.__getattribute__(name) def reset(self): """Reset simulation. """ - super().reset() + self.env.reset() - self.config.update({'goalPos': self.goal}) - init_states = self.np_random.uniform(low=-0.05, high=0.05, size=(4,)) - self.config.update({'init_states': list(init_states)}) + self.env.config.update({'goalPos': self.env.goal}) + init_states = self.env.np_random.uniform(low=-0.05, high=0.05, size=(4,)) + self.env.config.update({'init_states': list(init_states)}) + + if self.use_server: + obs = start_scene(self.env.config, self.nb_actions) + return np.array(obs['observation']) + return np.array(init_states) def get_available_actions(self): @@ -87,4 +103,4 @@ def get_available_actions(self): ------- list of the action available in the environment. """ - return self.action_space + return self.env.action_space From ec12b771e767285e426c24d01e1f60257e73f8cd Mon Sep 17 00:00:00 2001 From: samuelmyoussef Date: Thu, 26 Sep 2024 05:29:44 +0200 Subject: [PATCH 63/85] remove non-overriden methods --- sofagym/envs/Diamond/DiamondEnv.py | 48 +++++------------------------- 1 file changed, 7 insertions(+), 41 deletions(-) diff --git a/sofagym/envs/Diamond/DiamondEnv.py b/sofagym/envs/Diamond/DiamondEnv.py index 835a5a5..81751c3 100644 --- a/sofagym/envs/Diamond/DiamondEnv.py +++ b/sofagym/envs/Diamond/DiamondEnv.py @@ -57,7 +57,9 @@ class DiamondRobotEnv: "use_server": False } - def __init__(self, config = None, root=None, use_server: Optional[bool]=False): + def __init__(self, config = None, root=None, use_server: Optional[bool]=None): + if use_server is not None: + self.DEFAULT_CONFIG.update({'use_server': use_server}) self.use_server = self.DEFAULT_CONFIG["use_server"] self.env = ServerEnv(self.DEFAULT_CONFIG, config, root=root) if self.use_server else AbstractEnv(self.DEFAULT_CONFIG, config, root=root) @@ -66,13 +68,11 @@ def __init__(self, config = None, root=None, use_server: Optional[bool]=False): if self.env.config["goal"]: self.init_goal() - nb_actions = self.env.config["nb_actions"] - self.env.action_space = spaces.Discrete(nb_actions) - self.nb_actions = str(nb_actions) + self.env.action_space = spaces.Discrete(self.env.nb_actions) + self.nb_actions = str(self.env.nb_actions) - dim_state = self.env.config["dim_state"] - low_coordinates = np.ones(shape=dim_state)*-1 - high_coordinates = np.ones(shape=dim_state) + low_coordinates = np.ones(shape=self.env.dim_state)*-1 + high_coordinates = np.ones(shape=self.env.dim_state) self.env.observation_space = spaces.Box(low_coordinates, high_coordinates, dtype=np.float32) if self.env.root is None and not self.use_server: @@ -82,27 +82,6 @@ def __init__(self, config = None, root=None, use_server: Optional[bool]=False): def __getattr__(self, name): # assume it is implemented by self.instance return self.env.__getattribute__(name) - - def initialize_states(self): - if self.env.config["randomize_states"]: - self.init_states = self.randomize_init_states() - self.env.config.update({'init_states': list(self.init_states)}) - else: - self.init_states = self.env.config["init_states"] - - def randomize_init_states(self): - """Randomize initial states. - - Returns: - ------- - init_states: list - List of random initial states for the environment. - - Note: - ---- - This method should be implemented according to needed random initialization. - """ - return self.env.config["init_states"] def init_goal(self): # Set a new random goal from the list @@ -160,16 +139,3 @@ def render(self, mode='rgb_array'): self.env.viewer.render() else: self.env.render(mode) - - def get_available_actions(self): - """Gives the actions available in the environment. - - Parameters: - ---------- - None. - - Returns: - ------- - list of the action available in the environment. - """ - return self.env.action_space From 3a53e59ef9ec6aaa8a30f89807dbe93a78eeb98e Mon Sep 17 00:00:00 2001 From: samuelmyoussef Date: Thu, 26 Sep 2024 06:32:31 +0200 Subject: [PATCH 64/85] remove legacy viewer --- sofagym/envs/Diamond/DiamondEnv.py | 34 -------------------------- sofagym/envs/Diamond/DiamondToolbox.py | 4 +-- 2 files changed, 2 insertions(+), 36 deletions(-) diff --git a/sofagym/envs/Diamond/DiamondEnv.py b/sofagym/envs/Diamond/DiamondEnv.py index 81751c3..86c92f1 100644 --- a/sofagym/envs/Diamond/DiamondEnv.py +++ b/sofagym/envs/Diamond/DiamondEnv.py @@ -88,13 +88,6 @@ def init_goal(self): self.env.goal = [-30 + 60 * self.env.np_random.random(), -30 + 60 * self.env.np_random.random(), 125 + 20 * self.env.np_random.random()] self.env.config.update({'goalPos': self.env.goal}) - def step(self, action): - if self.use_server: - if self.env.viewer: - self.env.viewer.step(action) - - return self.env.step(action) - def reset(self): """Reset simulation. """ @@ -107,35 +100,8 @@ def reset(self): if self.use_server: obs = start_scene(self.env.config, self.nb_actions) - if self.env.viewer: - self.env.viewer.reset() state = np.array(obs['observation'], dtype=np.float32) else: state = np.array(self.env._getState(self.env.root), dtype=np.float32) return state - - def render(self, mode='rgb_array'): - """See the current state of the environment. - - Get the OpenGL Context to render an image (snapshot) of the simulation - state. - - Parameters: - ---------- - mode: string, default = 'rgb_array' - Type of representation. - - Returns: - ------- - None. - """ - if self.use_server: - if not self.env.viewer: - display_size = self.env.config["display_size"] # Sim display - self.env.viewer = LegacyViewer(self, display_size, startCmd=startCmd) - - # Use the viewer to display the environment. - self.env.viewer.render() - else: - self.env.render(mode) diff --git a/sofagym/envs/Diamond/DiamondToolbox.py b/sofagym/envs/Diamond/DiamondToolbox.py index aeb3921..7147706 100644 --- a/sofagym/envs/Diamond/DiamondToolbox.py +++ b/sofagym/envs/Diamond/DiamondToolbox.py @@ -347,7 +347,7 @@ def getPos(root): _: list The position(s) of the object(s) of the scene. """ - return + return root.Robot.tetras.position.value.tolist() def setPos(root, pos): @@ -369,4 +369,4 @@ def setPos(root, pos): Don't forget to init the new value of the position. """ - return + root.Robot.tetras.position.value = np.array(pos) From 346fb5badec653ba630e3ebf199da79fe6c13c19 Mon Sep 17 00:00:00 2001 From: samuelmyoussef Date: Thu, 26 Sep 2024 05:31:14 +0200 Subject: [PATCH 65/85] remove non-overriden methods --- sofagym/envs/CTR/CTREnv.py | 48 ++++++-------------------------------- 1 file changed, 7 insertions(+), 41 deletions(-) diff --git a/sofagym/envs/CTR/CTREnv.py b/sofagym/envs/CTR/CTREnv.py index 703078e..4688cd7 100644 --- a/sofagym/envs/CTR/CTREnv.py +++ b/sofagym/envs/CTR/CTREnv.py @@ -61,7 +61,9 @@ class ConcentricTubeRobotEnv: "use_server": False } - def __init__(self, config = None, root=None, use_server: Optional[bool]=False): + def __init__(self, config = None, root=None, use_server: Optional[bool]=None): + if use_server is not None: + self.DEFAULT_CONFIG.update({'use_server': use_server}) self.use_server = self.DEFAULT_CONFIG["use_server"] self.env = ServerEnv(self.DEFAULT_CONFIG, config, root=root) if self.use_server else AbstractEnv(self.DEFAULT_CONFIG, config, root=root) @@ -70,13 +72,11 @@ def __init__(self, config = None, root=None, use_server: Optional[bool]=False): if self.env.config["goal"]: self.init_goal() - nb_actions = self.env.config["nb_actions"] - self.env.action_space = spaces.Discrete(nb_actions) - self.nb_actions = str(nb_actions) + self.env.action_space = spaces.Discrete(self.env.nb_actions) + self.nb_actions = str(self.env.nb_actions) - dim_state = self.env.config["dim_state"] - low_coordinates = np.array([-1]*dim_state) - high_coordinates = np.array([1]*dim_state) + low_coordinates = np.array([-1]*self.env.dim_state) + high_coordinates = np.array([1]*self.env.dim_state) self.env.observation_space = spaces.Box(low_coordinates, high_coordinates, dtype=np.float32) self.default_action = 3 @@ -88,27 +88,6 @@ def __init__(self, config = None, root=None, use_server: Optional[bool]=False): def __getattr__(self, name): # assume it is implemented by self.instance return self.env.__getattribute__(name) - - def initialize_states(self): - if self.env.config["randomize_states"]: - self.init_states = self.randomize_init_states() - self.env.config.update({'init_states': list(self.init_states)}) - else: - self.init_states = self.env.config["init_states"] - - def randomize_init_states(self): - """Randomize initial states. - - Returns: - ------- - init_states: list - List of random initial states for the environment. - - Note: - ---- - This method should be implemented according to needed random initialization. - """ - return self.env.config["init_states"] def init_goal(self): # Set a new random goal from the list @@ -171,16 +150,3 @@ def render(self, mode='rgb_array'): self.env.viewer.render() else: self.env.render(mode) - - def get_available_actions(self): - """Gives the actions available in the environment. - - Parameters: - ---------- - None. - - Returns: - ------- - list of the action available in the environment. - """ - return self.env.action_space From 1bc7b25cfc6c9c6ab3a69b3a633b7c886cdbc4f6 Mon Sep 17 00:00:00 2001 From: samuelmyoussef Date: Thu, 26 Sep 2024 06:18:33 +0200 Subject: [PATCH 66/85] remove legacy viewer --- sofagym/envs/CTR/CTREnv.py | 38 ---------------------------------- sofagym/envs/CTR/CTRToolbox.py | 27 ++++++++++++++++++++++-- 2 files changed, 25 insertions(+), 40 deletions(-) diff --git a/sofagym/envs/CTR/CTREnv.py b/sofagym/envs/CTR/CTREnv.py index 4688cd7..0f175fb 100644 --- a/sofagym/envs/CTR/CTREnv.py +++ b/sofagym/envs/CTR/CTREnv.py @@ -95,13 +95,6 @@ def init_goal(self): self.env.goal = [0.0, y, abs(y) + 30 * self.env.np_random.random()] self.env.config.update({'goalPos': self.env.goal}) - def step(self, action): - if self.use_server: - if self.env.viewer: - self.env.viewer.step(action) - - return self.env.step(action) - def reset(self): """Reset simulation. """ @@ -114,39 +107,8 @@ def reset(self): if self.use_server: obs = start_scene(self.env.config, self.nb_actions) - if self.env.viewer: - self.env.viewer.reset() - - self.env.step(0) - self.env.step(4) - self.env.step(8) state = np.array(obs['observation'], dtype=np.float32) else: state = np.array(self.env._getState(self.env.root), dtype=np.float32) return state - - def render(self, mode='rgb_array'): - """See the current state of the environment. - - Get the OpenGL Context to render an image (snapshot) of the simulation - state. - - Parameters: - ---------- - mode: string, default = 'rgb_array' - Type of representation. - - Returns: - ------- - None. - """ - if self.use_server: - if not self.env.viewer: - display_size = self.env.config["display_size"] # Sim display - self.env.viewer = LegacyViewer(self, display_size, startCmd=startCmd) - - # Use the viewer to display the environment. - self.env.viewer.render() - else: - self.env.render(mode) diff --git a/sofagym/envs/CTR/CTRToolbox.py b/sofagym/envs/CTR/CTRToolbox.py index 6f988a4..5dbfc86 100644 --- a/sofagym/envs/CTR/CTRToolbox.py +++ b/sofagym/envs/CTR/CTRToolbox.py @@ -378,7 +378,17 @@ def getPos(root): _: list The position(s) of the object(s) of the scene. """ - return + cath_xtip = root.InstrumentCombined.m_ircontroller.xtip.value[0].tolist() + cath_rotation = root.InstrumentCombined.m_ircontroller.rotationInstrument.value[0].tolist() + guide_xtip = root.InstrumentCombined.m_ircontroller.xtip.value[1].tolist() + guide_rotation = root.InstrumentCombined.m_ircontroller.rotationInstrument.value[1].tolist() + coils_xtip = root.InstrumentCombined.m_ircontroller.xtip.value[2].tolist() + coils_rotation = root.InstrumentCombined.m_ircontroller.rotationInstrument.value[2].tolist() + + tip = root.InstrumentCombined.DOFs.position.value.tolist() + collis = root.InstrumentCombined.Collis.CollisionDOFs.position.value.tolist() + + return [cath_xtip, cath_rotation, guide_xtip, guide_rotation, coils_xtip, coils_rotation, tip, collis] def setPos(root, pos): @@ -400,4 +410,17 @@ def setPos(root, pos): Don't forget to init the new value of the position. """ - return + cath_xtip, cath_rotation, guide_xtip, guide_rotation, coils_xtip, coils_rotation, tip, collis = pos + + controller = root.InstrumentCombined.m_ircontroller + with controller.xtip.writeable() as xtip: + xtip[0] = np.array(cath_xtip) + xtip[1] = np.array(guide_xtip) + xtip[2] = np.array(coils_xtip) + + with controller.rotationInstrument.writeable() as rotation: + rotation[0] = np.array(cath_rotation) + rotation[1] = np.array(guide_rotation) + rotation[2] = np.array(coils_rotation) + + root.InstrumentCombined.DOFs.position.value = np.array(tip) From fb363c2edd963e35b45c6807594884841d6ef033 Mon Sep 17 00:00:00 2001 From: samuelmyoussef Date: Thu, 26 Sep 2024 05:41:03 +0200 Subject: [PATCH 67/85] remove non-overrinden methods --- sofagym/envs/CartStem/CartStemEnv.py | 51 +++++----------------------- 1 file changed, 9 insertions(+), 42 deletions(-) diff --git a/sofagym/envs/CartStem/CartStemEnv.py b/sofagym/envs/CartStem/CartStemEnv.py index b557e71..ece07f8 100644 --- a/sofagym/envs/CartStem/CartStemEnv.py +++ b/sofagym/envs/CartStem/CartStemEnv.py @@ -57,7 +57,9 @@ class CartStemEnv: "use_server": False } - def __init__(self, config = None, root=None, use_server: Optional[bool]=False): + def __init__(self, config = None, root=None, use_server: Optional[bool]=None): + if use_server is not None: + self.DEFAULT_CONFIG.update({'use_server': use_server}) self.use_server = self.DEFAULT_CONFIG["use_server"] self.env = ServerEnv(self.DEFAULT_CONFIG, config, root=root) if self.use_server else AbstractEnv(self.DEFAULT_CONFIG, config, root=root) @@ -66,13 +68,11 @@ def __init__(self, config = None, root=None, use_server: Optional[bool]=False): if self.env.config["goal"]: self.init_goal() - nb_actions = self.env.config["nb_actions"] - self.env.action_space = spaces.Discrete(nb_actions) - self.nb_actions = str(nb_actions) + self.env.action_space = spaces.Discrete(self.env.nb_actions) + self.nb_actions = str(self.env.nb_actions) - dim_state = self.env.config["dim_state"] - low_coordinates = np.array([-100]*dim_state) - high_coordinates = np.array([100]*dim_state) + low_coordinates = np.array([-100]*self.env.dim_state) + high_coordinates = np.array([100]*self.env.dim_state) self.env.observation_space = spaces.Box(low_coordinates, high_coordinates, dtype=np.float32) if self.env.root is None and not self.use_server: @@ -84,27 +84,9 @@ def __getattr__(self, name): return self.env.__getattribute__(name) def initialize_states(self): - if self.env.config["randomize_states"]: - self.init_states = self.randomize_init_states() - self.env.config.update({'init_states': list(self.init_states)}) - else: - self.init_states = self.env.config["init_states"] - + self.env.initialize_states() + self.env.config.update({'init_x': -(self.env.config["max_move"]/8) + (self.env.config["max_move"]/4)*self.env.np_random.random()}) - - def randomize_init_states(self): - """Randomize initial states. - - Returns: - ------- - init_states: list - List of random initial states for the environment. - - Note: - ---- - This method should be implemented according to needed random initialization. - """ - return self.env.config["init_states"] def step(self, action): obs, reward, done, info = self.env.step(action) @@ -130,18 +112,3 @@ def reset(self): state = np.array(self.env._getState(self.env.root), dtype=np.float32) return state - - def get_available_actions(self): - """Gives the actions available in the environment. - - Parameters: - ---------- - None. - - Returns: - ------- - list of the action available in the environment. - """ - return self.env.action_space - - From cef887f457b6c42046e250936dce55169063e272 Mon Sep 17 00:00:00 2001 From: samuelmyoussef Date: Thu, 26 Sep 2024 05:38:55 +0200 Subject: [PATCH 68/85] remove non-overriden methods --- .../CartStemContact/CartStemContactEnv.py | 57 +++---------------- 1 file changed, 8 insertions(+), 49 deletions(-) diff --git a/sofagym/envs/CartStemContact/CartStemContactEnv.py b/sofagym/envs/CartStemContact/CartStemContactEnv.py index 1568867..3eebd57 100644 --- a/sofagym/envs/CartStemContact/CartStemContactEnv.py +++ b/sofagym/envs/CartStemContact/CartStemContactEnv.py @@ -61,7 +61,9 @@ class CartStemContactEnv: "use_server": False } - def __init__(self, config = None, root=None, use_server: Optional[bool]=False): + def __init__(self, config = None, root=None, use_server: Optional[bool]=None): + if use_server is not None: + self.DEFAULT_CONFIG.update({'use_server': use_server}) self.use_server = self.DEFAULT_CONFIG["use_server"] self.env = ServerEnv(self.DEFAULT_CONFIG, config, root=root) if self.use_server else AbstractEnv(self.DEFAULT_CONFIG, config, root=root) @@ -70,15 +72,13 @@ def __init__(self, config = None, root=None, use_server: Optional[bool]=False): if self.env.config["goal"]: self.init_goal() - nb_actions = self.env.config["nb_actions"] low = np.array([-1]*1) high = np.array([1]*1) self.env.action_space = spaces.Box(low=low, high=high, shape=(1,), dtype=np.float32) - self.nb_actions = str(nb_actions) + self.nb_actions = str(self.env.nb_actions) - dim_state = self.env.config["dim_state"] - low_coordinates = np.array([-1]*dim_state) - high_coordinates = np.array([1]*dim_state) + low_coordinates = np.array([-1]*self.env.dim_state) + high_coordinates = np.array([1]*self.env.dim_state) self.env.observation_space = spaces.Box(low_coordinates, high_coordinates, dtype=np.float32) if self.env.root is None and not self.use_server: @@ -90,12 +90,8 @@ def __getattr__(self, name): return self.env.__getattribute__(name) def initialize_states(self): - if self.env.config["randomize_states"]: - self.init_states = self.randomize_init_states() - self.env.config.update({'init_states': list(self.init_states)}) - else: - self.init_states = self.env.config["init_states"] - + self.env.initialize_states() + low_cube, high_cube = -6+ 2*self.env.np_random.random(), 6 - 2*self.env.np_random.random() self.env.config.update({'cube_x': [low_cube, high_cube]}) self.env.config.update({'init_x': (low_cube + 3) + (high_cube-low_cube-3)*self.env.np_random.random()}) @@ -108,28 +104,6 @@ def initialize_states(self): self.env.config.update({'goalList': [[x_goal, 0, 20]]}) self.env.config.update({'max_move': max(abs(low_cube-1), high_cube+1)}) - def randomize_init_states(self): - """Randomize initial states. - - Returns: - ------- - init_states: list - List of random initial states for the environment. - - Note: - ---- - This method should be implemented according to needed random initialization. - """ - return self.env.config["init_states"] - - def init_goal(self): - # Set a new random goal from the list - goalList = self.env.config["goalList"] - id_goal = self.env.np_random.choice(range(len(goalList))) - self.env.config.update({'goal_node': id_goal}) - self.env.goal = goalList[id_goal] - self.env.config.update({'goalPos': self.env.goal}) - def reset(self): """Reset simulation. """ @@ -147,18 +121,3 @@ def reset(self): state = np.array(self.env._getState(self.env.root), dtype=np.float32) return state - - def get_available_actions(self): - """Gives the actions available in the environment. - - Parameters: - ---------- - None. - - Returns: - ------- - list of the action available in the environment. - """ - return self.env.action_space - - From 20fb38f8affbe9d894add2b38ddc4f723b89677b Mon Sep 17 00:00:00 2001 From: samuelmyoussef Date: Thu, 26 Sep 2024 05:35:57 +0200 Subject: [PATCH 69/85] remove non-overriden methods --- .../envs/CatchTheObject/CatchTheObjectEnv.py | 57 ++----------------- 1 file changed, 6 insertions(+), 51 deletions(-) diff --git a/sofagym/envs/CatchTheObject/CatchTheObjectEnv.py b/sofagym/envs/CatchTheObject/CatchTheObjectEnv.py index 8fd2e47..ef6bbc3 100644 --- a/sofagym/envs/CatchTheObject/CatchTheObjectEnv.py +++ b/sofagym/envs/CatchTheObject/CatchTheObjectEnv.py @@ -60,7 +60,9 @@ class CatchTheObject: "use_server": False } - def __init__(self, config = None, root=None, use_server: Optional[bool]=False): + def __init__(self, config = None, root=None, use_server: Optional[bool]=None): + if use_server is not None: + self.DEFAULT_CONFIG.update({'use_server': use_server}) self.use_server = self.DEFAULT_CONFIG["use_server"] self.env = ServerEnv(self.DEFAULT_CONFIG, config, root=root) if self.use_server else AbstractEnv(self.DEFAULT_CONFIG, config, root=root) @@ -69,15 +71,13 @@ def __init__(self, config = None, root=None, use_server: Optional[bool]=False): if self.env.config["goal"]: self.init_goal() - nb_actions = self.env.config["nb_actions"] low = np.array([-1]*1) high = np.array([1]*1) self.env.action_space = spaces.Box(low=low, high=high, shape=(1,), dtype=np.float32) - self.nb_actions = str(nb_actions) + self.nb_actions = str(self.env.nb_actions) - dim_state = self.env.config["dim_state"] - low_coordinates = np.array([-1]*dim_state) - high_coordinates = np.array([1]*dim_state) + low_coordinates = np.array([-1]*self.env.dim_state) + high_coordinates = np.array([1]*self.env.dim_state) self.env.observation_space = spaces.Box(low_coordinates, high_coordinates, dtype=np.float32) if self.env.root is None and not self.use_server: @@ -88,35 +88,6 @@ def __getattr__(self, name): # assume it is implemented by self.instance return self.env.__getattribute__(name) - def initialize_states(self): - if self.env.config["randomize_states"]: - self.init_states = self.randomize_init_states() - self.env.config.update({'init_states': list(self.init_states)}) - else: - self.init_states = self.env.config["init_states"] - - def randomize_init_states(self): - """Randomize initial states. - - Returns: - ------- - init_states: list - List of random initial states for the environment. - - Note: - ---- - This method should be implemented according to needed random initialization. - """ - return self.env.config["init_states"] - - def init_goal(self): - # Set a new random goal from the list - goalList = self.env.config["goalList"] - id_goal = self.env.np_random.choice(range(len(goalList))) - self.env.config.update({'goal_node': id_goal}) - self.env.goal = goalList[id_goal] - self.env.config.update({'goalPos': self.env.goal}) - def reset(self): """Reset simulation. """ @@ -134,19 +105,3 @@ def reset(self): state = np.array(self.env._getState(self.env.root), dtype=np.float32) return state - - def get_available_actions(self): - """Gives the actions available in the environment. - - Parameters: - ---------- - None. - - Returns: - ------- - list of the action available in the environment. - """ - return self.env.action_space - - - From ba90431383ef4654e4c140ca05dae051e75c84b0 Mon Sep 17 00:00:00 2001 From: samuelmyoussef Date: Thu, 26 Sep 2024 05:33:10 +0200 Subject: [PATCH 70/85] remove non-overriden methods --- sofagym/envs/CatheterBeam/CatheterBeamEnv.py | 56 +++----------------- 1 file changed, 7 insertions(+), 49 deletions(-) diff --git a/sofagym/envs/CatheterBeam/CatheterBeamEnv.py b/sofagym/envs/CatheterBeam/CatheterBeamEnv.py index efb1f8b..1400b14 100644 --- a/sofagym/envs/CatheterBeam/CatheterBeamEnv.py +++ b/sofagym/envs/CatheterBeam/CatheterBeamEnv.py @@ -51,7 +51,9 @@ class CatheterBeamEnv: "use_server": False } - def __init__(self, config = None, root=None, use_server: Optional[bool]=False): + def __init__(self, config = None, root=None, use_server: Optional[bool]=None): + if use_server is not None: + self.DEFAULT_CONFIG.update({'use_server': use_server}) self.use_server = self.DEFAULT_CONFIG["use_server"] self.env = ServerEnv(self.DEFAULT_CONFIG, config, root=root) if self.use_server else AbstractEnv(self.DEFAULT_CONFIG, config, root=root) @@ -60,13 +62,11 @@ def __init__(self, config = None, root=None, use_server: Optional[bool]=False): if self.env.config["goal"]: self.init_goal() - nb_actions = self.env.config["nb_actions"] - self.env.action_space = spaces.Discrete(nb_actions) - self.nb_actions = str(nb_actions) + self.env.action_space = spaces.Discrete(self.env.nb_actions) + self.nb_actions = str(self.env.nb_actions) - dim_state = self.env.config["dim_state"] - low_coordinates = np.array([-1]*dim_state) - high_coordinates = np.array([1]*dim_state) + low_coordinates = np.array([-1]*self.env.dim_state) + high_coordinates = np.array([1]*self.env.dim_state) self.env.observation_space = spaces.Box(low_coordinates, high_coordinates, dtype=np.float32) if self.env.root is None and not self.use_server: @@ -77,35 +77,6 @@ def __getattr__(self, name): # assume it is implemented by self.instance return self.env.__getattribute__(name) - def initialize_states(self): - if self.env.config["randomize_states"]: - self.init_states = self.randomize_init_states() - self.env.config.update({'init_states': list(self.init_states)}) - else: - self.init_states = self.env.config["init_states"] - - def randomize_init_states(self): - """Randomize initial states. - - Returns: - ------- - init_states: list - List of random initial states for the environment. - - Note: - ---- - This method should be implemented according to needed random initialization. - """ - return self.env.config["init_states"] - - def init_goal(self): - # Set a new random goal from the list - goalList = self.env.config["goalList"] - id_goal = self.env.np_random.choice(range(len(goalList))) - self.env.config.update({'goal_node': id_goal}) - self.env.goal = goalList[id_goal] - self.env.config.update({'goalPos': self.env.goal}) - def reset(self): """Reset simulation. """ @@ -123,16 +94,3 @@ def reset(self): state = np.array(self.env._getState(self.env.root), dtype=np.float32) return state - - def get_available_actions(self): - """Gives the actions available in the environment. - - Parameters: - ---------- - None. - - Returns: - ------- - list of the action available in the environment. - """ - return self.env.action_space From 5795fb73c21f9ae07cb9fcf2c260d73db83af0d0 Mon Sep 17 00:00:00 2001 From: samuelmyoussef Date: Thu, 26 Sep 2024 05:26:39 +0200 Subject: [PATCH 71/85] remove non-overriden methods --- sofagym/envs/Gripper/GripperEnv.py | 56 ++++-------------------------- 1 file changed, 7 insertions(+), 49 deletions(-) diff --git a/sofagym/envs/Gripper/GripperEnv.py b/sofagym/envs/Gripper/GripperEnv.py index 35e0d56..beb18cc 100644 --- a/sofagym/envs/Gripper/GripperEnv.py +++ b/sofagym/envs/Gripper/GripperEnv.py @@ -55,7 +55,9 @@ class GripperEnv: "use_server": False } - def __init__(self, config = None, root=None, use_server: Optional[bool]=False): + def __init__(self, config = None, root=None, use_server: Optional[bool]=None): + if use_server is not None: + self.DEFAULT_CONFIG.update({'use_server': use_server}) self.use_server = self.DEFAULT_CONFIG["use_server"] self.env = ServerEnv(self.DEFAULT_CONFIG, config, root=root) if self.use_server else AbstractEnv(self.DEFAULT_CONFIG, config, root=root) @@ -64,13 +66,11 @@ def __init__(self, config = None, root=None, use_server: Optional[bool]=False): if self.env.config["goal"]: self.init_goal() - nb_actions = self.env.config["nb_actions"] - self.env.action_space = spaces.Discrete(nb_actions) - self.nb_actions = str(nb_actions) + self.env.action_space = spaces.Discrete(self.env.nb_actions) + self.nb_actions = str(self.env.nb_actions) - dim_state = self.env.config["dim_state"] - low_coordinates = np.array([-1]*dim_state) - high_coordinates = np.array([1]*dim_state) + low_coordinates = np.array([-1]*self.env.dim_state) + high_coordinates = np.array([1]*self.env.dim_state) self.env.observation_space = spaces.Box(low_coordinates, high_coordinates, dtype=np.float32) if self.env.root is None and not self.use_server: @@ -81,35 +81,6 @@ def __getattr__(self, name): # assume it is implemented by self.instance return self.env.__getattribute__(name) - def initialize_states(self): - if self.env.config["randomize_states"]: - self.init_states = self.randomize_init_states() - self.env.config.update({'init_states': list(self.init_states)}) - else: - self.init_states = self.env.config["init_states"] - - def randomize_init_states(self): - """Randomize initial states. - - Returns: - ------- - init_states: list - List of random initial states for the environment. - - Note: - ---- - This method should be implemented according to needed random initialization. - """ - return self.env.config["init_states"] - - def init_goal(self): - # Set a new random goal from the list - goalList = self.env.config["goalList"] - id_goal = self.env.np_random.choice(range(len(goalList))) - self.env.config.update({'goal_node': id_goal}) - self.env.goal = goalList[id_goal] - self.env.config.update({'goalPos': self.env.goal}) - def reset(self): """Reset simulation. """ @@ -127,16 +98,3 @@ def reset(self): state = np.array(self.env._getState(self.env.root), dtype=np.float32) return state - - def get_available_actions(self): - """Gives the actions available in the environment. - - Parameters: - ---------- - None. - - Returns: - ------- - list of the action available in the environment. - """ - return self.env.action_space From 66d7458bac6dfb4d46c03e8338a4b1944dd8d19f Mon Sep 17 00:00:00 2001 From: samuelmyoussef Date: Thu, 26 Sep 2024 05:24:31 +0200 Subject: [PATCH 72/85] remove non-overriden methods --- sofagym/envs/Maze/MazeEnv.py | 56 +++++------------------------------- 1 file changed, 7 insertions(+), 49 deletions(-) diff --git a/sofagym/envs/Maze/MazeEnv.py b/sofagym/envs/Maze/MazeEnv.py index dcb7bb6..0d8df3a 100644 --- a/sofagym/envs/Maze/MazeEnv.py +++ b/sofagym/envs/Maze/MazeEnv.py @@ -58,7 +58,9 @@ class MazeEnv: "use_server": False } - def __init__(self, config = None, root=None, use_server: Optional[bool]=False): + def __init__(self, config = None, root=None, use_server: Optional[bool]=None): + if use_server is not None: + self.DEFAULT_CONFIG.update({'use_server': use_server}) self.use_server = self.DEFAULT_CONFIG["use_server"] self.env = ServerEnv(self.DEFAULT_CONFIG, config, root=root) if self.use_server else AbstractEnv(self.DEFAULT_CONFIG, config, root=root) @@ -67,13 +69,11 @@ def __init__(self, config = None, root=None, use_server: Optional[bool]=False): if self.env.config["goal"]: self.init_goal() - nb_actions = self.env.config["nb_actions"] - self.env.action_space = spaces.Discrete(nb_actions) - self.nb_actions = str(nb_actions) + self.env.action_space = spaces.Discrete(self.env.nb_actions) + self.nb_actions = str(self.env.nb_actions) - dim_state = self.env.config["dim_state"] - low_coordinates = np.array([-1]*dim_state) - high_coordinates = np.array([1]*dim_state) + low_coordinates = np.array([-1]*self.env.dim_state) + high_coordinates = np.array([1]*self.env.dim_state) self.env.observation_space = spaces.Box(low_coordinates, high_coordinates, dtype=np.float32) if self.env.root is None and not self.use_server: @@ -84,35 +84,6 @@ def __getattr__(self, name): # assume it is implemented by self.instance return self.env.__getattribute__(name) - def initialize_states(self): - if self.env.config["randomize_states"]: - self.init_states = self.randomize_init_states() - self.env.config.update({'init_states': list(self.init_states)}) - else: - self.init_states = self.env.config["init_states"] - - def randomize_init_states(self): - """Randomize initial states. - - Returns: - ------- - init_states: list - List of random initial states for the environment. - - Note: - ---- - This method should be implemented according to needed random initialization. - """ - return self.env.config["init_states"] - - def init_goal(self): - # Set a new random goal from the list - goalList = self.env.config["goalList"] - id_goal = self.env.np_random.choice(range(len(goalList))) - self.env.config.update({'goal_node': id_goal}) - self.env.goal = goalList[id_goal] - self.env.config.update({'goalPos': self.env.goal}) - def reset(self): """Reset simulation. """ @@ -130,16 +101,3 @@ def reset(self): state = np.array(self.env._getState(self.env.root), dtype=np.float32) return state - - def get_available_actions(self): - """Gives the actions available in the environment. - - Parameters: - ---------- - None. - - Returns: - ------- - list of the action available in the environment. - """ - return self.env.action_space From 85767d96c09c393fc88283bd5f8dd0cc5414431e Mon Sep 17 00:00:00 2001 From: samuelmyoussef Date: Thu, 26 Sep 2024 04:43:32 +0200 Subject: [PATCH 73/85] remove non-overriden methods --- sofagym/envs/SimpleMaze/SimpleMazeEnv.py | 47 +++++------------------- 1 file changed, 9 insertions(+), 38 deletions(-) diff --git a/sofagym/envs/SimpleMaze/SimpleMazeEnv.py b/sofagym/envs/SimpleMaze/SimpleMazeEnv.py index 3613c76..4af6110 100644 --- a/sofagym/envs/SimpleMaze/SimpleMazeEnv.py +++ b/sofagym/envs/SimpleMaze/SimpleMazeEnv.py @@ -56,7 +56,9 @@ class SimpleMazeEnv: "use_server": False } - def __init__(self, config = None, root=None, use_server: Optional[bool]=False): + def __init__(self, config = None, root=None, use_server: Optional[bool]=None): + if use_server is not None: + self.DEFAULT_CONFIG.update({'use_server': use_server}) self.use_server = self.DEFAULT_CONFIG["use_server"] self.env = ServerEnv(self.DEFAULT_CONFIG, config, root=root) if self.use_server else AbstractEnv(self.DEFAULT_CONFIG, config, root=root) @@ -65,13 +67,11 @@ def __init__(self, config = None, root=None, use_server: Optional[bool]=False): if self.env.config["goal"]: self.init_goal() - nb_actions = self.env.config["nb_actions"] - self.env.action_space = spaces.Discrete(nb_actions) - self.nb_actions = str(nb_actions) + self.env.action_space = spaces.Discrete(self.env.nb_actions) + self.nb_actions = str(self.env.nb_actions) - dim_state = self.env.config["dim_state"] - low_coordinates = np.array([-1]*dim_state) - high_coordinates = np.array([1]*dim_state) + low_coordinates = np.array([-1]*self.env.dim_state) + high_coordinates = np.array([1]*self.env.dim_state) self.env.observation_space = spaces.Box(low_coordinates, high_coordinates, dtype=np.float32) if self.env.root is None and not self.use_server: @@ -82,43 +82,14 @@ def __getattr__(self, name): # assume it is implemented by self.instance return self.env.__getattribute__(name) - def initialize_states(self): - if self.env.config["randomize_states"]: - self.init_states = self.randomize_init_states() - self.env.config.update({'init_states': list(self.init_states)}) - else: - self.init_states = self.env.config["init_states"] - - def randomize_init_states(self): - """Randomize initial states. - - Returns: - ------- - init_states: list - List of random initial states for the environment. - - Note: - ---- - This method should be implemented according to needed random initialization. - """ - return self.env.config["init_states"] - - def init_goal(self): - # Set a new random goal from the list - goalList = self.env.config["goalList"] - id_goal = self.env.np_random.choice(range(len(goalList))) - self.env.config.update({'goal_node': id_goal}) - self.env.goal = goalList[id_goal] - self.env.config.update({'goalPos': self.env.goal}) - def reset(self): """Reset simulation. """ self.initialize_states() - + if self.env.config["goal"]: self.init_goal() - + self.env.reset() if self.use_server: From b181d14904f708dfab7a1a615564897e0cd3a9bd Mon Sep 17 00:00:00 2001 From: samuelmyoussef Date: Thu, 26 Sep 2024 04:37:57 +0200 Subject: [PATCH 74/85] remove non-overriden methods --- sofagym/envs/StemPendulum/StemPendulumEnv.py | 52 ++++---------------- 1 file changed, 9 insertions(+), 43 deletions(-) diff --git a/sofagym/envs/StemPendulum/StemPendulumEnv.py b/sofagym/envs/StemPendulum/StemPendulumEnv.py index 6edb51a..54209a8 100644 --- a/sofagym/envs/StemPendulum/StemPendulumEnv.py +++ b/sofagym/envs/StemPendulum/StemPendulumEnv.py @@ -57,7 +57,9 @@ class StemPendulumEnv: "use_server": False } - def __init__(self, config = None, root=None, use_server: Optional[bool]=False): + def __init__(self, config = None, root=None, use_server: Optional[bool]=None): + if use_server is not None: + self.DEFAULT_CONFIG.update({'use_server': use_server}) self.use_server = self.DEFAULT_CONFIG["use_server"] self.env = ServerEnv(self.DEFAULT_CONFIG, config, root=root) if self.use_server else AbstractEnv(self.DEFAULT_CONFIG, config, root=root) @@ -65,16 +67,14 @@ def __init__(self, config = None, root=None, use_server: Optional[bool]=False): if self.env.config["goal"]: self.init_goal() - - nb_actions = self.env.config["nb_actions"] + low = np.array([-1]*1) high = np.array([1]*1) self.env.action_space = spaces.Box(low=low, high=high, shape=(1,), dtype=np.float32) - self.nb_actions = str(nb_actions) + self.nb_actions = str(self.env.nb_actions) - dim_state = self.env.config["dim_state"] - low_coordinates = np.array([-2]*dim_state) - high_coordinates = np.array([2]*dim_state) + low_coordinates = np.array([-2]*self.env.dim_state) + high_coordinates = np.array([2]*self.env.dim_state) self.env.observation_space = spaces.Box(low_coordinates, high_coordinates, dtype=np.float32) if self.env.root is None and not self.use_server: @@ -85,35 +85,14 @@ def __getattr__(self, name): # assume it is implemented by self.instance return self.env.__getattribute__(name) - def initialize_states(self): - if self.env.config["randomize_states"]: - self.init_states = self.randomize_init_states() - self.env.config.update({'init_states': list(self.init_states)}) - else: - self.init_states = self.env.config["init_states"] - - def randomize_init_states(self): - """Randomize initial states. - - Returns: - ------- - init_states: list - List of random initial states for the environment. - - Note: - ---- - This method should be implemented according to needed random initialization. - """ - return self.env.config["init_states"] - def reset(self): """Reset simulation. """ self.initialize_states() - + if self.env.config["goal"]: self.init_goal() - + self.env.reset() if self.use_server: @@ -123,16 +102,3 @@ def reset(self): state = np.array(self.env._getState(self.env.root), dtype=np.float32) return state - - def get_available_actions(self): - """Gives the actions available in the environment. - - Parameters: - ---------- - None. - - Returns: - ------- - list of the action available in the environment. - """ - return self.env.action_space From 6ff47bdf13c994b6bb37142009832e3a8f0fb5b1 Mon Sep 17 00:00:00 2001 From: samuelmyoussef Date: Thu, 26 Sep 2024 04:31:42 +0200 Subject: [PATCH 75/85] remove non-overriden methods --- sofagym/envs/TrunkCup/TrunkCupEnv.py | 62 +++++----------------------- 1 file changed, 10 insertions(+), 52 deletions(-) diff --git a/sofagym/envs/TrunkCup/TrunkCupEnv.py b/sofagym/envs/TrunkCup/TrunkCupEnv.py index 600818f..bd3f05a 100644 --- a/sofagym/envs/TrunkCup/TrunkCupEnv.py +++ b/sofagym/envs/TrunkCup/TrunkCupEnv.py @@ -55,22 +55,22 @@ class TrunkCupEnv: "use_server": False } - def __init__(self, config = None, root=None, use_server: Optional[bool]=False): + def __init__(self, config = None, root=None, use_server: Optional[bool]=None): + if use_server is not None: + self.DEFAULT_CONFIG.update({'use_server': use_server}) self.use_server = self.DEFAULT_CONFIG["use_server"] self.env = ServerEnv(self.DEFAULT_CONFIG, config, root=root) if self.use_server else AbstractEnv(self.DEFAULT_CONFIG, config, root=root) self.initialize_states() - + if self.env.config["goal"]: self.init_goal() - nb_actions = self.env.config["nb_actions"] - self.env.action_space = spaces.Discrete(nb_actions) - self.nb_actions = str(nb_actions) + self.env.action_space = spaces.Discrete(self.env.nb_actions) + self.nb_actions = str(self.env.nb_actions) - dim_state = self.env.config["dim_state"] - low_coordinates = np.array([-1]*dim_state) - high_coordinates = np.array([1]*dim_state) + low_coordinates = np.array([-1]*self.env.dim_state) + high_coordinates = np.array([1]*self.env.dim_state) self.env.observation_space = spaces.Box(low_coordinates, high_coordinates, dtype=np.float32) if self.env.root is None and not self.use_server: @@ -81,43 +81,14 @@ def __getattr__(self, name): # assume it is implemented by self.instance return self.env.__getattribute__(name) - def initialize_states(self): - if self.env.config["randomize_states"]: - self.init_states = self.randomize_init_states() - self.env.config.update({'init_states': list(self.init_states)}) - else: - self.init_states = self.env.config["init_states"] - - def randomize_init_states(self): - """Randomize initial states. - - Returns: - ------- - init_states: list - List of random initial states for the environment. - - Note: - ---- - This method should be implemented according to needed random initialization. - """ - return self.env.config["init_states"] - - def init_goal(self): - # Set a new random goal from the list - goalList = self.env.config["goalList"] - id_goal = self.env.np_random.choice(range(len(goalList))) - self.env.config.update({'goal_node': id_goal}) - self.env.goal = goalList[id_goal] - self.env.config.update({'goalPos': self.env.goal}) - def reset(self): """Reset simulation. """ self.initialize_states() - + if self.env.config["goal"]: self.init_goal() - + self.env.reset() if self.use_server: @@ -127,16 +98,3 @@ def reset(self): state = np.array(self.env._getState(self.env.root), dtype=np.float32) return state - - def get_available_actions(self): - """Gives the actions available in the environment. - - Parameters: - ---------- - None. - - Returns: - ------- - list of the action available in the environment. - """ - return self.env.action_space From 5de26b7b77de6590e2b7020df461d4c739067512 Mon Sep 17 00:00:00 2001 From: samuelmyoussef Date: Thu, 26 Sep 2024 04:51:51 +0200 Subject: [PATCH 76/85] remove non-overriden methods --- sofagym/envs/BubbleMotion/BubbleMotionEnv.py | 49 ++++---------------- 1 file changed, 9 insertions(+), 40 deletions(-) diff --git a/sofagym/envs/BubbleMotion/BubbleMotionEnv.py b/sofagym/envs/BubbleMotion/BubbleMotionEnv.py index 6756c44..67151c9 100644 --- a/sofagym/envs/BubbleMotion/BubbleMotionEnv.py +++ b/sofagym/envs/BubbleMotion/BubbleMotionEnv.py @@ -58,24 +58,24 @@ class BubbleMotionEnv: "use_server": False } - def __init__(self, config = None, root=None, use_server: Optional[bool]=False): + def __init__(self, config = None, root=None, use_server: Optional[bool]=None): + if use_server is not None: + self.DEFAULT_CONFIG.update({'use_server': use_server}) self.use_server = self.DEFAULT_CONFIG["use_server"] self.env = ServerEnv(self.DEFAULT_CONFIG, config, root=root) if self.use_server else AbstractEnv(self.DEFAULT_CONFIG, config, root=root) self.initialize_states() - + if self.env.config["goal"]: self.init_goal() - - nb_actions = self.env.config["nb_actions"] + low = np.array([-1]*9) high = np.array([1]*9) self.env.action_space = spaces.Box(low=low, high=high, shape=(9,), dtype=np.float32) - self.nb_actions = str(nb_actions) + self.nb_actions = str(self.env.nb_actions) - dim_state = self.env.config["dim_state"] - low_coordinates = np.array([0]*dim_state) - high_coordinates = np.array([80]*dim_state) + low_coordinates = np.array([0]*self.env.dim_state) + high_coordinates = np.array([80]*self.env.dim_state) self.env.observation_space = spaces.Box(low_coordinates, high_coordinates, dtype=np.float32) if self.env.root is None and not self.use_server: @@ -87,29 +87,11 @@ def __getattr__(self, name): return self.env.__getattribute__(name) def initialize_states(self): - if self.env.config["randomize_states"]: - self.init_states = self.randomize_init_states() - self.env.config.update({'init_states': list(self.init_states)}) - else: - self.init_states = self.env.config["init_states"] + self.env.initialize_states() bd = self.env.config["board_dim"] init_pos = [4+(bd-4)*self.env.np_random.random(), 4+(bd-4)*self.env.np_random.random(), 5] self.env.config.update({'init_pos': init_pos}) - - def randomize_init_states(self): - """Randomize initial states. - - Returns: - ------- - init_states: list - List of random initial states for the environment. - - Note: - ---- - This method should be implemented according to needed random initialization. - """ - return self.env.config["init_states"] def init_goal(self): bd = self.env.config["board_dim"] @@ -134,16 +116,3 @@ def reset(self): state = np.array(self.env._getState(self.env.root), dtype=np.float32) return state - - def get_available_actions(self): - """Gives the actions available in the environment. - - Parameters: - ---------- - None. - - Returns: - ------- - list of the action available in the environment. - """ - return self.env.action_space From b210a12fb809a0f52a5182167ff9d29c76f73655 Mon Sep 17 00:00:00 2001 From: samuelmyoussef Date: Thu, 26 Sep 2024 04:16:29 +0200 Subject: [PATCH 77/85] remove non-overriden methods --- sofagym/envs/Trunk/TrunkEnv.py | 62 ++++++---------------------------- 1 file changed, 10 insertions(+), 52 deletions(-) diff --git a/sofagym/envs/Trunk/TrunkEnv.py b/sofagym/envs/Trunk/TrunkEnv.py index c688a66..810e587 100644 --- a/sofagym/envs/Trunk/TrunkEnv.py +++ b/sofagym/envs/Trunk/TrunkEnv.py @@ -57,22 +57,22 @@ class TrunkEnv: "use_server": False } - def __init__(self, config = None, root=None, use_server: Optional[bool]=False): + def __init__(self, config = None, root=None, use_server: Optional[bool]=None): + if use_server is not None: + self.DEFAULT_CONFIG.update({'use_server': use_server}) self.use_server = self.DEFAULT_CONFIG["use_server"] self.env = ServerEnv(self.DEFAULT_CONFIG, config, root=root) if self.use_server else AbstractEnv(self.DEFAULT_CONFIG, config, root=root) self.initialize_states() - + if self.env.config["goal"]: self.init_goal() - nb_actions = self.env.config["nb_actions"] - self.env.action_space = spaces.Discrete(nb_actions) - self.nb_actions = str(nb_actions) + self.env.action_space = spaces.Discrete(self.env.nb_actions) + self.nb_actions = str(self.env.nb_actions) - dim_state = self.env.config["dim_state"] - low_coordinates = np.array([-1]*dim_state) - high_coordinates = np.array([1]*dim_state) + low_coordinates = np.array([-1]*self.env.dim_state) + high_coordinates = np.array([1]*self.env.dim_state) self.env.observation_space = spaces.Box(low_coordinates, high_coordinates, dtype=np.float32) if self.env.root is None and not self.use_server: @@ -83,43 +83,14 @@ def __getattr__(self, name): # assume it is implemented by self.instance return self.env.__getattribute__(name) - def initialize_states(self): - if self.env.config["randomize_states"]: - self.init_states = self.randomize_init_states() - self.env.config.update({'init_states': list(self.init_states)}) - else: - self.init_states = self.env.config["init_states"] - - def randomize_init_states(self): - """Randomize initial states. - - Returns: - ------- - init_states: list - List of random initial states for the environment. - - Note: - ---- - This method should be implemented according to needed random initialization. - """ - return self.env.config["init_states"] - - def init_goal(self): - # Set a new random goal from the list - goalList = self.env.config["goalList"] - id_goal = self.env.np_random.choice(range(len(goalList))) - self.env.config.update({'goal_node': id_goal}) - self.env.goal = goalList[id_goal] - self.env.config.update({'goalPos': self.env.goal}) - def reset(self): """Reset simulation. """ self.initialize_states() - + if self.env.config["goal"]: self.init_goal() - + self.env.reset() if self.use_server: @@ -129,16 +100,3 @@ def reset(self): state = np.array(self.env._getState(self.env.root), dtype=np.float32) return state - - def get_available_actions(self): - """Gives the actions available in the environment. - - Parameters: - ---------- - None. - - Returns: - ------- - list of the action available in the environment. - """ - return self.env.action_space From 995569f9733442f263088e3a571f462683d95318 Mon Sep 17 00:00:00 2001 From: samuelmyoussef Date: Fri, 8 Dec 2023 16:40:57 +0100 Subject: [PATCH 78/85] remove unused goal components --- sofagym/envs/CartPole/CartPoleEnv.py | 2 -- sofagym/envs/CartPole/CartPoleScene.py | 4 +--- sofagym/envs/CartPole/CartPoleToolbox.py | 13 +------------ 3 files changed, 2 insertions(+), 17 deletions(-) diff --git a/sofagym/envs/CartPole/CartPoleEnv.py b/sofagym/envs/CartPole/CartPoleEnv.py index ac64418..a6d40b3 100644 --- a/sofagym/envs/CartPole/CartPoleEnv.py +++ b/sofagym/envs/CartPole/CartPoleEnv.py @@ -22,7 +22,6 @@ class CartPoleEnv: "source": [0, 0, 160], "target": [0, 0, 0], "goal": False, - "goalList": [[0]], "start_node": None, "scale_factor": 10, "dt": 0.001, @@ -82,7 +81,6 @@ def reset(self): """ self.env.reset() - self.env.config.update({'goalPos': self.env.goal}) init_states = self.env.np_random.uniform(low=-0.05, high=0.05, size=(4,)) self.env.config.update({'init_states': list(init_states)}) diff --git a/sofagym/envs/CartPole/CartPoleScene.py b/sofagym/envs/CartPole/CartPoleScene.py index ebe358d..bc6773d 100644 --- a/sofagym/envs/CartPole/CartPoleScene.py +++ b/sofagym/envs/CartPole/CartPoleScene.py @@ -4,7 +4,7 @@ sys.path.insert(0, str(pathlib.Path(__file__).parent.absolute())+"/../") sys.path.insert(0, str(pathlib.Path(__file__).parent.absolute())) -from CartPoleToolbox import ApplyAction, GoalSetter, RewardShaper, StateInitializer +from CartPoleToolbox import ApplyAction, RewardShaper, StateInitializer from sofagym.header import addVisu from splib3.animation import AnimationManagerController from stlib3.physics.rigid import Floor @@ -51,7 +51,6 @@ def addRigidObject(node, filename, collisionFilename=None, position=[0, 0, 0, 0, def createScene(root, config={"source": [0, 0, 160], "target": [0, 0, 0], - "goalPos": None, "seed": None, "zFar":4000, "init_x": 0, @@ -168,5 +167,4 @@ def createScene(root, # SofaGym Env Components root.addObject(StateInitializer(name="StateInitializer", rootNode=root, pole_length=pole_length, init_states=config['init_states'])) root.addObject(RewardShaper(name="Reward", rootNode=root, max_angle=config['max_angle'], pole_length=pole_length)) - root.addObject(GoalSetter(name="GoalSetter")) root.addObject(ApplyAction(name="ApplyAction", root=root)) diff --git a/sofagym/envs/CartPole/CartPoleToolbox.py b/sofagym/envs/CartPole/CartPoleToolbox.py index 70e8460..d14b4ff 100644 --- a/sofagym/envs/CartPole/CartPoleToolbox.py +++ b/sofagym/envs/CartPole/CartPoleToolbox.py @@ -150,7 +150,7 @@ def getReward(self): return 1, pole_theta, self.max_angle - def update(self, goal): + def update(self, goal=None): """Update function. This function is used as an initialization function. @@ -212,17 +212,6 @@ def getState(rootNode): return state -class GoalSetter(Sofa.Core.Controller): - def __init__(self, *args, **kwargs): - Sofa.Core.Controller.__init__(self, *args, **kwargs) - - def update(self, goal): - pass - - def set_mo_pos(self, goal): - pass - - def getReward(rootNode): reward, theta, max_angle = rootNode.Reward.getReward() done = (theta > max_angle) or (theta < -max_angle) From 929ffcecc06a7e577940e3c706c69500c99cc1cb Mon Sep 17 00:00:00 2001 From: samuelmyoussef Date: Wed, 13 Dec 2023 11:53:00 +0100 Subject: [PATCH 79/85] add some default configs --- sofagym/envs/CartPole/CartPoleEnv.py | 23 ++++++++++++++++++++--- 1 file changed, 20 insertions(+), 3 deletions(-) diff --git a/sofagym/envs/CartPole/CartPoleEnv.py b/sofagym/envs/CartPole/CartPoleEnv.py index a6d40b3..ea47176 100644 --- a/sofagym/envs/CartPole/CartPoleEnv.py +++ b/sofagym/envs/CartPole/CartPoleEnv.py @@ -17,6 +17,7 @@ class CartPoleEnv: #Setting a default configuration path = path = os.path.dirname(os.path.abspath(__file__)) metadata = {'render.modes': ['human', 'rgb_array']} + dim_state = 4 DEFAULT_CONFIG = {"scene": "CartPole", "deterministic": True, "source": [0, 0, 160], @@ -39,11 +40,14 @@ class CartPoleEnv: "zFar": 4000, "time_before_start": 0, "seed": None, + "nb_actions": 2, + "dim_state": dim_state, "init_x": 0, + "x_threshold": 100, "max_move": 24, "max_angle": 0.418, "randomize_states": True, - "init_states": [0]*4, + "init_states": [0] * dim_state, "use_server": False } @@ -51,7 +55,11 @@ def __init__(self, config=None, root=None, use_server: Optional[bool]=False): self.use_server = self.DEFAULT_CONFIG["use_server"] self.env = ServerEnv(self.DEFAULT_CONFIG, config, root=root) if self.use_server else AbstractEnv(self.DEFAULT_CONFIG, config, root=root) - self.x_threshold = 100 + self.init_states = self.env.config["init_states"] + if self.env.config["randomize_states"]: + self.randomize_init_states() + + self.x_threshold = self.env.config["x_threshold"] self.theta_threshold_radians = self.env.config["max_move"] * math.pi / 180 self.env.config.update({'max_angle': self.theta_threshold_radians}) @@ -65,7 +73,7 @@ def __init__(self, config=None, root=None, use_server: Optional[bool]=False): dtype=np.float32, ) - nb_actions = 2 + nb_actions = self.env.config["nb_actions"] self.env.action_space = spaces.Discrete(nb_actions) self.nb_actions = str(nb_actions) @@ -76,9 +84,18 @@ def __getattr__(self, name): # assume it is implemented by self.instance return self.env.__getattribute__(name) + + def randomize_init_states(self): + self.init_states = self.env.np_random.uniform(low=-0.05, high=0.05, size=(self.env.config["dim_state"],)) + self.env.config.update({'init_states': list(self.init_states)}) + + def reset(self): """Reset simulation. """ + if self.env.config["randomize_states"]: + self.randomize_init_states() + self.env.reset() init_states = self.env.np_random.uniform(low=-0.05, high=0.05, size=(4,)) From ac19ebc4ec9dec31ce1e4330b75ce6c11a3b4d88 Mon Sep 17 00:00:00 2001 From: samuelmyoussef Date: Mon, 23 Sep 2024 16:56:27 +0200 Subject: [PATCH 80/85] initialize states --- sofagym/envs/CartPole/CartPoleEnv.py | 45 +++++++++++++++++++--------- 1 file changed, 31 insertions(+), 14 deletions(-) diff --git a/sofagym/envs/CartPole/CartPoleEnv.py b/sofagym/envs/CartPole/CartPoleEnv.py index ea47176..b2d3bb6 100644 --- a/sofagym/envs/CartPole/CartPoleEnv.py +++ b/sofagym/envs/CartPole/CartPoleEnv.py @@ -15,7 +15,7 @@ class CartPoleEnv: See the class AbstractEnv for arguments and methods. """ #Setting a default configuration - path = path = os.path.dirname(os.path.abspath(__file__)) + path = os.path.dirname(os.path.abspath(__file__)) metadata = {'render.modes': ['human', 'rgb_array']} dim_state = 4 DEFAULT_CONFIG = {"scene": "CartPole", @@ -53,11 +53,10 @@ class CartPoleEnv: def __init__(self, config=None, root=None, use_server: Optional[bool]=False): self.use_server = self.DEFAULT_CONFIG["use_server"] + self.env = ServerEnv(self.DEFAULT_CONFIG, config, root=root) if self.use_server else AbstractEnv(self.DEFAULT_CONFIG, config, root=root) - self.init_states = self.env.config["init_states"] - if self.env.config["randomize_states"]: - self.randomize_init_states() + self.initialize_states() self.x_threshold = self.env.config["x_threshold"] self.theta_threshold_radians = self.env.config["max_move"] * math.pi / 180 @@ -79,33 +78,51 @@ def __init__(self, config=None, root=None, use_server: Optional[bool]=False): self.env.observation_space = spaces.Box(-high, high, dtype=np.float32) + if self.env.root is None and not self.use_server: + self.env.init_root() + # called when an attribute is not found: def __getattr__(self, name): # assume it is implemented by self.instance return self.env.__getattribute__(name) - + def initialize_states(self): + if self.env.config["randomize_states"]: + self.init_states = self.randomize_init_states() + self.env.config.update({'init_states': list(self.init_states)}) + else: + self.init_states = self.env.config["init_states"] + def randomize_init_states(self): - self.init_states = self.env.np_random.uniform(low=-0.05, high=0.05, size=(self.env.config["dim_state"],)) - self.env.config.update({'init_states': list(self.init_states)}) + """Randomize initial states. + + Returns: + ------- + init_states: list + List of random initial states for the environment. + + Note: + ---- + This method should be implemented according to needed random initialization. + """ + init_states = self.env.np_random.uniform(low=-0.05, high=0.05, size=(self.env.config["dim_state"],)) + return init_states def reset(self): """Reset simulation. """ - if self.env.config["randomize_states"]: - self.randomize_init_states() + self.initialize_states() self.env.reset() - - init_states = self.env.np_random.uniform(low=-0.05, high=0.05, size=(4,)) - self.env.config.update({'init_states': list(init_states)}) if self.use_server: obs = start_scene(self.env.config, self.nb_actions) - return np.array(obs['observation']) + state = np.array(obs['observation'], dtype=np.float32) + else: + state = np.array(self.env._getState(self.env.root), dtype=np.float32) - return np.array(init_states) + return state def get_available_actions(self): """Gives the actions available in the environment. From 780f3a8ac533c638bcbe307a120ceb1314fc4050 Mon Sep 17 00:00:00 2001 From: samuelmyoussef Date: Thu, 11 Jan 2024 16:34:24 +0100 Subject: [PATCH 81/85] Update pole angle from position directly instead of calculation and decrease max angle allowed --- sofagym/envs/CartPole/CartPoleEnv.py | 2 +- sofagym/envs/CartPole/CartPoleToolbox.py | 28 ++++-------------------- 2 files changed, 5 insertions(+), 25 deletions(-) diff --git a/sofagym/envs/CartPole/CartPoleEnv.py b/sofagym/envs/CartPole/CartPoleEnv.py index b2d3bb6..c6fb573 100644 --- a/sofagym/envs/CartPole/CartPoleEnv.py +++ b/sofagym/envs/CartPole/CartPoleEnv.py @@ -44,7 +44,7 @@ class CartPoleEnv: "dim_state": dim_state, "init_x": 0, "x_threshold": 100, - "max_move": 24, + "max_move": 12, "max_angle": 0.418, "randomize_states": True, "init_states": [0] * dim_state, diff --git a/sofagym/envs/CartPole/CartPoleToolbox.py b/sofagym/envs/CartPole/CartPoleToolbox.py index d14b4ff..dac1582 100644 --- a/sofagym/envs/CartPole/CartPoleToolbox.py +++ b/sofagym/envs/CartPole/CartPoleToolbox.py @@ -145,8 +145,7 @@ def getReward(self): The reward and the cost. """ - cart_pos = self._getCartPos() - pole_theta, pole_theta_dot = self.calculatePoleTheta(cart_pos) + pole_theta = self.pole.MechanicalObject.position.value.tolist()[0][5] return 1, pole_theta, self.max_angle @@ -166,26 +165,6 @@ def update(self, goal=None): """ pass - def _getPolePos(self): - pos = self.pole.MechanicalObject.position.value.tolist()[0] - return pos[0], pos[1] - - def _getCartPos(self): - pos = self.cart.MechanicalObject.position.value.tolist()[0][0] - return pos - - def calculatePoleTheta(self, cart_pos): - x_pos, y_pos = self._getPolePos() - sin_theta = (y_pos/self.pole_length) - theta = abs((90*math.pi/180) - math.asin(sin_theta)) - - if x_pos < cart_pos: - theta = -theta - - theta_dot = self.pole.MechanicalObject.velocity.value.tolist()[0][5] - - return theta, theta_dot - def getState(rootNode): """Compute the state of the environment/agent. @@ -201,11 +180,12 @@ def getState(rootNode): The state of the environment/agent. """ cart = rootNode.Modeling.Cart - cart_pos = cart.MechanicalObject.position.value.tolist()[0][0] cart_vel = cart.MechanicalObject.velocity.value.tolist()[0][0] - pole_theta, pole_theta_dot = rootNode.Reward.calculatePoleTheta(cart_pos) + pole = rootNode.Modeling.Pole + pole_theta = pole.MechanicalObject.position.value.tolist()[0][5] + pole_theta_dot = pole.MechanicalObject.velocity.value.tolist()[0][5] state = [cart_pos, cart_vel, pole_theta, pole_theta_dot] From e9a316612cee64179f7ef864c20580536b992b17 Mon Sep 17 00:00:00 2001 From: samuelmyoussef Date: Wed, 25 Sep 2024 10:52:19 +0200 Subject: [PATCH 82/85] goal init --- sofagym/envs/CartPole/CartPoleEnv.py | 6 ++++++ 1 file changed, 6 insertions(+) diff --git a/sofagym/envs/CartPole/CartPoleEnv.py b/sofagym/envs/CartPole/CartPoleEnv.py index c6fb573..196297f 100644 --- a/sofagym/envs/CartPole/CartPoleEnv.py +++ b/sofagym/envs/CartPole/CartPoleEnv.py @@ -58,6 +58,9 @@ def __init__(self, config=None, root=None, use_server: Optional[bool]=False): self.initialize_states() + if self.env.config["goal"]: + self.init_goal() + self.x_threshold = self.env.config["x_threshold"] self.theta_threshold_radians = self.env.config["max_move"] * math.pi / 180 self.env.config.update({'max_angle': self.theta_threshold_radians}) @@ -113,6 +116,9 @@ def reset(self): """Reset simulation. """ self.initialize_states() + + if self.env.config["goal"]: + self.init_goal() self.env.reset() From 28c07b42b754eeb1d491e395f7a82e7b7fd6871d Mon Sep 17 00:00:00 2001 From: samuelmyoussef Date: Thu, 26 Sep 2024 05:44:17 +0200 Subject: [PATCH 83/85] remove non-overriden methods --- sofagym/envs/CartPole/CartPoleEnv.py | 23 +++++------------------ 1 file changed, 5 insertions(+), 18 deletions(-) diff --git a/sofagym/envs/CartPole/CartPoleEnv.py b/sofagym/envs/CartPole/CartPoleEnv.py index 196297f..a7683bb 100644 --- a/sofagym/envs/CartPole/CartPoleEnv.py +++ b/sofagym/envs/CartPole/CartPoleEnv.py @@ -51,9 +51,10 @@ class CartPoleEnv: "use_server": False } - def __init__(self, config=None, root=None, use_server: Optional[bool]=False): + def __init__(self, config = None, root=None, use_server: Optional[bool]=None): + if use_server is not None: + self.DEFAULT_CONFIG.update({'use_server': use_server}) self.use_server = self.DEFAULT_CONFIG["use_server"] - self.env = ServerEnv(self.DEFAULT_CONFIG, config, root=root) if self.use_server else AbstractEnv(self.DEFAULT_CONFIG, config, root=root) self.initialize_states() @@ -75,9 +76,8 @@ def __init__(self, config=None, root=None, use_server: Optional[bool]=False): dtype=np.float32, ) - nb_actions = self.env.config["nb_actions"] - self.env.action_space = spaces.Discrete(nb_actions) - self.nb_actions = str(nb_actions) + self.env.action_space = spaces.Discrete(self.env.nb_actions) + self.nb_actions = str(self.env.nb_actions) self.env.observation_space = spaces.Box(-high, high, dtype=np.float32) @@ -129,16 +129,3 @@ def reset(self): state = np.array(self.env._getState(self.env.root), dtype=np.float32) return state - - def get_available_actions(self): - """Gives the actions available in the environment. - - Parameters: - ---------- - None. - - Returns: - ------- - list of the action available in the environment. - """ - return self.env.action_space From 09f3817d0af9d38240fdfbf9cebcca6346acc0f1 Mon Sep 17 00:00:00 2001 From: samuelmyoussef <30751376+samuelmyoussef@users.noreply.github.com> Date: Fri, 18 Oct 2024 08:44:33 +0200 Subject: [PATCH 84/85] [Core] Add no server architecture implementation to the environments (#62) * change abstract class to no server * add server class * modify viewer * modify simulate & goal initializtion * simulation init and reset * remove unneeded goal and reward update and unused reward update param * add general methods to init and randomize states and goal * no server * class instantiation * remove unused goal components * add some default configs * initialize states * Update pole angle from position directly instead of calculation and decrease max angle allowed * goal init * remove non-overriden methods * separate AbstractEnv and ServerEnv into two files --- sofagym/AbstractEnv.py | 342 ++++++++++++++--------- sofagym/ServerEnv.py | 168 +++++++++++ sofagym/envs/CartPole/CartPoleEnv.py | 106 ++++--- sofagym/envs/CartPole/CartPoleScene.py | 9 +- sofagym/envs/CartPole/CartPoleToolbox.py | 56 +--- sofagym/simulate.py | 24 +- sofagym/viewer.py | 20 +- 7 files changed, 492 insertions(+), 233 deletions(-) create mode 100644 sofagym/ServerEnv.py diff --git a/sofagym/AbstractEnv.py b/sofagym/AbstractEnv.py index 20b42c8..17af7c8 100644 --- a/sofagym/AbstractEnv.py +++ b/sofagym/AbstractEnv.py @@ -20,7 +20,11 @@ import splib3 from sofagym.viewer import Viewer -from sofagym.rpc_server import start_server, add_new_step, get_result, clean_registry, close_scene + +import importlib + +import Sofa +import SofaRuntime class AbstractEnv(gym.Env): @@ -39,7 +43,6 @@ class AbstractEnv(gym.Env): while an action is still ongoing. close: Terminate the simulation. configure: Add element in the configuration. - clean: clean the registery. _formataction.. : transforme the type of action to use server. Arguments: @@ -112,7 +115,7 @@ class AbstractEnv(gym.Env): """ - def __init__(self, config=None, render_mode: Optional[str] = None): + def __init__(self, default_config, config=None, render_mode: Optional[str]=None, root=None): """ Classic initialization of a class in python. @@ -127,48 +130,63 @@ def __init__(self, config=None, render_mode: Optional[str] = None): """ # Define a DEFAULT_CONFIG in sub-class. - self.config = copy.deepcopy(self.DEFAULT_CONFIG) + self.config = copy.deepcopy(default_config) self.config["dt"] = self.config.get('dt', 0.01) if config is not None: self.config.update(config) - self.initialization() - - self.render_mode = render_mode - - def initialization(self): - """Initialization of all parameters. + self.scene = self.config['scene'] - Parameters: - ---------- - None. + self._getState = importlib.import_module("sofagym.envs."+self.scene+"."+self.scene+"Toolbox").getState + self._getReward = importlib.import_module("sofagym.envs."+self.scene+"."+self.scene+"Toolbox").getReward + self._startCmd = importlib.import_module("sofagym.envs."+self.scene+"."+self.scene+"Toolbox").startCmd + self._getPos = importlib.import_module("sofagym.envs."+self.scene+"."+self.scene+"Toolbox").getPos + + try: + self.create_scene = importlib.import_module("sofagym.envs."+self.scene+"." + self.scene + "Scene").createScene + except Exception as exc: + print("sofagym.envs."+self.scene+"." + self.scene + "Scene") + raise NotImplementedError("Importing your SOFA Scene Failed") from exc - Returns: - ------- - None. - """ + self.viewer = None + self.render_mode = render_mode - self.goalList = None - self.goal = None self.past_actions = [] - + self.pos = [] + self.past_pos = [] + self.num_envs = 40 self.np_random = None self.seed(self.config['seed']) - self.viewer = None - self.automatic_rendering_callback = None - - self.timer = 0 self.timeout = self.config["timeout"] - # Start the server which distributes the calculations to its clients - start_server(self.config) + self.init_save_paths() + + self.root = root + + self.init_states = None + + self.goal = None + + self.nb_actions = self.config["nb_actions"] + self.dim_state = self.config["dim_state"] + + def init_save_paths(self): + """Create directories to save results and images. + + Parameters: + ---------- + None. + Returns: + ------- + None. + """ if 'save_data' in self.config and self.config['save_data']: save_path_results = self.config['save_path']+"/data" os.makedirs(save_path_results, exist_ok=True) @@ -183,6 +201,38 @@ def initialization(self): self.configure({"save_path_image": save_path_image, "save_path_results": save_path_results}) + def init_root(self): + self.init_simulation() + + def initialize_states(self): + if self.config["randomize_states"]: + self.init_states = self.randomize_init_states() + self.config.update({'init_states': list(self.init_states)}) + else: + self.init_states = self.config["init_states"] + + def randomize_init_states(self): + """Randomize initial states. + + Returns: + ------- + init_states: list + List of random initial states for the environment. + + Note: + ---- + This method should be implemented according to needed random initialization. + """ + return self.config["init_states"] + + def init_goal(self): + # Set a new random goal from the list + goalList = self.config["goalList"] + id_goal = self.np_random.choice(range(len(goalList))) + self.config.update({'goal_node': id_goal}) + self.goal = goalList[id_goal] + self.config.update({'goalPos': self.goal}) + def seed(self, seed=None): """ Computes the random generators of the environment. @@ -200,6 +250,19 @@ def seed(self, seed=None): self.np_random, seed = seeding.np_random(seed) return [seed] + def get_available_actions(self): + """Gives the actions available in the environment. + + Parameters: + ---------- + None. + + Returns: + ------- + list of the action available in the environment. + """ + return self.action_space + def _formataction(self, action): """Change the type of action to be in [list, float, int]. @@ -258,23 +321,6 @@ def _formatactionDict(self, action): return action - def clean(self): - """Function to clean the registery . - - Close clients who are processing unused sequences of actions (for - planning) - - Parameters: - ---------- - None. - - Returns: - ------- - None. - """ - - clean_registry(self.past_actions) - def step(self, action): """Executes one action in the environment. @@ -298,72 +344,28 @@ def step(self, action): """ - # assert self.action_space.contains(action), "%r (%s) invalid" % (action, type(action)) action = self._formataction(action) - # Pass the actions to the server to launch the simulation. - result_id = add_new_step(self.past_actions, action) - self.past_actions.append(action) + self.pos = self.step_simulation(action) - # Request results from the server. - # print("[INFO] >>> Result id:", result_id) - results = get_result(result_id, timeout=self.timeout) + self.past_actions.append(action) + self.past_pos.append(self.pos) - obs = np.array(results["observation"]) # to work with baseline - reward = results["reward"] - done = results["done"] + obs = np.array(self._getState(self.root), dtype=np.float32) + done, reward = self._getReward(self.root) # Avoid long explorations by using a timer. self.timer += 1 if self.timer >= self.config["timer_limit"]: # reward = -150 truncated = True - info={}#(not use here) + + info = {} #(not use here) - if self.config["planning"]: - self.clean() return obs, reward, done, info - - def async_step(self, action): - """Executes one action in the environment. - - Apply action and execute scale_factor simulation steps of 0.01 s. - Like step but useful if you want to parallelise (blocking "get"). - Otherwise use step. - - Parameters: - ---------- - action: int - Action applied in the environment. - - Returns: - ------- - LateResult: - Class which allows to store the id of the client who performs - the calculation and to return later the usual information - (observation, reward, done) thanks to a get method. - - """ - assert self.action_space.contains(action), "%r (%s) invalid" % (action, type(action)) - - result_id = add_new_step(self.past_actions, action) - self.past_actions.append(action) - - class LateResult: - def __init__(self, result_id): - self.result_id = result_id - - def get(self, timeout=None): - results = get_result(self.result_id, timeout=timeout) - obs = results["observation"] - reward = results["reward"] - done = results["done"] - return obs, reward, done, {} - - return LateResult(copy.copy(result_id)) - + def reset(self): """Reset simulation. @@ -376,22 +378,22 @@ def reset(self): obs, info """ - self.clean() self.viewer = None splib3.animation.animate.manager = None - if not self.goalList: - self.goalList = self.config["goalList"] - - # Set a new random goal from the list - id_goal = self.np_random.choice(range(len(self.goalList))) - self.config.update({'goal_node': id_goal}) - self.goal = self.goalList[id_goal] self.timer = 0 self.past_actions = [] + self.pos = [] + self.past_pos = [] + + Sofa.Simulation.reset(self.root) + self.root = None + self.init_simulation() - return + obs = np.array(self._getState(self.root), dtype=np.float32) + + return obs def render(self, mode): """See the current state of the environment. @@ -411,40 +413,16 @@ def render(self, mode): self.render_mode = mode # Define the viewer at the first run of render. - if not self.viewer: + if self.viewer is None: display_size = self.config["display_size"] # Sim display if 'zFar' in self.config: zFar = self.config['zFar'] else: zFar = 0 - self.viewer = Viewer(self, display_size, zFar=zFar, save_path=self.config["save_path_image"]) + self.viewer = Viewer(self, self.root, display_size, zFar=zFar, save_path=self.config["save_path_image"]) # Use the viewer to display the environment. return self.viewer.render() - def _automatic_rendering(self): - """Automatically render the intermediate frames while an action is still ongoing. - - This allows to render the whole video and not only single steps - corresponding to agent decision-making. - If a callback has been set, use it to perform the rendering. This is - useful for the environment wrappers such as video-recording monitor that - need to access these intermediate renderings. - - Parameters: - ---------- - None. - - Returns: - ------- - None. - - """ - if self.viewer is not None: - if self.automatic_rendering_callback: - self.automatic_rendering_callback() - else: - self.render() - def close(self): """Terminate simulation. @@ -460,8 +438,7 @@ def close(self): """ if self.viewer is not None: self.viewer.close() - - close_scene() + print("All clients are closed. Bye Bye.") def configure(self, config): @@ -478,3 +455,104 @@ def configure(self, config): """ self.config.update(config) + + def init_simulation(self, mode='simu_and_visu'): + """Function to create scene and initialize all variables. + + Parameters: + ---------- + config: Dictionary. + Configuration of the environment. + _startCmd: function + Initialize the command. + mode: string, default = 'simu_and_visu' + Init a scene with or without visu and computations. + In ['simu', 'visu', 'simu_and_visu'] + + Returns: + ------- + root: + The loaded and initialized scene. + + """ + # Load the scene + self.root = Sofa.Core.Node("root") + + SofaRuntime.importPlugin("Sofa.Component") + self.create_scene(self.root, self.config, mode=mode) + Sofa.Simulation.init(self.root) + + # Realise action from history + if self.config['start_from_history'] is not None and self._startCmd is not None: + print(">> Start from history ...") + render = self.config['render'] + self.config.update({'render': 0}) + + for action in self.config['start_from_history']: + self.step_simulation(action) + + self.config.update({'render': render}) + print(">> ... Done.") + + if self.config["randomize_states"]: + self.root.StateInitializer.init_state(self.config["init_states"]) + + if 'time_before_start' in self.config: + print(">> Time before start:", self.config["time_before_start"], "steps. Initialization ...") + for _ in range(self.config["time_before_start"]): + Sofa.Simulation.animate(self.root, self.config["dt"]) + print(">> ... Done.") + + # Update Reward and GoalSetter + if self.config["goal"]: + self.root.GoalSetter.update(self.goal) + + self.root.Reward.update(self.goal) + + def step_simulation(self, action): + """Realise one step in the simulation. + + Apply action and execute 5 times scale_factor simulation steps of dt s. + + Parameters: + ---------- + root: + The scene. + config: Dictionary + The configuration of the environment. + action: int + The action to apply in the environment. + _startCmd: function + Initialize the command. + _getPos: function + Get the position of the object in the scene. + + Returns: + -------- + position(s): list + The positions of object(s) in the scene. + + """ + if self.config["goal"]: + goal = self.config['goalPos'] + self.root.GoalSetter.set_mo_pos(goal) + + render = self.config['render'] + surface_size = self.config['display_size'] + + # Create the command from action + self._startCmd(self.root, action, self.config["dt"]*(self.config["scale_factor"]-1)) + pos = [] + # Realise scale_factor simulation steps of 0.01 s + for _ in range(self.config["scale_factor"]): + Sofa.Simulation.animate(self.root, self.config["dt"]) + + #if render == 2: + # pos.append(self._getPos(self.root)) + # if self.viewer is not None: + # self.viewer.render_simulation(self.root) + + if render == 1: + pos.append(self._getPos(self.root)) + + return pos diff --git a/sofagym/ServerEnv.py b/sofagym/ServerEnv.py new file mode 100644 index 0000000..c2c8a92 --- /dev/null +++ b/sofagym/ServerEnv.py @@ -0,0 +1,168 @@ +# -*- coding: utf-8 -*- +"""ServerEnv to use the server-client architecture. +""" + +__authors__ = ("PSC", "dmarchal", "emenager") +__contact__ = ("pierre.schegg@robocath.com", "damien.marchal@univ-lille.fr", "etienne.menager@ens-rennes.fr") +__version__ = "1.0.0" +__copyright__ = "(c) 2020, Robocath, CNRS, Inria" +__date__ = "Oct 7 2020" + +from typing import Optional + +import numpy as np +import copy + +import splib3 + +from sofagym.rpc_server import start_server, add_new_step, get_result, clean_registry, close_scene + +from sofagym.AbstractEnv import AbstractEnv + + +class ServerEnv(AbstractEnv): + def __init__(self, default_config, config=None, render_mode: Optional[str]=None, root=None): + super().__init__(default_config, config, root=root) + + # Start the server which distributes the calculations to its clients + start_server(self.config) + + def step(self, action): + """Executes one action in the environment. + + Apply action and execute scale_factor simulation steps of 0.01 s. + + Parameters: + ---------- + action: int + Action applied in the environment. + + Returns: + ------- + obs(ObsType): + The new state of the agent. + reward(float): + The reward obtain after applying the action in the current state. + done(bool): + Whether the agent reaches the terminal state + info(dict): + additional information (not used here) + """ + # assert self.action_space.contains(action), "%r (%s) invalid" % (action, type(action)) + + action = self._formataction(action) + + # Pass the actions to the server to launch the simulation. + result_id = add_new_step(self.past_actions, action) + self.past_actions.append(action) + + # Request results from the server. + # print("[INFO] >>> Result id:", result_id) + results = get_result(result_id, timeout=self.timeout) + + obs = np.array(results["observation"]) # to work with baseline + reward = results["reward"] + done = results["done"] + + # Avoid long explorations by using a timer. + self.timer += 1 + if self.timer >= self.config["timer_limit"]: + # reward = -150 + truncated = True + + info = {} #(not use here) + + if self.config["planning"]: + self.clean() + return obs, reward, done, info + + def async_step(self, action): + """Executes one action in the environment. + + Apply action and execute scale_factor simulation steps of 0.01 s. + Like step but useful if you want to parallelise (blocking "get"). + Otherwise use step. + + Parameters: + ---------- + action: int + Action applied in the environment. + + Returns: + ------- + LateResult: + Class which allows to store the id of the client who performs + the calculation and to return later the usual information + (observation, reward, done) thanks to a get method. + + """ + assert self.action_space.contains(action), "%r (%s) invalid" % (action, type(action)) + + result_id = add_new_step(self.past_actions, action) + self.past_actions.append(action) + + class LateResult: + def __init__(self, result_id): + self.result_id = result_id + + def get(self, timeout=None): + results = get_result(self.result_id, timeout=timeout) + obs = results["observation"] + reward = results["reward"] + done = results["done"] + return obs, reward, done, {} + + return LateResult(copy.copy(result_id)) + + def reset(self): + """Reset simulation. + + Parameters: + ---------- + None. + + Returns: + ------- + obs, info + """ + self.clean() + self.viewer = None + + splib3.animation.animate.manager = None + + self.timer = 0 + self.past_actions = [] + + return + + def clean(self): + """Function to clean the registery . + + Close clients who are processing unused sequences of actions (for + planning) + + Parameters: + ---------- + None. + + Returns: + ------- + None. + """ + clean_registry(self.past_actions) + + def close(self): + """Terminate simulation. + + Close the viewer and the scene. + + Parametres: + ---------- + None. + + Returns: + ------- + None. + """ + super().close() + close_scene() diff --git a/sofagym/envs/CartPole/CartPoleEnv.py b/sofagym/envs/CartPole/CartPoleEnv.py index 75280b9..a8c98ac 100644 --- a/sofagym/envs/CartPole/CartPoleEnv.py +++ b/sofagym/envs/CartPole/CartPoleEnv.py @@ -1,26 +1,29 @@ import math import os import sys +from typing import Optional import numpy as np from gym import spaces from sofagym.AbstractEnv import AbstractEnv +from sofagym.ServerEnv import ServerEnv from sofagym.rpc_server import start_scene -class CartPoleEnv(AbstractEnv): +class CartPoleEnv: """Sub-class of AbstractEnv, dedicated to the cart pole scene. See the class AbstractEnv for arguments and methods. """ #Setting a default configuration - path = path = os.path.dirname(os.path.abspath(__file__)) + path = os.path.dirname(os.path.abspath(__file__)) metadata = {'render.modes': ['human', 'rgb_array']} + dim_state = 4 DEFAULT_CONFIG = {"scene": "CartPole", "deterministic": True, "source": [0, 0, 160], "target": [0, 0, 0], - "goalList": [[0]], + "goal": False, "start_node": None, "scale_factor": 10, "dt": 0.001, @@ -38,16 +41,31 @@ class CartPoleEnv(AbstractEnv): "zFar": 4000, "time_before_start": 0, "seed": None, + "nb_actions": 2, + "dim_state": dim_state, "init_x": 0, - "max_move": 24, + "x_threshold": 100, + "max_move": 12, + "max_angle": 0.418, + "randomize_states": True, + "init_states": [0] * dim_state, + "use_server": False } - def __init__(self, config=None): - super().__init__(config) + def __init__(self, config = None, root=None, use_server: Optional[bool]=None): + if use_server is not None: + self.DEFAULT_CONFIG.update({'use_server': use_server}) + self.use_server = self.DEFAULT_CONFIG["use_server"] + self.env = ServerEnv(self.DEFAULT_CONFIG, config, root=root) if self.use_server else AbstractEnv(self.DEFAULT_CONFIG, config, root=root) - self.x_threshold = 100 - self.theta_threshold_radians = self.config["max_move"] * math.pi / 180 - self.config.update({'max_angle': self.theta_threshold_radians}) + self.initialize_states() + + if self.env.config["goal"]: + self.init_goal() + + self.x_threshold = self.env.config["x_threshold"] + self.theta_threshold_radians = self.env.config["max_move"] * math.pi / 180 + self.env.config.update({'max_angle': self.theta_threshold_radians}) high = np.array( [ @@ -59,38 +77,56 @@ def __init__(self, config=None): dtype=np.float32, ) - nb_actions = 2 - self.action_space = spaces.Discrete(nb_actions) - self.nb_actions = str(nb_actions) + self.env.action_space = spaces.Discrete(self.env.nb_actions) + self.nb_actions = str(self.env.nb_actions) - self.observation_space = spaces.Box(-high, high, dtype=np.float32) + self.env.observation_space = spaces.Box(-high, high, dtype=np.float32) - def reset(self): - """Reset simulation. + if self.env.root is None and not self.use_server: + self.env.init_root() + + # called when an attribute is not found: + def __getattr__(self, name): + # assume it is implemented by self.instance + return self.env.__getattribute__(name) + def initialize_states(self): + if self.env.config["randomize_states"]: + self.init_states = self.randomize_init_states() + self.env.config.update({'init_states': list(self.init_states)}) + else: + self.init_states = self.env.config["init_states"] + + def randomize_init_states(self): + """Randomize initial states. + + Returns: + ------- + init_states: list + List of random initial states for the environment. + Note: ---- - We launch a client to create the scene. The scene of the program is - client_Env.py. - + This method should be implemented according to needed random initialization. """ - super().reset() + init_states = self.env.np_random.uniform(low=-0.05, high=0.05, size=(self.env.config["dim_state"],)) - self.config.update({'goalPos': self.goal}) - init_states = self.np_random.uniform(low=-0.05, high=0.05, size=(4,)) - self.config.update({'init_states': list(init_states)}) - obs = start_scene(self.config, self.nb_actions) - return np.array(obs['observation']) + return init_states - def get_available_actions(self): - """Gives the actions available in the environment. - - Parameters: - ---------- - None. - - Returns: - ------- - list of the action available in the environment. + def reset(self): + """Reset simulation. """ - return self.action_space + self.initialize_states() + + if self.env.config["goal"]: + self.init_goal() + + self.env.reset() + + if self.use_server: + obs = start_scene(self.env.config, self.nb_actions) + state = np.array(obs['observation'], dtype=np.float32) + else: + state = np.array(self.env._getState(self.env.root), dtype=np.float32) + + return state diff --git a/sofagym/envs/CartPole/CartPoleScene.py b/sofagym/envs/CartPole/CartPoleScene.py index 7d5a58b..bc6773d 100644 --- a/sofagym/envs/CartPole/CartPoleScene.py +++ b/sofagym/envs/CartPole/CartPoleScene.py @@ -4,7 +4,7 @@ sys.path.insert(0, str(pathlib.Path(__file__).parent.absolute())+"/../") sys.path.insert(0, str(pathlib.Path(__file__).parent.absolute())) -from CartPoleToolbox import ApplyAction, GoalSetter, RewardShaper, StateInitializer +from CartPoleToolbox import ApplyAction, RewardShaper, StateInitializer from sofagym.header import addVisu from splib3.animation import AnimationManagerController from stlib3.physics.rigid import Floor @@ -48,16 +48,16 @@ def addRigidObject(node, filename, collisionFilename=None, position=[0, 0, 0, 0, return object -def createScene(root, +def createScene(root, config={"source": [0, 0, 160], "target": [0, 0, 0], - "goalPos": None, "seed": None, "zFar":4000, "init_x": 0, "max_move": 24, "max_angle": 0.418, - "dt": 0.01}, + "dt": 0.01, + "init_states": [0]*4}, mode='simu_and_visu'): # Choose the mode: visualization or computations (or both) @@ -167,5 +167,4 @@ def createScene(root, # SofaGym Env Components root.addObject(StateInitializer(name="StateInitializer", rootNode=root, pole_length=pole_length, init_states=config['init_states'])) root.addObject(RewardShaper(name="Reward", rootNode=root, max_angle=config['max_angle'], pole_length=pole_length)) - root.addObject(GoalSetter(name="GoalSetter")) root.addObject(ApplyAction(name="ApplyAction", root=root)) diff --git a/sofagym/envs/CartPole/CartPoleToolbox.py b/sofagym/envs/CartPole/CartPoleToolbox.py index c323dc6..dac1582 100644 --- a/sofagym/envs/CartPole/CartPoleToolbox.py +++ b/sofagym/envs/CartPole/CartPoleToolbox.py @@ -52,13 +52,13 @@ def __init__(self, *args, **kwargs): if kwargs["init_states"] is not None: self.init_states = kwargs["init_states"] else: - print(">> ERROR: no inital states given.") + print(">> ERROR: no initial states given.") exit(1) self.cart = self.rootNode.Modeling.Cart self.pole = self.rootNode.Modeling.Pole - def init_state(self): + def init_state(self, init_states): """Randomly initialize the environment state. Parameters: @@ -70,6 +70,7 @@ def init_state(self): None. """ + self.init_states = init_states cart_pos, cart_vel, pole_theta, pole_theta_dot = self.init_states with self.cart.MechanicalObject.position.writeable() as position: @@ -124,14 +125,11 @@ def __init__(self, *args, **kwargs): self.rootNode = None if kwargs["rootNode"]: self.rootNode = kwargs["rootNode"] - if kwargs["max_angle"]: + if kwargs["max_angle"] is not None: self.max_angle = kwargs["max_angle"] - else: - print(">> ERROR: give a max angle for the normalization of the reward.") - exit(1) - if kwargs["pole_length"]: + if kwargs["pole_length"] is not None: self.pole_length = kwargs["pole_length"] - + self.cart = self.rootNode.Modeling.Cart self.pole = self.rootNode.Modeling.Pole @@ -147,12 +145,11 @@ def getReward(self): The reward and the cost. """ - cart_pos = self._getCartPos() - pole_theta, pole_theta_dot = self.calculatePoleTheta(cart_pos) + pole_theta = self.pole.MechanicalObject.position.value.tolist()[0][5] return 1, pole_theta, self.max_angle - def update(self): + def update(self, goal=None): """Update function. This function is used as an initialization function. @@ -168,26 +165,6 @@ def update(self): """ pass - def _getPolePos(self): - pos = self.pole.MechanicalObject.position.value.tolist()[0] - return pos[0], pos[1] - - def _getCartPos(self): - pos = self.cart.MechanicalObject.position.value.tolist()[0][0] - return pos - - def calculatePoleTheta(self, cart_pos): - x_pos, y_pos = self._getPolePos() - sin_theta = (y_pos/self.pole_length) - theta = abs((90*math.pi/180) - math.asin(sin_theta)) - - if x_pos < cart_pos: - theta = -theta - - theta_dot = self.pole.MechanicalObject.velocity.value.tolist()[0][5] - - return theta, theta_dot - def getState(rootNode): """Compute the state of the environment/agent. @@ -203,28 +180,18 @@ def getState(rootNode): The state of the environment/agent. """ cart = rootNode.Modeling.Cart - cart_pos = cart.MechanicalObject.position.value.tolist()[0][0] cart_vel = cart.MechanicalObject.velocity.value.tolist()[0][0] - pole_theta, pole_theta_dot = rootNode.Reward.calculatePoleTheta(cart_pos) + pole = rootNode.Modeling.Pole + pole_theta = pole.MechanicalObject.position.value.tolist()[0][5] + pole_theta_dot = pole.MechanicalObject.velocity.value.tolist()[0][5] state = [cart_pos, cart_vel, pole_theta, pole_theta_dot] return state -class GoalSetter(Sofa.Core.Controller): - def __init__(self, *args, **kwargs): - Sofa.Core.Controller.__init__(self, *args, **kwargs) - - def update(self): - pass - - def set_mo_pos(self, goal): - pass - - def getReward(rootNode): reward, theta, max_angle = rootNode.Reward.getReward() done = (theta > max_angle) or (theta < -max_angle) @@ -330,6 +297,7 @@ def startCmd_CartPole(rootNode, incr, duration): """ # Definition of the elements of the animation + def executeAnimation(rootNode, incr, factor): rootNode.ApplyAction.apply_action(incr) diff --git a/sofagym/simulate.py b/sofagym/simulate.py index 4ac2dab..b06b52f 100644 --- a/sofagym/simulate.py +++ b/sofagym/simulate.py @@ -61,22 +61,20 @@ def init_simulation(config, _startCmd=None, mode="simu_and_visu"): config.update({'render': render}) print(">> ... Done.") - # Init Reward and GoalSetter - root.GoalSetter.update() - root.Reward.update() - - try: - root.StateInitializer.init_state() - except AttributeError as error: - print(error) + if config["randomize_states"]: + root.StateInitializer.init_state(config["init_states"]) if 'time_before_start' in config: print(">> Time before start:", config["time_before_start"], "steps. Initialization ...") for i in range(config["time_before_start"]): Sofa.Simulation.animate(root, config["dt"]) print(">> ... Done.") - # Update Reward and GoalSetter - root.GoalSetter.update() + + # Update Reward and GoalSetter + if config["goal"]: + root.GoalSetter.update(config['goalPos']) + root.Reward.update(config['goalPos']) + else: root.Reward.update() return root @@ -106,11 +104,13 @@ def step_simulation(root, config, action, _startCmd, _getPos, viewer=None): The positions of object(s) in the scene. """ - goal = config['goalPos'] + if config["goal"]: + goal = config['goalPos'] + root.GoalSetter.set_mo_pos(goal) + render = config['render'] surface_size = config['display_size'] - root.GoalSetter.set_mo_pos(goal) # Create the command from action _startCmd(root, action, config["dt"]*(config["scale_factor"]-1)) diff --git a/sofagym/viewer.py b/sofagym/viewer.py index 7241f68..19c5866 100644 --- a/sofagym/viewer.py +++ b/sofagym/viewer.py @@ -62,7 +62,7 @@ class Viewer: """ - def __init__(self, env, surface_size, zFar=100, save_path=None, create_video=None, fps=10): + def __init__(self, env, root, surface_size, zFar=100, save_path=None, create_video=None, fps=10): """ Classic initialization of a class in python. @@ -97,7 +97,11 @@ def __init__(self, env, surface_size, zFar=100, save_path=None, create_video=Non self.agent_display = None self.frame = 0 - self.root = init_simulation(self.env.config, mode = 'visu') + if self.env.config["use_server"]: + self.root = init_simulation(self.env.config, mode = 'visu') + else: + self.root = root + scene = self.env.config['scene'] self._setPos = importlib.import_module("sofagym.envs."+scene+"."+scene+"Toolbox").setPos @@ -132,15 +136,21 @@ def render(self, pos=None): # Recovering an image and handling error cases try: if pos is None: - pos = get_position(self.env.past_actions)['position'] + if self.env.config["use_server"]: + pos = get_position(self.env.past_actions)['position'] + else: + pos = self.env.pos + self.frame += 1 + if pos == []: image = np.zeros((self.surface_size[0], self.surface_size[1], 3)) else: num_im = 0 for p in pos: - self._setPos(self.root, p) - Sofa.Simulation.animate(self.root, self.root.getDt()) + if self.env.config["use_server"]: + self._setPos(self.root, p) + Sofa.Simulation.animate(self.root, self.root.getDt()) glViewport(0, 0, self.surface_size[0], self.surface_size[1]) From 4acd33a3e3da576df3af3d6015970432d69a81da Mon Sep 17 00:00:00 2001 From: Samuel Youssef Date: Sun, 20 Oct 2024 12:17:11 +0200 Subject: [PATCH 85/85] Fixing ServerEnv import --- sofagym/envs/BubbleMotion/BubbleMotionEnv.py | 3 ++- sofagym/envs/CTR/CTREnv.py | 3 ++- sofagym/envs/CartStem/CartStemEnv.py | 3 ++- sofagym/envs/CartStemContact/CartStemContactEnv.py | 3 ++- sofagym/envs/CatchTheObject/CatchTheObjectEnv.py | 3 ++- sofagym/envs/CatheterBeam/CatheterBeamEnv.py | 3 ++- sofagym/envs/Diamond/DiamondEnv.py | 3 ++- sofagym/envs/Gripper/GripperEnv.py | 3 ++- sofagym/envs/Maze/MazeEnv.py | 3 ++- sofagym/envs/SimpleMaze/SimpleMazeEnv.py | 3 ++- sofagym/envs/StemPendulum/StemPendulumEnv.py | 3 ++- sofagym/envs/Trunk/TrunkEnv.py | 3 ++- sofagym/envs/TrunkCup/TrunkCupEnv.py | 3 ++- 13 files changed, 26 insertions(+), 13 deletions(-) diff --git a/sofagym/envs/BubbleMotion/BubbleMotionEnv.py b/sofagym/envs/BubbleMotion/BubbleMotionEnv.py index 67151c9..824315c 100644 --- a/sofagym/envs/BubbleMotion/BubbleMotionEnv.py +++ b/sofagym/envs/BubbleMotion/BubbleMotionEnv.py @@ -8,7 +8,8 @@ __copyright__ = "(c) 2021, Inria" __date__ = "Feb 3 2021" -from sofagym.AbstractEnv import AbstractEnv, ServerEnv +from sofagym.AbstractEnv import AbstractEnv +from sofagym.ServerEnv import ServerEnv from sofagym.rpc_server import start_scene from gym import spaces diff --git a/sofagym/envs/CTR/CTREnv.py b/sofagym/envs/CTR/CTREnv.py index 0f175fb..41baa0f 100644 --- a/sofagym/envs/CTR/CTREnv.py +++ b/sofagym/envs/CTR/CTREnv.py @@ -8,7 +8,8 @@ __copyright__ = "(c) 2021, Robocath, CNRS, Inria" __date__ = "Dec 01 2021" -from sofagym.AbstractEnv import AbstractEnv, ServerEnv +from sofagym.AbstractEnv import AbstractEnv +from sofagym.ServerEnv import ServerEnv from sofagym.rpc_server import start_scene from sofagym.viewer import LegacyViewer from sofagym.envs.CTR.CTRToolbox import startCmd diff --git a/sofagym/envs/CartStem/CartStemEnv.py b/sofagym/envs/CartStem/CartStemEnv.py index ece07f8..410fcbb 100644 --- a/sofagym/envs/CartStem/CartStemEnv.py +++ b/sofagym/envs/CartStem/CartStemEnv.py @@ -8,7 +8,8 @@ __copyright__ = "(c) 2021, Inria" __date__ = "Feb 3 2021" -from sofagym.AbstractEnv import AbstractEnv, ServerEnv +from sofagym.AbstractEnv import AbstractEnv +from sofagym.ServerEnv import ServerEnv from sofagym.rpc_server import start_scene from gym import spaces diff --git a/sofagym/envs/CartStemContact/CartStemContactEnv.py b/sofagym/envs/CartStemContact/CartStemContactEnv.py index 3eebd57..7b5ef88 100644 --- a/sofagym/envs/CartStemContact/CartStemContactEnv.py +++ b/sofagym/envs/CartStemContact/CartStemContactEnv.py @@ -10,7 +10,8 @@ import os, sys -from sofagym.AbstractEnv import AbstractEnv, ServerEnv +from sofagym.AbstractEnv import AbstractEnv +from sofagym.ServerEnv import ServerEnv from sofagym.rpc_server import start_scene from gym import spaces diff --git a/sofagym/envs/CatchTheObject/CatchTheObjectEnv.py b/sofagym/envs/CatchTheObject/CatchTheObjectEnv.py index ef6bbc3..1535a26 100644 --- a/sofagym/envs/CatchTheObject/CatchTheObjectEnv.py +++ b/sofagym/envs/CatchTheObject/CatchTheObjectEnv.py @@ -10,7 +10,8 @@ import os, sys -from sofagym.AbstractEnv import AbstractEnv, ServerEnv +from sofagym.AbstractEnv import AbstractEnv +from sofagym.ServerEnv import ServerEnv from sofagym.rpc_server import start_scene from gym import spaces diff --git a/sofagym/envs/CatheterBeam/CatheterBeamEnv.py b/sofagym/envs/CatheterBeam/CatheterBeamEnv.py index 1400b14..87c88e7 100644 --- a/sofagym/envs/CatheterBeam/CatheterBeamEnv.py +++ b/sofagym/envs/CatheterBeam/CatheterBeamEnv.py @@ -5,7 +5,8 @@ import numpy as np from gym import spaces -from sofagym.AbstractEnv import AbstractEnv, ServerEnv +from sofagym.AbstractEnv import AbstractEnv +from sofagym.ServerEnv import ServerEnv from sofagym.rpc_server import start_scene diff --git a/sofagym/envs/Diamond/DiamondEnv.py b/sofagym/envs/Diamond/DiamondEnv.py index 86c92f1..ebe296b 100644 --- a/sofagym/envs/Diamond/DiamondEnv.py +++ b/sofagym/envs/Diamond/DiamondEnv.py @@ -10,7 +10,8 @@ import os, sys -from sofagym.AbstractEnv import AbstractEnv, ServerEnv +from sofagym.AbstractEnv import AbstractEnv +from sofagym.ServerEnv import ServerEnv from sofagym.rpc_server import start_scene from sofagym.viewer import LegacyViewer from sofagym.envs.Diamond.DiamondToolbox import startCmd diff --git a/sofagym/envs/Gripper/GripperEnv.py b/sofagym/envs/Gripper/GripperEnv.py index beb18cc..00cafbd 100644 --- a/sofagym/envs/Gripper/GripperEnv.py +++ b/sofagym/envs/Gripper/GripperEnv.py @@ -11,7 +11,8 @@ import os, sys import numpy as np -from sofagym.AbstractEnv import AbstractEnv, ServerEnv +from sofagym.AbstractEnv import AbstractEnv +from sofagym.ServerEnv import ServerEnv from sofagym.rpc_server import start_scene from gym import spaces diff --git a/sofagym/envs/Maze/MazeEnv.py b/sofagym/envs/Maze/MazeEnv.py index 0d8df3a..ce1da19 100644 --- a/sofagym/envs/Maze/MazeEnv.py +++ b/sofagym/envs/Maze/MazeEnv.py @@ -11,7 +11,8 @@ import os, sys import numpy as np -from sofagym.AbstractEnv import AbstractEnv, ServerEnv +from sofagym.AbstractEnv import AbstractEnv +from sofagym.ServerEnv import ServerEnv from sofagym.rpc_server import start_scene from gym import spaces diff --git a/sofagym/envs/SimpleMaze/SimpleMazeEnv.py b/sofagym/envs/SimpleMaze/SimpleMazeEnv.py index 4af6110..155a951 100644 --- a/sofagym/envs/SimpleMaze/SimpleMazeEnv.py +++ b/sofagym/envs/SimpleMaze/SimpleMazeEnv.py @@ -11,7 +11,8 @@ import os, sys import numpy as np -from sofagym.AbstractEnv import AbstractEnv, ServerEnv +from sofagym.AbstractEnv import AbstractEnv +from sofagym.ServerEnv import ServerEnv from sofagym.rpc_server import start_scene from gym import spaces diff --git a/sofagym/envs/StemPendulum/StemPendulumEnv.py b/sofagym/envs/StemPendulum/StemPendulumEnv.py index 54209a8..b84e53d 100644 --- a/sofagym/envs/StemPendulum/StemPendulumEnv.py +++ b/sofagym/envs/StemPendulum/StemPendulumEnv.py @@ -11,7 +11,8 @@ import os, sys import numpy as np -from sofagym.AbstractEnv import AbstractEnv, ServerEnv +from sofagym.AbstractEnv import AbstractEnv +from sofagym.ServerEnv import ServerEnv from sofagym.rpc_server import start_scene from gym import spaces diff --git a/sofagym/envs/Trunk/TrunkEnv.py b/sofagym/envs/Trunk/TrunkEnv.py index 810e587..5d555aa 100644 --- a/sofagym/envs/Trunk/TrunkEnv.py +++ b/sofagym/envs/Trunk/TrunkEnv.py @@ -12,7 +12,8 @@ import numpy as np import sys -from sofagym.AbstractEnv import AbstractEnv, ServerEnv +from sofagym.AbstractEnv import AbstractEnv +from sofagym.ServerEnv import ServerEnv from sofagym.rpc_server import start_scene from gym import spaces diff --git a/sofagym/envs/TrunkCup/TrunkCupEnv.py b/sofagym/envs/TrunkCup/TrunkCupEnv.py index bd3f05a..3b08378 100644 --- a/sofagym/envs/TrunkCup/TrunkCupEnv.py +++ b/sofagym/envs/TrunkCup/TrunkCupEnv.py @@ -11,7 +11,8 @@ import os, sys import numpy as np -from sofagym.AbstractEnv import AbstractEnv, ServerEnv +from sofagym.AbstractEnv import AbstractEnv +from sofagym.ServerEnv import ServerEnv from sofagym.rpc_server import start_scene from gym import spaces