0
0
Fork 0
mirror of https://github.com/matrix-construct/construct synced 2025-01-17 01:51:53 +01:00
construct/include/ircd/db/cursor.h

328 lines
7.9 KiB
C++

/*
* Copyright (C) 2016 Charybdis Development Team
* Copyright (C) 2016 Jason Volk <jason@zemos.net>
*
* Permission to use, copy, modify, and/or distribute this software for any
* purpose with or without fee is hereby granted, provided that the above
* copyright notice and this permission notice is present in all copies.
*
* THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
* IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
* DISCLAIMED. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT,
* INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
* (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
* SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
* HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
* STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING
* IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
* POSSIBILITY OF SUCH DAMAGE.
*
*/
#pragma once
#define HAVE_IRCD_DB_CURSOR_H
namespace ircd::db
{
template<database *const &d, class tuple> struct cursor;
}
template<ircd::db::database *const &d,
class tuple>
struct ircd::db::cursor
{
template<class index_iterator> struct const_iterator_base;
struct const_reverse_iterator;
struct const_iterator;
template<enum where where = where::noop> using query_type = db::query<tuple, where>;
using iterator_type = const_iterator;
struct index index;
const query_type<> *query{nullptr};
const_iterator end(const string_view &key = {});
const_iterator begin(const string_view &key = {});
const_reverse_iterator rend(const string_view &key = {});
const_reverse_iterator rbegin(const string_view &key = {});
cursor(const string_view &index, const query_type<> *const &query = nullptr)
:index{*d, index}
,query{query}
{}
};
template<ircd::db::database *const &d,
class tuple>
template<class index_iterator>
struct ircd::db::cursor<d, tuple>::const_iterator_base
{
using value_type = const tuple;
using pointer = value_type *;
using reference = value_type &;
using difference_type = size_t;
using iterator_category = std::bidirectional_iterator_tag;
template<enum where where = where::noop> using query_type = cursor::query_type<where>;
const query_type<> *query{nullptr};
index_iterator idx;
std::array<db::cell, tuple::size()> cell;
db::row row;
mutable tuple v;
mutable bool stale{true};
bool invalid{true};
operator bool() const;
bool operator!() const;
bool operator==(const const_iterator_base &o) const;
bool operator!=(const const_iterator_base &o) const;
value_type &operator*() const
{
if(!stale)
return v;
assign(v, row, row_key());
stale = false;
return v;
}
value_type *operator->() const
{
return &this->operator*();
}
string_view row_key() const;
bool row_valid() const;
protected:
bool seek_row();
public:
const_iterator_base &operator++();
const_iterator_base &operator--();
const_iterator_base();
const_iterator_base(const cursor &, index_iterator, const gopts & = {});
};
template<ircd::db::database *const &d,
class tuple>
struct ircd::db::cursor<d, tuple>::const_iterator
:const_iterator_base<index::const_iterator>
{
using const_iterator_base<index::const_iterator>::const_iterator_base;
};
template<ircd::db::database *const &d,
class tuple>
struct ircd::db::cursor<d, tuple>::const_reverse_iterator
:const_iterator_base<index::const_reverse_iterator>
{
using const_iterator_base<index::const_reverse_iterator>::const_iterator_base;
};
//
// cursor
//
template<ircd::db::database *const &d,
class tuple>
typename ircd::db::cursor<d, tuple>::const_reverse_iterator
ircd::db::cursor<d, tuple>::rbegin(const string_view &key)
{
return const_reverse_iterator { *this, index.rbegin(key), {} };
}
template<ircd::db::database *const &d,
class tuple>
typename ircd::db::cursor<d, tuple>::const_reverse_iterator
ircd::db::cursor<d, tuple>::rend(const string_view &key)
{
return const_reverse_iterator { *this, index.rend(key), {} };
}
template<ircd::db::database *const &d,
class tuple>
typename ircd::db::cursor<d, tuple>::const_iterator
ircd::db::cursor<d, tuple>::begin(const string_view &key)
{
return const_iterator { *this, index.begin(key), {} };
}
template<ircd::db::database *const &d,
class tuple>
typename ircd::db::cursor<d, tuple>::const_iterator
ircd::db::cursor<d, tuple>::end(const string_view &key)
{
return const_iterator { *this, index.end(key), {} };
}
//
// const iterator
//
template<ircd::db::database *const &d,
class tuple>
template<class index_iterator>
ircd::db::cursor<d, tuple>::const_iterator_base<index_iterator>::const_iterator_base(const cursor &c,
index_iterator idx,
const gopts &opts)
:query{c.query}
,idx{std::move(idx)}
,cell{}
,row
{
*d,
bool(this->idx) && this->idx->second? this->idx->second:
bool(this->idx)? this->idx->first:
string_view{},
tuple{},
cell,
opts
}
,stale
{
true
}
,invalid
{
!this->idx || !row_valid()
}
{
if(invalid)
return;
if(!this->query)
return;
if(!(*this->query)(this->operator*()))
this->operator++();
}
template<ircd::db::database *const &d,
class tuple>
template<class index_iterator>
ircd::db::cursor<d, tuple>::const_iterator_base<index_iterator> &
ircd::db::cursor<d, tuple>::const_iterator_base<index_iterator>::operator++()
{
while(!(invalid = !bool(++idx)))
if(seek_row())
break;
return *this;
}
template<ircd::db::database *const &d,
class tuple>
template<class index_iterator>
ircd::db::cursor<d, tuple>::const_iterator_base<index_iterator> &
ircd::db::cursor<d, tuple>::const_iterator_base<index_iterator>::operator--()
{
while(!(invalid = !bool(--idx)))
if(seek_row())
break;
return *this;
}
template<ircd::db::database *const &d,
class tuple>
template<class index_iterator>
bool
ircd::db::cursor<d, tuple>::const_iterator_base<index_iterator>::seek_row()
{
if(!db::seek(row, row_key()))
return false;
stale = true;
if(this->query && !(*this->query)(this->operator*()))
return false;
return true;
}
template<ircd::db::database *const &d,
class tuple>
template<class index_iterator>
bool
ircd::db::cursor<d, tuple>::const_iterator_base<index_iterator>::row_valid()
const
{
return row.valid(row_key());
}
template<ircd::db::database *const &d,
class tuple>
template<class index_iterator>
ircd::string_view
ircd::db::cursor<d, tuple>::const_iterator_base<index_iterator>::row_key()
const
{
if(!idx)
return {};
if(idx->second)
return idx->second;
assert(bool(idx->first));
return idx->first;
}
template<ircd::db::database *const &d,
class tuple>
template<class index_iterator>
bool
ircd::db::cursor<d, tuple>::const_iterator_base<index_iterator>::operator!()
const
{
return !static_cast<bool>(*this);
}
template<ircd::db::database *const &d,
class tuple>
template<class index_iterator>
ircd::db::cursor<d, tuple>::const_iterator_base<index_iterator>::operator
bool()
const
{
if(invalid)
return false;
if(!idx)
return false;
return row_valid();
}
template<ircd::db::database *const &d,
class tuple>
template<class index_iterator>
bool
ircd::db::cursor<d, tuple>::const_iterator_base<index_iterator>::operator!=(const const_iterator_base<index_iterator> &o)
const
{
return !(*this == o);
}
template<ircd::db::database *const &d,
class tuple>
template<class index_iterator>
bool
ircd::db::cursor<d, tuple>::const_iterator_base<index_iterator>::operator==(const const_iterator_base<index_iterator> &o)
const
{
if(row_key() != o.row_key())
return false;
if(!row_valid() && !o.row_valid())
return true;
if(!row_valid() || !o.row_valid())
return false;
return true;
}