...and complete unit test coverage. This is complex stuff and we'd better be careful it actually works
124 lines
4.7 KiB
C++
124 lines
4.7 KiB
C++
/*
|
|
TREE-MUTATOR-NOOP-BINDING.hpp - diff::TreeMutator implementation building block
|
|
|
|
Copyright (C) Lumiera.org
|
|
2018, 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 tree-mutator-noop-binding.hpp
|
|
** Special binding implementation for TreeMutator, allowing to accept
|
|
** and ignore any tree diff without tangible effect. TreeMutator is a
|
|
** customisable intermediary, which enables otherwise opaque implementation
|
|
** data structures to receive and respond to generic structural change messages
|
|
** ("tree diff").
|
|
**
|
|
** Each concrete TreeMutator instance will be configured differently, and this
|
|
** adaptation is done by combining various building blocks. This header defines
|
|
** a special `/dev/null` building block, which behaves as if successfully consuming
|
|
** the given diff without actually doing anything. Obviously, such a "black hole layer"
|
|
** need to be below any other diff binding, and may be used to absorb any diff verbs
|
|
** not matched and consumed by a more specific binding. Contrast this to the TreeMutator
|
|
** default implementation, which likewise absorbs diff verbs, but in a way to trigger
|
|
** a lumiera::error::LUMIERA_ERROR_DIFF_CONFLICT.
|
|
**
|
|
** @note the header tree-mutator-attribute-binding.hpp was split off for sake of readability
|
|
** and is included automatically from bottom of tree-mutator.hpp
|
|
**
|
|
** @see tree-mutator-test.cpp
|
|
** @see TreeMutator::build()
|
|
**
|
|
*/
|
|
|
|
|
|
#ifndef LIB_DIFF_TREE_MUTATOR_NOOP_BINDING_H
|
|
#define LIB_DIFF_TREE_MUTATOR_NOOP_BINDING_H
|
|
|
|
|
|
#include "lib/error.hpp"
|
|
#include "lib/diff/gen-node.hpp"
|
|
#include "lib/diff/tree-mutator.hpp"
|
|
|
|
#include <type_traits>
|
|
|
|
namespace lib {
|
|
namespace diff{
|
|
|
|
namespace { // Mutator-Builder decorator components...
|
|
|
|
|
|
/**
|
|
* Diff binding for black holes, happily accepting anything.
|
|
* @note absorbs and silently ignores any diff verb.
|
|
*/
|
|
class BlackHoleMutation
|
|
: public TreeMutator
|
|
{
|
|
|
|
public:
|
|
BlackHoleMutation () = default;
|
|
|
|
|
|
/* ==== TreeMutator API ==== */
|
|
|
|
using Elm = GenNode const&;
|
|
using Buff = TreeMutator::Handle;
|
|
|
|
bool hasSrc() override { return true; } ///< always keen to do yet more
|
|
|
|
bool injectNew (Elm) override { return true; } ///< pretend to inject something new
|
|
bool matchSrc (Elm) override { return true; } ///< purport suitable element is waiting
|
|
bool acceptSrc (Elm) override { return true; } ///< claim to handle any diff task
|
|
bool accept_until (Elm) override { return true; } ///< profess to forward anywhere
|
|
bool findSrc (Elm) override { return true; } ///< sham to find anything
|
|
bool assignElm (Elm) override { return true; } ///< accept any assignment
|
|
|
|
bool
|
|
mutateChild (Elm, Buff buff) override ///< ignore inferiors, yet reproduce yourself
|
|
{
|
|
buff.create (BlackHoleMutation());
|
|
return true;
|
|
}
|
|
};
|
|
|
|
|
|
|
|
/** Entry point for DSL builder: create a binding which consumes any diff without effect.
|
|
* @warning must be used as bottom most layer in a custom TreeMutator, since it would
|
|
* otherwise shadow and disable any binding layer below.
|
|
* @note however it is possible to add a (typically selective) binding layer on top;
|
|
* any diff verb _not handled_ by such a top layer will fall through and
|
|
* be silently ignored. Such a setup can be used to "fish" for some specific
|
|
* attributes within a diff stream.
|
|
*/
|
|
template<class PAR>
|
|
inline auto
|
|
Builder<PAR>::ignoreAllChanges()
|
|
{
|
|
static_assert (std::is_same<PAR, TreeMutator>()
|
|
,"ignoreAllChanges() must be used as bottom layer.");
|
|
|
|
return Builder<BlackHoleMutation> (BlackHoleMutation{});
|
|
}
|
|
|
|
|
|
|
|
}//(END)Mutator-Builder decorator components...
|
|
|
|
}} // namespace lib::diff
|
|
#endif /*LIB_DIFF_TREE_MUTATOR_NOOP_BINDING_H*/
|