2011-05-17 12:00:33 +01:00
|
|
|
#include <cstdlib>
|
2011-07-22 21:33:30 +01:00
|
|
|
|
|
|
|
#include <iostream>
|
2011-05-17 12:00:33 +01:00
|
|
|
#include <sstream>
|
|
|
|
|
|
|
|
#include "FactorGraph.h"
|
2011-12-12 15:29:51 +00:00
|
|
|
#include "VarElimSolver.h"
|
2012-04-05 23:00:48 +01:00
|
|
|
#include "BpSolver.h"
|
2011-12-12 15:29:51 +00:00
|
|
|
#include "CbpSolver.h"
|
|
|
|
|
2011-05-17 12:00:33 +01:00
|
|
|
using namespace std;
|
|
|
|
|
2012-04-21 17:14:19 +01:00
|
|
|
int readHorusFlags (int, const char* []);
|
|
|
|
void readFactorGraph (FactorGraph&, const char*);
|
|
|
|
VarIds readQueryAndEvidence (FactorGraph&, int, const char* [], int);
|
|
|
|
|
2012-04-10 15:00:18 +01:00
|
|
|
void runSolver (const FactorGraph&, const VarIds&);
|
2011-05-17 12:00:33 +01:00
|
|
|
|
2012-04-21 17:14:19 +01:00
|
|
|
const string USAGE = "usage: ./hcli [HORUS_FLAG=VALUE] \
|
|
|
|
NETWORK_FILE [VARIABLE | OBSERVED_VARIABLE=EVIDENCE] ..." ;
|
2011-05-17 12:00:33 +01:00
|
|
|
|
|
|
|
|
|
|
|
int
|
|
|
|
main (int argc, const char* argv[])
|
2011-12-12 15:29:51 +00:00
|
|
|
{
|
2012-04-10 20:43:08 +01:00
|
|
|
if (argc <= 1) {
|
|
|
|
cerr << "error: no graphical model specified" << endl;
|
|
|
|
cerr << USAGE << endl;
|
|
|
|
exit (0);
|
|
|
|
}
|
2012-04-21 17:14:19 +01:00
|
|
|
int idx = readHorusFlags (argc, argv);
|
2012-04-05 19:34:37 +01:00
|
|
|
FactorGraph fg;
|
2012-04-21 17:14:19 +01:00
|
|
|
readFactorGraph (fg, argv[idx]);
|
|
|
|
VarIds queryIds = readQueryAndEvidence (fg, argc, argv, idx + 1);
|
|
|
|
runSolver (fg, queryIds);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
int
|
|
|
|
readHorusFlags (int argc, const char* argv[])
|
|
|
|
{
|
|
|
|
int i = 1;
|
|
|
|
for (; i < argc; i++) {
|
|
|
|
const string& arg = argv[i];
|
|
|
|
size_t pos = arg.find ('=');
|
|
|
|
if (pos == std::string::npos) {
|
|
|
|
return i;
|
|
|
|
}
|
|
|
|
string leftArg = arg.substr (0, pos);
|
|
|
|
string rightArg = arg.substr (pos + 1);
|
|
|
|
if (leftArg.empty()) {
|
|
|
|
cerr << "error: missing left argument" << endl;
|
|
|
|
cerr << USAGE << endl;
|
|
|
|
exit (0);
|
|
|
|
}
|
|
|
|
if (rightArg.empty()) {
|
|
|
|
cerr << "error: missing right argument" << endl;
|
|
|
|
cerr << USAGE << endl;
|
|
|
|
exit (0);
|
|
|
|
}
|
|
|
|
Util::setHorusFlag (leftArg, rightArg);
|
|
|
|
}
|
|
|
|
return i + 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
readFactorGraph (FactorGraph& fg, const char* s)
|
|
|
|
{
|
|
|
|
string fileName (s);
|
|
|
|
string extension = fileName.substr (fileName.find_last_of ('.') + 1);
|
2012-04-05 19:34:37 +01:00
|
|
|
if (extension == "uai") {
|
2012-04-10 20:43:08 +01:00
|
|
|
fg.readFromUaiFormat (fileName.c_str());
|
2011-12-12 15:29:51 +00:00
|
|
|
} else if (extension == "fg") {
|
2012-04-10 20:43:08 +01:00
|
|
|
fg.readFromLibDaiFormat (fileName.c_str());
|
2011-05-17 12:00:33 +01:00
|
|
|
} else {
|
|
|
|
cerr << "error: the graphical model must be defined either " ;
|
2012-04-05 19:34:37 +01:00
|
|
|
cerr << "in a UAI or libDAI file" << endl;
|
2011-05-17 12:00:33 +01:00
|
|
|
exit (0);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
2012-04-21 17:14:19 +01:00
|
|
|
VarIds
|
|
|
|
readQueryAndEvidence (
|
|
|
|
FactorGraph& fg,
|
|
|
|
int argc,
|
|
|
|
const char* argv[],
|
|
|
|
int start)
|
2011-05-17 12:00:33 +01:00
|
|
|
{
|
2012-04-05 23:00:48 +01:00
|
|
|
VarIds queryIds;
|
2012-04-21 17:14:19 +01:00
|
|
|
for (int i = start; i < argc; i++) {
|
2011-07-22 21:33:30 +01:00
|
|
|
const string& arg = argv[i];
|
2011-05-17 12:00:33 +01:00
|
|
|
if (arg.find ('=') == std::string::npos) {
|
2012-04-21 17:14:19 +01:00
|
|
|
if (Util::isInteger (arg) == false) {
|
2011-05-17 12:00:33 +01:00
|
|
|
cerr << "error: `" << arg << "' " ;
|
2012-04-21 17:14:19 +01:00
|
|
|
cerr << "is not a variable id" ;
|
2011-05-17 12:00:33 +01:00
|
|
|
cerr << endl;
|
|
|
|
exit (0);
|
|
|
|
}
|
2012-04-21 17:14:19 +01:00
|
|
|
VarId vid = Util::stringToUnsigned (arg);
|
2012-04-05 23:00:48 +01:00
|
|
|
VarNode* queryVar = fg.getVarNode (vid);
|
2012-04-21 17:14:19 +01:00
|
|
|
if (queryVar == false) {
|
|
|
|
cerr << "error: unknow variable with id " ;
|
|
|
|
cerr << "`" << vid << "'" << endl;
|
2011-05-17 12:00:33 +01:00
|
|
|
exit (0);
|
|
|
|
}
|
2012-04-21 17:14:19 +01:00
|
|
|
queryIds.push_back (vid);
|
2011-05-17 12:00:33 +01:00
|
|
|
} else {
|
|
|
|
size_t pos = arg.find ('=');
|
2012-04-21 17:14:19 +01:00
|
|
|
string leftArg = arg.substr (0, pos);
|
|
|
|
string rightArg = arg.substr (pos + 1);
|
|
|
|
if (leftArg.empty()) {
|
2011-05-17 12:00:33 +01:00
|
|
|
cerr << "error: missing left argument" << endl;
|
|
|
|
cerr << USAGE << endl;
|
|
|
|
exit (0);
|
|
|
|
}
|
2012-04-21 17:14:19 +01:00
|
|
|
if (Util::isInteger (leftArg) == false) {
|
|
|
|
cerr << "error: `" << leftArg << "' " ;
|
|
|
|
cerr << "is not a variable id" << endl ;
|
|
|
|
exit (0);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
VarId vid = Util::stringToUnsigned (leftArg);
|
|
|
|
VarNode* observedVar = fg.getVarNode (vid);
|
|
|
|
if (observedVar == false) {
|
|
|
|
cerr << "error: unknow variable with id " ;
|
|
|
|
cerr << "`" << vid << "'" << endl;
|
|
|
|
exit (0);
|
|
|
|
}
|
|
|
|
if (rightArg.empty()) {
|
2011-05-17 12:00:33 +01:00
|
|
|
cerr << "error: missing right argument" << endl;
|
|
|
|
cerr << USAGE << endl;
|
|
|
|
exit (0);
|
|
|
|
}
|
2012-04-21 17:14:19 +01:00
|
|
|
if (Util::isInteger (rightArg) == false) {
|
|
|
|
cerr << "error: `" << rightArg << "' " ;
|
|
|
|
cerr << "is not a state index" << endl ;
|
2011-05-17 12:00:33 +01:00
|
|
|
exit (0);
|
|
|
|
}
|
2012-04-21 17:14:19 +01:00
|
|
|
unsigned stateIdx = Util::stringToUnsigned (rightArg);
|
|
|
|
if (observedVar->isValidState (stateIdx) == false) {
|
|
|
|
cerr << "error: `" << stateIdx << "' " ;
|
|
|
|
cerr << "is not a valid state index for variable with id " ;
|
|
|
|
cerr << "`" << vid << "'" << endl;
|
2011-05-17 12:00:33 +01:00
|
|
|
exit (0);
|
|
|
|
}
|
2012-04-21 17:14:19 +01:00
|
|
|
observedVar->setEvidence (stateIdx);
|
2011-05-17 12:00:33 +01:00
|
|
|
}
|
|
|
|
}
|
2012-04-21 17:14:19 +01:00
|
|
|
return queryIds;
|
2012-04-10 15:00:18 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
runSolver (const FactorGraph& fg, const VarIds& queryIds)
|
|
|
|
{
|
2011-12-12 15:29:51 +00:00
|
|
|
Solver* solver = 0;
|
2012-03-31 23:27:37 +01:00
|
|
|
switch (Globals::infAlgorithm) {
|
2011-12-12 15:29:51 +00:00
|
|
|
case InfAlgorithms::VE:
|
|
|
|
solver = new VarElimSolver (fg);
|
|
|
|
break;
|
2012-04-05 23:00:48 +01:00
|
|
|
case InfAlgorithms::BP:
|
|
|
|
solver = new BpSolver (fg);
|
2011-12-12 15:29:51 +00:00
|
|
|
break;
|
|
|
|
case InfAlgorithms::CBP:
|
|
|
|
solver = new CbpSolver (fg);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
assert (false);
|
|
|
|
}
|
2012-04-21 17:14:19 +01:00
|
|
|
solver->printSolverFlags();
|
|
|
|
cout << endl;
|
2012-04-05 23:00:48 +01:00
|
|
|
if (queryIds.size() == 0) {
|
2011-07-22 21:33:30 +01:00
|
|
|
solver->printAllPosterioris();
|
2011-05-17 12:00:33 +01:00
|
|
|
} else {
|
2012-04-10 15:00:18 +01:00
|
|
|
solver->printAnswer (queryIds);
|
2011-05-17 12:00:33 +01:00
|
|
|
}
|
2011-07-22 21:33:30 +01:00
|
|
|
delete solver;
|
2011-05-17 12:00:33 +01:00
|
|
|
}
|
|
|
|
|