2009-10-14 05:48:24 +02:00
|
|
|
/*
|
|
|
|
|
QUERY-RESOLVER.hpp - interface for discovering contents of a scope
|
|
|
|
|
|
|
|
|
|
Copyright (C) Lumiera.org
|
|
|
|
|
2009, Hermann Vosseler <Ichthyostega@web.de>
|
|
|
|
|
|
|
|
|
|
This program 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 2 of the
|
|
|
|
|
License, or (at your option) any later version.
|
|
|
|
|
|
|
|
|
|
This program 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 this program; if not, write to the Free Software
|
|
|
|
|
Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
|
|
|
|
|
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
#ifndef MOBJECT_SESSION_QUERY_RESOLVER_H
|
|
|
|
|
#define MOBJECT_SESSION_QUERY_RESOLVER_H
|
|
|
|
|
|
|
|
|
|
//#include "proc/mobject/mobject.hpp"
|
|
|
|
|
//#include "proc/mobject/placement.hpp"
|
2009-10-25 16:18:53 +01:00
|
|
|
#include "lib/bool-checkable.hpp"
|
2009-10-24 00:23:22 +02:00
|
|
|
#include "lib/typed-counter.hpp"
|
2009-10-17 21:31:03 +02:00
|
|
|
#include "lib/iter-adapter.hpp"
|
2009-10-24 00:23:22 +02:00
|
|
|
#include "lib/util.hpp"
|
2009-10-14 05:48:24 +02:00
|
|
|
|
2009-10-26 01:39:25 +01:00
|
|
|
#include <boost/scoped_ptr.hpp>
|
2009-10-25 16:18:53 +01:00
|
|
|
#include <tr1/memory>
|
2009-10-14 05:48:24 +02:00
|
|
|
//#include <vector>
|
|
|
|
|
//#include <string>
|
|
|
|
|
|
|
|
|
|
//using std::vector;
|
|
|
|
|
//using std::string;
|
|
|
|
|
|
|
|
|
|
namespace mobject {
|
|
|
|
|
namespace session {
|
|
|
|
|
|
2009-10-24 00:23:22 +02:00
|
|
|
using util::unConst;
|
|
|
|
|
|
|
|
|
|
|
2009-10-25 16:18:53 +01:00
|
|
|
class Goal;
|
2009-10-29 04:32:00 +01:00
|
|
|
class Resolution;
|
2009-10-24 00:23:22 +02:00
|
|
|
class QueryResolver;
|
2009-10-26 01:39:25 +01:00
|
|
|
class QueryDispatcher;
|
2009-10-24 00:23:22 +02:00
|
|
|
|
2009-10-25 16:18:53 +01:00
|
|
|
/** Allow for taking ownership of a result set */
|
2009-10-29 04:32:00 +01:00
|
|
|
typedef std::tr1::shared_ptr<Resolution> PReso;
|
2009-10-25 16:18:53 +01:00
|
|
|
|
|
|
|
|
|
2009-10-17 21:31:03 +02:00
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* TODO type comment
|
2009-10-24 00:23:22 +02:00
|
|
|
* Query ABC
|
2009-10-17 21:31:03 +02:00
|
|
|
*/
|
2009-10-24 00:23:22 +02:00
|
|
|
class Goal
|
2009-10-17 21:31:03 +02:00
|
|
|
{
|
|
|
|
|
public:
|
2009-10-25 21:39:02 +01:00
|
|
|
virtual ~Goal() ;
|
2009-10-17 21:31:03 +02:00
|
|
|
|
2009-10-24 00:23:22 +02:00
|
|
|
enum Kind
|
|
|
|
|
{ GENERIC = 0
|
|
|
|
|
, DISCOVERY
|
|
|
|
|
};
|
2009-10-17 21:31:03 +02:00
|
|
|
|
2009-10-24 00:23:22 +02:00
|
|
|
struct QueryID
|
|
|
|
|
{
|
|
|
|
|
Kind kind;
|
|
|
|
|
size_t type;
|
|
|
|
|
};
|
2009-10-17 21:31:03 +02:00
|
|
|
|
2009-10-24 00:23:22 +02:00
|
|
|
QueryID getQID() { return id_; }
|
2009-10-17 21:31:03 +02:00
|
|
|
|
|
|
|
|
|
2009-10-24 00:23:22 +02:00
|
|
|
class Result
|
2009-10-25 16:18:53 +01:00
|
|
|
: public lib::BoolCheckable<Result>
|
2009-10-24 00:23:22 +02:00
|
|
|
{
|
|
|
|
|
void *cur_;
|
|
|
|
|
|
|
|
|
|
protected:
|
2009-10-26 01:39:25 +01:00
|
|
|
void point_at(void* p) { cur_ = p; }
|
2009-10-25 16:18:53 +01:00
|
|
|
|
2009-10-24 00:23:22 +02:00
|
|
|
template<typename RES>
|
2009-10-25 16:18:53 +01:00
|
|
|
RES&
|
|
|
|
|
access()
|
2009-10-24 00:23:22 +02:00
|
|
|
{
|
|
|
|
|
REQUIRE (cur_);
|
|
|
|
|
return *reinterpret_cast<RES*> (cur_);
|
|
|
|
|
}
|
2009-10-25 16:18:53 +01:00
|
|
|
|
|
|
|
|
public:
|
|
|
|
|
bool isValid() const { return bool(cur_); }
|
|
|
|
|
|
|
|
|
|
Result() : cur_(0) { } ///< create an NIL result
|
2009-10-24 00:23:22 +02:00
|
|
|
};
|
2009-10-17 21:31:03 +02:00
|
|
|
|
2009-10-25 16:18:53 +01:00
|
|
|
|
2009-10-17 21:31:03 +02:00
|
|
|
|
|
|
|
|
protected:
|
2009-10-24 00:23:22 +02:00
|
|
|
QueryID id_;
|
|
|
|
|
|
|
|
|
|
Goal (QueryID qid)
|
2009-10-29 04:32:00 +01:00
|
|
|
: id_(qid)
|
2009-10-24 00:23:22 +02:00
|
|
|
{ }
|
|
|
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/** Context used for generating type-IDs to denote
|
|
|
|
|
* the specific result types of issued queries */
|
|
|
|
|
typedef lib::TypedContext<Goal::Result> ResultType;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* TODO type comment
|
|
|
|
|
* Concrete query to yield specifically typed result elements
|
|
|
|
|
*/
|
|
|
|
|
template<class RES>
|
|
|
|
|
class Query
|
|
|
|
|
: public Goal
|
|
|
|
|
{
|
|
|
|
|
static QueryID
|
|
|
|
|
defineQueryTypeID ()
|
|
|
|
|
{
|
|
|
|
|
QueryID id = {Goal::GENERIC, ResultType::ID<RES>::get()};
|
|
|
|
|
return id;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public:
|
|
|
|
|
Query()
|
|
|
|
|
: Goal (defineQueryTypeID())
|
|
|
|
|
{ }
|
|
|
|
|
|
|
|
|
|
/* results retrieval */
|
2009-10-25 16:18:53 +01:00
|
|
|
class Cursor
|
|
|
|
|
: public Goal::Result
|
2009-10-24 00:23:22 +02:00
|
|
|
{
|
|
|
|
|
public:
|
2009-10-29 21:59:02 +01:00
|
|
|
typedef RES value_type;
|
|
|
|
|
typedef RES& reference;
|
|
|
|
|
typedef RES* pointer;
|
|
|
|
|
|
2009-10-26 01:39:25 +01:00
|
|
|
RES& operator* () { return access<RES>(); }
|
|
|
|
|
RES* operator->() { return & access<RES>(); }
|
2009-10-25 16:18:53 +01:00
|
|
|
|
2009-10-29 04:32:00 +01:00
|
|
|
void point_at(RES* r){ Goal::Result::point_at(r);}
|
2009-10-24 00:23:22 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
2009-10-29 04:32:00 +01:00
|
|
|
typedef lib::IterAdapter<Cursor,PReso> iterator;
|
2009-10-25 16:18:53 +01:00
|
|
|
|
2009-10-24 00:23:22 +02:00
|
|
|
iterator operator() (QueryResolver const& resolver);
|
|
|
|
|
|
2009-10-17 21:31:03 +02:00
|
|
|
};
|
2009-10-14 05:48:24 +02:00
|
|
|
|
|
|
|
|
|
2009-10-29 04:32:00 +01:00
|
|
|
/**
|
|
|
|
|
* ABC denoting the result set
|
|
|
|
|
* of an individual query resolution
|
|
|
|
|
*/
|
|
|
|
|
class Resolution
|
|
|
|
|
{
|
2009-10-29 21:59:02 +01:00
|
|
|
public:
|
2009-10-29 04:32:00 +01:00
|
|
|
typedef Goal::Result Result;
|
|
|
|
|
|
|
|
|
|
virtual ~Resolution();
|
|
|
|
|
|
2009-10-29 21:59:02 +01:00
|
|
|
|
|
|
|
|
friend bool
|
|
|
|
|
hasNext (PReso const&, Result const& pos) ////TICKET #375
|
2009-10-29 04:32:00 +01:00
|
|
|
{
|
|
|
|
|
return bool(pos);
|
|
|
|
|
}
|
|
|
|
|
|
2009-10-29 21:59:02 +01:00
|
|
|
friend void
|
2009-10-29 04:32:00 +01:00
|
|
|
iterNext (PReso& resultSet, Result& pos)
|
|
|
|
|
{
|
|
|
|
|
resultSet->nextResult(pos);
|
|
|
|
|
}
|
|
|
|
|
|
2009-10-29 21:59:02 +01:00
|
|
|
|
2009-10-29 04:32:00 +01:00
|
|
|
virtual Result prepareResolution() =0;
|
|
|
|
|
|
|
|
|
|
protected:
|
|
|
|
|
|
|
|
|
|
virtual void nextResult(Result& pos) =0;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
2009-10-14 05:48:24 +02:00
|
|
|
/**
|
|
|
|
|
* TODO type comment
|
|
|
|
|
*/
|
|
|
|
|
class QueryResolver
|
|
|
|
|
{
|
2009-10-26 01:39:25 +01:00
|
|
|
boost::scoped_ptr<QueryDispatcher> dispatcher_;
|
2009-10-14 05:48:24 +02:00
|
|
|
|
|
|
|
|
public:
|
|
|
|
|
|
2009-10-24 00:23:22 +02:00
|
|
|
virtual ~QueryResolver() ;
|
2009-10-14 05:48:24 +02:00
|
|
|
|
2009-10-25 16:18:53 +01:00
|
|
|
|
2009-10-25 21:39:02 +01:00
|
|
|
|
2009-10-17 21:31:03 +02:00
|
|
|
/** issue a query to retrieve contents
|
2009-10-24 00:23:22 +02:00
|
|
|
* The query is handed over internally to a suitable resolver implementation.
|
2009-10-25 21:39:02 +01:00
|
|
|
* @return concrete Resolution of the query (ResultSet), \em managed by smart-pointer.
|
2009-10-24 00:23:22 +02:00
|
|
|
* @throw lumiera::Error subclass if query evaluation flounders.
|
|
|
|
|
* This might be broken logic, invalid input, misconfiguration
|
|
|
|
|
* or failure of an external facility used for resolution.
|
|
|
|
|
* @note a query may yield no results, in which case the iterator is empty.
|
2009-10-17 21:31:03 +02:00
|
|
|
*/
|
2009-10-29 21:59:02 +01:00
|
|
|
PReso issue (Goal& query) const;
|
2009-10-17 21:31:03 +02:00
|
|
|
|
|
|
|
|
|
2009-10-24 00:23:22 +02:00
|
|
|
protected:
|
|
|
|
|
virtual bool canHandleQuery (Goal::QueryID qID) const =0;
|
|
|
|
|
|
2009-10-29 04:32:00 +01:00
|
|
|
QueryResolver();
|
2009-10-14 05:48:24 +02:00
|
|
|
};
|
|
|
|
|
///////////////////////////TODO currently just fleshing the API
|
|
|
|
|
|
|
|
|
|
|
2009-10-24 00:23:22 +02:00
|
|
|
template<typename RES>
|
|
|
|
|
inline typename Query<RES>::iterator
|
|
|
|
|
Query<RES>::operator() (QueryResolver const& resolver)
|
|
|
|
|
{
|
2009-10-25 21:39:02 +01:00
|
|
|
PReso resultSet = resolver.issue (*this);
|
2009-10-29 04:32:00 +01:00
|
|
|
Result first = resultSet->prepareResolution();
|
2009-10-29 21:59:02 +01:00
|
|
|
Cursor& start = static_cast<Cursor&> (first);
|
|
|
|
|
return iterator (resultSet, start);
|
2009-10-24 00:23:22 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2009-10-14 05:48:24 +02:00
|
|
|
}} // namespace mobject::session
|
|
|
|
|
#endif
|