Testcase (detect function invocation) passes now as expected
Some Library / Framework changes
- rename event-log-test.cpp
- allow the ExpectString also to work with concatenated expectation strings
Remark: there was a warning in the comment in event-log.hpp,
pointing out that negative assertions are shallow.
However, after the rework in 9/2018 (commit: d923138d1)
...this should no longer be true, since we perform proper backtracking,
leading to an exhaustive search.
366 lines
10 KiB
C++
366 lines
10 KiB
C++
/*
|
|
ACTIVITY-DETECTOR.hpp - test scaffolding to observe activities within the scheduler
|
|
|
|
Copyright (C) Lumiera.org
|
|
2023, 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.
|
|
|
|
*/
|
|
|
|
/** @file activity-detector.hpp
|
|
** Diagnostic setup to instrument and observe \ref Activity activations.
|
|
** The [Scheduler](\ref scheduler.hpp) powering the Lumiera render engine
|
|
** is implemented in terms of Activities, which can be time-bound and depend
|
|
** on each other. For performance reasons, these _operational atoms_ must be
|
|
** implemented as a tightly knit network of lightweight POD records without
|
|
** much indirection. This setup poses a challenge for unit tests and similar
|
|
** white box testing, due to the lack of a managed platform and any further
|
|
** means of indirection and extension. As a remedy, a set of preconfigured
|
|
** _detector Activity records_ is provided, which drop off event log messages
|
|
** by side effect. These detector probes can be wired in as decorators into
|
|
** a otherwise valid Activity-Term, allowing to watch and verify patterns
|
|
** of invocation -- which might even happen concurrently.
|
|
**
|
|
** @todo WIP-WIP-WIP 7/2023 right now this is a rather immature attempt
|
|
** towards a scaffolding to propel the build-up of the scheduler.
|
|
** @see SchedulerActivity_test
|
|
*/
|
|
|
|
|
|
#ifndef VAULT_GEAR_TEST_ACTIVITY_DETECTOR_H
|
|
#define VAULT_GEAR_TEST_ACTIVITY_DETECTOR_H
|
|
|
|
|
|
#include "vault/common.hpp"
|
|
//#include "lib/test/test-helper.hpp"
|
|
#include "lib/test/event-log.hpp"
|
|
|
|
//#include "steam/play/dummy-play-connection.hpp"
|
|
//#include "steam/fixture/node-graph-attachment.hpp"
|
|
//#include "steam/fixture/segmentation.hpp"
|
|
//#include "steam/mobject/model-port.hpp"
|
|
//#include "steam/engine/dispatcher.hpp"
|
|
//#include "steam/engine/job-ticket.hpp"
|
|
//#include "vault/gear/job.h"
|
|
//#include "vault/real-clock.hpp"
|
|
//#include "lib/allocator-handle.hpp"
|
|
//#include "lib/time/timevalue.hpp"
|
|
//#include "lib/diff/gen-node.hpp"
|
|
//#include "lib/linked-elements.hpp"
|
|
#include "lib/meta/variadic-helper.hpp"
|
|
#include "lib/wrapper.hpp"
|
|
#include "lib/format-cout.hpp"
|
|
#include "lib/format-util.hpp"
|
|
//#include "lib/itertools.hpp"
|
|
//#include "lib/depend.hpp"
|
|
#include "lib/util.hpp"
|
|
|
|
#include <functional>
|
|
#include <utility>
|
|
#include <string>
|
|
//#include <tuple>
|
|
//#include <map>
|
|
|
|
|
|
namespace vault{
|
|
namespace gear {
|
|
namespace test {
|
|
|
|
using std::string;
|
|
// using std::make_tuple;
|
|
// using lib::diff::GenNode;
|
|
// using lib::diff::MakeRec;
|
|
// using lib::time::TimeValue;
|
|
// using lib::time::Time;
|
|
// using lib::HashVal;
|
|
using lib::meta::RebindVariadic;
|
|
using util::isnil;
|
|
using std::forward;
|
|
using std::move;
|
|
// using util::isSameObject;
|
|
// using fixture::Segmentation;
|
|
// using vault::RealClock;
|
|
// using vault::gear::Job;
|
|
// using vault::gear::JobClosure;
|
|
|
|
|
|
/** Marker for invocation sequence */
|
|
class Seq
|
|
{
|
|
uint step_;
|
|
|
|
public:
|
|
Seq (uint start =0)
|
|
: step_{start}
|
|
{ }
|
|
|
|
operator uint() const
|
|
{
|
|
return step_;
|
|
}
|
|
operator string() const
|
|
{
|
|
return util::toString (step_);
|
|
}
|
|
|
|
uint
|
|
operator++()
|
|
{
|
|
++step_;
|
|
return step_;
|
|
}
|
|
|
|
bool
|
|
operator== (Seq const& o)
|
|
{
|
|
return step_ == o.step_;
|
|
}
|
|
};
|
|
|
|
|
|
namespace {// Event markers
|
|
const string MARK_INC{"IncSeq"};
|
|
const string MARK_SEQ{"Seq"};
|
|
}
|
|
|
|
class ActivityDetector;
|
|
|
|
|
|
class ActivityMatch
|
|
: private lib::test::EventMatch
|
|
{
|
|
using _Parent = lib::test::EventMatch;
|
|
|
|
ActivityMatch (lib::test::EventMatch&& matcher)
|
|
: _Parent{move (matcher)}
|
|
{ }
|
|
|
|
friend class ActivityDetector;
|
|
|
|
public:
|
|
// standard copy acceptable
|
|
|
|
operator bool() const { return _Parent::operator bool(); }
|
|
|
|
|
|
// EventMatch& locate (string match);
|
|
// EventMatch& locateMatch (string regExp);
|
|
// EventMatch& locateEvent (string match);
|
|
// EventMatch& locateEvent (string classifier, string match);
|
|
// EventMatch& locateCall (string match);
|
|
//
|
|
//
|
|
// /* query builders to find a match stepping forwards */
|
|
//
|
|
// EventMatch& before (string match);
|
|
// EventMatch& beforeMatch (string regExp);
|
|
// EventMatch& beforeEvent (string match);
|
|
// EventMatch& beforeEvent (string classifier, string match);
|
|
ActivityMatch& beforeInvocation (string match) { return delegate (&EventMatch::beforeCall, move(match)); }
|
|
//
|
|
//
|
|
// /* query builders to find a match stepping backwards */
|
|
//
|
|
// EventMatch& after (string match);
|
|
// EventMatch& afterMatch (string regExp);
|
|
// EventMatch& afterEvent (string match);
|
|
// EventMatch& afterEvent (string classifier, string match);
|
|
// EventMatch& afterCall (string match);
|
|
|
|
/** qualifier: additionally match the function arguments */
|
|
template<typename...ARGS>
|
|
ActivityMatch&
|
|
arg (ARGS const& ...args)
|
|
{
|
|
return delegate (&EventMatch::arg<ARGS...>, args...);
|
|
}
|
|
|
|
/** qualifier: additionally require the indicated sequence number */
|
|
ActivityMatch&
|
|
seq (uint seqNr)
|
|
{
|
|
_Parent::attrib (MARK_SEQ, util::toString (seqNr));
|
|
return *this;
|
|
}
|
|
|
|
ActivityMatch&
|
|
beforeSeqIncrement (uint seqNr)
|
|
{
|
|
_Parent::beforeEvent(MARK_INC, util::toString(seqNr));
|
|
return *this;
|
|
}
|
|
|
|
private:
|
|
template<typename...ARGS>
|
|
ActivityMatch&
|
|
delegate (_Parent& (_Parent::*fun) (ARGS...), ARGS&& ...args)
|
|
{
|
|
return static_cast<ActivityMatch&> (
|
|
(this->*fun) (forward<ARGS> (args)...));
|
|
}
|
|
};
|
|
|
|
|
|
/**
|
|
* Diagnostic context to record and evaluate activations within the Scheduler.
|
|
* @todo WIP-WIP-WIP 7/23 a new loopy hope
|
|
*/
|
|
class ActivityDetector
|
|
: util::NonCopyable
|
|
{
|
|
using EventLog = lib::test::EventLog;
|
|
|
|
EventLog eventLog_;
|
|
Seq invocationSeq_;
|
|
|
|
/**
|
|
* A Mock functor, logging all invocations into the EventLog
|
|
*/
|
|
template<typename RET, typename...ARGS>
|
|
class DiagnosticFun
|
|
{
|
|
using RetVal = lib::wrapper::ItemWrapper<RET>;
|
|
|
|
string id_;
|
|
EventLog* log_;
|
|
Seq const* seqNr_;
|
|
RetVal retVal_;
|
|
|
|
public:
|
|
DiagnosticFun (string id, EventLog& masterLog, Seq const& seqNr)
|
|
: id_{id}
|
|
, log_{&masterLog}
|
|
, seqNr_{&seqNr}
|
|
, retVal_{}
|
|
{ }
|
|
|
|
/** prepare a response value to return from the mock invocation */
|
|
template<typename VAL>
|
|
DiagnosticFun&&
|
|
returning (VAL&& riggedResponse)
|
|
{
|
|
retVal_ = std::forward<VAL> (riggedResponse);
|
|
return std::move (*this);
|
|
}
|
|
|
|
/** mock function call operator: logs all invocations */
|
|
RET
|
|
operator() (ARGS const& ...args)
|
|
{
|
|
log_->call (log_->getID(), id_, args...)
|
|
.addAttrib (MARK_SEQ, *seqNr_);
|
|
return *retVal_;
|
|
}
|
|
};
|
|
|
|
|
|
public:
|
|
ActivityDetector(string id ="")
|
|
: eventLog_{"ActivityDetector" + (isnil (id)? string{}: "("+id+")")}
|
|
, invocationSeq_{0}
|
|
{ }
|
|
|
|
operator string() const
|
|
{
|
|
return util::join (eventLog_);
|
|
}
|
|
|
|
void
|
|
clear(string newID)
|
|
{
|
|
if (isnil (newID))
|
|
eventLog_.clear();
|
|
else
|
|
eventLog_.clear (newID);
|
|
}
|
|
|
|
uint
|
|
operator++()
|
|
{
|
|
++invocationSeq_;
|
|
eventLog_.event (MARK_INC, invocationSeq_);
|
|
return invocationSeq_;
|
|
}
|
|
|
|
uint
|
|
currSeq() const
|
|
{
|
|
return invocationSeq_;
|
|
}
|
|
|
|
uint
|
|
markSequence()
|
|
{
|
|
return operator++();
|
|
}
|
|
|
|
|
|
/**
|
|
* Generic testing helper: build a λ-mock, logging all invocations
|
|
* @tparam SIG signature of the functor to be generated
|
|
* @param id human readable ID, to designate invocations in the log
|
|
* @return a function object with signature #SIG
|
|
*/
|
|
template<typename SIG>
|
|
auto
|
|
buildDiagnosticFun (string id)
|
|
{
|
|
using Ret = typename lib::meta::_Fun<SIG>::Ret;
|
|
using Args = typename lib::meta::_Fun<SIG>::Args;
|
|
using ArgsX = typename lib::meta::StripNullType<Args>::Seq; ////////////////////////////////////TICKET #987 : make lib::meta::Types<TYPES...> variadic
|
|
using SigTypes = typename lib::meta::Prepend<Ret, ArgsX>::Seq;
|
|
using Functor = typename RebindVariadic<DiagnosticFun, SigTypes>::Type;
|
|
|
|
return Functor{id, eventLog_, invocationSeq_};
|
|
}
|
|
|
|
|
|
|
|
ActivityMatch
|
|
verifyInvocation (string fun)
|
|
{
|
|
return ActivityMatch{move (eventLog_.verifyCall(fun))};
|
|
}
|
|
|
|
ActivityMatch
|
|
ensureNoInvocation (string fun)
|
|
{
|
|
return ActivityMatch{move (eventLog_.ensureNot(fun).locateCall(fun))};
|
|
}
|
|
|
|
ActivityMatch
|
|
verifySeqIncrement (uint seqNr)
|
|
{
|
|
return ActivityMatch{move (eventLog_.verifyEvent(MARK_INC, util::toString(seqNr)))};
|
|
}
|
|
|
|
|
|
string
|
|
showLog() const
|
|
{
|
|
return "\n____Event-Log___________________________\n"
|
|
+ util::join (eventLog_, "\n")
|
|
+ "\n────╼━━━━━━━━╾──────────────────────────"
|
|
;
|
|
}
|
|
|
|
|
|
private:
|
|
};
|
|
|
|
|
|
}}} // namespace vault::gear::test
|
|
#endif /*VAULT_GEAR_TEST_ACTIVITY_DETECTOR_H*/
|