us.ihmc.scs2.session.tools.RobotModelLoader Maven / Gradle / Ivy
package us.ihmc.scs2.session.tools;
import gnu.trove.map.hash.TLongObjectHashMap;
import us.ihmc.euclid.referenceFrame.ReferenceFrame;
import us.ihmc.euclid.tools.EuclidHashCodeTools;
import us.ihmc.log.LogTools;
import us.ihmc.mecano.multiBodySystem.interfaces.OneDoFJointBasics;
import us.ihmc.mecano.multiBodySystem.interfaces.SixDoFJointBasics;
import us.ihmc.mecano.multiBodySystem.iterators.SubtreeStreams;
import us.ihmc.robotDataLogger.handshake.YoVariableHandshakeParser;
import us.ihmc.robotDataLogger.jointState.JointState;
import us.ihmc.robotDataLogger.jointState.OneDoFState;
import us.ihmc.robotDataLogger.jointState.SixDoFState;
import us.ihmc.scs2.definition.robot.RobotDefinition;
import us.ihmc.scs2.definition.robot.sdf.SDFTools;
import us.ihmc.scs2.definition.robot.sdf.items.SDFRoot;
import us.ihmc.scs2.definition.robot.urdf.URDFTools;
import us.ihmc.scs2.definition.robot.urdf.items.URDFModel;
import us.ihmc.scs2.simulation.robot.Robot;
import us.ihmc.scs2.simulation.robot.RobotInterface;
import us.ihmc.yoVariables.exceptions.NameCollisionException;
import us.ihmc.yoVariables.registry.YoRegistry;
import us.ihmc.yoVariables.variable.YoVariable;
import jakarta.xml.bind.JAXBException;
import java.io.ByteArrayInputStream;
import java.io.File;
import java.io.IOException;
import java.net.URL;
import java.net.URLClassLoader;
import java.nio.file.Files;
import java.nio.file.Path;
import java.nio.file.Paths;
import java.nio.file.StandardCopyOption;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.List;
import java.util.Map;
import java.util.function.Function;
import java.util.stream.Collectors;
import java.util.zip.ZipEntry;
import java.util.zip.ZipInputStream;
public class RobotModelLoader
{
private final static String resourceDirectoryLocation = System.getProperty("user.home") + File.separator + ".ihmc" + File.separator + "resources";
private static final TLongObjectHashMap cachedImportedModels = new TLongObjectHashMap<>();
public static Runnable setupRobotUpdater(RobotDefinition robotDefinition,
YoVariableHandshakeParser handshakeParser,
YoRegistry rootRegistry,
ReferenceFrame inertialFrame)
{
return setupRobotUpdater(new Robot(robotDefinition, inertialFrame), handshakeParser, rootRegistry);
}
public static Runnable setupRobotUpdater(RobotInterface robot, YoVariableHandshakeParser handshakeParser, YoRegistry rootRegistry)
{
if (robot == null)
return null;
Map jointNameToState = handshakeParser.getJointStates().stream().collect(Collectors.toMap(JointState::getName, Function.identity()));
List jointStateUpdaters = new ArrayList<>();
SubtreeStreams.fromChildren(OneDoFJointBasics.class, robot.getRootBody()).forEach(oneDoFJoint ->
{
OneDoFState jointState = (OneDoFState) jointNameToState.get(
oneDoFJoint.getName());
if (jointState != null)
{
jointStateUpdaters.add(() ->
{
oneDoFJoint.setQ(jointState.getQ());
oneDoFJoint.setQd(jointState.getQd());
});
}
});
SixDoFJointBasics floatingJoint = (SixDoFJointBasics) robot.getRootBody().getChildrenJoints().get(0);
SixDoFState jointState = (SixDoFState) jointNameToState.get(floatingJoint.getSuccessor().getName());
if (jointState != null)
{
jointStateUpdaters.add(() ->
{
floatingJoint.getJointPose().set(jointState.getTranslation(), jointState.getRotation());
floatingJoint.getJointTwist().set(jointState.getTwistAngularPart(), jointState.getTwistLinearPart());
});
}
try
{
rootRegistry.addChild(robot.getRegistry());
}
catch (NameCollisionException e)
{
YoRegistry logRobotRegistry = rootRegistry.getChild(robot.getName());
// If there's no overlap in variable names, we can just add the robot variables to the logRobotRegistry.
for (YoVariable variable : new ArrayList<>(robot.getRegistry().getVariables())) // Need to copy the list to avoid concurrent modification exception.
{
if (logRobotRegistry.getVariable(variable.getName()) == null)
logRobotRegistry.addVariable(variable);
else
throw new NameCollisionException("Cannot add robot to log registry, name collision with variable " + variable.getFullNameString());
}
}
return () -> jointStateUpdaters.forEach(updater -> updater.run());
}
public static RobotDefinition loadModel(String modelName, String[] resourceDirectories, byte[] model, byte[] resourceZip)
{
return loadModel(modelName, null, resourceDirectories, model, resourceZip);
}
public static RobotDefinition loadModel(String modelName, String modelType, String[] resourceDirectories, byte[] model, byte[] resourceZip)
{
if (model == null)
return null;
long modelHashCode = computeModelHashCode(modelName, resourceDirectories, model, resourceZip);
RobotDefinition robotDefinition = cachedImportedModels.get(modelHashCode);
if (robotDefinition != null)
return robotDefinition;
ClassLoader resourceClassLoader = unpackResources(modelName, resourceZip);
if (modelType == null)
{
try
{
return loadSDFModel(resourceDirectories, model, modelHashCode, resourceClassLoader);
}
catch (JAXBException e)
{
try
{
return loadURDFModel(resourceDirectories, model, modelHashCode, resourceClassLoader);
}
catch (JAXBException e1)
{
LogTools.error("Could not load model {} with SDF or URDF model loaders", modelName);
LogTools.error("Stack trace from SDF model loader:");
e.printStackTrace();
LogTools.error("Stack trace from URDF model loader:");
e1.printStackTrace();
return null;
}
}
}
else
{
String modelTypeLowerCase = modelType.toLowerCase();
if (modelTypeLowerCase.contains("sdf"))
{
try
{
return loadSDFModel(resourceDirectories, model, modelHashCode, resourceClassLoader);
}
catch (JAXBException e)
{
e.printStackTrace();
return null;
}
}
else if (modelTypeLowerCase.contains("urdf"))
{
try
{
return loadURDFModel(resourceDirectories, model, modelHashCode, resourceClassLoader);
}
catch (JAXBException e)
{
e.printStackTrace();
return null;
}
}
else
{
LogTools.error("Unhandled model type: {}", modelType);
return null;
}
}
}
private static RobotDefinition loadSDFModel(String[] resourceDirectories, byte[] model, long modelHashCode, ClassLoader resourceClassLoader)
throws JAXBException
{
SDFRoot sdfRoot = SDFTools.loadSDFRoot(new ByteArrayInputStream(model), Arrays.asList(resourceDirectories), resourceClassLoader);
RobotDefinition robotDefinition = SDFTools.toFloatingRobotDefinition(sdfRoot.getModels().get(0));
robotDefinition.setResourceClassLoader(resourceClassLoader);
cachedImportedModels.put(modelHashCode, robotDefinition);
return robotDefinition;
}
private static RobotDefinition loadURDFModel(String[] resourceDirectories, byte[] model, long modelHashCode, ClassLoader resourceClassLoader)
throws JAXBException
{
URDFModel urdfModel = URDFTools.loadURDFModel(new ByteArrayInputStream(model), Arrays.asList(resourceDirectories), resourceClassLoader);
RobotDefinition robotDefinition = URDFTools.toRobotDefinition(urdfModel);
robotDefinition.setResourceClassLoader(resourceClassLoader);
cachedImportedModels.put(modelHashCode, robotDefinition);
return robotDefinition;
}
private static ClassLoader unpackResources(String modelName, byte[] resourceZip)
{
// Some of the model importers do not immediately release the files, triggering GC seems to help ensuring that these files are released.
System.gc();
if (resourceZip == null)
return null;
Path resourceDirectory = Paths.get(resourceDirectoryLocation, modelName);
try
{
if (!Files.exists(resourceDirectory))
Files.createDirectories(resourceDirectory);
}
catch (IOException e)
{
throw new RuntimeException(e);
}
ByteArrayInputStream is = new ByteArrayInputStream(resourceZip);
ZipInputStream zip = new ZipInputStream(is);
ZipEntry ze = null;
try
{
while ((ze = zip.getNextEntry()) != null)
{
// No need to explicitly handle directories, they will be created when copying files.
if (ze.isDirectory())
continue;
Path target = resourceDirectory.resolve(ze.getName());
Files.deleteIfExists(target);
if (!Files.exists(target.getParent()))
Files.createDirectories(target.getParent());
Files.copy(zip, target, StandardCopyOption.REPLACE_EXISTING);
}
}
catch (IOException e)
{
System.err.println("SDFModelLoader: Cannot load model zip file. Not unpacking robot model.");
e.printStackTrace();
}
finally
{
try
{
zip.close();
}
catch (IOException e)
{
e.printStackTrace();
}
try
{
is.close();
}
catch (IOException e)
{
e.printStackTrace();
}
}
try
{
return new URLClassLoader(new URL[] {resourceDirectory.toUri().toURL()});
}
catch (IOException e)
{
throw new RuntimeException(e);
}
}
private static long computeModelHashCode(String modelName, String[] resourceDirectories, byte[] model, byte[] resourceZip)
{
long hash = modelName == null ? 1L : modelName.hashCode();
hash = EuclidHashCodeTools.combineHashCode(hash, Arrays.hashCode(resourceDirectories));
hash = EuclidHashCodeTools.combineHashCode(hash, Arrays.hashCode(model));
hash = EuclidHashCodeTools.combineHashCode(hash, Arrays.hashCode(resourceZip));
return hash;
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy