2015-07-10 21:37:22 +00:00
|
|
|
#include "ClientDataModel.hpp"
|
|
|
|
|
|
2015-07-13 17:56:30 +00:00
|
|
|
#include <graphene/app/api.hpp>
|
|
|
|
|
#include <graphene/chain/protocol/protocol.hpp>
|
|
|
|
|
|
|
|
|
|
#include <fc/rpc/websocket_api.hpp>
|
|
|
|
|
|
2015-07-13 18:14:58 +00:00
|
|
|
using namespace graphene::app;
|
|
|
|
|
|
2015-07-13 17:56:30 +00:00
|
|
|
ChainDataModel::ChainDataModel( fc::thread& t, QObject* parent )
|
|
|
|
|
:QObject(parent),m_thread(&t){}
|
2015-07-10 21:37:22 +00:00
|
|
|
|
2015-07-13 20:21:01 +00:00
|
|
|
Account* ChainDataModel::getAccount(qint64 id)
|
2015-07-10 21:37:22 +00:00
|
|
|
{
|
2015-07-14 13:46:38 +00:00
|
|
|
auto& by_id_idx = m_accounts.get<::by_id>();
|
|
|
|
|
auto itr = by_id_idx.find(id);
|
|
|
|
|
if( itr == by_id_idx.end() )
|
|
|
|
|
{
|
|
|
|
|
auto tmp = new Account;
|
|
|
|
|
tmp->id = id; --m_account_query_num;
|
|
|
|
|
tmp->name = QString::number( --m_account_query_num);
|
|
|
|
|
auto result = m_accounts.insert( tmp );
|
|
|
|
|
assert( result.second );
|
|
|
|
|
|
|
|
|
|
/** execute in app thread */
|
|
|
|
|
m_thread->async( [this,id](){
|
|
|
|
|
try {
|
|
|
|
|
ilog( "look up names.." );
|
|
|
|
|
auto result = m_db_api->get_accounts( {account_id_type(id)} );
|
|
|
|
|
/** execute in main */
|
|
|
|
|
Q_EMIT queueExecute( [this,result,id](){
|
|
|
|
|
wlog( "process result" );
|
|
|
|
|
auto& by_id_idx = this->m_accounts.get<::by_id>();
|
|
|
|
|
auto itr = by_id_idx.find(id);
|
|
|
|
|
assert( itr != by_id_idx.end() );
|
|
|
|
|
|
|
|
|
|
if( result.size() == 0 || !result.front() )
|
|
|
|
|
{
|
|
|
|
|
elog( "delete later" );
|
|
|
|
|
(*itr)->deleteLater();
|
|
|
|
|
by_id_idx.erase( itr );
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
by_id_idx.modify( itr,
|
|
|
|
|
[=]( Account* a ){
|
|
|
|
|
a->setProperty("name", QString::fromStdString(result.front()->name) );
|
|
|
|
|
}
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
catch ( const fc::exception& e )
|
|
|
|
|
{
|
|
|
|
|
Q_EMIT exceptionThrown( QString::fromStdString(e.to_string()) );
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
return *result.first;
|
|
|
|
|
}
|
|
|
|
|
return *itr;
|
2015-07-10 21:37:22 +00:00
|
|
|
}
|
|
|
|
|
|
2015-07-13 20:21:01 +00:00
|
|
|
Account* ChainDataModel::getAccount(QString name)
|
2015-07-10 21:37:22 +00:00
|
|
|
{
|
2015-07-14 13:46:38 +00:00
|
|
|
auto& by_name_idx = m_accounts.get<by_account_name>();
|
|
|
|
|
auto itr = by_name_idx.find(name);
|
|
|
|
|
if( itr == by_name_idx.end() )
|
2015-07-13 21:24:25 +00:00
|
|
|
{
|
2015-07-14 13:46:38 +00:00
|
|
|
auto tmp = new Account;
|
|
|
|
|
tmp->id = --m_account_query_num;
|
|
|
|
|
tmp->name = name;
|
|
|
|
|
auto result = m_accounts.insert( tmp );
|
|
|
|
|
assert( result.second );
|
|
|
|
|
|
|
|
|
|
/** execute in app thread */
|
|
|
|
|
m_thread->async( [this,name](){
|
|
|
|
|
try {
|
|
|
|
|
ilog( "look up names.." );
|
|
|
|
|
auto result = m_db_api->lookup_account_names( {name.toStdString()} );
|
|
|
|
|
/** execute in main */
|
|
|
|
|
Q_EMIT queueExecute( [this,result,name](){
|
|
|
|
|
wlog( "process result" );
|
|
|
|
|
auto& by_name_idx = this->m_accounts.get<by_account_name>();
|
|
|
|
|
auto itr = by_name_idx.find(name);
|
|
|
|
|
assert( itr != by_name_idx.end() );
|
|
|
|
|
|
|
|
|
|
if( result.size() == 0 || !result.front() )
|
|
|
|
|
{
|
|
|
|
|
elog( "delete later" );
|
|
|
|
|
(*itr)->deleteLater();
|
|
|
|
|
by_name_idx.erase( itr );
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
by_name_idx.modify( itr,
|
|
|
|
|
[=]( Account* a ){
|
|
|
|
|
a->setProperty("id", result.front()->id.instance() );
|
|
|
|
|
}
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
catch ( const fc::exception& e )
|
|
|
|
|
{
|
|
|
|
|
Q_EMIT exceptionThrown( QString::fromStdString(e.to_string()) );
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
return *result.first;
|
2015-07-13 21:24:25 +00:00
|
|
|
}
|
2015-07-14 13:46:38 +00:00
|
|
|
return *itr;
|
2015-07-10 21:37:22 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
QQmlListProperty<Balance> Account::balances()
|
|
|
|
|
{
|
|
|
|
|
return QQmlListProperty<Balance>(this, m_balances);
|
|
|
|
|
}
|
2015-07-13 17:24:30 +00:00
|
|
|
|
2015-07-13 17:56:30 +00:00
|
|
|
GrapheneApplication::GrapheneApplication( QObject* parent )
|
|
|
|
|
:QObject( parent ),m_thread("app")
|
|
|
|
|
{
|
2015-07-13 18:14:58 +00:00
|
|
|
connect( this, &GrapheneApplication::queueExecute,
|
|
|
|
|
this, &GrapheneApplication::execute );
|
|
|
|
|
|
2015-07-13 17:56:30 +00:00
|
|
|
m_model = new ChainDataModel( m_thread, this );
|
2015-07-13 19:41:06 +00:00
|
|
|
|
|
|
|
|
connect( m_model, &ChainDataModel::queueExecute,
|
|
|
|
|
this, &GrapheneApplication::execute );
|
|
|
|
|
|
|
|
|
|
connect( m_model, &ChainDataModel::exceptionThrown,
|
|
|
|
|
this, &GrapheneApplication::exceptionThrown );
|
2015-07-13 17:56:30 +00:00
|
|
|
}
|
2015-07-13 18:14:58 +00:00
|
|
|
|
2015-07-13 17:56:30 +00:00
|
|
|
GrapheneApplication::~GrapheneApplication()
|
|
|
|
|
{
|
|
|
|
|
}
|
|
|
|
|
|
2015-07-13 18:14:58 +00:00
|
|
|
void GrapheneApplication::setIsConnected( bool v )
|
|
|
|
|
{
|
|
|
|
|
if( v != m_isConnected )
|
|
|
|
|
{
|
|
|
|
|
m_isConnected = v;
|
|
|
|
|
Q_EMIT isConnectedChanged( m_isConnected );
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2015-07-13 18:35:17 +00:00
|
|
|
void GrapheneApplication::start( QString apiurl, QString user, QString pass )
|
2015-07-13 17:24:30 +00:00
|
|
|
{
|
2015-07-13 17:56:30 +00:00
|
|
|
if( !m_thread.is_current() )
|
|
|
|
|
{
|
2015-07-13 18:35:17 +00:00
|
|
|
m_done = m_thread.async( [=](){ return start( apiurl, user, pass ); } );
|
2015-07-13 18:14:58 +00:00
|
|
|
return;
|
2015-07-13 17:56:30 +00:00
|
|
|
}
|
|
|
|
|
try {
|
2015-07-13 21:24:25 +00:00
|
|
|
m_client = std::make_shared<fc::http::websocket_client>();
|
|
|
|
|
ilog( "connecting...${s}", ("s",apiurl.toStdString()) );
|
|
|
|
|
auto con = m_client->connect( apiurl.toStdString() );
|
2015-07-13 17:56:30 +00:00
|
|
|
auto apic = std::make_shared<fc::rpc::websocket_api_connection>(*con);
|
2015-07-13 18:14:58 +00:00
|
|
|
auto remote_api = apic->get_remote_api< login_api >(1);
|
2015-07-13 19:41:06 +00:00
|
|
|
auto db_api = apic->get_remote_api< database_api >(0);
|
2015-07-13 18:14:58 +00:00
|
|
|
if( !remote_api->login( user.toStdString(), pass.toStdString() ) )
|
|
|
|
|
{
|
2015-07-13 21:24:25 +00:00
|
|
|
elog( "login failed" );
|
2015-07-13 19:41:06 +00:00
|
|
|
Q_EMIT loginFailed();
|
2015-07-13 18:14:58 +00:00
|
|
|
return;
|
|
|
|
|
}
|
2015-07-13 17:56:30 +00:00
|
|
|
|
2015-07-13 21:24:25 +00:00
|
|
|
ilog( "connecting..." );
|
2015-07-13 19:41:06 +00:00
|
|
|
queueExecute( [=](){
|
|
|
|
|
m_model->setDatabaseAPI( db_api );
|
|
|
|
|
});
|
|
|
|
|
|
2015-07-13 18:14:58 +00:00
|
|
|
queueExecute( [=](){setIsConnected( true );} );
|
2015-07-13 17:56:30 +00:00
|
|
|
} catch ( const fc::exception& e )
|
|
|
|
|
{
|
2015-07-13 19:41:06 +00:00
|
|
|
Q_EMIT exceptionThrown( QString::fromStdString(e.to_string()) );
|
2015-07-13 17:56:30 +00:00
|
|
|
}
|
2015-07-13 18:14:58 +00:00
|
|
|
}
|
|
|
|
|
Q_SLOT void GrapheneApplication::execute( const std::function<void()>& func )const
|
|
|
|
|
{
|
|
|
|
|
func();
|
2015-07-13 17:24:30 +00:00
|
|
|
}
|