Attempting to reduce the remaining pre-C++11 workarounds before upgrade to C++20... As a first step: rename the old type-sequence implementation into `TyOLD` to make it clearly distinguishable; a new variadic implementation `TySeq` was already introduced as partial workaround, and the next steps will be to switch over essential parts of the type-sequence library.
157 lines
5.4 KiB
C++
157 lines
5.4 KiB
C++
/*
|
||
TupleRecordInit(Test) - to build a tuple from a GenNode sequence
|
||
|
||
Copyright (C)
|
||
2016, Hermann Vosseler <Ichthyostega@web.de>
|
||
|
||
**Lumiera** 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. See the file COPYING for further details.
|
||
|
||
* *****************************************************************/
|
||
|
||
/** @file tuple-record-init-test.cpp
|
||
** unit test \ref TupleRecordInit_test
|
||
*/
|
||
|
||
|
||
#include "lib/test/run.hpp"
|
||
#include "lib/test/test-helper.hpp"
|
||
#include "lib/time/timevalue.hpp"
|
||
#include "lib/meta/tuple-record-init.hpp"
|
||
#include "lib/format-cout.hpp"
|
||
#include "lib/format-util.hpp"
|
||
|
||
#include <string>
|
||
|
||
using lib::Symbol;
|
||
using lib::Variant;
|
||
using lib::idi::EntryID;
|
||
using lib::diff::Rec;
|
||
using lib::diff::MakeRec;
|
||
using lib::diff::GenNode;
|
||
using lib::meta::TyOLD;
|
||
using lib::meta::Tuple;
|
||
using lib::meta::buildTuple;
|
||
using lib::time::Duration;
|
||
using lib::time::Time;
|
||
using lib::hash::LuidH;
|
||
|
||
using std::string;
|
||
using std::tuple;
|
||
using std::get;
|
||
|
||
namespace lib {
|
||
namespace meta {
|
||
namespace test {
|
||
|
||
using LERR_(WRONG_TYPE);
|
||
|
||
|
||
|
||
|
||
/*************************************************************************//**
|
||
* @test Metaprogramming: how to unload the contents of a runtime typed
|
||
* variant sequence into ctor arguments of a (compile time typed) tuple.
|
||
*
|
||
* This involves two problems
|
||
* - how to combine iteration, compile-time indexing and run-time access.
|
||
* - how to overcome the runtime-to-compile-time barrier, using a
|
||
* pre-generated double-dispatch (visitor).
|
||
*
|
||
* The concrete problem leading to the development of such a generic
|
||
* converter was the necessity to receive a command invocation
|
||
* parameter tuple from a Record<GenNode> sent via the UI-Bus.
|
||
* @see ElementExtractor
|
||
* @see GenNodeAccessor
|
||
* @see BusTerm_test::commandInvocation
|
||
* @see stage::test::Nexus::prepareDiagnosticCommandHandler
|
||
* @see ui-bus.hpp UI-Bus
|
||
*/
|
||
class TupleRecordInit_test : public Test
|
||
{
|
||
virtual void
|
||
run (Arg)
|
||
{
|
||
show_simpleUsage();
|
||
verify_errorHandling();
|
||
}
|
||
|
||
|
||
void
|
||
show_simpleUsage()
|
||
{
|
||
using NiceTypes = TyOLD<string, int>;
|
||
using UgglyTypes = TyOLD<EntryID<long>, Symbol, int, int64_t, double, Duration>; // various conversions and an immutable type (Duration)
|
||
|
||
Rec args = MakeRec().scope("lalü", 42);
|
||
Rec urgs = MakeRec().scope("lalü", "lala", 12, 34, 5.6, Time(7,8,9));
|
||
|
||
cout << args <<endl;
|
||
cout << urgs <<endl;
|
||
|
||
cout << buildTuple<NiceTypes> (args) <<endl;
|
||
cout << buildTuple<UgglyTypes> (urgs) <<endl;
|
||
}
|
||
|
||
|
||
void
|
||
verify_errorHandling()
|
||
{
|
||
Rec args = MakeRec().scope("surprise", 42);
|
||
|
||
using TooMany = TyOLD<string, int, long>;
|
||
VERIFY_ERROR (WRONG_TYPE, buildTuple<TooMany> (args)); // number of types in tuple exceeds capacity of the supplied argument record
|
||
|
||
using Unsigned = TyOLD<string, uint>;
|
||
using Floating = TyOLD<string, float>;
|
||
using Narrowing = TyOLD<string, short>;
|
||
VERIFY_ERROR (WRONG_TYPE, buildTuple<Unsigned> (args)); // dangerous conversion from signed to unsigned int is prohibited
|
||
VERIFY_ERROR (WRONG_TYPE, buildTuple<Floating> (args)); // conversion from integral to floating point element is prohibited
|
||
VERIFY_ERROR (WRONG_TYPE, buildTuple<Narrowing> (args)); // narrowing conversion from int to short is prohibited
|
||
|
||
// yet other (non-numeric) conversions are still possible
|
||
Rec timeArg = MakeRec().scope(Time(1,2,3,4));
|
||
using TupStr = TyOLD<string>;
|
||
Tuple<TupStr> tup = buildTuple<TupStr> (timeArg);
|
||
|
||
CHECK (std::get<string> (tup) == "4:03:02.001");
|
||
CHECK (string(Time(1,2,3,4)) == "4:03:02.001");
|
||
|
||
|
||
// conversions from LUID elements are handled restrictively
|
||
Rec hashArg = MakeRec().scope("random", LuidH());
|
||
VERIFY_ERROR (WRONG_TYPE, buildTuple<Unsigned> (args));
|
||
VERIFY_ERROR (WRONG_TYPE, buildTuple<Floating> (args));
|
||
VERIFY_ERROR (WRONG_TYPE, buildTuple<Narrowing> (args));
|
||
|
||
using ToSizeT = TyOLD<string, size_t>;
|
||
VERIFY_ERROR (WRONG_TYPE, (buildTuple<ToSizeT> (args))); // not even conversion to size_t is allowed
|
||
|
||
struct Dummy
|
||
{
|
||
HashVal hash;
|
||
|
||
Dummy (LuidH const& luid)
|
||
: hash(luid)
|
||
{ }
|
||
};
|
||
|
||
using WithDummy = TyOLD<string, Dummy>;
|
||
|
||
Tuple<WithDummy> tup2 = buildTuple<WithDummy> (hashArg); // while any type explicitly constructible from LUID are permitted.
|
||
VERIFY_ERROR (WRONG_TYPE, buildTuple<WithDummy> (args)); // building a Dummy from int(42) is disallowed, of course
|
||
|
||
HashVal h = get<Dummy>(tup2).hash;
|
||
CHECK (h == hashArg.child(1).data.get<LuidH>()); // note: the narrowing conversion happens within LuidH::operator HashVal()
|
||
}
|
||
};
|
||
|
||
|
||
/** Register this test class... */
|
||
LAUNCHER (TupleRecordInit_test, "unit meta");
|
||
|
||
|
||
|
||
}}} // namespace lib::meta::test
|