[Positioning] Add class OutputCSV

OutputCSV writes Result to a CSV file. It extends OutputFileMedium.

Update:
- Output.
- UserInterface (add option "output.csv-file").
- posexcept (add class no_output_csv_file).
This commit is contained in:
Matteo Cypriani 2010-03-08 17:36:16 +01:00
parent 84789f2ed2
commit 56c45b3334
9 changed files with 182 additions and 3 deletions

View File

@ -57,6 +57,7 @@ OBJ_LIST = \
userinterface.o \
output.o \
outputterminal.o \
outputcsv.o \
positioning.o \
input.o \
inputcsv.o \
@ -148,6 +149,10 @@ input.o: \
outputterminal.o: \
$(SRC_DIR)/outputmedium.hh \
$(OBJ_DIR)/result.o
outputcsv.o: \
$(SRC_DIR)/outputmedium.hh \
$(OBJ_DIR)/outputfilemedium.o \
$(OBJ_DIR)/result.o
output.o: \
$(OBJ_DIR)/outputterminal.o \
$(OBJ_DIR)/configuration.o \

View File

@ -1,8 +1,10 @@
#include "output.hh"
#include "outputterminal.hh"
#include "configuration.hh"
#include "posexcept.hh"
#include "outputterminal.hh"
#include "outputcsv.hh"
#include <string>
using namespace std ;
@ -48,6 +50,9 @@ void Output::initialise_output_media()
if (*i == "Terminal")
initialise_output_terminal() ;
else if (*i == "CSV")
initialise_output_csv() ;
else
throw output_medium_type_unknown(*i) ;
}
@ -60,6 +65,17 @@ void Output::initialise_output_terminal()
}
void Output::initialise_output_csv()
{
if (! Configuration::is_configured("output.csv-file"))
throw no_output_csv_file() ;
output_media.push_back(
new OutputCSV(
Configuration::string_value("output.csv-file"))) ;
}
void Output::write(const Result &result) const
{
for (vector<OutputMedium*>::const_iterator i = output_media.begin() ;

View File

@ -17,6 +17,7 @@ protected:
//@{
void initialise_output_media(void) ;
void initialise_output_terminal(void) ;
void initialise_output_csv(void) ;
//@}
public:

View File

@ -0,0 +1,36 @@
#include "outputcsv.hh"
#include "request.hh"
#include "mobile.hh"
#include <sstream>
using namespace std ;
/* *** Operations *** */
const string OutputCSV::result_to_csv(const Result &result)
{
ostringstream csv_line ;
const Request *const request = result.get_request() ;
if (request != NULL)
{
if (request->get_mobile() != NULL)
csv_line << request->get_mobile()->get_mac_addr() ;
csv_line << ';' << request->get_time_sent() ;
}
else
csv_line << ';' ;
Point3D position = result.get_position() ;
csv_line
<< ';' << position.get_x()
<< ';' << position.get_y()
<< ';' << position.get_z() ;
csv_line << '\n' ;
return csv_line.str() ;
}

View File

@ -0,0 +1,36 @@
#ifndef _OWLPS_POSITIONING_OUTPUTCSV_HH_
#define _OWLPS_POSITIONING_OUTPUTCSV_HH_
#include "outputmedium.hh"
#include "outputfilemedium.hh"
/// Writes a result to a CSV file
/**
* CSV format is:
* Mobile_MAC;Timestamp;X;Y;Z
*/
class OutputCSV: public OutputMedium, public OutputFileMedium
{
protected:
const std::string result_to_csv(const Result &result) ;
public:
OutputCSV(const std::string &filename):
OutputFileMedium(filename) {}
void write(const Result &result) ;
} ;
/* *** Operations *** */
inline void OutputCSV::write(const Result &result)
{
write_string_to_file(result_to_csv(result)) ;
}
#endif // _OWLPS_POSITIONING_OUTPUTCSV_HH_

View File

@ -113,6 +113,12 @@ const char* output_medium_type_unknown::what() const throw()
}
const char* no_output_csv_file::what() const throw()
{
return "No input CSV file specified in the configuration!" ;
}
/* *** Algorithms *** */

View File

@ -109,6 +109,13 @@ public:
} ;
class no_output_csv_file: public posexcept
{
public:
const char* what() const throw() ;
} ;
/* *** Algorithms *** */

View File

@ -110,8 +110,10 @@ this option more than once (but at least once). Allowed: Real.")
// Output options
("output.medium,O", po::value< vector<string> >()->composing(),
"Medium to which the results will be wrote. You can specify \
this option more than once. Allowed: Terminal. If this option is \
absent, results will be printed on the terminal.")
this option more than once. Allowed: Terminal, CSV. \
If this option is absent, results will be printed on the terminal.")
("output.csv-file", po::value<string>(),
"CSV file to use for output (when output.medium = CSV).")
; // End of options
}

View File

@ -0,0 +1,70 @@
#include <cxxtest/TestSuite.h>
#include "outputcsv.hh"
class OutputCSV_test: public CxxTest::TestSuite
{
private:
std::string csv_file_name ;
public:
OutputCSV_test(void)
{
Stock::clear() ;
csv_file_name = "/tmp/OutputCSV_test_csv_file.csv" ;
}
~OutputCSV_test(void)
{
TestUtil::remove_file(csv_file_name) ;
Stock::clear() ;
}
static OutputCSV_test* createSuite(void)
{
return new OutputCSV_test() ;
}
static void destroySuite(OutputCSV_test *suite)
{
delete suite ;
}
void test_outputcsv(void)
{
OutputCSV *outputcsv1 = new OutputCSV(csv_file_name) ;
for (std::vector<Result>::const_iterator i =
TestUtil::results.begin() ;
i != TestUtil::results.end() ; ++i)
outputcsv1->write(*i) ;
delete outputcsv1 ;
unsigned int result_count = 0 ;
std::ifstream inputcsv(csv_file_name.c_str()) ;
while (inputcsv)
{
std::string line ;
getline(inputcsv, line) ;
const Result &result = TestUtil::results.at(result_count) ;
std::ostringstream result_csv ;
result_csv
<< result.get_request()->get_mobile()->get_mac_addr() << ';'
<< result.get_request()->get_time_sent() << ';'
<< result.get_position().get_x() << ';'
<< result.get_position().get_y() << ';'
<< result.get_position().get_z() ;
TS_ASSERT_EQUALS(line, result_csv.str()) ;
if (++result_count >= TestUtil::results.size())
break ;
}
}
} ;