blob: b5c016130bbaba39d966ebddb5f21a82abe5cb7d [file] [log] [blame]
/** NDN-Atmos: Cataloging Service for distributed data originally developed
* for atmospheric science data
* Copyright (C) 2015 Colorado State University
*
* NDN-Atmos is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* NDN-Atmos is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with NDN-Atmos. If not, see <http://www.gnu.org/licenses/>.
**/
#include "query/query-adapter.hpp"
#include "boost-test.hpp"
#include "../../unit-test-time-fixture.hpp"
#include "util/config-file.hpp"
#include <boost/mpl/list.hpp>
#include <boost/thread.hpp>
#include <ndn-cxx/util/dummy-client-face.hpp>
#include <boost/property_tree/info_parser.hpp>
namespace atmos{
namespace tests{
using ndn::util::DummyClientFace;
using ndn::util::makeDummyClientFace;
class QueryAdapterTest : public query::QueryAdapter<std::string>
{
public:
QueryAdapterTest(const std::shared_ptr<ndn::util::DummyClientFace>& face,
const std::shared_ptr<ndn::KeyChain>& keyChain,
const std::shared_ptr<chronosync::Socket>& syncSocket)
: query::QueryAdapter<std::string>(face, keyChain, syncSocket)
{
}
virtual
~QueryAdapterTest()
{
}
void setDatabaseTable(const std::string& databaseTable)
{
m_databaseTable.assign(databaseTable);
}
void setNameFields(const std::vector<std::string>& nameFields)
{
m_nameFields = nameFields;
}
void setPrefix(const ndn::Name& prefix)
{
m_prefix = prefix;
}
void setSigningId(const ndn::Name& signingId)
{
m_signingId = signingId;
}
const ndn::Name
getPrefix()
{
return m_prefix;
}
const ndn::Name
getSigningId()
{
return m_signingId;
}
std::shared_ptr<ndn::Data>
getAckData(std::shared_ptr<const ndn::Interest> interest, const ndn::Name::Component& version)
{
return makeAckData(interest, version);
}
std::shared_ptr<ndn::Data>
getReplyData(const ndn::Name& segmentPrefix,
const Json::Value& value,
uint64_t segmentNo,
bool isFinalBlock,
bool isAutocomplete,
uint64_t resultCount,
uint64_t viewStart,
uint64_t viewEnd)
{
return makeReplyData(segmentPrefix, value, segmentNo, isFinalBlock,
isAutocomplete, resultCount, viewStart, viewEnd, false);
}
void
queryTest(std::shared_ptr<const ndn::Interest> interest)
{
runJsonQuery(interest);
}
void
prepareSegmentsByParams(std::vector<std::pair<std::string, std::string>>& queryParams,
const ndn::Name& segmentPrefix)
{
//BOOST_CHECK_EQUAL(sqlString, "SELECT name FROM cmip5 WHERE name=\'test\';");
for (auto it = queryParams.begin() ; it != queryParams.end(); ++it) {
std::cout << it->first << " " << it->second << std::endl;
}
Json::Value fileList;
fileList.append("/ndn/test1");
fileList.append("/ndn/test2");
fileList.append("/ndn/test3");
std::shared_ptr<ndn::Data> data = makeReplyData(segmentPrefix,
fileList, 0, true, false,
3, 0, 2, true);
m_mutex.lock();
m_cache.insert(*data);
m_mutex.unlock();
}
std::shared_ptr<const ndn::Data>
getDataFromCache(const ndn::Interest& interest)
{
return m_cache.find(interest);
}
void
configAdapter(const util::ConfigSection& section,
const ndn::Name& prefix)
{
onConfig(section, false, std::string("test.txt"), prefix);
}
bool
json2AutocompletionSqlTest(std::stringstream& sqlQuery,
Json::Value& jsonValue,
bool& lastComponent,
std::stringstream& nameField)
{
return json2AutocompletionSql(sqlQuery, jsonValue, lastComponent, nameField);
}
bool
testDoPrefixBasedSearch(Json::Value& jsonValue,
std::vector<std::pair<std::string, std::string>>& typedComponents)
{
return doPrefixBasedSearch(jsonValue, typedComponents);
}
bool
testDoFilterBasedSearch(Json::Value& jsonValue,
std::vector<std::pair<std::string, std::string>>& typedComponents)
{
return doFilterBasedSearch(jsonValue, typedComponents);
}
};
class QueryAdapterFixture : public UnitTestTimeFixture
{
public:
QueryAdapterFixture()
: face(makeDummyClientFace(io))
, keyChain(new ndn::KeyChain())
, databaseTable("cmip5")
, queryAdapterTest1(face, keyChain, syncSocket)
, queryAdapterTest2(face, keyChain, syncSocket)
{
std::string c1("activity"), c2("product"), c3("organization"), c4("model");
std::string c5("experiment"), c6("frequency"), c7("modeling_realm"), c8("variable_name");
std::string c9("ensemble"), c10("time");
nameFields.push_back(c1);
nameFields.push_back(c2);
nameFields.push_back(c3);
nameFields.push_back(c4);
nameFields.push_back(c5);
nameFields.push_back(c6);
nameFields.push_back(c7);
nameFields.push_back(c8);
nameFields.push_back(c9);
nameFields.push_back(c10);
queryAdapterTest1.setDatabaseTable(databaseTable);
queryAdapterTest1.setNameFields(nameFields);
queryAdapterTest2.setDatabaseTable(databaseTable);
queryAdapterTest2.setNameFields(nameFields);
}
virtual
~QueryAdapterFixture()
{
}
protected:
void
initializeQueryAdapterTest1()
{
util::ConfigSection section;
try {
std::stringstream ss;
ss << "signingId /test/signingId\
filterCategoryNames activity,product,organization,model,experiment,frequency,modeling_realm,variable_name,ensemble\
database \
{ \
dbServer localhost \
dbName testdb \
dbUser testuser \
dbPasswd testpwd \
}";
boost::property_tree::read_info(ss, section);
}
catch (boost::property_tree::info_parser_error &e) {
std::cout << "Failed to read config file " << e.what() << std::endl;;
}
queryAdapterTest1.configAdapter(section, ndn::Name("/test"));
}
void
initializeQueryAdapterTest2()
{
util::ConfigSection section;
try {
std::stringstream ss;
ss << "\
filterCategoryNames activity,product,organization,model,experiment,frequency,modeling_realm,variable_name,ensemble\
database\
{ \
dbServer localhost \
dbName testdb \
dbUser testuser \
dbPasswd testpwd \
}";
boost::property_tree::read_info(ss, section);
}
catch (boost::property_tree::info_parser_error &e) {
std::cout << "Failed to read config file " << e.what() << std::endl;;
}
queryAdapterTest2.configAdapter(section, ndn::Name("/test"));
}
protected:
std::shared_ptr<DummyClientFace> face;
std::shared_ptr<ndn::KeyChain> keyChain;
std::shared_ptr<chronosync::Socket> syncSocket;
std::string databaseTable;
std::vector<std::string> nameFields;
QueryAdapterTest queryAdapterTest1;
QueryAdapterTest queryAdapterTest2;
};
BOOST_FIXTURE_TEST_SUITE(QueryAdapterTestSuite, QueryAdapterFixture)
BOOST_AUTO_TEST_CASE(BasicQueryAdapterTest1)
{
BOOST_CHECK(queryAdapterTest1.getPrefix() == ndn::Name());
BOOST_CHECK(queryAdapterTest1.getSigningId() == ndn::Name());
}
BOOST_AUTO_TEST_CASE(BasicQueryAdapterTest2)
{
initializeQueryAdapterTest1();
BOOST_CHECK(queryAdapterTest1.getPrefix() == ndn::Name("/test"));
BOOST_CHECK(queryAdapterTest1.getSigningId() == ndn::Name("/test/signingId"));
}
// use real data instead of ack data
BOOST_AUTO_TEST_CASE(QueryAdapterMakeAckDataTest)
{
ndn::Interest interest(ndn::Name("/test/ack/data/json"));
interest.setInterestLifetime(ndn::time::milliseconds(1000));
interest.setMustBeFresh(true);
std::shared_ptr<const ndn::Interest> interestPtr = std::make_shared<ndn::Interest>(interest);
const ndn::name::Component version
= ndn::name::Component::fromVersion(1);
std::shared_ptr<ndn::Data> data = queryAdapterTest2.getAckData(interestPtr, version);
BOOST_CHECK_EQUAL(data->getName().toUri(), "/test/ack/data/json");
}
BOOST_AUTO_TEST_CASE(QueryAdapterMakeReplyDataTest1)
{
Json::Value fileList;
fileList.append("/ndn/test1");
fileList.append("/ndn/test2");
const ndn::Name prefix("/atmos/test/prefix");
std::shared_ptr<ndn::Data> data = queryAdapterTest2.getReplyData(prefix, fileList,
1, false, false, 2, 0, 1);
BOOST_CHECK_EQUAL(data->getName().toUri(), "/atmos/test/prefix/%00%01");
BOOST_CHECK_EQUAL(data->getFinalBlockId(), ndn::Name::Component(""));
const std::string jsonRes(reinterpret_cast<const char*>(data->getContent().value()),
data->getContent().value_size());
Json::Value parsedFromString;
Json::Reader reader;
BOOST_CHECK_EQUAL(reader.parse(jsonRes, parsedFromString), true);
BOOST_CHECK_EQUAL(parsedFromString["resultCount"], 2);
BOOST_CHECK_EQUAL(parsedFromString["results"].size(), 2);
BOOST_CHECK_EQUAL(parsedFromString["results"][0], "/ndn/test1");
BOOST_CHECK_EQUAL(parsedFromString["results"][1], "/ndn/test2");
}
BOOST_AUTO_TEST_CASE(QueryAdapterMakeReplyDataTest2)
{
Json::Value fileList;
fileList.append("/ndn/test1");
const ndn::Name prefix("/atmos/test/prefix");
std::shared_ptr<ndn::Data> data = queryAdapterTest2.getReplyData(prefix, fileList,
2, true, true, 1, 0, 0);
BOOST_CHECK_EQUAL(data->getName().toUri(), "/atmos/test/prefix/%00%02");
BOOST_CHECK_EQUAL(data->getFinalBlockId(), ndn::Name::Component::fromSegment(2));
const std::string jsonRes(reinterpret_cast<const char*>(data->getContent().value()),
data->getContent().value_size());
Json::Value parsedFromString;
Json::Reader reader;
BOOST_CHECK_EQUAL(reader.parse(jsonRes, parsedFromString), true);
BOOST_CHECK_EQUAL(parsedFromString["resultCount"], 1);
BOOST_CHECK_EQUAL(parsedFromString["next"].size(), 1);
BOOST_CHECK_EQUAL(parsedFromString["next"][0], "/ndn/test1");
}
BOOST_AUTO_TEST_CASE(QueryAdapterQueryProcessTest)
{
initializeQueryAdapterTest2();
Json::Value query;
query["name"] = "test";
Json::FastWriter fastWriter;
std::string jsonMessage = fastWriter.write(query);
jsonMessage.erase(std::remove(jsonMessage.begin(), jsonMessage.end(), '\n'), jsonMessage.end());
std::shared_ptr<ndn::Interest> queryInterest
= std::make_shared<ndn::Interest>(ndn::Name("/test/query").append(jsonMessage.c_str()));
queryAdapterTest2.queryTest(queryInterest);
// TODO: the code below should be enabled when queryAdapter can get the correct the
// ChronoSync state; currently, we don't need the activeQuery to save the ACK data;
//auto ackData = queryAdapterTest2.getDataFromActiveQuery(jsonMessage);
//
//BOOST_CHECK(ackData);
//if (ackData) {
// BOOST_CHECK_EQUAL(ackData->getName().getPrefix(3),
// ndn::Name("/test/query/%7B%22name%22%3A%22test%22%7D"));
// BOOST_CHECK_EQUAL(ackData->getName().at(ackData->getName().size() - 1),
// ndn::Name::Component("OK"));
// BOOST_CHECK_EQUAL(ackData->getContent().value_size(), 0);
//}
// there is no query-results namespace data
auto replyData = queryAdapterTest2.getDataFromCache(*queryInterest);
BOOST_CHECK(replyData);
if (replyData){
BOOST_CHECK_EQUAL(replyData->getName().getPrefix(2), ndn::Name("/test/query"));
const std::string jsonRes(reinterpret_cast<const char*>(replyData->getContent().value()),
replyData->getContent().value_size());
Json::Value parsedFromString;
Json::Reader reader;
BOOST_CHECK_EQUAL(reader.parse(jsonRes, parsedFromString), true);
BOOST_CHECK_EQUAL(parsedFromString["resultCount"], 3);
BOOST_CHECK_EQUAL(parsedFromString["results"].size(), 3);
BOOST_CHECK_EQUAL(parsedFromString["results"][0], "/ndn/test1");
BOOST_CHECK_EQUAL(parsedFromString["results"][1], "/ndn/test2");
BOOST_CHECK_EQUAL(parsedFromString["results"][2], "/ndn/test3");
}
}
BOOST_AUTO_TEST_CASE(QueryAdapterAutocompletionSqlSuccessTest)
{
initializeQueryAdapterTest2();
std::stringstream ss, nameField;
Json::Value testJson;
bool lastComponent = false;
testJson["?"] = "/";
BOOST_CHECK_EQUAL(true,
queryAdapterTest2.json2AutocompletionSqlTest(ss,
testJson,
lastComponent,
nameField));
BOOST_CHECK_EQUAL(lastComponent, false);
BOOST_CHECK_EQUAL(";", ss.str());
BOOST_CHECK_EQUAL("activity", nameField.str());
ss.str("");
ss.clear();
nameField.str("");
nameField.clear();
testJson.clear();
testJson["?"] = "/Activity/";
BOOST_CHECK_EQUAL(true,
queryAdapterTest2.json2AutocompletionSqlTest(ss,
testJson,
lastComponent,
nameField));
BOOST_CHECK_EQUAL(lastComponent, false);
BOOST_CHECK_EQUAL(" WHERE activity='Activity';", ss.str());
BOOST_CHECK_EQUAL("product", nameField.str());
ss.str("");
ss.clear();
nameField.str("");
nameField.clear();
testJson.clear();
testJson["?"] = "/Activity/Product/Organization/Model/Experiment/";
BOOST_CHECK_EQUAL(true,
queryAdapterTest2.json2AutocompletionSqlTest(ss,
testJson,
lastComponent,
nameField));
BOOST_CHECK_EQUAL(lastComponent, false);
BOOST_CHECK_EQUAL(" WHERE activity='Activity' AND \
experiment='Experiment' AND model='Model' AND organization='Organization' AND product='Product';",
ss.str());
BOOST_CHECK_EQUAL("frequency", nameField.str());
ss.str("");
ss.clear();
nameField.str("");
nameField.clear();
testJson.clear();
testJson["?"] = "/Activity/Product/Organization/Model/Experiment/Frequency/Modeling/\
Variable/Ensemble/";
BOOST_CHECK_EQUAL(true,
queryAdapterTest2.json2AutocompletionSqlTest(ss,
testJson,
lastComponent,
nameField));
BOOST_CHECK_EQUAL(lastComponent, true);
BOOST_CHECK_EQUAL(" WHERE activity='Activity' AND ensemble=\
'Ensemble' AND experiment='Experiment' AND frequency='Frequency' AND model='Model' AND \
modeling_realm='Modeling' AND organization='Organization' AND product='Product' AND variable_name=\
'Variable';",ss.str());
BOOST_CHECK_EQUAL("time", nameField.str());
}
BOOST_AUTO_TEST_CASE(QueryAdapterAutocompletionSqlFailTest)
{
initializeQueryAdapterTest2();
std::stringstream ss, nameField;
Json::Value testJson;
bool lastComponent = false;
testJson["?"] = "serchTest";
BOOST_CHECK_EQUAL(false,
queryAdapterTest2.json2AutocompletionSqlTest(ss,
testJson,
lastComponent,
nameField));
ss.str("");
ss.clear();
nameField.str("");
nameField.clear();
testJson.clear();
testJson["?"] = "/cmip5";
BOOST_CHECK_EQUAL(false,
queryAdapterTest2.json2AutocompletionSqlTest(ss,
testJson,
lastComponent,
nameField));
ss.str("");
ss.clear();
nameField.str("");
nameField.clear();
Json::Value testJson2; //simply clear does not work
testJson2[0] = "test";
BOOST_CHECK_EQUAL(false,
queryAdapterTest2.json2AutocompletionSqlTest(ss,
testJson,
lastComponent,
nameField));
ss.str("");
ss.clear();
nameField.str("");
nameField.clear();
Json::Value testJson3;
testJson3 = Json::Value(Json::arrayValue);
BOOST_CHECK_EQUAL(false,
queryAdapterTest2.json2AutocompletionSqlTest(ss,
testJson,
lastComponent,
nameField));
ss.str("");
ss.clear();
nameField.str("");
nameField.clear();
Json::Value testJson4;
Json::Value param;
param[0] = "test";
testJson4["name"] = param;
BOOST_CHECK_EQUAL(false,
queryAdapterTest2.json2AutocompletionSqlTest(ss,
testJson,
lastComponent,
nameField));
}
BOOST_AUTO_TEST_CASE(QueryAdapterDoFilterBasedSearchTest)
{
Json::Value testJson;
testJson["activity"] = "testActivity";
testJson["product"] = "testProduct";
std::vector<std::pair<std::string, std::string>> resultComponents;
BOOST_CHECK_EQUAL(true, queryAdapterTest1.testDoFilterBasedSearch(testJson, resultComponents));
BOOST_CHECK_EQUAL(2, resultComponents.size());
for (auto it=resultComponents.begin(); it != resultComponents.end(); it++) {
if (it->first == "activity")
BOOST_CHECK_EQUAL(it->second, "testActivity");
if (it->first == "product")
BOOST_CHECK_EQUAL(it->second, "testProduct");
}
testJson.clear();
resultComponents.clear();
BOOST_CHECK_EQUAL(true, queryAdapterTest1.testDoFilterBasedSearch(testJson, resultComponents));
BOOST_CHECK_EQUAL(0, resultComponents.size());
testJson.clear();
resultComponents.clear();
testJson["name"] = Json::nullValue;
BOOST_CHECK_EQUAL(false, queryAdapterTest1.testDoFilterBasedSearch(testJson, resultComponents));
testJson.clear();
resultComponents.clear();
Json::Value param;
param[0] = "test";
testJson["name"] = param;
BOOST_CHECK_EQUAL(false, queryAdapterTest1.testDoFilterBasedSearch(testJson, resultComponents));
testJson.clear();
resultComponents.clear();
Json::Value tmp;
tmp[0] = "test";
BOOST_CHECK_EQUAL(false, queryAdapterTest1.testDoFilterBasedSearch(tmp, resultComponents));
}
BOOST_AUTO_TEST_CASE(QueryAdapterDoPrefixBasedSearchTest)
{
Json::Value testJson;
testJson["??"] = "/";
std::vector<std::pair<std::string, std::string>> resultComponents;
BOOST_CHECK_EQUAL(true, queryAdapterTest2.testDoPrefixBasedSearch(testJson, resultComponents));
BOOST_CHECK_EQUAL(0, resultComponents.size());
testJson.clear();
resultComponents.clear();
testJson["??"] = "/Activity/Product";
BOOST_CHECK_EQUAL(true, queryAdapterTest2.testDoPrefixBasedSearch(testJson, resultComponents));
BOOST_CHECK_EQUAL(2, resultComponents.size());
for (auto it=resultComponents.begin(); it != resultComponents.end(); it++) {
if (it->first == "activity")
BOOST_CHECK_EQUAL(it->second, "Activity");
if (it->first == "product")
BOOST_CHECK_EQUAL(it->second, "Product");
}
testJson.clear();
resultComponents.clear();
testJson["??"] = "";
BOOST_CHECK_EQUAL(false, queryAdapterTest2.testDoPrefixBasedSearch(testJson, resultComponents));
testJson.clear();
resultComponents.clear();
testJson = Json::Value(Json::arrayValue);
BOOST_CHECK_EQUAL(false, queryAdapterTest2.testDoPrefixBasedSearch(testJson, resultComponents));
testJson.clear();
resultComponents.clear();
Json::Value tmp;
tmp[0] = "test";
BOOST_CHECK_EQUAL(false, queryAdapterTest2.testDoPrefixBasedSearch(tmp, resultComponents));
testJson.clear();
resultComponents.clear();
Json::Value param, testJson2;
param[0] = "test";
testJson2["activity"] = param;
BOOST_CHECK_EQUAL(false, queryAdapterTest2.testDoPrefixBasedSearch(testJson2, resultComponents));
}
BOOST_AUTO_TEST_SUITE_END()
}//tests
}//atmos