Skip to content

Latest commit

 

History

History
211 lines (156 loc) · 9.94 KB

lisdf.md

File metadata and controls

211 lines (156 loc) · 9.94 KB
Read/Write to Disk

Import .sdf and .lisdf files

Please read the documentation for custom models, custom robots, and custom composite objects.

It is possible to import data from an .sdf or .lisdf scene description file into TDW using the LisdfReader

As with all other import processes in TDW, objects referenced by .sdf and .lisdf file must be converted into asset bundles. We can do this by calling LisdfReader.read():

from pathlib import Path
from tdw.add_ons.lisdf_reader import LisdfReader
from tdw.backend.paths import EXAMPLE_CONTROLLER_OUTPUT_PATH

source_path = Path.home().joinpath("kitchen-worlds/assets/scenes/kitchen_basics.lisdf").resolve()
output_directory = EXAMPLE_CONTROLLER_OUTPUT_PATH.joinpath("")
lisdf_reader = LisdfReader()
lisdf_reader.read(lisdf_path=source_path, output_directory=output_directory)

The above example will take a long time to complete. However, once the asset bundles have been generated, subsequent read() calls will be nearly instantaneous.

The above example uses a .lisdf file from the kitchen-worlds repo; there are issues with cloning this repo in Windows. The best way to clone the repo is with WSL: call wsl git instead of git.

The Asset Bundle Creator Unity project

To create asset bundles, TDW uses Asset Bundle Creator, a Unity Editor project. It is possible to run the Unity project without any Python wrapper classes but there is usually no reason to do so. In the Python class, LisdfReader.read() launchesLisdfReader.Read() in the Unity project.

Asset Bundle Creator will be downloaded automatically the first time you use the Python wrapper class (see below).

Requirements

  • Windows 10, OS X, or Linux
  • (Windows only) Visual C++ 2012 Redistributable
  • The tdw module
  • Python 3.6+
  • git
  • Unity Hub
  • Unity Editor 2020.3.24f1
    • Build options must enabled for Windows, OS X, and Linux (these can be set when installing Unity).
    • Ideally, Unity Editor should be installed via Unity Hub; otherwise, you'll need to set the unity_editor_path parameter (see below).
    • To install on a Linux server, read this.

Unity Editor parameters

If you installed Unity Editor via Unity Hub, LisdfReader should be able to automatically find the Unity Editor executable.

If the Unity Editor executable is in an unexpected location, you will need to explicitly set the optional unity_editor_path parameter:

from pathlib import Path
from tdw.add_ons.lisdf_reader import LisdfReader
from tdw.backend.paths import EXAMPLE_CONTROLLER_OUTPUT_PATH

source_path = Path.home().joinpath("kitchen-worlds/assets/scenes/kitchen_basics.lisdf").resolve()
output_directory = EXAMPLE_CONTROLLER_OUTPUT_PATH.joinpath("")
lisdf_reader = LisdfReader()
lisdf_reader.read(lisdf_path=source_path,
                  output_directory=output_directory,
                  unity_editor_path="D:/Unity/2020.3.24f1/Editor/Unity.exe")

When you call read(), it automatically compares the version of your local Unity project to the one stored on GitHub. This requires an Internet connection and might not be desirable in all cases, especially on servers. To prevent the version check, set check_version=False:

from pathlib import Path
from tdw.add_ons.lisdf_reader import LisdfReader
from tdw.backend.paths import EXAMPLE_CONTROLLER_OUTPUT_PATH

source_path = Path.home().joinpath("kitchen-worlds/assets/scenes/kitchen_basics.lisdf").resolve()
output_directory = EXAMPLE_CONTROLLER_OUTPUT_PATH.joinpath("")
lisdf_reader = LisdfReader()
lisdf_reader.read(lisdf_path=source_path,
                  output_directory=output_directory,
                  check_version=False)

On Linux servers, set display to the correct X display:

from pathlib import Path
from tdw.add_ons.lisdf_reader import LisdfReader
from tdw.backend.paths import EXAMPLE_CONTROLLER_OUTPUT_PATH

source_path = Path.home().joinpath("kitchen-worlds/assets/scenes/kitchen_basics.lisdf").resolve()
output_directory = EXAMPLE_CONTROLLER_OUTPUT_PATH.joinpath("")
lisdf_reader = LisdfReader()
lisdf_reader.read(lisdf_path=source_path,
                  output_directory=output_directory,
                  display=":0")

Objects, composite objects, and robots

Within the Asset Bundle Creator Unity project, the parser reads a .lisdf file and builds asset bundles as objects, composite objects, and robots depending on the description:

  • If the model is described within the root file, it is assumed to be a static model, and will be generated using ModelCreator.
  • If the model is described in a referenced .urdf file, it is assumed to be a composite object and will be generated using CompositeObjectCreator.
  • Composite objects and robots are not the same in TDW. If a referenced .urdf file is describes a robot, you need to set this manually in your Python class, in which case asset bundles will be generated using RobotCreator.

To manually specify which .urdf files are robots, as opposed to articulated objects, set the robot_metadata parameter in read(), which accepts a list of LisdfRobotMetadata:

from pathlib import Path
from tdw.add_ons.lisdf_reader import LisdfReader
from tdw.lisdf_data.lisdf_robot_metadata import LisdfRobotMetadata
from tdw.backend.paths import EXAMPLE_CONTROLLER_OUTPUT_PATH

# Define a robot.
pr2 = LisdfRobotMetadata(name="pr2",
                         link_name_excludes_regex=["_gazebo_", "_cam_", "world_link", "imu_link",
                                                   "sensor_mount_link", "head_plate_frame",
                                                   "_gripper_motor_accelerometer_link"],
                         link_exclude_types=["camera", "laser"])

source_path = Path.home().joinpath("kitchen-worlds/assets/scenes/kitchen_basics.lisdf").resolve()
output_directory = EXAMPLE_CONTROLLER_OUTPUT_PATH.joinpath("")
lisdf_reader = LisdfReader()

# Read the .lisdf file. PR2 will be a robot.
lisdf_reader.read(lisdf_path=source_path, output_directory=output_directory, robot_metadata=[pr2])

link_name_excludes_regex and link_exclude_types are parameters used to "fix" an overly complicated .urdf file; read this for more information.

Overwrite and cleanup

If there are problems with the asset bundles, you may wish to regenerate them. In that case, set overwrite=true in read():

from pathlib import Path
from tdw.add_ons.lisdf_reader import LisdfReader
from tdw.backend.paths import EXAMPLE_CONTROLLER_OUTPUT_PATH

source_path = Path.home().joinpath("kitchen-worlds/assets/scenes/kitchen_basics.lisdf").resolve()
output_directory = EXAMPLE_CONTROLLER_OUTPUT_PATH.joinpath("")
lisdf_reader = LisdfReader()
lisdf_reader.read(lisdf_path=source_path, output_directory=output_directory, overwrite=True)

You might want to adjust or test the model and robot prefabs. To do this, set cleanup=False in read():

from pathlib import Path
from tdw.add_ons.lisdf_reader import LisdfReader
from tdw.backend.paths import EXAMPLE_CONTROLLER_OUTPUT_PATH

source_path = Path.home().joinpath("kitchen-worlds/assets/scenes/kitchen_basics.lisdf").resolve()
output_directory = EXAMPLE_CONTROLLER_OUTPUT_PATH.joinpath("")
lisdf_reader = LisdfReader()
lisdf_reader.read(lisdf_path=source_path, output_directory=output_directory, cleanup=False)

Having set cleanup=False, you can manually create asset bundles from the prefabs using ModelCreator, RobotCreator, and CompositeObjectCreator.

Write and send commands

After parsing a .sdf or .lisdf file and generating asset bundles, LisdfReader will also generate a commands.json file that contains a list of commands that can create the scene in TDW:

from pathlib import Path
import json
from tdw.add_ons.lisdf_reader import LisdfReader
from tdw.backend.paths import EXAMPLE_CONTROLLER_OUTPUT_PATH

source_path = Path.home().joinpath("kitchen-worlds/assets/scenes/kitchen_basics.lisdf").resolve()
output_directory = EXAMPLE_CONTROLLER_OUTPUT_PATH.joinpath("")
lisdf_reader = LisdfReader()
lisdf_reader.read(lisdf_path=source_path,
                  output_directory=output_directory)
commands_path = output_directory.joinpath("commands.json")
commands = json.loads(commands_path.read_text())

If you append LisdfReader to a controller, you don't need to read the commands manually; as an add-on, LisdfReader will automatically send the commands:

from pathlib import Path
from tdw.controller import Controller
from tdw.add_ons.lisdf_reader import LisdfReader
from tdw.backend.paths import EXAMPLE_CONTROLLER_OUTPUT_PATH

source_path = Path.home().joinpath("kitchen-worlds/assets/scenes/kitchen_basics.lisdf").resolve()
output_directory = EXAMPLE_CONTROLLER_OUTPUT_PATH.joinpath("")

# Create a controller.
c = Controller()
# Create an LISDF reader.
lisdf_reader = LisdfReader()
# Append the LISDF reader as an add-on.
c.add_ons.append(lisdf_reader)
# Read the .lisdf file. Create asset bundles if needed.
lisdf_reader.read(lisdf_path=source_path,
                  output_directory=output_directory)
# This will send the commands generated by `lisdf_reader`.
c.communicate([])

This is the last document in the "Read/Write to Disk" tutorial.

Return to the README


Python API: