WIP: first round of stubbing for diff::Record
This commit is contained in:
parent
b91734b0a6
commit
96e10faa84
2 changed files with 243 additions and 26 deletions
|
|
@ -67,10 +67,16 @@
|
||||||
|
|
||||||
|
|
||||||
#include "lib/error.hpp"
|
#include "lib/error.hpp"
|
||||||
|
#include "lib/iter-adapter.hpp"
|
||||||
|
#include "lib/iter-adapter-stl.hpp"
|
||||||
|
#include "lib/itertools.hpp"
|
||||||
|
|
||||||
//#include "lib/util.hpp"
|
//#include "lib/util.hpp"
|
||||||
//#include "lib/format-string.hpp"
|
//#include "lib/format-string.hpp"
|
||||||
|
|
||||||
//#include <vector>
|
#include <utility>
|
||||||
|
#include <vector>
|
||||||
|
#include <string>
|
||||||
//#include <map>
|
//#include <map>
|
||||||
|
|
||||||
|
|
||||||
|
|
@ -80,6 +86,7 @@ namespace diff{
|
||||||
namespace error = lumiera::error;
|
namespace error = lumiera::error;
|
||||||
|
|
||||||
//using util::_Fmt;
|
//using util::_Fmt;
|
||||||
|
using std::string;
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
@ -87,8 +94,191 @@ namespace diff{
|
||||||
template<typename VAL>
|
template<typename VAL>
|
||||||
class Record
|
class Record
|
||||||
{
|
{
|
||||||
|
using _Vec = std::vector<VAL>;
|
||||||
|
using Attrib = std::pair<Symbol, VAL>;
|
||||||
|
using Attribs = _Vec;
|
||||||
|
using Children = _Vec;
|
||||||
|
|
||||||
|
string type_;
|
||||||
|
Attribs attribs_;
|
||||||
|
Children children_;
|
||||||
|
|
||||||
public:
|
public:
|
||||||
|
Record()
|
||||||
|
: type_("NIL")
|
||||||
|
{ }
|
||||||
|
|
||||||
|
template<typename A, typename C>
|
||||||
|
Record(Symbol typeID, A&& att, C&& chi)
|
||||||
|
: type_(typeID)
|
||||||
|
, attribs_(std::forward<A> (att))
|
||||||
|
, children_(std::forward<C> (chi))
|
||||||
|
{ }
|
||||||
|
|
||||||
|
template<typename A, typename C>
|
||||||
|
Record(Symbol typeID, std::initializer_list<A> const&& att
|
||||||
|
, std::initializer_list<C> const&& chi)
|
||||||
|
: type_(typeID)
|
||||||
|
, attribs_(att)
|
||||||
|
, children_(chi)
|
||||||
|
{ }
|
||||||
|
|
||||||
|
template<typename SEQ>
|
||||||
|
explicit
|
||||||
|
Record (SEQ const& con)
|
||||||
|
: type_("NIL")
|
||||||
|
{
|
||||||
|
auto p = std::begin(con);
|
||||||
|
auto e = std::end(con);
|
||||||
|
if (p!=e && isTypeID (*p))
|
||||||
|
type_ = extractTypeID(*(p++));
|
||||||
|
for ( ; p!=e && isAttribute(*p); ++p)
|
||||||
|
attribs_.push_back (*p);
|
||||||
|
for ( ; p!=e; ++p)
|
||||||
|
children_.push_back (*p);
|
||||||
|
}
|
||||||
|
|
||||||
|
Record (std::initializer_list<VAL> const&& ili)
|
||||||
|
: Record(ili)
|
||||||
|
{ }
|
||||||
|
|
||||||
|
// all default copy operations acceptable
|
||||||
|
|
||||||
|
|
||||||
|
operator std::string() const
|
||||||
|
{
|
||||||
|
return "nebbich"; ////TODO
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
bool
|
||||||
|
empty() const
|
||||||
|
{
|
||||||
|
return attribs_.empty()
|
||||||
|
&& children_.empty();
|
||||||
|
}
|
||||||
|
|
||||||
|
string
|
||||||
|
getType() const
|
||||||
|
{
|
||||||
|
return type_;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool
|
||||||
|
hasAttribute (string key) const
|
||||||
|
{
|
||||||
|
return false; ////TODO
|
||||||
|
}
|
||||||
|
|
||||||
|
bool
|
||||||
|
contains (VAL const& ref) const
|
||||||
|
{
|
||||||
|
return false; ////TODO
|
||||||
|
}
|
||||||
|
|
||||||
|
VAL const&
|
||||||
|
get (string key) const
|
||||||
|
{
|
||||||
|
return "booo"; ////TODO
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/* ==== Exposing scope and contents for iteration ====== */
|
||||||
|
|
||||||
|
using iterator = IterAdapter<typename _Vec::const_iterator, const Record*>;
|
||||||
|
using scopeIter = typename iter_stl::_SeqT<_Vec>::Range;
|
||||||
|
using keyIter = TransformIter<scopeIter, string>;
|
||||||
|
using valIter = TransformIter<scopeIter, VAL>;
|
||||||
|
|
||||||
|
|
||||||
|
iterator begin () const { return iterator(this, attribs_.begin()); }
|
||||||
|
iterator end () const { return iterator(); }
|
||||||
|
|
||||||
|
|
||||||
|
scopeIter attribs() const { return iter_stl::eachElm(attribs_); }
|
||||||
|
scopeIter scope() const { return iter_stl::eachElm(children_); }
|
||||||
|
|
||||||
|
keyIter keys() const { return transformIterator(attribs(), extractKey); }
|
||||||
|
valIter vals() const { return transformIterator(attribs(), extractVal); }
|
||||||
|
|
||||||
|
protected: /* ==== API for the IterAdapter ==== */
|
||||||
|
|
||||||
|
/** Implementation of Iteration-logic: pull next element. */
|
||||||
|
template<class ITER>
|
||||||
|
friend void
|
||||||
|
iterNext (const Record* src, ITER& pos)
|
||||||
|
{
|
||||||
|
++pos;
|
||||||
|
checkPoint (src,pos);
|
||||||
|
}
|
||||||
|
|
||||||
|
/** Implementation of Iteration-logic: detect iteration end.
|
||||||
|
* @remarks seamless continuation of the iteration when reaching
|
||||||
|
* the end of the attribute collection. In this implementation,
|
||||||
|
* we use the default constructed \c ITER() to mark iteration end.
|
||||||
|
*/
|
||||||
|
template<class ITER>
|
||||||
|
friend bool
|
||||||
|
checkPoint (const Record* src, ITER& pos)
|
||||||
|
{
|
||||||
|
REQUIRE (src);
|
||||||
|
if ((pos != ITER()) && (pos != src->children_.end()))
|
||||||
|
return true;
|
||||||
|
else
|
||||||
|
if (pos != ITER() && (pos == src->attribs_.end()) && !src->children_.empty())
|
||||||
|
{
|
||||||
|
pos = src->children_.begin();
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
pos = ITER();
|
||||||
|
return false;
|
||||||
|
} }
|
||||||
|
|
||||||
|
private:
|
||||||
|
static bool
|
||||||
|
isAttribute (VAL const& v)
|
||||||
|
{
|
||||||
|
return false; ////TODO
|
||||||
|
}
|
||||||
|
|
||||||
|
static bool
|
||||||
|
isTypeID (VAL const& v)
|
||||||
|
{
|
||||||
|
return false; ////TODO
|
||||||
|
}
|
||||||
|
|
||||||
|
static string
|
||||||
|
extractTypeID (VAL const& v)
|
||||||
|
{
|
||||||
|
return "todo"; ////TODO
|
||||||
|
}
|
||||||
|
|
||||||
|
static string
|
||||||
|
extractKey (VAL const& v)
|
||||||
|
{
|
||||||
|
return "todo"; ////TODO
|
||||||
|
}
|
||||||
|
|
||||||
|
static VAL
|
||||||
|
extractVal (VAL const& v)
|
||||||
|
{
|
||||||
|
return VAL(); ///TODO
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
friend bool
|
||||||
|
operator== (Record const& r1, Record const& r2)
|
||||||
|
{
|
||||||
|
return false; ////TODO
|
||||||
|
}
|
||||||
|
|
||||||
|
friend bool
|
||||||
|
operator!= (Record const& r1, Record const& r2)
|
||||||
|
{
|
||||||
|
return ! (r1 == r2);
|
||||||
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
|
|
|
||||||
|
|
@ -24,15 +24,17 @@
|
||||||
#include "lib/test/run.hpp"
|
#include "lib/test/run.hpp"
|
||||||
#include "lib/test/test-helper.hpp"
|
#include "lib/test/test-helper.hpp"
|
||||||
#include "lib/diff/record.hpp"
|
#include "lib/diff/record.hpp"
|
||||||
|
#include "lib/itertools.hpp"
|
||||||
#include "lib/util.hpp" //////TODO necessary?
|
#include "lib/util.hpp" //////TODO necessary?
|
||||||
|
|
||||||
#include <iostream>
|
#include <iostream>
|
||||||
//#include <utility>
|
//#include <utility>
|
||||||
//#include <string>
|
#include <string>
|
||||||
#include <vector>
|
#include <vector>
|
||||||
|
|
||||||
//using std::string;
|
using std::string;
|
||||||
using util::isSameObject;
|
using util::isSameObject;
|
||||||
|
using util::isnil;
|
||||||
using std::vector;
|
using std::vector;
|
||||||
//using std::swap;
|
//using std::swap;
|
||||||
using std::cout;
|
using std::cout;
|
||||||
|
|
@ -44,16 +46,41 @@ namespace diff{
|
||||||
namespace test{
|
namespace test{
|
||||||
|
|
||||||
// using lumiera::error::LUMIERA_ERROR_LOGIC;
|
// using lumiera::error::LUMIERA_ERROR_LOGIC;
|
||||||
|
using lumiera::error::LUMIERA_ERROR_INVALID;
|
||||||
|
|
||||||
namespace {//Test fixture....
|
namespace {//Test fixture....
|
||||||
|
|
||||||
|
using Seq = vector<string>;
|
||||||
|
using RecS = Record<string>;
|
||||||
|
|
||||||
|
template<class IT>
|
||||||
|
inline Seq
|
||||||
|
contents (IT const& it)
|
||||||
|
{
|
||||||
|
Seq collected;
|
||||||
|
append_all (it, collected);
|
||||||
|
return collected;
|
||||||
|
}
|
||||||
|
|
||||||
|
inline Seq
|
||||||
|
contents (RecS const& rec_of_strings)
|
||||||
|
{
|
||||||
|
return contents (rec_of_strings.begin());
|
||||||
|
}
|
||||||
|
|
||||||
|
template<class X>
|
||||||
|
inline Seq
|
||||||
|
strings (std::initializer_list<X> const& con)
|
||||||
|
{
|
||||||
|
Seq collected;
|
||||||
|
for (auto elm : con)
|
||||||
|
collected.push_back(elm);
|
||||||
|
return collected;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
}//(End)Test fixture
|
}//(End)Test fixture
|
||||||
|
|
||||||
using Seq = vector<string>;
|
|
||||||
using RecS = Record<String>;
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
@ -85,13 +112,13 @@ namespace test{
|
||||||
simpleUsage()
|
simpleUsage()
|
||||||
{
|
{
|
||||||
RecS enterprise("starship"
|
RecS enterprise("starship"
|
||||||
, {"Name = USS Enterprise"
|
, strings ({"Name = USS Enterprise"
|
||||||
,"Registry = NCC-1701-D"
|
,"Registry = NCC-1701-D"
|
||||||
,"Class = Galaxy"
|
,"Class = Galaxy"
|
||||||
,"Owner = United Federation of Planets"
|
,"Owner = United Federation of Planets"
|
||||||
,"built=2363"
|
,"built=2363"
|
||||||
}
|
})
|
||||||
, {"Picard", "Riker", "Data", "Worf", "Troi", "Crusher", "La Forge"}
|
, strings ({"Picard", "Riker", "Data", "Troi", "Worf", "Crusher", "La Forge"})
|
||||||
);
|
);
|
||||||
|
|
||||||
CHECK (enterprise.getType() == "starship");
|
CHECK (enterprise.getType() == "starship");
|
||||||
|
|
@ -102,7 +129,7 @@ namespace test{
|
||||||
CHECK (!enterprise.hasAttribute("Owner "));
|
CHECK (!enterprise.hasAttribute("Owner "));
|
||||||
|
|
||||||
CHECK (enterprise.contains("Data"));
|
CHECK (enterprise.contains("Data"));
|
||||||
CHECK (!contains (enterprise, "Woof"));
|
CHECK (!enterprise.contains("Woof"));
|
||||||
CHECK (util::contains (enterprise, "Worf"));
|
CHECK (util::contains (enterprise, "Worf"));
|
||||||
|
|
||||||
VERIFY_ERROR (INVALID, enterprise.get("warp10"));
|
VERIFY_ERROR (INVALID, enterprise.get("warp10"));
|
||||||
|
|
@ -110,7 +137,7 @@ namespace test{
|
||||||
cout << "enterprise = " << string(enterprise)<<endl;
|
cout << "enterprise = " << string(enterprise)<<endl;
|
||||||
for (string elm : enterprise) cout << elm<<endl;
|
for (string elm : enterprise) cout << elm<<endl;
|
||||||
for (string mbr : enterprise.scope()) cout << mbr<<endl;
|
for (string mbr : enterprise.scope()) cout << mbr<<endl;
|
||||||
for (auto attr : enterprise.attributes()) cout << attr<<endl;
|
for (string att : enterprise.attribs()) cout << att<<endl;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
@ -130,14 +157,14 @@ namespace test{
|
||||||
CHECK ("NIL" == untyped.getType());
|
CHECK ("NIL" == untyped.getType());
|
||||||
CHECK (Seq{"x"} == contents(untyped));
|
CHECK (Seq{"x"} == contents(untyped));
|
||||||
CHECK (Seq{"x"} == contents(untyped.scope()));
|
CHECK (Seq{"x"} == contents(untyped.scope()));
|
||||||
CHECK (isnil (untyped.attributes()));
|
CHECK (isnil (untyped.attribs()));
|
||||||
|
|
||||||
RecS untyped2({"x=y", "z"});
|
RecS untyped2({"x=y", "z"});
|
||||||
CHECK (!isnil(untyped2));
|
CHECK (!isnil(untyped2));
|
||||||
CHECK ("NIL" == untyped2.getType());
|
CHECK ("NIL" == untyped2.getType());
|
||||||
CHECK (Seq({"x=y", "z"}) == contents(untyped2));
|
CHECK (Seq({"x=y", "z"}) == contents(untyped2));
|
||||||
CHECK (Seq{"x"} == contents (untyped2.keys()));
|
CHECK (Seq{"x"} == contents (untyped2.keys()));
|
||||||
CHECK (Seq{"y"} == contents (untyped2.values()));
|
CHECK (Seq{"y"} == contents (untyped2.vals()));
|
||||||
CHECK (Seq{"z"} == contents (untyped2.scope()));
|
CHECK (Seq{"z"} == contents (untyped2.scope()));
|
||||||
|
|
||||||
|
|
||||||
|
|
@ -146,7 +173,7 @@ namespace test{
|
||||||
CHECK ("thing" == something.getType());
|
CHECK ("thing" == something.getType());
|
||||||
CHECK (Seq({"type=thing", "a=1", "b=2", "c", "d"}) == contents(something));
|
CHECK (Seq({"type=thing", "a=1", "b=2", "c", "d"}) == contents(something));
|
||||||
CHECK (Seq({"a", "b"}) == contents (something.keys()));
|
CHECK (Seq({"a", "b"}) == contents (something.keys()));
|
||||||
CHECK (Seq({"1", "2"}) == contents (something.values()));
|
CHECK (Seq({"1", "2"}) == contents (something.vals()));
|
||||||
CHECK (Seq({"c", "d"}) == contents (something.scope()));
|
CHECK (Seq({"c", "d"}) == contents (something.scope()));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
@ -158,10 +185,10 @@ namespace test{
|
||||||
RecS b(a);
|
RecS b(a);
|
||||||
CHECK (a.getType() == b.getType());
|
CHECK (a.getType() == b.getType());
|
||||||
CHECK (contents(a) == contents(b));
|
CHECK (contents(a) == contents(b));
|
||||||
CHECK (contents(a.attributes()) == contents(b.attributes()));
|
CHECK (contents(a.attribs()) == contents(b.attribs()));
|
||||||
|
|
||||||
CHECK (!isSameOject (a.get("a"), b.get("a")));
|
CHECK (!isSameObject (a.get("a"), b.get("a")));
|
||||||
CHECK (!isSameOject (*a.scope(), *b.scope()));
|
CHECK (!isSameObject (*a.scope(), *b.scope()));
|
||||||
|
|
||||||
string& c = *b.scope();
|
string& c = *b.scope();
|
||||||
CHECK ("c" == c);
|
CHECK ("c" == c);
|
||||||
|
|
@ -179,7 +206,7 @@ namespace test{
|
||||||
CHECK (isnil (b));
|
CHECK (isnil (b));
|
||||||
b = bb;
|
b = bb;
|
||||||
CHECK (!isnil (b));
|
CHECK (!isnil (b));
|
||||||
CHECK (!isSameObject(*b.get("a"), *bb.get("a")));
|
CHECK (!isSameObject(b.get("a"), bb.get("a")));
|
||||||
CHECK (!isSameObject(*b.scope(), *bb.scope()));
|
CHECK (!isSameObject(*b.scope(), *bb.scope()));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
@ -204,7 +231,7 @@ namespace test{
|
||||||
RecS a2({"a","aa"});
|
RecS a2({"a","aa"});
|
||||||
CHECK (aa == a2); CHECK (a2 == aa);
|
CHECK (aa == a2); CHECK (a2 == aa);
|
||||||
|
|
||||||
RecS o1("oo", {"a=α", "b=β"}, {"γ", "δ", "ε"});
|
RecS o1("oo", strings({"a=α", "b=β"}), strings({"γ", "δ", "ε"}));
|
||||||
RecS o2({"type=oo", "a = α", "b = β", "γ", "δ", "ε"});
|
RecS o2({"type=oo", "a = α", "b = β", "γ", "δ", "ε"});
|
||||||
RecS o3({"type=oO", "a = α", "b = β", "γ", "δ", "ε"});
|
RecS o3({"type=oO", "a = α", "b = β", "γ", "δ", "ε"});
|
||||||
RecS o4({"type=oo", "a = α", "b = β", "c=γ", "δ", "ε"});
|
RecS o4({"type=oo", "a = α", "b = β", "c=γ", "δ", "ε"});
|
||||||
|
|
@ -251,7 +278,7 @@ namespace test{
|
||||||
CHECK ("u" == aa.getType());
|
CHECK ("u" == aa.getType());
|
||||||
CHECK (Seq({"type=u", "a=1", "a"}) == contents(aa));
|
CHECK (Seq({"type=u", "a=1", "a"}) == contents(aa));
|
||||||
CHECK (Seq({"a"}) == contents (aa.keys()));
|
CHECK (Seq({"a"}) == contents (aa.keys()));
|
||||||
CHECK (Seq({"1"}) == contents (aa.values()));
|
CHECK (Seq({"1"}) == contents (aa.vals()));
|
||||||
CHECK (Seq({"a"}) == contents (aa.scope()));
|
CHECK (Seq({"a"}) == contents (aa.scope()));
|
||||||
|
|
||||||
CHECK (mut == aa);
|
CHECK (mut == aa);
|
||||||
|
|
|
||||||
Loading…
Reference in a new issue