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.

752 lines
31 KiB

8 years ago
/******************************************************************************
8 years ago
* Copyright © 2014-2017 The SuperNET Developers. *
8 years ago
* *
* 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
8 years ago
// requestid is invariant for a specific request
// quoteid is invariant for a specific request after dest fields are set
8 years ago
#ifdef ENABLE_DEXPING
int32_t basilisk_ping_processDEX(struct supernet_info *myinfo,uint32_t senderipbits,uint8_t *data,int32_t datalen)
{
int32_t i,n,len=0; struct basilisk_relay *relay; struct basilisk_request R; uint8_t clen,serialized[256]; uint16_t sn; uint32_t crc;
portable_mutex_lock(&myinfo->DEX_reqmutex);
len += iguana_rwnum(0,&data[len],sizeof(sn),&sn);
if ( (relay= basilisk_request_ensure(myinfo,senderipbits,sn)) != 0 )
{
relay->numrequests = 0;
for (i=0; i<sn; i++)
{
clen = data[len++];
if ( len+clen <= datalen )
{
if ( relay->numrequests < relay->maxrequests )
{
memcpy(serialized,&data[len],clen);
8 years ago
printf("ping processDEX\n");
8 years ago
n = basilisk_rwDEXquote(0,serialized,&R);
if ( n != clen )
printf("n.%d clen.%d\n",n,clen);
len += clen;
crc = basilisk_requestid(&R);
if ( crc == R.requestid )
{
relay->requests[relay->numrequests++] = R;
//printf("[(%s %.8f) -> (%s %.8f) r.%u q.%u] ",R.src,dstr(R.srcamount),R.dest,dstr(R.destamount),R.requestid,R.quoteid);
} else printf("crc.%u error vs %u\n",crc,R.requestid);
} else printf("relay num.%d >= max.%d\n",relay->numrequests,relay->maxrequests);
} else len += clen;
}
}
else
{
for (i=0; i<sn; i++)
{
if ( len+clen <= datalen )
{
clen = data[len++];
len += clen;
}
}
}
portable_mutex_unlock(&myinfo->DEX_reqmutex);
return(len);
}
int32_t basilisk_ping_genDEX(struct supernet_info *myinfo,uint8_t *data,int32_t maxlen)
{
struct queueitem *item,*tmp; uint8_t clen; int32_t i,datalen = 0; uint16_t sn; uint32_t timestamp,now;
datalen += sizeof(uint16_t);
i = 0;
now = (uint32_t)time(NULL);
portable_mutex_lock(&myinfo->DEX_mutex);
DL_FOREACH_SAFE(myinfo->DEX_quotes,item,tmp)
{
memcpy(&clen,&item[1],sizeof(clen));
if ( datalen+clen < maxlen )
{
memcpy(&data[datalen],&item[1],clen+1), datalen += (clen + 1);
i++;
}
iguana_rwnum(0,(void *)((long)&item[1] + 1 + sizeof(uint32_t)),sizeof(timestamp),&timestamp);
if ( now > timestamp + BASILISK_DEXDURATION )
{
DL_DELETE(myinfo->DEX_quotes,item);
free(item);
} //else printf("now.%u vs timestamp.%u, lag.%d\n",now,timestamp,now-timestamp);
}
portable_mutex_unlock(&myinfo->DEX_mutex);
sn = i;
iguana_rwnum(1,data,sizeof(sn),&sn); // fill in at beginning
return(datalen);
}
#endif
8 years ago
int32_t basilisk_rwDEXquote(int32_t rwflag,uint8_t *serialized,struct basilisk_request *rp)
{
int32_t len = 0;
len += iguana_rwnum(rwflag,&serialized[len],sizeof(rp->requestid),&rp->requestid);
8 years ago
len += iguana_rwnum(rwflag,&serialized[len],sizeof(rp->timestamp),&rp->timestamp); // must be 2nd
8 years ago
len += iguana_rwnum(rwflag,&serialized[len],sizeof(rp->quoteid),&rp->quoteid);
8 years ago
len += iguana_rwnum(rwflag,&serialized[len],sizeof(rp->quotetime),&rp->quotetime);
8 years ago
len += iguana_rwnum(rwflag,&serialized[len],sizeof(rp->optionhours),&rp->optionhours);
8 years ago
len += iguana_rwnum(rwflag,&serialized[len],sizeof(rp->srcamount),&rp->srcamount);
8 years ago
len += iguana_rwnum(rwflag,&serialized[len],sizeof(rp->minamount),&rp->minamount);
8 years ago
len += iguana_rwbignum(rwflag,&serialized[len],sizeof(rp->srchash),rp->srchash.bytes);
8 years ago
len += iguana_rwbignum(rwflag,&serialized[len],sizeof(rp->desthash),rp->desthash.bytes);
len += iguana_rwnum(rwflag,&serialized[len],sizeof(rp->destamount),&rp->destamount);
8 years ago
if ( rwflag != 0 )
{
memcpy(&serialized[len],rp->src,sizeof(rp->src)), len += sizeof(rp->src);
memcpy(&serialized[len],rp->dest,sizeof(rp->dest)), len += sizeof(rp->dest);
}
else
{
memcpy(rp->src,&serialized[len],sizeof(rp->src)), len += sizeof(rp->src);
memcpy(rp->dest,&serialized[len],sizeof(rp->dest)), len += sizeof(rp->dest);
}
8 years ago
if ( rp->quoteid != 0 && basilisk_quoteid(rp) != rp->quoteid )
8 years ago
printf(" basilisk_rwDEXquote.%d: quoteid.%u mismatch calc %u rp.%p\n",rwflag,rp->quoteid,basilisk_quoteid(rp),rp);
8 years ago
if ( basilisk_requestid(rp) != rp->requestid )
8 years ago
printf(" basilisk_rwDEXquote.%d: requestid.%u mismatch calc %u rp.%p\n",rwflag,rp->requestid,basilisk_requestid(rp),rp);
8 years ago
return(len);
}
8 years ago
uint32_t basilisk_request_enqueue(struct supernet_info *myinfo,struct basilisk_request *rp)
8 years ago
{
8 years ago
uint8_t serialized[256]; int32_t len; struct queueitem *item;
8 years ago
//printf(" basilisk_request_enqueue\n");
8 years ago
len = basilisk_rwDEXquote(1,serialized+1,rp);
if ( (item= calloc(1,sizeof(*item) + len + 1)) != 0 )
8 years ago
{
8 years ago
serialized[0] = len;
8 years ago
memcpy(&item[1],serialized,len + 1);
portable_mutex_lock(&myinfo->DEX_mutex);
DL_APPEND(myinfo->DEX_quotes,item);
portable_mutex_unlock(&myinfo->DEX_mutex);
8 years ago
//printf("ENQUEUE.%u calc.%u\n",rp->requestid,basilisk_requestid(rp));
8 years ago
return(rp->requestid);
8 years ago
}
8 years ago
return(0);
8 years ago
}
8 years ago
cJSON *basilisk_requestjson(struct basilisk_request *rp)
8 years ago
{
8 years ago
cJSON *item = cJSON_CreateObject();
/*if ( rp->relaybits != 0 )
8 years ago
{
expand_ipbits(ipaddr,rp->relaybits);
jaddstr(item,"relay",ipaddr);
8 years ago
}*/
8 years ago
jaddbits256(item,"srchash",rp->srchash);
8 years ago
if ( bits256_nonz(rp->desthash) != 0 )
jaddbits256(item,"desthash",rp->desthash);
8 years ago
jaddstr(item,"src",rp->src);
if ( rp->srcamount != 0 )
8 years ago
jadd64bits(item,"srcamount",rp->srcamount);
8 years ago
if ( rp->minamount != 0 )
8 years ago
jadd64bits(item,"minamount",rp->minamount);
8 years ago
jaddstr(item,"dest",rp->dest);
if ( rp->destamount != 0 )
8 years ago
jadd64bits(item,"destamount",rp->destamount);
8 years ago
jaddnum(item,"quotetime",rp->quotetime);
8 years ago
jaddnum(item,"timestamp",rp->timestamp);
jaddnum(item,"requestid",rp->requestid);
jaddnum(item,"quoteid",rp->quoteid);
8 years ago
jaddnum(item,"optionhours",rp->optionhours);
jaddnum(item,"profit",(double)rp->profitmargin / 1000000.);
8 years ago
if ( rp->quoteid != 0 && basilisk_quoteid(rp) != rp->quoteid )
printf("quoteid mismatch %u vs %u\n",basilisk_quoteid(rp),rp->quoteid);
if ( basilisk_requestid(rp) != rp->requestid )
8 years ago
printf("requestid mismatch %u vs calc %u\n",rp->requestid,basilisk_requestid(rp));
8 years ago
{
int32_t i; struct basilisk_request R;
if ( basilisk_parsejson(&R,item) != 0 )
{
8 years ago
if ( memcmp(&R,rp,sizeof(*rp)-sizeof(uint32_t)) != 0 )
8 years ago
{
for (i=0; i<sizeof(*rp); i++)
printf("%02x",((uint8_t *)rp)[i]);
8 years ago
printf(" <- rp.%p\n",rp);
8 years ago
for (i=0; i<sizeof(R); i++)
printf("%02x",((uint8_t *)&R)[i]);
printf(" <- R mismatch\n");
8 years ago
for (i=0; i<sizeof(R); i++)
if ( ((uint8_t *)rp)[i] != ((uint8_t *)&R)[i] )
8 years ago
printf("(%02x %02x).%d ",((uint8_t *)rp)[i],((uint8_t *)&R)[i],i);
8 years ago
printf("mismatches\n");
8 years ago
} //else printf("matched JSON conv %u %u\n",basilisk_requestid(&R),basilisk_requestid(rp));
8 years ago
}
}
8 years ago
return(item);
}
8 years ago
int32_t basilisk_request_create(struct basilisk_request *rp,cJSON *valsobj,bits256 desthash,uint32_t timestamp)
8 years ago
{
8 years ago
char *dest,*src; uint32_t i;
8 years ago
memset(rp,0,sizeof(*rp));
if ( (dest= jstr(valsobj,"dest")) != 0 && (src= jstr(valsobj,"source")) != 0 && (rp->srcamount= j64bits(valsobj,"satoshis")) != 0 )
{
if ( (rp->destamount= j64bits(valsobj,"destsatoshis")) != 0 )
{
8 years ago
rp->desthash = desthash;
8 years ago
for (i=0; i<4; i++)
if ( rp->desthash.ulongs[i] != 0 )
break;
if ( i != 4 )
rp->destamount = 0;
}
8 years ago
rp->minamount = j64bits(valsobj,"minamount");
8 years ago
rp->timestamp = timestamp;
8 years ago
rp->srchash = jbits256(valsobj,"srchash");
8 years ago
rp->optionhours = jint(valsobj,"optionhours");
rp->profitmargin = jdouble(valsobj,"profit") * 1000000;
8 years ago
strncpy(rp->src,src,sizeof(rp->src)-1);
strncpy(rp->dest,dest,sizeof(rp->dest)-1);
8 years ago
//if ( jstr(valsobj,"relay") != 0 )
// rp->relaybits = (uint32_t)calc_ipbits(jstr(valsobj,"relay"));
8 years ago
rp->requestid = basilisk_requestid(rp);
8 years ago
//printf("set requestid <- %u\n",rp->requestid);
8 years ago
if ( rp->destamount != 0 && bits256_nonz(rp->desthash) != 0 )
{
rp->quoteid = basilisk_quoteid(rp);
8 years ago
//printf("set quoteid.%u\n",rp->quoteid);
8 years ago
}
//printf("create.%u calc.%u\n",rp->requestid,basilisk_requestid(rp));
return(0);
}
return(-1);
}
8 years ago
char *basilisk_start(struct supernet_info *myinfo,bits256 privkey,struct basilisk_request *_rp,uint32_t statebits,int32_t optionduration)
8 years ago
{
8 years ago
cJSON *retjson; bits256 tmpprivkey; struct basilisk_request *rp=0; int32_t i,srcmatch,destmatch;
8 years ago
if ( _rp->requestid == myinfo->lastdexrequestid )
{
8 years ago
//printf("filter duplicate r%u\n",_rp->requestid);
8 years ago
return(clonestr("{\"error\":\"filter duplicate requestid\"}"));
}
8 years ago
srcmatch = smartaddress_pubkey(myinfo,&tmpprivkey,_rp->srchash) >= 0;
destmatch = smartaddress_pubkey(myinfo,&tmpprivkey,_rp->desthash) >= 0;
8 years ago
if ( srcmatch != 0 || destmatch != 0 )
8 years ago
{
8 years ago
for (i=0; i<myinfo->numswaps; i++)
if ( myinfo->swaps[i]->I.req.requestid == _rp->requestid )
{
printf("basilisk_thread_start error trying to start requestid.%u which is already started\n",rp->requestid);
break;
}
if ( i == myinfo->numswaps )
8 years ago
{
8 years ago
rp = calloc(1,sizeof(*rp));
*rp = *_rp;
printf("START thread to complete %u/%u for (%s %.8f) <-> (%s %.8f) q.%u\n",rp->requestid,rp->quoteid,rp->src,dstr(rp->srcamount),rp->dest,dstr(rp->destamount),rp->quoteid);
8 years ago
myinfo->lastdexrequestid = rp->requestid;
8 years ago
if ( basilisk_thread_start(myinfo,privkey,rp,statebits,optionduration) != 0 )
8 years ago
{
basilisk_request_enqueue(myinfo,rp);
return(clonestr("{\"result\":\"started atomic swap thread\"}"));
8 years ago
} else return(clonestr("{\"error\":\"couldnt atomic swap thread\"}"));
8 years ago
}
else
{
printf("trying to start already pending swap.r%u\n",rp->requestid);
return(clonestr("{\"error\":\"cant start pending swap\"}"));
8 years ago
}
8 years ago
}
8 years ago
else if ( myinfo->IAMLP != 0 )
8 years ago
{
retjson = cJSON_CreateObject();
jaddstr(retjson,"result","basilisk node needs to start atomic thread locally");
return(jprint(retjson,1));
} else return(clonestr("{\"error\":\"unexpected basilisk_start not mine and amrelay\"}"));
8 years ago
}
8 years ago
void basilisk_requests_poll(struct supernet_info *myinfo)
8 years ago
{
static uint32_t lastpoll;
8 years ago
char *retstr; uint8_t data[32768]; cJSON *outerarray,*retjson; uint32_t msgid,channel; int32_t datalen,i,n; struct basilisk_request issueR; bits256 privkey; double hwm = 0.;
8 years ago
if ( myinfo->IAMNOTARY != 0 || time(NULL) < lastpoll+20 || (myinfo->IAMLP == 0 && myinfo->DEXactive < time(NULL)) )
8 years ago
return;
lastpoll = (uint32_t)time(NULL);
memset(&issueR,0,sizeof(issueR));
8 years ago
memset(&myinfo->DEXaccept,0,sizeof(myinfo->DEXaccept));
8 years ago
//printf("Call incoming\n");
if ( (retstr= InstantDEX_incoming(myinfo,0,0,0,0)) != 0 )
{
8 years ago
//printf("poll.(%s)\n",retstr);
8 years ago
if ( (retjson= cJSON_Parse(retstr)) != 0 )
{
if ( (outerarray= jarray(&n,retjson,"responses")) != 0 )
{
for (i=0; i<n; i++)
hwm = basilisk_process_results(myinfo,&issueR,jitem(outerarray,i),hwm);
} //else hwm = basilisk_process_results(myinfo,&issueR,outerarray,hwm);
free_json(retjson);
}
free(retstr);
} else printf("null incoming\n");
8 years ago
channel = 'D' + ((uint32_t)'E' << 8) + ((uint32_t)'X' << 16);
8 years ago
if ( hwm > 0. )
{
8 years ago
myinfo->DEXaccept = issueR;
8 years ago
if ( smartaddress_pubkey(myinfo,&privkey,issueR.srchash) >= 0 )
8 years ago
{
8 years ago
printf("matched dex_smartpubkey\n");
8 years ago
dex_channelsend(myinfo,issueR.srchash,issueR.desthash,channel,0x4000000,(void *)&issueR.requestid,sizeof(issueR.requestid)); // 60
8 years ago
dpow_nanomsg_update(myinfo);
dex_updateclient(myinfo);
8 years ago
if ( (retstr= basilisk_start(myinfo,privkey,&issueR,1,issueR.optionhours * 3600)) != 0 )
8 years ago
free(retstr);
8 years ago
}
8 years ago
else if ( issueR.requestid != myinfo->lastdexrequestid )//if ( issueR.quoteid == 0 )
8 years ago
{
issueR.quoteid = basilisk_quoteid(&issueR);
issueR.desthash = myinfo->myaddr.persistent;
datalen = basilisk_rwDEXquote(1,data,&issueR);
msgid = (uint32_t)time(NULL);
8 years ago
printf("other req hwm %f >>>>>>>>>>> send response (%llx -> %llx) last.%u r.%u quoteid.%u\n",hwm,(long long)issueR.desthash.txid,(long long)issueR.srchash.txid,myinfo->lastdexrequestid,issueR.requestid,issueR.quoteid);
8 years ago
dex_channelsend(myinfo,issueR.desthash,issueR.srchash,channel,msgid,data,datalen); //INSTANTDEX_LOCKTIME*2
8 years ago
dpow_nanomsg_update(myinfo);
dex_updateclient(myinfo);
8 years ago
if ( (retstr= basilisk_start(myinfo,myinfo->persistent_priv,&issueR,0,issueR.optionhours * 3600)) != 0 )
8 years ago
free(retstr);
8 years ago
} //else printf("basilisk_requests_poll unexpected hwm issueR\n");
}
}
8 years ago
struct basilisk_relay *basilisk_request_ensure(struct supernet_info *myinfo,uint32_t senderipbits,int32_t numrequests)
{
int32_t j; struct basilisk_relay *relay = 0;
if ( (j= basilisk_relayid(myinfo,senderipbits)) >= 0 )
{
8 years ago
relay = &myinfo->NOTARY.RELAYS[j];
8 years ago
if ( numrequests > relay->maxrequests )
{
relay->maxrequests = numrequests;
relay->requests = realloc(relay->requests,sizeof(*relay->requests) * numrequests);
}
}
return(relay);
}
8 years ago
static int _cmp_requests(const void *a,const void *b)
{
#define uint32_a (*(struct basilisk_request *)a).requestid
#define uint32_b (*(struct basilisk_request *)b).requestid
if ( uint32_b > uint32_a )
return(1);
else if ( uint32_b < uint32_a )
return(-1);
else
{
#undef uint32_a
#undef uint32_b
#define uint32_a (*(struct basilisk_request *)a).quoteid
#define uint32_b (*(struct basilisk_request *)b).quoteid
if ( uint32_b > uint32_a )
return(1);
else if ( uint32_b < uint32_a )
return(-1);
}
return(0);
#undef uint32_a
#undef uint32_b
}
8 years ago
struct basilisk_request *_basilisk_requests_uniq(struct supernet_info *myinfo,int32_t *nump,uint8_t *space,int32_t spacesize,struct basilisk_request *refrp)
8 years ago
{
8 years ago
int32_t i,j,n,k,m; struct basilisk_relay *relay; struct basilisk_request *requests,*rp;
8 years ago
m = 0;
if ( refrp != 0 )
m = 1;
for (j=0; j<myinfo->NOTARY.NUMRELAYS; j++)
8 years ago
m += myinfo->NOTARY.RELAYS[j].numrequests;
8 years ago
if ( m*sizeof(*requests) <= spacesize )
requests = (void *)space;
else requests = calloc(m,sizeof(*requests));
8 years ago
if ( refrp != 0 )
8 years ago
{
8 years ago
requests[0] = *refrp;
8 years ago
//for (i=0; i<sizeof(*refrp); i++)
// printf("%02x",((uint8_t *)refrp)[i]);
//printf(" uniq\n");
8 years ago
}
8 years ago
if ( refrp != 0 )
m = 1;
else m = 0;
for (j=0; j<myinfo->NOTARY.NUMRELAYS; j++)
8 years ago
{
8 years ago
relay = &myinfo->NOTARY.RELAYS[j];
8 years ago
if ( (n= relay->numrequests) > 0 )
{
for (i=0; i<n; i++)
{
rp = &relay->requests[i];
8 years ago
for (k=0; k<m; k++)
if ( memcmp(&requests[k],rp,sizeof(requests[k])) == 0 )
break;
if ( k == m )
8 years ago
{
8 years ago
//requests[m].relaybits = relay->ipbits;
8 years ago
requests[m++] = *rp;
8 years ago
}
}
}
}
8 years ago
qsort(requests,m,sizeof(*requests),_cmp_requests);
8 years ago
*nump = m;
return(requests);
}
8 years ago
/*char *basilisk_respond_swapstatus(struct supernet_info *myinfo,bits256 hash,uint32_t requestid,uint32_t quoteid)
8 years ago
{
cJSON *array,*retjson;
array = cJSON_CreateArray();
retjson = cJSON_CreateObject();
jadd(retjson,"result",array);
return(jprint(retjson,1));
8 years ago
}*/
8 years ago
8 years ago
char *basilisk_respond_requests(struct supernet_info *myinfo,bits256 hash,uint32_t requestid,uint32_t quoteid,struct basilisk_request *refrp)
8 years ago
{
8 years ago
int32_t i,qflag,num=0; cJSON *retjson,*array; struct basilisk_request *requests,*rp; uint8_t space[4096];
8 years ago
array = cJSON_CreateArray();
portable_mutex_lock(&myinfo->DEX_reqmutex);
8 years ago
if ( (requests= _basilisk_requests_uniq(myinfo,&num,space,sizeof(space),refrp)) != 0 )
8 years ago
{
for (i=0; i<num; i++)
{
rp = &requests[i];
8 years ago
if ( quoteid == 0 || (quoteid == rp->quoteid && (bits256_cmp(hash,rp->srchash) == 0 || bits256_cmp(hash,rp->desthash) == 0)) )
8 years ago
qflag = 1;
else qflag = 0;
8 years ago
//int32_t j; for (j=0; j<sizeof(*rp); j++)
// printf("%02x",((uint8_t *)rp)[j]);
//printf(" rp[%d] of %d qflag.%d\n",i,num,qflag);
8 years ago
if ( requestid == 0 || (rp->requestid == requestid && qflag != 0) )
8 years ago
jaddi(array,basilisk_requestjson(rp));
8 years ago
}
}
8 years ago
portable_mutex_unlock(&myinfo->DEX_reqmutex);
8 years ago
if ( requests != (void *)space )
free(requests);
8 years ago
retjson = cJSON_CreateObject();
jadd(retjson,"result",array);
return(jprint(retjson,1));
}
8 years ago
char *basilisk_respond_accept(struct supernet_info *myinfo,bits256 privkey,uint32_t requestid,uint32_t quoteid,struct basilisk_request *refrp)
8 years ago
{
8 years ago
int32_t i,num=0; char *retstr=0; struct basilisk_request *requests,*rp; uint8_t space[4096];
8 years ago
portable_mutex_lock(&myinfo->DEX_reqmutex);
8 years ago
if ( (requests= _basilisk_requests_uniq(myinfo,&num,space,sizeof(space),refrp)) != 0 )
8 years ago
{
8 years ago
for (i=0; i<num; i++)
8 years ago
{
8 years ago
rp = &requests[i];
8 years ago
if ( rp->requestid == requestid && rp->quoteid == quoteid )
8 years ago
{
8 years ago
printf("start from accept\n");
8 years ago
retstr = basilisk_start(myinfo,privkey,rp,1,0);
8 years ago
break;
8 years ago
}
}
}
8 years ago
portable_mutex_unlock(&myinfo->DEX_reqmutex);
8 years ago
if ( requests != (void *)space )
free(requests);
8 years ago
if ( retstr == 0 )
retstr = clonestr("{\"error\":\"couldnt find to requestid to choose\"}");
8 years ago
return(retstr);
}
#include "../includes/iguana_apidefs.h"
#include "../includes/iguana_apideclares.h"
8 years ago
THREE_STRINGS_AND_DOUBLE(tradebot,aveprice,comment,base,rel,basevolume)
{
double retvals[4],aveprice; cJSON *retjson = cJSON_CreateObject();
aveprice = instantdex_avehbla(myinfo,retvals,base,rel,basevolume);
jaddstr(retjson,"result","success");
jaddnum(retjson,"aveprice",aveprice);
jaddnum(retjson,"avebid",retvals[0]);
jaddnum(retjson,"bidvol",retvals[1]);
jaddnum(retjson,"aveask",retvals[2]);
jaddnum(retjson,"askvol",retvals[3]);
return(jprint(retjson,1));
}
8 years ago
ZERO_ARGS(InstantDEX,allcoins)
{
8 years ago
struct iguana_info *tmp; cJSON *native,*notarychains,*basilisk,*virtual,*full,*retjson = cJSON_CreateObject();
8 years ago
full = cJSON_CreateArray();
8 years ago
native = cJSON_CreateArray();
8 years ago
basilisk = cJSON_CreateArray();
virtual = cJSON_CreateArray();
8 years ago
notarychains = cJSON_CreateArray();
8 years ago
HASH_ITER(hh,myinfo->allcoins,coin,tmp)
{
8 years ago
if ( coin->FULLNODE < 0 )
jaddistr(native,coin->symbol);
8 years ago
//else if ( coin->virtualchain != 0 )
// jaddistr(virtual,coin->symbol);
else if ( coin->FULLNODE > 0 )//|| coin->VALIDATENODE > 0 )
8 years ago
jaddistr(full,coin->symbol);
8 years ago
//else if ( coin->notarychain >= 0 )
// jaddistr(notarychains,coin->symbol);
8 years ago
else jaddistr(basilisk,coin->symbol);
8 years ago
}
8 years ago
jadd(retjson,"native",native);
8 years ago
jadd(retjson,"basilisk",basilisk);
jadd(retjson,"full",full);
8 years ago
//jadd(retjson,"virtual",virtual);
//jadd(retjson,"notarychains",notarychains);
8 years ago
return(jprint(retjson,1));
}
8 years ago
cJSON *basilisk_unspents(struct supernet_info *myinfo,struct iguana_info *coin,char *coinaddr)
8 years ago
{
8 years ago
cJSON *unspents=0,*array=0; char *retstr;
8 years ago
if ( coin->FULLNODE > 0 )
{
array = cJSON_CreateArray();
jaddistr(array,coinaddr);
unspents = iguana_listunspents(myinfo,coin,array,0,0,"");
free_json(array);
}
else if ( coin->FULLNODE == 0 )
{
8 years ago
if ( (retstr= dex_listunspent(myinfo,coin,0,0,coin->symbol,coinaddr)) != 0 )
8 years ago
{
unspents = cJSON_Parse(retstr);
free(retstr);
}
}
else unspents = dpow_listunspent(myinfo,coin,coinaddr);
return(unspents);
}
8 years ago
char *basilisk_sendrawtransaction(struct supernet_info *myinfo,struct iguana_info *coin,char *signedtx)
{
char *retstr,buf[65]; bits256 txid;
if ( coin->FULLNODE > 0 )
{
txid = iguana_sendrawtransaction(myinfo,coin,signedtx);
if ( bits256_nonz(txid) )
{
bits256_str(buf,txid);
retstr = clonestr(buf);
} else retstr = clonestr("{\"error\":\"couldnt validate or send signedtx\"}");
}
else if ( coin->FULLNODE == 0 )
{
retstr = _dex_sendrawtransaction(myinfo,coin->symbol,signedtx);
}
else retstr = dpow_sendrawtransaction(myinfo,coin,signedtx);
return(retstr);
}
8 years ago
STRING_ARG(InstantDEX,available,source)
{
8 years ago
uint64_t total = 0; int32_t i,n=0; char coinaddr[64]; cJSON *item,*unspents,*retjson = 0;
8 years ago
if ( source != 0 && source[0] != 0 && (coin= iguana_coinfind(source)) != 0 )
8 years ago
{
if ( myinfo->expiration != 0 )
8 years ago
{
8 years ago
bitcoin_address(coinaddr,coin->chain->pubtype,myinfo->persistent_pubkey33,33);
if ( (unspents= basilisk_unspents(myinfo,coin,coinaddr)) != 0 )
8 years ago
{
8 years ago
//printf("available.(%s)\n",jprint(unspents,0));
8 years ago
if ( (n= cJSON_GetArraySize(unspents)) > 0 )
8 years ago
{
8 years ago
for (i=0; i<n; i++)
8 years ago
{
8 years ago
item = jitem(unspents,i);
8 years ago
//if ( is_cJSON_True(jobj(item,"spendable")) != 0 )
8 years ago
total += jdouble(item,"amount") * SATOSHIDEN;
8 years ago
//printf("(%s) -> %.8f\n",jprint(item,0),dstr(total));
8 years ago
}
8 years ago
}
8 years ago
free_json(unspents);
8 years ago
}
8 years ago
retjson = cJSON_CreateObject();
jaddnum(retjson,"result",dstr(total));
8 years ago
printf(" n.%d total %.8f (%s)\n",n,dstr(total),jprint(retjson,0));
8 years ago
return(jprint(retjson,1));
8 years ago
}
8 years ago
printf("InstantDEX_available: need to unlock wallet\n");
return(clonestr("{\"error\":\"need to unlock wallet\"}"));
}
printf("InstantDEX_available: %s is not active\n",source!=0?source:"");
return(clonestr("{\"error\":\"specified coin is not active\"}"));
8 years ago
}
8 years ago
HASH_ARRAY_STRING(InstantDEX,request,hash,vals,hexstr)
8 years ago
{
8 years ago
uint8_t serialized[512]; bits256 privkey; char buf[512],BTCaddr[64],KMDaddr[64]; struct basilisk_request R; int32_t jumblr,iambob,optionhours; cJSON *reqjson; uint32_t datalen=0,DEX_channel; struct iguana_info *bobcoin,*alicecoin;
8 years ago
myinfo->DEXactive = (uint32_t)time(NULL) + 3*BASILISK_TIMEOUT + 60;
8 years ago
jadd64bits(vals,"minamount",jdouble(vals,"minprice") * jdouble(vals,"amount") * SATOSHIDEN * SATOSHIDEN);
8 years ago
if ( jobj(vals,"desthash") == 0 )
jaddbits256(vals,"desthash",hash);
jadd64bits(vals,"satoshis",jdouble(vals,"amount") * SATOSHIDEN);
jadd64bits(vals,"destsatoshis",jdouble(vals,"destamount") * SATOSHIDEN);
8 years ago
jaddnum(vals,"timestamp",time(NULL));
8 years ago
if ( (jumblr= jint(vals,"usejumblr")) != 0 )
privkey = jumblr_privkey(myinfo,BTCaddr,KMDaddr,jumblr == 1 ? JUMBLR_DEPOSITPREFIX : "");
8 years ago
else privkey = myinfo->persistent_priv;
hash = curve25519(privkey,curve25519_basepoint9());
8 years ago
if ( jobj(vals,"srchash") == 0 )
jaddbits256(vals,"srchash",hash);
8 years ago
printf("service.(%s)\n",jprint(vals,0));
8 years ago
memset(&R,0,sizeof(R));
if ( basilisk_request_create(&R,vals,hash,juint(vals,"timestamp")) == 0 )
8 years ago
{
8 years ago
iambob = bitcoin_coinptrs(hash,&bobcoin,&alicecoin,R.src,R.dest,privkey,GENESIS_PUBKEY);
8 years ago
if ( (optionhours= jint(vals,"optionhours")) != 0 )
{
printf("iambob.%d optionhours.%d R.requestid.%u vs calc %u, q.%u\n",iambob,R.optionhours,R.requestid,basilisk_requestid(&R),R.quoteid);
if ( iambob != 0 && optionhours > 0 )
{
sprintf(buf,"{\"error\":\"illegal call option request hours.%d when iambob.%d\"}",optionhours,iambob);
printf("ERROR.(%s)\n",buf);
return(clonestr(buf));
}
else if ( iambob == 0 && optionhours < 0 )
{
sprintf(buf,"{\"error\":\"illegal put option request hours.%d when iambob.%d\"}",optionhours,iambob);
printf("ERROR.(%s)\n",buf);
return(clonestr(buf));
}
}
8 years ago
//if ( myinfo->IAMNOTARY != 0 || myinfo->NOTARY.RELAYID >= 0 )
// R.relaybits = myinfo->myaddr.myipbits;
if ( (reqjson= basilisk_requestjson(&R)) != 0 )
free_json(reqjson);
datalen = basilisk_rwDEXquote(1,serialized,&R);
8 years ago
//int32_t i; for (i=0; i<sizeof(R); i++)
// printf("%02x",((uint8_t *)&R)[i]);
8 years ago
printf(" R.requestid.%u vs calc %u, q.%u datalen.%d\n",R.requestid,basilisk_requestid(&R),R.quoteid,datalen);
basilisk_rwDEXquote(0,serialized,&R);
} else printf("error creating request\n");
8 years ago
if ( datalen > 0 )
{
8 years ago
uint32_t msgid;//,crc=0,crcs[2],numiters = 0; uint8_t buf[4096];
8 years ago
memset(hash.bytes,0,sizeof(hash));
8 years ago
msgid = (uint32_t)time(NULL);
8 years ago
DEX_channel = 'D' + ((uint32_t)'E' << 8) + ((uint32_t)'X' << 16);
8 years ago
basilisk_channelsend(myinfo,hash,hash,DEX_channel,msgid,serialized,datalen,60);
8 years ago
sleep(3);
8 years ago
/*while ( numiters < 10 && (crc= basilisk_crcsend(myinfo,0,buf,sizeof(buf),hash,myinfo->myaddr.persistent,DEX_channel,msgid,serialized,datalen,crcs)) == 0 )
8 years ago
{
8 years ago
//printf("didnt get back what was sent\n");
8 years ago
sleep(3);
basilisk_channelsend(myinfo,myinfo->myaddr.persistent,hash,DEX_channel,msgid,serialized,datalen,60);
numiters++;
8 years ago
}*/
8 years ago
//if ( crc != 0 )//basilisk_channelsend(myinfo,R.srchash,R.desthash,DEX_channel,(uint32_t)time(NULL),serialized,datalen,30) == 0 )
8 years ago
return(clonestr("{\"result\":\"DEX message sent\"}"));
8 years ago
//else return(clonestr("{\"error\":\"DEX message couldnt be sent\"}"));
8 years ago
}
return(clonestr("{\"error\":\"DEX message not sent\"}"));
8 years ago
}
8 years ago
INT_ARG(InstantDEX,automatched,requestid)
{
// return quoteid
8 years ago
myinfo->DEXactive = (uint32_t)time(NULL) + INSTANTDEX_LOCKTIME;
8 years ago
return(clonestr("{\"result\":\"automatched not yet\"}"));
}
8 years ago
int32_t InstantDEX_incoming_func(struct supernet_info *myinfo,void *ptr,uint8_t *data,int32_t datalen)
{
8 years ago
//int32_t i;
//for (i=0; i<datalen; i++)
// printf("%02x",data[i]);
//printf(" <- incoming\n");
8 years ago
return(0);
}
int32_t InstantDEX_process_channelget(struct supernet_info *myinfo,void *ptr,int32_t (*internal_func)(struct supernet_info *myinfo,void *ptr,uint8_t *data,int32_t datalen),uint32_t channel,uint32_t msgid,uint8_t *data,int32_t datalen,uint32_t expiration,uint32_t duration)
{
return((*internal_func)(myinfo,ptr,data,datalen));
}
8 years ago
INT_ARG(InstantDEX,incoming,requestid)
{
8 years ago
cJSON *retjson,*retarray; bits256 zero; uint32_t DEX_channel,msgid,now; int32_t retval,width,drift=3; uint8_t data[32768];
8 years ago
now = (uint32_t)time(NULL);
8 years ago
memset(&zero,0,sizeof(zero));
8 years ago
width = (now - myinfo->DEXpoll) + 2*drift;
8 years ago
if ( width < (drift+1) )
width = 2*drift+1;
else if ( width > 64 )
8 years ago
width = 64;
myinfo->DEXpoll = now;
8 years ago
retjson = cJSON_CreateObject();
DEX_channel = 'D' + ((uint32_t)'E' << 8) + ((uint32_t)'X' << 16);
8 years ago
msgid = (uint32_t)time(NULL) + drift;
8 years ago
if ( (retarray= basilisk_channelget(myinfo,zero,myinfo->myaddr.persistent,DEX_channel,msgid,width)) != 0 )
8 years ago
{
8 years ago
//printf("GOT.(%s)\n",jprint(retarray,0));
8 years ago
if ( (retval= basilisk_process_retarray(myinfo,0,InstantDEX_process_channelget,data,sizeof(data),DEX_channel,msgid,retarray,InstantDEX_incoming_func)) > 0 )
{
jaddstr(retjson,"result","success");
} else jaddstr(retjson,"error","cant process InstantDEX retarray");
jadd(retjson,"responses",retarray);
8 years ago
}
else
{
jaddstr(retjson,"error","cant do InstantDEX channelget");
printf("error channelget\n");
}
8 years ago
return(jprint(retjson,1));
8 years ago
}
8 years ago
/*TWO_INTS(InstantDEX,swapstatus,requestid,quoteid)
8 years ago
{
cJSON *vals; char *retstr;
8 years ago
myinfo->DEXactive = (uint32_t)time(NULL) + INSTANTDEX_LOCKTIME;
8 years ago
//if ( myinfo->IAMLP != 0 )
// return(basilisk_respond_swapstatus(myinfo,myinfo->myaddr.persistent,requestid,quoteid));
//else
8 years ago
{
vals = cJSON_CreateObject();
8 years ago
jaddnum(vals,"requestid",(uint32_t)requestid);
jaddnum(vals,"quoteid",(uint32_t)quoteid);
8 years ago
jaddbits256(vals,"hash",myinfo->myaddr.persistent);
8 years ago
retstr = basilisk_standardservice("SWP",myinfo,0,myinfo->myaddr.persistent,vals,"",1);
8 years ago
free_json(vals);
return(retstr);
}
8 years ago
}*/
8 years ago
8 years ago
TWO_INTS(InstantDEX,accept,requestid,quoteid)
8 years ago
{
8 years ago
cJSON *vals; char *retstr;
8 years ago
myinfo->DEXactive = (uint32_t)time(NULL) + INSTANTDEX_LOCKTIME;
8 years ago
if ( myinfo->IAMLP != 0 || myinfo->dexsock >= 0 || myinfo->subsock >= 0 )
8 years ago
return(basilisk_respond_accept(myinfo,myinfo->persistent_priv,requestid,quoteid,&myinfo->DEXaccept));
8 years ago
else
8 years ago
{
vals = cJSON_CreateObject();
8 years ago
jaddnum(vals,"quoteid",(uint32_t)quoteid);
jaddnum(vals,"requestid",(uint32_t)requestid);
8 years ago
retstr = basilisk_standardservice("ACC",myinfo,0,myinfo->myaddr.persistent,vals,"",1);
8 years ago
free_json(vals);
return(retstr);
}
}
#include "../includes/iguana_apiundefs.h"