2015-10-23 19:24:34 +02:00
|
|
|
/*
|
|
|
|
|
RECORD-CONTENT-MUTATOR.hpp - helper to remould record contents
|
|
|
|
|
|
|
|
|
|
Copyright (C) Lumiera.org
|
|
|
|
|
2015, 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 record-content-mutator.hpp
|
|
|
|
|
** Implementation helper for reshaping the contents of a lib::diff::Record.
|
|
|
|
|
** This technical helper is necessary to apply one level of a "Tree Diff"
|
|
|
|
|
** to an object represented as #Record::Mutator. Since records as such are
|
|
|
|
|
** designed as immutable value objects, we build a dedicated #Record::Mutator
|
|
|
|
|
** when it comes to reordering the contents of a given record. The technical
|
|
|
|
|
** details of doing so are highly coupled to the actual storage implementation
|
|
|
|
|
** of #Record, as well as to the actual procedure to apply a diff message, as
|
|
|
|
|
** implemented in lib::diff::DiffApplicationStrategy.
|
|
|
|
|
**
|
|
|
|
|
** @see generic-record-update-test.cpp
|
|
|
|
|
** @see tree-diff-application-test.cpp
|
|
|
|
|
** @see Record::Mutator
|
|
|
|
|
** @see diff-language.hpp
|
|
|
|
|
**
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
#ifndef LIB_DIFF_RECORD_CONTENT_MUTATOR_H
|
|
|
|
|
#define LIB_DIFF_RECORD_CONTENT_MUTATOR_H
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
#include "lib/error.hpp"
|
|
|
|
|
|
|
|
|
|
#include <boost/noncopyable.hpp>
|
|
|
|
|
#include <utility>
|
|
|
|
|
|
|
|
|
|
namespace lib {
|
|
|
|
|
namespace diff{
|
|
|
|
|
|
2015-10-24 01:49:07 +02:00
|
|
|
namespace error = lumiera::error;
|
2015-10-23 19:24:34 +02:00
|
|
|
using std::move;
|
|
|
|
|
using std::swap;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/** @internal helper for DiffApplicationStrategy<Rec::Mutator> */
|
|
|
|
|
template<class STO>
|
2015-10-23 20:55:02 +02:00
|
|
|
struct RecordContentMutator
|
2015-10-23 19:24:34 +02:00
|
|
|
: boost::noncopyable
|
|
|
|
|
{
|
2015-10-23 20:55:02 +02:00
|
|
|
using Iter = typename STO::iterator;
|
|
|
|
|
|
|
|
|
|
STO attribs;
|
|
|
|
|
STO children;
|
|
|
|
|
Iter pos;
|
|
|
|
|
|
2015-10-23 19:24:34 +02:00
|
|
|
RecordContentMutator()
|
2015-10-23 20:55:02 +02:00
|
|
|
: attribs()
|
|
|
|
|
, children()
|
|
|
|
|
, pos(attribs.begin())
|
|
|
|
|
{ }
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
bool
|
|
|
|
|
empty() const
|
2015-10-23 19:24:34 +02:00
|
|
|
{
|
2015-10-23 20:55:02 +02:00
|
|
|
return attribs.empty()
|
2015-10-24 01:49:07 +02:00
|
|
|
and children.empty();
|
2015-10-23 20:55:02 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool
|
|
|
|
|
currIsAttrib() const
|
|
|
|
|
{
|
2015-10-24 01:49:07 +02:00
|
|
|
return & *pos >= & *attribs.begin()
|
|
|
|
|
and & *pos < & *attribs.end();
|
2015-10-23 20:55:02 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool
|
|
|
|
|
currIsChild() const
|
|
|
|
|
{
|
2015-10-24 01:49:07 +02:00
|
|
|
return & *pos >= & *children.begin()
|
|
|
|
|
and & *pos < & *children.end();
|
2015-10-23 20:55:02 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Iter end() { return children.end(); }
|
|
|
|
|
Iter end() const { return children.end(); }
|
|
|
|
|
|
|
|
|
|
RecordContentMutator&
|
|
|
|
|
operator++()
|
|
|
|
|
{
|
2015-10-24 01:49:07 +02:00
|
|
|
if (pos == children.end())
|
|
|
|
|
throw error::State ("attempt to iterate beyond end of scope"
|
|
|
|
|
,error::LUMIERA_ERROR_ITER_EXHAUST);
|
|
|
|
|
if (pos == attribs.end())
|
|
|
|
|
pos = children.begin();
|
|
|
|
|
else
|
|
|
|
|
++pos;
|
|
|
|
|
if (pos == attribs.end())
|
|
|
|
|
pos = children.begin();
|
|
|
|
|
return *this;
|
2015-10-23 20:55:02 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
resetPos()
|
|
|
|
|
{
|
|
|
|
|
pos = attribs.begin();
|
2015-10-23 19:24:34 +02:00
|
|
|
}
|
2015-10-23 21:08:33 +02:00
|
|
|
|
2015-10-24 03:15:35 +02:00
|
|
|
void
|
|
|
|
|
jumpToChildScope()
|
|
|
|
|
{
|
|
|
|
|
pos = children.begin();
|
|
|
|
|
}
|
|
|
|
|
|
2015-10-23 21:08:33 +02:00
|
|
|
void
|
|
|
|
|
preAllocateStorage(size_t attribCnt, size_t childrenCnt)
|
|
|
|
|
{
|
|
|
|
|
// heuristics for storage pre-allocation (for tree diff application)
|
|
|
|
|
attribs.reserve (attribCnt * 120 / 100);
|
|
|
|
|
children.reserve (childrenCnt * 120 / 100);
|
|
|
|
|
ASSERT (this->empty());
|
|
|
|
|
}
|
2015-10-23 19:24:34 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
}} // namespace lib::diff
|
|
|
|
|
#endif /*LIB_DIFF_RECORD_CONTENT_MUTATOR_H*/
|