diff --git a/libdap-chain-mempool b/libdap-chain-mempool deleted file mode 160000 index e58e40b46acd4b3ea5265000bfa8b64ddfc5f07e..0000000000000000000000000000000000000000 --- a/libdap-chain-mempool +++ /dev/null @@ -1 +0,0 @@ -Subproject commit e58e40b46acd4b3ea5265000bfa8b64ddfc5f07e diff --git a/libdap-chain-mempool/CMakeLists.txt b/libdap-chain-mempool/CMakeLists.txt new file mode 100755 index 0000000000000000000000000000000000000000..5b64979025742cc8193d92468327aee6f77052ac --- /dev/null +++ b/libdap-chain-mempool/CMakeLists.txt @@ -0,0 +1,19 @@ +cmake_minimum_required(VERSION 2.8) +project (dap_chain_mempool) + +file(GLOB DAP_CHAIN_MEMPOOL_SRC *.c) +file(GLOB DAP_CHAIN_MEMPOOL_HDR *.h) + +if(WIN32) + include_directories(../3rdparty/wepoll/) + include_directories(../3rdparty/uthash/src/) +endif() + +add_library(${PROJECT_NAME} STATIC ${DAP_CHAIN_MEMPOOL_SRC} ${DAP_CHAIN_MEMPOOL_HDR}) + +target_link_libraries(dap_chain_mempool dap_http_server dap_client dap_chain_net dap_chain_global_db dap_core) +target_include_directories(dap_chain_mempool INTERFACE .) + +set(${PROJECT_NAME}_DEFINITIONS CACHE INTERNAL "${PROJECT_NAME}: Definitions" FORCE) + +set(${PROJECT_NAME}_INCLUDE_DIRS ${PROJECT_SOURCE_DIR} CACHE INTERNAL "${PROJECT_NAME}: Include Directories" FORCE) diff --git a/libdap-chain-mempool/client_mempool.c b/libdap-chain-mempool/client_mempool.c new file mode 100755 index 0000000000000000000000000000000000000000..9286fc8ae05c90edef5fdde4aa35e68de680b351 --- /dev/null +++ b/libdap-chain-mempool/client_mempool.c @@ -0,0 +1,334 @@ +/* + * Authors: + * Dmitriy A. Gearasimov <gerasimov.dmitriy@demlabs.net> + * Alexander Lysikov <alexander.lysikov@demlabs.net> + * DeM Labs Inc. https://demlabs.net + * Kelvin Project https://github.com/kelvinblockchain + * Copyright (c) 2017-2019 + * All rights reserved. + + This file is part of DAP (Deus Applications Prototypes) the open source project + + DAP (Deus Applicaions Prototypes) 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, either version 3 of the License, or + (at your option) any later version. + + DAP 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 any DAP based project. If not, see <http://www.gnu.org/licenses/>. +*/ +#include <stdlib.h> +#include <stdio.h> +#include <string.h> +#include <time.h> +#include <assert.h> +#include <errno.h> + +#ifdef _WIN32 +#include <winsock2.h> +#include <windows.h> +#include <mswsock.h> +#include <ws2tcpip.h> +#include <io.h> +#include <time.h> +#include <pthread.h> +#endif + +#include "dap_common.h" +#include "dap_config.h" +#include "dap_events.h" +#include "dap_chain_node.h" +#include "dap_client_pvt.h" +//#include "dap_http_client_simple.h" +#include "client_mempool.h" + +#define LOG_TAG "dap_client_mempool" + +static int listen_port_tcp = 8079; + +// send request to server +static int client_mempool_send_request(client_mempool_t *mempool, dap_datum_mempool_t *datum_mempool, + const char *action, bool is_last_req); + + +// callback for dap_client_new() in client_mempool_connect() +static void stage_status_callback(dap_client_t *a_client, void *a_arg) +{ + //printf("* stage_status_callback client=%x data=%x\n", a_client, a_arg); +} +// callback for dap_client_new() in client_mempool_connect() +static void stage_status_error_callback(dap_client_t *a_client, void *a_arg) +{ + //printf("* tage_status_error_callback client=%x data=%x\n", a_client, a_arg); +} + +// callback for dap_client_request_enc() in client_mempool_send_datum() +static void a_response_proc(dap_client_t *a_client, void *str, size_t str_len) +{ + //printf("a* _response_proc a_client=%x str=%s str_len=%d\n", a_client, str, str_len); + client_mempool_t *mempool = a_client->_inheritor; + assert(mempool); + if(mempool) { + if(str_len > 0) { + mempool->read_data_t.data = DAP_NEW_Z_SIZE(uint8_t, str_len + 1); + if(mempool->read_data_t.data) { + memcpy(mempool->read_data_t.data, str, str_len); + mempool->read_data_t.data_len = str_len; + } + } + pthread_mutex_lock(&mempool->wait_mutex); + mempool->state = CLIENT_MEMPOOL_SENDED; + +#ifndef _WIN32 + pthread_cond_signal(&mempool->wait_cond); +#else + SetEvent( mempool->wait_cond ); +#endif + + pthread_mutex_unlock(&mempool->wait_mutex); + } +} + +// callback for dap_client_request_enc() in client_mempool_send_datum() +static void a_response_error(dap_client_t *a_client, int val) +{ + //printf("* a_response_error a_client=%x val=%d\n", a_client, val); + client_mempool_t *mempool = a_client->_inheritor; + assert(mempool); + if(mempool) { + pthread_mutex_lock(&mempool->wait_mutex); + mempool->state = CLIENT_MEMPOOL_ERROR; +#ifndef _WIN32 + pthread_cond_signal(&mempool->wait_cond); +#else + SetEvent( mempool->wait_cond ); +#endif + pthread_mutex_unlock(&mempool->wait_mutex); + } +} + +// callback for the end of handshake in dap_client_go_stage() / client_mempool_connect() +static void a_stage_end_callback(dap_client_t *a_client, void *a_arg) +{ + client_mempool_t *mempool = a_client->_inheritor; + assert(mempool); + if(mempool) { + pthread_mutex_lock(&mempool->wait_mutex); + mempool->state = CLIENT_MEMPOOL_CONNECTED; +#ifndef _WIN32 + pthread_cond_signal(&mempool->wait_cond); +#else + SetEvent( mempool->wait_cond ); +#endif + pthread_mutex_unlock(&mempool->wait_mutex); + } +} + +int client_mempool_init(void) +{ + listen_port_tcp = dap_config_get_item_int32_default(g_config, "server", "listen_port_tcp", 8079); + return 0; +} + +void client_mempool_deinit() +{ + //dap_http_client_simple_deinit(); + dap_client_deinit(); +} + +client_mempool_t *client_mempool_connect(const char *addr) +{ + if(!addr || strlen(addr) < 1) + return NULL; + client_mempool_t *mempool = DAP_NEW_Z(client_mempool_t); + mempool->state = CLIENT_MEMPOOL_INIT; + + log_it(L_NOTICE, "======= client_mempool_connect( )" ); + +#ifndef _WIN32 + pthread_condattr_t attr; + pthread_condattr_init(&attr); + pthread_condattr_setclock(&attr, CLOCK_MONOTONIC); + pthread_cond_init(&mempool->wait_cond, &attr); +#else + mempool->wait_cond = CreateEventA( NULL, FALSE, FALSE, NULL ); +#endif + + pthread_mutex_init(&mempool->wait_mutex, NULL); + mempool->a_events = dap_events_new(); + mempool->a_client = dap_client_new(mempool->a_events, stage_status_callback, stage_status_error_callback); + mempool->a_client->_inheritor = mempool; + dap_client_pvt_t *l_client_internal = DAP_CLIENT_PVT(mempool->a_client); + + l_client_internal->uplink_addr = strdup(addr); + l_client_internal->uplink_port = listen_port_tcp; // reads from settings, default 8079 + l_client_internal->uplink_protocol_version = DAP_PROTOCOL_VERSION; + dap_client_stage_t a_stage_target = STAGE_ENC_INIT; + + mempool->state = CLIENT_MEMPOOL_CONNECT; + // Handshake + dap_client_go_stage(mempool->a_client, a_stage_target, a_stage_end_callback); + return mempool; +} + +/** + * wait for the complete of request + * + * timeout_ms timeout in milliseconds + * waited_state state which we will wait, sample CLIENT_MEMPOOL_CONNECTED or CLIENT_MEMPOOL_SENDED + * return -1 false, 0 timeout, 1 end of connection or sending data + */ +int client_mempool_wait(client_mempool_t *mempool, int waited_state, int timeout_ms) +{ + int ret = -1; + if( !mempool ) + return -1; + + log_it(L_NOTICE, "======= client_mempool_wait( ) tm %u ms", timeout_ms ); + + pthread_mutex_lock(&mempool->wait_mutex); +// have waited + if(mempool->state == waited_state) { + pthread_mutex_unlock(&mempool->wait_mutex); + return 1; + } +// prepare for signal waiting +#ifndef _WIN32 + struct timespec to; + clock_gettime(CLOCK_MONOTONIC, &to); + int64_t nsec_new = to.tv_nsec + timeout_ms * 1000000ll; +// if the new number of nanoseconds is more than a second + if(nsec_new > (long) 1e9) { + to.tv_sec += nsec_new / (long) 1e9; + to.tv_nsec = nsec_new % (long) 1e9; + } + else + to.tv_nsec = (long) nsec_new; +// signal waiting + int wait = pthread_cond_timedwait(&mempool->wait_cond, &mempool->wait_mutex, &to); + if(wait == 0) //0 + ret = 1; + else if(wait == ETIMEDOUT) // 110 260 + ret = 0; +#else + int wait = WaitForSingleObject( mempool->wait_cond, (uint32_t)timeout_ms ); + pthread_mutex_lock(&mempool->wait_mutex); + + if ( wait == WAIT_OBJECT_0 ) return 1; + else if ( wait == WAIT_TIMEOUT || wait == WAIT_FAILED ) { + ret = 0; + } +#endif + + pthread_mutex_unlock(&mempool->wait_mutex); + return ret; +} + +/** + * get read data from server + */ +uint8_t* client_mempool_read(client_mempool_t *mempool, int *data_len) +{ + if(mempool && mempool->read_data_t.data_len > 0) { + + uint8_t*data = DAP_NEW_Z_SIZE(uint8_t, mempool->read_data_t.data_len + 1); + if(mempool->read_data_t.data) { + memcpy(data, mempool->read_data_t.data, mempool->read_data_t.data_len); + if(data_len) + *data_len = mempool->read_data_t.data_len; + return data; + } + } + return NULL; +} + +void client_mempool_close(client_mempool_t *mempool) +{ + if(mempool) { + // send last request for dehandshake with "SessionCloseAfterRequest=true" + client_mempool_send_request(mempool, NULL, 0, true); + // wait close session + client_mempool_wait(mempool, CLIENT_MEMPOOL_SENDED, 500); + // clean mempool + dap_client_pvt_t *l_client_internal = DAP_CLIENT_PVT(mempool->a_client); + DAP_DELETE(l_client_internal->uplink_addr); + dap_client_delete(mempool->a_client); + dap_events_delete(mempool->a_events); + DAP_DELETE(mempool->read_data_t.data); + pthread_cond_destroy(&mempool->wait_cond); + pthread_mutex_destroy(&mempool->wait_mutex); + DAP_DELETE(mempool); + } +} + +// set new state and delete previous read data +static void client_mempool_reset(client_mempool_t *mempool, int new_state) +{ + if(!mempool) + return; + pthread_mutex_lock(&mempool->wait_mutex); + mempool->read_data_t.data_len = 0; + DAP_DELETE(mempool->read_data_t.data); + mempool->read_data_t.data = NULL; + mempool->state = new_state; + pthread_mutex_unlock(&mempool->wait_mutex); +} + +// send request to server +static int client_mempool_send_request(client_mempool_t *mempool, dap_datum_mempool_t *datum_mempool, + const char *action, bool is_last_req) +{ + if(!mempool || mempool->state < CLIENT_MEMPOOL_CONNECTED) + return -1; + const char * a_path = "mempool"; + const char *a_suburl = (action) ? action : "close"; + const char* a_query = NULL; + size_t a_request_size = 1; + uint8_t *a_request = (datum_mempool) ? dap_datum_mempool_serialize(datum_mempool, &a_request_size) : (uint8_t*) " "; + uint8_t *a_request_out = DAP_NEW_Z_SIZE(uint8_t, a_request_size * 2); // a_request + 1 byte for type action + dap_bin2hex(a_request_out, a_request, a_request_size); + client_mempool_reset(mempool, CLIENT_MEMPOOL_SEND); + dap_client_pvt_t * l_client_internal = DAP_CLIENT_PVT(mempool->a_client); + l_client_internal->is_close_session = is_last_req; + dap_client_request_enc(mempool->a_client, a_path, a_suburl, a_query, a_request_out, a_request_size * 2, + a_response_proc, a_response_error); + if(datum_mempool) + DAP_DELETE(a_request); + DAP_DELETE(a_request_out); + return 1; +} + +/** + * datum add in mempool + * + * return -1 not connected or error, 1 send packet OK + */ +int client_mempool_send_datum(client_mempool_t *mempool, dap_datum_mempool_t *datum_mempool) +{ + return client_mempool_send_request(mempool, datum_mempool, "add", false); +} + +/** + * datum check in mempool + * + * return -1 not connected or error, 1 send packet OK + */ +int client_mempool_check_datum(client_mempool_t *mempool, dap_datum_mempool_t *datum_mempool) +{ + return client_mempool_send_request(mempool, datum_mempool, "check", false); +} + +/** + * datum delete from mempool + * + * return -1 not connected or error, 1 send packet OK + */ +int client_mempool_del_datum(client_mempool_t *mempool, dap_datum_mempool_t *datum_mempool) +{ + return client_mempool_send_request(mempool, datum_mempool, "del", false); +} diff --git a/libdap-chain-mempool/client_mempool.h b/libdap-chain-mempool/client_mempool.h new file mode 100755 index 0000000000000000000000000000000000000000..475a02a396d11bebc526a26d1c0dab95461c756c --- /dev/null +++ b/libdap-chain-mempool/client_mempool.h @@ -0,0 +1,84 @@ +/* + * Authors: + * Dmitriy A. Gearasimov <gerasimov.dmitriy@demlabs.net> + * Alexander Lysikov <alexander.lysikov@demlabs.net> + * DeM Labs Inc. https://demlabs.net + * Kelvin Project https://github.com/kelvinblockchain + * Copyright (c) 2017-2019 + * All rights reserved. + + This file is part of DAP (Deus Applications Prototypes) the open source project + + DAP (Deus Applicaions Prototypes) 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, either version 3 of the License, or + (at your option) any later version. + + DAP 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 any DAP based project. If not, see <http://www.gnu.org/licenses/>. +*/ +#pragma once + +#include <pthread.h> +#include <stdbool.h> +#include "dap_client.h" +#include "dap_chain_mempool.h" + +// connection states +enum { + CLIENT_MEMPOOL_ERROR = -1, + CLIENT_MEMPOOL_INIT, + CLIENT_MEMPOOL_CONNECT, + CLIENT_MEMPOOL_CONNECTED, + CLIENT_MEMPOOL_SEND, + CLIENT_MEMPOOL_SENDED, + CLIENT_MEMPOOL_END +}; + +// state for a client connection with mempool +typedef struct client_mempool { + int state; + dap_events_t *a_events; + dap_client_t *a_client; + pthread_cond_t wait_cond; + pthread_mutex_t wait_mutex; + struct readed_data{ + uint8_t *data; + int data_len; + }read_data_t; +} client_mempool_t; + +int client_mempool_init(void); +void client_mempool_deinit(); + +client_mempool_t* client_mempool_connect(const char *addr); +void client_mempool_close(client_mempool_t *mempool); + +/** + * timeout_ms timeout in milliseconds + * return -1 false, 0 timeout, 1 end of connection or sending data + */ +int client_mempool_wait(client_mempool_t *mempool, int waited_state, int timeout_ms); + +/** + * get read data from server + */ +uint8_t* client_mempool_read(client_mempool_t *mempool, int *data_len); + +/** + * datum add in mempool + */ +int client_mempool_send_datum(client_mempool_t *mempool, dap_datum_mempool_t *datum_mempool); +/** + * datum check in mempool + */ +int client_mempool_check_datum(client_mempool_t *mempool, dap_datum_mempool_t *datum_mempool); +/** + * datum delete from mempool + */ +int client_mempool_del_datum(client_mempool_t *mempool, dap_datum_mempool_t *datum_mempool); diff --git a/libdap-chain-mempool/dap_chain_mempool.c b/libdap-chain-mempool/dap_chain_mempool.c new file mode 100755 index 0000000000000000000000000000000000000000..528ecf06c165a8f7b6f3ce7e60ed5bea601c07b8 --- /dev/null +++ b/libdap-chain-mempool/dap_chain_mempool.c @@ -0,0 +1,1112 @@ +/* + * Authors: + * Dmitriy A. Gearasimov <gerasimov.dmitriy@demlabs.net> + * Alexander Lysikov <alexander.lysikov@demlabs.net> + * DeM Labs Inc. https://demlabs.net + * Kelvin Project https://github.com/kelvinblockchain + * Copyright (c) 2017-2019 + * All rights reserved. + + This file is part of DAP (Deus Applications Prototypes) the open source project + + DAP (Deus Applicaions Prototypes) 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, either version 3 of the License, or + (at your option) any later version. + + DAP 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 any DAP based project. If not, see <http://www.gnu.org/licenses/>. + */ +#include <stddef.h> +#include <stdio.h> +#include <string.h> +#include <stdio.h> +#include <assert.h> +#include <memory.h> + +#ifdef _WIN32 +#include <winsock2.h> +#include <windows.h> +#include <mswsock.h> +#include <ws2tcpip.h> +#include <io.h> +#include <time.h> +#include <pthread.h> +#endif + +#include "dap_common.h" +#include "dap_hash.h" +#include "dap_http_client.h" +#include "dap_http_simple.h" +//#include "dap_enc_http.h" +#include "dap_enc_http.h" +//#include "dap_http.h" +#include "http_status_code.h" +#include "dap_chain_common.h" +#include "dap_chain_node.h" +#include "dap_chain_global_db.h" +#include "dap_enc.h" +#include <dap_enc_http.h> +#include <dap_enc_key.h> +#include <dap_enc_ks.h> +#include "dap_chain_mempool.h" + +#include "dap_common.h" +#include "dap_list.h" +#include "dap_chain.h" +#include "dap_chain_net.h" +#include "dap_sign.h" +#include "dap_chain_datum_tx.h" +#include "dap_chain_datum_tx_items.h" + +#define LOG_TAG "dap_chain_mempool" + +typedef struct list_used_item { + dap_chain_hash_fast_t tx_hash_fast; + int num_idx_out; + uint8_t padding[4]; + uint64_t value; +//dap_chain_tx_out_t *tx_out; +} list_used_item_t; + + +int dap_datum_mempool_init(void) +{ + return 0; +} + +/** + * @brief dap_chain_mempool_datum_add + * @param a_datum + * @return + */ +int dap_chain_mempool_datum_add(dap_chain_datum_t * a_datum, dap_chain_t * a_chain ) +{ + if( a_datum == NULL){ + log_it(L_ERROR, "NULL datum trying to add in mempool"); + return -1; + } + int ret =0; + + dap_chain_hash_fast_t l_key_hash; + dap_hash_fast(a_datum->data , a_datum->header.data_size, &l_key_hash); + + char * l_key_str = dap_chain_hash_fast_to_str_new(&l_key_hash); + char * l_gdb_group = dap_chain_net_get_gdb_group_mempool(a_chain); + if(dap_chain_global_db_gr_set(dap_strdup(l_key_str), (byte_t *) a_datum, dap_chain_datum_size(a_datum) + ,l_gdb_group)) { + log_it(L_NOTICE, "Datum with data's hash %s was placed in mempool", l_key_str); + }else{ + log_it(L_WARNING, "Can't place data's hash %s was placed in mempool", l_key_str); + } + DAP_DELETE(l_gdb_group); + DAP_DELETE(l_key_str); + return ret; +} + +/** + * Make transfer transaction & insert to cache + * + * return 0 Ok, -2 not enough funds to transfer, -1 other Error + */ +int dap_chain_mempool_tx_create(dap_chain_t * a_chain, dap_enc_key_t *a_key_from, + const dap_chain_addr_t* a_addr_from, const dap_chain_addr_t* a_addr_to, + const dap_chain_addr_t* a_addr_fee, + const char a_token_ticker[DAP_CHAIN_TICKER_SIZE_MAX], + uint64_t a_value, uint64_t a_value_fee) +{ + // check valid param + if(!a_chain | !a_key_from || ! a_addr_from || !a_key_from->priv_key_data || !a_key_from->priv_key_data_size || + !dap_chain_addr_check_sum(a_addr_from) || !dap_chain_addr_check_sum(a_addr_to) || + (a_addr_fee && !dap_chain_addr_check_sum(a_addr_fee)) || !a_value) + return -1; + + // find the transactions from which to take away coins + dap_list_t *l_list_used_out = NULL; // list of transaction with 'out' items + uint64_t l_value_transfer = 0; // how many coins to transfer + { + dap_chain_hash_fast_t l_tx_cur_hash = { 0 }; + uint64_t l_value_need = a_value + a_value_fee; + while(l_value_transfer < l_value_need) + { + // Get the transaction in the cache by the addr in out item + dap_chain_datum_tx_t *l_tx = dap_chain_ledger_tx_find_by_addr(a_chain->ledger,a_token_ticker, a_addr_from, + &l_tx_cur_hash); + if(!l_tx) + break; + // Get all item from transaction by type + int l_item_count = 0; + dap_list_t *l_list_out_items = dap_chain_datum_tx_items_get( l_tx, TX_ITEM_TYPE_OUT, + &l_item_count); + dap_list_t *l_list_tmp = l_list_out_items; + int l_out_idx_tmp = 0; // current index of 'out' item + while(l_list_tmp) { + dap_chain_tx_out_t *out_item = l_list_tmp->data; + // if 'out' item has addr = a_addr_from + if(out_item && !memcmp(a_addr_from, &out_item->addr, sizeof(dap_chain_addr_t))) { + + // Check whether used 'out' items + if(!dap_chain_ledger_tx_hash_is_used_out_item (a_chain->ledger, &l_tx_cur_hash, l_out_idx_tmp)) { + + list_used_item_t *item = DAP_NEW(list_used_item_t); + memcpy(&item->tx_hash_fast, &l_tx_cur_hash, sizeof(dap_chain_hash_fast_t)); + item->num_idx_out = l_out_idx_tmp; + item->value = out_item->header.value; + l_list_used_out = dap_list_append(l_list_used_out, item); + l_value_transfer += item->value; + // already accumulated the required value, finish the search for 'out' items + if(l_value_transfer >= l_value_need) { + break; + } + } + } + // go to the next 'out' item in l_tx transaction + l_out_idx_tmp++; + l_list_tmp = dap_list_next(l_list_tmp); + } + dap_list_free(l_list_out_items); + } + + // nothing to tranfer (not enough funds) + if(!l_list_used_out || l_value_transfer < l_value_need) { + dap_list_free_full(l_list_used_out, free); + return -2; + } + } + + // create empty transaction + dap_chain_datum_tx_t *l_tx = dap_chain_datum_tx_create(); + // add 'in' items + { + dap_list_t *l_list_tmp = l_list_used_out; + uint64_t l_value_to_items = 0; // how many datoshi to transfer + while(l_list_tmp) { + list_used_item_t *item = l_list_tmp->data; + if(dap_chain_datum_tx_add_in_item(&l_tx, &item->tx_hash_fast,(uint32_t) item->num_idx_out) == 1) { + l_value_to_items += item->value; + } + l_list_tmp = dap_list_next(l_list_tmp); + } + assert(l_value_to_items == l_value_transfer); + dap_list_free_full(l_list_used_out, free); + } + // add 'out' items + { + uint64_t l_value_pack = 0; // how much datoshi add to 'out' items + if(dap_chain_datum_tx_add_out_item(&l_tx, a_addr_to, a_value) == 1) { + l_value_pack += a_value; + // transaction fee + if(a_addr_fee) { + if(dap_chain_datum_tx_add_out_item(&l_tx, a_addr_fee, a_value_fee) == 1) + l_value_pack += a_value_fee; + } + } + // coin back + uint64_t l_value_back = l_value_transfer - l_value_pack; + if(l_value_back) { + if(dap_chain_datum_tx_add_out_item(&l_tx, a_addr_from, l_value_back) != 1) { + dap_chain_datum_tx_delete(l_tx); + return -1; + } + } + } + + // add 'sign' items + if(dap_chain_datum_tx_add_sign_item(&l_tx, a_key_from) != 1) { + dap_chain_datum_tx_delete(l_tx); + return -1; + } + + size_t l_tx_size = dap_chain_datum_tx_get_size(l_tx); + dap_chain_datum_t *l_datum = dap_chain_datum_create(DAP_CHAIN_DATUM_TX, l_tx, l_tx_size); + DAP_DELETE(l_tx); + if(dap_chain_mempool_datum_add (l_datum, a_chain) == 0){ + return 0; + }else{ + DAP_DELETE( l_datum ); + return -4; + } +} + +/** + * Make transfer transaction & insert to cache + * + * return 0 Ok, -2 not enough funds to transfer, -1 other Error + */ +int dap_chain_mempool_tx_create_massive( dap_chain_t * a_chain, dap_enc_key_t *a_key_from, + const dap_chain_addr_t* a_addr_from, const dap_chain_addr_t* a_addr_to, + const dap_chain_addr_t* a_addr_fee, + const char a_token_ticker[10], + uint64_t a_value, uint64_t a_value_fee,size_t a_tx_num) +{ + // check valid param + if(!a_chain | !a_key_from || !a_addr_from || !a_key_from->priv_key_data || !a_key_from->priv_key_data_size || + !dap_chain_addr_check_sum(a_addr_from) || !dap_chain_addr_check_sum(a_addr_to) || + (a_addr_fee && !dap_chain_addr_check_sum(a_addr_fee)) || !a_value || !a_tx_num){ + log_it(L_ERROR, "Wrong parameters in dap_chain_mempool_tx_create_massive() call"); + return -1; + + } + dap_global_db_obj_t * l_objs = DAP_NEW_Z_SIZE(dap_global_db_obj_t, (a_tx_num+1)*sizeof (dap_global_db_obj_t) ); + + + + // Search unused out: + uint64_t l_value_need =a_tx_num*( a_value + a_value_fee ); + dap_chain_hash_fast_t l_tx_prev_hash = { 0 }; + dap_list_t *l_list_used_out = NULL; // list of transaction with 'out' items + uint64_t l_value_transfer = 0; // how many coins to transfer + + log_it(L_DEBUG,"Create %lu transactions, summary %Lf.7", a_tx_num,dap_chain_balance_to_coins(l_value_need) ) ; + + while(l_value_transfer < l_value_need){ + // Get the transaction in the cache by the addr in out item + dap_chain_datum_tx_t *l_tx = dap_chain_ledger_tx_find_by_addr(a_chain->ledger, a_token_ticker,a_addr_from, + &l_tx_prev_hash); + if(!l_tx) + break; + // Get all item from transaction by type + int l_item_count = 0; + dap_list_t *l_list_out_items = dap_chain_datum_tx_items_get( l_tx, TX_ITEM_TYPE_OUT, + &l_item_count); + dap_list_t *l_list_tmp = l_list_out_items; + int l_out_idx_tmp = 0; // current index of 'out' item + while(l_list_tmp) { + dap_chain_tx_out_t *out_item = l_list_tmp->data; + // if 'out' item has addr = a_addr_from + if(out_item && out_item->header.value && !memcmp(a_addr_from, &out_item->addr, sizeof(dap_chain_addr_t))) { + + // Check whether used 'out' items + if(!dap_chain_ledger_tx_hash_is_used_out_item (a_chain->ledger, &l_tx_prev_hash, l_out_idx_tmp)) { + + list_used_item_t *l_it = DAP_NEW(list_used_item_t); + memcpy(&l_it->tx_hash_fast, &l_tx_prev_hash, sizeof(dap_chain_hash_fast_t)); + l_it->num_idx_out = l_out_idx_tmp; + l_it->value = out_item->header.value; + log_it(L_DEBUG," Found output with value %llu", l_it->value ); + l_list_used_out = dap_list_append(l_list_used_out, l_it); + l_value_transfer += l_it->value; + // already accumulated the required value, finish the search for 'out' items + if(l_value_transfer >= l_value_need) { + l_out_idx_tmp++; + break; + } + } + } + // go to the next 'out' item in l_tx transaction + l_out_idx_tmp++; + l_list_tmp = dap_list_next(l_list_tmp); + } + } + + // nothing to tranfer (not enough funds) + if(!l_list_used_out || l_value_transfer < l_value_need) { + log_it(L_WARNING,"Not enough funds to transfer"); + dap_list_free_full(l_list_used_out, free); + return -2; + } + + for (size_t i=0; i< a_tx_num ; i++){ + log_it(L_DEBUG, "Prepare tx %u",i); + // find the transactions from which to take away coins + + // create empty transaction + dap_chain_datum_tx_t *l_tx_new = dap_chain_datum_tx_create(); + uint64_t l_value_back=0; + // add 'in' items + dap_list_t *l_list_tmp = l_list_used_out; + uint64_t l_value_to_items = 0; // how many coins to transfer + + // Add in and remove out used items + while(l_list_tmp) { + list_used_item_t *item = l_list_tmp->data; + char l_in_hash_str[70]; + + dap_chain_hash_fast_to_str(&item->tx_hash_fast,l_in_hash_str,sizeof (l_in_hash_str) ); + + if(dap_chain_datum_tx_add_in_item(&l_tx_new, &item->tx_hash_fast, (uint32_t) item->num_idx_out) == 1) { + l_value_to_items += item->value; + log_it(L_DEBUG,"Added input %s with %llu datoshi",l_in_hash_str, item->value); + }else{ + log_it(L_WARNING,"Can't add input from %s with %llu datoshi",l_in_hash_str, item->value); + } + l_list_used_out = l_list_tmp->next; + DAP_DELETE(l_list_tmp->data); + dap_list_free1(l_list_tmp); + l_list_tmp = l_list_used_out; + if ( l_value_to_items >= l_value_transfer ) + break; + } + if ( l_value_to_items < (a_value + a_value_fee) ){ + log_it(L_ERROR,"Not enought values on output %llu to produce enought ins %llu when need %llu", + l_value_to_items, l_value_transfer, + l_value_need); + return -5; + } + + // add 'out' items + uint64_t l_value_pack = 0; // how much coin add to 'out' items + if(dap_chain_datum_tx_add_out_item(&l_tx_new, a_addr_to, a_value) == 1) { + l_value_pack += a_value; + // transaction fee + if(a_addr_fee) { + if(dap_chain_datum_tx_add_out_item(&l_tx_new, a_addr_fee, a_value_fee) == 1) + l_value_pack += a_value_fee; + } + } + // coin back + l_value_back = l_value_transfer - l_value_pack; + if(l_value_back) { + //log_it(L_DEBUG,"Change back %llu", l_value_back); + if(dap_chain_datum_tx_add_out_item(&l_tx_new, a_addr_from, l_value_back) != 1) { + dap_chain_datum_tx_delete(l_tx_new); + return -3; + } + } + + // add 'sign' items + if(dap_chain_datum_tx_add_sign_item(&l_tx_new, a_key_from) != 1) { + dap_chain_datum_tx_delete(l_tx_new); + return -1; + } + // now tx is formed - calc size and hash + size_t l_tx_size = dap_chain_datum_tx_get_size(l_tx_new); + + dap_chain_hash_fast_t l_tx_new_hash; + dap_hash_fast(l_tx_new, l_tx_size, &l_tx_new_hash); + // If we have value back - update balance cache + if(l_value_back) { + //log_it(L_DEBUG,"We have value back %llu now lets see how many outputs we have", l_value_back); + int l_item_count = 0; + dap_list_t *l_list_out_items = dap_chain_datum_tx_items_get( l_tx_new, TX_ITEM_TYPE_OUT, + &l_item_count); + dap_list_t *l_list_tmp = l_list_out_items; + int l_out_idx_tmp = 0; // current index of 'out' item + //log_it(L_DEBUG,"We have %d outputs in new TX", l_item_count); + while(l_list_tmp) { + dap_chain_tx_out_t * l_out = l_list_tmp->data ; + if( ! l_out){ + log_it(L_WARNING, "Output is NULL, continue check outputs..."); + l_out_idx_tmp++; + continue; + } + if ( memcmp(&l_out->addr, a_addr_from, sizeof (*a_addr_from))==0 ){ + list_used_item_t *l_item_back = DAP_NEW(list_used_item_t); + memcpy(&l_item_back->tx_hash_fast, &l_tx_new_hash, sizeof(dap_chain_hash_fast_t)); + l_item_back->num_idx_out = l_out_idx_tmp; + l_item_back->value = l_value_back; + l_list_used_out = dap_list_prepend(l_list_used_out, l_item_back); + log_it(L_DEBUG,"Found change back output, stored back in UTXO table"); + break; + } + l_list_tmp = l_list_tmp->next; + l_out_idx_tmp++; + } + //log_it(L_DEBUG,"Checked all outputs"); + dap_list_free( l_list_out_items); + } + l_value_transfer -= l_value_pack; + + // Now produce datum + dap_chain_datum_t *l_datum = dap_chain_datum_create(DAP_CHAIN_DATUM_TX, l_tx_new, l_tx_size); + + dap_chain_datum_tx_delete(l_tx_new); + //dap_chain_ledger_tx_add( a_chain->ledger, l_tx); + + l_objs[i].key = dap_chain_hash_fast_to_str_new(&l_tx_new_hash); + //continue; + l_objs[i].value = (uint8_t*) l_datum; + l_objs[i].value_len = l_tx_size + sizeof(l_datum->header); + log_it(L_DEBUG, "Prepared obj with key %s (value_len = %llu)", + l_objs[i].key? l_objs[i].key :"NULL" , l_objs[i].value_len ); + + } + dap_list_free_full(l_list_used_out, free); + + char * l_gdb_group = dap_chain_net_get_gdb_group_mempool(a_chain); + + //return 0; + if( dap_chain_global_db_gr_save(l_objs,a_tx_num,l_gdb_group) ) { + log_it(L_NOTICE, "%u transaction are placed in mempool", a_tx_num); + //DAP_DELETE(l_objs); + DAP_DELETE(l_gdb_group); + return 0; + }else{ + log_it(L_ERROR, "Can't place %u transactions in mempool", a_tx_num); + //DAP_DELETE(l_objs); + DAP_DELETE(l_gdb_group); + return -4; + } + + +} + +dap_chain_hash_fast_t* dap_chain_mempool_tx_create_cond_input(dap_chain_net_t * a_net,dap_chain_hash_fast_t *a_tx_prev_hash, + + const dap_chain_addr_t* a_addr_to, dap_enc_key_t *l_key_tx_sign, dap_chain_datum_tx_receipt_t * l_receipt, size_t l_receipt_size) +{ + dap_ledger_t * l_ledger = a_net ? dap_chain_ledger_by_net_name( a_net->pub.name ) : NULL; + if ( ! a_net || ! l_ledger || ! a_addr_to ) + return NULL; + if ( ! dap_chain_addr_check_sum (a_addr_to) ){ + log_it(L_ERROR, "Wrong address_to checksum"); + return NULL; + } + + // create empty transaction + dap_chain_datum_tx_t *l_tx = dap_chain_datum_tx_create(); + + uint16_t pos=0; + dap_chain_datum_tx_add_item(&l_tx, (byte_t*) l_receipt); + pos++; + // add 'in_cond' items + // TODO - find first cond_item occurance, not just set it to 1 + if (dap_chain_datum_tx_add_in_cond_item(&l_tx,a_tx_prev_hash,1,pos-1) != 0 ){ + dap_chain_datum_tx_delete(l_tx); + log_it( L_ERROR, "Cant add tx cond input"); + return NULL; + } + + if(dap_chain_datum_tx_add_out_item(&l_tx, a_addr_to, l_receipt->receipt_info.value_datoshi) != 1) { + dap_chain_datum_tx_delete(l_tx); + log_it( L_ERROR, "Cant add tx output"); + return NULL; + } + + // add 'sign' items + if (l_key_tx_sign){ + if(dap_chain_datum_tx_add_sign_item(&l_tx, l_key_tx_sign) != 1) { + dap_chain_datum_tx_delete(l_tx); + log_it( L_ERROR, "Can't add sign output"); + return NULL; + } + } + size_t l_tx_size = dap_chain_datum_tx_get_size( l_tx ); + dap_chain_datum_t *l_datum = dap_chain_datum_create( DAP_CHAIN_DATUM_TX, l_tx, l_tx_size ); + + dap_chain_hash_fast_t *l_key_hash = DAP_NEW_Z( dap_chain_hash_fast_t ); + dap_hash_fast( l_tx, l_tx_size, l_key_hash ); + DAP_DELETE( l_tx ); + + char * l_key_str = dap_chain_hash_fast_to_str_new( l_key_hash ); + char * l_gdb_group = dap_chain_net_get_gdb_group_mempool_by_chain_type( a_net ,CHAIN_TYPE_TX); + if( dap_chain_global_db_gr_set( dap_strdup(l_key_str), (uint8_t *) l_datum, dap_chain_datum_size(l_datum) + , l_gdb_group ) ) { + log_it(L_NOTICE, "Transaction %s placed in mempool", l_key_str); + } + DAP_DELETE(l_gdb_group); + DAP_DELETE(l_key_str); + + return l_key_hash; +} + + +/** + * Make transfer transaction + * + * return dap_chain_datum_t, NULL if Error + */ +static dap_chain_datum_t* dap_chain_tx_create_cond(dap_chain_net_t * a_net, + dap_enc_key_t *a_key_from, dap_enc_key_t *a_key_cond, + const dap_chain_addr_t* a_addr_from, + const char a_token_ticker[DAP_CHAIN_TICKER_SIZE_MAX], + uint64_t a_value,uint64_t a_value_per_unit_max, dap_chain_net_srv_price_unit_uid_t a_unit, + dap_chain_net_srv_uid_t a_srv_uid, uint64_t a_value_fee, const void *a_cond, size_t a_cond_size) +{ + dap_ledger_t * l_ledger = a_net ? dap_chain_ledger_by_net_name( a_net->pub.name ) : NULL; + // check valid param + if(!a_net || ! l_ledger || !a_key_from || !a_key_from->priv_key_data || !a_key_from->priv_key_data_size || + !dap_chain_addr_check_sum(a_addr_from) || + !a_value) + return NULL; + + // find the transactions from which to take away coins + dap_list_t *l_list_used_out = NULL; // list of transaction with 'out' items + uint64_t l_value_transfer = 0; // how many coins to transfer + { + dap_chain_hash_fast_t l_tx_cur_hash = { 0 }; + uint64_t l_value_need = a_value + a_value_fee; + while(l_value_transfer < l_value_need) + { + // Get the transaction in the cache by the addr in out item + dap_chain_datum_tx_t *l_tx = dap_chain_ledger_tx_find_by_addr(l_ledger, a_token_ticker,a_addr_from, + &l_tx_cur_hash); + if(!l_tx) + break; + // Get all item from transaction by type + int l_item_count = 0; + dap_list_t *l_list_out_items = dap_chain_datum_tx_items_get( l_tx, TX_ITEM_TYPE_OUT, + &l_item_count); + dap_list_t *l_list_tmp = l_list_out_items; + int l_out_idx_tmp = 0; // current index of 'out' item + while(l_list_tmp) { + dap_chain_tx_out_t *out_item = l_list_tmp->data; + // if 'out' item has addr = a_addr_from + if(out_item && !memcmp(a_addr_from, &out_item->addr, sizeof(dap_chain_addr_t))) { + + // Check whether used 'out' items + if(!dap_chain_ledger_tx_hash_is_used_out_item(l_ledger, &l_tx_cur_hash, l_out_idx_tmp)) { + + list_used_item_t *item = DAP_NEW(list_used_item_t); + memcpy(&item->tx_hash_fast, &l_tx_cur_hash, sizeof(dap_chain_hash_fast_t)); + item->num_idx_out = l_out_idx_tmp; + item->value = out_item->header.value; + l_list_used_out = dap_list_append(l_list_used_out, item); + l_value_transfer += item->value; + // already accumulated the required value, finish the search for 'out' items + if(l_value_transfer >= l_value_need) { + break; + } + } + } + // go to the next 'out' item in l_tx transaction + l_out_idx_tmp++; + l_list_tmp = dap_list_next(l_list_tmp); + } + dap_list_free(l_list_out_items); + } + + // nothing to tranfer (not enough funds) + if(!l_list_used_out || l_value_transfer < l_value_need) { + dap_list_free_full(l_list_used_out, free); + log_it( L_ERROR, "nothing to tranfer (not enough funds)"); + return NULL; + } + } + + // create empty transaction + dap_chain_datum_tx_t *l_tx = dap_chain_datum_tx_create(); + // add 'in' items + { + dap_list_t *l_list_tmp = l_list_used_out; + uint64_t l_value_to_items = 0; // how many coins to transfer + while(l_list_tmp) { + list_used_item_t *item = l_list_tmp->data; + if(dap_chain_datum_tx_add_in_item(&l_tx, &item->tx_hash_fast,(uint32_t) item->num_idx_out) == 1) { + l_value_to_items += item->value; + } + l_list_tmp = dap_list_next(l_list_tmp); + } + assert(l_value_to_items == l_value_transfer); + dap_list_free_full(l_list_used_out, free); + } + // add 'out_cond' and 'out' items + { + uint64_t l_value_pack = 0; // how much coin add to 'out' items + if(dap_chain_datum_tx_add_out_cond_item(&l_tx, a_key_cond, a_srv_uid, a_value, a_value_per_unit_max, a_unit, a_cond, + a_cond_size) == 1) { + l_value_pack += a_value; + // transaction fee + if(a_value_fee) { + // TODO add condition with fee for mempool-as-service + } + } + // coin back + uint64_t l_value_back = l_value_transfer - l_value_pack; + if(l_value_back) { + if(dap_chain_datum_tx_add_out_item(&l_tx, a_addr_from, l_value_back) != 1) { + dap_chain_datum_tx_delete(l_tx); + log_it( L_ERROR, "Cant add coin back output"); + return NULL; + } + } + } + + // add 'sign' items + if(dap_chain_datum_tx_add_sign_item(&l_tx, a_key_from) != 1) { + dap_chain_datum_tx_delete(l_tx); + log_it( L_ERROR, "Can't add sign output"); + return NULL; + } + + size_t l_tx_size = dap_chain_datum_tx_get_size( l_tx ); + dap_chain_datum_t *l_datum = dap_chain_datum_create( DAP_CHAIN_DATUM_TX, l_tx, l_tx_size ); + + return l_datum; + /*dap_chain_hash_fast_t *l_key_hash = DAP_NEW_Z( dap_chain_hash_fast_t ); + dap_hash_fast( l_tx, l_tx_size, l_key_hash ); + DAP_DELETE( l_tx ); + + char * l_key_str = dap_chain_hash_fast_to_str_new( l_key_hash ); + char * l_gdb_group = dap_chain_net_get_gdb_group_mempool_by_chain_type( a_net ,CHAIN_TYPE_TX); + if( dap_chain_global_db_gr_set( dap_strdup(l_key_str), (uint8_t *) l_datum, dap_chain_datum_size(l_datum) + , l_gdb_group ) ) { + log_it(L_NOTICE, "Transaction %s placed in mempool", l_key_str); + } + DAP_DELETE(l_gdb_group); + DAP_DELETE(l_key_str); + + return l_key_hash;*/ +} + +/** + * Make transfer transaction & insert to database + * + * return 0 Ok, -2 not enough funds to transfer, -1 other Error + */ +dap_chain_hash_fast_t* dap_chain_proc_tx_create_cond(dap_chain_net_t * a_net, + dap_enc_key_t *a_key_from, dap_enc_key_t *a_key_cond, + const dap_chain_addr_t* a_addr_from, + const char a_token_ticker[DAP_CHAIN_TICKER_SIZE_MAX], + uint64_t a_value,uint64_t a_value_per_unit_max, dap_chain_net_srv_price_unit_uid_t a_unit, + dap_chain_net_srv_uid_t a_srv_uid, uint64_t a_value_fee, const void *a_cond, size_t a_cond_size) +{ + + dap_chain_t *l_chain = dap_chain_net_get_chain_by_chain_type(a_net, CHAIN_TYPE_TX); + if(!l_chain) + return NULL; + // Make transfer transaction + dap_chain_datum_t *l_datum = dap_chain_tx_create_cond(a_net,a_key_from, a_key_cond, a_addr_from, + a_token_ticker,a_value,a_value_per_unit_max, a_unit, + a_srv_uid, a_value_fee, a_cond, a_cond_size); + + if(!l_datum) + return NULL; + size_t l_datums_number = l_chain->callback_datums_pool_proc(l_chain, &l_datum, 1); + if(!l_datums_number) + return NULL; + + dap_chain_datum_tx_t *l_tx = (dap_chain_datum_tx_t*)&(l_datum->data); + size_t l_tx_size = l_datum->header.data_size; + + dap_chain_hash_fast_t *l_key_hash = DAP_NEW_Z( dap_chain_hash_fast_t ); + dap_hash_fast( l_tx, l_tx_size, l_key_hash ); + //DAP_DELETE( l_tx ); + + return l_key_hash; +} + +/** + * Make transfer transaction & insert to cache + * + * return 0 Ok, -2 not enough funds to transfer, -1 other Error + */ +dap_chain_hash_fast_t* dap_chain_mempool_tx_create_cond(dap_chain_net_t * a_net, + dap_enc_key_t *a_key_from, dap_enc_key_t *a_key_cond, + const dap_chain_addr_t* a_addr_from, + const char a_token_ticker[DAP_CHAIN_TICKER_SIZE_MAX], + uint64_t a_value,uint64_t a_value_per_unit_max, dap_chain_net_srv_price_unit_uid_t a_unit, + dap_chain_net_srv_uid_t a_srv_uid, uint64_t a_value_fee, const void *a_cond, size_t a_cond_size) +{ + // Make transfer transaction + dap_chain_datum_t *l_datum = dap_chain_tx_create_cond(a_net,a_key_from, a_key_cond, a_addr_from, + a_token_ticker,a_value,a_value_per_unit_max, a_unit, + a_srv_uid, a_value_fee, a_cond, a_cond_size); + + if(!l_datum) + return NULL; + + dap_chain_datum_tx_t *l_tx = (dap_chain_datum_tx_t*)&(l_datum->data); + size_t l_tx_size = l_datum->header.data_size;//dap_chain_datum_tx_get_size( l_tx ); + + dap_chain_hash_fast_t *l_key_hash = DAP_NEW_Z( dap_chain_hash_fast_t ); + dap_hash_fast( l_tx, l_tx_size, l_key_hash ); + //DAP_DELETE( l_tx ); + + char * l_key_str = dap_chain_hash_fast_to_str_new( l_key_hash ); + char * l_gdb_group = dap_chain_net_get_gdb_group_mempool_by_chain_type( a_net ,CHAIN_TYPE_TX); + if( dap_chain_global_db_gr_set( dap_strdup(l_key_str), (uint8_t *) l_datum, dap_chain_datum_size(l_datum) + , l_gdb_group ) ) { + log_it(L_NOTICE, "Transaction %s placed in mempool", l_key_str); + } + DAP_DELETE(l_gdb_group); + DAP_DELETE(l_key_str); + + return l_key_hash; +} + +/** + * Make receipt transaction & insert to cache + * + * return 0 Ok, -2 not enough funds to transfer, -1 other Error + */ +int dap_chain_mempool_tx_create_receipt(uint64_t a_value) +//(dap_enc_key_t *a_key_from, dap_enc_key_t *a_key_cond, +// const dap_chain_addr_t* a_addr_from, const dap_chain_addr_t* a_addr_cond, +// const dap_chain_addr_t* a_addr_fee, const char a_token_ticker[DAP_CHAIN_TICKER_SIZE_MAX], +// uint64_t a_value, uint64_t a_value_fee, const void *a_cond, size_t a_cond_size) +{ + // check valid param +/* if(!a_key_from || !a_key_from->priv_key_data || !a_key_from->priv_key_data_size || + !dap_chain_addr_check_sum(a_addr_from) || !dap_chain_addr_check_sum(a_addr_cond) || + (a_addr_fee && !dap_chain_addr_check_sum(a_addr_fee)) || !a_value) + return -1;*/ +/* + // find the transactions from which to take away coins + dap_list_t *l_list_used_out = NULL; // list of transaction with 'out' items + uint64_t l_value_transfer = 0; // how many coins to transfer + { + dap_chain_hash_fast_t l_tx_cur_hash = { 0 }; + uint64_t l_value_need = a_value + a_value_fee; + while(l_value_transfer < l_value_need) + { + // Get the transaction in the cache by the addr in out item + const dap_chain_datum_tx_t *l_tx = dap_chain_ledger_tx_find_by_addr(a_addr_from, + &l_tx_cur_hash); + if(!l_tx) + break; + // Get all item from transaction by type + int l_item_count = 0; + dap_list_t *l_list_out_items = dap_chain_datum_tx_items_get((dap_chain_datum_tx_t*) l_tx, TX_ITEM_TYPE_OUT, + &l_item_count); + dap_list_t *l_list_tmp = l_list_out_items; + int l_out_idx_tmp = 0; // current index of 'out' item + while(l_list_tmp) { + dap_chain_tx_out_t *out_item = l_list_tmp->data; + // if 'out' item has addr = a_addr_from + if(out_item && &out_item->addr && !memcmp(a_addr_from, &out_item->addr, sizeof(dap_chain_addr_t))) { + + // Check whether used 'out' items + if(!dap_chain_ledger_tx_hash_is_used_out_item(&l_tx_cur_hash, l_out_idx_tmp)) { + + list_used_item_t *item = DAP_NEW(list_used_item_t); + memcpy(&item->tx_hash_fast, &l_tx_cur_hash, sizeof(dap_chain_hash_fast_t)); + item->num_idx_out = l_out_idx_tmp; + item->value = out_item->header.value; + l_list_used_out = dap_list_append(l_list_used_out, item); + l_value_transfer += item->value; + // already accumulated the required value, finish the search for 'out' items + if(l_value_transfer >= l_value_need) { + break; + } + } + } + // go to the next 'out' item in l_tx transaction + l_out_idx_tmp++; + l_list_tmp = dap_list_next(l_list_tmp); + } + dap_list_free(l_list_out_items); + } + + // nothing to tranfer (not enough funds) + if(!l_list_used_out || l_value_transfer < l_value_need) { + dap_list_free_full(l_list_used_out, free); + return -2; + } + } + + // create empty transaction + dap_chain_datum_tx_t *l_tx = dap_chain_datum_tx_create(); + // add 'in' items + { + dap_list_t *l_list_tmp = l_list_used_out; + uint64_t l_value_to_items = 0; // how many coins to transfer + while(l_list_tmp) { + list_used_item_t *item = l_list_tmp->data; + if(dap_chain_datum_tx_add_in_item(&l_tx, &item->tx_hash_fast, item->num_idx_out) == 1) { + l_value_to_items += item->value; + } + l_list_tmp = dap_list_next(l_list_tmp); + } + assert(l_value_to_items == l_value_transfer); + dap_list_free_full(l_list_used_out, free); + } + // add 'out_cond' and 'out' items + { + uint64_t l_value_pack = 0; // how much coin add to 'out' items + if(dap_chain_datum_tx_add_out_cond_item(&l_tx, a_key_cond, (dap_chain_addr_t*) a_addr_cond, a_value, a_cond, + a_cond_size) == 1) { + l_value_pack += a_value; + // transaction fee + if(a_addr_fee) { + if(dap_chain_datum_tx_add_out_item(&l_tx, a_addr_fee, a_value_fee) == 1) + l_value_pack += a_value_fee; + } + } + // coin back + uint64_t l_value_back = l_value_transfer - l_value_pack; + if(l_value_back) { + if(dap_chain_datum_tx_add_out_item(&l_tx, a_addr_from, l_value_back) != 1) { + dap_chain_datum_tx_delete(l_tx); + return -1; + } + } + } + + // add 'sign' items + if(dap_chain_datum_tx_add_sign_item(&l_tx, a_key_from) != 1) { + dap_chain_datum_tx_delete(l_tx); + return -1; + } + + size_t l_tx_size = dap_chain_datum_tx_get_size(l_tx); + dap_chain_datum_t *l_datum = dap_chain_datum_create(DAP_CHAIN_DATUM_TX, l_tx, l_tx_size); + + dap_chain_hash_fast_t l_key_hash; + dap_hash_fast(l_tx, l_tx_size, &l_key_hash); + DAP_DELETE(l_tx); + + char * l_key_str = dap_chain_hash_fast_to_str_new(&l_key_hash); + if(dap_chain_global_db_gr_set(dap_strdup(l_key_str), (uint8_t *) l_datum, dap_chain_datum_size(l_datum) + , c_dap_datum_mempool_gdb_group)) { + log_it(L_NOTICE, "Transaction %s placed in mempool", l_key_str); + // add transaction to ledger + if(dap_chain_ledger_tx_add((dap_chain_datum_tx_t*) l_datum->data) < 0) + log_it(L_ERROR, "Transaction %s not placed in LEDGER", l_key_str); + } + DAP_DELETE(l_key_str);*/ + + return 0; +} + +uint8_t* dap_datum_mempool_serialize(dap_datum_mempool_t *datum_mempool, size_t *size) +{ + size_t a_request_size = 2 * sizeof(uint16_t), shift_size = 0; + for(int i = 0; i < datum_mempool->datum_count; i++) { + a_request_size += dap_chain_datum_size(datum_mempool->data[i]) + sizeof(uint16_t); + } + uint8_t *a_request = DAP_NEW_SIZE(uint8_t, a_request_size); + memcpy(a_request + shift_size, &(datum_mempool->version), sizeof(uint16_t)); + shift_size += sizeof(uint16_t); + memcpy(a_request + shift_size, &(datum_mempool->datum_count), sizeof(uint16_t)); + shift_size += sizeof(uint16_t); + for(int i = 0; i < datum_mempool->datum_count; i++) { + size_t size_one = dap_chain_datum_size(datum_mempool->data[i]); + memcpy(a_request + shift_size, &size_one, sizeof(uint16_t)); + shift_size += sizeof(uint16_t); + memcpy(a_request + shift_size, datum_mempool->data[i], size_one); + shift_size += size_one; + } + assert(shift_size == a_request_size); + if(size) + *size = a_request_size; + return a_request; +} + +dap_datum_mempool_t * dap_datum_mempool_deserialize(uint8_t *a_datum_mempool_ser, size_t a_datum_mempool_ser_size) +{ + size_t shift_size = 0; + //uint8_t *a_datum_mempool_ser = DAP_NEW_Z_SIZE(uint8_t, datum_mempool_size / 2 + 1); + //datum_mempool_size = hex2bin(a_datum_mempool_ser, datum_mempool_str_in, datum_mempool_size) / 2; + dap_datum_mempool_t *datum_mempool = DAP_NEW_Z(dap_datum_mempool_t); + memcpy(&(datum_mempool->version), a_datum_mempool_ser + shift_size, sizeof(uint16_t)); + shift_size += sizeof(uint16_t); + memcpy(&(datum_mempool->datum_count), a_datum_mempool_ser + shift_size, sizeof(uint16_t)); + shift_size += sizeof(uint16_t); + datum_mempool->data = DAP_NEW_Z_SIZE(dap_chain_datum_t*, datum_mempool->datum_count * sizeof(dap_chain_datum_t*)); + for(int i = 0; i < datum_mempool->datum_count; i++) { + size_t size_one = 0; + memcpy(&size_one, a_datum_mempool_ser + shift_size, sizeof(uint16_t)); + shift_size += sizeof(uint16_t); + datum_mempool->data[i] = (dap_chain_datum_t*) DAP_NEW_Z_SIZE(uint8_t, size_one); + memcpy(datum_mempool->data[i], a_datum_mempool_ser + shift_size, size_one); + shift_size += size_one; + datum_mempool->data[i]; + } + assert(shift_size == a_datum_mempool_ser_size); + DAP_DELETE(a_datum_mempool_ser); + return datum_mempool; +} + +void dap_datum_mempool_clean(dap_datum_mempool_t *datum) +{ + if(!datum) + return; + for(int i = 0; i < datum->datum_count; i++) { + DAP_DELETE(datum->data[i]); + } + DAP_DELETE(datum->data); + datum->data = NULL; +} + +void dap_datum_mempool_free(dap_datum_mempool_t *datum) +{ + dap_datum_mempool_clean(datum); + DAP_DELETE(datum); +} + +/** + * + */ +static char* calc_datum_hash(const char *datum_str, size_t datum_size) +{ + dap_chain_hash_fast_t a_hash; + dap_hash_fast( datum_str, datum_size, &a_hash); + size_t a_str_max = (sizeof(a_hash.raw) + 1) * 2 + 2; /* heading 0x */ + char *a_str = DAP_NEW_Z_SIZE(char, a_str_max); + +// size_t hash_len = dap_chain_hash_fast_to_str(&a_hash, a_str, a_str_max); + dap_chain_hash_fast_to_str(&a_hash, a_str, a_str_max); + +// if(!hash_len) { +// DAP_DELETE(a_str); +// return NULL; +// } + + return a_str; +} + +static void enc_http_reply_encode_new(struct dap_http_simple *a_http_simple, dap_enc_key_t * key, + enc_http_delegate_t * a_http_delegate) +{ + //dap_enc_key_t * key = dap_enc_ks_find_http(a_http_simple->http); + if(key == NULL) { + log_it(L_ERROR, "Can't find http key."); + return; + } + if(a_http_delegate->response) { + + if(a_http_simple->reply) + free(a_http_simple->reply); + + size_t l_reply_size_max = dap_enc_code_out_size(a_http_delegate->key, + a_http_delegate->response_size, + DAP_ENC_DATA_TYPE_RAW); + + a_http_simple->reply = DAP_NEW_SIZE(void, l_reply_size_max); + a_http_simple->reply_size = dap_enc_code(a_http_delegate->key, + a_http_delegate->response, a_http_delegate->response_size, + a_http_simple->reply, l_reply_size_max, + DAP_ENC_DATA_TYPE_RAW); + + /*/ decode test + size_t l_response_dec_size_max = a_http_simple->reply_size ? a_http_simple->reply_size * 2 + 16 : 0; + char * l_response_dec = a_http_simple->reply_size ? DAP_NEW_Z_SIZE(char, l_response_dec_size_max) : NULL; + size_t l_response_dec_size = 0; + if(a_http_simple->reply_size) + l_response_dec_size = dap_enc_decode(a_http_delegate->key, + a_http_simple->reply, a_http_simple->reply_size, + l_response_dec, l_response_dec_size_max, + DAP_ENC_DATA_TYPE_RAW); + l_response_dec_size_max = 0;*/ + } + +} + +/** + * @brief + * @param cl_st HTTP server instance + * @param arg for return code + */ +void chain_mempool_proc(struct dap_http_simple *cl_st, void * arg) +{ + http_status_code_t * return_code = (http_status_code_t*) arg; + // save key while it alive, i.e. still exist + dap_enc_key_t *key = dap_enc_ks_find_http(cl_st->http); + //dap_enc_key_serealize_t *key_ser = dap_enc_key_serealize(key_tmp); + //dap_enc_key_t *key = dap_enc_key_deserealize(key_ser, sizeof(dap_enc_key_serealize_t)); + + // read header + dap_http_header_t *hdr_session_close_id = + (cl_st->http) ? dap_http_header_find(cl_st->http->in_headers, "SessionCloseAfterRequest") : NULL; + dap_http_header_t *hdr_key_id = + (hdr_session_close_id && cl_st->http) ? dap_http_header_find(cl_st->http->in_headers, "KeyID") : NULL; + + enc_http_delegate_t *dg = enc_http_request_decode(cl_st); + if(dg) { + char *suburl = dg->url_path; + char *request_str = dg->request_str; + int request_size = (int) dg->request_size; + //printf("!!***!!! chain_mempool_proc arg=%d suburl=%s str=%s len=%d\n", arg, suburl, request_str, request_size); + if(request_str && request_size > 1) { + // find what to do + uint8_t action = DAP_DATUM_MEMPOOL_NONE; //*(uint8_t*) request_str; + if(dg->url_path_size > 0) { + if(!strcmp(suburl, "add")) + action = DAP_DATUM_MEMPOOL_ADD; + else if(!strcmp(suburl, "check")) + action = DAP_DATUM_MEMPOOL_CHECK; + else if(!strcmp(suburl, "del")) + action = DAP_DATUM_MEMPOOL_DEL; + } + dap_datum_mempool_t *datum_mempool = + (action != DAP_DATUM_MEMPOOL_NONE) ? + dap_datum_mempool_deserialize((uint8_t*) request_str, (size_t) request_size) : NULL; + if(datum_mempool) + { + dap_datum_mempool_free(datum_mempool); + char *a_key = calc_datum_hash(request_str, (size_t) request_size); + char *a_value; + switch (action) + { + case DAP_DATUM_MEMPOOL_ADD: // add datum in base + //a_value = DAP_NEW_Z_SIZE(char, request_size * 2); + //bin2hex((char*) a_value, (const unsigned char*) request_str, request_size); + if(dap_chain_global_db_gr_set(dap_strdup(a_key), request_str,(size_t) request_size, + dap_config_get_item_str_default(g_config, "mempool", "gdb_group", "datum-pool"))) { + *return_code = Http_Status_OK; + } + log_it(L_INFO, "Insert hash: key=%s result:%s", a_key, + (*return_code == Http_Status_OK) ? "OK" : "False!"); + DAP_DELETE(a_key); + break; + + case DAP_DATUM_MEMPOOL_CHECK: // check datum in base + + strcpy(cl_st->reply_mime, "text/text"); + char *str = dap_chain_global_db_gr_get( dap_strdup(a_key) , NULL, + dap_config_get_item_str_default(g_config, "mempool", "gdb_group", "datum-pool")); + if(str) { + dg->response = strdup("1"); + DAP_DELETE(str); + log_it(L_INFO, "Check hash: key=%s result: Present", a_key); + } + else + { + dg->response = strdup("0"); + log_it(L_INFO, "Check hash: key=%s result: Absent", a_key); + } + dg->response_size = strlen(dg->response); + *return_code = Http_Status_OK; + enc_http_reply_encode_new(cl_st, key, dg); + break; + + case DAP_DATUM_MEMPOOL_DEL: // delete datum in base + strcpy(cl_st->reply_mime, "text/text"); + if(dap_chain_global_db_gr_del( dap_strdup(a_key), + dap_config_get_item_str_default(g_config, "mempool", "gdb_group", "datum-pool"))) { + dg->response = strdup("1"); + + log_it(L_INFO, "Delete hash: key=%s result: Ok", a_key); + } + else + { + dg->response = strdup("0"); + log_it(L_INFO, "Delete hash: key=%s result: False!", a_key); + } + *return_code = Http_Status_OK; + enc_http_reply_encode_new(cl_st, key, dg); + break; + + default: // unsupported command + log_it(L_INFO, "Unknown request=%s! key=%s", (suburl) ? suburl : "-", a_key); + DAP_DELETE(a_key); + enc_http_delegate_delete(dg); + if(key) + dap_enc_key_delete(key); + return; + } + DAP_DELETE(a_key); + } + else + *return_code = Http_Status_BadRequest; + } + else + *return_code = Http_Status_BadRequest; + enc_http_delegate_delete(dg); + } + else { + *return_code = Http_Status_Unauthorized; + } + if(hdr_session_close_id && hdr_session_close_id->value && !strcmp(hdr_session_close_id->value, "yes")) { + // close session + if(hdr_key_id && hdr_key_id->value) { + dap_enc_ks_delete(hdr_key_id->value); + } + } +} + +/** + * @brief chain_mempool_add_proc + * @param sh HTTP server instance + * @param url URL string + */ +void dap_chain_mempool_add_proc(dap_http_t * a_http_server, const char * a_url) +{ + dap_http_simple_proc_add(a_http_server, a_url, 4096, chain_mempool_proc); +} diff --git a/libdap-chain-mempool/dap_chain_mempool.h b/libdap-chain-mempool/dap_chain_mempool.h new file mode 100755 index 0000000000000000000000000000000000000000..abe82a61fcae79d6afb4fc9cb3fa33f91b98688d --- /dev/null +++ b/libdap-chain-mempool/dap_chain_mempool.h @@ -0,0 +1,75 @@ +#pragma once + +#include <stdint.h> +#include "dap_chain_datum.h" +#include "dap_chain_net.h" +#include "dap_chain_ledger.h" +#include "dap_http.h" +/* + // datum mempool structure + typedef struct dap_datum_mempool { + int16_t version; // structure version + uint16_t datum_count; // datums count + struct { + int32_t datum_size; + dap_chain_datum_t *datum; + }DAP_ALIGN_PACKED data[]; // mass of datums + }DAP_ALIGN_PACKED dap_datum_mempool_t; + */ + +#define DAP_DATUM_MEMPOOL_VERSION "01" + +// action +enum { + DAP_DATUM_MEMPOOL_NONE = 0, DAP_DATUM_MEMPOOL_ADD, DAP_DATUM_MEMPOOL_CHECK, DAP_DATUM_MEMPOOL_DEL +}; + +// datum mempool structure +typedef struct dap_datum_mempool { + uint16_t version; // structure version + uint16_t datum_count; // datums count + dap_chain_datum_t **data;// mass of datums +}DAP_ALIGN_PACKED dap_datum_mempool_t; + +int dap_datum_mempool_init(void); + +extern const char* c_dap_datum_mempool_gdb_group; + +uint8_t* dap_datum_mempool_serialize(dap_datum_mempool_t *datum_mempool, size_t *size); +dap_datum_mempool_t * dap_datum_mempool_deserialize(uint8_t *datum_mempool_str, size_t size); + +void dap_datum_mempool_clean(dap_datum_mempool_t *datum); +void dap_datum_mempool_free(dap_datum_mempool_t *datum); + +void dap_chain_mempool_add_proc(dap_http_t * a_http_server, const char * a_url); + +int dap_chain_mempool_tx_create(dap_chain_t * a_chain, dap_enc_key_t *a_key_from, + const dap_chain_addr_t* a_addr_from, const dap_chain_addr_t* a_addr_to, + const dap_chain_addr_t* a_addr_fee, + const char a_token_ticker[DAP_CHAIN_TICKER_SIZE_MAX], + uint64_t a_value, uint64_t a_value_fee); + +// Make transfer transaction & insert to cache +dap_chain_hash_fast_t* dap_chain_proc_tx_create_cond(dap_chain_net_t * a_net, + dap_enc_key_t *a_key_from, dap_enc_key_t *a_key_cond, + const dap_chain_addr_t* a_addr_from, + const char a_token_ticker[DAP_CHAIN_TICKER_SIZE_MAX], + uint64_t a_value,uint64_t a_value_per_unit_max, dap_chain_net_srv_price_unit_uid_t a_unit, + dap_chain_net_srv_uid_t a_srv_uid, uint64_t a_value_fee, const void *a_cond, size_t a_cond_size); +dap_chain_hash_fast_t* dap_chain_mempool_tx_create_cond(dap_chain_net_t * a_net, + dap_enc_key_t *a_key_from, dap_enc_key_t *a_key_cond, + const dap_chain_addr_t* a_addr_from, + const char a_token_ticker[DAP_CHAIN_TICKER_SIZE_MAX], + uint64_t a_value,uint64_t a_value_per_unit_max, dap_chain_net_srv_price_unit_uid_t a_unit, + dap_chain_net_srv_uid_t a_srv_uid, uint64_t a_value_fee, const void *a_cond, size_t a_cond_size); + +dap_chain_hash_fast_t* dap_chain_mempool_tx_create_cond_input(dap_chain_net_t * a_net,dap_chain_hash_fast_t *a_tx_prev_hash, + const dap_chain_addr_t* a_addr_to, dap_enc_key_t * l_key_tx_sign, dap_chain_datum_tx_receipt_t * l_receipt, size_t l_receipt_size); + + +int dap_chain_mempool_datum_add(dap_chain_datum_t * a_datum, dap_chain_t * a_chain ); +int dap_chain_mempool_tx_create_massive( dap_chain_t * a_chain, dap_enc_key_t *a_key_from, + const dap_chain_addr_t* a_addr_from, const dap_chain_addr_t* a_addr_to, + const dap_chain_addr_t* a_addr_fee, + const char a_token_ticker[DAP_CHAIN_TICKER_SIZE_MAX], + uint64_t a_value, uint64_t a_value_fee,size_t a_tx_num);