2018-02-04 03:22:01 +01:00
|
|
|
// Matrix Construct
|
|
|
|
//
|
|
|
|
// Copyright (C) Matrix Construct Developers, Authors & Contributors
|
|
|
|
// Copyright (C) 2016-2018 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. The
|
|
|
|
// full license for this software is available in the LICENSE file.
|
2017-09-08 11:23:11 +02:00
|
|
|
|
|
|
|
#pragma once
|
2017-11-16 02:48:25 +01:00
|
|
|
#define HAVE_IRCD_M_VM_CURSOR_H
|
2017-09-08 11:23:11 +02:00
|
|
|
|
2018-01-26 21:26:09 +01:00
|
|
|
namespace ircd::m::dbs
|
2017-09-08 11:23:11 +02:00
|
|
|
{
|
2017-11-16 02:48:25 +01:00
|
|
|
struct cursor;
|
2017-09-08 11:23:11 +02:00
|
|
|
}
|
|
|
|
|
2018-01-26 21:26:09 +01:00
|
|
|
struct ircd::m::dbs::cursor
|
2017-09-08 11:23:11 +02:00
|
|
|
{
|
2017-09-22 12:31:44 +02:00
|
|
|
template<class index_iterator> struct const_iterator_base;
|
|
|
|
struct const_reverse_iterator;
|
2017-09-08 11:23:11 +02:00
|
|
|
struct const_iterator;
|
|
|
|
|
2018-01-26 21:26:09 +01:00
|
|
|
template<enum where where = where::noop> using query_type = dbs::query<where>;
|
2017-09-08 11:23:11 +02:00
|
|
|
using iterator_type = const_iterator;
|
|
|
|
|
2017-11-16 02:48:25 +01:00
|
|
|
db::index index;
|
2017-09-28 03:24:36 +02:00
|
|
|
const query_type<> *query{nullptr};
|
2017-09-08 11:23:11 +02:00
|
|
|
|
2017-09-26 06:37:32 +02:00
|
|
|
const_iterator end(const string_view &key = {});
|
|
|
|
const_iterator begin(const string_view &key = {});
|
2017-09-08 11:23:11 +02:00
|
|
|
|
2017-09-26 06:37:32 +02:00
|
|
|
const_reverse_iterator rend(const string_view &key = {});
|
|
|
|
const_reverse_iterator rbegin(const string_view &key = {});
|
2017-09-22 12:31:44 +02:00
|
|
|
|
2017-09-28 03:24:36 +02:00
|
|
|
cursor(const string_view &index, const query_type<> *const &query = nullptr)
|
2017-11-16 02:48:25 +01:00
|
|
|
:index{*event::events, index}
|
2017-09-28 03:24:36 +02:00
|
|
|
,query{query}
|
2017-09-08 11:23:11 +02:00
|
|
|
{}
|
|
|
|
};
|
|
|
|
|
2017-09-22 12:31:44 +02:00
|
|
|
template<class index_iterator>
|
2018-01-26 21:26:09 +01:00
|
|
|
struct ircd::m::dbs::cursor::const_iterator_base
|
2017-09-08 11:23:11 +02:00
|
|
|
{
|
2017-11-16 02:48:25 +01:00
|
|
|
using value_type = const event;
|
2017-09-08 11:23:11 +02:00
|
|
|
using pointer = value_type *;
|
|
|
|
using reference = value_type &;
|
|
|
|
using difference_type = size_t;
|
|
|
|
using iterator_category = std::bidirectional_iterator_tag;
|
2017-09-28 03:24:36 +02:00
|
|
|
template<enum where where = where::noop> using query_type = cursor::query_type<where>;
|
2017-09-08 11:23:11 +02:00
|
|
|
|
2017-09-28 03:24:36 +02:00
|
|
|
const query_type<> *query{nullptr};
|
2017-09-08 11:23:11 +02:00
|
|
|
index_iterator idx;
|
2017-11-16 02:48:25 +01:00
|
|
|
std::array<db::cell, event::size()> cell;
|
2017-09-08 11:23:11 +02:00
|
|
|
db::row row;
|
2017-11-16 02:48:25 +01:00
|
|
|
mutable event v;
|
2017-09-08 11:23:11 +02:00
|
|
|
mutable bool stale{true};
|
2017-09-22 12:31:44 +02:00
|
|
|
bool invalid{true};
|
2017-09-08 11:23:11 +02:00
|
|
|
|
2017-09-16 20:48:09 +02:00
|
|
|
operator bool() const;
|
|
|
|
bool operator!() const;
|
2017-09-08 11:23:11 +02:00
|
|
|
|
2017-09-22 12:31:44 +02:00
|
|
|
bool operator==(const const_iterator_base &o) const;
|
|
|
|
bool operator!=(const const_iterator_base &o) const;
|
2017-09-08 11:23:11 +02:00
|
|
|
|
2017-09-22 12:31:44 +02:00
|
|
|
value_type &operator*() const
|
|
|
|
{
|
|
|
|
if(!stale)
|
|
|
|
return v;
|
2017-09-08 11:23:11 +02:00
|
|
|
|
2017-09-25 02:59:23 +02:00
|
|
|
assign(v, row, row_key());
|
2017-09-22 12:31:44 +02:00
|
|
|
stale = false;
|
|
|
|
return v;
|
|
|
|
}
|
|
|
|
|
|
|
|
value_type *operator->() const
|
|
|
|
{
|
|
|
|
return &this->operator*();
|
|
|
|
}
|
|
|
|
|
2017-09-25 02:59:23 +02:00
|
|
|
string_view row_key() const;
|
|
|
|
bool row_valid() const;
|
2017-09-26 07:13:11 +02:00
|
|
|
|
|
|
|
protected:
|
2017-09-22 12:31:44 +02:00
|
|
|
bool seek_row();
|
2017-09-08 11:23:11 +02:00
|
|
|
|
|
|
|
public:
|
2017-09-22 12:31:44 +02:00
|
|
|
const_iterator_base &operator++();
|
|
|
|
const_iterator_base &operator--();
|
2017-09-08 11:23:11 +02:00
|
|
|
|
2017-09-22 12:31:44 +02:00
|
|
|
const_iterator_base();
|
2017-11-16 02:48:25 +01:00
|
|
|
const_iterator_base(const cursor &, index_iterator, const db::gopts & = {});
|
2017-09-22 12:31:44 +02:00
|
|
|
};
|
|
|
|
|
2018-01-26 21:26:09 +01:00
|
|
|
struct ircd::m::dbs::cursor::const_iterator
|
2017-11-16 02:48:25 +01:00
|
|
|
:const_iterator_base<db::index::const_iterator>
|
2017-09-22 12:31:44 +02:00
|
|
|
{
|
2017-11-16 02:48:25 +01:00
|
|
|
using const_iterator_base<db::index::const_iterator>::const_iterator_base;
|
2017-09-22 12:31:44 +02:00
|
|
|
};
|
|
|
|
|
2018-01-26 21:26:09 +01:00
|
|
|
struct ircd::m::dbs::cursor::const_reverse_iterator
|
2017-11-16 02:48:25 +01:00
|
|
|
:const_iterator_base<db::index::const_reverse_iterator>
|
2017-09-22 12:31:44 +02:00
|
|
|
{
|
2017-11-16 02:48:25 +01:00
|
|
|
using const_iterator_base<db::index::const_reverse_iterator>::const_iterator_base;
|
2017-09-08 11:23:11 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
//
|
|
|
|
// cursor
|
|
|
|
//
|
|
|
|
|
2018-01-26 21:26:09 +01:00
|
|
|
inline ircd::m::dbs::cursor::const_reverse_iterator
|
|
|
|
ircd::m::dbs::cursor::rbegin(const string_view &key)
|
2017-09-08 11:23:11 +02:00
|
|
|
{
|
2017-09-22 12:31:44 +02:00
|
|
|
return const_reverse_iterator { *this, index.rbegin(key), {} };
|
2017-09-08 11:23:11 +02:00
|
|
|
}
|
|
|
|
|
2018-01-26 21:26:09 +01:00
|
|
|
inline ircd::m::dbs::cursor::const_reverse_iterator
|
|
|
|
ircd::m::dbs::cursor::rend(const string_view &key)
|
2017-09-08 11:23:11 +02:00
|
|
|
{
|
2017-09-22 12:31:44 +02:00
|
|
|
return const_reverse_iterator { *this, index.rend(key), {} };
|
2017-09-08 11:23:11 +02:00
|
|
|
}
|
|
|
|
|
2018-01-26 21:26:09 +01:00
|
|
|
inline ircd::m::dbs::cursor::const_iterator
|
|
|
|
ircd::m::dbs::cursor::begin(const string_view &key)
|
2017-09-08 11:23:11 +02:00
|
|
|
{
|
2017-09-22 12:31:44 +02:00
|
|
|
return const_iterator { *this, index.begin(key), {} };
|
2017-09-08 11:23:11 +02:00
|
|
|
}
|
|
|
|
|
2018-01-26 21:26:09 +01:00
|
|
|
inline ircd::m::dbs::cursor::const_iterator
|
|
|
|
ircd::m::dbs::cursor::end(const string_view &key)
|
2017-09-08 11:23:11 +02:00
|
|
|
{
|
2017-09-22 12:31:44 +02:00
|
|
|
return const_iterator { *this, index.end(key), {} };
|
2017-09-08 11:23:11 +02:00
|
|
|
}
|
|
|
|
|
2017-09-22 12:31:44 +02:00
|
|
|
//
|
|
|
|
// const iterator
|
|
|
|
//
|
|
|
|
|
|
|
|
template<class index_iterator>
|
2018-01-26 21:26:09 +01:00
|
|
|
ircd::m::dbs::cursor::const_iterator_base<index_iterator>::const_iterator_base(const cursor &c,
|
2017-11-16 02:48:25 +01:00
|
|
|
index_iterator idx,
|
|
|
|
const db::gopts &opts)
|
2017-09-28 03:24:36 +02:00
|
|
|
:query{c.query}
|
2017-09-08 11:23:11 +02:00
|
|
|
,idx{std::move(idx)}
|
2017-09-29 06:19:02 +02:00
|
|
|
,cell{}
|
2017-09-08 11:23:11 +02:00
|
|
|
,row
|
|
|
|
{
|
2017-11-16 02:48:25 +01:00
|
|
|
*event::events,
|
2017-09-25 02:59:23 +02:00
|
|
|
bool(this->idx) && this->idx->second? this->idx->second:
|
|
|
|
bool(this->idx)? this->idx->first:
|
|
|
|
string_view{},
|
2017-11-16 02:48:25 +01:00
|
|
|
event{},
|
2017-09-29 06:19:02 +02:00
|
|
|
cell,
|
2017-09-25 02:59:23 +02:00
|
|
|
opts
|
|
|
|
}
|
|
|
|
,stale
|
|
|
|
{
|
|
|
|
true
|
2017-09-08 11:23:11 +02:00
|
|
|
}
|
|
|
|
,invalid
|
|
|
|
{
|
2017-09-25 02:59:23 +02:00
|
|
|
!this->idx || !row_valid()
|
2017-09-08 11:23:11 +02:00
|
|
|
}
|
|
|
|
{
|
2017-09-22 01:59:58 +02:00
|
|
|
if(invalid)
|
|
|
|
return;
|
|
|
|
|
2017-09-28 03:24:36 +02:00
|
|
|
if(!this->query)
|
2017-09-22 01:59:58 +02:00
|
|
|
return;
|
|
|
|
|
2017-09-28 03:24:36 +02:00
|
|
|
if(!(*this->query)(this->operator*()))
|
2017-09-22 12:31:44 +02:00
|
|
|
this->operator++();
|
2017-09-08 11:23:11 +02:00
|
|
|
}
|
|
|
|
|
2017-09-22 12:31:44 +02:00
|
|
|
template<class index_iterator>
|
2018-01-26 21:26:09 +01:00
|
|
|
ircd::m::dbs::cursor::const_iterator_base<index_iterator> &
|
|
|
|
ircd::m::dbs::cursor::const_iterator_base<index_iterator>::operator++()
|
2017-09-08 11:23:11 +02:00
|
|
|
{
|
2017-09-22 12:31:44 +02:00
|
|
|
while(!(invalid = !bool(++idx)))
|
|
|
|
if(seek_row())
|
|
|
|
break;
|
2017-09-08 11:23:11 +02:00
|
|
|
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
2017-09-22 12:31:44 +02:00
|
|
|
template<class index_iterator>
|
2018-01-26 21:26:09 +01:00
|
|
|
ircd::m::dbs::cursor::const_iterator_base<index_iterator> &
|
|
|
|
ircd::m::dbs::cursor::const_iterator_base<index_iterator>::operator--()
|
2017-09-08 11:23:11 +02:00
|
|
|
{
|
2017-09-22 12:31:44 +02:00
|
|
|
while(!(invalid = !bool(--idx)))
|
|
|
|
if(seek_row())
|
|
|
|
break;
|
2017-09-08 11:23:11 +02:00
|
|
|
|
2017-09-22 12:31:44 +02:00
|
|
|
return *this;
|
2017-09-08 11:23:11 +02:00
|
|
|
}
|
|
|
|
|
2017-09-22 12:31:44 +02:00
|
|
|
template<class index_iterator>
|
|
|
|
bool
|
2018-01-26 21:26:09 +01:00
|
|
|
ircd::m::dbs::cursor::const_iterator_base<index_iterator>::seek_row()
|
2017-09-08 11:23:11 +02:00
|
|
|
{
|
2017-09-25 02:59:23 +02:00
|
|
|
if(!db::seek(row, row_key()))
|
2017-09-22 12:31:44 +02:00
|
|
|
return false;
|
2017-09-08 11:23:11 +02:00
|
|
|
|
2017-09-22 12:31:44 +02:00
|
|
|
stale = true;
|
2017-09-28 03:24:36 +02:00
|
|
|
if(this->query && !(*this->query)(this->operator*()))
|
2017-09-22 12:31:44 +02:00
|
|
|
return false;
|
2017-09-08 11:23:11 +02:00
|
|
|
|
2017-09-22 12:31:44 +02:00
|
|
|
return true;
|
2017-09-08 11:23:11 +02:00
|
|
|
}
|
|
|
|
|
2017-09-25 02:59:23 +02:00
|
|
|
template<class index_iterator>
|
|
|
|
bool
|
2018-01-26 21:26:09 +01:00
|
|
|
ircd::m::dbs::cursor::const_iterator_base<index_iterator>::row_valid()
|
2017-09-25 02:59:23 +02:00
|
|
|
const
|
|
|
|
{
|
|
|
|
return row.valid(row_key());
|
|
|
|
}
|
|
|
|
|
|
|
|
template<class index_iterator>
|
|
|
|
ircd::string_view
|
2018-01-26 21:26:09 +01:00
|
|
|
ircd::m::dbs::cursor::const_iterator_base<index_iterator>::row_key()
|
2017-09-25 02:59:23 +02:00
|
|
|
const
|
|
|
|
{
|
|
|
|
if(!idx)
|
|
|
|
return {};
|
|
|
|
|
|
|
|
if(idx->second)
|
|
|
|
return idx->second;
|
|
|
|
|
|
|
|
assert(bool(idx->first));
|
|
|
|
return idx->first;
|
|
|
|
}
|
|
|
|
|
2017-09-22 12:31:44 +02:00
|
|
|
template<class index_iterator>
|
2017-09-16 20:48:09 +02:00
|
|
|
bool
|
2018-01-26 21:26:09 +01:00
|
|
|
ircd::m::dbs::cursor::const_iterator_base<index_iterator>::operator!()
|
2017-09-16 20:48:09 +02:00
|
|
|
const
|
|
|
|
{
|
|
|
|
return !static_cast<bool>(*this);
|
|
|
|
}
|
|
|
|
|
2017-09-22 12:31:44 +02:00
|
|
|
template<class index_iterator>
|
2018-01-26 21:26:09 +01:00
|
|
|
ircd::m::dbs::cursor::const_iterator_base<index_iterator>::operator
|
2017-09-16 20:48:09 +02:00
|
|
|
bool()
|
|
|
|
const
|
|
|
|
{
|
|
|
|
if(invalid)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
if(!idx)
|
|
|
|
return false;
|
|
|
|
|
2017-09-25 02:59:23 +02:00
|
|
|
return row_valid();
|
2017-09-16 20:48:09 +02:00
|
|
|
}
|
|
|
|
|
2017-09-22 12:31:44 +02:00
|
|
|
template<class index_iterator>
|
2017-09-08 11:23:11 +02:00
|
|
|
bool
|
2018-01-26 21:26:09 +01:00
|
|
|
ircd::m::dbs::cursor::const_iterator_base<index_iterator>::operator!=(const const_iterator_base<index_iterator> &o)
|
2017-09-08 11:23:11 +02:00
|
|
|
const
|
|
|
|
{
|
|
|
|
return !(*this == o);
|
|
|
|
}
|
|
|
|
|
2017-09-22 12:31:44 +02:00
|
|
|
template<class index_iterator>
|
2017-09-08 11:23:11 +02:00
|
|
|
bool
|
2018-01-26 21:26:09 +01:00
|
|
|
ircd::m::dbs::cursor::const_iterator_base<index_iterator>::operator==(const const_iterator_base<index_iterator> &o)
|
2017-09-08 11:23:11 +02:00
|
|
|
const
|
|
|
|
{
|
2017-09-25 02:59:23 +02:00
|
|
|
if(row_key() != o.row_key())
|
|
|
|
return false;
|
|
|
|
|
|
|
|
if(!row_valid() && !o.row_valid())
|
2017-09-08 11:23:11 +02:00
|
|
|
return true;
|
|
|
|
|
2017-09-25 02:59:23 +02:00
|
|
|
if(!row_valid() || !o.row_valid())
|
2017-09-08 11:23:11 +02:00
|
|
|
return false;
|
|
|
|
|
2017-09-25 02:59:23 +02:00
|
|
|
return true;
|
2017-09-08 11:23:11 +02:00
|
|
|
}
|