All Downloads are FREE. Search and download functionalities are using the official Maven repository.

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