/****************************************************************************** * Copyright © 2014-2015 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. * * * ******************************************************************************/ #ifndef xcode_trades_h #define xcode_trades_h struct tradehistory { uint64_t assetid,purchased,sold; }; struct tradehistory *_update_tradehistory(struct tradehistory *hist,uint64_t assetid,uint64_t purchased,uint64_t sold) { int32_t i = 0; if ( hist == 0 ) hist = calloc(1,sizeof(*hist)); if ( hist[i].assetid != 0 ) { for (i=0; hist[i].assetid!=0; i++) if ( hist[i].assetid == assetid ) break; } if ( hist[i].assetid == 0 ) { hist = realloc(hist,(i+2) * sizeof(*hist)); memset(&hist[i],0,2 * sizeof(hist[i])); hist[i].assetid = assetid; } if ( hist[i].assetid == assetid ) { hist[i].purchased += purchased; hist[i].sold += sold; printf("hist[%d] %llu +%llu -%llu -> (%llu %llu)\n",i,(long long)hist[i].assetid,(long long)purchased,(long long)sold,(long long)hist[i].purchased,(long long)hist[i].sold); } else printf("_update_tradehistory: impossible case!\n"); return(hist); } struct tradehistory *update_tradehistory(struct tradehistory *hist,uint64_t srcasset,uint64_t srcamount,uint64_t destasset,uint64_t destamount) { hist = _update_tradehistory(hist,srcasset,0,srcamount); hist = _update_tradehistory(hist,destasset,destamount,0); return(hist); } cJSON *_tradehistory_json(struct tradehistory *asset) { cJSON *json = cJSON_CreateObject(); char numstr[64]; sprintf(numstr,"%llu",(long long)asset->assetid), cJSON_AddItemToObject(json,"assetid",cJSON_CreateString(numstr)); sprintf(numstr,"%.8f",dstr(asset->purchased)), cJSON_AddItemToObject(json,"purchased",cJSON_CreateString(numstr)); sprintf(numstr,"%.8f",dstr(asset->sold)), cJSON_AddItemToObject(json,"sold",cJSON_CreateString(numstr)); sprintf(numstr,"%.8f",dstr(asset->purchased) - dstr(asset->sold)), cJSON_AddItemToObject(json,"net",cJSON_CreateString(numstr)); return(json); } cJSON *tradehistory_json(struct tradehistory *hist,cJSON *array) { int32_t i; char assetname[64],numstr[64]; cJSON *assets,*netpos,*item,*json = cJSON_CreateObject(); cJSON_AddItemToObject(json,"rawtrades",array); assets = cJSON_CreateArray(); netpos = cJSON_CreateArray(); for (i=0; hist[i].assetid!=0; i++) { cJSON_AddItemToArray(assets,_tradehistory_json(&hist[i])); item = cJSON_CreateObject(); get_assetname(assetname,hist[i].assetid); cJSON_AddItemToObject(item,"asset",cJSON_CreateString(assetname)); sprintf(numstr,"%.8f",dstr(hist[i].purchased) - dstr(hist[i].sold)), cJSON_AddItemToObject(item,"net",cJSON_CreateString(numstr)); cJSON_AddItemToArray(netpos,item); } cJSON_AddItemToObject(json,"assets",assets); cJSON_AddItemToObject(json,"netpositions",netpos); return(json); } cJSON *tabulate_trade_history(uint64_t mynxt64bits,cJSON *array) { int32_t i,n; cJSON *item; long balancing; struct tradehistory *hist = 0; uint64_t src64bits,srcamount,srcasset,dest64bits,destamount,destasset,jump64bits,jumpamount,jumpasset; //{"requestType":"processjumptrade","NXT":"5277534112615305538","assetA":"5527630","amountA":"6700000000","other":"1510821971811852351","assetB":"12982485703607823902","amountB":"100000000","feeA":"250000000","balancing":0,"feeAtxid":"1234468909119892020","triggerhash":"34ea5aaeeeb62111a825a94c366b4ae3d12bb73f9a3413a27d1b480f6029a73c"} if ( array != 0 && is_cJSON_Array(array) != 0 && (n= cJSON_GetArraySize(array)) > 0 ) { for (i=0; i<n; i++) { item = cJSON_GetArrayItem(array,i); src64bits = get_API_nxt64bits(cJSON_GetObjectItem(item,"NXT")); srcamount = get_API_nxt64bits(cJSON_GetObjectItem(item,"amountA")); srcasset = get_API_nxt64bits(cJSON_GetObjectItem(item,"assetA")); dest64bits = get_API_nxt64bits(cJSON_GetObjectItem(item,"other")); destamount = get_API_nxt64bits(cJSON_GetObjectItem(item,"amountB")); destasset = get_API_nxt64bits(cJSON_GetObjectItem(item,"assetB")); jump64bits = get_API_nxt64bits(cJSON_GetObjectItem(item,"jumper")); jumpamount = get_API_nxt64bits(cJSON_GetObjectItem(item,"jumpasset")); jumpasset = get_API_nxt64bits(cJSON_GetObjectItem(item,"jumpamount")); balancing = (long)get_API_nxt64bits(cJSON_GetObjectItem(item,"balancing")); if ( src64bits != 0 && srcamount != 0 && srcasset != 0 && dest64bits != 0 && destamount != 0 && destasset != 0 ) { if ( src64bits == mynxt64bits ) hist = update_tradehistory(hist,srcasset,srcamount,destasset,destamount); else if ( dest64bits == mynxt64bits ) hist = update_tradehistory(hist,destasset,destamount,srcasset,srcamount); else if ( jump64bits == mynxt64bits ) continue; else printf("illegal tabulate_trade_entry %llu: (%llu -> %llu) via %llu\n",(long long)mynxt64bits,(long long)src64bits,(long long)dest64bits,(long long)jump64bits); } else printf("illegal tabulate_trade_entry %llu: %llu %llu %llu || %llu %llu %llu\n",(long long)mynxt64bits,(long long)src64bits,(long long)srcamount,(long long)srcasset,(long long)dest64bits,(long long)destamount,(long long)destasset); } } if ( hist != 0 ) { array = tradehistory_json(hist,array); free(hist); } return(array); } cJSON *get_tradehistory(char *refNXTaddr,uint32_t timestamp) { char cmdstr[1024],NXTaddr[64],*jsonstr; struct destbuf receiverstr,message,newtriggerhash,triggerhash; cJSON *json,*array,*txobj,*msgobj,*attachment,*retjson = 0,*histarray = 0; int32_t i,j,n,m,duplicates = 0; uint64_t senderbits; if ( timestamp == 0 ) timestamp = 38785003; sprintf(cmdstr,"requestType=getBlockchainTransactions&account=%s×tamp=%u&withMessage=true",refNXTaddr,timestamp); if ( (jsonstr= issue_NXTPOST(cmdstr)) != 0 ) { if ( (json= cJSON_Parse(jsonstr)) != 0 ) { if ( (array= cJSON_GetObjectItem(json,"transactions")) != 0 && is_cJSON_Array(array) != 0 && (n= cJSON_GetArraySize(array)) > 0 ) { for (i=0; i<n; i++) { txobj = cJSON_GetArrayItem(array,i); copy_cJSON(&receiverstr,cJSON_GetObjectItem(txobj,"recipient")); if ( (senderbits = get_API_nxt64bits(cJSON_GetObjectItem(txobj,"sender"))) != 0 ) { expand_nxt64bits(NXTaddr,senderbits); if ( refNXTaddr != 0 && strcmp(NXTaddr,refNXTaddr) == 0 ) { if ( (attachment= cJSON_GetObjectItem(txobj,"attachment")) != 0 && (msgobj= cJSON_GetObjectItem(attachment,"message")) != 0 ) { copy_cJSON(&message,msgobj); //printf("(%s) -> ",message); unstringify(message.buf); if ( (msgobj= cJSON_Parse(message.buf)) != 0 ) { //printf("(%s)\n",message); if ( histarray == 0 ) histarray = cJSON_CreateArray(), j = m = 0; else { copy_cJSON(&newtriggerhash,cJSON_GetObjectItem(msgobj,"triggerhash")); m = cJSON_GetArraySize(histarray); for (j=0; j<m; j++) { copy_cJSON(&triggerhash,cJSON_GetObjectItem(cJSON_GetArrayItem(histarray,j),"triggerhash")); if ( strcmp(triggerhash.buf,newtriggerhash.buf) == 0 ) { duplicates++; break; } } } if ( j == m ) cJSON_AddItemToArray(histarray,msgobj); } else printf("parse error on.(%s)\n",message.buf); } } } } } free_json(json); } free(jsonstr); } if ( histarray != 0 ) retjson = tabulate_trade_history(calc_nxt64bits(refNXTaddr),histarray); printf("duplicates.%d\n",duplicates); return(retjson); } void free_pending(struct pending_trade *pend) { struct InstantDEX_quote *iQ; if ( (iQ= find_iQ(pend->quoteid)) != 0 ) { iQ->s.closed = 1; delete_iQ(pend->quoteid); } else printf("free_pending: cant find pending tx for %llu\n",(long long)pend->quoteid); if ( pend->triggertx != 0 ) free(pend->triggertx); if ( pend->txbytes != 0 ) free(pend->txbytes); if ( pend->tradesjson != 0 ) free_json(pend->tradesjson); free(pend); } /*void oldInstantDEX_history(int32_t action,struct pending_trade *pend,char *str) { uint8_t txbuf[32768]; char *tmpstr; uint16_t n; long len = 0; // struct pending_trade { struct queueitem DL; struct prices777_order order; uint64_t triggertxid,txid,quoteid,orderid; struct prices777 *prices; char *triggertx,*txbytes; cJSON *tradesjson; double price,volume; uint32_t timestamp; int32_t dir,type; }; memcpy(&txbuf[len],&action,sizeof(action)), len += sizeof(action); if ( action == 0 ) { memcpy(&txbuf[len],pend,sizeof(*pend)), len += sizeof(*pend); if ( pend->triggertx != 0 ) { n = (uint16_t)strlen(pend->triggertx) + 1; memcpy(&txbuf[len],&n,sizeof(n)), len += sizeof(n); memcpy(&txbuf[len],pend->triggertx,n), len += n; } if ( pend->txbytes != 0 ) { n = (uint16_t)strlen(pend->txbytes) + 1; memcpy(&txbuf[len],&n,sizeof(n)), len += sizeof(n); memcpy(&txbuf[len],pend->txbytes,n), len += n; } if ( pend->tradesjson != 0 ) { tmpstr = jprint(pend->tradesjson,0); n = (uint16_t)strlen(tmpstr) + 1; memcpy(&txbuf[len],&n,sizeof(n)), len += sizeof(n); memcpy(&txbuf[len],tmpstr,n), len += n; free(tmpstr); } } else { memcpy(&txbuf[len],&pend->orderid,sizeof(pend->orderid)), len += sizeof(pend->orderid); memcpy(&txbuf[len],&pend->quoteid,sizeof(pend->quoteid)), len += sizeof(pend->quoteid); } if ( str != 0 ) { n = (uint16_t)strlen(str) + 1; memcpy(&txbuf[len],&n,sizeof(n)), len += sizeof(n); memcpy(&txbuf[len],str,n), len += n; } else { n = 0; memcpy(&txbuf[len],&n,sizeof(n)), len += sizeof(n); } txind777_create(INSTANTDEX.history,INSTANTDEX.numhist,pend->timestamp,txbuf,len); txinds777_flush(INSTANTDEX.history,INSTANTDEX.numhist,pend->timestamp); INSTANTDEX.numhist++; }*/ char *InstantDEX_loadhistory(struct pending_trade *pend,int32_t *actionp,uint8_t *txbuf,int32_t size) { char *tmpstr,*str = 0; uint16_t n; long len = 0; memcpy(actionp,&txbuf[len],sizeof(*actionp)), len += sizeof(*actionp); if ( *actionp == 0 ) { memcpy(pend,&txbuf[len],sizeof(*pend)), len += sizeof(*pend); //printf("pendsize.%ld trigger.%p tx.%p json.%p\n",(long)sizeof(*pend),pend->triggertx,pend->txbytes,pend->tradesjson); if ( pend->triggertx != 0 ) { memcpy(&n,&txbuf[len],sizeof(n)), len += sizeof(n); pend->triggertx = calloc(1,n); memcpy(pend->triggertx,&txbuf[len],n), len += n; } if ( pend->txbytes != 0 ) { memcpy(&n,&txbuf[len],sizeof(n)), len += sizeof(n); pend->txbytes = calloc(1,n); memcpy(pend->txbytes,&txbuf[len],n), len += n; } if ( pend->tradesjson != 0 ) { memcpy(&n,&txbuf[len],sizeof(n)), len += sizeof(n); tmpstr = calloc(1,n); memcpy(tmpstr,&txbuf[len],n), len += n; if ( (pend->tradesjson= cJSON_Parse(tmpstr)) == 0 ) printf("cant parse.(%s)\n",tmpstr); free(tmpstr); } } else { memcpy(&pend->orderid,&txbuf[len],sizeof(pend->orderid)), len += sizeof(pend->orderid); memcpy(&pend->quoteid,&txbuf[len],sizeof(pend->quoteid)), len += sizeof(pend->quoteid); } memcpy(&n,&txbuf[len],sizeof(n)), len += sizeof(n); if ( n != 0 ) { str = calloc(1,n); memcpy(str,&txbuf[len],n), len += n; } if ( len != size ) printf("loadhistory warning: len.%ld != size.%d\n",len,size); return(str); } struct pending_trade *InstantDEX_historyi(int32_t *actionp,char **strp,int32_t i,uint8_t *txbuf,int32_t maxsize) { struct pending_trade *pend = 0; /* void *ptr; int32_t size; *strp = 0; txinds777_seek(INSTANTDEX.history,i); if ( (ptr= txinds777_read(&size,txbuf,INSTANTDEX.history)) == 0 || size <= 0 || size > maxsize ) { printf("InstantDEX_inithistory: error reading entry.%d | ptr.%p size.%d\n",i,ptr,maxsize); return(0); } pend = calloc(1,sizeof(*pend)); *strp = InstantDEX_loadhistory(pend,actionp,ptr,size);*/ return(pend); } int32_t oldInstantDEX_inithistory(int32_t firsti,int32_t endi) { int32_t i,action; uint8_t txbuf[32768]; char *str; struct pending_trade *pend; printf("InstantDEX_inithistory firsti.%d endi.%d\n",firsti,endi); for (i=firsti; i<endi; i++) { if ( (pend= InstantDEX_historyi(&action,&str,i,txbuf,sizeof(txbuf))) != 0 ) { printf("type.%d (%c) action.%d orderid.%llu quoteid.%llu (%s)\n",pend->type,pend->type!=0?pend->type:'0',action,(long long)pend->orderid,(long long)pend->quoteid,str!=0?str:""); if ( str != 0 ) free(str); free_pending(pend); } } return(i); } cJSON *InstantDEX_tradeitem(struct pending_trade *pend) { // struct pending_trade { struct queueitem DL; struct prices777_order order; uint64_t triggertxid,txid,quoteid,orderid; struct prices777 *prices; char *triggertx,*txbytes; cJSON *tradesjson; double price,volume; uint32_t timestamp; int32_t dir,type; }; struct InstantDEX_quote *iQ; char str[64]; cJSON *json = cJSON_CreateObject(); str[0] = (pend->type == 0) ? '0' : pend->type; str[1] = 0; jaddstr(json,"type",str); jaddnum(json,"timestamp",pend->timestamp); jadd64bits(json,"orderid",pend->orderid), jadd64bits(json,"quoteid",pend->quoteid); if ( (iQ= find_iQ(pend->quoteid)) != 0 ) { if ( iQ->s.baseid != 0 && iQ->s.relid != 0 ) jadd64bits(json,"baseid",iQ->s.baseid), jadd64bits(json,"relid",iQ->s.relid); if ( iQ->s.baseamount != 0 && iQ->s.relamount != 0 ) jaddnum(json,"baseqty",iQ->s.baseamount), jaddnum(json,"relqty",iQ->s.relamount); } else printf("tradeitem cant find quoteid.%llu\n",(long long)pend->quoteid); if ( pend->dir != 0 ) jaddnum(json,"dir",pend->dir); if ( pend->price > SMALLVAL && pend->volume > SMALLVAL ) jaddnum(json,"price",pend->price), jaddnum(json,"volume",pend->volume); if ( pend->triggertxid != 0 ) jadd64bits(json,"triggertxid",pend->triggertxid); if ( pend->txid != 0 ) jadd64bits(json,"txid",pend->txid); if ( pend->triggertx != 0 ) jaddstr(json,"triggertx",pend->triggertx); if ( pend->txbytes != 0 ) jaddstr(json,"txbytes",pend->txbytes); return(json); } char *InstantDEX_withdraw(cJSON *argjson) { char *exchangestr,*str; struct exchange_info *exchange; int32_t exchangeid; if ( (exchangestr= jstr(argjson,"exchange")) != 0 && (exchange= find_exchange(&exchangeid,exchangestr)) != 0 ) { if ( exchange->issue.withdraw != 0 ) { if ( (str= (*exchange->issue.withdraw)(&exchange->cHandle,exchange,argjson)) == 0 ) str = clonestr("{\"result\":\"nothing returned from exchange\"}"); return(str); } else return(clonestr("{\"error\":\"no withdraw function\"}")); } return(clonestr("{\"error\":\"withdraw is not yet\"}")); } char *InstantDEX_tradehistory(cJSON *argjson,int32_t firsti,int32_t endi) { /* cJSON *json,*array,*item,*tmp; int32_t exchangeid,i,action; uint8_t txbuf[32768]; char *str,*exchangestr; struct pending_trade *pend; struct exchange_info *exchange; if ( (exchangestr= jstr(argjson,"exchange")) != 0 && (exchange= find_exchange(&exchangeid,exchangestr)) != 0 ) { if ( exchange->issue.tradehistory != 0 ) { if ( (str= (*exchange->issue.tradehistory)(&exchange->cHandle,exchange,argjson)) == 0 ) str = clonestr("{\"result\":\"nothing returned from exchange\"}"); return(str); } else return(clonestr("{\"error\":\"no tradehistory function\"}")); } json = cJSON_CreateObject(); array = cJSON_CreateArray(); if ( endi == 0 ) endi = INSTANTDEX.numhist-1; if ( endi < firsti ) endi = firsti; for (i=firsti; i<=endi; i++) { if ( (pend= InstantDEX_historyi(&action,&str,i,txbuf,sizeof(txbuf))) != 0 ) { item = cJSON_CreateObject(); jaddnum(item,"i",i); jaddnum(item,"action",action); jadd(item,"trade",InstantDEX_tradeitem(pend)); if ( pend->tradesjson != 0 ) jadd(item,"trades",cJSON_Duplicate(pend->tradesjson,1)); if ( str != 0 ) { if ( (tmp= cJSON_Parse(str)) != 0 ) jadd(item,"str",tmp); free(str); } free_pending(pend); jaddi(array,item); } } jadd(json,"tradehistory",array); jaddnum(json,"numentries",INSTANTDEX.numhist); return(jprint(json,1));*/ return(0); } int32_t substr128(char *dest,char *src) { char zeroes[129],*match; int32_t i; for (i=0; i<128; i++) zeroes[i] = '0'; zeroes[i] = 0; strcpy(dest,src); if ( (match= strstr(dest,zeroes)) != 0 ) { strcpy(match,"Z"); for (i=0; match[128+i]!=0; i++) match[i+1] = match[128+i]; match[i+1] = 0; } //printf("substr128.(%s) -> (%s)\n",src,dest); return(0); } uint64_t gen_NXTtx(struct NXTtx *tx,uint64_t dest64bits,uint64_t assetidbits,uint64_t qty,uint64_t orderid,uint64_t quoteid,int32_t deadline,char *reftx,char *phaselink,uint32_t finishheight,char *phasesecret) { char secret[8192],cmd[16384],destNXTaddr[64],assetidstr[64],hexstr[64],*retstr; uint8_t msgbuf[17]; cJSON *json; int32_t len; uint64_t phasecost = 0; if ( deadline > 1000 ) deadline = 1000; expand_nxt64bits(destNXTaddr,dest64bits); memset(tx,0,sizeof(*tx)); if ( ((phasesecret != 0 && phasesecret[0] != 0) || (phaselink!= 0 && phaselink[0] != 0)) && finishheight <= _get_NXTheight(0) ) { printf("finish height.%u must be in the future.%u\n",finishheight,_get_NXTheight(0)); return(0); } if ( phaselink != 0 || phasesecret != 0 ) phasecost = MIN_NQTFEE; cmd[0] = 0; if ( assetidbits == NXT_ASSETID ) sprintf(cmd,"requestType=sendMoney&amountNQT=%lld",(long long)qty); else { expand_nxt64bits(assetidstr,assetidbits); if ( is_mscoin(assetidstr) == 0 ) sprintf(cmd,"requestType=transferAsset&asset=%s&quantityQNT=%lld",assetidstr,(long long)qty); else sprintf(cmd,"requestType=transferCurrency¤cy=%s&units=%lld",assetidstr,(long long)qty); } if ( quoteid != 0 ) { len = 0; printf("serialize buffer\n"); //len = txind777_txbuf(msgbuf,len,orderid,sizeof(orderid)); //len = txind777_txbuf(msgbuf,len,quoteid,sizeof(quoteid)); init_hexbytes_noT(hexstr,msgbuf,len); sprintf(cmd+strlen(cmd),"&messageIsText=true&message=%s",hexstr); } if ( cmd[0] != 0 ) { escape_code(secret,IGUANA_NXTACCTSECRET); sprintf(cmd+strlen(cmd),"&deadline=%u&feeNQT=%lld&secretPhrase=%s&recipient=%s&broadcast=false",deadline,(long long)MIN_NQTFEE+phasecost,secret,destNXTaddr); if ( reftx != 0 && reftx[0] != 0 ) sprintf(cmd+strlen(cmd),"&referencedTransactionFullHash=%s",reftx); if ( phaselink != 0 && phaselink[0] != 0 ) sprintf(cmd+strlen(cmd),"&phased=true&phasingFinishHeight=%u&phasingVotingModel=4&phasingQuorum=1&phasingLinkedFullHash=%s",finishheight,phaselink); else if ( phasesecret != 0 && phasesecret[0] != 0 ) sprintf(cmd+strlen(cmd),"&phased=true&phasingFinishHeight=%u&phasingVotingModel=5&phasingHashedSecretAlgorithm=62&phasingQuorum=1&phasingHashedSecret=%s",finishheight,phasesecret); //printf("generated cmd.(%s)\n",cmd); if ( (retstr= issue_NXTPOST(cmd)) != 0 ) { //printf("(%s)\n",retstr); if ( (json= cJSON_Parse(retstr)) != 0 ) { if ( extract_cJSON_str(tx->txbytes,MAX_JSON_FIELD,json,"transactionBytes") > 0 && extract_cJSON_str(tx->utxbytes,MAX_JSON_FIELD,json,"unsignedTransactionBytes") > 0 && extract_cJSON_str(tx->fullhash,MAX_JSON_FIELD,json,"fullHash") > 0 && extract_cJSON_str(tx->sighash,MAX_JSON_FIELD,json,"signatureHash") > 0 ) { tx->txid = j64bits(json,"transaction"); substr128(tx->utxbytes2,tx->utxbytes); } free_json(json); } free(retstr); } } return(tx->txid); } struct NXTtx *fee_triggerhash(char *triggerhash,uint64_t orderid,uint64_t quoteid,int32_t deadline) { static struct NXTtx fee; if ( fee.fullhash[0] == 0 ) gen_NXTtx(&fee,calc_nxt64bits(INSTANTDEX_ACCT),NXT_ASSETID,INSTANTDEX_FEE,orderid,quoteid,deadline,0,0,0,0); strcpy(triggerhash,fee.fullhash); return(&fee); } uint64_t InstantDEX_swapstr(char *sendphased,char *phasesecret,uint64_t *txidp,char *triggertx,char *txbytes,char *swapstr,uint64_t orderid,struct prices777_order *order,char *triggerhash,char *phaselink,int32_t finishheight) { struct NXTtx fee,sendtx; uint64_t otherqty = 0,otherassetbits = 0,assetidbits = 0,qty = 0; int32_t deadline = INSTANTDEX_TRIGGERDEADLINE; if ( finishheight != 0 ) { if ( finishheight > FINISH_HEIGHT ) deadline *= (finishheight / FINISH_HEIGHT); finishheight += _get_NXTheight(0); } swapstr[0] = triggertx[0] = txbytes[0] = 0; *txidp = 0; gen_NXTtx(&fee,calc_nxt64bits(INSTANTDEX_ACCT),NXT_ASSETID,INSTANTDEX_FEE,orderid,order->s.quoteid,deadline,triggerhash,0,0,0); strcpy(triggertx,fee.txbytes); if ( order->s.baseamount < 0 ) assetidbits = order->s.baseid, qty = -order->s.baseamount, otherassetbits = order->s.relid, otherqty = order->s.relamount; else if ( order->s.relamount < 0 ) assetidbits = order->s.relid, qty = -order->s.relamount, otherassetbits = order->s.baseid, otherqty = order->s.baseamount; printf("genNXTtx.(%llu/%llu) finish at %u vs %u lag %u deadline %d assetidbits.%llu sendphased.(%s)\n",(long long)orderid,(long long)order->s.quoteid,finishheight,_get_NXTheight(0),finishheight-_get_NXTheight(0),deadline,(long long)assetidbits,sendphased!=0?sendphased:""); if ( sendphased != 0 && assetidbits != 0 && qty != 0 ) { if ( triggerhash == 0 || triggerhash[0] == 0 ) triggerhash = fee.fullhash; gen_NXTtx(&sendtx,order->s.offerNXT,assetidbits,qty,orderid,order->s.quoteid,deadline,triggerhash,phaselink,finishheight,phasesecret); *txidp = sendtx.txid; strcpy(txbytes,sendtx.txbytes); sprintf(swapstr,",\"F\":\"%u\",\"T\":\"%s\",\"FH\":\"%s\",\"U\":\"%s\",\"S\":\"%s\",\"a\":\"%llu\",\"q\":\"%llu\"}",finishheight,fee.fullhash,sendtx.fullhash,sendtx.utxbytes2,sendtx.sighash,(long long)otherassetbits,(long long)otherqty); } else sprintf(swapstr,",\"F\":\"%u\",\"T\":\"%s\",\"a\":\"%llu\",\"q\":\"%llu\"}",finishheight,fee.fullhash,(long long)otherassetbits,(long long)otherqty); return(fee.txid); } uint64_t prices777_swapbuf(char *sendphased,char *phasesecret,uint64_t *txidp,char *triggertx,char *txbytes,char *swapbuf,char *exchangestr,char *base,char *rel,struct prices777_order *order,uint64_t orderid,int32_t finishoffset,char *triggerhash) { char swapstr[4096],*str; uint64_t txid = 0; *txidp = 0; if ( strcmp(exchangestr,"wallet") == 0 ) str = "swap"; else { str = order->wt > 0. ? "buy" : (order->wt < 0. ? "sell" : "swap"); //printf("not wallet!\n"); getchar(); } if ( finishoffset == 0 ) finishoffset = FINISH_HEIGHT; sprintf(swapbuf,"{\"orderid\":\"%llu\",\"quoteid\":\"%llu\",\"offerNXT\":\"%llu\",\"fillNXT\":\"%s\",\"plugin\":\"relay\",\"destplugin\":\"InstantDEX\",\"method\":\"busdata\",\"submethod\":\"%s\",\"exchange\":\"%s\",\"base\":\"%s\",\"rel\":\"%s\",\"baseid\":\"%llu\",\"relid\":\"%llu\",\"baseqty\":\"%lld\",\"relqty\":\"%lld\"}",(long long)orderid,(long long)order->s.quoteid,(long long)order->s.offerNXT,IGUANA_NXTADDR,str,exchangestr,base,rel,(long long)order->s.baseid,(long long)order->s.relid,(long long)order->s.baseamount,(long long)order->s.relamount); if ( order->s.price > SMALLVAL ) sprintf(swapbuf + strlen(swapbuf) - 1,",\"price\":%.8f,\"volume\":%.8f}",order->s.price,order->s.vol); txid = InstantDEX_swapstr(sendphased,phasesecret,txidp,triggertx,txbytes,swapstr,orderid,order,triggerhash,0,finishoffset); strcpy(swapbuf+strlen(swapbuf)-1,swapstr); //printf("swapbuf.(%s)\n",swapbuf); return(txid); } char *prices777_finishswap(int32_t dotrade,int32_t type,struct pending_trade *pend,char *swapbuf,char *triggertx,char *txbytes) { uint32_t nonce; char *str; if ( triggertx[0] != 0 ) pend->triggertx = clonestr(triggertx); if ( txbytes[0] != 0 ) pend->txbytes = clonestr(txbytes); pend->order.s.swap = 1; pend->tradesjson = cJSON_Parse(swapbuf); pend->type = type; printf("quoteid.%llu and pending.%d\n",(long long)pend->order.s.quoteid,pend->order.s.pending); if ( dotrade != 0 ) { if ( (str= busdata_sync(&nonce,swapbuf,"allnodes",0)) != 0 ) free(str); pend->queueflag = 1; queue_enqueue("PendingQ",&Pending_offersQ,&pend->DL,0); } //InstantDEX_history(0,pend,swapbuf); return(clonestr(swapbuf)); } /*int32_t subatomic_pubkeyhash(char *pubkeystr,char *pkhash,struct coin777 *coin,uint64_t quoteid) { printf("subatomic pubkeyhash not yet\n"); char tmpswapaddr[128],swapacct[128]; uint8_t tmpbuf[128]; struct destbuf pubkey; sprintf(swapacct,"atomic.%llu",(long long)quoteid); pkhash[0] = pubkeystr[0] = 0; if ( get_acct_coinaddr(tmpswapaddr,coin->name,coin->serverport,coin->userpass,swapacct) != 0 ) { get_pubkey(&pubkey,coin->name,coin->serverport,coin->userpass,tmpswapaddr); strcpy(pubkeystr,pubkey.buf); calc_OP_HASH160(pkhash,tmpbuf,pubkey.buf); return(0); } return(-1); }*/ int32_t complete_swap(struct InstantDEX_quote *iQ,uint64_t orderid,uint64_t quoteid,int32_t err) { /*int32_t errcode=-1,errcode2=-2; char *txstr,*txstr2; int32_t iter; struct pending_trade *pend; for (iter=0; iter<2; iter++) { while ( (pend= queue_dequeue(&Pending_offersQ.pingpong[iter],0)) != 0 ) { if ( pend->quoteid == quoteid ) { if ( err == 0 && issue_broadcastTransaction(&errcode2,&txstr2,pend->txbytes,IGUANA_NXTACCTSECRET) == pend->txid && errcode2 == 0 ) { if ( err == 0 && (issue_broadcastTransaction(&errcode,&txstr,pend->triggertx,IGUANA_NXTACCTSECRET) != pend->triggertxid || errcode != 0) ) err = -13; } if ( err == 0 && errcode == 0 && errcode2 == 0 ) { iQ->s.matched = 1; //InstantDEX_history(1,pend,0); } //else InstantDEX_history(-1,pend,0); printf("errs.(%d %d %d) COMPLETED %llu/%llu %d %f %f with txids %llu %llu\n",err,errcode,errcode2,(long long)pend->orderid,(long long)pend->quoteid,pend->dir,pend->price,pend->volume,(long long)pend->triggertxid,(long long)pend->txid); pend->queueflag = 1; pend->finishtime = (uint32_t)time(NULL); return(1); } queue_enqueue("requeue",&Pending_offersQ.pingpong[iter ^ 1],&pend->DL,0); } }*/ printf("complete swap is notyet\n"); return(-1); } char *prices777_tradewallet(struct pending_trade *pend) { printf("tradewallet is not yet\n"); return(0); /* struct coin777 *recvcoin,*sendcoin; cJSON *walletitem,*item; char fieldA[64],fieldB[64],triggertx[4096],txbytes[4096],fieldpkhash[64],refredeemscript[2048],scriptPubKey[128],p2shaddr[64]; char swapbuf[8192],buf[1024],*rpubA=0,*rpubB=0,*rpkhash=0,*spubA=0,*spubB=0,*spkhash=0,*recvstr=0; char *sendstr=0,*refundtx,*redeemscript,*str; int32_t finishin,deadline; uint32_t nonce; uint64_t sendamount,recvamount,sendasset,recvasset; struct destbuf base,rel; if ( pend->item != 0 && (item= jitem(pend->item,0)) != 0 && (walletitem= jobj(item,"wallet")) != 0 ) { finishin = (pend->extra[0] == 0) ? 200 : myatoi(pend->extra,10000); if ( finishin < FINISH_HEIGHT ) finishin = FINISH_HEIGHT; copy_cJSON(&base,jobj(item,"base")); copy_cJSON(&rel,jobj(item,"rel")); if ( (recvamount= j64bits(item,"recvbase")) != 0 && (sendamount= j64bits(item,"sendrel")) != 0 ) recvstr = base.buf, sendstr = rel.buf, recvasset = pend->order.s.baseid, sendasset = pend->order.s.relid; else if ( (recvamount= j64bits(item,"recvrel")) != 0 && (sendamount= j64bits(item,"sendbase")) != 0 ) recvstr = rel.buf, sendstr = base.buf, recvasset = pend->order.s.relid, sendasset = pend->order.s.baseid; else { return(clonestr("{\"error\":\"need recvbase/sendrel or recvrel/sendbase\"}\n")); } recvcoin = coin777_find(recvstr,1), sendcoin = coin777_find(sendstr,1); // placeask -> recvbase/sendrel, placebid -> sendbase/recvrel, it is relative to the one that placed quote if ( strcmp(recvstr,"NXT") != 0 ) // placeask COIN/NXT or placebid NXT/COIN { if ( recvamount < recvcoin->mgw.txfee ) { printf("recvamount %.8f < txfee %.8f\n",dstr(recvamount),dstr(recvcoin->mgw.txfee)); return(clonestr("{\"error\":\"amount too small\"}\n")); } sprintf(fieldA,"%spubA",recvstr), rpubA = jstr(walletitem,fieldA); sprintf(fieldB,"%spubB",recvstr), rpubB = jstr(walletitem,fieldB); sprintf(fieldpkhash,"%spkhash",recvstr), rpkhash = jstr(walletitem,fieldpkhash); if ( rpubA[0] != 0 && rpubB != 0 && rpkhash != 0 ) // Alice for recvcoin -> Bob, Bob sends NXT -> Alice { if ( recvcoin->funding.signedtransaction[0] == 0 && (refundtx= subatomic_fundingtx(refredeemscript,&recvcoin->funding,recvcoin,rpubA,rpubB,rpkhash,recvamount,finishin)) != 0 ) { deadline = 3600; gen_NXTtx(&recvcoin->trigger,calc_nxt64bits(INSTANTDEX_ACCT),NXT_ASSETID,INSTANTDEX_FEE,pend->orderid,pend->order.s.quoteid,deadline,0,0,0,0); sprintf(swapbuf,"{\"orderid\":\"%llu\",\"quoteid\":\"%llu\",\"offerNXT\":\"%llu\",\"fillNXT\":\"%s\",\"plugin\":\"relay\",\"destplugin\":\"InstantDEX\",\"method\":\"busdata\",\"submethod\":\"swap\",\"exchange\":\"wallet\",\"recvamount\":\"%lld\",\"rtx\":\"%s\",\"rs\":\"%s\",\"recvcoin\":\"%s\",\"%s\":\"%s\",\"%s\":\"%s\",\"%s\":\"%s\",\"trigger\":\"%s\",\"sendasset\":\"%llu\",\"sendqty\":\"%llu\",\"base\":\"%s\",\"rel\":\"%s\"}",(long long)pend->orderid,(long long)pend->order.s.quoteid,(long long)pend->order.s.offerNXT,SUPERNET.NXTADDR,(long long)recvamount,refundtx,refredeemscript,recvstr,fieldA,rpubA,fieldB,rpubB,fieldpkhash,rpkhash,recvcoin->trigger.fullhash,(long long)sendasset,(long long)sendamount,pend->prices->base,pend->prices->rel); recvcoin->refundtx = refundtx; pend->order.s.swap = 1; if ( pend->dotrade != 0 && (str= busdata_sync(&nonce,swapbuf,"allnodes",0)) != 0 ) { pend->queueflag = 1; queue_enqueue("PendingQ",&Pending_offersQ.pingpong[0],&pend->DL,0); } return(clonestr(swapbuf)); } else return(clonestr("{\"error\":\"cant create refundtx, maybe already pending\"}\n")); } else { sprintf(buf,"{\"error\":\"sendNXT recvstr.(%s) rpubA.(%s) without %s rpubB.%p or %s rpkhash.%p\"}\n",recvstr,rpubA,fieldB,rpubB,fieldpkhash,rpkhash); return(clonestr(buf)); } } else if ( strcmp(sendstr,"NXT") != 0 ) { if ( sendamount < sendcoin->mgw.txfee ) { printf("sendamount %.8f < txfee %.8f\n",dstr(sendamount),dstr(sendcoin->mgw.txfee)); return(clonestr("{\"error\":\"amount too small\"}\n")); } sprintf(fieldA,"%spubA",sendstr), spubA = jstr(walletitem,fieldA); sprintf(fieldB,"%spubB",sendstr), spubB = jstr(walletitem,fieldB); sprintf(fieldpkhash,"%spkhash",sendstr), spkhash = jstr(walletitem,fieldpkhash); if ( spubA != 0 && spubB != 0 && spkhash[0] != 0 ) // Bob <- sendcoin from Alice, send NXT -> Alice { if ( (redeemscript= create_atomictx_scripts(sendcoin->p2shtype,scriptPubKey,p2shaddr,spubA,spubB,spkhash)) != 0 ) { pend->triggertxid = prices777_swapbuf("yes",spkhash,&pend->txid,triggertx,txbytes,swapbuf,"wallet",pend->prices->base,pend->prices->rel,&pend->order,pend->orderid,finishin,0); sprintf(swapbuf+strlen(swapbuf)-1,",\"sendcoin\":\"%s\",\"sendamount\":\"%llu\",\"%s\":\"%s\",\"%s\":\"%s\",\"%s\":\"%s\",\"recvasset\":\"%llu\",\"recvqty\":\"%llu\"}",sendstr,(long long)sendamount,fieldA,spubA,fieldB,spubB,fieldpkhash,spkhash,(long long)recvasset,(long long)recvamount); free(redeemscript); pend->order.s.swap = 1; if ( pend->dotrade != 0 && (str= busdata_sync(&nonce,swapbuf,"allnodes",0)) != 0 ) { free(str); pend->queueflag = 1; queue_enqueue("PendingQ",&Pending_offersQ.pingpong[0],&pend->DL,0); } return(clonestr(swapbuf)); } } else { sprintf(buf,"{\"error\":\"recvNXT sendstr.(%s) spubA.(%s) without %s spubB.(%s) or %s spkhash.(%s)\"}\n",sendstr,spubA,fieldB,spubB,fieldpkhash,spkhash); return(clonestr(buf)); } } else if ( rpubA[0] != 0 && rpubB != 0 && rpkhash != 0 && spubA != 0 && spubB != 0 && spkhash[0] != 0 && (strcmp(sendstr,"BTC") == 0 || strcmp(recvstr,"BTC") == 0) ) { if ( recvcoin->funding.signedtransaction[0] == 0 && (refundtx= subatomic_fundingtx(refredeemscript,&recvcoin->funding,recvcoin,rpubA,rpubB,rpkhash,recvamount,finishin)) != 0 ) { if ( (redeemscript= create_atomictx_scripts(sendcoin->p2shtype,scriptPubKey,p2shaddr,spubA,spubB,spkhash)) != 0 ) { pend->triggertxid = prices777_swapbuf(0,0,&pend->txid,triggertx,txbytes,swapbuf,"wallet",pend->prices->base,pend->prices->rel,&pend->order,pend->orderid,finishin,0); sprintf(swapbuf,"{\"orderid\":\"%llu\",\"quoteid\":\"%llu\",\"offerNXT\":\"%llu\",\"fillNXT\":\"%s\",\"plugin\":\"relay\",\"destplugin\":\"InstantDEX\",\"method\":\"busdata\",\"submethod\":\"swap\",\"exchange\":\"wallet\",\"sendcoin\":\"%s\",\"recvcoin\":\"%s\",\"sendamount\":\"%lld\",\"recvamount\":\"%lld\",\"base\":\"%s\",\"rel\":\"%s\"}",(long long)pend->orderid,(long long)pend->order.s.quoteid,(long long)pend->order.s.offerNXT,SUPERNET.NXTADDR,sendstr,recvstr,(long long)sendamount,(long long)recvamount,pend->prices->base,pend->prices->rel); sprintf(swapbuf+strlen(swapbuf)-1,",\"rtx\":\"%s\",\"rs\":\"%s\",\"rpubA\":\"%s\",\"rpubB\":\"%s\",\"rpkhash\":\"%s\",\"pubA\":\"%s\",\"pubB\":\"%s\",\"pkhash\":\"%s\"}",refundtx,refredeemscript,rpubA,rpubB,rpkhash,spubA,spubB,spkhash); free(redeemscript); free(refundtx); pend->order.s.swap = 1; if ( pend->dotrade != 0 && (str= busdata_sync(&nonce,swapbuf,"allnodes",0)) != 0 ) { free(str); pend->queueflag = 1; queue_enqueue("PendingQ",&Pending_offersQ.pingpong[0],&pend->DL,0); } return(clonestr(swapbuf)); } free(refundtx); } else return(clonestr("{\"error\":\"cant create refundtx, maybe already pending\"}\n")); } else return(clonestr("{\"error\":\"one of wallets must be NXT or BTC\"}\n")); printf("wallet swap finishin.%d trigger.%llu swapbuf.(%s)\n",finishin,(long long)pend->triggertxid,swapbuf); return(prices777_finishswap(pend->dotrade,'A',pend,swapbuf,triggertx,txbytes)); } else return(clonestr("{\"error\":\"need to have trades[] json item\"}\n"));*/ } struct pending_trade *prices777_createpending(int32_t *curlingp,void *bot,void **cHandlep,int32_t dotrade,cJSON *item,char *activenxt,char *secret,struct prices777 *prices,int32_t dir,double price,double volume,struct InstantDEX_quote *iQ,struct prices777_order *order,uint64_t orderid,char *extra) { struct InstantDEX_quote _iQ; struct exchange_info *exchange; struct pending_trade *pend; char swapbuf[8192],triggertx[4096],txbytes[4096]; if ( (exchange= find_exchange(0,prices->exchange)) == 0 && exchange->issue.trade != 0 ) { printf("prices777_trade: need to have supported exchange\n"); return(0); } if ( cHandlep == 0 ) cHandlep = &exchange->cHandle; if ( iQ == 0 && order == 0 ) { printf("prices777_trade: need to have either iQ or order\n"); return(0); } else if ( iQ == 0 && (iQ= find_iQ(order->s.quoteid)) == 0 ) { iQ = &_iQ; memset(&_iQ,0,sizeof(_iQ)); iQ->s = order->s; iQ->exchangeid = prices->exchangeid; if ( iQ->s.timestamp == 0 ) iQ->s.timestamp = (uint32_t)time(NULL); iQ = create_iQ(iQ,0); } else iQ = create_iQ(iQ,0); pend = calloc(1,sizeof(*pend)); pend->bot = bot; safecopy((char *)pend->nxtsecret,secret,sizeof(pend->nxtsecret)); pend->size = (int32_t)sizeof(*pend); pend->my64bits = calc_nxt64bits(activenxt); triggertx[0] = txbytes[0] = swapbuf[0] = 0; pend->prices = prices, pend->dir = dir, pend->price = price, pend->volume = volume, pend->orderid = orderid; iQ->s.pending = 1; pend->curlingp = curlingp; pend->quoteid = iQ->s.quoteid; if ( order != 0 ) pend->order = *order; else pend->order.s = iQ->s; pend->timestamp = (uint32_t)time(NULL); pend->expiration = pend->timestamp + 60; pend->cHandlep = cHandlep; pend->dotrade = dotrade; pend->item = item; pend->exchange = exchange; safecopy(pend->extra,extra,sizeof(pend->extra)); return(pend); } char *prices777_issuepending(struct pending_trade *pend) { char swapbuf[8192],triggertx[4096],txbytes[4096],*retstr; struct prices777 *prices; struct exchange_info *exchange; if ( (prices= pend->prices) == 0 || (exchange= pend->exchange) == 0 ) retstr = clonestr("{\"error\":\"no prices ptr\"}"); else if ( strcmp(prices->exchange,"wallet") == 0 ) retstr = prices777_tradewallet(pend); else if ( strcmp(prices->exchange,INSTANTDEX_NAME) == 0 ) { pend->expiration = pend->timestamp + INSTANTDEX_TRIGGERDEADLINE*60; pend->triggertxid = prices777_swapbuf("yes",0,&pend->txid,triggertx,txbytes,swapbuf,prices->exchange,prices->base,prices->rel,&pend->order,pend->orderid,myatoi(pend->extra,10000),0); retstr = prices777_finishswap(pend->dotrade,'T',pend,swapbuf,triggertx,txbytes); } else if ( strcmp(prices->exchange,"nxtae") == 0 ) { pend->type = 'N'; retstr = fill_nxtae(pend->dotrade,&pend->txid,pend->my64bits,(char *)pend->nxtsecret,pend->dir,pend->price,pend->volume,prices->baseid,prices->relid); if ( pend->dotrade != 0 ) { pend->queueflag = 1; queue_enqueue("PendingQ",&Pending_offersQ,&pend->DL,0); } } else { if ( exchange->issue.trade != 0 ) { printf(" issue dir.%d %s/%s price %f vol %f -> %s\n",pend->dir,prices->base,prices->rel,pend->price,pend->volume,prices->exchange); retstr = pend->extra; if ( pend->curlingp != 0 ) *pend->curlingp = 1; if ( (pend->txid= (*exchange->issue.trade)(pend->cHandlep,pend->dotrade,&retstr,exchange,prices->base,prices->rel,pend->dir,pend->price,pend->volume)) != 0 ) { pend->queueflag = 1; pend->finishtime = (uint32_t)time(NULL); } else printf("no txid from trade\n"); if ( pend->curlingp != 0 ) *pend->curlingp = 0; if ( retstr != 0 ) { if ( pend->dotrade != 0 ) { pend->queueflag = 1; queue_enqueue("PendingQ",&Pending_offersQ,&pend->DL,0); } printf("returning.%p (%s)\n",retstr,retstr); } } else retstr = clonestr("{\"error\":\"no trade function for exchange\"}\n"); } if ( retstr == 0 ) retstr = clonestr("{\"error\":\"no response\"}"); return(retstr); } char *prices777_trade(int32_t *curlingp,void *bot,struct pending_trade **pendp,void **cHandlep,int32_t dotrade,cJSON *item,char *activenxt,char *secret,struct prices777 *prices,int32_t dir,double price,double volume,struct InstantDEX_quote *iQ,struct prices777_order *order,uint64_t orderid,char *extra) { struct pending_trade *pend; char *retstr; if ( pendp != 0 ) *pendp = 0; if ( (pend= prices777_createpending(curlingp,bot,cHandlep,dotrade,item,activenxt,secret,prices,dir,price,volume,iQ,order,orderid,extra)) != 0 ) { if ( bot == 0 || dotrade == 0 ) retstr = prices777_issuepending(pend); else if ( pend->queueflag != 0 ) retstr = clonestr("{\"result\":\"pending_trade created\"}"); else retstr = clonestr("{\"error\":\"pending_trade couldnt be created\"}"); if ( pend->queueflag == 0 ) free_pending(pend), pend = 0; else if ( pendp != 0 ) *pendp = pend; return(retstr); } else return(clonestr("{\"error\":\"couldnt createpending\"}")); } char *issue_calculateFullHash(char *unsignedtxbytes,char *sighash) { char cmd[4096]; sprintf(cmd,"requestType=calculateFullHash&unsignedTransactionBytes=%s&signatureHash=%s",unsignedtxbytes,sighash); return(issue_NXTPOST(cmd)); } char *issue_parseTransaction(char *txbytes) { char cmd[4096],*retstr = 0; sprintf(cmd,"requestType=parseTransaction&transactionBytes=%s",txbytes); retstr = issue_NXTPOST(cmd); //printf("issue_parseTransaction.%s %s\n",txbytes,retstr); if ( retstr != 0 ) { //printf("(%s) -> (%s)\n",cmd,retstr); //retstr = parse_NXTresults(0,"sender","",results_processor,jsonstr,strlen(jsonstr)); //free(jsonstr); } else printf("error getting txbytes.%s\n",txbytes); return(retstr); } uint64_t issue_broadcastTransaction(int32_t *errcodep,char **retstrp,char *txbytes,char *NXTACCTSECRET) { cJSON *json,*errjson; uint64_t txid = 0; char cmd[4096],secret[8192],*retstr; escape_code(secret,NXTACCTSECRET); sprintf(cmd,"requestType=broadcastTransaction&secretPhrase=%s&transactionBytes=%s",secret,txbytes); retstr = issue_NXTPOST(cmd); *errcodep = -1; if ( retstrp != 0 ) *retstrp = retstr; if ( retstr != 0 ) { //printf("(%s) -> (%s)\n",cmd,retstr); //printf("broadcast got.(%s)\n",retstr); if ( (json= cJSON_Parse(retstr)) != 0 ) { errjson = cJSON_GetObjectItem(json,"errorCode"); if ( errjson != 0 ) { //printf("ERROR broadcasting.(%s)\n",retstr); *errcodep = (int32_t)get_cJSON_int(json,"errorCode"); } else { if ( (txid = get_satoshi_obj(json,"transaction")) != 0 ) *errcodep = 0; } } if ( retstrp == 0 ) free(retstr); } return(txid); } char *issue_signTransaction(char *txbytes,char *NXTACCTSECRET) { char cmd[4096],secret[8192]; escape_code(secret,NXTACCTSECRET); sprintf(cmd,"requestType=signTransaction&secretPhrase=%s&unsignedTransactionBytes=%s",secret,txbytes); return(issue_NXTPOST(cmd)); } char *issue_approveTransaction(char *fullhash,char *revealed,char *message,char *NXTACCTSECRET) { char cmd[4096],secret[8192]; escape_code(secret,NXTACCTSECRET); sprintf(cmd,"requestType=approveTransaction&secretPhrase=%s&transactionFullHash=%s&revealedSecret=%s&messageIsText=true&feeNQT=%lld&deadline=%d&message=%s",secret,fullhash,revealed,(long long)MIN_NQTFEE,DEFAULT_NXT_DEADLINE,message); printf("submit approve.(%s)\n",cmd); return(issue_NXTPOST(cmd)); } uint32_t issue_getTime() { char cmd[4096],*jsonstr; cJSON *json; uint32_t timestamp = 0; //sprintf(cmd,"requestType=getAsset&asset=%s",assetidstr); sprintf(cmd,"requestType=getTime"); if ( (jsonstr= issue_NXTPOST(cmd)) != 0 ) { if ( (json= cJSON_Parse(jsonstr)) != 0 ) timestamp = juint(json,"time"), free_json(json); free(jsonstr); } return(timestamp); } int32_t swap_verifyNXT(uint32_t *finishp,uint32_t *deadlinep,cJSON *origjson,char *offerNXT,char *exchangestr,uint64_t orderid,uint64_t quoteid,struct InstantDEX_quote *iQ,char *phasedtx) { char UTX[32768],*triggerhash,*utx,*sighash,*jsonstr=0,*parsed,*fullhash,*cmpstr; cJSON *json=0,*txobj,*attachment; int32_t retval = -1; uint64_t otherbits,otherqty,recvasset; struct destbuf calchash; int64_t recvqty; uint32_t i,j,timestamp,now,finishheight; *finishp = 0; if ( (triggerhash= jstr(origjson,"T")) == 0 ) triggerhash = jstr(origjson,"trigger"); otherbits = j64bits(origjson,"a"); otherqty = j64bits(origjson,"q"); fullhash = jstr(origjson,"FH"); finishheight = juint(origjson,"F"); if ( phasedtx == 0 ) { utx = jstr(origjson,"U"); if ( utx != 0 && strlen(utx) > sizeof(UTX) ) { printf("UTX overflow\n"); return(-1); } else if ( utx != 0 ) { for (i=0; utx[i]!=0; i++) if ( utx[i] == 'Z' ) { memcpy(UTX,utx,i); for (j=0; j<128; j++) UTX[i+j] = '0'; UTX[i+j] = 0; strcat(UTX,utx+i+1); break; } } sighash = jstr(origjson,"S"); if ( iQ->s.isask == 0 ) recvasset = iQ->s.baseid, recvqty = iQ->s.baseamount / get_assetmult(recvasset); else recvasset = iQ->s.relid, recvqty = iQ->s.relamount / get_assetmult(recvasset); printf("utx.(%s) -> UTX.(%s) sighash.(%s)\n",utx,UTX,sighash); } else { recvqty = otherqty; recvasset = otherbits; } if ( phasedtx != 0 || (jsonstr= issue_calculateFullHash(UTX,sighash)) != 0 ) { if ( phasedtx != 0 || (json= cJSON_Parse(jsonstr)) != 0 ) { copy_cJSON(&calchash,jobj(json,"fullHash")); if ( phasedtx != 0 || strcmp(calchash.buf,fullhash) == 0 ) { if ( (parsed= issue_parseTransaction(phasedtx != 0 ? phasedtx : UTX)) != 0 ) { _stripwhite(parsed,' '); //printf("iQ (%llu/%llu) otherbits.%llu qty %llu PARSED OFFER.(%s) triggerhash.(%s) (%s) offer sender.%s\n",(long long)iQ->s.baseid,(long long)iQ->s.relid,(long long)otherbits,(long long)otherqty,parsed,fullhash,calchash,sender); if ( (txobj= cJSON_Parse(parsed)) != 0 ) { *deadlinep = juint(txobj,"deadline"); timestamp = juint(txobj,"timestamp"); now = issue_getTime(); if ( (attachment= jobj(txobj,"attachment")) != 0 ) *finishp = juint(attachment,"phasingFinishHeight"); cmpstr = jstr(txobj,"referencedTransactionFullHash"); if ( *deadlinep >= INSTANTDEX_TRIGGERDEADLINE/2 && ((long)now - timestamp) < 60 && (cmpstr == 0 || triggerhash == 0 || (cmpstr != 0 && triggerhash != 0 && strcmp(cmpstr,triggerhash) == 0)) ) { // https://nxtforum.org/nrs-releases/nrs-v1-5-15/msg191715/#msg191715 printf("GEN RESPONDTX lag.%d deadline.%d (recv.%llu %lld) recv.(%llu %lld) orderid.%llu/%llx quoteid.%llu/%llx\n",now-timestamp,*deadlinep,(long long)recvasset,(long long)recvqty,(long long)recvasset,(long long)recvqty,(long long)orderid,(long long)orderid,(long long)quoteid,(long long)quoteid); if ( InstantDEX_verify(IGUANA_MY64BITS,recvasset,recvqty,txobj,recvasset,recvqty) == 0 ) retval = 0; else printf("(%s) didnt validate against quoteid.%llu\n",parsed,(long long)quoteid); } else fprintf(stderr,"swap rejects tx deadline %d >= INSTANTDEX_TRIGGERDEADLINE/2 && (now %d - %d timestamp) %d < 60\n",*deadlinep,now,timestamp,now-timestamp); free_json(txobj); } else fprintf(stderr,"swap cant parse tx.(%s)\n",parsed); free(parsed); } else fprintf(stderr,"swap cant parse UTX.(%s)\n",UTX); } else fprintf(stderr,"mismatch (%s) != (%s)\n",calchash.buf,fullhash); if ( json != 0 ) free_json(json); } else fprintf(stderr,"swap cant parse.(%s)\n",jsonstr); if ( jsonstr != 0 ) free(jsonstr); } else fprintf(stderr,"calchash.(%s)\n",jsonstr); return(retval); } struct pending_trade *pending_swap(char **strp,int32_t type,uint64_t orderid,uint64_t quoteid,char *triggerhash,char *fullhash,char *txstr,char *txstr2) { struct pending_trade *pend; cJSON *retjson; pend = calloc(1,sizeof(*pend)); pend->orderid = orderid, pend->quoteid = quoteid; if ( triggerhash != 0 ) pend->triggertx = clonestr(triggerhash); if ( fullhash != 0 ) pend->txbytes = clonestr(fullhash); pend->type = type; if ( txstr != 0 && txstr2 != 0 ) { retjson = cJSON_CreateObject(); jadd(retjson,"fee",cJSON_Parse(txstr)); jadd(retjson,"responsetx",cJSON_Parse(txstr2)); *strp = jprint(retjson,0); pend->tradesjson = retjson; } pend->timestamp = (uint32_t)time(NULL); return(pend); } char *swap_responseNXT(int32_t type,char *offerNXT,uint64_t otherbits,uint64_t otherqty,uint64_t orderid,uint64_t quoteid,int32_t deadline,char *triggerhash,char *phaselink,int32_t finishheight,struct InstantDEX_quote *iQ) { struct NXTtx fee,responsetx; int32_t errcode,errcode2; char *txstr,*txstr2,*str = 0; struct pending_trade *pend; gen_NXTtx(&fee,calc_nxt64bits(INSTANTDEX_ACCT),NXT_ASSETID,INSTANTDEX_FEE,orderid,quoteid,deadline,triggerhash,0,0,0); gen_NXTtx(&responsetx,calc_nxt64bits(offerNXT),otherbits,otherqty,orderid,quoteid,deadline,triggerhash,phaselink,finishheight,0); if ( (fee.txid= issue_broadcastTransaction(&errcode,&txstr,fee.txbytes,IGUANA_NXTACCTSECRET)) != 0 ) { if ( (responsetx.txid= issue_broadcastTransaction(&errcode2,&txstr2,responsetx.txbytes,IGUANA_NXTACCTSECRET)) != 0 ) { if ( (pend= pending_swap(&str,type,orderid,quoteid,triggerhash,phaselink,txstr,txstr2)) != 0 && str != 0 ) { iQ->s.pending = iQ->s.swap = 1; //InstantDEX_history(0,pend,str); pend->queueflag = 1; queue_enqueue("PendingQ",&Pending_offersQ,&pend->DL,0); printf("BROADCAST fee.txid %llu and %llu (%s %s)\n",(long long)fee.txid,(long long)responsetx.txid,fee.fullhash,responsetx.fullhash); } } else printf("error.%d broadcasting responsetx.(%s) %s\n",errcode2,responsetx.txbytes,txstr2); } else printf("error.%d broadcasting feetx.(%s) %s\n",errcode,fee.txbytes,txstr); if ( str == 0 ) str = clonestr("{\"error\":\"swap_responseNXT error responding\"}"); return(str); } int32_t extract_pkhash(char *pubkeystr,char *pkhash,char *script) { int32_t len; uint8_t rmd160[20],data[4096],*ptr; decode_hex(data,(int32_t)strlen(script)>>1,script); len = data[0]; ptr = &data[len + 1]; len = *ptr++; if ( len == 33 ) { init_hexbytes_noT(pubkeystr,ptr,33); calc_OP_HASH160(pkhash,rmd160,pubkeystr); printf("pkhash.(%s)\n",pkhash); return(0); } return(-1); } char *swap_func(int32_t localaccess,int32_t valid,char *sender,cJSON *origjson,char *origargstr) { /*char script[4096],hexstr[128],*str,*base,*rel,*txstr,*phasedtx,*cointxid,*signedtx,*jsonstr; uint8_t msgbuf[512]; struct pending_trade *pend; struct prices777_order order; struct InstantDEX_quote *iQ,_iQ; cJSON *json; uint32_t deadline,finishheight,nonce,isask; int32_t errcode,myoffer,myfill,len; struct NXTtx sendtx,fee; struct destbuf spendtxid,reftx; struct destbuf offerNXT,exchange; uint64_t otherbits,otherqty,quoteid,orderid,recvasset,recvqty,sendasset,sendqty,fillNXT,destbits,value; char pubkeystr[128],pkhash[64],swapbuf[4096],refredeemscript[1024],vintxid[128],*triggerhash,*fullhash,*dest,deststr[64];*/ struct destbuf offerNXT,exchange; uint32_t deadline,finishheight,isask; char *triggerhash,*fullhash; int32_t myoffer,myfill; struct InstantDEX_quote *iQ,_iQ; struct prices777_order order; uint64_t otherbits,otherqty,quoteid,orderid,fillNXT; copy_cJSON(&offerNXT,jobj(origjson,"offerNXT")); fillNXT = j64bits(origjson,"fillNXT"); copy_cJSON(&exchange,jobj(origjson,"exchange")); finishheight = juint(origjson,"F"); if ( (triggerhash= jstr(origjson,"T")) == 0 ) triggerhash = jstr(origjson,"trigger"); myoffer = strcmp(IGUANA_NXTADDR,offerNXT.buf) == 0; myfill = (IGUANA_MY64BITS == fillNXT); //printf("swap_func got (%s)\n",origargstr); if ( myoffer+myfill != 0 ) { orderid = j64bits(origjson,"orderid"); quoteid = j64bits(origjson,"quoteid"); if ( (iQ= find_iQ(quoteid)) == 0 ) { fprintf(stderr,"swap_func: cant find quoteid.%llu\n",(long long)quoteid); iQ = &_iQ, memset(iQ,0,sizeof(*iQ)); //return(clonestr("{\"error\":\"cant find quoteid\"}")); } if ( iQ->s.responded != 0 ) { fprintf(stderr,"already responded quoteid.%llu\n",(long long)iQ->s.quoteid); return(0); } isask = iQ->s.isask; memset(&order,0,sizeof(order)); order.s = iQ->s; #ifdef notyet if ( strcmp("wallet",exchange.buf) == 0 ) { uint64_t sendamount,recvamount; struct coin777 *recvcoin,*sendcoin; char refundsig[512],fieldA[64],fieldB[64],fieldpkhash[64]; char *recvstr,*sendstr,*spendtx,*refundtx,*redeemscript,*rpubA,*rpubB,*rpkhash,*spubA,*spubB,*spkhash; recvcoin = sendcoin = 0; sendamount = recvamount = 0; if ( (recvstr= jstr(origjson,"recvcoin")) != 0 ) recvcoin = coin777_find(recvstr,0); if ( (sendstr= jstr(origjson,"sendcoin")) != 0 ) sendcoin = coin777_find(sendstr,0); if ( iQ->s.baseid == NXT_ASSETID ) isask ^= 1; //printf("recvstr.%p sendstr.%p\n",recvstr,sendstr); if ( recvstr != 0 && sendstr != 0 ) { if ( (sendamount= j64bits(origjson,"sendamount")) != 0 && (recvamount= j64bits(origjson,"recvamount")) != 0 && sendcoin != 0 && recvcoin != 0 && (refundtx= jstr(origjson,"rtx")) != 0 && (redeemscript= jstr(origjson,"rs")) != 0 && (rpubA= jstr(origjson,"rpubA")) != 0 && (rpubB= jstr(origjson,"rpubB")) != 0 && (rpkhash= jstr(origjson,"rpkhash")) != 0 && triggerhash != 0 && (spubA= jstr(origjson,"spubA")) != 0 && (spubB= jstr(origjson,"spubB")) != 0 && (spkhash= jstr(origjson,"spkhash")) != 0 ) { } } else if ( recvstr != 0 ) { //printf("INCOMINGRECV.(%s)\n",origargstr); sprintf(fieldA,"%spubA",recvcoin->name); sprintf(fieldB,"%spubB",recvcoin->name); sprintf(fieldpkhash,"%spkhash",recvcoin->name); if ( (recvamount= j64bits(origjson,"recvamount")) != 0 && recvcoin != 0 && (rpubA= jstr(origjson,fieldA)) != 0 && (rpubB= jstr(origjson,fieldB)) != 0 && (rpkhash= jstr(origjson,fieldpkhash)) != 0 ) { if ( ((isask != 0 && myoffer != 0) || (isask == 0 && myfill != 0)) && j64bits(origjson,"fill") != IGUANA_MY64BITS && (refundtx= jstr(origjson,"rtx")) != 0 && (redeemscript= jstr(origjson,"rs")) != 0 ) // Bob: sends NXT to Alice, recvs recvcoin { subatomic_pubkeyhash(pubkeystr,pkhash,recvcoin,quoteid); //printf("CALC >>>>>>>>>> (%s) vs (%s)\n",pkhash,rpkhash); if ( (base= jstr(origjson,"base")) != 0 && (rel= jstr(origjson,"rel")) != 0 && (sendasset= j64bits(origjson,"sendasset")) != 0 && (sendqty= j64bits(origjson,"sendqty")) != 0 ) { //printf("inside (%s/%s) sendasset.%llu sendqty.%llu rpkhash.(%s)\n",base,rel,(long long)sendasset,(long long)sendqty,rpkhash); if ( (spendtx= subatomic_spendtx(&spendtxid,vintxid,refundsig,recvcoin,rpubA,rpubB,pubkeystr,recvamount,refundtx,redeemscript)) != 0 ) { finishheight = 60; deadline = 3600*4; if ( (pend= pending_swap(&str,'A',orderid,quoteid,0,0,0,0)) != 0 ) { if ( isask == 0 ) destbits = calc_nxt64bits(offerNXT.buf); else destbits = fillNXT; gen_NXTtx(&fee,calc_nxt64bits(INSTANTDEX_ACCT),NXT_ASSETID,INSTANTDEX_FEE,orderid,quoteid,deadline,triggerhash,0,0,0); issue_broadcastTransaction(&errcode,&txstr,fee.txbytes,IGUANA_NXTACCTSECRET); gen_NXTtx(&sendtx,destbits,sendasset,sendqty,orderid,quoteid,deadline,triggerhash,0,_get_NXTheight(0)+finishheight,rpkhash); //issue_broadcastTransaction(&errcode,&txstr,sendtx.txbytes,IGUANA_NXTACCTSECRET); printf(">>>>>>>>>>>> broadcast fee and phased.(%s) trigger.%s\n",sendtx.txbytes,triggerhash); sprintf(swapbuf,"{\"orderid\":\"%llu\",\"quoteid\":\"%llu\",\"offerNXT\":\"%s\",\"fillNXT\":\"%llu\",\"plugin\":\"relay\",\"destplugin\":\"InstantDEX\",\"method\":\"busdata\",\"submethod\":\"swap\",\"exchange\":\"wallet\",\"recvcoin\":\"%s\",\"recvamount\":\"%lld\",\"%s\":\"%s\",\"%s\":\"%s\",\"%s\":\"%s\",\"refundsig\":\"%s\",\"phasedtx\":\"%s\",\"spendtxid\":\"%s\",\"a\":\"%llu\",\"q\":\"%llu\",\"trigger\":\"%s\",\"fill\":\"%llu\"}",(long long)orderid,(long long)quoteid,offerNXT.buf,(long long)fillNXT,recvcoin->name,(long long)recvamount,fieldA,rpubA,fieldB,rpubB,fieldpkhash,rpkhash,refundsig,sendtx.txbytes,spendtxid.buf,(long long)sendasset,(long long)sendqty,triggerhash,(long long)IGUANA_MY64BITS); if ( (str= busdata_sync(&nonce,swapbuf,"allnodes",0)) != 0 ) free(str); // poll for vin then broadcast spendtx printf(">>>>>>>>>>>>>>>>>>>> wait for (%s) then send SPENDTX.(%s)\n",vintxid,spendtx); if ( (value= wait_for_txid(script,recvcoin,vintxid,0,recvamount,recvcoin->minconfirms,0)) != 0 ) { signedtx = malloc(strlen(spendtx) + 16); sprintf(signedtx,"[\"%s\"]",spendtx); if ( (cointxid= bitcoind_passthru(recvcoin->name,recvcoin->serverport,recvcoin->userpass,"sendrawtransaction",signedtx)) != 0 ) { printf(">>>>>>>>>>>>> BROADCAST SPENDTX.(%s) (%s)\n",signedtx,cointxid); free(cointxid); } free(signedtx); } printf("ATOMIC SWAP.%llu finished\n",(long long)quoteid); iQ->s.closed = 1; delete_iQ(quoteid); } else printf("cant get pending_swap pend.%p\n",pend); free(spendtx); return(clonestr(swapbuf)); } else printf("error generating spendtx\n"); } else printf("mismatched recv (%s vs %s) or (%s)\n",recvcoin->atomicrecvpubkey,rpubB,rpkhash); } else if ( j64bits(origjson,"fill") != IGUANA_MY64BITS && (str= jstr(origjson,"refundsig")) != 0 && str[0] != 0 && (phasedtx= jstr(origjson,"phasedtx")) != 0 && phasedtx[0] != 0 ) // Alice to verify NXTtx and send recvcoin { if ( isask != 0 ) dest = offerNXT.buf; else { expand_nxt64bits(deststr,fillNXT); dest = deststr; } if ( swap_verifyNXT(&finishheight,&deadline,origjson,dest,exchange.buf,orderid,quoteid,iQ,phasedtx) == 0 ) { if ( recvcoin->refundtx != 0 && (recvcoin->signedrefund= subatomic_validate(recvcoin,rpubA,rpubB,rpkhash,recvcoin->refundtx,str)) != 0 ) { free(recvcoin->refundtx), recvcoin->refundtx = 0; issue_broadcastTransaction(&errcode,&txstr,recvcoin->trigger.txbytes,IGUANA_NXTACCTSECRET); issue_broadcastTransaction(&errcode,&txstr,phasedtx,IGUANA_NXTACCTSECRET); printf(">>>>>>>>>>>>>>>>>>>>>>>>>>> ISSUE TRIGGER.(%s) phased.(%s).%d | signedrefund.(%s)\n",recvcoin->trigger.txbytes,txstr!=0?txstr:"phasedsubmit error",errcode,recvcoin->signedrefund); signedtx = malloc(strlen(recvcoin->funding.signedtransaction) + 16); sprintf(signedtx,"[\"%s\"]",recvcoin->funding.signedtransaction); if ( (cointxid= bitcoind_passthru(recvcoin->name,recvcoin->serverport,recvcoin->userpass,"sendrawtransaction",signedtx)) != 0 ) { printf(">>>>>>>>>>>>> FUNDING BROADCAST.(%s) (%s)\n",recvcoin->funding.signedtransaction,cointxid); free(cointxid); } else printf("error sendrawtransaction.(%s)\n",recvcoin->funding.signedtransaction); free(signedtx); copy_cJSON(&spendtxid,jobj(origjson,"spendtxid")); printf("wait for spendtx.(%s)\n",spendtxid.buf); if ( (value= wait_for_txid(script,recvcoin,spendtxid.buf,0,recvamount-recvcoin->mgw.txfee,0,0)) != 0 ) { iQ->s.responded = 1; if ( extract_pkhash(pubkeystr,pkhash,script) == 0 ) { if ( strcmp(pkhash,rpkhash) == 0 ) { reftx.buf[0] = 0; if ( (jsonstr= issue_parseTransaction(phasedtx)) != 0 ) { if ( (json= cJSON_Parse(jsonstr)) != 0 ) { copy_cJSON(&reftx,jobj(json,"fullHash")); free_json(json); } free(jsonstr); } len = 0; len = txind777_txbuf(msgbuf,len,orderid,sizeof(orderid)); len = txind777_txbuf(msgbuf,len,quoteid,sizeof(quoteid)); init_hexbytes_noT(hexstr,msgbuf,len); if ( (str= issue_approveTransaction(reftx.buf,pubkeystr,hexstr,IGUANA_NXTACCTSECRET)) != 0 ) { printf("fullhash.(%s) pubkey.(%s) pkhash.(%s) APPROVED.(%s)\n",reftx.buf,pubkeystr,pkhash,str); free(str); } else printf("error sending in approval\n"); } else printf("script.(%s) -> pkhash.(%s) vs rpkhash.(%s)\n",script,pkhash,rpkhash); } else printf("unexpected end of script.(%s) (%s)\n",script,str); } printf("FINISHED ATOMIC SWAP of quoteid.%llu\n",(long long)quoteid); iQ->s.closed = 1; memset(&recvcoin->trigger,0,sizeof(recvcoin->trigger)); memset(&recvcoin->funding,0,sizeof(recvcoin->funding)); free(recvcoin->signedrefund), recvcoin->signedrefund = 0; delete_iQ(quoteid); } else printf("refund tx didnt verify\n"); } else printf("NXT tx didnt verify\n"); } //else printf("myfill.%d myoffer.%d recv mismatch isask.%d\n",myfill,myoffer,iQ->s.isask); //printf("recv failed\n"); return(clonestr("{\"result\":\"recv failed\"}")); } } else if ( sendstr != 0 ) // Alice sendcoin -> Bob, recvs NXT { //printf("INCOMINGSEND.(%s)\n",origargstr); sprintf(fieldA,"%spubA",sendcoin->name); sprintf(fieldB,"%spubB",sendcoin->name); sprintf(fieldpkhash,"%spkhash",sendcoin->name); if ( ((isask == 0 && myoffer != 0) || (isask != 0 && myfill != 0)) && (sendamount= j64bits(origjson,"sendamount")) != 0 && sendcoin != 0 && triggerhash != 0 && (spubA= jstr(origjson,fieldA)) != 0 && (spubB= jstr(origjson,fieldB)) != 0 && (spkhash= jstr(origjson,fieldpkhash)) != 0 ) { if ( (base= jstr(origjson,"base")) != 0 && (rel= jstr(origjson,"rel")) != 0 && (recvasset= j64bits(origjson,"recvasset")) != 0 && (recvqty= j64bits(origjson,"recvqty")) != 0 ) { if ( sendcoin->funding.signedtransaction[0] == 0 && (refundtx= subatomic_fundingtx(refredeemscript,&sendcoin->funding,sendcoin,spubA,spubB,spkhash,sendamount,10)) != 0 ) { deadline = 3600; gen_NXTtx(&sendcoin->trigger,calc_nxt64bits(INSTANTDEX_ACCT),NXT_ASSETID,INSTANTDEX_FEE,orderid,quoteid,deadline,0,0,0,0); sprintf(swapbuf,"{\"orderid\":\"%llu\",\"quoteid\":\"%llu\",\"offerNXT\":\"%s\",\"fillNXT\":\"%llu\",\"plugin\":\"relay\",\"destplugin\":\"InstantDEX\",\"method\":\"busdata\",\"submethod\":\"swap\",\"exchange\":\"%s\",\"recvamount\":\"%lld\",\"rtx\":\"%s\",\"rs\":\"%s\",\"recvcoin\":\"%s\",\"%s\":\"%s\",\"%s\":\"%s\",\"%s\":\"%s\",\"trigger\":\"%s\",\"sendasset\":\"%llu\",\"sendqty\":\"%llu\",\"base\":\"%s\",\"rel\":\"%s\",\"fill\":\"%llu\"}",(long long)orderid,(long long)quoteid,offerNXT.buf,(long long)fillNXT,exchange.buf,(long long)sendamount,refundtx,refredeemscript,sendstr,fieldA,spubA,fieldB,spubB,fieldpkhash,spkhash,sendcoin->trigger.fullhash,(long long)recvasset,(long long)recvqty,base,rel,(long long)IGUANA_MY64BITS); sendcoin->refundtx = refundtx; if ( (str= busdata_sync(&nonce,swapbuf,"allnodes",0)) != 0 ) free(str); return(clonestr(swapbuf)); //printf("BUSDATA.(%s)\n",swapbuf); } else return(clonestr("{\"error\":\"cant create refundtx, maybe already pending\"}\n")); } //else printf("mismatched send (%s vs %s) or (%s)\n",sendcoin->atomicrecvpubkey,spubB,spkhash); } else printf("myfill.%d myoffer.%d send mismatch isask.%d\n",myfill,myoffer,iQ->s.isask); } return(clonestr("{\"result\":\"processed wallet swap\"}")); } #endif if ( myoffer != 0 && swap_verifyNXT(&finishheight,&deadline,origjson,offerNXT.buf,exchange.buf,orderid,quoteid,iQ,0) == 0 ) { otherbits = j64bits(origjson,"a"); otherqty = j64bits(origjson,"q"); fullhash = jstr(origjson,"FH"); finishheight = juint(origjson,"F"); return(swap_responseNXT('R',offerNXT.buf,otherbits,otherqty,orderid,quoteid,deadline,triggerhash,fullhash,finishheight,iQ)); } else printf("myfill.%d myoffer.%d swap mismatch\n",myfill,myoffer); } return(clonestr("{\"result\":\"processed swap\"}")); } int32_t match_unconfirmed(char *sender,char *hexstr,cJSON *txobj,char *txidstr,char *account,uint64_t amount,uint64_t qty,uint64_t assetid,char *recipient) { // ok, the bug here is that on a delayed respondtx, the originator should refuse to release the trigger (and the money tx) uint64_t orderid,quoteid,recvasset,sendasset; int64_t recvqty,sendqty; uint32_t bidask,deadline,timestamp,now; struct InstantDEX_quote *iQ; decode_hex((void *)&orderid,sizeof(orderid),hexstr); decode_hex((void *)"eid,sizeof(quoteid),hexstr+16); //printf("match_unconfirmed.(%s) orderid.%llu %llx quoteid.%llu %llx\n",hexstr,(long long)orderid,(long long)orderid,(long long)quoteid,(long long)quoteid); deadline = juint(txobj,"deadline"); timestamp = juint(txobj,"timestamp"); now = issue_getTime(); //printf("deadline.%u now.%u timestamp.%u lag %ld\n",deadline,now,timestamp,((long)now - timestamp)); if ( deadline < INSTANTDEX_TRIGGERDEADLINE/2 || ((long)now - timestamp) > 60*2 ) return(0); if ( (iQ= find_iQ(quoteid)) != 0 && iQ->s.closed == 0 && iQ->s.pending != 0 && (iQ->s.responded == 0 || iQ->s.feepaid == 0) ) { if ( Debuglevel > 2 ) printf("match unconfirmed %llu/%llu %p swap.%d feepaid.%d responded.%d sender.(%s) -> recv.(%s) me.(%s) offer.(%llu)\n",(long long)orderid,(long long)quoteid,iQ,iQ->s.swap,iQ->s.feepaid,iQ->s.responded,sender,recipient,IGUANA_NXTADDR,(long long)iQ->s.offerNXT); if ( iQ->s.swap != 0 && (strcmp(recipient,INSTANTDEX_ACCT) == 0 || strcmp(recipient,IGUANA_NXTADDR) == 0) ) { if ( iQ->s.feepaid == 0 ) { if ( verify_NXTtx(txobj,NXT_ASSETID,INSTANTDEX_FEE,calc_nxt64bits(INSTANTDEX_ACCT)) == 0 ) { iQ->s.feepaid = 1; printf("FEE DETECTED\n"); } else printf("notfee: dest.%s src.%s amount.%llu qty.%llu assetid.%llu\n",recipient,sender,(long long)amount,(long long)qty,(long long)assetid); } if ( iQ->s.responded == 0 ) { bidask = iQ->s.isask; if ( iQ->s.offerNXT == IGUANA_MY64BITS ) bidask ^= 1; if ( bidask != 0 ) { sendasset = iQ->s.relid, sendqty = iQ->s.relamount; recvasset = iQ->s.baseid, recvqty = iQ->s.baseamount; } else { sendasset = iQ->s.baseid, sendqty = iQ->s.baseamount; recvasset = iQ->s.relid, recvqty = iQ->s.relamount; } sendqty /= get_assetmult(sendasset); recvqty /= get_assetmult(recvasset); if ( Debuglevel > 2 ) printf("sendasset.%llu sendqty.%llu mult.%llu, recvasset.%llu recvqty.%llu mult.%llu\n",(long long)sendasset,(long long)sendqty,(long long)get_assetmult(sendasset),(long long)recvasset,(long long)recvqty,(long long)get_assetmult(recvasset)); if ( InstantDEX_verify(IGUANA_MY64BITS,sendasset,sendqty,txobj,recvasset,recvqty) == 0 ) { iQ->s.responded = 1; printf("iQ: %llu/%llu %lld/%lld | recv %llu %lld offerNXT.%llu\n",(long long)iQ->s.baseid,(long long)iQ->s.relid,(long long)iQ->s.baseamount,(long long)iQ->s.relamount,(long long)recvasset,(long long)recvqty,(long long)iQ->s.offerNXT); printf("RESPONSE DETECTED\n"); } } if ( iQ->s.responded != 0 && iQ->s.feepaid != 0 ) { printf("both detected offer.%llu my64bits.%llu\n",(long long)iQ->s.offerNXT,(long long)IGUANA_MY64BITS); complete_swap(iQ,orderid,quoteid,iQ->s.offerNXT == IGUANA_MY64BITS); } } } return(-1); } int32_t is_unfunded_order(uint64_t nxt64bits,uint64_t assetid,uint64_t amount) { char assetidstr[64],NXTaddr[64],cmd[1024],*jsonstr; int64_t ap_mult,unconfirmed,balance = 0; cJSON *json; expand_nxt64bits(NXTaddr,nxt64bits); if ( assetid == NXT_ASSETID ) { sprintf(cmd,"requestType=getAccount&account=%s",NXTaddr); if ( (jsonstr= issue_NXTPOST(cmd)) != 0 ) { if ( (json= cJSON_Parse(jsonstr)) != 0 ) { balance = get_API_nxt64bits(cJSON_GetObjectItem(json,"balanceNQT")); free_json(json); } free(jsonstr); } strcpy(assetidstr,"NXT"); } else { expand_nxt64bits(assetidstr,assetid); if ( (ap_mult= assetmult(assetidstr)) != 0 ) { expand_nxt64bits(NXTaddr,nxt64bits); balance = ap_mult * get_asset_quantity(&unconfirmed,NXTaddr,assetidstr); } } if ( balance < amount ) { printf("balance %.8f < amount %.8f for asset.%s\n",dstr(balance),dstr(amount),assetidstr); return(1); } return(0); } cJSON *InstantDEX_tradejson(int32_t *curlingp,void *bot,struct pending_trade **pendp,void **cHandlep,cJSON *item,char *activenxt,char *secret,struct prices777_order *order,int32_t dotrade,uint64_t orderid,char *extra) { char swapbuf[8192],buf[8192],triggertx[4096],txbytes[4096],*retstr,*exchange; uint64_t txid,qty,avail,priceNQT; struct prices777 *prices; cJSON *json = 0; if ( pendp != 0 ) *pendp = 0; if ( (prices= order->source) != 0 ) { exchange = prices->exchange; swapbuf[0] = 0; if ( dotrade == 0 ) { if ( strcmp(exchange,INSTANTDEX_NAME) != 0 && strcmp(exchange,"wallet") != 0 ) { sprintf(buf,"{\"orderid\":\"%llu\",\"trade\":\"%s\",\"exchange\":\"%s\",\"base\":\"%s\",\"rel\":\"%s\",\"baseid\":\"%llu\",\"relid\":\"%llu\",\"price\":%.8f,\"volume\":%.8f,\"extra\":\"%s\"}",(long long)orderid,order->wt > 0. ? "buy" : "sell",exchange,prices->base,prices->rel,(long long)prices->baseid,(long long)prices->relid,order->s.price,order->s.vol,extra!=0?extra:""); if ( strcmp(exchange,"nxtae") == 0 ) { qty = calc_asset_qty(&avail,&priceNQT,activenxt,0,prices->baseid,order->s.price,order->s.vol); sprintf(buf+strlen(buf)-1,",\"priceNQT\":\"%llu\",\"quantityQNT\":\"%llu\",\"avail\":\"%llu\"}",(long long)priceNQT,(long long)qty,(long long)avail); if ( qty == 0 ) sprintf(buf+strlen(buf)-1,",\"error\":\"insufficient balance\"}"); } return(cJSON_Parse(buf)); } else { //{"inverted":0,"contract":"MMNXT/Jay","baseid":"979292558519844732","relid":"8688289798928624137","bids":[{"plugin":"Inst // antDEX","method":"tradesequence","dotrade":1,"price":2,"volume":2,"trades":[]}],"asks":[],"numbids":1,"numasks":0,"lastb // id":2,"lastask":0,"NXT":"11471677413693100042","timestamp":1440587058,"maxdepth":25} prices777_swapbuf("yes",0,&txid,triggertx,txbytes,swapbuf,prices->exchange,prices->base,prices->rel,order,orderid,extra==0?0:myatoi(extra,10000),0); return(cJSON_Parse(swapbuf)); } } retstr = prices777_trade(curlingp,bot,pendp,cHandlep,dotrade,item,activenxt,secret,prices,order->wt,order->s.price,order->s.vol,0,order,orderid,extra); if ( retstr != 0 ) { json = cJSON_Parse(retstr); free(retstr); } } return(json); } char *InstantDEX_dotrades(int32_t curlings[],void *bot,void *cHandles[],char *activenxt,char *secret,cJSON *json,struct prices777_order *trades,int32_t numtrades,int32_t dotrade,char *extra) { struct destbuf exchangestr,gui,name,base,rel; struct InstantDEX_quote iQ; cJSON *retjson,*retarray; int32_t i; bidask_parse(1,&exchangestr,&name,&base,&rel,&gui,&iQ,json); retjson = cJSON_CreateObject(), retarray = cJSON_CreateArray(); for (i=0; i<numtrades; i++) { //printf("GOT%d.(%s)\n",i,jprint(json,0)); if ( trades[i].retitem != 0 ) free_json(trades[i].retitem ); trades[i].retitem = InstantDEX_tradejson(curlings!=0?&curlings[i]:0,bot,&trades[i].pend,cHandles!=0?cHandles[i]:0,jobj(json,"trades"),activenxt,secret,&trades[i],dotrade,iQ.s.quoteid,extra); jaddi(retarray,trades[i].retitem); } jadd(retjson,"traderesults",retarray); return(jprint(retjson,0)); } char *InstantDEX_tradesequence(int32_t curlings[],void *bot,void *cHandles[],int32_t *nump,struct prices777_order *trades,int32_t maxtrades,int32_t dotrade,char *activenxt,char *secret,cJSON *json) { //"trades":[[{"basket":"bid","rootwt":-1,"groupwt":1,"wt":-1,"price":40000,"volume":0.00015000,"group":0,"trade":"buy","exchange":"nxtae","asset":"17554243582654188572","base":"BTC","rel":"NXT","orderid":"3545444239044461477","orderprice":40000,"ordervolume":0.00015000}], [{"basket":"bid","rootwt":-1,"groupwt":1,"wt":1,"price":0.00376903,"volume":1297.41480000,"group":10,"trade":"sell","exchange":"coinbase","name":"BTC/USD","base":"BTC","rel":"USD","orderid":"1","orderprice":265.32000000,"ordervolume":4.89000000}]]} cJSON *array,*item; int32_t i,n,dir; char *tradestr,*exchangestr; struct prices777_order *order; uint64_t orderid,assetid,currency,baseid,relid,quoteid; int64_t sendbase,recvbase,sendrel,recvrel; struct destbuf base,rel,name; double orderprice,ordervolume; struct prices777 *prices; uint32_t timestamp; if ( (array= jarray(&n,json,"trades")) != 0 ) { if ( n > maxtrades ) return(clonestr("{\"error\":\"exceeded max trades possible in a tradesequence\"}")); if ( n == 1 && is_cJSON_Array(jitem(array,0)) != 0 ) { //printf("NESTED ARRAY DETECTED\n"); array = jitem(array,0); n = cJSON_GetArraySize(array); } *nump = n; timestamp = (uint32_t)time(NULL); for (i=0; i<n; i++) { order = &trades[i]; memset(order,0,sizeof(*order)); item = jitem(array,i); tradestr = jstr(item,"trade"), exchangestr = jstr(item,"exchange"); copy_cJSON(&base,jobj(item,"base")), copy_cJSON(&rel,jobj(item,"rel")), copy_cJSON(&name,jobj(item,"name")); orderid = j64bits(item,"orderid"), quoteid = j64bits(item,"quoteid"); if ( orderid == 0 ) orderid = quoteid; if ( quoteid == 0 ) quoteid = orderid; order->id = orderid, order->s.quoteid = quoteid; assetid = j64bits(item,"asset"), currency = j64bits(item,"currency"); baseid = j64bits(item,"baseid"), relid = j64bits(item,"relid"); sendbase = j64bits(item,"sendbase"), recvbase = j64bits(item,"recvbase"); sendrel = j64bits(item,"sendrel"), recvrel = j64bits(item,"recvrel"); order->s.baseamount = (recvbase - sendbase); order->s.relamount = (recvrel - sendrel); orderprice = jdouble(item,"orderprice"), ordervolume = jdouble(item,"ordervolume"); order->s.timestamp = juint(item,"timestamp"); order->s.duration = juint(item,"duration"); order->s.minperc = juint(item,"minperc"); order->s.baseid = baseid; order->s.relid = relid; //printf("ITEM.(%s)\n",jprint(item,0)); if ( tradestr != 0 ) { if ( strcmp(tradestr,"buy") == 0 ) dir = 1; else if ( strcmp(tradestr,"sell") == 0 ) dir = -1; else if ( strcmp(tradestr,"swap") == 0 ) dir = 0; else return(clonestr("{\"error\":\"invalid trade direction\"}")); if ( (prices= prices777_initpair(1,exchangestr,base.buf,rel.buf,0.,name.buf,baseid,relid,0)) != 0 ) { order->source = prices; order->s.offerNXT = j64bits(item,"offerNXT"); order->wt = dir, order->s.price = orderprice, order->s.vol = ordervolume; printf("item[%d] dir.%d (price %.8f vol %.4f) %s/%s baseid.%llu relid.%llu sendbase.%llu recvbase.%llu sendrel.%llu recvrel.%llu | baseqty.%lld relqty.%lld\n",i,dir,order->s.price,order->s.vol,prices->base,prices->rel,(long long)order->s.baseid,(long long)order->s.relid,(long long)sendbase,(long long)recvbase,(long long)sendrel,(long long)recvrel,(long long)order->s.baseamount,(long long)order->s.relamount); } else return(clonestr("{\"error\":\"invalid exchange or contract pair\"}")); } else { printf("item.(%s)\n",jprint(item,0)); return(clonestr("{\"error\":\"no trade specified\"}")); } } return(InstantDEX_dotrades(curlings,bot,cHandles,activenxt,secret,json,trades,n,dotrade,jstr(json,"extra"))); } printf("error parsing.(%s)\n",jprint(json,0)); return(clonestr("{\"error\":\"couldnt process trades\"}")); } #endif