323 lines
13 KiB
323 lines
13 KiB
/******************************************************************************
|
|
* 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_tradebots_h
|
|
#define xcode_tradebots_h
|
|
|
|
#define TRADEBOT_DEFAULT_DURATION (600)
|
|
struct tradebot_info
|
|
{
|
|
char buf[512],name[64],*prevobookstr,NXTADDR[64],NXTACCTSECRET[64];
|
|
uint32_t starttime,expiration,finishtime,startedtrades,apitag;
|
|
int32_t numtrades,havetrade,numlinks;
|
|
double price,volume;
|
|
struct prices777_order trades[256]; void *cHandles[256]; int32_t curlings[256];
|
|
struct tradebot_info *linkedbots[8];
|
|
struct apitag_info *api;
|
|
struct tradebot_info *oppo;
|
|
struct InstantDEX_quote iQ;
|
|
};
|
|
|
|
// ./SNapi "{\"allfields\":1,\"agent\":\"InstantDEX\",\"method\":\"orderbook\",\"exchange\":\"active\",\"base\":\"NXT\",\"rel\":\"BTC\"}"
|
|
|
|
// test balance verifier
|
|
// test tradeleg verifier
|
|
// test pass through quotes
|
|
// user lockin addrs
|
|
// atomic swaps using 2of3 msig
|
|
// broadcast request to all marketmakers
|
|
// pick best response and do BTC <-> NXT and NXT <-> ABC
|
|
|
|
int32_t tradebot_havealltrades(struct tradebot_info *bot)
|
|
{
|
|
int32_t i;
|
|
if ( bot->havetrade != 0 )
|
|
{
|
|
if ( bot->numlinks > 0 )
|
|
{
|
|
for (i=0; i<bot->numlinks; i++)
|
|
if ( bot->linkedbots[i] == 0 || bot->linkedbots[i]->havetrade == 0 )
|
|
return(0);
|
|
}
|
|
return(1);
|
|
}
|
|
return(0);
|
|
}
|
|
|
|
struct tradebot_info *tradebot_compile(cJSON *argjson,struct InstantDEX_quote *iQ,struct apitag_info *api)
|
|
{
|
|
static uint64_t lastmonce;
|
|
uint64_t monce; char *name,*tmp,*tmp2; int32_t duration; struct tradebot_info *bot = calloc(1,sizeof(*bot));
|
|
monce = (long long)(1000*time(NULL) + milliseconds());
|
|
if ( monce == lastmonce )
|
|
monce++;
|
|
lastmonce = monce;
|
|
bot->iQ = *iQ;
|
|
bot->api = api;
|
|
if ( (duration= juint(argjson,"duration")) == 0 )
|
|
duration = TRADEBOT_DEFAULT_DURATION;
|
|
bot->expiration = (uint32_t)time(NULL) + duration;
|
|
if ( (name= jstr(argjson,"name")) != 0 )
|
|
safecopy(bot->name,name,sizeof(bot->name));
|
|
else sprintf(bot->name,"bot.%llu",monce);
|
|
if ( (tmp= jstr(argjson,"botnxt")) == 0 || (tmp2= jstr(argjson,"secret")) == 0 )
|
|
{
|
|
safecopy(bot->NXTADDR,SUPERNET.NXTADDR,sizeof(bot->NXTADDR));
|
|
safecopy(bot->NXTACCTSECRET,SUPERNET.NXTACCTSECRET,sizeof(bot->NXTACCTSECRET));
|
|
}
|
|
else
|
|
{
|
|
safecopy(bot->NXTADDR,tmp,sizeof(bot->NXTADDR));
|
|
safecopy(bot->NXTACCTSECRET,tmp2,sizeof(bot->NXTACCTSECRET));
|
|
}
|
|
//bot->arbmargin = jdouble(argjson,"arbmargin");
|
|
return(bot);
|
|
}
|
|
|
|
int32_t tradebot_acceptable(struct tradebot_info *bot,cJSON *item)
|
|
{
|
|
double price,volume; int32_t dir,i,n; cJSON *trades,*trade;
|
|
if ( bot->iQ.s.isask != 0 )
|
|
dir = -1;
|
|
else dir = 1;
|
|
bot->price = price = jdouble(item,"price");
|
|
bot->volume = volume = jdouble(item,"volume");
|
|
if ( (trades= jarray(&n,item,"trades")) != 0 )
|
|
{
|
|
/*{
|
|
"plugin": "InstantDEX",
|
|
"method": "tradesequence",
|
|
"dotrade": 1,
|
|
"price": 0.00001858,
|
|
"volume": 484.39181916,
|
|
"trades": [
|
|
{
|
|
"basket": "bid",
|
|
"price": 0.00001858,
|
|
"volume": 484.39181916,
|
|
"group": 0,
|
|
"exchange": "bittrex",
|
|
"base": "NXT",
|
|
"rel": "BTC",
|
|
"trade": "sell",
|
|
"name": "NXT/BTC",
|
|
"orderprice": 0.00001858,
|
|
"ordervolume": 484.39181916
|
|
}
|
|
]
|
|
}*/
|
|
if ( n == 1 && is_cJSON_Array(jitem(trades,0)) != 0 )
|
|
{
|
|
//printf("NESTED ARRAY DETECTED\n");
|
|
trades = jitem(trades,0);
|
|
n = cJSON_GetArraySize(trades);
|
|
}
|
|
sprintf(bot->buf,"[%s %s%s %.8f %.4f] <- ",bot->iQ.s.isask != 0 ? "sell" : "buy ",bot->iQ.base,bot->iQ.rel,price,volume);
|
|
for (i=0; i<n; i++)
|
|
{
|
|
trade = jitem(trades,i);
|
|
sprintf(bot->buf+strlen(bot->buf),"[%s %s %.8f %.4f] ",jstr(trade,"exchange"),jstr(trade,"trade"),jdouble(trade,"orderprice"),jdouble(trade,"ordervolume"));
|
|
}
|
|
sprintf(bot->buf+strlen(bot->buf),"n.%d\n",n);
|
|
if ( bot->iQ.s.isask == 0 && bot->oppo != 0 && bot->price > 0. && bot->oppo->price > 0 )
|
|
{
|
|
//if ( bot->price < bot->oppo->price )
|
|
{
|
|
printf("%s%s%.8f -> %.8f = gain %.3f%%\n\n",bot->buf,bot->oppo->buf,bot->price,bot->oppo->price,(bot->oppo->price/bot->price - 1)*100);
|
|
}
|
|
}
|
|
}
|
|
//printf("%s: dir.%d price %.8f vol %f vs bot price %.8f vol %f\n",bot->name,dir,price,volume,bot->iQ.s.price,bot->iQ.s.vol);
|
|
//if ( (dir > 0 && price < bot->iQ.s.price) || (dir < 0 && price >= bot->iQ.s.price) )
|
|
return(1);
|
|
return(0);
|
|
}
|
|
|
|
int32_t tradebot_isvalidtrade(struct tradebot_info *bot,struct prices777_order *order,cJSON *retjson)
|
|
{
|
|
cJSON *array,*item; char *resultval; double balance,required; int32_t i,n,valid = 0;
|
|
if ( (array= jarray(&n,retjson,"traderesults")) != 0 )
|
|
{
|
|
for (i=0; i<n; i++)
|
|
{
|
|
item = jitem(array,i);
|
|
if ( jstr(item,"error") == 0 && (resultval= jstr(item,"success")) != 0 )
|
|
{
|
|
balance = jdouble(item,"balance");
|
|
required = jdouble(item,"required");
|
|
printf("[%s %f R%f] ",resultval,balance,required);
|
|
valid++;
|
|
}
|
|
}
|
|
//printf("valid.%d of %d\n",valid,n);
|
|
if ( valid == n )
|
|
return(0);
|
|
}
|
|
return(-1);
|
|
}
|
|
|
|
int32_t tradebot_tradedone(struct tradebot_info *bot,struct prices777_order *order)
|
|
{
|
|
struct pending_trade *pend;
|
|
if ( (pend= order->pend) != 0 && pend->finishtime != 0 )
|
|
return(1);
|
|
else return(0);
|
|
}
|
|
|
|
int32_t tradebot_haspending(struct tradebot_info *bot)
|
|
{
|
|
int32_t i,finished;
|
|
for (i=finished=0; i<bot->numtrades; i++)
|
|
{
|
|
if ( tradebot_tradedone(bot,&bot->trades[i]) > 0 )
|
|
finished++;
|
|
}
|
|
return(finished < bot->numtrades);
|
|
}
|
|
|
|
void tradebot_free(struct tradebot_info *bot)
|
|
{
|
|
int32_t i; struct pending_trade *pend;
|
|
for (i=0; i<bot->numtrades; i++)
|
|
{
|
|
if ( (pend= bot->trades[i].pend) != 0 )
|
|
free_pending(pend);
|
|
if ( bot->trades[i].retitem != 0 )
|
|
free_json(bot->trades[i].retitem);
|
|
if ( bot->cHandles[i] != 0 )
|
|
{
|
|
while ( bot->curlings[i] != 0 )
|
|
{
|
|
fprintf(stderr,"%s: wait for curlrequest[%d] to finish\n",bot->name,i);
|
|
sleep(3);
|
|
}
|
|
curlhandle_free(bot->cHandles[i]);
|
|
}
|
|
}
|
|
if ( bot->prevobookstr != 0 )
|
|
free(bot->prevobookstr);
|
|
free(bot);
|
|
}
|
|
|
|
void Tradebot_loop(void *ptr)
|
|
{
|
|
int32_t i,n,dotrade; char *obookstr,*retstr; cJSON *json,*array,*item,*retjson,*submit;
|
|
char jsonstr[1024]; struct tradebot_info *bot = ptr;
|
|
printf("START Tradebot.(%s)\n",bot->name);
|
|
while ( bot->finishtime == 0 && time(NULL) < bot->expiration )
|
|
{
|
|
if ( bot->startedtrades == 0 )
|
|
{
|
|
sprintf(jsonstr,"{\"allfields\":1,\"agent\":\"InstantDEX\",\"method\":\"orderbook\",\"exchange\":\"active\",\"base\":\"%s\",\"rel\":\"%s\"}",bot->iQ.base,bot->iQ.rel);
|
|
if ( (json= cJSON_Parse(jsonstr)) == 0 )
|
|
{
|
|
printf("cant parse.(%s)\n",jsonstr);
|
|
exit(-1);
|
|
}
|
|
obookstr = SuperNET_SNapi(bot->api,json,0,1);
|
|
//printf("GOT.(%s)\n",obookstr);
|
|
free_json(json);
|
|
if ( bot->prevobookstr == 0 || strcmp(obookstr,bot->prevobookstr) != 0 )
|
|
{
|
|
if ( bot->prevobookstr != 0 )
|
|
free(bot->prevobookstr);
|
|
bot->prevobookstr = obookstr;
|
|
//printf("UPDATE.(%s)\n",obookstr);
|
|
submit = 0;
|
|
if ( (json= cJSON_Parse(obookstr)) != 0 )
|
|
{
|
|
array = (bot->iQ.s.isask != 0) ? jarray(&n,json,"bids") : jarray(&n,json,"asks");
|
|
if ( array != 0 && n > 0 )
|
|
{
|
|
dotrade = 0;
|
|
for (i=0; i<1; i++)
|
|
{
|
|
item = jitem(array,i);
|
|
if ( tradebot_acceptable(bot,item) > 0 )
|
|
{
|
|
submit = cJSON_Duplicate(item,1);
|
|
if ( jobj(submit,"dotrade") == 0 )
|
|
jaddnum(submit,"dotrade",0);
|
|
else cJSON_ReplaceItemInObject(submit,"dotrade",cJSON_CreateNumber(0));
|
|
retstr = SuperNET_SNapi(bot->api,submit,0,1);
|
|
free_json(submit);
|
|
//retstr = InstantDEX_tradesequence(bot->curlings,bot,bot->cHandles,&bot->numtrades,bot->trades,(int32_t)( sizeof(bot->trades)/sizeof(*bot->trades)),dotrade,bot->NXTADDR,bot->NXTACCTSECRET,item);
|
|
if ( retstr != 0 )
|
|
{
|
|
if ( (retjson= cJSON_Parse(retstr)) != 0 )
|
|
{
|
|
if ( tradebot_isvalidtrade(bot,&bot->trades[i],retjson) > 0 )
|
|
bot->havetrade = 1;
|
|
free_json(retjson);
|
|
}
|
|
free(retstr);
|
|
if ( bot->havetrade == 0 )
|
|
continue;
|
|
}
|
|
}
|
|
break;
|
|
}
|
|
if ( 0 && submit != 0 && tradebot_havealltrades(bot) != 0 )
|
|
{
|
|
dotrade = 1;
|
|
cJSON_ReplaceItemInObject(submit,"dotrade",cJSON_CreateNumber(1));
|
|
bot->startedtrades = (uint32_t)time(NULL);
|
|
retstr = InstantDEX_tradesequence(bot->curlings,bot,bot->cHandles,&bot->numtrades,bot->trades,(int32_t)(sizeof(bot->trades)/sizeof(*bot->trades)),dotrade,bot->NXTADDR,bot->NXTACCTSECRET,item);
|
|
printf("TRADE RESULT.(%s)\n",retstr);
|
|
break;
|
|
}
|
|
}
|
|
free_json(json);
|
|
}
|
|
}
|
|
}
|
|
else if ( bot->startedtrades != 0 )
|
|
{
|
|
if ( tradebot_haspending(bot) > 0 && bot->finishtime == 0 )
|
|
bot->finishtime = (uint32_t)time(NULL);
|
|
}
|
|
usleep(5000000);
|
|
}
|
|
while ( tradebot_haspending(bot) != 0 )
|
|
sleep(60);
|
|
printf("FINISHED Tradebot.(%s) at %u finishtime.%u expiration.%u\n",bot->name,(uint32_t)time(NULL),bot->finishtime,bot->expiration);
|
|
tradebot_free(bot);
|
|
}
|
|
|
|
char *Tradebot_parser(cJSON *argjson,struct InstantDEX_quote *iQ,struct apitag_info *api)
|
|
{
|
|
char *submethod,*exchange; struct tradebot_info *bot,*oppobot;
|
|
printf("InstantDEX_tradebot.(%s) (%s/%s)\n",jprint(argjson,0),iQ->base,iQ->rel);
|
|
if ( (submethod= jstr(argjson,"submethod")) != 0 && (exchange= jstr(argjson,"exchange")) != 0 && strcmp(exchange,"active") == 0 && iQ != 0 )
|
|
{
|
|
if ( strcmp(submethod,"simplebot") == 0 )
|
|
{
|
|
if ( (bot= tradebot_compile(argjson,iQ,api)) == 0 )
|
|
return(clonestr("{\"error\":\"tradebot compiler error\"}"));
|
|
iQ->s.isask ^= 1;
|
|
if ( (oppobot= tradebot_compile(argjson,iQ,api)) == 0 )
|
|
return(clonestr("{\"error\":\"tradebot compiler error\"}"));
|
|
bot->oppo = oppobot;
|
|
oppobot->oppo = bot;
|
|
iguana_launch("bot",(void *)Tradebot_loop,bot);
|
|
iguana_launch("oppobot",(void *)Tradebot_loop,oppobot);
|
|
return(clonestr("{\"result\":\"tradebot started\"}"));
|
|
} else return(clonestr("{\"error\":\"unrecognized tradebot command\"}"));
|
|
return(clonestr("{\"result\":\"tradebot command processed\"}"));
|
|
} else return(clonestr("{\"error\":\"no prices777 or no tradebot submethod or not active exchange\"}"));
|
|
}
|
|
|
|
#endif
|
|
|