Skip to content

Commit

Permalink
Merge pull request #250 from Naoki-Hiraoka/custom_limb-checkjoint
Browse files Browse the repository at this point in the history
enable to read custom limb name from .yaml file (fix for #249)
  • Loading branch information
k-okada authored Apr 16, 2023
2 parents 6749740 + 6cb9aa6 commit f327d15
Show file tree
Hide file tree
Showing 5 changed files with 304 additions and 39 deletions.
56 changes: 56 additions & 0 deletions euscollada/pr2_custom_limb.yaml
Original file line number Diff line number Diff line change
@@ -0,0 +1,56 @@
##
## - collada_joint_name : euslisp_joint_name (start with :)
##

torso:
- torso_lift_joint : torso-waist-z
larm:
- l_shoulder_pan_joint : larm-collar-y
- l_shoulder_lift_joint : larm-shoulder-p
- l_upper_arm_roll_joint : larm-shoulder-r
- l_elbow_flex_joint : larm-elbow-p
- l_forearm_roll_joint : larm-elbow-r
- l_wrist_flex_joint : larm-wrist-p
- l_wrist_roll_joint : larm-wrist-r
rarm:
- r_shoulder_pan_joint : rarm-collar-y
- r_shoulder_lift_joint : rarm-shoulder-p
- r_upper_arm_roll_joint : rarm-shoulder-r
- r_elbow_flex_joint : rarm-elbow-p
- r_forearm_roll_joint : rarm-elbow-r
- r_wrist_flex_joint : rarm-wrist-p
- r_wrist_roll_joint : rarm-wrist-r
head:
- head_pan_joint : head-neck-y
- head_tilt_joint : head-neck-p
rhand:
- r_gripper_r_finger_joint : rhand-r-finger
- r_gripper_r_finger_tip_joint : rhand-r-finger-tip
- r_gripper_l_finger_joint : rhand-l-finger
- r_gripper_l_finger_tip_joint : rhand-l-finger-tip
lhand:
- l_gripper_r_finger_joint : lhand-r-finger
- l_gripper_r_finger_tip_joint : lhand-r-finger-tip
- l_gripper_l_finger_joint : lhand-l-finger
- l_gripper_l_finger_tip_joint : lhand-l-finger-tip

##
## end-coords
##
larm-end-coords:
parent : l_gripper_tool_frame
rotate : [0, 1, 0, 00]
rarm-end-coords:
parent : r_gripper_tool_frame
rotate : [0, 1, 0, 00]
head-end-coords:
translate : [0.08, 0, 0.13]
rotate : [0, 1, 0, 90]

##
## reset-pose
##
angle-vector:
reset-manip-pose : [300.0, 75.0, 50.0, 110.0, -110.0, -20.0, -10.0, -10.0, -75.0, 50.0, -110.0, -110.0, 20.0, -10.0, -10.0, 0.0, 50.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0]
reset-pose : [50.0, 60.0, 74.0, 70.0, -120.0, 20.0, -30.0, 180.0, -60.0, 74.0, -70.0, -120.0, -20.0, -30.0, 180.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0]

141 changes: 107 additions & 34 deletions euscollada/src/collada2eus_urdfmodel.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -19,6 +19,7 @@
#include <algorithm>


#include <boost/algorithm/string/predicate.hpp>
#include <boost/program_options.hpp>
#include <boost/filesystem/path.hpp>
#include <boost/filesystem/operations.hpp>
Expand Down Expand Up @@ -219,7 +220,7 @@ class ModelEuslisp {
void addLinkCoords();
void printMesh(const aiScene* scene, const aiNode* node, const Vector3 &scale,
const string &material_name, vector<coordT> &store_pt, bool printq);
void readYaml(string &config_file);
void readYaml(std::vector<string> &config_file);

#if URDFDOM_1_0_0_API
Pose getLinkPose(LinkConstSharedPtr link) {
Expand Down Expand Up @@ -257,6 +258,7 @@ class ModelEuslisp {
void printSensors();
void printSensorLists();
void printGeometries();
void printUniqueLimbs();

// print methods
void copyRobotClassDefinition ();
Expand Down Expand Up @@ -589,36 +591,64 @@ void ModelEuslisp::printMesh(const aiScene* scene, const aiNode* node, const Vec
}

bool limb_order_asc(const pair<string, size_t>& left, const pair<string, size_t>& right) { return left.second < right.second; }
void ModelEuslisp::readYaml (string &config_file) {
void ModelEuslisp::readYaml (std::vector<string> &config_files) {
// read yaml
string limb_candidates[] = {"torso", "larm", "rarm", "lleg", "rleg", "head"}; // candidates of limb names
std::vector<string> limb_candidates; // limb names is given from yaml files

vector<pair<string, size_t> > limb_order;
#ifndef USE_CURRENT_YAML
ifstream fin(config_file.c_str());
if (fin.fail()) {
fprintf(stderr, "%c[31m;; Could not open %s%c[m\n", 0x1b, config_file.c_str(), 0x1b);
} else {
YAML::Parser parser(fin);
parser.GetNextDocument(doc);

doc = YAML::Node(YAML::NodeType::Map);
for(int i = config_files.size() - 1; i >= 0; i--) { // override values from later config_file with values from earlier config_file
YAML::Node tmp_doc;
#ifndef USE_CURRENT_YAML
ifstream fin(config_files[i].c_str());
if (fin.fail()) {
fprintf(stderr, "%c[31m;; Could not open %s%c[m\n", 0x1b, config_files[i].c_str(), 0x1b);
} else {
YAML::Parser parser(fin);
parser.GetNextDocument(tmp_doc);
for(YAML::const_iterator it=tmp_doc.begin();it != tmp_doc.end();++it) {
doc[it->first] = it->second;
}
}
#else
{
// yaml-cpp is greater than 0.5.0
doc = YAML::LoadFile(config_file.c_str());
tmp_doc = YAML::LoadFile(config_files[i].c_str());
for(YAML::const_iterator it=tmp_doc.begin();it != tmp_doc.end();++it) {
doc[it->first] = it->second;
}
#endif
}

{
// set limb_candidates from yaml files
for(YAML::const_iterator it=doc.begin();it != doc.end();++it) {
// -end-coords, *-vector, sensors
if ( boost::algorithm::ends_with(it->first.as<std::string>(), "-coords") ||
boost::algorithm::ends_with(it->first.as<std::string>(), "-vector") ||
it->first.as<std::string>() == "sensors") {
} else {
limb_candidates.push_back(it->first.as<std::string>());
}
}
/* re-order limb name by lines of yaml */
BOOST_FOREACH(string& limb, limb_candidates) {
#ifdef USE_CURRENT_YAML
if (doc[limb]) {
int line=0;
ifstream fin2(config_file.c_str()); // super ugry hack until yaml-cpp 0.5.2
string buffer;
for (;fin2;) {
getline(fin2, buffer); line++;
if(buffer == limb+":") break;
bool found = false;
for(int i = 0; i < config_files.size() && !found; i++) { // super ugry hack until yaml-cpp 0.5.2
ifstream fin2(config_files[i].c_str());
string buffer;
for (;fin2;) {
getline(fin2, buffer); line++;
if(buffer == limb+":") {
found = true;
break;
}
}
fin2.close();
}
fin2.close();
std::cerr << limb << "@" << line << std::endl;
limb_order.push_back(pair<string, size_t>(limb, line));
}
Expand All @@ -633,37 +663,55 @@ void ModelEuslisp::readYaml (string &config_file) {
}

// generate limbs including limb_name, link_names, and joint_names
for (size_t i = 0; i < limb_order.size(); i++) {
string limb_name = limb_order[i].first;
vector<string> tmp_link_names, tmp_joint_names;
for (size_t l = 0; l < limb_order.size(); l++) {
string limb_name = limb_order[l].first;
vector<string> tmp_link_names, tmp_joint_names, tmp_method_names;
try {
const YAML::Node& limb_doc = doc[limb_name];
for(unsigned int i = 0; i < limb_doc.size(); i++) {
const YAML::Node& n = limb_doc[i];
if(n.Type() != YAML::NodeType::Map) {
ROS_WARN_STREAM("invalid yaml representation. ignore limb " << limb_name);
goto nextlimb;
}
#ifdef USE_CURRENT_YAML
for(YAML::const_iterator it=n.begin();it!=n.end();it++) {
if(it->second.Type() != YAML::NodeType::Scalar) {
ROS_WARN_STREAM("invalid yaml representation. ignore limb " << limb_name);
goto nextlimb;
}
string key, value;
key = it->first.as<std::string>();
value = it->second.as<std::string>();
#else
for(YAML::Iterator it=n.begin();it!=n.end();it++) {
if(it->second.Type() != YAML::NodeType::Scalar) {
ROS_WARN_STREAM("invalid yaml representation. ignore limb " << limb_name);
goto nextlimb;
}
string key, value; it.first() >> key; it.second() >> value;
#endif
tmp_joint_names.push_back(key);
#if URDFDOM_1_0_0_API
JointConstSharedPtr jnt = robot->getJoint(key);
#else
boost::shared_ptr<const Joint> jnt = robot->getJoint(key);
#endif
if (!!jnt) {
tmp_joint_names.push_back(key);
tmp_method_names.push_back(value);
tmp_link_names.push_back(jnt->child_link_name);
} else {
// error
ROS_WARN_STREAM("joint " << key << " is not found. ignore limb " << limb_name);
goto nextlimb;
}
g_all_link_names.push_back(pair<string, string>(key, value));
}
}
for(unsigned int j = 0; j < limb_doc.size(); j++) {
g_all_link_names.push_back(pair<string, string>(tmp_joint_names[j], tmp_method_names[j]));
}
limbs.push_back(link_joint_pair(limb_name, link_joint(tmp_link_names, tmp_joint_names)));
nextlimb:
;
} catch(YAML::RepresentationException& e) {
}
}
Expand Down Expand Up @@ -742,6 +790,14 @@ void ModelEuslisp::printRobotDefinition() {
for (vector<daeSensor>::iterator it = m_sensors.begin(); it != m_sensors.end(); it++) {
fprintf(fp, " %s-sensor-coords", it->name.c_str());
}
fprintf(fp, "\n ;; non-default limb names\n");
fprintf(fp, " ");
BOOST_FOREACH(link_joint_pair& limb, limbs) {
if( limb.first == "torso" || limb.first == "larm" || limb.first == "rarm" || limb.first == "lleg" || limb.first == "rleg" || limb.first == "head" ) {
continue;
}
fprintf(fp, " %s %s-end-coords %s-root-link", limb.first.c_str(), limb.first.c_str(), limb.first.c_str());
}
fprintf(fp, "\n )\n )\n");
// TODO: add openrave manipulator tip frame
}
Expand All @@ -768,6 +824,8 @@ void ModelEuslisp::printRobotMethods() {

printSensors();

printUniqueLimbs();

printGeometries();

fprintf(fp, " )\n");
Expand Down Expand Up @@ -1582,6 +1640,19 @@ void ModelEuslisp::printSensorLists() {
fprintf(fp, "))\n");
}

void ModelEuslisp::printUniqueLimbs() {
fprintf(fp, "\n ;; non-default limbs\n");
BOOST_FOREACH(link_joint_pair& limb, limbs) {
if( limb.first == "torso" || limb.first == "larm" || limb.first == "rarm" || limb.first == "lleg" || limb.first == "rleg" || limb.first == "head" ) {
continue;
}
fprintf(fp, " (:%s (&rest args) (unless args (setq args (list nil))) (send* self :limb :%s args))\n", limb.first.c_str(), limb.first.c_str());
fprintf(fp, " (:%s-end-coords () %s-end-coords)\n", limb.first.c_str(), limb.first.c_str());
fprintf(fp, " (:%s-root-link () %s-root-link)\n", limb.first.c_str(), limb.first.c_str());
}
}


#if URDFDOM_1_0_0_API
void ModelEuslisp::printGeometry (GeometrySharedPtr g, const Pose &pose,
#else
Expand Down Expand Up @@ -1935,7 +2006,7 @@ int main(int argc, char** argv)
string arobot_name;

string input_file;
string config_file;
std::vector<string> config_files;
string pconfig_file; // backward compatibility (positional option)
string output_file;

Expand All @@ -1948,7 +2019,7 @@ int main(int argc, char** argv)
("use_collision,U", "use collision geometry (default collision is the same as visual)")
("robot_name,N", po::value< vector<string> >(), "output robot name")
("input_file,I", po::value< vector<string> >(), "input file")
("config_file,C", po::value< vector<string> >(), "configuration yaml file")
("config_file,C", po::value< vector<string> >()->multitoken(), "configuration yaml files")
("pconfig_file", po::value< vector<string> >(), "not used (used internally)")
("output_file,O", po::value< vector<string> >(), "output file")
;
Expand Down Expand Up @@ -1978,7 +2049,7 @@ int main(int argc, char** argv)
}
if (vm.count("config_file")) {
vector<string> aa = vm["config_file"].as< vector<string> >();
config_file = aa[0];
config_files = aa;
}
if (vm.count("pconfig_file")) {
vector<string> aa = vm["pconfig_file"].as< vector<string> >();
Expand Down Expand Up @@ -2055,12 +2126,14 @@ int main(int argc, char** argv)
pconfig_file.clear();
}
}
if (config_file.empty() && !pconfig_file.empty()) {
config_file = pconfig_file;
if (config_files.empty() && !pconfig_file.empty()) {
config_files.push_back(pconfig_file);
}
if (!config_file.empty()) {
cerr << ";; Config file is: "
<< config_file << endl;
if(!config_files.empty()) {
cerr << ";; Config file is:" << endl;;
for(int i = 0; i < config_files.size(); i++) {
cerr << ";; " << config_files[i] << endl;
}
}
cerr << ";; Output file is: "
<< output_file << endl;
Expand All @@ -2077,8 +2150,8 @@ int main(int argc, char** argv)
//eusmodel.setAddJointSuffix();
//eusmodel.setAddLinkSuffix();

if (!config_file.empty()) {
eusmodel.readYaml(config_file);
if (!config_files.empty()) {
eusmodel.readYaml(config_files);
}
eusmodel.writeToFile (output_file);
}
47 changes: 42 additions & 5 deletions euscollada/test/euscollada-model-conversion-test.l
Original file line number Diff line number Diff line change
Expand Up @@ -4,11 +4,18 @@

(defun generate-eusmodel-from-collada (col-file eus-file &key (yaml-file))
(let ((strm (piped-fork
(if yaml-file
(format nil "rosrun euscollada collada2eus ~A ~A ~A && echo finished"
col-file yaml-file eus-file)
(format nil "rosrun euscollada collada2eus ~A ~A && echo finished"
col-file eus-file)))))
(cond ((stringp yaml-file)
(format nil "rosrun euscollada collada2eus ~A ~A ~A && echo finished"
col-file yaml-file eus-file))
((null yaml-file)
(format nil "rosrun euscollada collada2eus ~A ~A && echo finished"
col-file eus-file))
(t ;; list of filename
(concatenate string
(format nil "rosrun euscollada collada2eus -I ~A -O ~A -C "
col-file eus-file)
(apply #'concatenate string (mapcar #'(lambda (f) (format nil "~A " f)) yaml-file))
"&& echo finished"))))))
;; (assert (select-stream (list strm) 240) (format nil ";; faild to generate ~A within 240 second" eus-file)) ;; why return nil immidiately
(dotimes (i 240)
(when
Expand Down Expand Up @@ -51,6 +58,18 @@
(deftest pr2-eusmodel-validity-check
(eusmodel-validity-check-one *pr2*))

(deftest generate-pr2-custom-limb-test
(generate-eusmodel-from-collada "/tmp/pr2.dae" "/tmp/pr2_custom_limb.l"
:yaml-file "$(rospack find euscollada)/pr2_custom_limb.yaml")
(assert (probe-file "/tmp/pr2_custom_limb.l") ";; faild to generate /tmp/pr2_custom_limb.l, file does not exist")
(load "/tmp/pr2_custom_limb.l")
(pr2)
(eusmodel-validity-check-one *pr2*)
(send *pr2* :reset-pose)
(assert (equal (send-all (send *pr2* :rhand :joint-list) :name) (list "r_gripper_r_finger_joint" "r_gripper_r_finger_tip_joint" "r_gripper_l_finger_joint" "r_gripper_l_finger_tip_joint")))
(assert (subsetp (send *pr2* :rhand :joint-list) (send *pr2* :joint-list) :test #'equal))
)

(deftest generate-sample-robot-model-test
(when (ros::rospack-find "openhrp3")
(generate-eusmodel-from-collada
Expand Down Expand Up @@ -89,5 +108,23 @@
";; sample robot has not valid centroid ~A" (send *samplerobot* :centroid))
))

(deftest generate-sample-robot-multi-yaml-test
(when (ros::rospack-find "openhrp3")
(generate-eusmodel-from-collada
"$(rospack find openhrp3)/share/OpenHRP-3.1/sample/model/sample1.dae"
"/tmp/samplerobot_multi_yaml.l" :yaml-file (list "$(rospack find euscollada)/test/samplerobot1.yaml" "$(rospack find euscollada)/test/samplerobot2.yaml"))

(assert (probe-file "/tmp/samplerobot_multi_yaml.l") ";; faild to generate /tmp/samplerobot_multi_yaml.l, file does not exist")
(load "/tmp/samplerobot_multi_yaml.l")
(samplerobot)
(send *samplerobot* :init-pose)
(eusmodel-validity-check-one *samplerobot*)
(send *samplerobot* :fix-leg-to-coords (make-coords))
(assert (eps= 1.308000e+05 (send *samplerobot* :weight))
";; sample robot has not valid weight ~A" (send *samplerobot* :weight))
(assert (eps-v= #f(2.32416 0.0 791.942) (send *samplerobot* :centroid))
";; sample robot has not valid centroid ~A" (send *samplerobot* :centroid))
))

(run-all-tests)
(exit)
Loading

0 comments on commit f327d15

Please sign in to comment.