/****************************************************************************** * 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. * * * ******************************************************************************/ // // LP_prices.c // marketmaker // struct LP_orderbookentry { bits256 txid,txid2,pubkey; double price; uint64_t basesatoshis; int32_t vout,vout2,age; }; #define LP_MAXPRICEINFOS 256 struct LP_priceinfo { char symbol[16]; uint64_t coinbits; int32_t ind,pad; double diagval,high[2],low[2],last[2],bid[2],ask[2]; //volume,btcvolume,prevday; // mostly bittrex info double relvals[LP_MAXPRICEINFOS]; double myprices[LP_MAXPRICEINFOS]; double minprices[LP_MAXPRICEINFOS]; double margins[LP_MAXPRICEINFOS]; } LP_priceinfos[LP_MAXPRICEINFOS]; int32_t LP_numpriceinfos; struct LP_cacheinfo { UT_hash_handle hh; struct LP_quoteinfo Q; uint8_t key[sizeof(bits256)+sizeof(uint64_t)*2+sizeof(int32_t)]; double price; uint32_t timestamp; } *LP_cacheinfos; struct LP_pubkeyinfo { UT_hash_handle hh; bits256 pubkey; double matrix[LP_MAXPRICEINFOS][LP_MAXPRICEINFOS]; uint32_t timestamp,istrusted,numerrors; } *LP_pubkeyinfos; struct LP_priceinfo *LP_priceinfofind(char *symbol) { int32_t i; struct LP_priceinfo *pp; uint64_t coinbits; if ( symbol == 0 || symbol[0] == 0 ) return(0); if ( LP_numpriceinfos > 0 ) { coinbits = stringbits(symbol); pp = LP_priceinfos; for (i=0; icoinbits == coinbits ) return(pp); } return(0); } struct LP_priceinfo *LP_priceinfoptr(int32_t *indp,char *base,char *rel) { struct LP_priceinfo *basepp,*relpp; if ( (basepp= LP_priceinfofind(base)) != 0 && (relpp= LP_priceinfofind(rel)) != 0 ) { *indp = relpp->ind; return(basepp); } else { *indp = -1; return(0); } } int32_t LP_cachekey(uint8_t *key,char *base,char *rel,bits256 txid,int32_t vout) { uint64_t basebits,relbits; int32_t offset = 0; basebits = stringbits(base); relbits = stringbits(rel); memcpy(&key[offset],&basebits,sizeof(basebits)), offset += sizeof(basebits); memcpy(&key[offset],&relbits,sizeof(relbits)), offset += sizeof(relbits); memcpy(&key[offset],&txid,sizeof(txid)), offset += sizeof(txid); memcpy(&key[offset],&vout,sizeof(vout)), offset += sizeof(vout); return(offset); } struct LP_cacheinfo *LP_cachefind(char *base,char *rel,bits256 txid,int32_t vout) { struct LP_cacheinfo *ptr=0; uint8_t key[sizeof(bits256)+sizeof(uint64_t)*2+sizeof(vout)]; if ( base == 0 || rel == 0 ) return(0); if ( LP_cachekey(key,base,rel,txid,vout) == sizeof(key) ) { portable_mutex_lock(&LP_cachemutex); HASH_FIND(hh,LP_cacheinfos,key,sizeof(key),ptr); portable_mutex_unlock(&LP_cachemutex); } else printf("LP_cachefind keysize mismatch?\n"); if ( 0 && ptr != 0 && ptr->timestamp != 0 && ptr->timestamp < time(NULL)-LP_CACHEDURATION ) { printf("expire price %.8f\n",ptr->price); ptr->price = 0.; ptr->timestamp = 0; memset(&ptr->Q,0,sizeof(ptr->Q)); } return(ptr); } struct LP_pubkeyinfo *LP_pubkeyfind(bits256 pubkey) { struct LP_pubkeyinfo *pubp=0; portable_mutex_lock(&LP_pubkeymutex); HASH_FIND(hh,LP_pubkeyinfos,&pubkey,sizeof(pubkey),pubp); portable_mutex_unlock(&LP_pubkeymutex); return(pubp); } struct LP_pubkeyinfo *LP_pubkeyadd(bits256 pubkey) { struct LP_pubkeyinfo *pubp=0; if ( (pubp= LP_pubkeyfind(pubkey)) == 0 ) { portable_mutex_lock(&LP_pubkeymutex); pubp = calloc(1,sizeof(*pubp)); pubp->pubkey = pubkey; HASH_ADD_KEYPTR(hh,LP_pubkeyinfos,&pubp->pubkey,sizeof(pubp->pubkey),pubp); portable_mutex_unlock(&LP_pubkeymutex); if ( (pubp= LP_pubkeyfind(pubkey)) == 0 ) printf("pubkeyadd find error after add\n"); } return(pubp); } int32_t LP_pubkey_istrusted(bits256 pubkey) { struct LP_pubkeyinfo *pubp; if ( (pubp= LP_pubkeyfind(pubkey)) != 0 ) return(pubp->istrusted != 0); return(0); } char *LP_pubkey_trustset(bits256 pubkey,uint32_t trustval) { struct LP_pubkeyinfo *pubp; if ( (pubp= LP_pubkeyfind(pubkey)) != 0 ) { pubp->istrusted = trustval; return(clonestr("{\"result\":\"success\"}")); } return(clonestr("{\"error\":\"pubkey not found\"}")); } cJSON *LP_pubkeyjson(struct LP_pubkeyinfo *pubp) { int32_t baseid,relid; char *base; double price; cJSON *item,*array,*obj; obj = cJSON_CreateObject(); array = cJSON_CreateArray(); for (baseid=0; baseidmatrix[baseid][relid]) > SMALLVAL ) { item = cJSON_CreateArray(); jaddistr(item,base); jaddistr(item,LP_priceinfos[relid].symbol); jaddinum(item,price); jaddi(array,item); } } } jaddbits256(obj,"pubkey",pubp->pubkey); jaddnum(obj,"timestamp",pubp->timestamp); jadd(obj,"asks",array); if ( pubp->istrusted != 0 ) jaddnum(obj,"istrusted",pubp->istrusted); return(obj); } char *LP_prices() { struct LP_pubkeyinfo *pubp,*tmp; cJSON *array = cJSON_CreateArray(); HASH_ITER(hh,LP_pubkeyinfos,pubp,tmp) { jaddi(array,LP_pubkeyjson(pubp)); } return(jprint(array,1)); } void LP_prices_parse(cJSON *obj) { struct LP_pubkeyinfo *pubp; struct LP_priceinfo *basepp; uint32_t timestamp; bits256 pubkey; cJSON *asks,*item; int32_t i,n,relid; char *base,*rel; double askprice; pubkey = jbits256(obj,"pubkey"); if ( bits256_nonz(pubkey) != 0 && (pubp= LP_pubkeyadd(pubkey)) != 0 ) { if ( (timestamp= juint(obj,"timestamp")) > pubp->timestamp && (asks= jarray(&n,obj,"asks")) != 0 ) { pubp->timestamp = timestamp; for (i=0; iind,relid,askprice); pubp->matrix[basepp->ind][relid] = askprice; } } } } } void LP_peer_pricesquery(char *destipaddr,uint16_t destport) { char *retstr; cJSON *array; int32_t i,n; if ( (retstr= issue_LP_getprices(destipaddr,destport)) != 0 ) { if ( (array= cJSON_Parse(retstr)) != 0 ) { if ( is_cJSON_Array(array) && (n= cJSON_GetArraySize(array)) > 0 ) { for (i=0; iQ; if ( ptr->price == 0. && ptr->Q.satoshis != 0 ) { printf("LP_pricecache: null ptr->price? "); ptr->price = (double)ptr->Q.destsatoshis / ptr->Q.satoshis; } //printf("found %s/%s %.8f\n",base,rel,ptr->price); return(ptr->price); } //char str[65]; printf("cachemiss %s/%s %s/v%d\n",base,rel,bits256_str(str,txid),vout); return(0.); } void LP_priceinfoupdate(char *base,char *rel,double price) { struct LP_priceinfo *basepp,*relpp; if ( (basepp= LP_priceinfofind(base)) != 0 && (relpp= LP_priceinfofind(rel)) != 0 ) { //dxblend(&basepp->relvals[relpp->ind],price,0.9); //dxblend(&relpp->relvals[basepp->ind],1. / price,0.9); basepp->relvals[relpp->ind] = price; relpp->relvals[basepp->ind] = 1. / price; } } double LP_myprice(double *bidp,double *askp,char *base,char *rel) { struct LP_priceinfo *basepp,*relpp; double val; *bidp = *askp = 0.; if ( (basepp= LP_priceinfofind(base)) != 0 && (relpp= LP_priceinfofind(rel)) != 0 ) { if ( (*askp= basepp->myprices[relpp->ind]) > SMALLVAL ) { if ( (val= relpp->myprices[basepp->ind]) > SMALLVAL ) { *bidp = 1. / val; return((*askp + *bidp) * 0.5); } else { *bidp = 0.; return(*askp); } } else { if ( (val= relpp->myprices[basepp->ind]) > SMALLVAL ) { *bidp = 1. / val; *askp = 0.; return(*bidp); } } } return(0.); } char *LP_myprices() { int32_t baseid,relid; double bid,ask; char *base,*rel; cJSON *item,*array; array = cJSON_CreateArray(); for (baseid=0; baseid SMALLVAL ) { item = cJSON_CreateObject(); jaddstr(item,"base",base); jaddstr(item,"rel",rel); jaddnum(item,"bid",bid); jaddnum(item,"ask",ask); jaddi(array,item); } } } return(jprint(array,1)); } int32_t LP_mypriceset(int32_t *changedp,char *base,char *rel,double price) { struct LP_priceinfo *basepp,*relpp; struct LP_pubkeyinfo *pubp; *changedp = 0; if ( base != 0 && rel != 0 && price > SMALLVAL && (basepp= LP_priceinfofind(base)) != 0 && (relpp= LP_priceinfofind(rel)) != 0 ) { if ( fabs(basepp->myprices[relpp->ind] - price) > SMALLVAL ) *changedp = 1; basepp->myprices[relpp->ind] = price; // ask //relpp->myprices[basepp->ind] = (1. / price); // bid if ( (pubp= LP_pubkeyadd(LP_mypub25519)) != 0 ) { pubp->matrix[basepp->ind][relpp->ind] = price; //pubp->matrix[relpp->ind][basepp->ind] = (1. / price); pubp->timestamp = (uint32_t)time(NULL); } return(0); } else return(-1); } double LP_price(char *base,char *rel) { struct LP_priceinfo *basepp; int32_t relind; double price = 0.; if ( (basepp= LP_priceinfoptr(&relind,base,rel)) != 0 ) { if ( (price= basepp->myprices[relind]) == 0. ) price = basepp->relvals[relind]; } return(price); } cJSON *LP_priceinfomatrix(int32_t usemyprices) { int32_t i,j,n,m; double total,sum,val; struct LP_priceinfo *pp; uint32_t now; struct LP_cacheinfo *ptr,*tmp; cJSON *vectorjson = cJSON_CreateObject(); now = (uint32_t)time(NULL); HASH_ITER(hh,LP_cacheinfos,ptr,tmp) { if ( ptr->timestamp < now-3600*2 || ptr->price == 0. ) continue; LP_priceinfoupdate(ptr->Q.srccoin,ptr->Q.destcoin,ptr->price); } pp = LP_priceinfos; total = m = 0; for (i=0; idiagval = sum = n = 0; for (j=0; jmyprices[j]) == 0. ) val = pp->relvals[j]; if ( val > SMALLVAL ) { sum += val; n++; } } if ( n > 0 ) { pp->diagval = sum / n; total += pp->diagval, m++; } } if ( m > 0 ) { pp = LP_priceinfos; for (i=0; idiagval > SMALLVAL ) { pp->diagval /= total; jaddnum(vectorjson,pp->symbol,pp->diagval); } } } return(vectorjson); } struct LP_priceinfo *LP_priceinfoadd(char *symbol) { struct LP_priceinfo *pp; cJSON *retjson; if ( symbol == 0 ) return(0); if ( LP_numpriceinfos >= sizeof(LP_priceinfos)/sizeof(*LP_priceinfos) ) { printf("cant add any more priceinfos\n"); return(0); } pp = &LP_priceinfos[LP_numpriceinfos]; memset(pp,0,sizeof(*pp)); safecopy(pp->symbol,symbol,sizeof(pp->symbol)); pp->coinbits = stringbits(symbol); pp->ind = LP_numpriceinfos++; LP_numpriceinfos++; if ( (retjson= LP_priceinfomatrix(0)) != 0 ) free_json(retjson); return(pp); } struct LP_cacheinfo *LP_cacheadd(char *base,char *rel,bits256 txid,int32_t vout,double price,struct LP_quoteinfo *qp) { char str[65]; struct LP_cacheinfo *ptr=0; if ( base == 0 || rel == 0 ) return(0); if ( (ptr= LP_cachefind(base,rel,txid,vout)) == 0 ) { ptr = calloc(1,sizeof(*ptr)); if ( LP_cachekey(ptr->key,base,rel,txid,vout) == sizeof(ptr->key) ) { portable_mutex_lock(&LP_cachemutex); HASH_ADD(hh,LP_cacheinfos,key,sizeof(ptr->key),ptr); portable_mutex_unlock(&LP_cachemutex); } else printf("LP_cacheadd keysize mismatch?\n"); } ptr->Q = *qp; ptr->timestamp = (uint32_t)time(NULL); if ( price != ptr->price ) { ptr->price = price; LP_priceinfoupdate(base,rel,price); printf("updated %s/v%d %s/%s %llu price %.8f\n",bits256_str(str,txid),vout,base,rel,(long long)qp->satoshis,price); } else ptr->price = price; return(ptr); } static int _cmp_orderbook(const void *a,const void *b) { int32_t retval = 0; #define ptr_a (*(struct LP_orderbookentry **)a)->price #define ptr_b (*(struct LP_orderbookentry **)b)->price if ( ptr_b > ptr_a ) retval = -1; else if ( ptr_b < ptr_a ) retval = 1; else { #undef ptr_a #undef ptr_b #define ptr_a ((struct LP_orderbookentry *)a)->basesatoshis #define ptr_b ((struct LP_orderbookentry *)b)->basesatoshis if ( ptr_b > ptr_a ) return(-1); else if ( ptr_b < ptr_a ) return(1); } // printf("%.8f vs %.8f -> %d\n",ptr_a,ptr_b,retval); return(retval); #undef ptr_a #undef ptr_b } cJSON *LP_orderbookjson(struct LP_orderbookentry *op) { cJSON *item = cJSON_CreateObject(); if ( op->price > SMALLVAL ) { jaddnum(item,"price",op->price); jaddnum(item,"volume",dstr(op->basesatoshis)); jaddbits256(item,"txid",op->txid); jaddnum(item,"vout",op->vout); jaddbits256(item,"pubkey",op->pubkey); jaddnum(item,"age",op->age); } return(item); } struct LP_orderbookentry *LP_orderbookentry(char *base,char *rel,bits256 txid,int32_t vout,bits256 txid2,int32_t vout2,double price,uint64_t basesatoshis,bits256 pubkey,int32_t age) { struct LP_orderbookentry *op; if ( (op= calloc(1,sizeof(*op))) != 0 ) { op->txid = txid; op->vout = vout; op->txid2 = txid2; op->vout2 = vout2; op->price = price; op->basesatoshis = basesatoshis; op->pubkey = pubkey; op->age = age; } return(op); } int32_t LP_orderbookfind(struct LP_orderbookentry **array,int32_t num,bits256 txid,int32_t vout) { int32_t i; for (i=0; ivout == vout && bits256_cmp(array[i]->txid,txid) == 0) || (array[i]->vout2 == vout && bits256_cmp(array[i]->txid2,txid) == 0) ) return(i); return(-1); } int32_t LP_orderbook_utxoentries(uint32_t now,int32_t polarity,char *base,char *rel,struct LP_orderbookentry *(**arrayp),int32_t num,int32_t cachednum,int32_t duration) { struct LP_utxoinfo *utxo,*tmp; struct LP_pubkeyinfo *pubp=0; struct LP_priceinfo *basepp; struct LP_orderbookentry *op; uint32_t oldest; double price; int32_t baseid,relid; uint64_t basesatoshis,val,val2; if ( (basepp= LP_priceinfoptr(&relid,base,rel)) != 0 ) baseid = basepp->ind; else return(num); now = (uint32_t)time(NULL); oldest = now - duration; HASH_ITER(hh,LP_utxoinfos[1],utxo,tmp) { if ( pubp == 0 || bits256_cmp(pubp->pubkey,utxo->pubkey) != 0 ) pubp = LP_pubkeyfind(utxo->pubkey); if ( pubp != 0 && pubp->numerrors >= LP_MAXPUBKEY_ERRORS ) continue; //char str[65],str2[65]; printf("check utxo.%s/v%d from %s\n",bits256_str(str,utxo->payment.txid),utxo->payment.vout,bits256_str(str2,utxo->pubkey)); if ( strcmp(base,utxo->coin) == 0 && LP_isavailable(utxo) > 0 && pubp != 0 && (price= pubp->matrix[baseid][relid]) > SMALLVAL && pubp->timestamp > oldest && pubp->timestamp <= now ) { if ( LP_orderbookfind(*arrayp,cachednum,utxo->payment.txid,utxo->payment.vout) < 0 ) { if ( LP_iseligible(&val,&val2,utxo->iambob,utxo->coin,utxo->payment.txid,utxo->payment.vout,utxo->S.satoshis,utxo->deposit.txid,utxo->deposit.vout) == 0 ) continue; if ( polarity > 0 ) basesatoshis = utxo->S.satoshis; else basesatoshis = utxo->S.satoshis * price; //char str[65]; printf("found utxo not in orderbook %s/v%d %.8f %.8f\n",bits256_str(str,utxo->payment.txid),utxo->payment.vout,dstr(basesatoshis),polarity > 0 ? price : 1./price); if ( (op= LP_orderbookentry(base,rel,utxo->payment.txid,utxo->payment.vout,utxo->deposit.txid,utxo->deposit.vout,polarity > 0 ? price : 1./price,basesatoshis,utxo->pubkey,now - pubp->timestamp)) != 0 ) { *arrayp = realloc(*arrayp,sizeof(*(*arrayp)) * (num+1)); (*arrayp)[num++] = op; if ( LP_ismine(utxo) > 0 && utxo->T.lasttime == 0 ) LP_utxo_clientpublish(utxo); } } } } return(num); } char *LP_orderbook(char *base,char *rel,int32_t duration) { uint32_t now,i; struct LP_priceinfo *basepp=0,*relpp=0; struct LP_orderbookentry **bids = 0,**asks = 0; cJSON *retjson,*array; int32_t numbids=0,numasks=0,cachenumbids,cachenumasks,baseid,relid; if ( (basepp= LP_priceinfofind(base)) == 0 || (relpp= LP_priceinfofind(rel)) == 0 ) return(clonestr("{\"error\":\"base or rel not added\"}")); if ( duration <= 0 ) duration = LP_ORDERBOOK_DURATION; baseid = basepp->ind; relid = relpp->ind; now = (uint32_t)time(NULL); cachenumbids = numbids, cachenumasks = numasks; //printf("start cache.(%d %d) numbids.%d numasks.%d\n",cachenumbids,cachenumasks,numbids,numasks); numasks = LP_orderbook_utxoentries(now,1,base,rel,&asks,numasks,cachenumasks,duration); numbids = LP_orderbook_utxoentries(now,-1,rel,base,&bids,numbids,cachenumbids,duration); retjson = cJSON_CreateObject(); array = cJSON_CreateArray(); if ( numbids > 1 ) qsort(bids,numbids,sizeof(*bids),_cmp_orderbook); if ( numasks > 1 ) { for (i=0; iprice); printf(" -> "); qsort(asks,numasks,sizeof(*asks),_cmp_orderbook); for (i=0; iprice); printf("sorted asks.%d\n",numasks); } for (i=0; i SMALLVAL && origprice < price ) price = origprice; } if ( price > SMALLVAL ) { retjson = cJSON_CreateObject(); jaddstr(retjson,"result","success"); jaddstr(retjson,"method","postprice"); jaddbits256(retjson,"pubkey",LP_mypub25519); jaddstr(retjson,"base",base); jaddstr(retjson,"rel",rel); jaddnum(retjson,"price",price); jadd(retjson,"theoretical",LP_priceinfomatrix(0)); jadd(retjson,"quotes",LP_priceinfomatrix(1)); return(jprint(retjson,1)); } else return(clonestr("{\"error\":\"cant find baserel pair\"}")); } void LP_priceupdate(char *base,char *rel,double price,double avebid,double aveask,double highbid,double lowask,double PAXPRICES[32]) { LP_priceinfoupdate(base,rel,price); } void LP_pricefeedupdate(bits256 pubkey,char *base,char *rel,double price) { struct LP_priceinfo *basepp,*relpp; struct LP_pubkeyinfo *pubp; char str[65]; //printf("check PRICEFEED UPDATE.(%s/%s) %.8f %s\n",base,rel,price,bits256_str(str,pubkey)); if ( price > SMALLVAL && (basepp= LP_priceinfofind(base)) != 0 && (relpp= LP_priceinfofind(rel)) != 0 ) { printf("PRICEFEED UPDATE.(%s/%s) %.8f %s\n",base,rel,price,bits256_str(str,pubkey)); if ( (pubp= LP_pubkeyadd(pubkey)) != 0 ) { pubp->matrix[basepp->ind][relpp->ind] = price; pubp->matrix[relpp->ind][basepp->ind] = 1. / price; pubp->timestamp = (uint32_t)time(NULL); } else printf("error creating pubkey entry\n"); } else printf("error finding %s/%s %.8f\n",base,rel,price); } int32_t LP_autoprices; int32_t LP_autoprice(char *base,char *rel,double minprice,double margin,char *type) { struct LP_priceinfo *basepp,*relpp; if ( (basepp= LP_priceinfofind(base)) != 0 && (relpp= LP_priceinfofind(rel)) != 0 ) { basepp->minprices[relpp->ind] = minprice; basepp->margins[relpp->ind] = margin; LP_autoprices++; return(0); } return(-1); } void LP_autopriceset(void *ctx,int32_t dir,struct LP_priceinfo *relpp,struct LP_priceinfo *basepp,double price) { double margin,minprice; int32_t changed; if ( (margin= basepp->margins[relpp->ind]) != 0. ) { if ( dir > 0 ) price = 1. / (price * (1. - margin)); else price = (price * (1. + margin)); //printf("%s/%s %.8f dir.%d margin %.8f\n",relpp->symbol,basepp->symbol,price,dir,margin); if ( (minprice= basepp->minprices[relpp->ind]) == 0. || price >= minprice ) { LP_mypriceset(&changed,relpp->symbol,basepp->symbol,price); //printf("changed.%d\n",changed); if ( changed != 0 ) LP_pricepings(ctx,LP_myipaddr,LP_mypubsock,relpp->symbol,basepp->symbol,price); } } } double LP_pricesparse(void *ctx,int32_t trexflag,char *retstr,struct LP_priceinfo *btcpp) { //{"success":true,"message":"","result":[{"MarketName":"BTC-KMD","High":0.00040840,"Low":0.00034900,"Volume":328042.46061669,"Last":0.00037236,"BaseVolume":123.36439511,"TimeStamp":"2017-07-15T13:50:21.87","Bid":0.00035721,"Ask":0.00037069,"OpenBuyOrders":343,"OpenSellOrders":1690,"PrevDay":0.00040875,"Created":"2017-02-11T23:04:01.853"}, //{"TradePairId":4762,"Label":"WAVES/BTC","AskPrice":0.00099989,"BidPrice":0.00097350,"Low":0.00095000,"High":0.00108838,"Volume":6501.24403100,"LastPrice":0.00098028,"BuyVolume":1058994.86554882,"SellVolume":2067.87377158,"Change":-7.46,"Open":0.00105926,"Close":0.00098028,"BaseVolume":6.52057452,"BuyBaseVolume":2.33098660,"SellBaseVolume":1167.77655709}, int32_t i,j,n,iter; double price,kmdbtc,bid,ask,nxtkmd=0.; struct LP_priceinfo *coinpp,*refpp; char symbol[16],*name,*refcoin; cJSON *retjson,*array,*item; if ( (retjson= cJSON_Parse(retstr)) != 0 ) { //printf("got.(%s)\n",retstr); kmdbtc = 0.; refcoin = "BTC"; refpp = btcpp; if ( (array= jarray(&n,retjson,trexflag != 0 ? "result" : "Data")) != 0 ) { for (iter=0; iter<2; iter++) { for (i=0; i SMALLVAL && strcmp(symbol,"NXT") == 0 ) nxtkmd = 0.5 * (bid + ask) / kmdbtc; if ( (coinpp= LP_priceinfofind(symbol)) != 0 ) { coinpp->high[trexflag] = jdouble(item,"High"); coinpp->low[trexflag] = jdouble(item,"Low"); //coinpp->volume = jdouble(item,"Volume"); //coinpp->btcvolume = jdouble(item,"BaseVolume"); coinpp->last[trexflag] = jdouble(item,trexflag != 0 ? "Last" : "LastPrice"); coinpp->bid[trexflag] = bid; coinpp->ask[trexflag] = ask; //coinpp->prevday = jdouble(item,"PrevDay"); //printf("iter.%d trexflag.%d %s high %.8f, low %.8f, last %.8f hbla.(%.8f %.8f)\n",iter,trexflag,symbol,coinpp->high[trexflag],coinpp->low[trexflag],coinpp->last[trexflag],coinpp->bid[trexflag],coinpp->ask[trexflag]); if ( coinpp->bid[trexflag] > SMALLVAL && coinpp->ask[trexflag] > SMALLVAL ) { price = 0.5 * (coinpp->bid[trexflag] + coinpp->ask[trexflag]); if ( iter == 0 ) { if ( strcmp(symbol,"KMD") == 0 ) kmdbtc = price; } else { if ( strcmp(symbol,"KMD") == 0 ) continue; //printf("(%s/%s) iter.%d trexflag.%d %s %.8f %.8f\n",refpp->symbol,coinpp->symbol,iter,trexflag,symbol,price,price/kmdbtc); price /= kmdbtc; } if ( trexflag == 0 && coinpp->bid[1] > SMALLVAL && coinpp->ask[1] > SMALLVAL ) { //printf("have trex: iter.%d trexflag.%d %s %.8f %.8f\n",iter,trexflag,symbol,coinpp->bid[1],coinpp->ask[1]); continue; } LP_autopriceset(ctx,1,refpp,coinpp,price); LP_autopriceset(ctx,-1,coinpp,refpp,price); } } } } } refcoin = "KMD"; if ( kmdbtc == 0. || (refpp= LP_priceinfofind("KMD")) == 0 ) break; } } free_json(retjson); } return(nxtkmd); } static char *assetids[][3] = { { "12071612744977229797", "UNITY", "10000" }, { "15344649963748848799", "DEX", "1" }, { "6883271355794806507", "PANGEA", "10000" }, { "17911762572811467637", "JUMBLR", "10000" }, { "17083334802666450484", "BET", "10000" }, { "13476425053110940554", "CRYPTO", "1000" }, { "6932037131189568014", "HODL", "1" }, { "3006420581923704757", "SHARK", "10000" }, { "17571711292785902558", "BOTS", "1" }, { "10524562908394749924", "MGW", "1" }, }; void prices_loop(void *ignore) { char *retstr; cJSON *retjson,*bid,*ask; uint64_t bidsatoshis,asksatoshis; int32_t i; double nxtkmd,price; struct LP_priceinfo *btcpp,*kmdpp,*fiatpp,*nxtpp; void *ctx = bitcoin_ctx(); while ( 1 ) { if ( LP_autoprices == 0 ) { sleep(60); continue; } if ( (btcpp= LP_priceinfofind("BTC")) == 0 ) { sleep(60); continue; } if ( (retstr= issue_curlt("https://bittrex.com/api/v1.1/public/getmarketsummaries",LP_HTTP_TIMEOUT*10)) == 0 ) { printf("error getting marketsummaries\n"); sleep(60); continue; } nxtkmd = LP_pricesparse(ctx,1,retstr,btcpp); free(retstr); if ( (retstr= issue_curlt("https://www.cryptopia.co.nz/api/GetMarkets",LP_HTTP_TIMEOUT*10)) == 0 ) { printf("error getting marketsummaries\n"); sleep(60); continue; } LP_pricesparse(ctx,0,retstr,btcpp); free(retstr); if ( (kmdpp= LP_priceinfofind("KMD")) != 0 ) { for (i=0; i<32; i++) { if ( (fiatpp= LP_priceinfofind(CURRENCIES[i])) != 0 ) { if ( (retjson= LP_paxprice(CURRENCIES[i])) != 0 ) { //printf("(%s %.8f %.8f) ",CURRENCIES[i],jdouble(retjson,"price"),jdouble(retjson,"invprice")); price = jdouble(retjson,"price"); LP_autopriceset(ctx,1,kmdpp,fiatpp,price); LP_autopriceset(ctx,-1,fiatpp,kmdpp,price); free_json(retjson); } } } } if ( nxtkmd > SMALLVAL ) { for (i=0; i (%s) nxtkmd %.8f %.8f %.8f\n",assetids[i][1],assetids[i][0],jprint(retjson,0),nxtkmd,0.5*dstr(bidsatoshis + asksatoshis),price); free_json(retjson); } } } } sleep(60); } }