/* Copyright (C) 2007-2013 Arjen G Lentz & Antony T Curtis for Open Query 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; 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ /* ====================================================================== Open Query Graph Computation Engine, based on a concept by Arjen Lentz v3 implementation by Antony Curtis, Arjen Lentz, Andrew McDonnell For more information, documentation, support, enhancement engineering, see http://openquery.com/graph or contact graph@openquery.com ====================================================================== */ #pragma once #include #include #include #include #include "graphcore-config.h" #include #include #include #include #include #include #include "graphcore-types.h" namespace oqgraph3 { typedef open_query::VertexID vertex_id; typedef open_query::EdgeWeight weight_t; typedef size_t vertices_size_type; typedef size_t edges_size_type; typedef size_t degree_size_type; struct graph; struct cursor; typedef boost::intrusive_ptr graph_ptr; struct cursor_ptr : public boost::intrusive_ptr { cursor_ptr() : boost::intrusive_ptr() { } cursor_ptr(cursor* pos) : boost::intrusive_ptr(pos) { } operator const std::string&() const; bool operator==(const cursor_ptr&) const; bool operator!=(const cursor_ptr&) const; }; struct edge_info { cursor_ptr _cursor; edge_info() : _cursor(0) { } explicit edge_info(const cursor_ptr& pos) : _cursor(pos) { } edge_info& operator=(const cursor_ptr& pos) { _cursor= pos; return *this; } vertex_id origid() const; vertex_id destid() const; weight_t weight() const; bool operator==(const edge_info&) const; bool operator!=(const edge_info&) const; }; struct cursor { mutable int _ref_count; graph_ptr _graph; int _index; unsigned _parts; std::string _key; std::string _position; int _debugid; boost::optional _origid; boost::optional _destid; cursor(const graph_ptr& graph); cursor(const cursor& src); ~cursor(); operator bool() const { return !_position.empty(); } operator edge_info() const { return edge_info(const_cast(this)); } vertex_id get_origid(); vertex_id get_destid(); weight_t get_weight(); int seek_to( boost::optional origid, boost::optional destid); int seek_next(); int seek_prev(); void save_position(); int restore_position(); const std::string& record_position() const; void clear_position(); int clear_position(int rc) { clear_position(); return rc; } bool operator==(const cursor& x) const; bool operator!=(const cursor& x) const; friend void intrusive_ptr_add_ref(cursor* ptr) { ++ptr->_ref_count; } friend void intrusive_ptr_release(cursor* ptr) { if (!--(ptr->_ref_count)) delete ptr; } }; struct graph { mutable int _ref_count; cursor* _cursor; bool _stale; cursor_ptr _rnd_cursor; size_t _rnd_pos; ::TABLE* _table; ::Field* _source; ::Field* _target; ::Field* _weight; graph( ::TABLE* table, ::Field* source, ::Field* target, ::Field* weight= 0); ~graph(); edges_size_type num_edges() const; friend edges_size_type num_edges(const graph& g) { return g.num_edges(); } friend void intrusive_ptr_add_ref(graph* ptr) { ptr->_ref_count++; } friend void intrusive_ptr_release(graph* ptr) { ptr->_ref_count--; } }; }