diff --git a/iguana/exchanges/LP_RTmetrics.c b/iguana/exchanges/LP_RTmetrics.c index 3876a61cc..f0b813a9f 100644 --- a/iguana/exchanges/LP_RTmetrics.c +++ b/iguana/exchanges/LP_RTmetrics.c @@ -18,9 +18,139 @@ // marketmaker // -cJSON *LP_RTmetrics_sort(cJSON *rawasks,int32_t numasks,double maxprice,double relvolume) +struct LP_metricinfo { - cJSON *array,*item,*statsjson,*swaps=0; int32_t i,numswaps=0; bits256 zero; uint32_t futuretime; double price; char *retstr; + bits256 pubkey; + double metric,price,balance,minvol,maxvol; + int32_t ind,numutxos,age,pendingswaps; +}; + +struct LP_RTmetrics_pendings +{ + char refbase[16],refrel[16]; + int32_t numswaps,numavoidtxids,numwhitelist,numblacklist,numpendings,pending_swaps[1024]; + bits256 avoidtxids[8192],whitelist[1024],blacklist[1024],pending_pubkeys[1024]; +} LP_RTmetrics; + +int32_t LP_bits256_find(bits256 *list,int32_t num,bits256 val) +{ + int32_t i; + if ( bits256_nonz(val) != 0 ) + { + for (i=0; i= 0 ) + LP_RTmetrics.pending_swaps[ind]++; + return(ind); +} + +int32_t LP_RTmetrics_pendingswaps(bits256 pubkey) +{ + int32_t ind; + if ( (ind= LP_bits256_find(LP_RTmetrics.pending_pubkeys,LP_RTmetrics.numpendings,pubkey)) >= 0 ) + return(LP_RTmetrics.pending_swaps[ind]); + else return(0); +} + +int32_t LP_RTmetrics_avoidtxid(bits256 txid) +{ + return(LP_bits256_find(LP_RTmetrics.avoidtxids,LP_RTmetrics.numavoidtxids,txid)); +} + +int32_t LP_RTmetrics_whitelisted(bits256 pubkey) +{ + return(LP_bits256_find(LP_RTmetrics.whitelist,LP_RTmetrics.numwhitelist,pubkey)); +} + +int32_t LP_RTmetrics_blacklisted(bits256 pubkey) +{ + return(LP_bits256_find(LP_RTmetrics.blacklist,LP_RTmetrics.numblacklist,pubkey)); +} + +void LP_RTmetrics_swapsinfo(char *refbase,char *refrel,cJSON *swaps,int32_t numswaps) +{ + int32_t i; char *base,*rel,*retstr; cJSON *item,*swapjson; bits256 srcpub,destpub; uint64_t aliceid,basesatoshis,relsatoshis; uint32_t requestid,quoteid; double price; + for (i=0; iistrusted > 0 ) + LP_RTmetrics_whitelistadd(pubp->pubkey); + else if ( pubp->istrusted < 0 ) + LP_RTmetrics_blacklistadd(pubp->pubkey); + } futuretime = (uint32_t)time(NULL) + 3600*100; memset(zero.bytes,0,sizeof(zero)); if ( (retstr= LP_statslog_disp(100,futuretime,futuretime,"",zero)) != 0 ) @@ -29,37 +159,110 @@ cJSON *LP_RTmetrics_sort(cJSON *rawasks,int32_t numasks,double maxprice,double r { if ( (swaps= jarray(&numswaps,statsjson,"swaps")) != 0 ) { + printf("LP_RTmetrics_update for (%s)\n",jprint(swaps,0)); if ( numswaps > 0 ) - swaps = jduplicate(swaps); - else swaps = 0; + LP_RTmetrics_swapsinfo(base,rel,swaps,numswaps); } free_json(statsjson); } free(retstr); } - //if ( numswaps == 0 || swaps == 0 ) - return(0); - printf("calc RTmetrics for (%s)\n",jprint(swaps,0)); - /*jadd64bits(item,"aliceid",sp->aliceid); - jaddbits256(item,"src",sp->Q.srchash); - jaddstr(item,"base",sp->Q.srccoin); - jaddnum(item,"basevol",dstr(sp->Q.satoshis)); - jaddbits256(item,"dest",sp->Q.desthash); - jaddstr(item,"rel",sp->Q.destcoin); - jaddnum(item,"relvol",dstr(sp->Q.destsatoshis)); - jaddnum(item,"price",sp->qprice); - jaddnum(item,"requestid",sp->Q.R.requestid); - jaddnum(item,"quoteid",sp->Q.R.quoteid); - */ - array = cJSON_CreateArray(); - for (i=0; i LP_MAXPENDING_SWAPS ) + { + char str[65]; printf("%s has %d pending swaps! which is more than %d\n",bits256_str(str,LP_RTmetrics.pending_pubkeys[i]),LP_RTmetrics.pending_swaps[i],LP_MAXPENDING_SWAPS); + LP_RTmetrics_blacklistadd(LP_RTmetrics.pending_pubkeys[i]); + } + printf("%d pubkeys have pending swaps, whitelist.%d blacklist.%d avoidtxids.%d\n",LP_RTmetrics.numpendings,LP_RTmetrics.numwhitelist,LP_RTmetrics.numblacklist,LP_RTmetrics.numavoidtxids); +} + +double _LP_RTmetric_calc(struct LP_metricinfo *mp,double bestprice,double maxprice,double relvolume) +{ + int32_t n; double metric,origmetric = (bestprice / mp->price); + metric = origmetric; + if ( mp->numutxos == 0 || relvolume == 0. || mp->maxvol == 0. || mp->balance == 0. ) + return(metric * 100.); + if ( relvolume < mp->minvol ) + { + metric *= (mp->minvol / relvolume); + } + else if ( relvolume > mp->maxvol ) + { + metric *= (relvolume / mp->maxvol); + } + if ( relvolume < mp->balance/LP_MINVOL ) + { + metric *= (mp->balance / relvolume); + } + else if ( relvolume > mp->balance/mp->numutxos ) + { + metric *= (relvolume / (mp->balance/mp->numutxos)); + } + if ( mp->age > LP_ORDERBOOK_DURATION*0.8 ) + metric *= 2; + else if ( mp->age > 60 ) + metric *= 1.03; + if ( (n= mp->pendingswaps) > 0 ) + while ( n-- > 0 ) + metric *= 1.1; + if ( metric != origmetric ) + printf("price %.8f orig %.8f -> %.8f relvol %.8f min %.8f max %.8f bal %.8f age.%d pend.%d\n",mp->price,origmetric,metric,relvolume,mp->minvol,mp->maxvol,mp->balance,mp->age,mp->pendingswaps); + return(metric); +} + +void LP_RTmetric_calc(struct LP_metricinfo *sortbuf,int32_t ind,cJSON *item,double bestprice,double maxprice,double relvolume,double prevdepth) +{ + sortbuf[ind].pubkey = jbits256(item,"pubkey"); + sortbuf[ind].price = jdouble(item,"price"); + sortbuf[ind].maxvol = jdouble(item,"maxvolume"); + sortbuf[ind].minvol = jdouble(item,"minvolume"); + sortbuf[ind].balance = jdouble(item,"depth") - prevdepth; + sortbuf[ind].numutxos = juint(item,"numutxos"); + sortbuf[ind].age = juint(item,"age"); + sortbuf[ind].ind = ind; + sortbuf[ind].pendingswaps = LP_RTmetrics_pendingswaps(sortbuf[ind].pubkey); + sortbuf[ind].metric = _LP_RTmetric_calc(&sortbuf[ind],bestprice,maxprice,relvolume); +} + +cJSON *LP_RTmetrics_sort(char *base,char *rel,cJSON *rawasks,int32_t numasks,double maxprice,double relvolume) +{ + cJSON *array,*item; int32_t i,num,groupi; double price,prevdepth,bestprice; struct LP_metricinfo *sortbuf; + groupi = -1; + bestprice = 0.; + for (num=i=0; i maxprice ) break; - jaddi(array,jduplicate(item)); + if ( i == 0 ) + bestprice = price; + else if ( price < bestprice*LP_RTMETRICS_TOPGROUP ) + groupi = i; + num++; } - free_json(swaps); + if ( groupi > 0 ) + { + sortbuf = calloc(groupi+1,sizeof(*sortbuf)); + prevdepth = 0.; + for (i=0; i<=groupi; i++) + { + item = jitem(rawasks,i); + LP_RTmetric_calc(sortbuf,i,item,bestprice,maxprice,relvolume,prevdepth); + prevdepth = jdouble(item,"depth"); + } + revsortds(&sortbuf[0].metric,groupi+1,sizeof(*sortbuf)); + array = cJSON_CreateArray(); + for (i=0; i<=groupi; i++) + { + printf("(%d -> %d) ",i,sortbuf[i].ind); + item = jitem(rawasks,sortbuf[i].ind); + jaddi(array,jduplicate(item)); + } + free(sortbuf); + for (; i pubkeys, reputations, bonds etc. -// // verify portfolio, interest to KMD withdraw, pricebroadcast loop, trade to pubkey // dPoW security -> 4: KMD notarized, 5: BTC notarized, after next notary elections // bigendian architectures need to use little endian for sighash calcs @@ -96,6 +91,7 @@ char *blocktrail_listtransactions(char *symbol,char *coinaddr,int32_t num,int32_ #include "LP_bitcoin.c" #include "LP_coins.c" #include "LP_rpc.c" +#include "LP_RTmetrics.c" #include "LP_utxo.c" #include "LP_prices.c" #include "LP_scan.c" @@ -107,7 +103,6 @@ char *blocktrail_listtransactions(char *symbol,char *coinaddr,int32_t num,int32_ #include "LP_utxos.c" #include "LP_forwarding.c" #include "LP_signatures.c" -#include "LP_RTmetrics.c" #include "LP_ordermatch.c" #include "LP_portfolio.c" #include "LP_messages.c" @@ -732,7 +727,7 @@ void LP_pubkeysloop(void *ctx) LP_counter += 100; //printf("LP_pubkeysloop %d\n",LP_counter); LP_notify_pubkeys(ctx,LP_mypubsock); - sleep(60); + sleep(LP_ORDERBOOK_DURATION * .777); } } @@ -771,7 +766,7 @@ void LP_privkeysloop(void *ctx) LP_counter += 1000; //printf("LP_privkeysloop %u\n",LP_counter); LP_privkey_updates(ctx,LP_mypubsock,0); - sleep(60); + sleep(LP_ORDERBOOK_DURATION * .777); } } diff --git a/iguana/exchanges/LP_ordermatch.c b/iguana/exchanges/LP_ordermatch.c index 831bbe44d..31558980d 100644 --- a/iguana/exchanges/LP_ordermatch.c +++ b/iguana/exchanges/LP_ordermatch.c @@ -656,6 +656,12 @@ int32_t LP_tradecommand(void *ctx,char *myipaddr,int32_t pubsock,cJSON *argjson, printf("request.(%s)\n",jprint(argjson,0)); if ( 1 )//LP_allocated(butxo->payment.txid,butxo->payment.vout) != 0 || LP_allocated(butxo->deposit.txid,butxo->deposit.vout) != 0 || (qprice= LP_quote_validate(autxo,butxo,&Q,1)) <= SMALLVAL ) { + LP_RTmetrics_update(Q.srccoin,Q.destcoin); + if ( LP_RTmetrics_blacklisted(Q.desthash) >= 0 ) + { + printf("request from blacklisted %s, ignore\n",bits256_str(str,Q.desthash)); + return(retval); + } printf("butxo.%p replace path %p %s, %p %s, %.8f\n",butxo,LP_allocated(butxo->payment.txid,butxo->payment.vout),bits256_str(str,butxo->payment.txid),LP_allocated(butxo->deposit.txid,butxo->deposit.vout),bits256_str(str2,butxo->deposit.txid),LP_quote_validate(autxo,butxo,&Q,1)); LP_listunspent_both(Q.srccoin,Q.coinaddr,0); if ( (butxo= LP_address_utxopair(1,utxos,max,LP_coinfind(Q.srccoin),Q.coinaddr,Q.txfee,dstr(Q.destsatoshis),price,Q.desttxfee)) != 0 ) @@ -787,7 +793,7 @@ struct LP_utxoinfo *LP_buyutxo(double *ordermatchpricep,int64_t *bestsatoshisp,i { if ( (rawasks= jarray(&numasks,orderbook,"asks")) != 0 ) { - if ( (asks= LP_RTmetrics_sort(rawasks,numasks,maxprice,dstr(autxo->S.satoshis))) == 0 ) + if ( (asks= LP_RTmetrics_sort(base,autxo->coin,rawasks,numasks,maxprice,dstr(autxo->S.satoshis))) == 0 ) asks = rawasks; for (i=0; i= 0 ) + continue; //printf("[%d/%d] %s pubcmp %d price %.8f vs maxprice %.8f\n",i,numasks,jprint(item,0),bits256_cmp(pubkey,G.LP_mypub25519),price,maxprice); if ( LP_pricevalid(price) > 0 && price <= maxprice ) { @@ -904,6 +912,7 @@ char *LP_autobuy(void *ctx,char *myipaddr,int32_t mypubsock,char *base,char *rel printf("destsatoshis %.8f vs utxo %.8f this would have triggered an quote error -13\n",dstr(destsatoshis),dstr(autxo->payment.value)); return(clonestr("{\"error\":\"cant find alice utxo that is small enough\"}")); } + LP_RTmetrics_update(base,rel); while ( 1 ) { if ( (bestutxo= LP_buyutxo(&ordermatchprice,&bestsatoshis,&bestdestsatoshis,autxo,base,maxprice,duration,txfee,desttxfee,gui,pubkeys,numpubs,destpubkey)) == 0 || ordermatchprice == 0. || bestdestsatoshis == 0 ) diff --git a/iguana/exchanges/LP_statemachine.c b/iguana/exchanges/LP_statemachine.c index bc05ecb71..c3cd25927 100644 --- a/iguana/exchanges/LP_statemachine.c +++ b/iguana/exchanges/LP_statemachine.c @@ -2349,6 +2349,20 @@ struct LP_utxoinfo *LP_bestutxo(double *ordermatchpricep,int64_t *bestsatoshisp, LP_mypriceset(&changed,autxo->coin,base,1. / *ordermatchpricep); return(bestutxo); } + +/*static int _LP_metric_eval(const void *a,const void *b) + { + #define aptr (*(struct LP_metricinfo **)a) + #define bptr (*(struct LP_metricinfo **)b) + if ( bptr->metric > aptr->metric ) + return(1); + else if ( bptr->metric < aptr->metric ) + return(-1); + return(0); + #undef aptr + #undef bptr + }*/ + /*portable_mutex_lock(&ep->pendingQ.mutex); if ( ep->pendingQ.list != 0 ) { diff --git a/iguana/exchanges/LP_stats.c b/iguana/exchanges/LP_stats.c index e31cdd6db..1fcea0355 100644 --- a/iguana/exchanges/LP_stats.c +++ b/iguana/exchanges/LP_stats.c @@ -191,7 +191,10 @@ int32_t LP_swapstats_update(struct LP_swapstats *sp,struct LP_quoteinfo *qp,cJSO sp->Apaymentspent = LP_swapstats_txid(lineobj,"Apaymentspent",sp->Apaymentspent); sp->depositspent = LP_swapstats_txid(lineobj,"depositspent",sp->depositspent); if ( (statusstr= jstr(lineobj,"status")) != 0 && strcmp(statusstr,"finished") == 0 ) - sp->finished = juint(lineobj,"timestamp"); + { + if ( (sp->finished= juint(lineobj,"timestamp")) == 0 ) + sp->finished = (uint32_t)time(NULL); + } if ( sp->finished == 0 && time(NULL) > sp->Q.timestamp+INSTANTDEX_LOCKTIME*2 ) sp->expired = (uint32_t)time(NULL); return(0); @@ -231,7 +234,6 @@ int32_t LP_statslog_parsequote(char *method,cJSON *lineobj) quoteid = juint(lineobj,"quoteid"); if ( (sp= LP_swapstats_find(aliceid)) != 0 ) { - flag = 1; sp->methodind = methodind; if ( LP_swapstats_update(sp,&Q,lineobj) == 0 ) flag = 1; diff --git a/iguana/exchanges/LP_utxo.c b/iguana/exchanges/LP_utxo.c index b031eeec2..cb3d90a6e 100644 --- a/iguana/exchanges/LP_utxo.c +++ b/iguana/exchanges/LP_utxo.c @@ -149,7 +149,7 @@ int32_t LP_address_utxo_ptrs(struct iguana_info *coin,int32_t iambob,struct LP_a DL_FOREACH_SAFE(ap->utxos,up,tmp) { //char str[65]; printf("LP_address_utxo_ptrs %s n.%d %.8f %s v%d spendheight.%d allocated.%p\n",ap->coinaddr,n,dstr(up->U.value),bits256_str(str,up->U.txid),up->U.vout,up->spendheight,LP_allocated(up->U.txid,up->U.vout)); - if ( up->spendheight <= 0 ) + if ( up->spendheight <= 0 && LP_RTmetrics_avoidtxid(up->U.txid) < 0 ) { if ( coin->electrum == 0 ) {