|
|
@ -18,6 +18,15 @@ |
|
|
|
// LP_ordermatch.c
|
|
|
|
// marketmaker
|
|
|
|
//
|
|
|
|
|
|
|
|
struct LP_gtcorder |
|
|
|
{ |
|
|
|
struct LP_gtcorder *next,*prev; |
|
|
|
struct LP_quoteinfo Q; |
|
|
|
double maxprice; |
|
|
|
uint32_t cancelled,pending; |
|
|
|
} *GTCorders; |
|
|
|
|
|
|
|
struct LP_quoteinfo LP_Alicequery,LP_Alicereserved; |
|
|
|
double LP_Alicemaxprice; |
|
|
|
bits256 LP_Alicedestpubkey,LP_bobs_reserved; |
|
|
@ -567,10 +576,43 @@ int32_t LP_connectstartbob(void *ctx,int32_t pubsock,char *base,char *rel,double |
|
|
|
return(retval); |
|
|
|
} |
|
|
|
|
|
|
|
void LP_gtc_iteration(void *ctx,char *myipaddr,int32_t mypubsock) |
|
|
|
{ |
|
|
|
struct LP_gtcorder *gtc,*tmp; struct LP_quoteinfo *qp; uint64_t destvalue,destvalue2; |
|
|
|
if ( Alice_expiration != 0 ) |
|
|
|
return; |
|
|
|
DL_FOREACH_SAFE(GTCorders,gtc,tmp) |
|
|
|
{ |
|
|
|
qp = >c->Q; |
|
|
|
if ( gtc->cancelled == 0 && LP_iseligible(&destvalue,&destvalue2,0,qp->destcoin,qp->desttxid,qp->destvout,qp->destsatoshis,qp->feetxid,qp->feevout) == 0 ) |
|
|
|
{ |
|
|
|
gtc->cancelled = (uint32_t)time(NULL); |
|
|
|
LP_failedmsg(qp->R.requestid,qp->R.quoteid,-9997,qp->uuidstr); |
|
|
|
} |
|
|
|
if ( gtc->cancelled != 0 ) |
|
|
|
{ |
|
|
|
portable_mutex_lock(&LP_gtcmutex); |
|
|
|
DL_DELETE(GTCorders,gtc); |
|
|
|
free(gtc); |
|
|
|
portable_mutex_unlock(&LP_gtcmutex); |
|
|
|
} |
|
|
|
else |
|
|
|
{ |
|
|
|
if ( time(NULL) > gtc->pending+LP_AUTOTRADE_TIMEOUT*10 ) |
|
|
|
{ |
|
|
|
qp->timestamp = (uint32_t)time(NULL); |
|
|
|
LP_query(ctx,myipaddr,mypubsock,"request",qp); |
|
|
|
LP_Alicequery = *qp, LP_Alicemaxprice = gtc->maxprice, Alice_expiration = qp->timestamp + 2*LP_AUTOTRADE_TIMEOUT, LP_Alicedestpubkey = qp->srchash; |
|
|
|
char str[65]; printf("LP_gtc fill.%d gtc.%d %s/%s %.8f vol %.8f dest.(%s) maxprice %.8f etomicdest.(%s) uuid.%s fill.%d gtc.%d\n",qp->fill,qp->gtc,qp->srccoin,qp->destcoin,dstr(qp->satoshis),dstr(qp->destsatoshis),bits256_str(str,LP_Alicedestpubkey),gtc->maxprice,qp->etomicdest,qp->uuidstr,qp->fill,qp->gtc); |
|
|
|
break; |
|
|
|
} |
|
|
|
} |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
char *LP_trade(void *ctx,char *myipaddr,int32_t mypubsock,struct LP_quoteinfo *qp,double maxprice,int32_t timeout,int32_t duration,uint32_t tradeid,bits256 destpubkey,char *uuidstr) |
|
|
|
{ |
|
|
|
double price; |
|
|
|
price = 0.; |
|
|
|
struct LP_gtcorder *gtc; |
|
|
|
memset(qp->txid.bytes,0,sizeof(qp->txid)); |
|
|
|
qp->txid2 = qp->txid; |
|
|
|
qp->aliceid = LP_aliceid_calc(qp->desttxid,qp->destvout,qp->feetxid,qp->feevout); |
|
|
@ -578,8 +620,20 @@ char *LP_trade(void *ctx,char *myipaddr,int32_t mypubsock,struct LP_quoteinfo *q |
|
|
|
qp->tradeid = LP_rand(); |
|
|
|
qp->srchash = destpubkey; |
|
|
|
strncpy(qp->uuidstr,uuidstr,sizeof(qp->uuidstr)-1); |
|
|
|
qp->timestamp = (uint32_t)time(NULL); |
|
|
|
if ( qp->gtc != 0 ) |
|
|
|
{ |
|
|
|
strcpy(&qp->uuidstr[strlen(qp->uuidstr)-6],"cccccc"); |
|
|
|
gtc = calloc(1,sizeof(*gtc)); |
|
|
|
gtc->Q = *qp; |
|
|
|
gtc->maxprice = maxprice; |
|
|
|
gtc->pending = (uint32_t)time(NULL); |
|
|
|
portable_mutex_lock(&LP_gtcmutex); |
|
|
|
DL_APPEND(GTCorders,gtc); |
|
|
|
portable_mutex_unlock(&LP_gtcmutex); |
|
|
|
} |
|
|
|
LP_query(ctx,myipaddr,mypubsock,"request",qp); |
|
|
|
LP_Alicequery = *qp, LP_Alicemaxprice = maxprice, Alice_expiration = qp->timestamp + timeout, LP_Alicedestpubkey = destpubkey; |
|
|
|
LP_Alicequery = *qp, LP_Alicemaxprice = maxprice, Alice_expiration = qp->timestamp + timeout, LP_Alicedestpubkey = qp->srchash; |
|
|
|
char str[65]; printf("LP_trade fill.%d gtc.%d %s/%s %.8f vol %.8f dest.(%s) maxprice %.8f etomicdest.(%s) uuid.%s fill.%d gtc.%d\n",qp->fill,qp->gtc,qp->srccoin,qp->destcoin,dstr(qp->satoshis),dstr(qp->destsatoshis),bits256_str(str,LP_Alicedestpubkey),maxprice,qp->etomicdest,qp->uuidstr,qp->fill,qp->gtc); |
|
|
|
return(LP_recent_swaps(0,uuidstr)); |
|
|
|
} |
|
|
@ -628,16 +682,45 @@ char *LP_cancel_order(char *uuidstr) |
|
|
|
int32_t num = 0; cJSON *retjson; |
|
|
|
if ( uuidstr != 0 ) |
|
|
|
{ |
|
|
|
num = LP_trades_canceluuid(uuidstr); |
|
|
|
retjson = cJSON_CreateObject(); |
|
|
|
jaddstr(retjson,"result","success"); |
|
|
|
jaddnum(retjson,"numentries",num); |
|
|
|
if ( strcmp(LP_Alicequery.uuidstr,uuidstr) == 0 ) |
|
|
|
if ( uuidstr[0] == 'G' ) |
|
|
|
{ |
|
|
|
LP_failedmsg(LP_Alicequery.R.requestid,LP_Alicequery.R.quoteid,-9998,LP_Alicequery.uuidstr); |
|
|
|
LP_alicequery_clear(); |
|
|
|
jaddstr(retjson,"status","uuid canceled"); |
|
|
|
} else jaddstr(retjson,"status","will stop trade negotiation, but if swap started it wont cancel"); |
|
|
|
struct LP_gtcorder *gtc,*tmp; |
|
|
|
DL_FOREACH_SAFE(GTCorders,gtc,tmp) |
|
|
|
{ |
|
|
|
if ( strcmp(gtc->Q.uuidstr,uuidstr) == 0 ) |
|
|
|
{ |
|
|
|
retjson = cJSON_CreateObject(); |
|
|
|
jaddstr(retjson,"result","success"); |
|
|
|
jaddstr(retjson,"cancelled",uuidstr); |
|
|
|
jaddnum(retjson,"pending",gtc->pending); |
|
|
|
if ( gtc->cancelled == 0 ) |
|
|
|
{ |
|
|
|
gtc->cancelled = (uint32_t)time(NULL); |
|
|
|
jaddstr(retjson,"status","uuid canceled"); |
|
|
|
LP_failedmsg(gtc->Q.R.requestid,gtc->Q.R.quoteid,-9997,gtc->Q.uuidstr); |
|
|
|
} |
|
|
|
else |
|
|
|
{ |
|
|
|
jaddstr(retjson,"status","uuid already canceled"); |
|
|
|
LP_failedmsg(gtc->Q.R.requestid,gtc->Q.R.quoteid,-9996,gtc->Q.uuidstr); |
|
|
|
} |
|
|
|
} |
|
|
|
} |
|
|
|
return(clonestr("{\"error\":\"gtc uuid not found\"}")); |
|
|
|
} |
|
|
|
else |
|
|
|
{ |
|
|
|
num = LP_trades_canceluuid(uuidstr); |
|
|
|
retjson = cJSON_CreateObject(); |
|
|
|
jaddstr(retjson,"result","success"); |
|
|
|
jaddnum(retjson,"numentries",num); |
|
|
|
if ( strcmp(LP_Alicequery.uuidstr,uuidstr) == 0 ) |
|
|
|
{ |
|
|
|
LP_failedmsg(LP_Alicequery.R.requestid,LP_Alicequery.R.quoteid,-9998,LP_Alicequery.uuidstr); |
|
|
|
LP_alicequery_clear(); |
|
|
|
jaddstr(retjson,"status","uuid canceled"); |
|
|
|
} else jaddstr(retjson,"status","will stop trade negotiation, but if swap started it wont cancel"); |
|
|
|
} |
|
|
|
return(jprint(retjson,1)); |
|
|
|
} |
|
|
|
return(clonestr("{\"error\":\"uuid not cancellable\"}")); |
|
|
@ -917,7 +1000,7 @@ double LP_trades_pricevalidate(struct LP_quoteinfo *qp,struct iguana_info *coin, |
|
|
|
struct LP_quoteinfo *LP_trades_gotrequest(void *ctx,struct LP_quoteinfo *qp,struct LP_quoteinfo *newqp,char *pairstr) |
|
|
|
{ |
|
|
|
int32_t voliters=10,priceiters=33; |
|
|
|
double price=0.,p=0.,qprice,myprice,bestprice,range,bid,ask; struct iguana_info *coin,*othercoin; struct LP_utxoinfo A,B,*autxo,*butxo; cJSON *reqjson; char str[65]; struct LP_address_utxo *utxos[4096]; int32_t i,j,r,counter,max = (int32_t)(sizeof(utxos)/sizeof(*utxos)); |
|
|
|
double price=0.,p=0.,qprice,myprice,bestprice,range,bid,ask; uint64_t satoshis; struct iguana_info *coin,*othercoin; struct LP_utxoinfo A,B,*autxo,*butxo; cJSON *reqjson,*retjson; char str[65],*retstr,*txidstr,*hexstr; struct LP_address_utxo *utxos[4096]; int32_t i,j,notarized,r,num,counter,max = (int32_t)(sizeof(utxos)/sizeof(*utxos)); |
|
|
|
*newqp = *qp; |
|
|
|
qp = newqp; |
|
|
|
printf("bob %s received REQUEST.(%s) fill.%d gtc.%d\n",bits256_str(str,G.LP_mypub25519),qp->uuidstr+32,qp->fill,qp->gtc); |
|
|
@ -1017,6 +1100,8 @@ printf("bob %s received REQUEST.(%s) fill.%d gtc.%d\n",bits256_str(str,G.LP_mypu |
|
|
|
qp->quotetime = (uint32_t)time(NULL); |
|
|
|
break; |
|
|
|
} |
|
|
|
if ( qp->fill != 0 ) |
|
|
|
break; |
|
|
|
qp->destsatoshis = (qp->destsatoshis * 2) / 3; |
|
|
|
} |
|
|
|
if ( butxo != 0 && j < voliters ) |
|
|
@ -1049,7 +1134,29 @@ printf("bob %s received REQUEST.(%s) fill.%d gtc.%d\n",bits256_str(str,G.LP_mypu |
|
|
|
} |
|
|
|
else if ( qp->fill != 0 || i == priceiters ) |
|
|
|
{ |
|
|
|
printf("i.%d cant find utxopair aliceid.%llu %s/%s %.8f -> relvol %.8f\n",i,(long long)qp->aliceid,qp->srccoin,qp->destcoin,dstr(LP_basesatoshis(dstr(qp->destsatoshis),price,qp->txfee,qp->desttxfee)),dstr(qp->destsatoshis)); |
|
|
|
printf("i.%d cant find utxopair aliceid.%llu %s/%s %.8f -> relvol %.8f txfee %.8f\n",i,(long long)qp->aliceid,qp->srccoin,qp->destcoin,dstr(LP_basesatoshis(dstr(qp->destsatoshis),price,qp->txfee,qp->desttxfee)),dstr(qp->destsatoshis),dstr(qp->txfee)); |
|
|
|
if ( qp->gtc != 0 && qp->fill != 0 && coin != 0 && LP_getheight(¬arized,coin) > coin->bobfillheight+3 ) |
|
|
|
{ |
|
|
|
satoshis = LP_basesatoshis(dstr(qp->destsatoshis),price,qp->txfee,qp->desttxfee) + 3*qp->txfee; |
|
|
|
LP_address_utxo_reset(&num,coin); |
|
|
|
if ( (retstr= LP_autofillbob(coin,satoshis*1.02)) != 0 ) |
|
|
|
{ |
|
|
|
if ( (retjson= cJSON_Parse(retstr)) != 0 ) |
|
|
|
{ |
|
|
|
if ( (hexstr= jstr(retjson,"hex")) != 0 ) |
|
|
|
{ |
|
|
|
if ( (txidstr= LP_sendrawtransaction(coin->symbol,hexstr,0)) != 0 ) |
|
|
|
{ |
|
|
|
printf("autofill created %s\n",txidstr); |
|
|
|
free(txidstr); |
|
|
|
coin->bobfillheight = LP_getheight(¬arized,coin); |
|
|
|
} |
|
|
|
} |
|
|
|
free_json(retjson); |
|
|
|
} |
|
|
|
free(retstr); |
|
|
|
} |
|
|
|
} |
|
|
|
return(0); |
|
|
|
} |
|
|
|
else |
|
|
@ -1668,8 +1775,8 @@ char *LP_autobuy(void *ctx,int32_t fomoflag,char *myipaddr,int32_t mypubsock,cha |
|
|
|
} |
|
|
|
} |
|
|
|
int32_t changed; |
|
|
|
Q.gtc = gtcflag; |
|
|
|
Q.fill = fillflag; |
|
|
|
Q.gtc = gtcflag; |
|
|
|
LP_mypriceset(&changed,rel,base,1. / maxprice); |
|
|
|
LP_mypriceset(&changed,base,rel,0.); |
|
|
|
if ( uuidstr == 0 || uuidstr[0] == 0 ) |
|
|
|