Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Feature rapid neem creation #216

Open
wants to merge 49 commits into
base: dev
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from 32 commits
Commits
Show all changes
49 commits
Select commit Hold shift + click to select a range
8d0d50d
Adds the initial conceptual idea, which seems to be working
Leusmann Aug 16, 2024
fce61af
Adds first ProcThor environment
Leusmann Aug 16, 2024
2f113ff
Kind of working demo
Leusmann Aug 20, 2024
e523797
Stop ignoring the Giskard answer when spawning the mesh
Leusmann Aug 20, 2024
93e1cad
Merge branch 'feature/rapid-neem-creation' into HEAD
Leusmann Aug 20, 2024
15666ba
Adds file// to the replaced key words
Leusmann Aug 20, 2024
c144628
Adds the functionallty to create NEEMs of the task executed.
Leusmann Aug 29, 2024
2c9ad5b
Adds first Multivers experiment
Leusmann Aug 29, 2024
6d126f7
Quick fix to store the object pose at this moment and not the object …
Leusmann Aug 29, 2024
d81e87f
Quick and dirty fix for object poses of believed object. Shouldn't be…
Leusmann Aug 29, 2024
e1b7abd
Changes camera starting postion
Leusmann Aug 29, 2024
9c91788
Adds the Micheal function and the second version
Leusmann Sep 16, 2024
3139a6b
Changes from script so function and adds a second version where the e…
Leusmann Sep 16, 2024
38e1910
Adds the download helper functions. Maybe need to check the paths etc
Leusmann Sep 16, 2024
568cd18
Quick fix for equality. Apparently pybullet creates ids in such a way…
Leusmann Sep 17, 2024
8e62e9c
Merge remote-tracking branch 'origin/dev' into feature-rapid-neem-cre…
Leusmann Oct 4, 2024
7d1feb7
small change to refelct the changes in the Enum
Leusmann Oct 18, 2024
3d737ae
Merge remote-tracking branch 'origin/dev' into feature-rapid-neem-cre…
Leusmann Oct 18, 2024
62af001
Move from functions to class
Leusmann Oct 29, 2024
ea85ba6
Changed from functions to class and simplyfies the download
Leusmann Oct 29, 2024
e2b43e1
Delete house_2
Leusmann Oct 29, 2024
5ebd4e6
Adds documentation strings etc
Leusmann Nov 7, 2024
3518551
Removed unnecessary imports and switched to pycramros lib instead of …
Leusmann Nov 7, 2024
37e2c4a
Changes requester to procthor interface and store it at the proper place
Leusmann Nov 7, 2024
71a469f
Refelct the changes from moving 'requester' to 'external_interface/pr…
Leusmann Nov 7, 2024
0a6db3a
Deleted old version of function and some other stuff
Leusmann Nov 8, 2024
23134bd
Adds the first iteration of the procthor interface example
Leusmann Nov 8, 2024
1c9190d
Refactor function name
Leusmann Nov 20, 2024
cd3bf47
Removes unnecessary comments and refactors function name
Leusmann Nov 20, 2024
700b7a3
Removes old iteration, which somehow made it this far
Leusmann Nov 20, 2024
6a056a5
Removes old iteration, which somehow made it this far
Leusmann Nov 20, 2024
39537b9
Renames CEO in to demo
Leusmann Nov 20, 2024
87e3a98
First iteration of the experiment Interface
Leusmann Nov 27, 2024
200eb14
Changes prints to log messages
Leusmann Dec 4, 2024
f095935
First draft of the PyCRAMGym class
Leusmann Dec 4, 2024
bf5a530
Delete unnecessary files again
Leusmann Dec 4, 2024
b26e94c
Adds the tests for the procthor interface
Leusmann Dec 5, 2024
fa38542
Merge branch 'dev' of github.com:cram2/pycram into feature-rapid-neem…
Leusmann Dec 5, 2024
123d14c
Change function name and remove prints
Leusmann Dec 5, 2024
35593c8
Changes print to log
Leusmann Dec 9, 2024
242ae86
Updates the example to properly log instead of print and use the newe…
Leusmann Dec 9, 2024
924f9e1
Adds the example code for testig purposes
Leusmann Dec 9, 2024
bf2c1a4
Moves example code to the proper file and import that one
Leusmann Dec 9, 2024
a7f7113
Merge remote-tracking branch 'origin/dev' into feature-rapid-neem-cre…
Leusmann Dec 10, 2024
e2725ee
Switches from Enum to pyCRAP
Leusmann Dec 10, 2024
cb46fc6
Changes imports to be realtive
Leusmann Dec 11, 2024
3cff20c
Changes from ObjectTypes to PyCRAP and adds some documentation
Leusmann Dec 11, 2024
f518323
Adds gymnasium to the requirements
Leusmann Dec 11, 2024
00105e5
Restes the camera back to its default position
Leusmann Dec 11, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
97 changes: 97 additions & 0 deletions demos/pycram_procthor_demo/demo.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,97 @@
import rospy

from pycram.worlds.bullet_world import BulletWorld
from pycram.datastructures.enums import ObjectType, WorldMode
from pycram.world_concepts.world_object import Object
from pycram.datastructures.pose import Pose

from pycram.datastructures.dataclasses import Color
from demos.playingfield.sandcastle import generic_plan
from pycram import World
from pycram.failures import PlanFailure
from pycram.tasktree import task_tree

import traceback
import sqlalchemy
import pycram.orm.base
from pycram.external_interfaces.procthor import ProcThorInterface

engine = sqlalchemy.create_engine("sqlite+pysqlite:///:memory:", echo=False)
session_maker = sqlalchemy.orm.sessionmaker(bind=engine)
session = session_maker()
# ToDo: Many Plans missing, but this needs function pointers and I am not comfortable with it
# plans=[]
robots = []
robot_name = "pr2.urdf"
robots.append(robot_name)
pycram.orm.base.Base.metadata.create_all(engine)
session.commit()
# Set up for ProcThor stuff Example usage
procThorInterface = ProcThorInterface(base_url="http://procthor.informatik.uni-bremen.de:5000/")
number_of_test_environment = 5

# Get Environments
number_of_known_environment = len(
procThorInterface.get_all_environments_stored_below_dictionary(procThorInterface.source_folder))
print("Number of known Environments:{}".format(number_of_known_environment))
print("Number of needed Testenvironments:{}".format(number_of_test_environment))

if number_of_known_environment < number_of_test_environment:
print("Downloading missing environments...")
procThorInterface.download_num_random_environment(number_of_test_environment - number_of_known_environment)

def runWorld():
counter = 0
works = 0
fails = 0
known_environments = procThorInterface.get_all_environments_stored_below_dictionary(procThorInterface.source_folder)
world = BulletWorld(WorldMode.GUI)
apartment = None
milk_pos = Pose([1, -1.78, 0.55], [1, 0, 0, 0])
milk = Object("milk", ObjectType.MILK, "milk.stl", pose=Pose([1, -1.78, 0.55], [1, 0, 0, 0]),
color=Color(1, 0, 0, 1))
for robot in robots:
robot_obj = Object("pr2", ObjectType.ROBOT, robot, pose=Pose([1, 2, 0]))
for environment in known_environments:
counter += 1
print("Trying plan: {} with robot: {} in: {}".format("param_plan", robot, environment["name"]))
try:
apartment = Object(environment["name"], ObjectType.ENVIRONMENT, environment["storage_place"])
generic_plan(world)
works += 1
print("Successfully!\n Overall successful tries: {}".format(works))

except PlanFailure as e:
traceback.print_exc()
fails += 1
print("Plan Fail!\n Overall failed tries: {}".format(fails))

except FileNotFoundError as e2:
traceback.print_exc()
fails += 1
print("Fail!\n Overall failed tries: {}".format(fails))

finally:
try:
process_meta_data = pycram.orm.base.ProcessMetaData()
process_meta_data.description = "CEO Test run number {} robot:{} enviroment:{}".format(counter,
robot,
environment)
process_meta_data.insert(session)
task_tree.root.insert(session)
except Exception as e:
traceback.print_exc()
print("Error while storing the NEEM. This should not happen.")
task_tree.reset_tree()
if World.current_world is not None and apartment is not None:
world.remove_object(apartment)
milk.set_pose(milk_pos)
print("reseting world")
if World.current_world is None:
world.remove_object(robot_obj)

print("Resume:\nOverall successful tries: {}\nOverall failed tries: {}".format(works, fails))
World.current_world.exit()


runWorld()
121 changes: 121 additions & 0 deletions examples/interface_examples/procthor.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,121 @@
# Procthor Interface - Creating a CEO for testing multible robots in mutliple enviromnents
Leusmann marked this conversation as resolved.
Show resolved Hide resolved

This Notebook aims to provide an overview of the ProcThor Interface, by giving an easy-to-understand example on how to
use it. We will be executing a simple pick and place plan with hard coded values and store the NEEMs of these
experiments within a local memory database.

Please be aware that this Notebook (currently) works only when you are connected to the network at the university in
Bremen, otherwise we will not be able to connect to the REST service and the webserver itself.
Leusmann marked this conversation as resolved.
Show resolved Hide resolved

If you want to check if you are able to run the Notebook simply try to ping the following server.

```batch

ping procthor.informatik.uni-bremen.de

```

If you get a positive answer back from the server everything seems to be all right.

Now we need to set up the local memory database.
```python
import sqlalchemy
import pycram.orm.base

engine = sqlalchemy.create_engine("sqlite+pysqlite:///:memory:", echo=False)
session_maker = sqlalchemy.orm.sessionmaker(bind=engine)
session = session_maker()
pycram.orm.base.Base.metadata.create_all(engine)
session.commit()
```
Time for some more setup, while we can technically use multiple robots in the different environments we for now stick
with only PR2. Afterwards, we will start to set up our ProcThorInterface. Our example will work with 5 different
environments.
```python
from pycram.external_interfaces.procthor import ProcThorInterface
from pycram.ros import ros_tools
pycram_path=ros_tools.get_ros_package_path('pycram')

robots =[]
robot_name="pr2.urdf"
robots.append(robot_name)


procthor_rest_endpoint = "http://procthor.informatik.uni-bremen.de:5000/"
source_folder = pycram_path + "resources/tmp/"
procThorInterface=ProcThorInterface(base_url=procthor_rest_endpoint,source_folder=source_folder)
number_of_test_environment=5
```
First let us check how many different environments are already known.

```python
number_of_known_environment = len(
procThorInterface.get_all_environments_stored_below_dictionary(procThorInterface.source_folder))
print("Number of known Environments:{}".format(number_of_known_environment))
print("Number of needed Testenvironments:{}".format(number_of_test_environment))
```

Now depending on if we have to get some more, we will download some additional:
```python
if number_of_known_environment < number_of_test_environment:
print("Downloading missing environments...")
procThorInterface.download_num_random_environment(number_of_test_environment-number_of_known_environment)
```

Now that the preparations are met we can start to run our plan for each robot (even though it is only one) and each
environment we know:

```python
v = 0
works = 0
fails = 0
known_environments = procThorInterface.get_all_environments_stored_below_dictionary(procThorInterface.source_folder)
world = BulletWorld(WorldMode.GUI)
apartment = None
milk_pos = Pose([1, -1.78, 0.55], [1, 0, 0, 0])
milk = Object("milk", ObjectType.MILK, "milk.stl", pose=Pose([1, -1.78, 0.55], [1, 0, 0, 0]),
color=Color(1, 0, 0, 1))
for robot in robots:
robot_obj = Object("pr2", ObjectType.ROBOT, robot, pose=Pose([1, 2, 0]))
for environment in known_environments:
v += 1
print("Trying plan: {} with robot: {} in: {}".format("param_plan", robot, environment["name"]))
try:
apartment = Object(environment["name"], ObjectType.ENVIRONMENT, environment["storage_place"])
generic_plan(world)
works += 1
print("Successfully!\n Overall successful tries: {}".format(works))

except PlanFailure as e:
traceback.print_exc()
fails += 1
print("Plan Fail!\n Overall failed tries: {}".format(fails))

except FileNotFoundError as e2:
traceback.print_exc()
fails += 1
print("Fail!\n Overall failed tries: {}".format(fails))

finally:
try:
process_meta_data = pycram.orm.base.ProcessMetaData()
process_meta_data.description = "CEO Test run number {} robot:{} enviroment:{}".format(v, robot,
environment)
process_meta_data.insert(session)
task_tree.root.insert(session)
except Exception as e:
traceback.print_exc()
print("Error while storing the NEEM. This should not happen.")
task_tree.reset_tree()
if World.current_world is not None and apartment is not None:
world.remove_object(apartment)
milk.set_pose(milk_pos)
print("reseting world")
if World.current_world is None:
world.remove_object(robot_obj)

print("Resume:\nOverall successful tries: {}\nOverall failed tries: {}".format(works, fails))
World.current_world.exit()
```


5 changes: 5 additions & 0 deletions src/pycram/designators/action_designator.py
Original file line number Diff line number Diff line change
Expand Up @@ -924,8 +924,13 @@ class DetectActionPerformable(ActionAbstract):
"""
orm_class: Type[ActionAbstract] = field(init=False, default=ORMDetectAction)

object_at_execution: Optional[ObjectDesignatorDescription.Object] = field(init=False)

@with_tree
def perform(self) -> None:
# Store the object's data copy at execution
self.object_at_execution = self.object_designator.frozen_copy()

return DetectingMotion(object_type=self.object_designator.obj_type).perform()


Expand Down
3 changes: 2 additions & 1 deletion src/pycram/external_interfaces/giskard.py
Original file line number Diff line number Diff line change
Expand Up @@ -173,7 +173,8 @@ def spawn_object(object: Object) -> None:
filename = geometry.file_name
spawn_mesh(object.name, filename, object.get_pose())
else:
spawn_urdf(object.name, object.path, object.get_pose())
ww = spawn_urdf(object.name, object.path, object.get_pose())
print("GiskardSpawnURDF Return value: {} ObjectName:{}".format(ww,object.name))
Leusmann marked this conversation as resolved.
Show resolved Hide resolved


@init_giskard_interface
Expand Down
145 changes: 145 additions & 0 deletions src/pycram/external_interfaces/procthor.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,145 @@
import fnmatch
import os

import requests
from bs4 import BeautifulSoup
from pycram.ros import ros_tools
Leusmann marked this conversation as resolved.
Show resolved Hide resolved
from typing_extensions import Dict, Set

filename = ros_tools.get_ros_package_path('pycram')


class ProcThorInterface:
def __init__(self, base_url="http://procthor.informatik.uni-bremen.de:5000",
source_folder=os.path.join(ros_tools.get_ros_package_path('pycram'), "resources/tmp/")):
self.base_url = base_url
self.source_folder = source_folder
self.stored_environments = []
self.stored_environments.extend(self.get_all_environments_stored_below_dictionary(self.source_folder))

def _download_file(self, base_url: str, full_url: str, folder: str) -> str:
"""
Downloads the file given in full_url and stores it into folder. If necessary it will create the same folder
structure. For this purpose the base_url is necessary to decide what is folder structure and what is just url.


:param base_url: Base url as string from
:param full_url: Full url of the file to be downloaded
:param folder: Folder where the file should be stored
:return: The local file name of the downloaded file
"""
print("base_url:{} full_url:{} folder:{}".format(base_url, full_url, folder))
tree_structure = full_url.replace(base_url, '')
if tree_structure.startswith("/"):
tree_structure = tree_structure[1:]
full_storage_path = os.path.join(folder, tree_structure)
if not os.path.exists(full_storage_path[:full_storage_path.rfind('/') + 1]):
print(full_storage_path[:full_storage_path.rfind('/') + 1])
os.makedirs(full_storage_path[:full_storage_path.rfind('/') + 1], exist_ok=True)
local_filename = os.path.join(folder, full_url.split('/')[-1])
# Send HTTP GET request to fetch the file
with requests.get(full_url, stream=True) as r:
r.raise_for_status()
# Write the content to a file
with open(full_storage_path, 'wb') as f:
for chunk in r.iter_content(chunk_size=8192):
f.write(chunk)
return local_filename

# Function to get the list of files in a directory on the web server
def _get_files_list(self, base_url: str) -> Set[str]:
"""
Function to get the list of files in a directory on the web server.

:param base_url: Base url as string from
:return: Set of all files found below the given url
"""
print("get_files_list({})".format(base_url))
# Send GET request to the URL
response = requests.get(base_url)
# Parse the HTML response
soup = BeautifulSoup(response.text, 'html.parser')
# Find all links on the page
links = soup.find_all('a')
file_list = set()
files_urls = set()
for link in links:
if link.get('href').endswith('/') and not link.get('href').startswith('/'):
files_urls = files_urls.union(self._get_files_list(os.path.join(base_url, link['href'])))

elif link.get('href').endswith(('.usda', '.urdf', '.stl', '.usd', '.hdr')):
files_urls.add(os.path.join(base_url, link['href']))
else:
print("Ignored File: {}".format(link.get('href')))
return files_urls

# Main function to download all files from a directory
def download_all_files_from_URL(self, base_url: str, folder: str) -> None:
"""
Main function to download all files from a given path. Files will be stored in folder.

:param base_url: Base url as string from
:param folder: folder where the files should be stored
:return: None
"""
# Ensure the folder exists
os.makedirs(folder, exist_ok=True)
# Get the list of files
files = self._get_files_list(base_url)
# create_folder_structure(base_url,folder,files)
# Download each file
print(files)
for file_url in files:
print(f"Downloading {file_url}...")
self._download_file(base_url, file_url, folder)
print("All files downloaded.")
return None

# Returns a list of all the urdf files in a given folder structure if non is given uses the base_source_folder
# only look for urdf, but can be easly adapted
def get_all_environments_stored_below_dictionary(self, source_folder: str) -> list:
Leusmann marked this conversation as resolved.
Show resolved Hide resolved
"""
Returns a list of dictionaries that contains the name and the storage_place of all urdf files below a given
folder. Only looks for urdf.

:param source_folder: Base path as string from
:return: map from name to storage place
"""
urdf_files = []
urdf_file = {}
for root, dirs, files in os.walk(source_folder):
for filename in fnmatch.filter(files, '*.urdf'):
urdf_file["name"] = filename.rsplit('.', 1)[0]
urdf_file["storage_place"] = os.path.join(root, filename)
urdf_files.append(urdf_file.copy())
return urdf_files

def download_one_random_environment(self) -> Dict[str, str]:
Leusmann marked this conversation as resolved.
Show resolved Hide resolved
"""
Downloads one random environment currently not present.

:param: None
:return: Dictionary of name and storage place
"""
endpoint = "GetRandomEnvironment"
full_url = os.path.join(self.base_url, endpoint)
response = requests.get(full_url)
self.download_all_files_from_URL(response.json()["storage_place"], self.source_folder)
self.stored_environments.append(response.json())
return response.json()

def download_num_random_environment(self, num_of_environments: int) -> list:
"""
Downloads given amount of random environment currently not present.

:param num_of_environments: Amount of environments that get downloaded
:return: List of Dictionaries of name and storage place
"""
endpoint = "GetNumEnvironment"
full_url = os.path.join(self.base_url, endpoint, str(num_of_environments))
response = requests.get(full_url)
for env in response.json():
print(env)
self.download_all_files_from_URL(env["storage_place"], self.source_folder)
self.stored_environments.append(env)
return response.json()
Loading
Loading