diff --git a/.github/workflows/macos.yml b/.github/workflows/macos.yml index 0842e3e5d..ff4cb40c2 100644 --- a/.github/workflows/macos.yml +++ b/.github/workflows/macos.yml @@ -6,7 +6,7 @@ jobs: build: env: - PACKAGE: sdformat13 + PACKAGE: sdformat14 runs-on: macos-latest steps: - uses: actions/checkout@v3 diff --git a/.gitignore b/.gitignore index 08e5eba3e..fe7c191e7 100644 --- a/.gitignore +++ b/.gitignore @@ -3,3 +3,4 @@ build build_* *.*.sw? .vscode +__pycache__ diff --git a/CMakeLists.txt b/CMakeLists.txt index 48c227d90..af09e1629 100644 --- a/CMakeLists.txt +++ b/CMakeLists.txt @@ -5,11 +5,11 @@ if(COMMAND CMAKE_POLICY) CMAKE_POLICY(SET CMP0004 NEW) endif(COMMAND CMAKE_POLICY) -project (sdformat13 VERSION 13.5.0) +project (sdformat14 VERSION 14.0.0) # The protocol version has nothing to do with the package version. # It represents the current version of SDFormat implemented by the software -set (SDF_PROTOCOL_VERSION 1.10) +set (SDF_PROTOCOL_VERSION 1.11) OPTION(SDFORMAT_DISABLE_CONSOLE_LOGFILE "Disable the sdformat console logfile" OFF) @@ -162,7 +162,7 @@ if (BUILD_SDF) add_subdirectory(conf) add_subdirectory(doc) if (pybind11_FOUND AND NOT SKIP_PYBIND11) - add_subdirectory(python) + add_subdirectory(python) endif() endif(BUILD_SDF) diff --git a/Changelog.md b/Changelog.md index caa6addf6..b1332f0e8 100644 --- a/Changelog.md +++ b/Changelog.md @@ -1,3 +1,7 @@ +## libsdformat 14.X + +### libsdformat 14.0.0 (202X-XX-XX) + ## libsdformat 13.X ### libsdformat 13.5.0 (2023-05-18) diff --git a/README.md b/README.md index 5802cd718..b5413885e 100644 --- a/README.md +++ b/README.md @@ -114,7 +114,7 @@ conda install libsdformat=12.5.0 --channel conda-forge ## Source Installation -**Note:** the `main` branch is under development for `libsdformat13` and is +**Note:** the `main` branch is under development for `libsdformat14` and is currently unstable. A release branch (`sdf12`, `sdf11`, `sdf10`, `sdf9`, etc.) is recommended for most users. diff --git a/examples/CMakeLists.txt b/examples/CMakeLists.txt index 39caedb85..7541a5de6 100644 --- a/examples/CMakeLists.txt +++ b/examples/CMakeLists.txt @@ -1,6 +1,6 @@ cmake_minimum_required(VERSION 3.10.2 FATAL_ERROR) -find_package(sdformat13 REQUIRED) +find_package(sdformat14 REQUIRED) set (CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -std=c++17") diff --git a/include/sdf/ParticleEmitter.hh b/include/sdf/ParticleEmitter.hh index 3b824590c..bf592de24 100644 --- a/include/sdf/ParticleEmitter.hh +++ b/include/sdf/ParticleEmitter.hh @@ -37,7 +37,7 @@ namespace sdf /// \enum ParticleEmitterType /// \brief The set of particle emitter types. - // Developer note: Make sure to update emitterTypeStrs in the source + // Developer note: Make sure to update kEmitterTypeStrs in the source // file when changing this enum. enum class ParticleEmitterType { diff --git a/include/sdf/Plugin.hh b/include/sdf/Plugin.hh index b7f790e40..def211593 100644 --- a/include/sdf/Plugin.hh +++ b/include/sdf/Plugin.hh @@ -238,7 +238,7 @@ namespace sdf const std::string &_name, const std::string &_xmlContent); /// \brief Private data pointer. - public: std::unique_ptr dataPtr; + private: std::unique_ptr dataPtr; }; /// \brief A vector of Plugin. diff --git a/include/sdf/Sensor.hh b/include/sdf/Sensor.hh index 0b998c1f1..7b62572ec 100644 --- a/include/sdf/Sensor.hh +++ b/include/sdf/Sensor.hh @@ -49,7 +49,7 @@ namespace sdf /// \enum SensorType /// \brief The set of sensor types. - // Developer note: Make sure to update sensorTypeStrs in the source file + // Developer note: Make sure to update kSensorTypeStrs in the source file // when changing this enum. enum class SensorType { diff --git a/include/sdf/Types.hh b/include/sdf/Types.hh index f27d50bf7..f2f19248c 100644 --- a/include/sdf/Types.hh +++ b/include/sdf/Types.hh @@ -25,6 +25,7 @@ #include #include +#include #include #include "sdf/system_util.hh" #include "sdf/Error.hh" @@ -35,7 +36,21 @@ namespace sdf inline namespace SDF_VERSION_NAMESPACE { // - const std::string kSdfScopeDelimiter = "::"; + namespace internal + { + /// \brief Initializes the scope delimiter as a function-local static + /// variable so it can be used to initialize kSdfScopeDelimiter. + /// \note This should not be used directly in user code. It will likely be + /// removed in libsdformat 15 with kSdfScopeDelimiter. + SDFORMAT_VISIBLE const std::string &SdfScopeDelimiter(); + } // namespace internal + + constexpr std::string_view kScopeDelimiter{"::"}; + + // Deprecated because it violates the Google Style Guide as it is not + // trivially destructible. Please use sdf::kScopeDelimiter instead. + GZ_DEPRECATED(14) + inline const std::string &kSdfScopeDelimiter = internal::SdfScopeDelimiter(); /// \brief The source path replacement if it was parsed from a string, /// instead of a file. diff --git a/python/test/gz_test_deps/sdformat.py b/python/test/gz_test_deps/sdformat.py index 241e91374..51cec2889 100644 --- a/python/test/gz_test_deps/sdformat.py +++ b/python/test/gz_test_deps/sdformat.py @@ -1 +1 @@ -from sdformat13 import * +from sdformat14 import * diff --git a/python/test/pyModel_TEST.py b/python/test/pyModel_TEST.py index 126c1b579..111f2d5b1 100644 --- a/python/test/pyModel_TEST.py +++ b/python/test/pyModel_TEST.py @@ -255,7 +255,7 @@ def test_add_modify_frame(self): def test_uri(self): model = Model() - uri = "https:#fuel.ignitionrobotics.org/1.0/openrobotics/models/my-model" + uri = "https:#fuel.gazebosim.org/1.0/openrobotics/models/my-model" model.set_uri(uri) self.assertEqual(uri, model.uri()) diff --git a/sdf/1.10/CMakeLists.txt b/sdf/1.10/CMakeLists.txt index 531202e38..55a906f71 100644 --- a/sdf/1.10/CMakeLists.txt +++ b/sdf/1.10/CMakeLists.txt @@ -73,8 +73,8 @@ foreach(FIL ${sdfs}) add_custom_command( OUTPUT "${CMAKE_CURRENT_BINARY_DIR}/${FIL_WE}.xsd" - COMMAND ${RUBY} ${CMAKE_SOURCE_DIR}/tools/xmlschema.rb - ARGS -s ${CMAKE_CURRENT_SOURCE_DIR} -i ${ABS_FIL} -o ${CMAKE_CURRENT_BINARY_DIR} + COMMAND ${Python3_EXECUTABLE} ${CMAKE_SOURCE_DIR}/tools/xmlschema.py + ARGS --sdf-dir ${CMAKE_CURRENT_SOURCE_DIR} --input-file ${ABS_FIL} --output-dir ${CMAKE_CURRENT_BINARY_DIR} DEPENDS ${ABS_FIL} COMMENT "Running xml schema compiler on ${FIL}" VERBATIM) diff --git a/sdf/1.11/1_10.convert b/sdf/1.11/1_10.convert new file mode 100644 index 000000000..380267d62 --- /dev/null +++ b/sdf/1.11/1_10.convert @@ -0,0 +1,2 @@ + + diff --git a/sdf/1.11/CMakeLists.txt b/sdf/1.11/CMakeLists.txt new file mode 100644 index 000000000..cb67a9e06 --- /dev/null +++ b/sdf/1.11/CMakeLists.txt @@ -0,0 +1,88 @@ +set (sdfs + actor.sdf + air_pressure.sdf + air_speed.sdf + altimeter.sdf + atmosphere.sdf + audio_source.sdf + audio_sink.sdf + battery.sdf + box_shape.sdf + camera.sdf + capsule_shape.sdf + collision.sdf + contact.sdf + cylinder_shape.sdf + ellipsoid_shape.sdf + frame.sdf + forcetorque.sdf + geometry.sdf + gps.sdf + gripper.sdf + gui.sdf + heightmap_shape.sdf + image_shape.sdf + imu.sdf + inertial.sdf + joint.sdf + lidar.sdf + light.sdf + light_state.sdf + link.sdf + link_state.sdf + logical_camera.sdf + magnetometer.sdf + material.sdf + mesh_shape.sdf + model.sdf + model_state.sdf + navsat.sdf + noise.sdf + particle_emitter.sdf + physics.sdf + plane_shape.sdf + plugin.sdf + polyline_shape.sdf + population.sdf + pose.sdf + projector.sdf + ray.sdf + rfidtag.sdf + rfid.sdf + road.sdf + root.sdf + scene.sdf + sensor.sdf + spherical_coordinates.sdf + sphere_shape.sdf + sonar.sdf + state.sdf + surface.sdf + transceiver.sdf + visual.sdf + world.sdf +) + +set (SDF_SCHEMA) + +foreach(FIL ${sdfs}) + get_filename_component(ABS_FIL ${FIL} ABSOLUTE) + get_filename_component(FIL_WE ${FIL} NAME_WE) + + list(APPEND SDF_SCHEMA "${CMAKE_CURRENT_BINARY_DIR}/${FIL_WE}.xsd") + + add_custom_command( + OUTPUT "${CMAKE_CURRENT_BINARY_DIR}/${FIL_WE}.xsd" + COMMAND ${Python3_EXECUTABLE} ${CMAKE_SOURCE_DIR}/tools/xmlschema.py + ARGS --sdf-dir ${CMAKE_CURRENT_SOURCE_DIR} --input-file ${ABS_FIL} --output-dir ${CMAKE_CURRENT_BINARY_DIR} + DEPENDS ${ABS_FIL} + COMMENT "Running xml schema compiler on ${FIL}" + VERBATIM) +endforeach() + +add_custom_target(schema1_11 ALL DEPENDS ${SDF_SCHEMA}) + +set_source_files_properties(${SDF_SCHEMA} PROPERTIES GENERATED TRUE) + +install(FILES 1_10.convert ${sdfs} DESTINATION ${CMAKE_INSTALL_FULL_DATAROOTDIR}/sdformat${PROJECT_VERSION_MAJOR}/1.11) +install(FILES ${SDF_SCHEMA} DESTINATION ${CMAKE_INSTALL_FULL_DATAROOTDIR}/sdformat${PROJECT_VERSION_MAJOR}/1.11) diff --git a/sdf/1.11/actor.sdf b/sdf/1.11/actor.sdf new file mode 100644 index 000000000..4ff1c4cc6 --- /dev/null +++ b/sdf/1.11/actor.sdf @@ -0,0 +1,86 @@ + + + A special kind of model which can have a scripted motion. This includes both global waypoint type animations and skeleton animations. + + + A unique name for the actor. + + + + + + Skin file which defines a visual and the underlying skeleton which moves it. + + + Path to skin file, accepted formats: COLLADA, BVH. + + + + Scale the skin's size. + + + + + Animation file defines an animation for the skeleton in the skin. The skeleton must be compatible with the skin skeleton. + + + Unique name for animation. + + + + Path to animation file. Accepted formats: COLLADA, BVH. + + + Scale for the animation skeleton. + + + Set to true so the animation is interpolated on X. + + + + + Adds scripted trajectories to the actor. + + + Set this to true for the script to be repeated in a loop. For a fluid continuous motion, make sure the last waypoint matches the first one. + + + + This is the time to wait before starting the script. If running in a loop, this time will be waited before starting each cycle. + + + + Set to true if the animation should start as soon as the simulation starts playing. It is useful to set this to false if the animation should only start playing only when triggered by a plugin, for example. + + + + The trajectory contains a series of keyframes to be followed. + + Unique id for a trajectory. + + + + If it matches the type of an animation, they will be played at the same time. + + + + The tension of the trajectory spline. The default value of zero equates to a Catmull-Rom spline, which may also cause the animation to overshoot keyframes. A value of one will cause the animation to stick to the keyframes. + + + + Each point in the trajectory. + + The time in seconds, counted from the beginning of the script, when the pose should be reached. + + + The pose which should be reached at the given time. + + + + + + + + + + diff --git a/sdf/1.11/air_pressure.sdf b/sdf/1.11/air_pressure.sdf new file mode 100644 index 000000000..07fe5b564 --- /dev/null +++ b/sdf/1.11/air_pressure.sdf @@ -0,0 +1,15 @@ + + These elements are specific to an air pressure sensor. + + + The initial altitude in meters. This value can be used by a sensor implementation to augment the altitude of the sensor. For example, if you are using simulation instead of creating a 1000 m mountain model on which to place your sensor, you could instead set this value to 1000 and place your model on a ground plane with a Z height of zero. + + + + + Noise parameters for the pressure data. + + + + + diff --git a/sdf/1.11/air_speed.sdf b/sdf/1.11/air_speed.sdf new file mode 100644 index 000000000..1313f705a --- /dev/null +++ b/sdf/1.11/air_speed.sdf @@ -0,0 +1,11 @@ + + These elements are specific to an air speed sensor. This sensor determines speed based on the differential between static and dynamic pressure. + + + + Noise parameters for the pressure data. + + + + + diff --git a/sdf/1.11/altimeter.sdf b/sdf/1.11/altimeter.sdf new file mode 100644 index 000000000..66ddee579 --- /dev/null +++ b/sdf/1.11/altimeter.sdf @@ -0,0 +1,18 @@ + + These elements are specific to an altimeter sensor. + + + + Noise parameters for vertical position + + + + + + + Noise parameters for vertical velocity + + + + + diff --git a/sdf/1.11/atmosphere.sdf b/sdf/1.11/atmosphere.sdf new file mode 100644 index 000000000..641faccb9 --- /dev/null +++ b/sdf/1.11/atmosphere.sdf @@ -0,0 +1,21 @@ + + + The atmosphere tag specifies the type and properties of the atmosphere model. + + + The type of the atmosphere engine. Current options are adiabatic. Defaults to adiabatic if left unspecified. + + + + Temperature at sea level in kelvins. + + + + Pressure at sea level in pascals. + + + + Temperature gradient with respect to increasing altitude at sea level in units of K/m. + + + diff --git a/sdf/1.11/audio_sink.sdf b/sdf/1.11/audio_sink.sdf new file mode 100644 index 000000000..d3bd0711d --- /dev/null +++ b/sdf/1.11/audio_sink.sdf @@ -0,0 +1,4 @@ + + + An audio sink. + diff --git a/sdf/1.11/audio_source.sdf b/sdf/1.11/audio_source.sdf new file mode 100644 index 000000000..b6ee6069a --- /dev/null +++ b/sdf/1.11/audio_source.sdf @@ -0,0 +1,30 @@ + + + An audio source. + + + URI of the audio media. + + + + Pitch for the audio media, in Hz + + + + Gain for the audio media, in dB. + + + + List of collision objects that will trigger audio playback. + + Name of child collision element that will trigger audio playback. + + + + + True to make the audio source loop playback. + + + + + diff --git a/sdf/1.11/battery.sdf b/sdf/1.11/battery.sdf new file mode 100644 index 000000000..dc6f8deb6 --- /dev/null +++ b/sdf/1.11/battery.sdf @@ -0,0 +1,12 @@ + + + Description of a battery. + + + Unique name for the battery. + + + + Initial voltage in volts. + + diff --git a/sdf/1.11/box_shape.sdf b/sdf/1.11/box_shape.sdf new file mode 100644 index 000000000..826ab37a2 --- /dev/null +++ b/sdf/1.11/box_shape.sdf @@ -0,0 +1,6 @@ + + Box shape + + The three side lengths of the box. The origin of the box is in its geometric center (inside the center of the box). + + diff --git a/sdf/1.11/camera.sdf b/sdf/1.11/camera.sdf new file mode 100644 index 000000000..8fd218caf --- /dev/null +++ b/sdf/1.11/camera.sdf @@ -0,0 +1,230 @@ + + These elements are specific to camera sensors. + + + An optional name for the camera. + + + + If the camera will be triggered by a topic + + + + Name of the camera info + + + + Name of the topic that will trigger the camera if enabled + + + + Horizontal field of view + + + + The image size in pixels and format. + + Width in pixels + + + Height in pixels + + + (L8|L16|R_FLOAT16|R_FLOAT32|R8G8B8|B8G8R8|BAYER_RGGB8|BAYER_BGGR8|BAYER_GBRG8|BAYER_GRBG8) + + + Value used for anti-aliasing + + + + + The near and far clip planes. Objects closer or farther than these planes are not rendered. + + + Near clipping plane + + + + Far clipping plane + + + + + Enable or disable saving of camera frames. + + True = saving enabled + + + The path name which will hold the frame data. If path name is relative, then directory is relative to current working directory. + + + + + Depth camera parameters + + Type of output + + + The near and far clip planes. Objects closer or farther than these planes are not detected by the depth camera. + + + Near clipping plane for depth camera + + + + Far clipping plane for depth camera + + + + + + + The segmentation type of the segmentation camera. Valid options are: + - semantic: Semantic segmentation, which provides 2 images: + 1. A grayscale image, with the pixel values representing the label of an object + 2. A colored image, with the pixel values being a unique color for each label + + - panoptic | instance: Panoptic segmentation, which provides an image where each pixel + has 1 channel for label value of the object and 2 channels for the + number of the instances of that label, and a colored image which its + pixels have a unique color for each instance. + + + + + + The boundingbox type of the boundingbox camera. Valid options are: + - 2d | visible_2d | visible_box_2d: a visible 2d box mode which provides axis aligned 2d boxes + on the visible parts of the objects + + - full_2d | full_box_2d: a full 2d box mode which provides axis aligned 2d boxes that fills the + object dimentions, even if it has an occluded part + + - 3d: a 3d mode which provides oriented 3d boxes + + + + + The properties of the noise model that should be applied to generated images + + The type of noise. Currently supported types are: "gaussian" (draw additive noise values independently for each pixel from a Gaussian distribution). + + + For type "gaussian," the mean of the Gaussian distribution from which noise values are drawn. + + + For type "gaussian," the standard deviation of the Gaussian distribution from which noise values are drawn. + + + + + Lens distortion to be applied to camera images. See http://en.wikipedia.org/wiki/Distortion_(optics)#Software_correction + + The radial distortion coefficient k1 + + + The radial distortion coefficient k2 + + + The radial distortion coefficient k3 + + + The tangential distortion coefficient p1 + + + The tangential distortion coefficient p2 + + + The distortion center or principal point + + + + + Lens projection description + + + Type of the lens mapping. Supported values are gnomonical, stereographic, equidistant, equisolid_angle, orthographic, custom. For gnomonical (perspective) projection, it is recommended to specify a horizontal_fov of less than or equal to 90° + + + If true the image will be scaled to fit horizontal FOV, otherwise it will be shown according to projection type parameters + + + + Definition of custom mapping function in a form of r=c1*f*fun(theta/c2 + c3). See https://en.wikipedia.org/wiki/Fisheye_lens#Mapping_function + + Linear scaling constant + + + Angle scaling constant + + + Angle offset constant + + + Focal length of the optical system. Note: It's not a focal length of the lens in a common sense! This value is ignored if 'scale_to_fov' is set to true + + + Possible values are 'sin', 'tan' and 'id' + + + + + Everything outside of the specified angle will be hidden, 90° by default + + + + Resolution of the environment cube map used to draw the world + + + + Camera intrinsic parameters for setting a custom perspective projection matrix (cannot be used with WideAngleCamera since this class uses image stitching from 6 different cameras for achieving a wide field of view). The focal lengths can be computed using focal_length_in_pixels = (image_width_in_pixels * 0.5) / tan(field_of_view_in_degrees * 0.5 * PI/180) + + X focal length (in pixels, overrides horizontal_fov) + + + Y focal length (in pixels, overrides horizontal_fov) + + + X principal point (in pixels) + + + Y principal point (in pixels) + + + XY axis skew + + + + + Camera projection matrix P for overriding camera intrinsic matrix K values so that users can configure P independently of K. This is useful when working with stereo cameras where P may be different from K due to the transform between the two cameras. + + X focal length for projection matrix(in pixels, overrides fx) + + + Y focal length for projection matrix(in pixels, overrides fy) + + + X principal point for projection matrix(in pixels, overrides cx) + + + Y principal point for projection matrix(in pixels, overrides cy) + + + X translation for projection matrix (in pixels) + + + Y translation for projection matrix (in pixels) + + + + + + + + + + An optional frame id name to be used in the camera_info message header. + + + + diff --git a/sdf/1.11/capsule_shape.sdf b/sdf/1.11/capsule_shape.sdf new file mode 100644 index 000000000..2831099ba --- /dev/null +++ b/sdf/1.11/capsule_shape.sdf @@ -0,0 +1,9 @@ + + Capsule shape + + Radius of the capsule + + + Length of the cylindrical portion of the capsule along the z axis + + diff --git a/sdf/1.11/collision.sdf b/sdf/1.11/collision.sdf new file mode 100644 index 000000000..e83406df1 --- /dev/null +++ b/sdf/1.11/collision.sdf @@ -0,0 +1,22 @@ + + + The collision properties of a link. Note that this can be different from the visual properties of a link, for example, simpler collision models are often used to reduce computation time. + + + Unique name for the collision element within the scope of the parent link. + + + + intensity value returned by laser sensor. + + + + Maximum number of contacts allowed between two entities. This value overrides the max_contacts element defined in physics. + + + + + + + + diff --git a/sdf/1.11/contact.sdf b/sdf/1.11/contact.sdf new file mode 100644 index 000000000..d1cde50ac --- /dev/null +++ b/sdf/1.11/contact.sdf @@ -0,0 +1,12 @@ + + These elements are specific to the contact sensor. + + + name of the collision element within a link that acts as the contact sensor. + + + + Topic on which contact data is published. + + + diff --git a/sdf/1.11/cylinder_shape.sdf b/sdf/1.11/cylinder_shape.sdf new file mode 100644 index 000000000..771596ba4 --- /dev/null +++ b/sdf/1.11/cylinder_shape.sdf @@ -0,0 +1,9 @@ + + Cylinder shape + + Radius of the cylinder + + + Length of the cylinder along the z axis + + diff --git a/sdf/1.11/ellipsoid_shape.sdf b/sdf/1.11/ellipsoid_shape.sdf new file mode 100644 index 000000000..821aadf12 --- /dev/null +++ b/sdf/1.11/ellipsoid_shape.sdf @@ -0,0 +1,6 @@ + + Ellipsoid shape + + The three radii of the ellipsoid. The origin of the ellipsoid is in its geometric center (inside the center of the ellipsoid). + + diff --git a/sdf/1.11/forcetorque.sdf b/sdf/1.11/forcetorque.sdf new file mode 100644 index 000000000..5a1846951 --- /dev/null +++ b/sdf/1.11/forcetorque.sdf @@ -0,0 +1,54 @@ + + These elements are specific to the force torque sensor. + + + Frame in which to report the wrench values. Currently supported frames are: + "parent" report the wrench expressed in the orientation of the parent link frame, + "child" report the wrench expressed in the orientation of the child link frame, + "sensor" report the wrench expressed in the orientation of the joint sensor frame. + Note that for each option the point with respect to which the + torque component of the wrench is expressed is the joint origin. + + + + + Direction of the wrench measured by the sensor. The supported options are: + "parent_to_child" if the measured wrench is the one applied by the parent link on the child link, + "child_to_parent" if the measured wrench is the one applied by the child link on the parent link. + + + + + These elements are specific to measurement-frame force, + which is expressed in Newtons + + Force along the X axis + + + + Force along the Y axis + + + + Force along the Z axis + + + + + + These elements are specific to measurement-frame torque, + which is expressed in Newton-meters + + Torque about the X axis + + + + Force about the Y axis + + + + Torque about the Z axis + + + + diff --git a/sdf/1.11/frame.sdf b/sdf/1.11/frame.sdf new file mode 100644 index 000000000..74802fd7e --- /dev/null +++ b/sdf/1.11/frame.sdf @@ -0,0 +1,35 @@ + + + A frame of reference in which poses may be expressed. + + + + Name of the frame. It must be unique whithin its scope (model/world), + i.e., it must not match the name of another frame, link, joint, or model + within the same scope. + + + + + + If specified, this frame is attached to the specified frame. The specified + frame must be within the same scope and may be defined implicitly, i.e., + the name of any //frame, //model, //joint, or //link within the same scope + may be used. + + If missing, this frame is attached to the containing scope's frame. Within + a //world scope this is the implicit world frame, and within a //model + scope this is the implicit model frame. + + A frame moves jointly with the frame it is @attached_to. This is different + from //pose/@relative_to. @attached_to defines how the frame is attached + to a //link, //model, or //world frame, while //pose/@relative_to defines + how the frame's pose is represented numerically. As a result, following + the chain of @attached_to attributes must always lead to a //link, + //model, //world, or //joint (implicitly attached_to its child //link). + + + + + + diff --git a/sdf/1.11/geometry.sdf b/sdf/1.11/geometry.sdf new file mode 100644 index 000000000..884902afb --- /dev/null +++ b/sdf/1.11/geometry.sdf @@ -0,0 +1,20 @@ + + + The shape of the visual or collision object. + + + You can use the empty tag to make empty geometries. + + + + + + + + + + + + + + diff --git a/sdf/1.11/gps.sdf b/sdf/1.11/gps.sdf new file mode 100644 index 000000000..95e004925 --- /dev/null +++ b/sdf/1.11/gps.sdf @@ -0,0 +1,40 @@ + + These elements are specific to the GPS sensor. + + + + Parameters related to GPS position measurement. + + + + Noise parameters for horizontal position measurement, in units of meters. + + + + + + Noise parameters for vertical position measurement, in units of meters. + + + + + + + + Parameters related to GPS position measurement. + + + + Noise parameters for horizontal velocity measurement, in units of meters/second. + + + + + + Noise parameters for vertical velocity measurement, in units of meters/second. + + + + + + diff --git a/sdf/1.11/gripper.sdf b/sdf/1.11/gripper.sdf new file mode 100644 index 000000000..12fd0b341 --- /dev/null +++ b/sdf/1.11/gripper.sdf @@ -0,0 +1,30 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/sdf/1.11/gui.sdf b/sdf/1.11/gui.sdf new file mode 100644 index 000000000..3a87763a7 --- /dev/null +++ b/sdf/1.11/gui.sdf @@ -0,0 +1,60 @@ + + + + + + + + + + + + + + + + + + + + + Set the type of projection for the camera. Valid values are "perspective" and "orthographic". + + + + + + + Name of the tracked visual. If no name is provided, the remaining settings will be applied whenever tracking is triggered in the GUI. + + + + Minimum distance between the camera and the tracked visual. This parameter is only used if static is set to false. + + + + Maximum distance between the camera and the tracked visual. This parameter is only used if static is set to false. + + + + If set to true, the position of the camera is fixed relatively to the model or to the world, depending on the value of the use_model_frame element. Otherwise, the position of the camera may vary but the distance between the camera and the model will depend on the value of the min_dist and max_dist elements. In any case, the camera will always follow the model by changing its orientation. + + + + If set to true, the position of the camera is relative to the model reference frame, which means that its position relative to the model will not change. Otherwise, the position of the camera is relative to the world reference frame, which means that its position relative to the world will not change. This parameter is only used if static is set to true. + + + + The position of the camera's reference frame. This parameter is only used if static is set to true. If use_model_frame is set to true, the position is relative to the model reference frame, otherwise it represents world coordinates. + + + + If set to true, the camera will inherit the yaw rotation of the tracked model. This parameter is only used if static and use_model_frame are set to true. + + + + + + + + diff --git a/sdf/1.11/heightmap_shape.sdf b/sdf/1.11/heightmap_shape.sdf new file mode 100644 index 000000000..7a7462298 --- /dev/null +++ b/sdf/1.11/heightmap_shape.sdf @@ -0,0 +1,44 @@ + + A heightmap based on a 2d grayscale image. + + URI to a grayscale image file + + + The size of the heightmap in world units. + When loading an image: "size" is used if present, otherwise defaults to 1x1x1. + When loading a DEM: "size" is used if present, otherwise defaults to true size of DEM. + + + + A position offset. + + + + The heightmap can contain multiple textures. The order of the texture matters. The first texture will appear at the lowest height, and the last texture at the highest height. Use blend to control the height thresholds and fade between textures. + + Size of the applied texture in meters. + + + Diffuse texture image filename + + + Normalmap texture image filename + + + + The blend tag controls how two adjacent textures are mixed. The number of blend elements should equal one less than the number of textures. + + Min height of a blend layer + + + Distance over which the blend occurs + + + + Set if the rendering engine will use terrain paging + + + Samples per heightmap datum. For rasterized heightmaps, this indicates the number of samples to take per pixel. Using a higher value, e.g. 2, will generally improve the quality of the heightmap but lower performance. + + + diff --git a/sdf/1.11/image_shape.sdf b/sdf/1.11/image_shape.sdf new file mode 100644 index 000000000..369de7c5e --- /dev/null +++ b/sdf/1.11/image_shape.sdf @@ -0,0 +1,18 @@ + + Extrude a set of boxes from a grayscale image. + + URI of the grayscale image file + + + Scaling factor applied to the image + + + Grayscale threshold + + + Height of the extruded boxes + + + The amount of error in the model + + diff --git a/sdf/1.11/imu.sdf b/sdf/1.11/imu.sdf new file mode 100644 index 000000000..25690c8f1 --- /dev/null +++ b/sdf/1.11/imu.sdf @@ -0,0 +1,120 @@ + + These elements are specific to the IMU sensor. + + + + + + This string represents special hardcoded use cases that are commonly seen with typical robot IMU's: + - CUSTOM: use Euler angle custom_rpy orientation specification. + The orientation of the IMU's reference frame is defined by adding the custom_rpy rotation + to the parent_frame. + - NED: The IMU XYZ aligns with NED, where NED orientation relative to Gazebo world + is defined by the SphericalCoordinates class. + - ENU: The IMU XYZ aligns with ENU, where ENU orientation relative to Gazebo world + is defined by the SphericalCoordinates class. + - NWU: The IMU XYZ aligns with NWU, where NWU orientation relative to Gazebo world + is defined by the SphericalCoordinates class. + - GRAV_UP: where direction of gravity maps to IMU reference frame Z-axis with Z-axis pointing in + the opposite direction of gravity. IMU reference frame X-axis direction is defined by grav_dir_x. + Note if grav_dir_x is parallel to gravity direction, this configuration fails. + Otherwise, IMU reference frame X-axis is defined by projection of grav_dir_x onto a plane + normal to the gravity vector. IMU reference frame Y-axis is a vector orthogonal to both + X and Z axis following the right hand rule. + - GRAV_DOWN: where direction of gravity maps to IMU reference frame Z-axis with Z-axis pointing in + the direction of gravity. IMU reference frame X-axis direction is defined by grav_dir_x. + Note if grav_dir_x is parallel to gravity direction, this configuration fails. + Otherwise, IMU reference frame X-axis is defined by projection of grav_dir_x onto a plane + normal to the gravity vector. IMU reference frame Y-axis is a vector orthogonal to both + X and Z axis following the right hand rule. + + + + + This field and parent_frame are used when localization is set to CUSTOM. + Orientation (fixed axis roll, pitch yaw) transform from parent_frame to this IMU's reference frame. + Some common examples are: + - IMU reports in its local frame on boot. IMU sensor frame is the reference frame. + Example: parent_frame="", custom_rpy="0 0 0" + - IMU reports in Gazebo world frame. + Example sdf: parent_frame="world", custom_rpy="0 0 0" + - IMU reports in NWU frame. + Uses SphericalCoordinates class to determine world frame in relation to magnetic north and gravity; + i.e. rotation between North-West-Up and world (+X,+Y,+Z) frame is defined by SphericalCoordinates class. + Example sdf given world is NWU: parent_frame="world", custom_rpy="0 0 0" + - IMU reports in NED frame. + Uses SphericalCoordinates class to determine world frame in relation to magnetic north and gravity; + i.e. rotation between North-East-Down and world (+X,+Y,+Z) frame is defined by SphericalCoordinates class. + Example sdf given world is NWU: parent_frame="world", custom_rpy="M_PI 0 0" + - IMU reports in ENU frame. + Uses SphericalCoordinates class to determine world frame in relation to magnetic north and gravity; + i.e. rotation between East-North-Up and world (+X,+Y,+Z) frame is defined by SphericalCoordinates class. + Example sdf given world is NWU: parent_frame="world", custom_rpy="0 0 -0.5*M_PI" + - IMU reports in ROS optical frame as described in http://www.ros.org/reps/rep-0103.html#suffix-frames, which is + (z-forward, x-left to right when facing +z, y-top to bottom when facing +z). + (default gazebo camera is +x:view direction, +y:left, +z:up). + Example sdf: parent_frame="local", custom_rpy="-0.5*M_PI 0 -0.5*M_PI" + + + + Name of parent frame which the custom_rpy transform is defined relative to. + It can be any valid fully scoped Gazebo Link name or the special reserved "world" frame. + If left empty, use the sensor's own local frame. + + + + + + Used when localization is set to GRAV_UP or GRAV_DOWN, a projection of this vector + into a plane that is orthogonal to the gravity vector + defines the direction of the IMU reference frame's X-axis. + grav_dir_x is defined in the coordinate frame as defined by the parent_frame element. + + + + Name of parent frame in which the grav_dir_x vector is defined. + It can be any valid fully scoped Gazebo Link name or the special reserved "world" frame. + If left empty, use the sensor's own local frame. + + + + + + + These elements are specific to body-frame angular velocity, + which is expressed in radians per second + + Angular velocity about the X axis + + + + Angular velocity about the Y axis + + + + Angular velocity about the Z axis + + + + + + These elements are specific to body-frame linear acceleration, + which is expressed in meters per second squared + + Linear acceleration about the X axis + + + + Linear acceleration about the Y axis + + + + Linear acceleration about the Z axis + + + + + + Some IMU sensors rely on external filters to produce orientation estimates. True to generate and output orientation data, false to disable orientation data generation. + + diff --git a/sdf/1.11/inertial.sdf b/sdf/1.11/inertial.sdf new file mode 100644 index 000000000..f4d31ef59 --- /dev/null +++ b/sdf/1.11/inertial.sdf @@ -0,0 +1,209 @@ + + + + The link's mass, position of its center of mass, its central inertia + properties, and optionally its fluid added mass. + + + + The mass of the link. + + + + + This pose (translation, rotation) describes the position and orientation + of the link's center-of-mass-frame C relative to the link-frame L. + The first three components (x y z) specify the position vector from Lo + (the link-frame origin) to Co (the link's center of mass) as + `x L̂x + y L̂y + z L̂ᴢ`, where L̂x, L̂y, L̂ᴢ are link-frame L's orthogonal unit + vectors. The subsequent values characterize C's orientation relative to + link-frame L as a sequence of Euler rotations + (r p y) documented in http://sdformat.org/tutorials?tut=specify_pose, + or as a quaternion (x y z w), where w is the scalar component. + + + + 'euler_rpy' by default. Supported rotation formats are + 'euler_rpy', Euler angles representation in roll, pitch, yaw. The pose is expected to have 6 values. + 'quat_xyzw', Quaternion representation in x, y, z, w. The pose is expected to have 7 values. + + + + + + Whether or not the euler angles are in degrees, otherwise they will be interpreted as radians by default. + + + + + + + + This link's moments of inertia ixx, iyy, izz and products of inertia + ixy, ixz, iyz about Co (the link's center of mass) for the unit vectors + Ĉx, Ĉy, Ĉᴢ fixed in the center-of-mass-frame C. + Note: the orientation of Ĉx, Ĉy, Ĉᴢ relative to L̂x, L̂y, L̂ᴢ is specified + by the `pose` tag. + To avoid compatibility issues associated with the negative sign + convention for product of inertia, align Ĉx, Ĉy, Ĉᴢ with principal + inertia directions so that all the products of inertia are zero. + For more information about this sign convention, see the following + MathWorks documentation for working with CAD tools: + https://www.mathworks.com/help/releases/R2021b/physmod/sm/ug/specify-custom-inertia.html#mw_b043ec69-835b-4ca9-8769-af2e6f1b190c + + + + The link's moment of inertia about Co (the link's center of mass) for Ĉx. + + + + + The link's product of inertia about Co (the link's center of mass) for + Ĉx and Ĉy, where the product of inertia convention -m x y (not +m x y) + is used. If Ĉx or Ĉy is a principal inertia direction, ixy = 0. + + + + + The link's product of inertia about Co (the link's center of mass) for + Ĉx and Ĉz, where the product of inertia convention -m x z (not +m x z) + is used. If Ĉx or Ĉz is a principal inertia direction, ixz = 0. + + + + + The link's moment of inertia about Co (the link's center of mass) for Ĉy. + + + + + The link's product of inertia about Co (the link's center of mass) for + Ĉy and Ĉz, where the product of inertia convention -m y z (not +m y z) + is used. If Ĉy or Ĉz is a principal inertia direction, iyz = 0. + + + + + The link's moment of inertia about Co (the link's center of mass) for Ĉz. + + + + + + + This link's fluid added mass matrix about the link's origin. + This matrix represents the inertia of the fluid that is dislocated when the + body moves. Added mass should be zero if the density of the surrounding + fluid is negligible with respect to the body's density. + The 6x6 matrix is symmetric, therefore only 21 unique elements can be set. + The elements of the matrix follow the [x, y, z, p, q, r] notation, where + [x, y, z] correspond to translation and [p, q, r] to rotation + (i.e. roll, pitch, yaw). + + + + Added mass in the X axis due to linear acceleration in the X axis, in kg. + + + + + Added mass in the X axis due to linear acceleration in the Y axis, and vice-versa, in kg. + + + + + Added mass in the X axis due to linear acceleration in the Z axis, and vice-versa, in kg. + + + + + Added mass in the X axis due to angular acceleration about the X axis, and vice-versa, in kg * m. + + + + + Added mass in the X axis due to angular acceleration about the Y axis, and vice-versa, in kg * m. + + + + + Added mass in the X axis due to angular acceleration about the Z axis, and vice-versa, in kg * m. + + + + + Added mass in the Y axis due to linear acceleration in the Y axis, in kg. + + + + + Added mass in the Y axis due to linear acceleration in the Z axis, and vice-versa, in kg. + + + + + Added mass in the Y axis due to angular acceleration about the X axis, and vice-versa, in kg * m. + + + + + Added mass in the Y axis due to angular acceleration about the Y axis, and vice-versa, in kg * m. + + + + + Added mass in the Y axis due to angular acceleration about the Z axis, and vice-versa, in kg * m. + + + + + Added mass in the Z axis due to linear acceleration in the Z axis, in kg. + + + + + Added mass in the Z axis due to angular acceleration about the X axis, and vice-versa, in kg * m. + + + + + Added mass in the Z axis due to angular acceleration about the Y axis, and vice-versa, in kg * m. + + + + + Added mass in the Z axis due to angular acceleration about the Z axis, and vice-versa, in kg * m. + + + + + Added mass moment about the X axis due to angular acceleration about the X axis, in kg * m^2. + + + + + Added mass moment about the X axis due to angular acceleration about the Y axis, and vice-versa, in kg * m^2. + + + + + Added mass moment about the X axis due to angular acceleration about the Z axis, and vice-versa, in kg * m^2. + + + + + Added mass moment about the Y axis due to angular acceleration about the Y axis, in kg * m^2. + + + + + Added mass moment about the Y axis due to angular acceleration about the Z axis, and vice-versa, in kg * m^2. + + + + + Added mass moment about the Z axis due to angular acceleration about the Z axis, in kg * m^2. + + + + diff --git a/sdf/1.11/joint.sdf b/sdf/1.11/joint.sdf new file mode 100644 index 000000000..b757614e9 --- /dev/null +++ b/sdf/1.11/joint.sdf @@ -0,0 +1,240 @@ + + + A joint connects two links with kinematic and dynamic properties. By default, the pose of a joint is expressed in the child link frame. + + + A unique name for the joint within its scope. + + + + The type of joint, which must be one of the following: + (continuous) a hinge joint that rotates on a single axis with a continuous range of motion, + (revolute) a hinge joint that rotates on a single axis with a fixed range of motion, + (gearbox) geared revolute joints, + (revolute2) same as two revolute joints connected in series, + (prismatic) a sliding joint that slides along an axis with a limited range specified by upper and lower limits, + (ball) a ball and socket joint, + (screw) a single degree of freedom joint with coupled sliding and rotational motion, + (universal) like a ball joint, but constrains one degree of freedom, + (fixed) a joint with zero degrees of freedom that rigidly connects two links. + + + + + Name of the parent frame or "world". + + + + Name of the child frame. The value "world" may not be specified. + + + + Parameter for gearbox joints. Given theta_1 and theta_2 defined in description for gearbox_reference_body, theta_2 = -gearbox_ratio * theta_1. + + + + Parameter for gearbox joints. Gearbox ratio is enforced over two joint angles. First joint angle (theta_1) is the angle from the gearbox_reference_body to the parent link in the direction of the axis element and the second joint angle (theta_2) is the angle from the gearbox_reference_body to the child link in the direction of the axis2 element. + + + + + Parameter for screw joints representing the ratio between rotation + and translation of the joint. This parameter has been interpreted by + gazebo-classic as having units of radians / meter with a positive value + corresponding to a left-handed thread. + The parameter is now deprecated in favor of `screw_thread_pitch`. + + + + + + A parameter for screw joint kinematics, representing the + axial distance traveled for each revolution of the joint, + with units of meters / revolution with a positive value corresponding + to a right-handed thread. + This parameter supersedes `thread_pitch`. + + + + + + Parameters related to the axis of rotation for revolute joints, + the axis of translation for prismatic joints. + + + + Represents the x,y,z components of the axis unit vector. The axis is + expressed in the joint frame unless a different frame is expressed in + the expressed_in attribute. The vector should be normalized. + + + + Name of frame in whose coordinates the xyz unit vector is expressed. + + + + + An element specifying physical properties of the joint. These values are used to specify modeling properties of the joint, particularly useful for simulation. + + The physical velocity dependent viscous damping coefficient of the joint. + + + The physical static friction value of the joint. + + + The spring reference position for this joint axis. + + + The spring stiffness for this joint axis. + + + + specifies the limits of this joint + + Specifies the lower joint limit (radians for revolute joints, meters for prismatic joints). Omit if joint is continuous. + + + Specifies the upper joint limit (radians for revolute joints, meters for prismatic joints). Omit if joint is continuous. + + + A value for enforcing the maximum joint effort applied. Limit is not enforced if value is negative. + + + A value for enforcing the maximum joint velocity. + + + + Joint stop stiffness. + + + + Joint stop dissipation. + + + + + + + + Parameters related to the second axis of rotation for revolute2 joints and universal joints. + + + + Represents the x,y,z components of the axis unit vector. The axis is + expressed in the joint frame unless a different frame is expressed in + the expressed_in attribute. The vector should be normalized. + + + + Name of frame in whose coordinates the xyz unit vector is expressed. + + + + + An element specifying physical properties of the joint. These values are used to specify modeling properties of the joint, particularly useful for simulation. + + The physical velocity dependent viscous damping coefficient of the joint. EXPERIMENTAL: if damping coefficient is negative and implicit_spring_damper is true, adaptive damping is used. + + + The physical static friction value of the joint. + + + The spring reference position for this joint axis. + + + The spring stiffness for this joint axis. + + + + + + + An attribute specifying the lower joint limit (radians for revolute joints, meters for prismatic joints). Omit if joint is continuous. + + + An attribute specifying the upper joint limit (radians for revolute joints, meters for prismatic joints). Omit if joint is continuous. + + + An attribute for enforcing the maximum joint effort applied by Joint::SetForce. Limit is not enforced if value is negative. + + + (not implemented) An attribute for enforcing the maximum joint velocity. + + + + Joint stop stiffness. Supported physics engines: SimBody. + + + + Joint stop dissipation. Supported physics engines: SimBody. + + + + + + + Parameters that are specific to a certain physics engine. + + Simbody specific parameters + + Force cut in the multibody graph at this joint. + + + + ODE specific parameters + + If cfm damping is set to true, ODE will use CFM to simulate damping, allows for infinite damping, and one additional constraint row (previously used for joint limit) is always active. + + + + If implicit_spring_damper is set to true, ODE will use CFM, ERP to simulate stiffness and damping, allows for infinite damping, and one additional constraint row (previously used for joint limit) is always active. This replaces cfm_damping parameter in SDFormat 1.4. + + + + Scale the excess for in a joint motor at joint limits. Should be between zero and one. + + + Constraint force mixing for constrained directions + + + Error reduction parameter for constrained directions + + + Bounciness of the limits + + + Maximum force or torque used to reach the desired velocity. + + + The desired velocity of the joint. Should only be set if you want the joint to move on load. + + + + + + Constraint force mixing parameter used by the joint stop + + + Error reduction parameter used by the joint stop + + + + + + + Suspension constraint force mixing parameter + + + Suspension error reduction parameter + + + + + + If provide feedback is set to true, physics engine will compute the constraint forces at this joint. + + + + + + diff --git a/sdf/1.11/lidar.sdf b/sdf/1.11/lidar.sdf new file mode 100644 index 000000000..f0100a9a3 --- /dev/null +++ b/sdf/1.11/lidar.sdf @@ -0,0 +1,78 @@ + + These elements are specific to the lidar sensor. + + + + + + + + The number of simulated lidar rays to generate per complete laser sweep cycle. + + + + This number is multiplied by samples to determine the number of range data points returned. If resolution is not equal to one, range data is interpolated. + + + + + + + + Must be greater or equal to min_angle + + + + + + + + The number of simulated lidar rays to generate per complete laser sweep cycle. + + + + This number is multiplied by samples to determine the number of range data points returned. If resolution is not equal to one, range data is interpolated. + + + + + + + + Must be greater or equal to min_angle + + + + + + + specifies range properties of each simulated lidar + + The minimum distance for each lidar ray. + + + The maximum distance for each lidar ray. + + + Linear resolution of each lidar ray. + + + + + The properties of the noise model that should be applied to generated scans + + The type of noise. Currently supported types are: "gaussian" (draw noise values independently for each beam from a Gaussian distribution). + + + For type "gaussian," the mean of the Gaussian distribution from which noise values are drawn. + + + For type "gaussian," the standard deviation of the Gaussian distribution from which noise values are drawn. + + + + + + + + diff --git a/sdf/1.11/light.sdf b/sdf/1.11/light.sdf new file mode 100644 index 000000000..356bad19a --- /dev/null +++ b/sdf/1.11/light.sdf @@ -0,0 +1,71 @@ + + + The light element describes a light source. + + + A unique name for the light. + + + + The light type: point, directional, spot. + + + + When true, the light will cast shadows. + + + + When true, the light is on. + + + + If true, the light is visualized in the GUI + + + + Scale factor to set the relative power of a light. + + + + + + Diffuse light color + + + Specular light color + + + + Light attenuation + + Range of the light + + + The linear attenuation factor: 1 means attenuate evenly over the distance. + + + The constant attenuation factor: 1.0 means never attenuate, 0.0 is complete attenutation. + + + The quadratic attenuation factor: adds a curvature to the attenuation. + + + + + Direction of the light, only applicable for spot and directional lights. + + + + Spot light parameters + + Angle covered by the bright inner cone + + + Angle covered by the outer cone + + + The rate of falloff between the inner and outer cones. 1.0 means a linear falloff, less means slower falloff, higher means faster falloff. + + + + diff --git a/sdf/1.11/light_state.sdf b/sdf/1.11/light_state.sdf new file mode 100644 index 000000000..673efcdb4 --- /dev/null +++ b/sdf/1.11/light_state.sdf @@ -0,0 +1,10 @@ + + + Light state + + + Name of the light + + + + diff --git a/sdf/1.11/link.sdf b/sdf/1.11/link.sdf new file mode 100644 index 000000000..b518ff3c5 --- /dev/null +++ b/sdf/1.11/link.sdf @@ -0,0 +1,51 @@ + + + A physical link with inertia, collision, and visual properties. A link must be a child of a model, and any number of links may exist in a model. + + + A unique name for the link within the scope of the model. + + + + If true, the link is affected by gravity. + + + + If true, the link is affected by the wind. + + + + If true, the link can collide with other links in the model. Two links within a model will collide if link1.self_collide OR link2.self_collide. Links connected by a joint will never collide. + + + + If true, the link is kinematic only + + + + If true, the link will have 6DOF and be a direct child of world. + + + + Exponential damping of the link's velocity. + + Linear damping + + + Angular damping + + + + + + + + + + + + + + + + diff --git a/sdf/1.11/link_state.sdf b/sdf/1.11/link_state.sdf new file mode 100644 index 000000000..28fff465d --- /dev/null +++ b/sdf/1.11/link_state.sdf @@ -0,0 +1,40 @@ + + + Link state + + + Name of the link + + + + Velocity of the link. The x, y, z components of the pose + correspond to the linear velocity of the link, and the roll, pitch, yaw + components correspond to the angular velocity of the link + + + + + Acceleration of the link. The x, y, z components of the pose + correspond to the linear acceleration of the link, and the roll, + pitch, yaw components correspond to the angular acceleration of the link + + + + + Force and torque applied to the link. The x, y, z components + of the pose correspond to the force applied to the link, and the roll, + pitch, yaw components correspond to the torque applied to the link + + + + + Collision state + + + Name of the collision + + + + + + diff --git a/sdf/1.11/logical_camera.sdf b/sdf/1.11/logical_camera.sdf new file mode 100644 index 000000000..de47703c0 --- /dev/null +++ b/sdf/1.11/logical_camera.sdf @@ -0,0 +1,19 @@ + + These elements are specific to logical camera sensors. A logical camera reports objects that fall within a frustum. Computation should be performed on the CPU. + + + Near clipping distance of the view frustum + + + + Far clipping distance of the view frustum + + + + Aspect ratio of the near and far planes. This is the width divided by the height of the near or far planes. + + + + Horizontal field of view of the frustum, in radians. This is the angle between the frustum's vertex and the edges of the near or far plane. + + diff --git a/sdf/1.11/magnetometer.sdf b/sdf/1.11/magnetometer.sdf new file mode 100644 index 000000000..2f3bfeeee --- /dev/null +++ b/sdf/1.11/magnetometer.sdf @@ -0,0 +1,21 @@ + + These elements are specific to a Magnetometer sensor. + + + Parameters related to the body-frame X axis of the magnetometer + + + + + + Parameters related to the body-frame Y axis of the magnetometer + + + + + + Parameters related to the body-frame Z axis of the magnetometer + + + + \ No newline at end of file diff --git a/sdf/1.11/material.sdf b/sdf/1.11/material.sdf new file mode 100644 index 000000000..3459654e4 --- /dev/null +++ b/sdf/1.11/material.sdf @@ -0,0 +1,166 @@ + + + The material of the visual element. + + + Name of material from an installed script file. This will override the color element if the script exists. + + + URI of the material script file + + + + Name of the script within the script file + + + + + + + vertex, pixel, normal_map_object_space, normal_map_tangent_space + + + + filename of the normal map + + + + + Set render order for coplanar polygons. The higher value will be rendered on top of the other coplanar polygons + + + + If false, dynamic lighting will be disabled + + + + The ambient color of a material specified by set of four numbers representing red/green/blue, each in the range of [0,1]. + + + + The diffuse color of a material specified by set of four numbers representing red/green/blue/alpha, each in the range of [0,1]. + + + + The specular color of a material specified by set of four numbers representing red/green/blue/alpha, each in the range of [0,1]. + + + + The specular exponent of a material + + + + The emissive color of a material specified by set of four numbers representing red/green/blue, each in the range of [0,1]. + + + + If true, the mesh that this material is applied to will be rendered as double sided + + + + + Physically Based Rendering (PBR) material. There are two PBR workflows: metal and specular. While both workflows and their parameters can be specified at the same time, typically only one of them will be used (depending on the underlying renderer capability). It is also recommended to use the same workflow for all materials in the world. + + + PBR using the Metallic/Roughness workflow. + + + Filename of the diffuse/albedo map. + + + + Filename of the roughness map. + + + + Material roughness in the range of [0,1], where 0 represents a smooth surface and 1 represents a rough surface. This is the inverse of a specular map in a PBR specular workflow. + + + + Filename of the metalness map. + + + + Material metalness in the range of [0,1], where 0 represents non-metal and 1 represents raw metal + + + + Filename of the environment / reflection map, typically in the form of a cubemap + + + + Filename of the ambient occlusion map. The map defines the amount of ambient lighting on the surface. + + + + + The space that the normals are in. Values are: 'object' or 'tangent' + + + Filename of the normal map. The normals can be in the object space or tangent space as specified in the 'type' attribute + + + + Filename of the emissive map. + + + + + Index of the texture coordinate set to use. + + Filename of the light map. The light map is a prebaked light texture that is applied over the albedo map + + + + + + PBR using the Specular/Glossiness workflow. + + + Filename of the diffuse/albedo map. + + + + Filename of the specular map. + + + + Filename of the glossiness map. + + + + Material glossiness in the range of [0-1], where 0 represents a rough surface and 1 represents a smooth surface. This is the inverse of a roughness map in a PBR metal workflow. + + + + Filename of the environment / reflection map, typically in the form of a cubemap + + + + Filename of the ambient occlusion map. The map defines the amount of ambient lighting on the surface. + + + + + The space that the normals are in. Values are: 'object' or 'tangent' + + + Filename of the normal map. The normals can be in the object space or tangent space as specified in the 'type' attribute + + + + Filename of the emissive map. + + + + + Index of the texture coordinate set to use. + + Filename of the light map. The light map is a prebaked light texture that is applied over the albedo map + + + + + + + diff --git a/sdf/1.11/mesh_shape.sdf b/sdf/1.11/mesh_shape.sdf new file mode 100644 index 000000000..61bce76dd --- /dev/null +++ b/sdf/1.11/mesh_shape.sdf @@ -0,0 +1,20 @@ + + Mesh shape + + Mesh uri + + + + Use a named submesh. The submesh must exist in the mesh specified by the uri + + Name of the submesh within the parent mesh + + + Set to true to center the vertices of the submesh at 0,0,0. This will effectively remove any transformations on the submesh before the poses from parent links and models are applied. + + + + + Scaling factor applied to the mesh + + diff --git a/sdf/1.11/model.sdf b/sdf/1.11/model.sdf new file mode 100644 index 000000000..57dc83a01 --- /dev/null +++ b/sdf/1.11/model.sdf @@ -0,0 +1,92 @@ + + + The model element defines a complete robot or any other physical object. + + + + The name of the model and its implicit frame. This name must be unique + among all elements defining frames within the same scope, i.e., it must + not match another //model, //frame, //joint, or //link within the same + scope. + + + + + + The name of the model's canonical link, to which the model's implicit + coordinate frame is attached. If unset or set to an empty string, the + first `/link` listed as a direct child of this model is chosen as the + canonical link. If the model has no direct `/link` children, it will + instead be attached to the first nested (or included) model's implicit + frame. + + + + The frame inside this model whose pose will be set by the pose element of the model. i.e, the pose element specifies the pose of this frame instead of the model frame. + + + + + If set to true, the model is immovable; i.e., a dynamics engine will not + update its position. This will also overwrite this model's `@canonical_link` + and instead attach the model's implicit frame to the world's implicit frame. + This holds even if this model is nested (or included) by another model + instead of being a direct child of `//world`. + + + + + If set to true, all links in the model will collide with each other (except those connected by a joint). Can be overridden by the link or collision element self_collide property. Two links within a model will collide if link1.self_collide OR link2.self_collide. Links connected by a joint will never collide. + + + + Allows a model to auto-disable, which is means the physics engine can skip updating the model when the model is at rest. This parameter is only used by models with no joints. + + + + + + + + + + + + Include resources from a URI. This can be used to nest models. The included resource can only contain one 'model' element. The URI can point to a directory or a file. If the URI is a directory, it must conform to the model database structure (see /tutorials?tut=composition&cat=specification&#defining-models-in-separate-files). + + + Merge the included nested model into the top model + + + + URI to a resource, such as a model + + + + + + + Override the name of the included model. + + + + Override the static value of the included model. + + + + The frame inside the included model whose pose will be set by the specified pose element. If this element is specified, the pose must be specified. + + + + + A nested model element + + A unique name for the model. This name must not match another nested model in the same level as this model. + + + + + If set to true, all links in the model will be affected by the wind. Can be overriden by the link wind property. + + + diff --git a/sdf/1.11/model_state.sdf b/sdf/1.11/model_state.sdf new file mode 100644 index 000000000..3fd296ff2 --- /dev/null +++ b/sdf/1.11/model_state.sdf @@ -0,0 +1,41 @@ + + + Model state + + + Name of the model + + + + Joint angle + + + Name of the joint + + + + + Index of the axis. + + + Angle of an axis + + + + + A nested model state element + + Name of the model. + + + + + + + + Scale for the 3 dimensions of the model. + + + + + diff --git a/sdf/1.11/navsat.sdf b/sdf/1.11/navsat.sdf new file mode 100644 index 000000000..8a8f1719b --- /dev/null +++ b/sdf/1.11/navsat.sdf @@ -0,0 +1,40 @@ + + These elements are specific to the NAVSAT sensor. + + + + Parameters related to NAVSAT position measurement. + + + + Noise parameters for horizontal position measurement, in units of meters. + + + + + + Noise parameters for vertical position measurement, in units of meters. + + + + + + + + Parameters related to NAVSAT position measurement. + + + + Noise parameters for horizontal velocity measurement, in units of meters/second. + + + + + + Noise parameters for vertical velocity measurement, in units of meters/second. + + + + + + diff --git a/sdf/1.11/noise.sdf b/sdf/1.11/noise.sdf new file mode 100644 index 000000000..ceeac5037 --- /dev/null +++ b/sdf/1.11/noise.sdf @@ -0,0 +1,43 @@ + + The properties of a sensor noise model. + + + + The type of noise. Currently supported types are: + "none" (no noise). + "gaussian" (draw noise values independently for each measurement from a Gaussian distribution). + "gaussian_quantized" ("gaussian" plus quantization of outputs (ie. rounding)) + + + + + For type "gaussian*", the mean of the Gaussian distribution from which + noise values are drawn. + + + + For type "gaussian*", the standard deviation of the Gaussian distribution from which noise values are drawn. + + + For type "gaussian*", the mean of the Gaussian distribution from which bias values are drawn. + + + For type "gaussian*", the standard deviation of the Gaussian distribution from which bias values are drawn. + + + + For type "gaussian*", the standard deviation of the noise used to drive a process to model slow variations in a sensor bias. + + + + For type "gaussian*", the correlation time in seconds of the noise used to drive a process to model slow variations in a sensor bias. A typical value, when used, would be on the order of 3600 seconds (1 hour). + + + + + For type "gaussian_quantized", the precision of output signals. A value + of zero implies infinite precision / no quantization. + + + + diff --git a/sdf/1.11/particle_emitter.sdf b/sdf/1.11/particle_emitter.sdf new file mode 100755 index 000000000..d6a2fdf7e --- /dev/null +++ b/sdf/1.11/particle_emitter.sdf @@ -0,0 +1,120 @@ + + + A particle emitter that can be used to describe fog, smoke, and dust. + + + A unique name for the particle emitter. + + + + The type of a particle emitter. One of "box", "cylinder", "ellipsoid", or "point". + + + + True indicates that the particle emitter should generate particles when loaded + + + + The number of seconds the emitter is active. A value less than or equal to zero means infinite duration. + + + + + The size of the emitter where the particles are sampled. + Default value is (1, 1, 1). + Note that the interpretation of the emitter area varies + depending on the emmiter type: + - point: The area is ignored. + - box: The area is interpreted as width X height X depth. + - cylinder: The area is interpreted as the bounding box of the + cylinder. The cylinder is oriented along the Z-axis. + - ellipsoid: The area is interpreted as the bounding box of an + ellipsoid shaped area, i.e. a sphere or + squashed-sphere area. The parameters are again + identical to EM_BOX, except that the dimensions + describe the widest points along each of the axes. + + + + + The particle dimensions (width, height, depth). + + + + The number of seconds each particle will ’live’ for before being destroyed. This value must be greater than zero. + + + + The number of particles per second that should be emitted. + + + + Sets a minimum velocity for each particle (m/s). + + + + Sets a maximum velocity for each particle (m/s). + + + + Sets the amount by which to scale the particles in both x and y direction per second. + + + + + Sets the starting color for all particles emitted. + The actual color will be interpolated between this color + and the one set under color_end. + Color::White is the default color for the particles + unless a specific function is used. + To specify a color, RGB values should be passed in. + For example, to specify red, a user should enter: + 1 0 0 + Note that this function overrides the particle colors set + with color_range_image. + + + + + + Sets the end color for all particles emitted. + The actual color will be interpolated between this color + and the one set under color_start. + Color::White is the default color for the particles + unless a specific function is used (see color_start for + more information about defining custom colors with RGB + values). + Note that this function overrides the particle colors set + with color_range_image. + + + + + + Sets the path to the color image used as an affector. This affector modifies the color of particles in flight. The colors are taken from a specified image file. The range of color values begins from the left side of the image and moves to the right over the lifetime of the particle, therefore only the horizontal dimension of the image is used. Note that this function overrides the particle colors set with color_start and color_end. + + + + + + Topic used to update particle emitter properties at runtime. + The default topic is + /model/{model_name}/particle_emitter/{emitter_name} + Note that the emitter id and name may not be changed. + + + + + + This is used to determine the ratio of particles that will be detected + by sensors. Increasing the ratio means there is a higher chance of + particles reflecting and interfering with depth sensing, making the + emitter appear more dense. Decreasing the ratio decreases the chance + of particles reflecting and interfering with depth sensing, making it + appear less dense. + + + + + + diff --git a/sdf/1.11/physics.sdf b/sdf/1.11/physics.sdf new file mode 100644 index 000000000..f644ae9fd --- /dev/null +++ b/sdf/1.11/physics.sdf @@ -0,0 +1,238 @@ + + + The physics tag specifies the type and properties of the dynamics engine. + + + The name of this set of physics parameters. + + + + If true, this physics element is set as the default physics profile for the world. If multiple default physics elements exist, the first element marked as default is chosen. If no default physics element exists, the first physics element is chosen. + + + + The type of the dynamics engine. Current options are ode, bullet, simbody and dart. Defaults to ode if left unspecified. + + + + Maximum time step size at which every system in simulation can interact with the states of the world. (was physics.sdf's dt). + + + + + target simulation speedup factor, defined by ratio of simulation time to real-time. + + + + + Rate at which to update the physics engine (UpdatePhysics calls per real-time second). (was physics.sdf's update_rate). + + + + Maximum number of contacts allowed between two entities. This value can be over ridden by a max_contacts element in a collision element. + + + + DART specific physics properties + + + + One of the following types: pgs, dantzig. PGS stands for Projected Gauss-Seidel. + + + + Specify collision detector for DART to use. Can be dart, fcl, bullet or ode. + + + + + Simbody specific physics properties + + (Currently not used in simbody) The time duration which advances with each iteration of the dynamics engine, this has to be no bigger than max_step_size under physics block. If left unspecified, min_step_size defaults to max_step_size. + + + Roughly the relative error of the system. + -LOG(accuracy) is roughly the number of significant digits. + + + Tolerable "slip" velocity allowed by the solver when static + friction is supposed to hold object in place. + + + vc + Assume real COR=1 when v=0. + e_min = given minimum COR, at v >= vp (a.k.a. plastic_coef_restitution) + d = slope = (1-e_min)/vp + OR, e_min = 1 - d*vp + e_max = maximum COR = 1-d*vc, reached at v=vc + e = 0, v <= vc + = 1 - d*v, vc < v < vp + = e_min, v >= vp + + dissipation factor = d*min(v,vp) [compliant] + cor = e [rigid] + + Combining rule e = 0, e1==e2==0 + = 2*e1*e2/(e1+e2), otherwise]]> + + + + Default contact material stiffness + (force/dist or torque/radian). + + + dissipation coefficient to be used in compliant contact; + if not given it is (1-min_cor)/plastic_impact_velocity + + + + this is the COR to be used at high velocities for rigid + impacts; if not given it is 1 - dissipation*plastic_impact_velocity + + + + + smallest impact velocity at which min COR is reached; set + to zero if you want the min COR always to be used + + + + static friction (mu_s) as described by this plot: http://gazebosim.org/wiki/File:Stribeck_friction.png + + + dynamic friction (mu_d) as described by this plot: http://gazebosim.org/wiki/File:Stribeck_friction.png + + + viscous friction (mu_v) with units of (1/velocity) as described by this plot: http://gazebosim.org/wiki/File:Stribeck_friction.png + + + + for rigid impacts only, impact velocity at which + COR is set to zero; normally inherited from global default but can + be overridden here. Combining rule: use larger velocity + + + + This is the largest slip velocity at which + we'll consider a transition to stiction. Normally inherited + from a global default setting. For a continuous friction model + this is the velocity at which the max static friction force + is reached. Combining rule: use larger velocity + + + + + + + Bullet specific physics properties + + + + One of the following types: sequential_impulse only. + + + The time duration which advances with each iteration of the dynamics engine, this has to be no bigger than max_step_size under physics block. If left unspecified, min_step_size defaults to max_step_size. + + + Number of iterations for each step. A higher number produces greater accuracy at a performance cost. + + + Set the successive over-relaxation parameter. + + + + + Bullet constraint parameters. + + Constraint force mixing parameter. See the ODE page for more information. + + + Error reduction parameter. See the ODE page for more information. + + + The depth of the surface layer around all geometry objects. Contacts are allowed to sink into the surface layer up to the given depth before coming to rest. The default value is zero. Increasing this to some small value (e.g. 0.001) can help prevent jittering problems due to contacts being repeatedly made and broken. + + + Similar to ODE's max_vel implementation. See http://web.archive.org/web/20120430155635/http://bulletphysics.org/mediawiki-1.5.8/index.php/BtContactSolverInfo#Split_Impulse for more information. + + + Similar to ODE's max_vel implementation. See http://web.archive.org/web/20120430155635/http://bulletphysics.org/mediawiki-1.5.8/index.php/BtContactSolverInfo#Split_Impulse for more information. + + + + + + ODE specific physics properties + + + + One of the following types: world, quick + + + The time duration which advances with each iteration of the dynamics engine, this has to be no bigger than max_step_size under physics block. If left unspecified, min_step_size defaults to max_step_size. + + + Number of threads to use for "islands" of disconnected models. + + + Number of iterations for each step. A higher number produces greater accuracy at a performance cost. + + + Experimental parameter. + + + Set the successive over-relaxation parameter. + + + Flag to use threading to speed up position correction computation. + + + + Flag to enable dynamic rescaling of moment of inertia in constrained directions. + See gazebo pull request 1114 for the implementation of this feature. + https://osrf-migration.github.io/gazebo-gh-pages/#!/osrf/gazebo/pull-request/1114 + + + + + Name of ODE friction model to use. Valid values include: + + pyramid_model: (default) friction forces limited in two directions + in proportion to normal force. + box_model: friction forces limited to constant in two directions. + cone_model: friction force magnitude limited in proportion to normal force. + + See gazebo pull request 1522 for the implementation of this feature. + https://osrf-migration.github.io/gazebo-gh-pages/#!/osrf/gazebo/pull-request/1522 + https://github.com/osrf/gazebo/commit/968dccafdfbfca09c9b3326f855612076fed7e6f + + + + + + ODE constraint parameters. + + Constraint force mixing parameter. See the ODE page for more information. + + + Error reduction parameter. See the ODE page for more information. + + + The maximum correcting velocities allowed when resolving contacts. + + + The depth of the surface layer around all geometry objects. Contacts are allowed to sink into the surface layer up to the given depth before coming to rest. The default value is zero. Increasing this to some small value (e.g. 0.001) can help prevent jittering problems due to contacts being repeatedly made and broken. + + + + diff --git a/sdf/1.11/plane_shape.sdf b/sdf/1.11/plane_shape.sdf new file mode 100644 index 000000000..2a82be152 --- /dev/null +++ b/sdf/1.11/plane_shape.sdf @@ -0,0 +1,9 @@ + + Plane shape + + Normal direction for the plane. When a Plane is used as a geometry for a Visual or Collision object, then the normal is specified in the Visual or Collision frame, respectively. + + + Length of each side of the plane. Note that this property is meaningful only for visualizing the Plane, i.e., when the Plane is used as a geometry for a Visual object. The Plane has infinite size when used as a geometry for a Collision object. + + diff --git a/sdf/1.11/plugin.sdf b/sdf/1.11/plugin.sdf new file mode 100644 index 000000000..4564fe4fb --- /dev/null +++ b/sdf/1.11/plugin.sdf @@ -0,0 +1,13 @@ + + + A plugin is a dynamically loaded chunk of code. It can exist as a child of world, model, and sensor. + + A name for the plugin. + + + Name of the shared library to load. If the filename is not a full path name, the file will be searched for in the configuration paths. + + + This is a special element that should not be specified in an SDFormat file. It automatically copies child elements into the SDFormat element so that a plugin can access the data. + + diff --git a/sdf/1.11/polyline_shape.sdf b/sdf/1.11/polyline_shape.sdf new file mode 100644 index 000000000..665884354 --- /dev/null +++ b/sdf/1.11/polyline_shape.sdf @@ -0,0 +1,14 @@ + + Defines an extruded polyline shape + + + + A series of points that define the path of the polyline. + + + + + Height of the polyline + + + diff --git a/sdf/1.11/population.sdf b/sdf/1.11/population.sdf new file mode 100644 index 000000000..b14ad739c --- /dev/null +++ b/sdf/1.11/population.sdf @@ -0,0 +1,58 @@ + + + + The population element defines how and where a set of models will + be automatically populated in Gazebo. + + + + + A unique name for the population. This name must not match + another population in the world. + + + + + + + + The number of models to place. + + + + + Specifies the type of object distribution and its optional parameters. + + + + + Define how the objects will be placed in the specified region. + - random: Models placed at random. + - uniform: Models approximately placed in a 2D grid pattern with control + over the number of objects. + - grid: Models evenly placed in a 2D grid pattern. The number of objects + is not explicitly specified, it is based on the number of rows and + columns of the grid. + - linear-x: Models evently placed in a row along the global x-axis. + - linear-y: Models evently placed in a row along the global y-axis. + - linear-z: Models evently placed in a row along the global z-axis. + + + + + Number of rows in the grid. + + + Number of columns in the grid. + + + Distance between elements of the grid. + + + + + + + + + diff --git a/sdf/1.11/pose.sdf b/sdf/1.11/pose.sdf new file mode 100644 index 000000000..81bc62c3e --- /dev/null +++ b/sdf/1.11/pose.sdf @@ -0,0 +1,43 @@ + + + A pose (translation, rotation) expressed in the frame named by + @relative_to. The first three components (x, y, z) represent the position of + the element's origin (in the @relative_to frame). The rotation component + represents the orientation of the element as either a sequence of Euler + rotations (r, p, y), see http://sdformat.org/tutorials?tut=specify_pose, + or as a quaternion (x, y, z, w), where w is the real component. + + + + If specified, this pose is expressed in the named frame. The named frame + must be declared within the same scope (world/model) as the element that + has its pose specified by this tag. + + If missing, the pose is expressed in the frame of the parent XML element + of the element that contains the pose. For exceptions to this rule and + more details on the default behavior, see + http://sdformat.org/tutorials?tut=pose_frame_semantics. + + Note that @relative_to merely affects an element's initial pose and + does not affect the element's dynamic movement thereafter. + + New in v1.8: @relative_to may use frames of nested scopes. In this case, + the frame is specified using `::` as delimiter to define the scope of the + frame, e.g. `nested_model_A::nested_model_B::awesome_frame`. + + + + + 'euler_rpy' by default. Supported rotation formats are + 'euler_rpy', Euler angles representation in roll, pitch, yaw. The pose is expected to have 6 values. + 'quat_xyzw', Quaternion representation in x, y, z, w. The pose is expected to have 7 values. + + + + + + Whether or not the euler angles are in degrees, otherwise they will be interpreted as radians by default. + + + + diff --git a/sdf/1.11/projector.sdf b/sdf/1.11/projector.sdf new file mode 100644 index 000000000..ca8cc8fd3 --- /dev/null +++ b/sdf/1.11/projector.sdf @@ -0,0 +1,32 @@ + + + + Name of the projector + + + + Texture name + + + + Field of view + + + + + Near clip distance + + + + + far clip distance + + + + + + + + + + diff --git a/sdf/1.11/ray.sdf b/sdf/1.11/ray.sdf new file mode 100644 index 000000000..58cf68b39 --- /dev/null +++ b/sdf/1.11/ray.sdf @@ -0,0 +1,78 @@ + + These elements are specific to the ray (laser) sensor. + + + + + + + + The number of simulated rays to generate per complete laser sweep cycle. + + + + This number is multiplied by samples to determine the number of range data points returned. If resolution is less than one, range data is interpolated. If resolution is greater than one, range data is averaged. + + + + + + + + Must be greater or equal to min_angle + + + + + + + + The number of simulated rays to generate per complete laser sweep cycle. + + + + This number is multiplied by samples to determine the number of range data points returned. If resolution is less than one, range data is interpolated. If resolution is greater than one, range data is averaged. + + + + + + + + Must be greater or equal to min_angle + + + + + + + specifies range properties of each simulated ray + + The minimum distance for each ray. + + + The maximum distance for each ray. + + + Linear resolution of each ray. + + + + + The properties of the noise model that should be applied to generated scans + + The type of noise. Currently supported types are: "gaussian" (draw noise values independently for each beam from a Gaussian distribution). + + + For type "gaussian," the mean of the Gaussian distribution from which noise values are drawn. + + + For type "gaussian," the standard deviation of the Gaussian distribution from which noise values are drawn. + + + + + + + + diff --git a/sdf/1.11/rfid.sdf b/sdf/1.11/rfid.sdf new file mode 100644 index 000000000..61351dd8a --- /dev/null +++ b/sdf/1.11/rfid.sdf @@ -0,0 +1,2 @@ + + diff --git a/sdf/1.11/rfidtag.sdf b/sdf/1.11/rfidtag.sdf new file mode 100644 index 000000000..55699dc08 --- /dev/null +++ b/sdf/1.11/rfidtag.sdf @@ -0,0 +1,2 @@ + + diff --git a/sdf/1.11/road.sdf b/sdf/1.11/road.sdf new file mode 100644 index 000000000..172e48045 --- /dev/null +++ b/sdf/1.11/road.sdf @@ -0,0 +1,16 @@ + + + + Name of the road + + + + Width of the road + + + + A series of points that define the path of the road. + + + + diff --git a/sdf/1.11/root.sdf b/sdf/1.11/root.sdf new file mode 100644 index 000000000..94ecc8731 --- /dev/null +++ b/sdf/1.11/root.sdf @@ -0,0 +1,21 @@ + + SDFormat base element that can include one model, actor, light, or worlds. A user of multiple worlds could run parallel instances of simulation, or offer selection of a world at runtime. + + + + Version number of the SDFormat specification, consisting of major + and minor versions delimited by a `.` character. + A major version bump is required if older versions cannot be + automatically converted to this version. + A minor version bump is required when there are breaking changes that + can be handled by the automatic conversion functionality encoded in the + `*.convert` files. + + + + + + + + + diff --git a/sdf/1.11/scene.sdf b/sdf/1.11/scene.sdf new file mode 100644 index 000000000..1428d2c92 --- /dev/null +++ b/sdf/1.11/scene.sdf @@ -0,0 +1,86 @@ + + + Specifies the look of the environment. + + + Color of the ambient light. + + + + Color of the background. + + + + Properties for the sky + + Time of day [0..24] + + + Sunrise time [0..24] + + + Sunset time [0..24] + + + + Information about clouds in the sky. + + Speed of the clouds + + + + Direction of the cloud movement + + + Density of clouds + + + + Average size of the clouds + + + + Ambient cloud color + + + + + The URI to a cubemap texture for a skybox. A .dds file is typically used for the cubemap. + + + + + Enable/disable shadows + + + + Controls fog + + Fog color + + + Fog type: constant, linear, quadratic + + + Distance to start of fog + + + Distance to end of fog + + + Density of fog + + + + + Enable/disable the grid + + + + Show/hide world origin indicator + + + diff --git a/sdf/1.11/schema/types.xsd b/sdf/1.11/schema/types.xsd new file mode 100644 index 000000000..3d1215ace --- /dev/null +++ b/sdf/1.11/schema/types.xsd @@ -0,0 +1,46 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/sdf/1.11/sensor.sdf b/sdf/1.11/sensor.sdf new file mode 100644 index 000000000..78954738b --- /dev/null +++ b/sdf/1.11/sensor.sdf @@ -0,0 +1,82 @@ + + + The sensor tag describes the type and properties of a sensor. + + + A unique name for the sensor. This name must not match another model in the model. + + + + The type name of the sensor. By default, SDFormat supports types + air_pressure, + air_speed, + altimeter, + camera, + contact, + boundingbox_camera, boundingbox, + custom, + depth_camera, depth, + force_torque, + gps, + gpu_lidar, + gpu_ray, + imu, + lidar, + logical_camera, + magnetometer, + multicamera, + navsat, + ray, + rfid, + rfidtag, + rgbd_camera, rgbd, + segmentation_camera, segmentation, + sonar, + thermal_camera, thermal, + wireless_receiver, and + wireless_transmitter. + The "ray", "gpu_ray", and "gps" types are equivalent to "lidar", "gpu_lidar", and "navsat", respectively. It is preferred to use "lidar", "gpu_lidar", and "navsat" since "ray", "gpu_ray", and "gps" will be deprecated. The "ray", "gpu_ray", and "gps" types are maintained for legacy support. + + + + + If true the sensor will always be updated according to the update rate. + + + + The frequency at which the sensor data is generated. If left unspecified, the sensor will generate data every cycle. + + + + If true, the sensor is visualized in the GUI + + + + Name of the topic on which data is published. This is necessary for visualization + + + + If true, the sensor will publish performance metrics + + + + + + + + + + + + + + + + + + + + + + + diff --git a/sdf/1.11/sonar.sdf b/sdf/1.11/sonar.sdf new file mode 100644 index 000000000..0a048cda8 --- /dev/null +++ b/sdf/1.11/sonar.sdf @@ -0,0 +1,16 @@ + + These elements are specific to the sonar sensor. + + The sonar collision shape. Currently supported geometries are: "cone" and "sphere". + + + Minimum range + + + Max range + + + + Radius of the sonar cone at max range. This parameter is only used if geometry is "cone". + + diff --git a/sdf/1.11/sphere_shape.sdf b/sdf/1.11/sphere_shape.sdf new file mode 100644 index 000000000..73ad03e5d --- /dev/null +++ b/sdf/1.11/sphere_shape.sdf @@ -0,0 +1,6 @@ + + Sphere shape + + radius of the sphere + + diff --git a/sdf/1.11/spherical_coordinates.sdf b/sdf/1.11/spherical_coordinates.sdf new file mode 100644 index 000000000..978b15fbd --- /dev/null +++ b/sdf/1.11/spherical_coordinates.sdf @@ -0,0 +1,65 @@ + + + + Name of planetary surface model, used to determine the surface altitude + at a given latitude and longitude. The default is an ellipsoid model of + the earth based on the WGS-84 standard. It is used in Gazebo's GPS sensor + implementation. + + + + + + This field identifies how Gazebo world frame is aligned in Geographical + sense. The final Gazebo world frame orientation is obtained by rotating + a frame aligned with following notation by the field heading_deg. + Options are: + - ENU (East-North-Up) + + + + + Geodetic latitude at origin of gazebo reference frame, specified + in units of degrees. + + + + + + Longitude at origin of gazebo reference frame, specified in units + of degrees. + + + + + + Elevation of origin of gazebo reference frame, specified in meters. + + + + + + Equatorial axis of a custom surface type, specified in meters. + This is only required for custom surfaces. + + + + + + Polar axis of a custom surface type, specified in meters. + This is only required for custom surfaces. + + + + + + Heading offset of gazebo reference frame, measured as angle between + Gazebo world frame and the world_frame_orientation type. + The direction of rotation follows the right-hand rule, so a positive + angle indicates clockwise rotation (from east to north) when viewed from top-down. Note + that this is not consistent with compass heading convention. + The angle is specified in degrees. + + + + diff --git a/sdf/1.11/state.sdf b/sdf/1.11/state.sdf new file mode 100644 index 000000000..6a13e28bf --- /dev/null +++ b/sdf/1.11/state.sdf @@ -0,0 +1,41 @@ + + + + + Name of the world this state applies to + + + + Simulation time stamp of the state [seconds nanoseconds] + + + + Wall time stamp of the state [seconds nanoseconds] + + + + Real time stamp of the state [seconds nanoseconds] + + + + Number of simulation iterations. + + + + A list containing the entire description of entities inserted. + + + + + + A list of names of deleted entities/ + + The name of a deleted entity. + + + + + + + + diff --git a/sdf/1.11/surface.sdf b/sdf/1.11/surface.sdf new file mode 100644 index 000000000..a1631368d --- /dev/null +++ b/sdf/1.11/surface.sdf @@ -0,0 +1,245 @@ + + The surface parameters + + + + Bounciness coefficient of restitution, from [0...1], where 0=no bounciness. + + + Bounce capture velocity, below which effective coefficient of restitution is 0. + + + + + + + + Parameters for torsional friction + + + Torsional friction coefficient, unitless maximum ratio of + tangential stress to normal stress. + + + + + If this flag is true, + torsional friction is calculated using the "patch_radius" parameter. + If this flag is set to false, + "surface_radius" (R) and contact depth (d) + are used to compute the patch radius as sqrt(R*d). + + + + Radius of contact patch surface. + + + Surface radius on the point of contact. + + + Torsional friction parameters for ODE + + + Force dependent slip for torsional friction, + equivalent to inverse of viscous damping coefficient + with units of rad/s/(Nm). + A slip value of 0 is infinitely viscous. + + + + + + + ODE friction parameters + + + Coefficient of friction in first friction pyramid direction, + the unitless maximum ratio of force in first friction pyramid + direction to normal force. + + + + + Coefficient of friction in second friction pyramid direction, + the unitless maximum ratio of force in second friction pyramid + direction to normal force. + + + + + Unit vector specifying first friction pyramid direction in + collision-fixed reference frame. + If the friction pyramid model is in use, + and this value is set to a unit vector for one of the + colliding surfaces, + the ODE Collide callback function will align the friction pyramid directions + with a reference frame fixed to that collision surface. + If both surfaces have this value set to a vector of zeros, + the friction pyramid directions will be aligned with the world frame. + If this value is set for both surfaces, the behavior is undefined. + + + + + Force dependent slip in first friction pyramid direction, + equivalent to inverse of viscous damping coefficient + with units of m/s/N. + A slip value of 0 is infinitely viscous. + + + + + Force dependent slip in second friction pyramid direction, + equivalent to inverse of viscous damping coefficient + with units of m/s/N. + A slip value of 0 is infinitely viscous. + + + + + + + Coefficient of friction in first friction pyramid direction, + the unitless maximum ratio of force in first friction pyramid + direction to normal force. + + + + + Coefficient of friction in second friction pyramid direction, + the unitless maximum ratio of force in second friction pyramid + direction to normal force. + + + + + Unit vector specifying first friction pyramid direction in + collision-fixed reference frame. + If the friction pyramid model is in use, + and this value is set to a unit vector for one of the + colliding surfaces, + the friction pyramid directions will be aligned + with a reference frame fixed to that collision surface. + If both surfaces have this value set to a vector of zeros, + the friction pyramid directions will be aligned with the world frame. + If this value is set for both surfaces, the behavior is undefined. + + + + Coefficient of rolling friction + + + + + + + + Flag to disable contact force generation, while still allowing collision checks and contact visualization to occur. + + + Bitmask for collision filtering when collide_without_contact is on + + + + Bitmask for collision filtering. This will override collide_without_contact. Parsed as 16-bit unsigned integer. + + + + + + + + + Poisson's ratio is the unitless ratio between transverse and axial strain. + This value must lie between (-1, 0.5). Defaults to 0.3 for typical steel. + Note typical silicone elastomers have Poisson's ratio near 0.49 ~ 0.50. + + For reference, approximate values for Material:(Young's Modulus, Poisson's Ratio) + for some of the typical materials are: + Plastic: (1e8 ~ 3e9 Pa, 0.35 ~ 0.41), + Wood: (4e9 ~ 1e10 Pa, 0.22 ~ 0.50), + Aluminum: (7e10 Pa, 0.32 ~ 0.35), + Steel: (2e11 Pa, 0.26 ~ 0.31). + + + + + Young's Modulus in SI derived unit Pascal. + Defaults to -1. If value is less or equal to zero, + contact using elastic modulus (with Poisson's Ratio) is disabled. + + For reference, approximate values for Material:(Young's Modulus, Poisson's Ratio) + for some of the typical materials are: + Plastic: (1e8 ~ 3e9 Pa, 0.35 ~ 0.41), + Wood: (4e9 ~ 1e10 Pa, 0.22 ~ 0.50), + Aluminum: (7e10 Pa, 0.32 ~ 0.35), + Steel: (2e11 Pa, 0.26 ~ 0.31). + + + + + ODE contact parameters + + Soft constraint force mixing. + + + Soft error reduction parameter + + + dynamically "stiffness"-equivalent coefficient for contact joints + + + dynamically "damping"-equivalent coefficient for contact joints + + + maximum contact correction velocity truncation term. + + + minimum allowable depth before contact correction impulse is applied + + + + Bullet contact parameters + + Soft constraint force mixing. + + + Soft error reduction parameter + + + dynamically "stiffness"-equivalent coefficient for contact joints + + + dynamically "damping"-equivalent coefficient for contact joints + + + Similar to ODE's max_vel implementation. See http://bulletphysics.org/mediawiki-1.5.8/index.php/BtContactSolverInfo#Split_Impulse for more information. + + + Similar to ODE's max_vel implementation. See http://bulletphysics.org/mediawiki-1.5.8/index.php/BtContactSolverInfo#Split_Impulse for more information. + + + + + + + + soft contact pamameters based on paper: + http://www.cc.gatech.edu/graphics/projects/Sumit/homepage/papers/sigasia11/jain_softcontacts_siga11.pdf + + + This is variable k_v in the soft contacts paper. Its unit is N/m. + + + This is variable k_e in the soft contacts paper. Its unit is N/m. + + + Viscous damping of point velocity in body frame. Its unit is N/m/s. + + + Fraction of mass to be distributed among deformable nodes. + + + + + diff --git a/sdf/1.11/transceiver.sdf b/sdf/1.11/transceiver.sdf new file mode 100644 index 000000000..9f05b069d --- /dev/null +++ b/sdf/1.11/transceiver.sdf @@ -0,0 +1,34 @@ + + These elements are specific to a wireless transceiver. + + + Service set identifier (network name) + + + + Specifies the frequency of transmission in MHz + + + + Only a frequency range is filtered. Here we set the lower bound (MHz). + + + + + Only a frequency range is filtered. Here we set the upper bound (MHz). + + + + + Specifies the antenna gain in dBi + + + + Specifies the transmission power in dBm + + + + Mininum received signal power in dBm + + + diff --git a/sdf/1.11/visual.sdf b/sdf/1.11/visual.sdf new file mode 100644 index 000000000..b07f45025 --- /dev/null +++ b/sdf/1.11/visual.sdf @@ -0,0 +1,38 @@ + + + The visual properties of the link. This element specifies the shape of the object (box, cylinder, etc.) for visualization purposes. + + + Unique name for the visual element within the scope of the parent link. + + + + If true the visual will cast shadows. + + + + will be implemented in the future release. + + + + The amount of transparency( 0=opaque, 1 = fully transparent) + + + + + + + + Optional meta information for the visual. The information contained within this element should be used to provide additional feedback to an end user. + + + The layer in which this visual is displayed. The layer number is useful for programs, such as Gazebo, that put visuals in different layers for enhanced visualization. + + + + + + + + + diff --git a/sdf/1.11/world.sdf b/sdf/1.11/world.sdf new file mode 100644 index 000000000..6235e193b --- /dev/null +++ b/sdf/1.11/world.sdf @@ -0,0 +1,75 @@ + + The world element encapsulates an entire world description including: models, scene, physics, and plugins. + + + Unique name of the world + + + + Global audio properties. + + + Device to use for audio playback. A value of "default" will use the system's default audio device. Otherwise, specify a an audio device file" + + + + + The wind tag specifies the type and properties of the wind. + + + Linear velocity of the wind. + + + + + + Include resources from a URI. Included resources can only contain one 'model', 'light' or 'actor' element. The URI can point to a directory or a file. If the URI is a directory, it must conform to the model database structure (see /tutorials?tut=composition&cat=specification&#defining-models-in-separate-files). + + + Merge the included model into the top model. Only elements valid in 'world' are allowed in the included model + + + URI to a resource, such as a model + + + + Override the name of the included entity. + + + + Override the static value of the included entity. + + + + + + + The frame inside the included entity whose pose will be set by the specified pose element. If this element is specified, the pose must be specified. + + + + + The gravity vector in m/s^2, expressed in a coordinate frame defined by the spherical_coordinates tag. + + + + The magnetic vector in Tesla, expressed in a coordinate frame defined by the spherical_coordinates tag. + + + + + + + + + + + + + + + + + + + diff --git a/sdf/1.5/CMakeLists.txt b/sdf/1.5/CMakeLists.txt index be630c5c4..68ceba50f 100644 --- a/sdf/1.5/CMakeLists.txt +++ b/sdf/1.5/CMakeLists.txt @@ -67,8 +67,8 @@ foreach(FIL ${sdfs}) add_custom_command( OUTPUT "${CMAKE_CURRENT_BINARY_DIR}/${FIL_WE}.xsd" - COMMAND ${RUBY} ${CMAKE_SOURCE_DIR}/tools/xmlschema.rb - ARGS -s ${CMAKE_CURRENT_SOURCE_DIR} -i ${ABS_FIL} -o ${CMAKE_CURRENT_BINARY_DIR} + COMMAND ${Pyhton3_EXECUTABLE} ${CMAKE_SOURCE_DIR}/tools/xmlschema.py + ARGS --sdf-dir ${CMAKE_CURRENT_SOURCE_DIR} --input-file ${ABS_FIL} --output-dir ${CMAKE_CURRENT_BINARY_DIR} DEPENDS ${ABS_FIL} COMMENT "Running xml schema compiler on ${FIL}" VERBATIM) diff --git a/sdf/1.6/CMakeLists.txt b/sdf/1.6/CMakeLists.txt index 0ed72231d..ed41bf3b1 100644 --- a/sdf/1.6/CMakeLists.txt +++ b/sdf/1.6/CMakeLists.txt @@ -71,8 +71,8 @@ foreach(FIL ${sdfs}) add_custom_command( OUTPUT "${CMAKE_CURRENT_BINARY_DIR}/${FIL_WE}.xsd" - COMMAND ${RUBY} ${CMAKE_SOURCE_DIR}/tools/xmlschema.rb - ARGS -s ${CMAKE_CURRENT_SOURCE_DIR} -i ${ABS_FIL} -o ${CMAKE_CURRENT_BINARY_DIR} + COMMAND ${Python3_EXECUTABLE} ${CMAKE_SOURCE_DIR}/tools/xmlschema.py + ARGS --sdf-dir ${CMAKE_CURRENT_SOURCE_DIR} --input-file ${ABS_FIL} --output-dir ${CMAKE_CURRENT_BINARY_DIR} DEPENDS ${ABS_FIL} COMMENT "Running xml schema compiler on ${FIL}" VERBATIM) diff --git a/sdf/1.7/CMakeLists.txt b/sdf/1.7/CMakeLists.txt index ce971c310..51e6fe9af 100644 --- a/sdf/1.7/CMakeLists.txt +++ b/sdf/1.7/CMakeLists.txt @@ -72,8 +72,8 @@ foreach(FIL ${sdfs}) add_custom_command( OUTPUT "${CMAKE_CURRENT_BINARY_DIR}/${FIL_WE}.xsd" - COMMAND ${RUBY} ${CMAKE_SOURCE_DIR}/tools/xmlschema.rb - ARGS -s ${CMAKE_CURRENT_SOURCE_DIR} -i ${ABS_FIL} -o ${CMAKE_CURRENT_BINARY_DIR} + COMMAND ${Python3_EXECUTABLE} ${CMAKE_SOURCE_DIR}/tools/xmlschema.py + ARGS --sdf-dir ${CMAKE_CURRENT_SOURCE_DIR} --input-file ${ABS_FIL} --output-dir ${CMAKE_CURRENT_BINARY_DIR} DEPENDS ${ABS_FIL} COMMENT "Running xml schema compiler on ${FIL}" VERBATIM) diff --git a/sdf/1.8/CMakeLists.txt b/sdf/1.8/CMakeLists.txt index 162b2941a..ed7f874df 100644 --- a/sdf/1.8/CMakeLists.txt +++ b/sdf/1.8/CMakeLists.txt @@ -74,8 +74,8 @@ foreach(FIL ${sdfs}) add_custom_command( OUTPUT "${CMAKE_CURRENT_BINARY_DIR}/${FIL_WE}.xsd" - COMMAND ${RUBY} ${CMAKE_SOURCE_DIR}/tools/xmlschema.rb - ARGS -s ${CMAKE_CURRENT_SOURCE_DIR} -i ${ABS_FIL} -o ${CMAKE_CURRENT_BINARY_DIR} + COMMAND ${Python3_EXECUTABLE} ${CMAKE_SOURCE_DIR}/tools/xmlschema.py + ARGS --sdf-dir ${CMAKE_CURRENT_SOURCE_DIR} --input-file ${ABS_FIL} --output-dir ${CMAKE_CURRENT_BINARY_DIR} DEPENDS ${ABS_FIL} COMMENT "Running xml schema compiler on ${FIL}" VERBATIM) diff --git a/sdf/1.9/CMakeLists.txt b/sdf/1.9/CMakeLists.txt index 28ae14907..193f82c23 100644 --- a/sdf/1.9/CMakeLists.txt +++ b/sdf/1.9/CMakeLists.txt @@ -74,8 +74,8 @@ foreach(FIL ${sdfs}) add_custom_command( OUTPUT "${CMAKE_CURRENT_BINARY_DIR}/${FIL_WE}.xsd" - COMMAND ${RUBY} ${CMAKE_SOURCE_DIR}/tools/xmlschema.rb - ARGS -s ${CMAKE_CURRENT_SOURCE_DIR} -i ${ABS_FIL} -o ${CMAKE_CURRENT_BINARY_DIR} + COMMAND ${Python3_EXECUTABLE} ${CMAKE_SOURCE_DIR}/tools/xmlschema.py + ARGS --sdf-dir ${CMAKE_CURRENT_SOURCE_DIR} --input-file ${ABS_FIL} --output-dir ${CMAKE_CURRENT_BINARY_DIR} DEPENDS ${ABS_FIL} COMMENT "Running xml schema compiler on ${FIL}" VERBATIM) diff --git a/sdf/CMakeLists.txt b/sdf/CMakeLists.txt index 983e7804b..90509ffa4 100644 --- a/sdf/CMakeLists.txt +++ b/sdf/CMakeLists.txt @@ -8,32 +8,22 @@ add_subdirectory(1.7) add_subdirectory(1.8) add_subdirectory(1.9) add_subdirectory(1.10) +add_subdirectory(1.11) add_custom_target(schema) -add_dependencies(schema schema1_10) +add_dependencies(schema schema1_11) -# Optionally use the python script (default in sdf14+) -if (EMBEDSDF_PY) - if (NOT Python3_Interpreter_FOUND) - gz_build_error("Python is required to generate the C++ file with the SDF content") - endif() +if (NOT Python3_Interpreter_FOUND) + gz_build_error("Python is required to generate the C++ file with the SDF content") +endif() # Generate the EmbeddedSdf.cc file, which contains all the supported SDF # descriptions in a map of strings. The parser.cc file uses EmbeddedSdf.hh. - execute_process( - COMMAND ${Python3_EXECUTABLE} ${CMAKE_SOURCE_DIR}/sdf/embedSdf.py - --output-file "${PROJECT_BINARY_DIR}/src/EmbeddedSdf.cc" - WORKING_DIRECTORY "${CMAKE_SOURCE_DIR}/sdf" - OUTPUT_FILE "${PROJECT_BINARY_DIR}/src/EmbeddedSdf.cc" - ) -else() - execute_process( - COMMAND ${RUBY} ${CMAKE_SOURCE_DIR}/sdf/embedSdf.rb - WORKING_DIRECTORY "${CMAKE_SOURCE_DIR}/sdf" - OUTPUT_FILE "${PROJECT_BINARY_DIR}/src/EmbeddedSdf.cc" - ) -endif() - +execute_process( + COMMAND ${Python3_EXECUTABLE} ${CMAKE_SOURCE_DIR}/sdf/embedSdf.py + --output-file "${PROJECT_BINARY_DIR}/src/EmbeddedSdf.cc" + WORKING_DIRECTORY "${CMAKE_SOURCE_DIR}/sdf" +) # Generate aggregated SDF description files for use by the sdformat.org # website. If the description files change, the generated full*.sdf files need @@ -41,7 +31,7 @@ endif() if (GZ_PROGRAM) # Update this list as new sdformat spec versions are added. - set(sdf_desc_versions 1.4 1.5 1.6 1.7 1.8 1.9 1.10) + set(sdf_desc_versions 1.4 1.5 1.6 1.7 1.8 1.9 1.10 1.11) set(description_targets) foreach(desc_ver ${sdf_desc_versions}) diff --git a/sdf/embedSdf.py b/sdf/embedSdf.py index 292958712..9764ee230 100644 --- a/sdf/embedSdf.py +++ b/sdf/embedSdf.py @@ -8,12 +8,13 @@ import argparse import inspect import sys -from pathlib import Path +from pathlib import Path, PurePosixPath # The list of supported SDF specification versions. This will let us drop # versions without removing the directories. SUPPORTED_SDF_VERSIONS = [ + "1.11", "1.10", "1.9", "1.8", @@ -27,7 +28,7 @@ # The list of supported SDF conversions. This list includes versions that # a user can convert an existing SDF version to. -SUPPORTED_SDF_CONVERSIONS = ["1.10", "1.9", "1.8", "1.7", "1.6", "1.5", "1.4", "1.3"] +SUPPORTED_SDF_CONVERSIONS = ["1.11", "1.10", "1.9", "1.8", "1.7", "1.6", "1.5", "1.4", "1.3"] # whitespace indentation for C++ code INDENTATION = " " @@ -164,7 +165,9 @@ def generate_map_content(paths: List[Path], relative_to: Optional[str] = None) - # Strip relative path if requested if relative_to is not None: path = path.relative_to(relative_to) - content.append(embed_sdf_content(str(path), file_content)) + # dir separator is hardcoded to '/' in C++ mapping + posix_path = PurePosixPath(path) + content.append(embed_sdf_content(str(posix_path), file_content)) return ",".join(content) @@ -200,4 +203,4 @@ def main(args=None) -> int: if __name__ == "__main__": - sys.exit(main()) + sys.exit(main()) \ No newline at end of file diff --git a/sdf/embedSdf.rb b/sdf/embedSdf.rb deleted file mode 100755 index 488ff698d..000000000 --- a/sdf/embedSdf.rb +++ /dev/null @@ -1,45 +0,0 @@ -#!/usr/bin/env ruby - -# The list of supported SDF specification versions. This will let us drop -# versions without removing the directories. -supportedSdfVersions = ['1.10', '1.9', '1.8', '1.7', '1.6', '1.5', '1.4', '1.3', '1.2'] - -# The list of supported SDF conversions. This list includes versions that -# a user can convert an existing SDF version to. -supportedSdfConversions = ['1.10', '1.9', '1.8', '1.7', '1.6', '1.5', '1.4', '1.3'] - -puts %q! -#include "EmbeddedSdf.hh" - -namespace sdf { -inline namespace SDF_VERSION_NAMESPACE { - -const std::map &GetEmbeddedSdf() { - static const std::map result{ -! - -# Stores the contents of the file in the map. -def embed(pathname) - puts "{\"#{pathname}\", R\"__sdf_literal__(" - infile = File.open(pathname) - puts infile.read - puts ")__sdf_literal__\"}," -end - -# Embed the supported *.sdf files. -supportedSdfVersions.each do |version| - Dir.glob("#{version}/*.sdf").sort.each { |file| embed(file) } -end - -# Embed the supported *.convert files. -supportedSdfConversions.each do |version| - Dir.glob("#{version}/*.convert").sort.each { |file| embed(file) } -end -puts %q! - }; - return result; -} - -} -} -! diff --git a/src/Camera.cc b/src/Camera.cc index c41850a0d..2f2ea7a2b 100644 --- a/src/Camera.cc +++ b/src/Camera.cc @@ -15,6 +15,8 @@ * */ #include +#include + #include "sdf/Camera.hh" #include "sdf/parser.hh" #include "Utils.hh" @@ -23,7 +25,7 @@ using namespace sdf; /// \brief String names for the pixel formats. /// \sa Image::PixelFormat. -static std::array kPixelFormatNames = +constexpr std::array kPixelFormatNames = { "UNKNOWN_PIXEL_FORMAT", "L_INT8", @@ -1146,9 +1148,9 @@ std::string Camera::ConvertPixelFormat(PixelFormatType _type) { unsigned int index = static_cast(_type); if (index < kPixelFormatNames.size()) - return kPixelFormatNames[static_cast(_type)]; + return std::string(kPixelFormatNames[static_cast(_type)]); - return kPixelFormatNames[0]; + return std::string(kPixelFormatNames[0]); } ///////////////////////////////////////////////// diff --git a/src/Filesystem.cc b/src/Filesystem.cc index 257103366..03544231c 100644 --- a/src/Filesystem.cc +++ b/src/Filesystem.cc @@ -90,7 +90,7 @@ class DirIter::Implementation #ifndef _WIN32 -static const char preferred_separator = '/'; +static const char kSdfFileSystemPreferredSeparator = '/'; ////////////////////////////////////////////////// bool exists(const std::string &_path) @@ -210,7 +210,7 @@ void DirIter::close_handle() #else // Windows -static const char preferred_separator = '\\'; +static const char kSdfFileSystemPreferredSeparator = '\\'; ////////////////////////////////////////////////// static bool not_found_error(int _errval) @@ -496,7 +496,7 @@ void DirIter::close_handle() ////////////////////////////////////////////////// const std::string separator(const std::string &_p) { - return _p + preferred_separator; + return _p + kSdfFileSystemPreferredSeparator; } ////////////////////////////////////////////////// @@ -509,7 +509,7 @@ std::string basename(const std::string &_path) for (size_t i = 0; i < _path.length(); ++i) { - if (_path[i] == preferred_separator) + if (_path[i] == kSdfFileSystemPreferredSeparator) { if (i == (_path.length() - 1)) { @@ -519,7 +519,7 @@ std::string basename(const std::string &_path) // basename is empty, we return just a "/". if (basename.size() == 0) { - basename.push_back(preferred_separator); + basename.push_back(kSdfFileSystemPreferredSeparator); } break; } @@ -556,7 +556,7 @@ DirIter::DirIter() : dataPtr(gz::utils::MakeUniqueImpl()) ////////////////////////////////////////////////// std::string DirIter::operator*() const { - return this->dataPtr->dirname + preferred_separator + + return this->dataPtr->dirname + kSdfFileSystemPreferredSeparator + this->dataPtr->current; } diff --git a/src/Model_TEST.cc b/src/Model_TEST.cc index 14e26bf5c..cb4325d76 100644 --- a/src/Model_TEST.cc +++ b/src/Model_TEST.cc @@ -445,7 +445,7 @@ TEST(DOMModel, Uri) std::string name = "my-model"; gz::math::Pose3d pose(1, 2, 3, 0.1, 0.2, 0.3); std::string uri = - "https://fuel.ignitionrobotics.org/1.0/openrobotics/models/my-model"; + "https://fuel.gazebosim.org/1.0/openrobotics/models/my-model"; model.SetName(name); model.SetRawPose(pose); diff --git a/src/ParticleEmitter.cc b/src/ParticleEmitter.cc index 347df7f4e..605b58812 100644 --- a/src/ParticleEmitter.cc +++ b/src/ParticleEmitter.cc @@ -14,10 +14,11 @@ * limitations under the License. * */ +#include #include #include #include -#include +#include #include "sdf/Error.hh" #include "sdf/parser.hh" @@ -31,8 +32,9 @@ using namespace sdf; /// Particle emitter type strings. These should match the data in -/// `enum class ParticleEmitterType` located in ParticleEmitter.hh. -const std::vector emitterTypeStrs = +/// `enum class ParticleEmitterType` located in ParticleEmitter.hh, and the size +/// template parameter should match the number of elements as well. +constexpr std::array kEmitterTypeStrs = { "point", "box", @@ -251,9 +253,9 @@ void ParticleEmitter::SetType(const ParticleEmitterType _type) ///////////////////////////////////////////////// bool ParticleEmitter::SetType(const std::string &_typeStr) { - for (size_t i = 0; i < emitterTypeStrs.size(); ++i) + for (size_t i = 0; i < kEmitterTypeStrs.size(); ++i) { - if (_typeStr == emitterTypeStrs[i]) + if (_typeStr == kEmitterTypeStrs[i]) { this->dataPtr->type = static_cast(i); return true; @@ -266,8 +268,8 @@ bool ParticleEmitter::SetType(const std::string &_typeStr) std::string ParticleEmitter::TypeStr() const { size_t index = static_cast(this->dataPtr->type); - if (index < emitterTypeStrs.size()) - return emitterTypeStrs[index]; + if (index < kEmitterTypeStrs.size()) + return std::string(kEmitterTypeStrs[index]); return "point"; } diff --git a/src/Root_TEST.cc b/src/Root_TEST.cc index fd169a7c3..f7a643743 100644 --- a/src/Root_TEST.cc +++ b/src/Root_TEST.cc @@ -543,7 +543,7 @@ TEST(DOMRoot, CopyConstructor) auto testFrame1 = [](const sdf::Root &_root) { - EXPECT_EQ("1.10", _root.Version()); + EXPECT_EQ("1.11", _root.Version()); const sdf::World *world = _root.WorldByIndex(0); ASSERT_NE(nullptr, world); diff --git a/src/Sensor.cc b/src/Sensor.cc index 6c6dde9d6..78ac7087b 100644 --- a/src/Sensor.cc +++ b/src/Sensor.cc @@ -14,8 +14,10 @@ * limitations under the License. * */ +#include #include #include +#include #include #include #include @@ -39,8 +41,9 @@ using namespace sdf; /// Sensor type strings. These should match the data in -/// `enum class SensorType` located in Sensor.hh. -const std::vector sensorTypeStrs = +/// `enum class SensorType` located in Sensor.hh, and the size template +/// parameter should match the number of elements as well. +constexpr std::array kSensorTypeStrs = { "none", "altimeter", @@ -528,9 +531,9 @@ void Sensor::SetType(const SensorType _type) ///////////////////////////////////////////////// bool Sensor::SetType(const std::string &_typeStr) { - for (size_t i = 0; i < sensorTypeStrs.size(); ++i) + for (size_t i = 0; i < kSensorTypeStrs.size(); ++i) { - if (_typeStr == sensorTypeStrs[i]) + if (_typeStr == kSensorTypeStrs[i]) { this->dataPtr->type = static_cast(i); return true; @@ -645,8 +648,8 @@ void Sensor::SetUpdateRate(double _hz) std::string Sensor::TypeStr() const { size_t index = static_cast(this->dataPtr->type); - if (index > 0 && index < sensorTypeStrs.size()) - return sensorTypeStrs[index]; + if (index > 0 && index < kSensorTypeStrs.size()) + return std::string(kSensorTypeStrs[index]); return "none"; } diff --git a/src/Types.cc b/src/Types.cc index 60604bffc..6e202b936 100644 --- a/src/Types.cc +++ b/src/Types.cc @@ -96,12 +96,12 @@ std::ostream &operator<<(std::ostream &_out, const sdf::Errors &_errs) std::pair SplitName( const std::string &_absoluteName) { - const auto pos = _absoluteName.rfind(kSdfScopeDelimiter); + const auto pos = _absoluteName.rfind(kScopeDelimiter); if (pos != std::string::npos) { const std::string first = _absoluteName.substr(0, pos); const std::string second = - _absoluteName.substr(pos + kSdfScopeDelimiter.size()); + _absoluteName.substr(pos + kScopeDelimiter.size()); return {first, second}; } return {"", _absoluteName}; @@ -109,20 +109,20 @@ std::pair SplitName( static bool EndsWithDelimiter(const std::string &_s) { - if (_s.size() < kSdfScopeDelimiter.size()) + if (_s.size() < kScopeDelimiter.size()) return false; - const size_t startPosition = _s.size() - kSdfScopeDelimiter.size(); + const size_t startPosition = _s.size() - kScopeDelimiter.size(); return _s.compare( - startPosition, kSdfScopeDelimiter.size(), kSdfScopeDelimiter) == 0; + startPosition, kScopeDelimiter.size(), kScopeDelimiter) == 0; } static bool StartsWithDelimiter(const std::string &_s) { - if (_s.size() < kSdfScopeDelimiter.size()) + if (_s.size() < kScopeDelimiter.size()) return false; - return _s.compare(0, kSdfScopeDelimiter.size(), kSdfScopeDelimiter) == 0; + return _s.compare(0, kScopeDelimiter.size(), kScopeDelimiter) == 0; } // Join a scope name prefix with a local name using the scope delimeter @@ -138,11 +138,19 @@ std::string JoinName( const bool localNameStartsWithDelimiter = StartsWithDelimiter(_localName); if (scopeNameEndsWithDelimiter && localNameStartsWithDelimiter) - return _scopeName + _localName.substr(kSdfScopeDelimiter.size()); + return _scopeName + _localName.substr(kScopeDelimiter.size()); else if (scopeNameEndsWithDelimiter || localNameStartsWithDelimiter) return _scopeName + _localName; else - return _scopeName + kSdfScopeDelimiter + _localName; + return _scopeName + std::string(kScopeDelimiter) + _localName; +} + +///////////////////////////////////////////////// +const std::string &internal::SdfScopeDelimiter() +{ + static const gz::utils::NeverDestroyed delimiter{ + kScopeDelimiter}; + return delimiter.Access(); } } } diff --git a/src/Types_TEST.cc b/src/Types_TEST.cc index 392dfb137..37b951739 100644 --- a/src/Types_TEST.cc +++ b/src/Types_TEST.cc @@ -120,6 +120,7 @@ TEST(Types, ErrorsOutputStream) EXPECT_EQ(expected, output.str()); } +///////////////////////////////////////////////// TEST(Types, SplitName) { { @@ -159,6 +160,7 @@ TEST(Types, SplitName) } } +///////////////////////////////////////////////// TEST(Types, JoinName) { { @@ -198,3 +200,11 @@ TEST(Types, JoinName) EXPECT_EQ(joinedName, ""); } } + +///////////////////////////////////////////////// +TEST(Types, ScopeDelimiters) +{ + GZ_UTILS_WARN_IGNORE__DEPRECATED_DECLARATION + EXPECT_EQ(sdf::kScopeDelimiter, sdf::kSdfScopeDelimiter); + GZ_UTILS_WARN_RESUME__DEPRECATED_DECLARATION +} diff --git a/src/Utils.cc b/src/Utils.cc index 6adbdc669..0097b0ed1 100644 --- a/src/Utils.cc +++ b/src/Utils.cc @@ -207,7 +207,7 @@ static std::optional computeAbsoluteName( std::advance(it, 1); for (; it != names.rend(); ++it) { - absoluteParentName.append(kSdfScopeDelimiter); + absoluteParentName.append(kScopeDelimiter); absoluteParentName.append(*it); } diff --git a/test/integration/geometry_dom.cc b/test/integration/geometry_dom.cc index b3079b1da..07d50d570 100644 --- a/test/integration/geometry_dom.cc +++ b/test/integration/geometry_dom.cc @@ -179,7 +179,7 @@ TEST(DOMGeometry, Shapes) EXPECT_EQ(sdf::GeometryType::MESH, meshCol->Geom()->Type()); const sdf::Mesh *meshColGeom = meshCol->Geom()->MeshShape(); ASSERT_NE(nullptr, meshColGeom); - EXPECT_EQ("https://fuel.ignitionrobotics.org/1.0/an_org/models/a_model/mesh/" + EXPECT_EQ("https://fuel.gazebosim.org/1.0/an_org/models/a_model/mesh/" "mesh.dae", meshColGeom->Uri()); EXPECT_TRUE(gz::math::Vector3d(0.1, 0.2, 0.3) == meshColGeom->Scale()); @@ -193,7 +193,7 @@ TEST(DOMGeometry, Shapes) EXPECT_EQ(sdf::GeometryType::MESH, meshVis->Geom()->Type()); const sdf::Mesh *meshVisGeom = meshVis->Geom()->MeshShape(); ASSERT_NE(nullptr, meshVisGeom); - EXPECT_EQ("https://fuel.ignitionrobotics.org/1.0/an_org/models/a_model/mesh" + EXPECT_EQ("https://fuel.gazebosim.org/1.0/an_org/models/a_model/mesh" "/mesh.dae", meshVisGeom->Uri()); EXPECT_TRUE(gz::math::Vector3d(1.2, 2.3, 3.4) == meshVisGeom->Scale()); @@ -207,7 +207,7 @@ TEST(DOMGeometry, Shapes) EXPECT_EQ(sdf::GeometryType::HEIGHTMAP, heightmapCol->Geom()->Type()); auto heightmapColGeom = heightmapCol->Geom()->HeightmapShape(); ASSERT_NE(nullptr, heightmapColGeom); - EXPECT_EQ("https://fuel.ignitionrobotics.org/1.0/an_org/models/a_model/" + EXPECT_EQ("https://fuel.gazebosim.org/1.0/an_org/models/a_model/" "materials/textures/heightmap.png", heightmapColGeom->Uri()); EXPECT_EQ(gz::math::Vector3d(500, 500, 100), heightmapColGeom->Size()); EXPECT_EQ(gz::math::Vector3d(1, 2, 3), heightmapColGeom->Position()); @@ -221,7 +221,7 @@ TEST(DOMGeometry, Shapes) EXPECT_EQ(sdf::GeometryType::HEIGHTMAP, heightmapVis->Geom()->Type()); auto heightmapVisGeom = heightmapVis->Geom()->HeightmapShape(); ASSERT_NE(nullptr, heightmapVisGeom); - EXPECT_EQ("https://fuel.ignitionrobotics.org/1.0/an_org/models/a_model/" + EXPECT_EQ("https://fuel.gazebosim.org/1.0/an_org/models/a_model/" "materials/textures/heightmap.png", heightmapVisGeom->Uri()); EXPECT_EQ(gz::math::Vector3d(500, 500, 100), heightmapVisGeom->Size()); EXPECT_EQ(gz::math::Vector3d(1, 2, 3), heightmapVisGeom->Position()); @@ -229,23 +229,23 @@ TEST(DOMGeometry, Shapes) EXPECT_EQ(2u, heightmapVisGeom->BlendCount()); auto texture0 = heightmapVisGeom->TextureByIndex(0u); - EXPECT_EQ("https://fuel.ignitionrobotics.org/1.0/an_org/models/a_model/" + EXPECT_EQ("https://fuel.gazebosim.org/1.0/an_org/models/a_model/" "materials/textures/diffuse0.png", texture0->Diffuse()); - EXPECT_EQ("https://fuel.ignitionrobotics.org/1.0/an_org/models/a_model/" + EXPECT_EQ("https://fuel.gazebosim.org/1.0/an_org/models/a_model/" "materials/textures/normal0.png", texture0->Normal()); EXPECT_DOUBLE_EQ(5.0, texture0->Size()); auto texture1 = heightmapVisGeom->TextureByIndex(1u); - EXPECT_EQ("https://fuel.ignitionrobotics.org/1.0/an_org/models/a_model/" + EXPECT_EQ("https://fuel.gazebosim.org/1.0/an_org/models/a_model/" "materials/textures/diffuse1.png", texture1->Diffuse()); - EXPECT_EQ("https://fuel.ignitionrobotics.org/1.0/an_org/models/a_model/" + EXPECT_EQ("https://fuel.gazebosim.org/1.0/an_org/models/a_model/" "materials/textures/normal1.png", texture1->Normal()); EXPECT_DOUBLE_EQ(10.0, texture1->Size()); auto texture2 = heightmapVisGeom->TextureByIndex(2u); - EXPECT_EQ("https://fuel.ignitionrobotics.org/1.0/an_org/models/a_model/" + EXPECT_EQ("https://fuel.gazebosim.org/1.0/an_org/models/a_model/" "materials/textures/diffuse2.png", texture2->Diffuse()); - EXPECT_EQ("https://fuel.ignitionrobotics.org/1.0/an_org/models/a_model/" + EXPECT_EQ("https://fuel.gazebosim.org/1.0/an_org/models/a_model/" "materials/textures/normal2.png", texture2->Normal()); EXPECT_DOUBLE_EQ(20.0, texture2->Size()); diff --git a/test/integration/resolve_uris.cc b/test/integration/resolve_uris.cc index c3498beb9..e7d2a67d2 100644 --- a/test/integration/resolve_uris.cc +++ b/test/integration/resolve_uris.cc @@ -33,10 +33,10 @@ constexpr const int kNumCallbacks = 13; constexpr const char* kMeshUri = - "https://fuel.ignitionrobotics.org/1.0/an_org/models/a_model/mesh/mesh.dae"; + "https://fuel.gazebosim.org/1.0/an_org/models/a_model/mesh/mesh.dae"; constexpr const char* kHeightmapUri = - "https://fuel.ignitionrobotics.org/1.0/an_org/models/a_model/materials/textures/heightmap.png"; // NOLINT + "https://fuel.gazebosim.org/1.0/an_org/models/a_model/materials/textures/heightmap.png"; // NOLINT constexpr const char* kCubemapUri = "dummyUri"; @@ -44,10 +44,10 @@ constexpr const char* kMaterialScriptUri = "file://media/materials/scripts/gazebo.material"; constexpr const char* kDiffuseUri = - "https://fuel.ignitionrobotics.org/1.0/an_org/models/a_model/materials/textures/diffuse0.png"; // NOLINT + "https://fuel.gazebosim.org/1.0/an_org/models/a_model/materials/textures/diffuse0.png"; // NOLINT constexpr const char* kNormalUri = - "https://fuel.ignitionrobotics.org/1.0/an_org/models/a_model/materials/textures/normal0.png"; // NOLINT + "https://fuel.gazebosim.org/1.0/an_org/models/a_model/materials/textures/normal0.png"; // NOLINT ///////////////////////////////////////////////// TEST(ResolveURIs, StoreResolvedDisabled) diff --git a/test/sdf/shapes.sdf b/test/sdf/shapes.sdf index 49d393314..1f5f4fa27 100644 --- a/test/sdf/shapes.sdf +++ b/test/sdf/shapes.sdf @@ -121,7 +121,7 @@ - https://fuel.ignitionrobotics.org/1.0/an_org/models/a_model/mesh/mesh.dae + https://fuel.gazebosim.org/1.0/an_org/models/a_model/mesh/mesh.dae my_submesh
true
@@ -134,7 +134,7 @@ - https://fuel.ignitionrobotics.org/1.0/an_org/models/a_model/mesh/mesh.dae + https://fuel.gazebosim.org/1.0/an_org/models/a_model/mesh/mesh.dae another_submesh
false
@@ -147,7 +147,7 @@ - https://fuel.ignitionrobotics.org/1.0/an_org/models/a_model/materials/textures/heightmap.png + https://fuel.gazebosim.org/1.0/an_org/models/a_model/materials/textures/heightmap.png 500 500 100 1 2 3 @@ -157,12 +157,12 @@ - https://fuel.ignitionrobotics.org/1.0/an_org/models/a_model/materials/textures/heightmap.png + https://fuel.gazebosim.org/1.0/an_org/models/a_model/materials/textures/heightmap.png 500 500 100 1 2 3 - https://fuel.ignitionrobotics.org/1.0/an_org/models/a_model/materials/textures/diffuse0.png - https://fuel.ignitionrobotics.org/1.0/an_org/models/a_model/materials/textures/normal0.png + https://fuel.gazebosim.org/1.0/an_org/models/a_model/materials/textures/diffuse0.png + https://fuel.gazebosim.org/1.0/an_org/models/a_model/materials/textures/normal0.png 5 @@ -170,8 +170,8 @@ 5 - https://fuel.ignitionrobotics.org/1.0/an_org/models/a_model/materials/textures/diffuse1.png - https://fuel.ignitionrobotics.org/1.0/an_org/models/a_model/materials/textures/normal1.png + https://fuel.gazebosim.org/1.0/an_org/models/a_model/materials/textures/diffuse1.png + https://fuel.gazebosim.org/1.0/an_org/models/a_model/materials/textures/normal1.png 10 @@ -179,8 +179,8 @@ 10 - https://fuel.ignitionrobotics.org/1.0/an_org/models/a_model/materials/textures/diffuse2.png - https://fuel.ignitionrobotics.org/1.0/an_org/models/a_model/materials/textures/normal2.png + https://fuel.gazebosim.org/1.0/an_org/models/a_model/materials/textures/diffuse2.png + https://fuel.gazebosim.org/1.0/an_org/models/a_model/materials/textures/normal2.png 20 diff --git a/test/sdf/shapes_world.sdf b/test/sdf/shapes_world.sdf index bfbcfcd18..5eb324490 100644 --- a/test/sdf/shapes_world.sdf +++ b/test/sdf/shapes_world.sdf @@ -123,7 +123,7 @@ - https://ignitionfuel.org/an_org/models/a_model/mesh/mesh.dae + https://fuel.gazebosim.org/an_org/models/a_model/mesh/mesh.dae my_submesh
true
@@ -136,7 +136,7 @@ - https://ignitionfuel.org/an_org/models/a_model/mesh/mesh.dae + https://fuel.gazebosim.org/an_org/models/a_model/mesh/mesh.dae another_submesh
false
diff --git a/tools/xmlschema.py b/tools/xmlschema.py new file mode 100755 index 000000000..b3518f9ec --- /dev/null +++ b/tools/xmlschema.py @@ -0,0 +1,316 @@ +#!/usr/bin/env python3 +# Copyright 2023 Open Source Robotics Foundation, Inc. +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. + +""" +Conversion script for SDF definitions to XML XSD files +""" + +from xml.etree import ElementTree + +import os + +from typing import List, Dict, Tuple, Optional + + +# Mapping between "type" values found in SDF files to the corresponding +# XSD standard datatypes as defined by https://www.w3.org/TR/xmlschema11-2/ +SDF_TYPES_TO_XSD_STD_TYPES = { + "bool": "boolean", + "char": "char", + "int": "int", + "double": "double", + "float": "float", + "string": "string", + "unsigned int": "unsignedInt", + "unsigned long": "unsignedLong", +} + +# Mapping between "required" values found in SDF files to the corresponding +# minOccurs and maxOccurs found in XSD +SDF_REQUIRED_TO_MIN_MAX_OCCURS: Dict[str, Tuple[str, str]] = { + "0": ("0", "1"), # Required: 0, (minOccurs: 0, maxOccurs: 1) + "1": ("1", "1"), # Required: 1, (minOccurs: 1, maxOccurs: 1) + "+": ("1", "unbounded"), # Required: +, (minOccurs: 1, maxOccurs: inf) + "*": ("0", "unbounded"), # Required: *, (minOccurs: 0, maxOccurs: inf) + "-1": ("0", "unbounded"), # Required: -1, (minOccurs: 0, maxOccurs: inf) +} + + +def indent_lines(lines: List[str], indent: int) -> List[str]: + """ + Indent a list of xml lines group of lines by a number (indent) of spaces + + """ + return [" " * indent + line for line in lines] + + +def get_attribute(element: ElementTree.Element, attrib: str) -> Optional[str]: + """ + Retrieve XML attribute from an element + """ + return element.attrib[attrib] if attrib in element.attrib else None + + +def is_std_type(sdf_type: str) -> bool: + """ + Check if sdf_type is a known XSD standard type. + Return true if the sdf_type is in the set of known types, false otherwise. + """ + return sdf_type in SDF_TYPES_TO_XSD_STD_TYPES + + +def xsd_type_string(sdf_type: str) -> Optional[str]: + """ + Check if xsd_type is a known XSD standard type. + If it is, return 'xsd:' + type, None otherwise. + """ + if is_std_type(sdf_type): + xsd_type = SDF_TYPES_TO_XSD_STD_TYPES[sdf_type] + return "xsd:" + xsd_type + return None + + +def print_documentation(element: ElementTree.Element) -> List[str]: + """ + Print the documentation associated with an element + """ + lines = [] + description = element.find("description") + if ( + description is not None + and description.text is not None + and len(description.text) + ): + lines.append("") + lines.append(" ") + lines.append(f" ") + lines.append(" ") + lines.append("") + return lines + + +def print_include(element: ElementTree.Element) -> List[str]: + """ + Print include tag information + """ + lines = [] + filename = get_attribute(element, "filename") + if filename is not None: + loc = "http://sdformat.org/schemas/" + loc += filename.replace(".sdf", ".xsd") + lines.append(f"") + return lines + + +def print_include_ref(element: ElementTree.Element, sdf_root_dir: str) -> List[str]: + """ + Print include tag reference information + """ + lines = [] + filename = get_attribute(element, "filename") + if filename is not None: + sdf_path = os.path.join(sdf_root_dir, filename) + if not os.path.exists(sdf_path): + raise RuntimeError(f"Attempted to include non-existent file: {sdf_path}") + + include_tree = ElementTree.parse(sdf_path) + root = include_tree.getroot() + include_element_name = root.attrib["name"] + lines.append(f"") + return lines + + +def print_plugin_element(element: ElementTree.Element) -> List[str]: + """ + Separate handling of the 'plugin' element + """ + lines = [] + # Short circuit for plugin.sdf copy_data element + if "copy_data" in element.attrib: + lines.append("") + lines.append( + " " + ) + lines.append("") + return lines + + +def print_element(element: ElementTree.Element) -> List[str]: + """ + Print a child element of the sdf definition + """ + lines = [] + + elem_name = get_attribute(element, "name") + elem_type = get_attribute(element, "type") + elem_reqd = get_attribute(element, "required") + + if elem_type and is_std_type(elem_type): + elem_type = xsd_type_string(elem_type) + + if not elem_reqd: + raise RuntimeError("Cannot process element missing 'required' attribute") + + min_occurs, max_occurs = SDF_REQUIRED_TO_MIN_MAX_OCCURS[elem_reqd] + lines.append(f"") + + if elem_type is None: + lines.append(f"") + lines.extend(indent_lines(print_documentation(element), 2)) + lines.append(" ") + lines.append(" ") + + for child_element in element.findall("element"): + lines.extend(indent_lines(print_element(child_element), 6)) + + lines.append(" ") + + for attribute in element.findall("attribute"): + lines.extend(indent_lines(print_attribute(attribute), 4)) + + lines.append(" ") + else: + lines.append(f"") + lines.extend(indent_lines(print_documentation(element), 2)) + + lines.append("") + lines.append("") + return lines + + +def print_attribute(element: ElementTree.Element) -> List[str]: + """ + Print an attribute of the sdf definition + """ + lines = [] + + elem_name = get_attribute(element, "name") + elem_type = get_attribute(element, "type") + elem_reqd = get_attribute(element, "required") + elem_default = get_attribute(element, "default") + + if elem_type and is_std_type(elem_type): + elem_type = xsd_type_string(elem_type) + + use = "" + default = "" + + if elem_reqd == "1": + use = "use='required'" + elif elem_reqd == "0": + use = "use='optional'" + if elem_default is not None: + default = f"default='{elem_default}'" + + lines.append( + f"" + ) + lines.extend(indent_lines(print_documentation(element), 2)) + lines.append("") + return lines + + +def print_xsd(element: ElementTree.Element, sdf_root_dir: str) -> List[str]: + """ + Print xsd for top level SDF element + """ + lines = [] + + elem_name = get_attribute(element, "name") + elem_type = get_attribute(element, "type") + + elements = element.findall("element") + attributes = element.findall("attribute") + includes = element.findall("include") + + lines.extend(print_documentation(element)) + lines.append( + "" + ) + + # Reference any includes in the SDF file + for include in includes: + lines.extend(print_include(include)) + + if len(elements) or len(attributes) or len(includes): + lines.append(f"") + lines.append(" ") + + if elem_name != "plugin" and (len(elements) or len(includes)): + lines.append(" ") + + for child_element in elements: + if "copy_data" in child_element.attrib: + element_lines = print_plugin_element(child_element) + lines.extend(indent_lines(element_lines, 4)) + else: + element_lines = print_element(child_element) + lines.extend(indent_lines(element_lines, 6)) + + for include_element in includes: + element_lines = print_include_ref(include_element, sdf_root_dir) + lines.extend(indent_lines(element_lines, 6)) + + if elem_name != "plugin" and (len(elements) or len(includes)): + lines.append(" ") + + for attribute_element in attributes: + lines.extend(indent_lines(print_attribute(attribute_element), 4)) + + lines.append(" ") + lines.append("") + else: + if elem_type and is_std_type(elem_type): + elem_type = f' type={xsd_type_string(elem_type)}' + else: + elem_type = "" + + lines.append(f"") + return lines + + +def process(input_file_sdf: str, sdf_dir: str) -> List[str]: + """ + Produce an XSD file from an input SDF file + """ + lines = [] + tree = ElementTree.parse(input_file_sdf) + root = tree.getroot() + lines.append("") + lines.append("") + lines.extend(indent_lines(print_xsd(root, sdf_dir), 2)) + lines.append("") + return lines + + +if __name__ == "__main__": + import argparse + + parser = argparse.ArgumentParser("xmlschema.py") + parser.add_argument("--input-file") + parser.add_argument("--sdf-dir") + parser.add_argument("--output-dir") + args = parser.parse_args() + + input_file = os.path.abspath(args.input_file) + output_lines = process(input_file, args.sdf_dir) + fname = os.path.splitext(os.path.basename(args.input_file))[0] + os.makedirs(args.output_dir, exist_ok=True) + + output_file = os.path.join(args.output_dir, f"{fname}.xsd") + + with open(output_file, "w", encoding="utf8") as f: + f.write("\n".join(output_lines)) + f.write("\n") diff --git a/tools/xmlschema.rb b/tools/xmlschema.rb deleted file mode 100755 index cadfa59fb..000000000 --- a/tools/xmlschema.rb +++ /dev/null @@ -1,302 +0,0 @@ -#!/usr/bin/env ruby - -require "rexml/document" -require "optparse" - -$path = nil - -################################################# -# \brief A not very elegant way to convert to schema types -def xsdType(_type) - if _type == "unsigned int" - return "unsignedInt" - elsif _type == "unsigned long" - return "unsignedLog" - elsif _type == "bool" - return "boolean" - else - return _type - end -end - -################################################# -def isStdType(_type) - return _type == "string" || _type == "int" || _type == "double" || - _type == "float" || _type == "bool" || _type == "char" || - _type == "unsigned int" -end - -################################################# -def printElem(_file, _spaces, _elem) - - # this currently short-circuits the plugin.sdf copy_data element. - if _elem.attributes["name"].nil? - _file.printf("%*s\n", _spaces-2, "") - _file.printf("%*s\n", _spaces, "") - _file.printf("%*s\n", _spaces-2, "") - return - end - - type = _elem.attributes["type"] - if isStdType(type) - type = "xsd:" + xsdType(type) - end - - minOccurs = '0' - maxOccurs = 'unbounded' - if _elem.attributes["required"] == '0' - minOccurs='0' - maxOccurs='1' - elsif _elem.attributes["required"] == '1' - minOccurs='1' - maxOccurs='1' - elsif _elem.attributes["required"] == '+' - minOccurs='1' - maxOccurs='unbounded' - elsif _elem.attributes["required"] == '*' - minOccurs='0' - maxOccurs='unbounded' - end - - _file.printf("%*s\n", - _spaces, "", minOccurs, maxOccurs) - - # Print the complex type with a name - if type.nil? || type == "" - - _file.printf("%*s\n", - _spaces, "", _elem.attributes["name"]) - - if !_elem.elements["description"].nil? && - !_elem.elements["description"].text.nil? - printDocumentation(_file, _spaces+2, _elem.elements["description"].text) - end - - _file.printf("%*s\n", _spaces+2, "") - - _file.printf("%*s\n", _spaces+4, "") - - _elem.get_elements("element").each do |elem| - printElem(_file, _spaces+6, elem) - end - _file.printf("%*s\n", _spaces+4, "") - - # Print the attributes for the complex type - # Attributes must go at the end of the complex type. - _elem.get_elements("attribute").each do |attr| - printAttribute(_file, _spaces+4, attr); - end - - _file.printf("%*s\n", _spaces+2, "") - else - _file.printf("%*s\n", - _spaces, "", _elem.attributes["name"], type) - - if !_elem.elements["description"].nil? && - !_elem.elements["description"].text.nil? - printDocumentation(_file, _spaces+2, _elem.elements["description"].text) - end - end - - _file.printf("%*s\n", _spaces, "") - _file.printf("%*s\n", _spaces, "") - -end - -################################################# -def printDocumentation(_file, _spaces, _doc) - _file.printf("%*s\n", _spaces, "") - - _spaces += 2 - _file.printf("%*s\n", _spaces, "") - - _spaces += 2 - _file.printf("%*s\n",_spaces, "", _doc); - _spaces -= 2 - - _file.printf("%*s\n", _spaces, "") - _spaces -= 2 - - _file.printf("%*s\n", _spaces, "") -end - -################################################# -def printIncludeRef(_file, _spaces, _inc) - path = File.join($path, _inc.attributes["filename"]) - doc = REXML::Document.new File.new(path) - incElemName = doc.root.attributes['name'] - _file.printf("%*s\n", _spaces, "", incElemName) -end - -################################################# -def printInclude(_file, _spaces, _attr) - loc = "http://sdformat.org/schemas/" - loc += _attr.attributes['filename'].sub("\.sdf","\.xsd") - _file.printf("%*s\n", _spaces, "", loc) -end - -################################################# -def printAttribute(_file, _spaces, _attr) - name = _attr.attributes["name"] - type = _attr.attributes["type"] - use = "" - default = "" - - if !_attr.attributes["required"].nil? - if _attr.attributes["required"] == "1" - use = "use='required'" - elsif _attr.attributes["required"] == "0" - use = "use='optional'" - - # Default is only valid if use is optional - if !_attr.attributes["default"].nil? - default="default='#{_attr.attributes["default"]}'" - end - end - end - - if isStdType(type) - type = "xsd:" + xsdType(type) - end - - _file.printf("%*s\n", _spaces, - "", name, type, use, default) - - if !_attr.elements["description"].nil? && - !_attr.elements["description"].text.nil? - printDocumentation(_file, _spaces+2, _attr.elements["description"].text) - end - _file.printf("%*s\n", _spaces, "") -end - -################################################# -# \brief Print the complete schema for an element into a file. -# \param[in] _file File pointer in which to print the schema. -# \param[in] _spaces Number of spaces to prepend to each line. -# \param[in] _elem The SDF element to convert to an xml schema. -def printXSD(_file, _spaces, _elem) - - if !_elem.elements["description"].nil? && - !_elem.elements["description"].text.nil? - printDocumentation(_file, _spaces, _elem.elements["description"].text) - end - - _file.printf("%*s\n", _spaces, "") - - # Print the inclues for the complex type - # The includes must appear first - _elem.get_elements("include").each do |inc| - printInclude(_file, _spaces, inc); - end - - if _elem.get_elements("element").size > 0 || - _elem.get_elements("attribute").size > 0 || - _elem.get_elements("include").size > 0 - - # Print the complex type with a name - _file.printf("%*s\n", _spaces, "", - _elem.attributes["name"]) - _file.printf("%*s\n", _spaces+2, "") - - if _elem.attributes['name'] != "plugin" && - (_elem.get_elements("element").size > 0 || - _elem.get_elements("include").size > 0) - _file.printf("%*s\n", _spaces+4, "") - end - - # Print all the child elements - _elem.get_elements("element").each do |elem| - printElem(_file, _spaces+6, elem); - end - - # Print all the included sdf's root elements - _elem.get_elements("include").each do |inc| - printIncludeRef(_file, _spaces+6, inc); - end - if _elem.attributes['name'] != "plugin" && - (_elem.get_elements("element").size > 0 || - _elem.get_elements("include").size > 0) - _file.printf("%*s\n", _spaces+4, "") - end - - # Print the attributes for the complex type - # Attributes must go at the end of the complex type. - _elem.get_elements("attribute").each do |attr| - printAttribute(_file, _spaces+4, attr); - end - - # Close the complex type - _file.printf("%*s\n", _spaces+2, "") - _file.printf("%*s\n", _spaces, "") - else - type = _elem.attributes["type"] - - if isStdType(type) - type = "xsd:" + type - end - - if !type.nil? - type = "type='" + type + "'" - end - - _file.printf("%*s\n", _spaces, "", - _elem.attributes["name"], type) - end -end - - -infile = nil -outdir = nil - -opt_parser = OptionParser.new do |o| - o.on("-i", "--in [path]", String, - "SDF file to compile") {|path| infile = path} - o.on("-o", "--out [path]", String, - "Output directory for source and header files") {|path| outdir = path} - o.on("-s", "--sdf [path]", String, - "Directory containing all the SDF files") {|path| $path = path} - o.on("-h", "--help", "Display this help message") do - puts opt_parser - exit - end -end -opt_parser.parse! - -if infile.nil? - puts "Missing option -i." - exit -elsif !File.exist?(infile) - puts "Input file[#{infile}] does not exist\n" - exit -end - -if $path.nil? - puts "Missing option -s." - exit -elsif !Dir.exist?($path) - puts "SDF source dir[#{$path}] does not exist\n" - exit -end - -if outdir.nil? - puts "Missing output directory, option -o." - exit -elsif !Dir.exist?(outdir) - Dir.mkdir(outdir) -end - -doc = REXML::Document.new File.new(infile) - -spaces = 2 -doc.elements.each_with_index("element") do |elem, i| - out_xsd = infile.split("/").last.sub("\.sdf","\.xsd") - file = File.open(File.join(outdir, out_xsd), "w") - - file.print("\n") - file.print("\n") - - printXSD(file, spaces, elem) - - file.print("\n") - file.close() -end