You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
887 lines
38 KiB
887 lines
38 KiB
/******************************************************************************
|
|
* Copyright © 2014-2017 The SuperNET Developers. *
|
|
* *
|
|
* See the AUTHORS, DEVELOPER-AGREEMENT and LICENSE files at *
|
|
* the top-level directory of this distribution for the individual copyright *
|
|
* holder information and the developer policies on copyright and licensing. *
|
|
* *
|
|
* Unless otherwise agreed in a custom licensing agreement, no part of the *
|
|
* SuperNET software, including this file may be copied, modified, propagated *
|
|
* or distributed except according to the terms contained in the LICENSE file *
|
|
* *
|
|
* Removal or modification of this copyright notice is prohibited. *
|
|
* *
|
|
******************************************************************************/
|
|
|
|
// included from basilisk.c
|
|
// connect DEX to jumblr-core
|
|
|
|
/*
|
|
z_exportkey "zaddr"
|
|
z_exportwallet "filename"
|
|
z_getoperationstatus (["operationid", ... ])
|
|
z_gettotalbalance ( minconf )
|
|
z_importkey "zkey" ( rescan )
|
|
z_importwallet "filename"
|
|
z_listaddresses
|
|
z_sendmany "fromaddress" [{"address":... ,"amount":..., "memo":"<hex>"},...] ( minconf ) ( fee )
|
|
*/
|
|
|
|
#define JUMBLR_INCR 99.65
|
|
#define JUMBLR_TXFEE 0.01
|
|
#define JUMBLR_ADDR "RGhxXpXSSBTBm9EvNsXnTQczthMCxHX91t"
|
|
#define JUMBLR_BTCADDR "18RmTJe9qMech8siuhYfMtHo8RtcN1obC6"
|
|
#define JUMBLR_FEE 0.001
|
|
|
|
int32_t jumblr_addresstype(struct supernet_info *myinfo,struct iguana_info *coin,char *addr)
|
|
{
|
|
if ( strcmp(coin->symbol,"KMD") == 0 )
|
|
{
|
|
if ( addr[0] == 'z' && addr[1] == 'c' && strlen(addr) >= 40 )
|
|
return('z');
|
|
else if ( strlen(addr) < 40 )
|
|
return('t');
|
|
else return(-1);
|
|
} else return('t');
|
|
}
|
|
|
|
struct jumblr_item *jumblr_opidfind(struct supernet_info *myinfo,char *opid)
|
|
{
|
|
struct jumblr_item *ptr;
|
|
HASH_FIND(hh,myinfo->jumblrs,opid,(int32_t)strlen(opid),ptr);
|
|
return(ptr);
|
|
}
|
|
|
|
struct jumblr_item *jumblr_opidadd(struct supernet_info *myinfo,struct iguana_info *coin,char *opid)
|
|
{
|
|
struct jumblr_item *ptr;
|
|
if ( (ptr= jumblr_opidfind(myinfo,opid)) == 0 )
|
|
{
|
|
ptr = calloc(1,sizeof(*ptr));
|
|
safecopy(ptr->opid,opid,sizeof(ptr->opid));
|
|
HASH_ADD_KEYPTR(hh,myinfo->jumblrs,ptr->opid,(int32_t)strlen(ptr->opid),ptr);
|
|
if ( ptr != jumblr_opidfind(myinfo,opid) )
|
|
printf("jumblr_opidadd.(%s) ERROR, couldnt find after add\n",opid);
|
|
}
|
|
return(ptr);
|
|
}
|
|
|
|
char *jumblr_validateaddress(struct supernet_info *myinfo,struct iguana_info *coin,char *addr)
|
|
{
|
|
char params[1024];
|
|
if ( coin->FULLNODE < 0 )
|
|
{
|
|
sprintf(params,"[\"%s\"]",addr);
|
|
return(bitcoind_passthru(coin->symbol,coin->chain->serverport,coin->chain->userpass,"validateaddress",params));
|
|
} else return(_dex_validateaddress(myinfo,coin->symbol,addr));
|
|
}
|
|
|
|
int32_t jumblr_ismine(struct supernet_info *myinfo,struct iguana_info *coin,char *addr)
|
|
{
|
|
char params[1024],*retstr; cJSON *retjson,*obj; int32_t retval = -1;
|
|
sprintf(params,"[\"%s\"]",addr);
|
|
if ( (retstr= jumblr_validateaddress(myinfo,coin,addr)) != 0 )
|
|
{
|
|
if ( (retjson= cJSON_Parse(retstr)) != 0 )
|
|
{
|
|
if ( (obj= jobj(retjson,"ismine")) != 0 && is_cJSON_True(obj) != 0 )
|
|
retval = 1;
|
|
else retval = 0;
|
|
free_json(retjson);
|
|
}
|
|
free(retstr);
|
|
}
|
|
return(retval);
|
|
}
|
|
|
|
char *jumblr_zgetnewaddress(struct supernet_info *myinfo,struct iguana_info *coin)
|
|
{
|
|
return(bitcoind_passthru(coin->symbol,coin->chain->serverport,coin->chain->userpass,"z_getnewaddress",""));
|
|
}
|
|
|
|
char *jumblr_zlistoperationids(struct supernet_info *myinfo,struct iguana_info *coin)
|
|
{
|
|
return(bitcoind_passthru(coin->symbol,coin->chain->serverport,coin->chain->userpass,"z_listoperationids",""));
|
|
}
|
|
|
|
char *jumblr_zgetoperationresult(struct supernet_info *myinfo,struct iguana_info *coin,char *opid)
|
|
{
|
|
char params[1024];
|
|
sprintf(params,"[[\"%s\"]]",opid);
|
|
return(bitcoind_passthru(coin->symbol,coin->chain->serverport,coin->chain->userpass,"z_getoperationresult",params));
|
|
}
|
|
|
|
char *jumblr_zgetoperationstatus(struct supernet_info *myinfo,struct iguana_info *coin,char *opid)
|
|
{
|
|
char params[1024];
|
|
sprintf(params,"[[\"%s\"]]",opid);
|
|
return(bitcoind_passthru(coin->symbol,coin->chain->serverport,coin->chain->userpass,"z_getoperationstatus",params));
|
|
}
|
|
|
|
char *jumblr_sendt_to_z(struct supernet_info *myinfo,struct iguana_info *coin,char *taddr,char *zaddr,double amount)
|
|
{
|
|
char params[1024]; double fee = (amount-3*JUMBLR_TXFEE) * JUMBLR_FEE;
|
|
if ( jumblr_addresstype(myinfo,coin,zaddr) != 'z' || jumblr_addresstype(myinfo,coin,taddr) != 't' )
|
|
return(clonestr("{\"error\":\"illegal address in t to z\"}"));
|
|
sprintf(params,"[\"%s\", [{\"address\":\"%s\",\"amount\":%.8f}, {\"address\":\"%s\",\"amount\":%.8f}], 1, %.8f]",taddr,zaddr,amount-fee-JUMBLR_TXFEE,JUMBLR_ADDR,fee,JUMBLR_TXFEE);
|
|
return(bitcoind_passthru(coin->symbol,coin->chain->serverport,coin->chain->userpass,"z_sendmany",params));
|
|
}
|
|
|
|
char *jumblr_sendz_to_z(struct supernet_info *myinfo,struct iguana_info *coin,char *zaddrS,char *zaddrD,double amount)
|
|
{
|
|
char params[1024]; double fee = (amount-2*JUMBLR_TXFEE) * JUMBLR_FEE;
|
|
if ( jumblr_addresstype(myinfo,coin,zaddrS) != 'z' || jumblr_addresstype(myinfo,coin,zaddrD) != 'z' )
|
|
return(clonestr("{\"error\":\"illegal address in z to z\"}"));
|
|
sprintf(params,"[\"%s\", [{\"address\":\"%s\",\"amount\":%.8f}, {\"address\":\"%s\",\"amount\":%.8f}], 1, %.8f]",zaddrS,zaddrD,amount-fee-JUMBLR_TXFEE,JUMBLR_ADDR,fee,JUMBLR_TXFEE);
|
|
return(bitcoind_passthru(coin->symbol,coin->chain->serverport,coin->chain->userpass,"z_sendmany",params));
|
|
}
|
|
|
|
char *jumblr_sendz_to_t(struct supernet_info *myinfo,struct iguana_info *coin,char *zaddr,char *taddr,double amount)
|
|
{
|
|
char params[1024]; double fee = (amount-JUMBLR_TXFEE) * JUMBLR_FEE;
|
|
if ( jumblr_addresstype(myinfo,coin,zaddr) != 'z' || jumblr_addresstype(myinfo,coin,taddr) != 't' )
|
|
return(clonestr("{\"error\":\"illegal address in z to t\"}"));
|
|
sprintf(params,"[\"%s\", [{\"address\":\"%s\",\"amount\":%.8f}, {\"address\":\"%s\",\"amount\":%.8f}], 1, %.8f]",zaddr,taddr,amount-fee-JUMBLR_TXFEE,JUMBLR_ADDR,fee,JUMBLR_TXFEE);
|
|
return(bitcoind_passthru(coin->symbol,coin->chain->serverport,coin->chain->userpass,"z_sendmany",params));
|
|
}
|
|
|
|
char *jumblr_zlistreceivedbyaddress(struct supernet_info *myinfo,struct iguana_info *coin,char *addr)
|
|
{
|
|
char params[1024];
|
|
sprintf(params,"[\"%s\", 1]",addr);
|
|
return(bitcoind_passthru(coin->symbol,coin->chain->serverport,coin->chain->userpass,"z_listreceivedbyaddress",params));
|
|
}
|
|
|
|
char *jumblr_getreceivedbyaddress(struct supernet_info *myinfo,struct iguana_info *coin,char *addr)
|
|
{
|
|
char params[1024];
|
|
sprintf(params,"[\"%s\", 1]",addr);
|
|
return(bitcoind_passthru(coin->symbol,coin->chain->serverport,coin->chain->userpass,"getreceivedbyaddress",params));
|
|
}
|
|
|
|
char *jumblr_importprivkey(struct supernet_info *myinfo,struct iguana_info *coin,char *wifstr)
|
|
{
|
|
char params[1024];
|
|
sprintf(params,"[\"%s\", \"\", false]",wifstr);
|
|
return(bitcoind_passthru(coin->symbol,coin->chain->serverport,coin->chain->userpass,"importprivkey",params));
|
|
}
|
|
|
|
char *jumblr_zgetbalance(struct supernet_info *myinfo,struct iguana_info *coin,char *addr)
|
|
{
|
|
char params[1024];
|
|
sprintf(params,"[\"%s\", 1]",addr);
|
|
return(bitcoind_passthru(coin->symbol,coin->chain->serverport,coin->chain->userpass,"z_getbalance",params));
|
|
}
|
|
|
|
char *jumblr_listunspent(struct supernet_info *myinfo,struct iguana_info *coin,char *coinaddr)
|
|
{
|
|
char params[1024];
|
|
if ( coin->FULLNODE == 0 )
|
|
return(dex_listunspent(myinfo,coin,0,0,coin->symbol,coinaddr));
|
|
sprintf(params,"[1, 99999999, [\"%s\"]]",coinaddr);
|
|
return(bitcoind_passthru(coin->symbol,coin->chain->serverport,coin->chain->userpass,"listunspent",params));
|
|
}
|
|
|
|
int64_t jumblr_receivedby(struct supernet_info *myinfo,struct iguana_info *coin,char *addr)
|
|
{
|
|
char *retstr; int64_t total = 0;
|
|
if ( (retstr= jumblr_getreceivedbyaddress(myinfo,coin,addr)) != 0 )
|
|
{
|
|
total = atof(retstr) * SATOSHIDEN;
|
|
free(retstr);
|
|
}
|
|
return(total);
|
|
}
|
|
|
|
int64_t jumblr_balance(struct supernet_info *myinfo,struct iguana_info *coin,char *addr)
|
|
{
|
|
char *retstr; double val; cJSON *retjson; int32_t i,n; int64_t balance = 0;
|
|
if ( jumblr_addresstype(myinfo,coin,addr) == 't' )
|
|
{
|
|
if ( coin->FULLNODE < 0 && jumblr_ismine(myinfo,coin,addr) > 0 )
|
|
{
|
|
if ( (retstr= jumblr_listunspent(myinfo,coin,addr)) != 0 )
|
|
{
|
|
printf("jumblr.[%s].(%s)\n",coin->symbol,retstr);
|
|
if ( (retjson= cJSON_Parse(retstr)) != 0 )
|
|
{
|
|
if ( (n= cJSON_GetArraySize(retjson)) > 0 )
|
|
for (i=0; i<n; i++)
|
|
balance += SATOSHIDEN * jdouble(jitem(retjson,i),"amount");
|
|
free_json(retjson);
|
|
}
|
|
free(retstr);
|
|
}
|
|
}
|
|
else if ( (retstr= dex_getbalance(myinfo,coin,0,0,coin->symbol,addr)) != 0 )
|
|
{
|
|
//printf("DEX retstr.(%s)\n",retstr);
|
|
if ( (retjson= cJSON_Parse(retstr)) != 0 )
|
|
{
|
|
balance = jdouble(retjson,"balance") * SATOSHIDEN;
|
|
free_json(retjson);
|
|
}
|
|
free(retstr);
|
|
}
|
|
}
|
|
else if ( (retstr= jumblr_zgetbalance(myinfo,coin,addr)) != 0 )
|
|
{
|
|
if ( (val= atof(retstr)) > SMALLVAL )
|
|
balance = val * SATOSHIDEN;
|
|
free(retstr);
|
|
}
|
|
return(balance);
|
|
}
|
|
|
|
int32_t jumblr_itemset(struct jumblr_item *ptr,cJSON *item,char *status)
|
|
{
|
|
cJSON *params,*amounts,*dest; char *from,*addr; int32_t i,n; int64_t amount;
|
|
/*"params" : {
|
|
"fromaddress" : "RDhEGYScNQYetCyG75Kf8Fg61UWPdwc1C5",
|
|
"amounts" : [
|
|
{
|
|
"address" : "zc9s3UdkDFTnnwHrMCr1vYy2WmkjhmTxXNiqC42s7BjeKBVUwk766TTSsrRPKfnX31Bbu8wbrTqnjDqskYGwx48FZMPHvft",
|
|
"amount" : 3.00000000
|
|
}
|
|
],
|
|
"minconf" : 1,
|
|
"fee" : 0.00010000
|
|
}*/
|
|
if ( (params= jobj(item,"params")) != 0 )
|
|
{
|
|
//printf("params.(%s)\n",jprint(params,0));
|
|
if ( (from= jstr(params,"fromaddress")) != 0 )
|
|
{
|
|
safecopy(ptr->src,from,sizeof(ptr->src));
|
|
}
|
|
if ( (amounts= jarray(&n,params,"amounts")) != 0 )
|
|
{
|
|
for (i=0; i<n; i++)
|
|
{
|
|
dest = jitem(amounts,i);
|
|
//printf("%s ",jprint(dest,0));
|
|
if ( (addr= jstr(dest,"address")) != 0 && (amount= jdouble(dest,"amount")*SATOSHIDEN) > 0 )
|
|
{
|
|
if ( strcmp(addr,JUMBLR_ADDR) == 0 )
|
|
ptr->fee = amount;
|
|
else
|
|
{
|
|
ptr->amount = amount;
|
|
safecopy(ptr->dest,addr,sizeof(ptr->dest));
|
|
}
|
|
}
|
|
}
|
|
}
|
|
ptr->txfee = jdouble(params,"fee") * SATOSHIDEN;
|
|
}
|
|
return(1);
|
|
}
|
|
|
|
void jumblr_opidupdate(struct supernet_info *myinfo,struct iguana_info *coin,struct jumblr_item *ptr)
|
|
{
|
|
char *retstr,*status,KMDjumblr[64],KMDdeposit[64],BTCaddr[64]; cJSON *retjson,*item;
|
|
if ( ptr->status == 0 )
|
|
{
|
|
if ( (retstr= jumblr_zgetoperationstatus(myinfo,coin,ptr->opid)) != 0 )
|
|
{
|
|
if ( (retjson= cJSON_Parse(retstr)) != 0 )
|
|
{
|
|
if ( cJSON_GetArraySize(retjson) == 1 )
|
|
{
|
|
item = jitem(retjson,0);
|
|
//printf("%s\n",jprint(item,0));
|
|
if ( (status= jstr(item,"status")) != 0 )
|
|
{
|
|
if ( strcmp(status,"success") == 0 )
|
|
{
|
|
ptr->status = jumblr_itemset(ptr,item,status);
|
|
jumblr_privkey(myinfo,BTCaddr,0,KMDdeposit,JUMBLR_DEPOSITPREFIX);
|
|
jumblr_privkey(myinfo,BTCaddr,0,KMDjumblr,"");
|
|
if ( (jumblr_addresstype(myinfo,coin,ptr->src) == 't' && jumblr_addresstype(myinfo,coin,ptr->src) == 'z' && strcmp(ptr->src,KMDdeposit) != 0) || (jumblr_addresstype(myinfo,coin,ptr->src) == 'z' && jumblr_addresstype(myinfo,coin,ptr->src) == 't' && strcmp(ptr->dest,KMDjumblr) != 0) )
|
|
{
|
|
printf("a non-jumblr t->z pruned\n");
|
|
free(jumblr_zgetoperationresult(myinfo,coin,ptr->opid));
|
|
ptr->status = -1;
|
|
}
|
|
|
|
}
|
|
else if ( strcmp(status,"failed") == 0 )
|
|
{
|
|
printf("%s failed\n",ptr->opid);
|
|
free(jumblr_zgetoperationresult(myinfo,coin,ptr->opid));
|
|
ptr->status = -1;
|
|
}
|
|
}
|
|
}
|
|
free_json(retjson);
|
|
}
|
|
free(retstr);
|
|
}
|
|
}
|
|
}
|
|
|
|
void jumblr_prune(struct supernet_info *myinfo,struct iguana_info *coin,struct jumblr_item *ptr)
|
|
{
|
|
struct jumblr_item *tmp; char oldsrc[128]; int32_t flag = 1;
|
|
printf("PRUNE %s\n",ptr->opid);
|
|
strcpy(oldsrc,ptr->src);
|
|
free(jumblr_zgetoperationresult(myinfo,coin,ptr->opid));
|
|
while ( flag != 0 )
|
|
{
|
|
flag = 0;
|
|
HASH_ITER(hh,myinfo->jumblrs,ptr,tmp)
|
|
{
|
|
if ( strcmp(oldsrc,ptr->dest) == 0 )
|
|
{
|
|
printf("prune %s (%s -> %s) matched oldsrc\n",ptr->opid,ptr->src,ptr->dest);
|
|
free(jumblr_zgetoperationresult(myinfo,coin,ptr->opid));
|
|
strcpy(oldsrc,ptr->src);
|
|
flag = 1;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void jumblr_opidsupdate(struct supernet_info *myinfo,struct iguana_info *coin)
|
|
{
|
|
char *retstr; cJSON *array; int32_t i,n; struct jumblr_item *ptr;
|
|
if ( (retstr= jumblr_zlistoperationids(myinfo,coin)) != 0 )
|
|
{
|
|
if ( (array= cJSON_Parse(retstr)) != 0 )
|
|
{
|
|
if ( (n= cJSON_GetArraySize(array)) > 0 )
|
|
{
|
|
for (i=0; i<n; i++)
|
|
{
|
|
if ( (ptr= jumblr_opidadd(myinfo,coin,jstri(array,i))) != 0 )
|
|
{
|
|
if ( ptr->status == 0 )
|
|
jumblr_opidupdate(myinfo,coin,ptr);
|
|
//printf("%d: %s -> %s %.8f\n",ptr->status,ptr->src,ptr->dest,dstr(ptr->amount));
|
|
if ( jumblr_addresstype(myinfo,coin,ptr->src) == 'z' && jumblr_addresstype(myinfo,coin,ptr->dest) == 't' )
|
|
jumblr_prune(myinfo,coin,ptr);
|
|
}
|
|
}
|
|
}
|
|
free_json(array);
|
|
}
|
|
free(retstr);
|
|
}
|
|
}
|
|
|
|
bits256 jumblr_privkey(struct supernet_info *myinfo,char *coinaddr,uint8_t pubtype,char *KMDaddr,char *prefix)
|
|
{
|
|
bits256 privkey,pubkey; uint8_t pubkey33[33]; char passphrase[sizeof(myinfo->jumblr_passphrase) + 64];
|
|
sprintf(passphrase,"%s%s",prefix,myinfo->jumblr_passphrase);
|
|
if ( myinfo->jumblr_passphrase[0] == 0 )
|
|
strcpy(myinfo->jumblr_passphrase,"password");
|
|
conv_NXTpassword(privkey.bytes,pubkey.bytes,(uint8_t *)passphrase,(int32_t)strlen(passphrase));
|
|
bitcoin_pubkey33(myinfo->ctx,pubkey33,privkey);
|
|
bitcoin_address(coinaddr,pubtype,pubkey33,33);
|
|
bitcoin_address(KMDaddr,60,pubkey33,33);
|
|
//printf("(%s) -> (%s %s)\n",passphrase,coinaddr,KMDaddr);
|
|
return(privkey);
|
|
}
|
|
|
|
int64_t jumblr_DEXsplit(struct supernet_info *myinfo,struct iguana_info *coin,bits256 *splittxidp,char *coinaddr,bits256 txid,int32_t vout,int64_t remaining,double bigprice,double middleprice,double smallprice,double fees[4],cJSON *privkeys,double esttxfee)
|
|
{
|
|
int64_t values[4],outputs[64],value,total,estfee; int32_t i,n,success=0,completed,sendflag,numoutputs = 0; char *retstr; cJSON *retjson,*utxo,*item;
|
|
total = 0;
|
|
estfee = SATOSHIDEN * esttxfee;
|
|
memset(values,0,sizeof(values));
|
|
memset(outputs,0,sizeof(outputs));
|
|
if ( bigprice > SMALLVAL )
|
|
values[0] = SATOSHIDEN * bigprice;
|
|
if ( middleprice > SMALLVAL )
|
|
values[1] = SATOSHIDEN * middleprice;
|
|
if ( smallprice > SMALLVAL )
|
|
values[2] = SATOSHIDEN * smallprice;
|
|
for (i=0; i<4; i++)
|
|
{
|
|
if ( fees[i] > SMALLVAL )
|
|
values[3+i] = SATOSHIDEN * fees[i];
|
|
}
|
|
for (i=0; i<7; i++)
|
|
{
|
|
if ( (value= values[i]) != 0 )
|
|
{
|
|
n = 0;
|
|
while ( n < 10 && remaining > value && numoutputs < sizeof(outputs)/sizeof(*outputs) )
|
|
{
|
|
outputs[numoutputs++] = value;
|
|
remaining -= value;
|
|
total += value;
|
|
printf("%.8f ",dstr(value));
|
|
n++;
|
|
}
|
|
}
|
|
}
|
|
char str[65]; printf("numoutputs.%d total %.8f %s/v%d\n",numoutputs,dstr(total),bits256_str(str,txid),vout);
|
|
if ( numoutputs > 1 ) // no point to make just one
|
|
{
|
|
if ( (retstr= _dex_gettxout(myinfo,coin->symbol,txid,vout)) != 0 )
|
|
{
|
|
item = cJSON_Parse(retstr);
|
|
jaddbits256(item,"txid",txid);
|
|
jaddnum(item,"vout",vout);
|
|
free(retstr);
|
|
if ( item != 0 )
|
|
{
|
|
utxo = cJSON_CreateArray();
|
|
jaddi(utxo,item);
|
|
sendflag = 0;
|
|
///printf("jitem.(%s)\n",jprint(utxo,0));
|
|
if ( (retstr= iguana_utxorawtx(myinfo,coin,0,coinaddr,coinaddr,outputs,numoutputs,0,&completed,sendflag,utxo,privkeys)) != 0 )
|
|
{
|
|
if ( completed != 0 )
|
|
{
|
|
if ( (retjson= cJSON_Parse(retstr)) != 0 )
|
|
{
|
|
if ( jobj(retjson,"error") == 0 && jobj(retjson,"sent") != 0 )
|
|
{
|
|
*splittxidp = jbits256(retjson,"sent");
|
|
success = 1;
|
|
printf("DEXsplit success %.8f\n",dstr(total));
|
|
}
|
|
free_json(retjson);
|
|
}
|
|
}
|
|
free(retstr);
|
|
}
|
|
free_json(utxo);
|
|
}
|
|
}
|
|
}
|
|
return(success * total);
|
|
}
|
|
|
|
double jumblr_DEXutxosize(double *targetvolBp,double *targetvolMp,double *targetvolSp,int32_t isbob,double kmdprice)
|
|
{
|
|
double fee,depositfactor = (isbob == 0) ? 1. : 1.2;
|
|
fee = JUMBLR_INCR * JUMBLR_FEE;
|
|
*targetvolBp = depositfactor * kmdprice * ((JUMBLR_INCR + 3*fee)*100 + 3*JUMBLR_TXFEE);
|
|
*targetvolMp = depositfactor * kmdprice * ((JUMBLR_INCR + 3*fee)*10 + 3*JUMBLR_TXFEE);
|
|
*targetvolSp = depositfactor * kmdprice * ((JUMBLR_INCR + 3*fee) + 3*JUMBLR_TXFEE);
|
|
return(depositfactor);
|
|
}
|
|
|
|
int32_t jumblr_DEXutxoind(int32_t *shouldsplitp,double targetvolB,double targetvolM,double targetvolS,double amount,double margin,double dexfeeratio,double esttxfee)
|
|
{
|
|
*shouldsplitp = 0;
|
|
if ( amount >= targetvolB )
|
|
{
|
|
if ( amount > margin * (targetvolB + targetvolS) )
|
|
*shouldsplitp = 1;
|
|
return(0);
|
|
}
|
|
else
|
|
{
|
|
if ( amount >= targetvolM )
|
|
{
|
|
if ( amount > margin * (targetvolM + targetvolS) )
|
|
*shouldsplitp = 1;
|
|
return(1);
|
|
}
|
|
else
|
|
{
|
|
if ( amount >= targetvolS )
|
|
{
|
|
if ( amount > margin * targetvolS )
|
|
*shouldsplitp = 1;
|
|
return(2);
|
|
}
|
|
else if ( amount >= targetvolB/dexfeeratio )
|
|
{
|
|
if ( amount > margin * targetvolB/dexfeeratio )
|
|
*shouldsplitp = 1;
|
|
return(3);
|
|
}
|
|
else if ( amount >= targetvolM/dexfeeratio )
|
|
{
|
|
if ( amount > margin * targetvolM/dexfeeratio )
|
|
*shouldsplitp = 1;
|
|
return(4);
|
|
}
|
|
else if ( amount >= targetvolS/dexfeeratio )
|
|
{
|
|
if ( amount > margin * targetvolS/dexfeeratio )
|
|
*shouldsplitp = 1;
|
|
return(5);
|
|
}
|
|
else if ( amount >= esttxfee )
|
|
{
|
|
if ( amount > esttxfee*4 )
|
|
*shouldsplitp = 1;
|
|
return(6);
|
|
}
|
|
else return(-1);
|
|
}
|
|
}
|
|
}
|
|
|
|
int32_t jumblr_DEXutxoupdate(struct supernet_info *myinfo,struct iguana_info *coin,int32_t *shouldsplitp,bits256 *splittxidp,char *coinaddr,bits256 privkey,bits256 txid,int32_t vout,uint64_t value,int32_t isbob,double kmdprice,double estfee)
|
|
{
|
|
double fees[4],targetvolB,amount,targetvolM,targetvolS,depositfactor,dexfeeratio,margin; int32_t ind = -1,i; cJSON *privkeys; char wifstr[128];
|
|
*shouldsplitp = 0;
|
|
margin = 1.1;
|
|
depositfactor = (isbob == 0) ? 1. : 1.2;
|
|
dexfeeratio = 500.;
|
|
amount = dstr(value);
|
|
memset(splittxidp,0,sizeof(*splittxidp));
|
|
depositfactor = jumblr_DEXutxosize(&targetvolB,&targetvolM,&targetvolS,isbob,kmdprice);
|
|
printf("depositfactor %.8f targetvols %.8f %.8f %.8f\n",depositfactor,targetvolB,targetvolM,targetvolS);
|
|
fees[0] = (margin * targetvolB) / dexfeeratio;
|
|
fees[1] = (margin * targetvolM) / dexfeeratio;
|
|
fees[2] = (margin * targetvolS) / dexfeeratio;
|
|
fees[3] = (strcmp("BTC",coin->symbol) == 0) ? 50000 : 10000;
|
|
for (i=0; i<4; i++)
|
|
if ( fees[i] < 10000 )
|
|
fees[i] = 10000;
|
|
if ( (ind= jumblr_DEXutxoind(shouldsplitp,targetvolB,targetvolM,targetvolS,amount,margin,dexfeeratio,fees[3])) >= 0 )
|
|
{
|
|
printf("shouldsplit.%d ind.%d\n",*shouldsplitp,ind);
|
|
if ( *shouldsplitp != 0 )
|
|
{
|
|
privkeys = cJSON_CreateArray();
|
|
bitcoin_priv2wif(wifstr,privkey,coin->chain->wiftype);
|
|
jaddistr(privkeys,wifstr);
|
|
if ( jumblr_DEXsplit(myinfo,coin,splittxidp,coinaddr,txid,vout,value,margin * targetvolB,margin * targetvolM,margin * targetvolS,fees,privkeys,estfee) > 0 )
|
|
ind = -1;
|
|
else *shouldsplitp = 0;
|
|
free_json(privkeys);
|
|
}
|
|
} // else printf("negative ind\n");
|
|
return(ind);
|
|
}
|
|
|
|
/*struct DEXcoin_info
|
|
{
|
|
bits256 deposit_privkey,jumblr_privkey;
|
|
struct iguana_info *coin;
|
|
cJSON *utxos,*spentutxos,*bigutxos,*normalutxos,*smallutxos,*feeutxos,*otherutxos;
|
|
double btcprice,USD_average,DEXpending,maxbid,minask,avail,KMDavail;
|
|
uint32_t lasttime;
|
|
char CMCname[32],symbol[16],depositaddr[64],KMDdepositaddr[64],KMDjumblraddr[64],jumblraddr[64];
|
|
};*/
|
|
|
|
int32_t jumblr_utxotxidpending(struct supernet_info *myinfo,bits256 *splittxidp,int32_t *indp,struct iguana_info *coin,bits256 txid,int32_t vout)
|
|
{
|
|
int32_t i;
|
|
memset(splittxidp,0,sizeof(*splittxidp));
|
|
for (i=0; i<coin->DEXinfo.numpending; i++)
|
|
{
|
|
if ( coin->DEXinfo.pending[i].vout == vout && bits256_cmp(coin->DEXinfo.pending[i].txid,txid) == 0 )
|
|
{
|
|
*indp = coin->DEXinfo.pending[i].ind;
|
|
*splittxidp = coin->DEXinfo.pending[i].splittxid;
|
|
// printf("jumblr_utxotxidpending found txid in slot.%d\n",i);
|
|
return(i);
|
|
}
|
|
}
|
|
// printf("jumblr_utxotxidpending cant find txid\n");
|
|
return(-1);
|
|
}
|
|
|
|
void jumblr_utxotxidpendingadd(struct supernet_info *myinfo,char *dest,struct iguana_info *coin,bits256 txid,int32_t vout,uint64_t value,bits256 splittxid,int32_t ind,double price,double estfee,int32_t shouldsplit)
|
|
{
|
|
struct jumblr_pending pend; cJSON *vals,*retjson; bits256 hash; char *retstr;
|
|
memset(&pend,0,sizeof(pend));
|
|
pend.splittxid = splittxid;
|
|
pend.txid = txid;
|
|
pend.vout = vout;
|
|
pend.ind = ind;
|
|
if ( 0 && myinfo->IAMLP == 0 && shouldsplit == 0 && ind < 3 )
|
|
{
|
|
static uint32_t num;
|
|
if ( num == 0 && price > SMALLVAL )
|
|
{
|
|
num++;
|
|
vals = cJSON_CreateObject();
|
|
jaddstr(vals,"source",coin->symbol);
|
|
jaddstr(vals,"dest",dest);
|
|
jaddnum(vals,"amount",price * 100);//dstr(value) - estfee);
|
|
jaddnum(vals,"minprice",price);
|
|
jaddnum(vals,"usejumblr",1);
|
|
memset(hash.bytes,0,sizeof(hash));
|
|
if ( (retstr= InstantDEX_request(myinfo,coin,0,0,hash,vals,"")) != 0 )
|
|
{
|
|
if ( (retjson= cJSON_Parse(retstr)) != 0 )
|
|
{
|
|
printf("request.(%s) -> (%s)\n",jprint(vals,0),retstr);
|
|
free_json(retjson);
|
|
}
|
|
free(retstr);
|
|
}
|
|
free_json(vals);
|
|
}
|
|
}
|
|
coin->DEXinfo.pending = realloc(coin->DEXinfo.pending,sizeof(*coin->DEXinfo.pending) * (1 + coin->DEXinfo.numpending));
|
|
coin->DEXinfo.pending[coin->DEXinfo.numpending++] = pend;
|
|
}
|
|
|
|
void jumblr_utxoupdate(struct supernet_info *myinfo,char *dest,struct iguana_info *coin,double price,char *coinaddr,bits256 privkey,double estfee)
|
|
{
|
|
char *retstr; cJSON *array,*item; int32_t shouldsplit,i,n,vout,ind; bits256 txid,splittxid; uint64_t value;
|
|
if ( (retstr= jumblr_listunspent(myinfo,coin,coinaddr)) != 0 )
|
|
{
|
|
//printf("%s.(%s)\n",coin->symbol,retstr);
|
|
if ( (array= cJSON_Parse(retstr)) != 0 )
|
|
{
|
|
if ( (n= cJSON_GetArraySize(array)) > 0 )
|
|
{
|
|
for (i=0; i<n; i++)
|
|
{
|
|
if ( (item= jitem(array,i)) == 0 )
|
|
continue;
|
|
txid = jbits256(item,"txid");
|
|
vout = jint(item,"vout");
|
|
value = SATOSHIDEN * jdouble(item,"amount");
|
|
//printf("price %.8f %llx/v%d %.8f %d of %d\n",price,(long long)txid.txid,vout,dstr(value),i,n);
|
|
if ( jumblr_utxotxidpending(myinfo,&splittxid,&ind,coin,txid,vout) < 0 )
|
|
{
|
|
ind = jumblr_DEXutxoupdate(myinfo,coin,&shouldsplit,&splittxid,coinaddr,privkey,txid,vout,value,myinfo->IAMLP,price,estfee);
|
|
jumblr_utxotxidpendingadd(myinfo,dest,coin,txid,vout,value,splittxid,ind,price,estfee,shouldsplit);
|
|
}
|
|
else
|
|
{
|
|
// update status of utxo
|
|
}
|
|
}
|
|
}
|
|
free_json(array);
|
|
}
|
|
free(retstr);
|
|
}
|
|
}
|
|
|
|
void jumblr_DEXupdate(struct supernet_info *myinfo,struct iguana_info *coin,char *symbol,char *CMCname,double BTC2KMD,double KMDavail)
|
|
{
|
|
double avebid,aveask,highbid,lowask,CMC_average,changes[3],estfee,estbtcfee; struct iguana_info *kmdcoin; struct DEXcoin_info *ptr = &coin->DEXinfo;
|
|
// wait for one confirmation to clear most in mempool (ha, ha)
|
|
// deal with changing addresses, ie all pendings?
|
|
estfee = 0.0001;
|
|
estbtcfee = 0.0015;
|
|
if ( coin != 0 && (kmdcoin= iguana_coinfind("KMD")) != 0 && time(NULL) > ptr->lasttime+60 )
|
|
{
|
|
ptr->coin = coin;
|
|
if ( strcmp(symbol,ptr->symbol) != 0 )
|
|
{
|
|
safecopy(ptr->symbol,symbol,sizeof(ptr->symbol));
|
|
safecopy(ptr->CMCname,CMCname,sizeof(ptr->CMCname));
|
|
}
|
|
//if ( ptr->depositaddr[0] == 0 )
|
|
{
|
|
if ( strcmp("KMD",symbol) == 0 )
|
|
ptr->deposit_privkey = jumblr_privkey(myinfo,ptr->depositaddr,0,ptr->KMDdepositaddr,JUMBLR_DEPOSITPREFIX);
|
|
else ptr->deposit_privkey = jumblr_privkey(myinfo,ptr->depositaddr,ptr->coin->chain->pubtype,ptr->KMDdepositaddr,JUMBLR_DEPOSITPREFIX);
|
|
}
|
|
//if ( ptr->jumblraddr[0] == 0 )
|
|
{
|
|
if ( strcmp("KMD",symbol) == 0 )
|
|
ptr->jumblr_privkey = jumblr_privkey(myinfo,ptr->jumblraddr,0,ptr->KMDjumblraddr,"");
|
|
else ptr->jumblr_privkey = jumblr_privkey(myinfo,ptr->jumblraddr,ptr->coin->chain->pubtype,ptr->KMDjumblraddr,"");
|
|
}
|
|
ptr->avail = dstr(jumblr_balance(myinfo,ptr->coin,ptr->depositaddr));
|
|
ptr->btcprice = get_theoretical(&avebid,&aveask,&highbid,&lowask,&CMC_average,changes,CMCname,symbol,"BTC",&ptr->USD_average);
|
|
KMDavail = dstr(jumblr_balance(myinfo,ptr->coin,ptr->KMDjumblraddr));
|
|
if ( strcmp("KMD",symbol) == 0 )
|
|
{
|
|
ptr->BTC2KMD = ptr->btcprice;
|
|
ptr->kmdprice = 1.;
|
|
ptr->KMDavail = KMDavail;
|
|
/*if ( (btccoin= iguana_coinfind("BTC")) != 0 )
|
|
jumblr_utxoupdate(myinfo,"KMD",btccoin,ptr->btcprice,ptr->depositaddr,ptr->deposit_privkey,estbtcfee);
|
|
jumblr_utxoupdate(myinfo,"BTC",kmdcoin,1.,ptr->KMDdepositaddr,ptr->deposit_privkey,estfee);*/
|
|
}
|
|
else if ( (ptr->BTC2KMD= BTC2KMD) > SMALLVAL )
|
|
{
|
|
ptr->kmdprice = ptr->btcprice / BTC2KMD;
|
|
ptr->KMDavail = KMDavail;
|
|
//jumblr_utxoupdate(myinfo,"KMD",ptr->coin,ptr->kmdprice,ptr->depositaddr,ptr->deposit_privkey,estfee);
|
|
}
|
|
ptr->lasttime = (uint32_t)time(NULL);
|
|
printf("%s avail %.8f KMDavail %.8f btcprice %.8f deposit.(%s %s) -> jumblr.(%s %s)\n",symbol,ptr->avail,KMDavail,ptr->btcprice,ptr->depositaddr,ptr->KMDdepositaddr,ptr->jumblraddr,ptr->KMDjumblraddr);
|
|
} // else printf("skip\n");
|
|
}
|
|
|
|
void jumblr_CMCname(char *CMCname,char *symbol)
|
|
{
|
|
if ( strcmp(symbol,"KMD") == 0 )
|
|
strcpy(CMCname,"komodo");
|
|
}
|
|
|
|
void jumblr_DEXcheck(struct supernet_info *myinfo,struct iguana_info *coin,int32_t toKMD)
|
|
{
|
|
double vol,avail; struct iguana_info *kmdcoin,*coinbtc = 0;
|
|
kmdcoin = iguana_coinfind("KMD");
|
|
coinbtc = iguana_coinfind("BTC");
|
|
//printf("jumblr_DEXcheck numswaps.%d notary.%d IAMLP.%d %p %p %f\n",myinfo->numswaps,myinfo->IAMNOTARY,myinfo->IAMLP,kmdcoin,coinbtc,kmdcoin->DEXinfo.btcprice);
|
|
if ( myinfo->IAMNOTARY != 0 || myinfo->IAMLP != 0 )
|
|
return;
|
|
if ( kmdcoin == 0 || coinbtc == 0 )
|
|
return;
|
|
jumblr_DEXupdate(myinfo,kmdcoin,"KMD","komodo",0.,0.);
|
|
if ( strcmp(coin->symbol,"KMD") != 0 && kmdcoin->DEXinfo.btcprice > 0. )
|
|
{
|
|
if ( coin->CMCname[0] == 0 )
|
|
jumblr_CMCname(coin->CMCname,coin->symbol);
|
|
if ( coin->CMCname[0] != 0 )
|
|
jumblr_DEXupdate(myinfo,coin,coin->symbol,coin->CMCname,kmdcoin->DEXinfo.btcprice,kmdcoin->DEXinfo.avail);
|
|
}
|
|
if ( myinfo->numswaps == 0 && myinfo->IAMLP == 0 && kmdcoin->DEXinfo.btcprice > SMALLVAL )
|
|
{
|
|
double minbtc,minkmd,btcavail; char *retstr; cJSON *vals; bits256 hash;
|
|
minbtc = (kmdcoin->DEXinfo.btcprice * 1.2) * (JUMBLR_INCR + 3*(JUMBLR_INCR * JUMBLR_FEE + JUMBLR_TXFEE));
|
|
btcavail = dstr(jumblr_balance(myinfo,coinbtc,kmdcoin->DEXinfo.depositaddr));
|
|
if ( toKMD == 0 && coinbtc != 0 && btcavail > (minbtc + kmdcoin->DEXinfo.DEXpending) )
|
|
{
|
|
avail = (btcavail - kmdcoin->DEXinfo.DEXpending);
|
|
printf("BTC deposits %.8f, min %.8f avail %.8f\n",btcavail,minbtc,avail);
|
|
/*if ( avail >= (100. * minbtc) )
|
|
vol = (100. * minbtc);
|
|
else if ( avail >= (10. * minbtc) )
|
|
vol = (10. * minbtc);
|
|
else*/
|
|
if ( avail >= minbtc )
|
|
vol = minbtc;
|
|
else vol = 0.;
|
|
if ( vol > 0. )
|
|
{
|
|
vals = cJSON_CreateObject();
|
|
jaddstr(vals,"source","BTC");
|
|
jaddstr(vals,"dest","KMD");
|
|
jaddnum(vals,"amount",vol);
|
|
jaddnum(vals,"minprice",kmdcoin->DEXinfo.btcprice);
|
|
jaddnum(vals,"usejumblr",1);
|
|
jaddnum(vals,"DEXselector",1);
|
|
memset(hash.bytes,0,sizeof(hash));
|
|
kmdcoin->DEXinfo.DEXpending += vol;
|
|
if ( (retstr= InstantDEX_request(myinfo,coinbtc,0,0,hash,vals,"")) != 0 )
|
|
{
|
|
printf("request.(%s) -> (%s)\n",jprint(vals,0),retstr);
|
|
free(retstr);
|
|
}
|
|
free_json(vals);
|
|
// curl --url "http://127.0.0.1:7778" --data "{\"agent\":\"InstantDEX\",\"method\":\"request\",\"vals\":{\"source\":\"KMD\",\"amount\":20,\"dest\":\"USD\",\"minprice\":0.08}}"
|
|
}
|
|
} //else printf("btcavail %.8f pending %.8f\n",btcavail,pending);
|
|
minkmd = 100.;
|
|
if ( toKMD != 0 && coinbtc != 0 && kmdcoin->DEXinfo.KMDavail > (minkmd + kmdcoin->DEXinfo.KMDpending) )
|
|
{
|
|
avail = (kmdcoin->DEXinfo.KMDavail - kmdcoin->DEXinfo.KMDpending);
|
|
printf("KMD deposits %.8f, min %.8f, avail %.8f\n",kmdcoin->DEXinfo.KMDavail,minkmd,avail);
|
|
/*if ( avail > 100.*JUMBLR_INCR )
|
|
vol = 100.*JUMBLR_INCR;
|
|
else if ( avail > 10.*JUMBLR_INCR )
|
|
vol = 10.*JUMBLR_INCR;
|
|
else*/ if ( avail >= JUMBLR_INCR )
|
|
vol = JUMBLR_INCR;
|
|
else vol = 0.;
|
|
if ( vol > 0. )
|
|
{
|
|
vals = cJSON_CreateObject();
|
|
jaddstr(vals,"source","KMD");
|
|
jaddstr(vals,"dest","BTC");
|
|
jaddnum(vals,"amount",vol);
|
|
//jaddnum(vals,"destamount",JUMBLR_INCR*kmdcoin->DEXinfo.btcprice);
|
|
jaddnum(vals,"minprice",kmdcoin->DEXinfo.btcprice);
|
|
jaddnum(vals,"usejumblr",2);
|
|
memset(hash.bytes,0,sizeof(hash));
|
|
kmdcoin->DEXinfo.KMDpending += vol;
|
|
jaddnum(vals,"DEXselector",2);
|
|
if ( (retstr= InstantDEX_request(myinfo,coinbtc,0,0,hash,vals,"")) != 0 )
|
|
{
|
|
printf("request.(%s) -> (%s)\n",jprint(vals,0),retstr);
|
|
free(retstr);
|
|
}
|
|
free_json(vals);
|
|
}
|
|
} else printf("kmdavail %.8f pending %.8f\n",kmdcoin->DEXinfo.KMDavail,kmdcoin->DEXinfo.KMDpending);
|
|
} else printf("notlp.%d kmdprice %.8f\n",myinfo->IAMLP,kmdcoin->DEXinfo.btcprice);
|
|
}
|
|
|
|
void jumblr_iteration(struct supernet_info *myinfo,struct iguana_info *coin,int32_t selector,int32_t modval)
|
|
{
|
|
//static uint32_t lasttime;
|
|
char BTCaddr[64],KMDaddr[64],*zaddr,*retstr; bits256 privkey; uint64_t amount=0,total=0; double fee; struct jumblr_item *ptr,*tmp; uint8_t r;
|
|
if ( myinfo->IAMNOTARY != 0 )
|
|
return;
|
|
fee = JUMBLR_INCR * JUMBLR_FEE;
|
|
OS_randombytes(&r,sizeof(r));
|
|
//r = 0;
|
|
if ( strcmp(coin->symbol,"KMD") == 0 && coin->FULLNODE < 0 )
|
|
{
|
|
//printf("JUMBLR selector.%d modval.%d r.%d\n",selector,modval,r&7);
|
|
switch ( selector )
|
|
{
|
|
case 0: // public -> z, need to importprivkey
|
|
jumblr_privkey(myinfo,BTCaddr,0,KMDaddr,JUMBLR_DEPOSITPREFIX);
|
|
if ( (total= jumblr_balance(myinfo,coin,KMDaddr)) >= (JUMBLR_INCR + 3*(fee+JUMBLR_TXFEE))*SATOSHIDEN )
|
|
{
|
|
if ( (r & 1) == 0 )
|
|
{
|
|
if ( (zaddr= jumblr_zgetnewaddress(myinfo,coin)) != 0 )
|
|
{
|
|
if ( total >= SATOSHIDEN * ((JUMBLR_INCR + 3*fee)*100 + 3*JUMBLR_TXFEE) )
|
|
amount = SATOSHIDEN * ((JUMBLR_INCR + 3*fee)*100 + 3*JUMBLR_TXFEE);
|
|
else if ( total >= SATOSHIDEN * ((JUMBLR_INCR + 3*fee)*10 + 3*JUMBLR_TXFEE) )
|
|
amount = SATOSHIDEN * ((JUMBLR_INCR + 3*fee)*10 + 3*JUMBLR_TXFEE);
|
|
else amount = SATOSHIDEN * ((JUMBLR_INCR + 3*fee) + 3*JUMBLR_TXFEE);
|
|
if ( (retstr= jumblr_sendt_to_z(myinfo,coin,KMDaddr,zaddr,dstr(amount))) != 0 )
|
|
{
|
|
printf("sendt_to_z.(%s)\n",retstr);
|
|
free(retstr);
|
|
}
|
|
free(zaddr);
|
|
} else printf("no zaddr from jumblr_zgetnewaddress\n");
|
|
}
|
|
} else printf("%s total %.8f vs %.8f\n",KMDaddr,dstr(total),(JUMBLR_INCR + 3*(fee+JUMBLR_TXFEE)));
|
|
break;
|
|
case 1: // z -> z
|
|
jumblr_opidsupdate(myinfo,coin);
|
|
HASH_ITER(hh,myinfo->jumblrs,ptr,tmp)
|
|
{
|
|
if ( jumblr_addresstype(myinfo,coin,ptr->src) == 't' && jumblr_addresstype(myinfo,coin,ptr->dest) == 'z' )
|
|
{
|
|
if ( (r & 1) == 0 && ptr->spent == 0 && (total= jumblr_balance(myinfo,coin,ptr->dest)) >= (fee + JUMBLR_FEE)*SATOSHIDEN )
|
|
{
|
|
if ( (zaddr= jumblr_zgetnewaddress(myinfo,coin)) != 0 )
|
|
{
|
|
if ( (retstr= jumblr_sendz_to_z(myinfo,coin,ptr->dest,zaddr,dstr(total))) != 0 )
|
|
{
|
|
printf("sendz_to_z.(%s)\n",retstr);
|
|
free(retstr);
|
|
}
|
|
ptr->spent = (uint32_t)time(NULL);
|
|
free(zaddr);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
break;
|
|
case 2: // z -> public
|
|
jumblr_opidsupdate(myinfo,coin);
|
|
HASH_ITER(hh,myinfo->jumblrs,ptr,tmp)
|
|
{
|
|
if ( jumblr_addresstype(myinfo,coin,ptr->src) == 'z' && jumblr_addresstype(myinfo,coin,ptr->dest) == 'z' )
|
|
{
|
|
if ( (r & 1) == 0 && ptr->spent == 0 && (total= jumblr_balance(myinfo,coin,ptr->dest)) >= (fee + JUMBLR_FEE)*SATOSHIDEN )
|
|
{
|
|
privkey = jumblr_privkey(myinfo,BTCaddr,0,KMDaddr,"");
|
|
if ( (retstr= jumblr_sendz_to_t(myinfo,coin,ptr->dest,KMDaddr,dstr(total))) != 0 )
|
|
{
|
|
printf("sendz_to_t.(%s)\n",retstr);
|
|
free(retstr);
|
|
}
|
|
ptr->spent = (uint32_t)time(NULL);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
|