Branch data Line data Source code
1 : : /***********************************************************************\ 2 : : * gnc-sql-result.hpp: Encapsulate the results of a SQL query. * 3 : : * * 4 : : * Copyright 2016 John Ralls <jralls@ceridwen.us> * 5 : : * * 6 : : * This program is free software; you can redistribute it and/or * 7 : : * modify it under the terms of the GNU General Public License as * 8 : : * published by the Free Software Foundation; either version 2 of * 9 : : * the License, or (at your option) any later version. * 10 : : * * 11 : : * This program is distributed in the hope that it will be useful, * 12 : : * but WITHOUT ANY WARRANTY; without even the implied warranty of * 13 : : * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * 14 : : * GNU General Public License for more details. * 15 : : * * 16 : : * You should have received a copy of the GNU General Public License * 17 : : * along with this program; if not, contact: * 18 : : * * 19 : : * Free Software Foundation Voice: +1-617-542-5942 * 20 : : * 51 Franklin Street, Fifth Floor Fax: +1-617-542-2652 * 21 : : * Boston, MA 02110-1301, USA gnu@gnu.org * 22 : : \***********************************************************************/ 23 : : 24 : : #ifndef __GNC_SQL_RESULT_HPP__ 25 : : #define __GNC_SQL_RESULT_HPP__ 26 : : 27 : : #include <qof.h> 28 : : 29 : : #include <cstdint> 30 : : #include <optional> 31 : : #include <string> 32 : : #include <vector> 33 : : 34 : : class GncSqlRow; 35 : : /** 36 : : * Pure virtual class to iterate over a query result set. 37 : : */ 38 : : 39 : : class GncSqlResult 40 : : { 41 : : public: 42 : 0 : virtual ~GncSqlResult() = default; 43 : : virtual uint64_t size() const noexcept = 0; 44 : : virtual GncSqlRow& begin() = 0; 45 : : virtual GncSqlRow& end() = 0; 46 : : friend GncSqlRow; 47 : : protected: 48 : : class IteratorImpl { 49 : : public: 50 : 0 : virtual ~IteratorImpl() = default; 51 : : virtual GncSqlRow& operator++() = 0; 52 : : virtual GncSqlResult* operator*() = 0; 53 : : virtual std::optional<int64_t> get_int_at_col (const char* col) const = 0; 54 : : virtual std::optional<double> get_float_at_col (const char* col) const = 0; 55 : : virtual std::optional<double> get_double_at_col (const char* col) const = 0; 56 : : virtual std::optional<std::string> get_string_at_col (const char* col) const = 0; 57 : : virtual std::optional<time64> get_time64_at_col (const char* col) const = 0; 58 : : virtual bool is_col_null (const char* col) const noexcept = 0; 59 : : }; 60 : : }; 61 : : 62 : : /** 63 : : * Row of SQL Query results. 64 : : * 65 : : * This is a "pointer" class of a pimpl pattern, the implementation being 66 : : * GncSqlResult::IteratorImpl. It's designed to present a std::forward_iterator 67 : : * like interface for use with range-for while allowing for wrapping a C API. 68 : : * 69 : : * Important Implementation Note: Operator++() as written requires that the 70 : : * sentinel GncSqlRow returned by GncSqlResult::end() has m_impl = nullptr in 71 : : * order to terminate the loop condition. This is a bit of a hack and might be a 72 : : * problem with a different SQL interface library from libdbi. 73 : : * 74 : : * Note that GncSqlResult::begin and GncSqlRow::operator++() return 75 : : * GncSqlRow&. This is necessary for operator++() to be called: Using operator 76 : : * ++() on a pointer performs pointer arithmetic rather than calling the 77 : : * pointed-to-class's operator++() and C++'s range-for uses operator++() 78 : : * directly on whatever begin() gives it. 79 : : */ 80 : : class GncSqlRow 81 : : { 82 : : public: 83 : 0 : GncSqlRow (GncSqlResult::IteratorImpl* iter) : m_iter{iter} {} 84 : 0 : ~GncSqlRow() { } 85 : : GncSqlRow& operator++(); 86 : 275 : GncSqlRow& operator*() { return *this; } 87 : : friend bool operator!=(const GncSqlRow&, const GncSqlRow&); 88 : 778 : std::optional<int64_t> get_int_at_col (const char* col) const { 89 : 778 : return m_iter->get_int_at_col (col); } 90 : 58 : std::optional<double> get_float_at_col (const char* col) const { 91 : 58 : return m_iter->get_float_at_col (col); } 92 : 52 : std::optional<double> get_double_at_col (const char* col) const { 93 : 52 : return m_iter->get_double_at_col (col); } 94 : 1175 : std::optional<std::string> get_string_at_col (const char* col) const { 95 : 1175 : return m_iter->get_string_at_col (col); } 96 : 0 : std::optional<time64> get_time64_at_col (const char* col) const { 97 : 0 : return m_iter->get_time64_at_col (col); } 98 : 71 : bool is_col_null (const char* col) const noexcept { 99 : 71 : return m_iter->is_col_null (col); } 100 : : private: 101 : : GncSqlResult::IteratorImpl* m_iter; 102 : : }; 103 : : 104 : 745 : inline bool operator!=(const GncSqlRow& lr, const GncSqlRow& rr) { 105 : 745 : return lr.m_iter != rr.m_iter; 106 : : } 107 : : 108 : 26 : inline bool operator==(const GncSqlRow& lr, const GncSqlRow& rr) { 109 : 26 : return !(lr != rr); 110 : : } 111 : : 112 : : 113 : : #endif //__GNC_SQL_RESULT_HPP__