Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Adds an api to collect results into direct object and/or its stl-container #122

Open
wants to merge 2 commits into
base: master
Choose a base branch
from
Open
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
82 changes: 82 additions & 0 deletions hdr/collectors.h
Original file line number Diff line number Diff line change
@@ -0,0 +1,82 @@

#ifndef SQLITE_COLLECTORS_H
#define SQLITE_COLLECTORS_H

#include <sqlite_modern_cpp.h>

#if __cplusplus > 201402
#define CPP14_SUPPORTED
#endif

#ifdef CPP14_SUPPORTED
#define __COLLECTOR_RETURN_TYPE auto
#define __COLLECTOR_CTOR_OF_T(T,X) {X}
#else
#define __COLLECTOR_RETURN_TYPE std::function<void(Args...)>
#define __COLLECTOR_CTOR_OF_T(T,X) T(X)

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Don't these names violate the standard?

#endif

template <typename ...Args>
struct collect
{
template<typename T, template<class, class...> class C>
inline static
__COLLECTOR_RETURN_TYPE
to(C<T> &container)
{
return [&container](const Args&... args)
{
container.push_back(__COLLECTOR_CTOR_OF_T(T,args...));
};
}

template<typename T, template<class, class...> class C>
inline static
__COLLECTOR_RETURN_TYPE
to(C<std::shared_ptr<T>> &container)
{
return [&container](const Args&... args)
{
container.push_back(std::make_shared<T>(args...));
};
}

template<template<class, class...> class C, typename T >
struct as
{
friend C<T> operator>>(sqlite::database_binder& db_binder, as<C,T>)
{
C<T> container;
db_binder>>to(container);
return container;
}

friend C<T> operator>>(sqlite::database_binder&& db_binder, as<C,T>)
{
return db_binder>>as<C,T>();
}
};

template<typename T >
struct as<std::shared_ptr,T>
{
friend std::shared_ptr<T> operator>>(sqlite::database_binder& db_binder, as<std::shared_ptr,T>)
{
std::shared_ptr<T> objP;
db_binder>>[&objP](Args... args)
{
objP = std::make_shared<T>(args...);
};
return objP;
}

friend std::shared_ptr<T> operator>>(sqlite::database_binder&& db_binder, as<std::shared_ptr,T> whatever)
{
return db_binder>>whatever;
}

};
};


#endif
88 changes: 88 additions & 0 deletions tests/collector.cc
Original file line number Diff line number Diff line change
@@ -0,0 +1,88 @@

#include "collectors.h"
#include <list>
#include <algorithm>

using namespace std;
using namespace sqlite;

struct user
{
int id;
string name;

user(int _id, string _name):id(_id),name(_name)
{
clog<<"object created : "<<*this<<"\n";
}

friend std::ostream& operator <<(std::ostream& out, const user& a)
{
out<<"<"<<a.id<<","<<a.name<<">";
return out;
}
};


database init_db()
{
database db(":memory:");
db << "CREATE TABLE tbl (id integer, name string);";
db << "INSERT INTO tbl VALUES (?, ?);" << 1 << "hello";
db << "INSERT INTO tbl VALUES (?, ?);" << 2 << "world";

return db;
}


void test_single_user(database &db)
{
auto may_be_user = db<<"SELECT * FROM tbl WHERE id = 1 ;"
>> collect<int,string>::as<shared_ptr,user>();
assert(may_be_user);
assert(may_be_user->name == "hello");
}

void test_multiple_user(database &db)
{
auto all_users = db<<"SELECT * FROM tbl ;"
>> collect<int,string>::as<list,user>();
assert(not all_users.empty());
assert(any_of(all_users.begin(), all_users.end(), [](user u)
{
return u.name == "hello";
}));
}

void test_multiple_usernames(database &db)
{
auto names = db<<"SELECT name FROM tbl ORDER BY id DESC;"
>> collect<string>::as<vector,string>();
assert(not names.empty());
assert(names[0] == "world");
assert(names[1] == "hello");
}

void test_append_results_to_container(database &db)
{
list<int> ids = {-1,0,11};

db<<"SELECT id FROM tbl ;"
>>collect<int>::to(ids);

list<int> expected = { -1,0,11, 1, 2};
assert(ids == expected);
}

int main()
{
database db=init_db();

test_single_user(db);
test_multiple_user(db);
test_multiple_usernames(db);

test_append_results_to_container(db);

return 0;
}