2008-08-08 07:25:10 +00:00
|
|
|
#include <boost/optional.hpp>
|
|
|
|
|
2008-08-08 00:01:22 +00:00
|
|
|
#include "request.h"
|
|
|
|
|
2008-08-08 03:23:34 +00:00
|
|
|
namespace afp {
|
|
|
|
|
2008-08-08 06:31:46 +00:00
|
|
|
/* Perhaps, these should be configurable */
|
|
|
|
static const int MAX_FILTER_LEN = 1024 * 10; /* 10 KB */
|
|
|
|
static const int MAX_VARNAME_LEN = 255;
|
|
|
|
static const int MAX_VALUE_LEN = 1024 * 256; /* 256 KB */
|
|
|
|
|
2008-08-08 00:01:22 +00:00
|
|
|
// Protocol:
|
|
|
|
// code NULL <key> NULL <value> NULL ... <value> NULL NULL
|
|
|
|
|
2008-08-08 07:25:10 +00:00
|
|
|
template<typename charT, typename Traits>
|
|
|
|
struct basic_nul_terminated_string_reader {
|
|
|
|
typedef std::istream_iterator<charT> iterator_t;
|
|
|
|
typedef std::basic_istream<charT, Traits> stream_t;
|
|
|
|
typedef std::basic_string<charT, Traits> string_t;
|
|
|
|
|
|
|
|
basic_nul_terminated_string_reader(stream_t &stream)
|
|
|
|
: stream_(stream)
|
|
|
|
, it_(stream)
|
|
|
|
, first_(true)
|
|
|
|
{
|
|
|
|
}
|
2008-08-08 00:01:22 +00:00
|
|
|
|
2008-08-08 07:25:10 +00:00
|
|
|
boost::optional<string_t> read(std::size_t max_len = 0) {
|
|
|
|
string_t ret;
|
2008-08-08 00:01:22 +00:00
|
|
|
|
2008-08-08 07:25:10 +00:00
|
|
|
if (first_)
|
|
|
|
first_ = false;
|
|
|
|
else
|
|
|
|
++it_;
|
2008-08-08 00:01:22 +00:00
|
|
|
|
2008-08-08 07:25:10 +00:00
|
|
|
for (; it_ != end_; ++it_) {
|
|
|
|
if (*it_ == stream_.widen('\0')) {
|
|
|
|
return ret;
|
|
|
|
}
|
2008-08-08 06:31:46 +00:00
|
|
|
|
2008-08-08 07:25:10 +00:00
|
|
|
if (max_len && (ret.size() > max_len))
|
|
|
|
return boost::optional<string_t>();
|
|
|
|
|
|
|
|
ret += *it_;
|
|
|
|
}
|
|
|
|
|
|
|
|
return boost::optional<string_t>();
|
2008-08-08 00:01:22 +00:00
|
|
|
}
|
|
|
|
|
2008-08-08 07:25:10 +00:00
|
|
|
private:
|
|
|
|
stream_t &stream_;
|
|
|
|
iterator_t it_, end_;
|
|
|
|
bool first_;
|
|
|
|
};
|
2008-08-08 00:01:22 +00:00
|
|
|
|
2008-08-08 07:25:10 +00:00
|
|
|
typedef basic_nul_terminated_string_reader<char, std::char_traits<char> >
|
|
|
|
nul_terminated_string_reader;
|
2008-08-08 00:01:22 +00:00
|
|
|
|
2008-08-08 07:25:10 +00:00
|
|
|
bool
|
|
|
|
request::load(std::istream &inp) {
|
|
|
|
inp.unsetf(std::ios_base::skipws);
|
|
|
|
|
|
|
|
nul_terminated_string_reader reader(inp);
|
|
|
|
boost::optional<std::string> str;
|
|
|
|
|
|
|
|
if (!(str = reader.read(MAX_FILTER_LEN)))
|
|
|
|
return false;
|
|
|
|
filter = *str;
|
|
|
|
|
|
|
|
for (;;) {
|
2008-08-08 00:01:22 +00:00
|
|
|
std::string key, value;
|
|
|
|
|
|
|
|
/* read the key */
|
2008-08-08 07:25:10 +00:00
|
|
|
if (!(str = reader.read(MAX_VARNAME_LEN)))
|
2008-08-08 00:01:22 +00:00
|
|
|
return false;
|
2008-08-08 07:25:10 +00:00
|
|
|
key = *str;
|
2008-08-08 00:01:22 +00:00
|
|
|
|
|
|
|
if (key.empty())
|
|
|
|
/* empty string means end of input */
|
|
|
|
return true;
|
|
|
|
|
|
|
|
/* read the value */
|
2008-08-08 07:25:10 +00:00
|
|
|
if (!(str = reader.read(MAX_VALUE_LEN)))
|
2008-08-08 00:01:22 +00:00
|
|
|
return false;
|
2008-08-08 07:25:10 +00:00
|
|
|
value = *str;
|
2008-08-08 00:01:22 +00:00
|
|
|
|
2008-08-08 03:23:34 +00:00
|
|
|
f.add_variable(key, datum(value));
|
2008-08-08 00:01:22 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool
|
|
|
|
request::evaluate()
|
|
|
|
{
|
|
|
|
return f.evaluate(filter);
|
|
|
|
}
|
|
|
|
|
2008-08-08 03:23:34 +00:00
|
|
|
} // namespace afp
|