362 lines
14 KiB

/******************************************************************************
* Copyright © 2014-2018 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. *
* *
******************************************************************************/
#include "exchanges777.h"
#define TRADEBOTS_GAPTIME 60
struct tradebot_trade
{
double price,volume;
uint64_t orderid;
uint32_t started,finished,dir;
char exchangestr[32],base[32],rel[32];
};
struct tradebot_info
{
struct tradebot_info *next,*prev;
struct supernet_info *myinfo;
char name[128],exchangestr[32],base[32],rel[32];
int32_t dir,numtrades,estimatedtrades;
double price,volume,pricesum,totalvolume;
uint32_t dead,pause,started,expiration;
struct tradebot_trade trades[];
};
cJSON *tradebot_json(struct supernet_info *myinfo,struct exchange_info *exchange,struct tradebot_info *bot)
{
char str[65]; int32_t i,numpending; double pendsum,pendvolume,vol; cJSON *json,*array,*item;
json = cJSON_CreateObject();
jaddstr(json,"exchange",exchange->name);
jaddstr(json,"started",utc_str(str,bot->started));
if ( bot->pause != 0 )
jaddstr(json,"paused",utc_str(str,bot->pause));
if ( bot->dead != 0 )
jaddstr(json,"stopped",utc_str(str,bot->dead));
jaddstr(json,"base",bot->base);
jaddstr(json,"rel",bot->rel);
jaddstr(json,"type",bot->dir > 0 ? "buy" : "sell");
jaddnum(json,"price",bot->price);
jaddnum(json,"volume",bot->volume);
if ( (vol= bot->totalvolume) > SMALLVAL )
{
jaddnum(json,"aveprice",bot->pricesum/vol);
jaddnum(json,"totalvolume",vol);
}
array = cJSON_CreateArray();
for (pendsum=pendvolume=numpending=i=0; i<bot->numtrades; i++)
{
item = cJSON_CreateObject();
jadd64bits(item,"orderid",bot->trades[i].orderid);
jaddstr(item,"type",bot->trades[i].dir > 0 ? "buy" : "sell");
jaddstr(item,"base",bot->trades[i].base);
jaddstr(item,"rel",bot->trades[i].rel);
jaddnum(item,"price",bot->trades[i].price);
jaddnum(item,"volume",bot->trades[i].volume);
jaddstr(item,"started",utc_str(str,bot->trades[i].started));
if ( bot->trades[i].finished == 0 )
{
jaddnum(item,"elapsed",time(NULL) - bot->trades[i].started);
pendsum += bot->trades[i].price * bot->trades[i].volume;
pendvolume += bot->trades[i].volume;
numpending++;
} else jaddnum(item,"duration",bot->trades[i].finished - bot->trades[i].started);
jaddi(array,item);
}
jadd(json,"trades",array);
if ( (vol= pendvolume) > SMALLVAL )
{
jaddnum(json,"pending",numpending);
jaddnum(json,"pendingprice",pendsum/vol);
jaddnum(json,"pendingvolume",vol);
}
return(json);
}
struct tradebot_info *tradebot_find(struct supernet_info *myinfo,struct exchange_info *exchange,char *botname,cJSON *array,char *base,char *rel)
{
struct tradebot_info *tmp,*bot,*retbot = 0;
portable_mutex_lock(&exchange->mutexT);
DL_FOREACH_SAFE(exchange->tradebots,bot,tmp)
{
if ( botname != 0 && strcmp(botname,bot->name) == 0 )
retbot = bot;
if ( array != 0 )
jaddi(array,tradebot_json(myinfo,exchange,bot));
if ( base != 0 && rel != 0 && strcmp(base,bot->base) == 0 && strcmp(rel,bot->rel) == 0 )
retbot = bot;
}
portable_mutex_unlock(&exchange->mutexT);
return(retbot);
}
void tradebot_add(struct exchange_info *exchange,struct tradebot_info *bot)
{
portable_mutex_lock(&exchange->mutexT);
DL_APPEND(exchange->tradebots,bot);
portable_mutex_unlock(&exchange->mutexT);
}
struct tradebot_info *tradebot_create(struct supernet_info *myinfo,struct exchange_info *exchange,char *base,char *rel,int32_t dir,double price,double volume,int32_t duration)
{
struct tradebot_info *bot;
if ( (bot= calloc(1,sizeof(*bot))) != 0 )
{
bot->myinfo = myinfo;
safecopy(bot->exchangestr,exchange->name,sizeof(bot->exchangestr));
safecopy(bot->base,base,sizeof(bot->base));
safecopy(bot->rel,rel,sizeof(bot->rel));
bot->dir = dir, bot->price = price, bot->volume = volume;
bot->started = (uint32_t)time(NULL);
if ( duration < 1 )
duration = 1;
bot->expiration = bot->started + duration;
bot->estimatedtrades = (duration / TRADEBOTS_GAPTIME) + 1;
sprintf(bot->name,"%s_%s_%s.%d",exchange->name,base,rel,bot->started);
tradebot_add(exchange,bot);
}
return(bot);
}
struct tradebot_trade *tradebot_issuetrade(struct exchange_info *exchange,struct tradebot_info *bot,char *base,char *rel,double price,double volume,int32_t dir)
{
struct tradebot_trade *tr; char *str; int32_t maxseconds = 30,dotrade = 1;
bot = realloc(bot,sizeof(*bot) + (bot->numtrades + 1) * sizeof(bot->trades[0]));
tr = &bot->trades[bot->numtrades++];
memset(tr,0,sizeof(*tr));
tr->price = price, tr->volume = volume, tr->dir = dir;
safecopy(tr->exchangestr,exchange->name,sizeof(tr->exchangestr));
safecopy(tr->base,base,sizeof(tr->base));
safecopy(tr->rel,rel,sizeof(tr->rel));
if ( (str= exchanges777_Qtrade(exchange,base,rel,maxseconds,dotrade,dir,price,volume,0)) != 0 )
free(str);
return(tr);
}
void tradebot_timeslice(struct exchange_info *exchange,void *_bot)
{
double volume; struct tradebot_info *bot = _bot;
if ( time(NULL) < bot->expiration && bot->dead == 0 )
{
if ( bot->pause == 0 )
{
if ( bot->numtrades == 0 || bot->trades[bot->numtrades-1].finished != 0 || time(NULL) > bot->trades[bot->numtrades-1].started+60 )
{
if ( bot->estimatedtrades > 0 )
{
volume = bot->volume / bot->estimatedtrades;
tradebot_issuetrade(exchange,bot,bot->base,bot->rel,bot->price,volume,bot->dir);
}
}
}
}
else
{
DL_DELETE(exchange->tradebots,bot);
free(bot);
}
}
void tradebot_timeslices(struct exchange_info *exchange)
{
struct tradebot_info *bot,*tmp;
portable_mutex_lock(&exchange->mutexT);
DL_FOREACH_SAFE(exchange->tradebots,bot,tmp)
{
tradebot_timeslice(exchange,bot);
}
portable_mutex_unlock(&exchange->mutexT);
}
char *tradebot_launch(struct supernet_info *myinfo,char *exchangestr,char *base,char *rel,int32_t dir,double price,double volume,int32_t duration,char *remoteaddr,cJSON *json)
{
struct exchange_info *exchange; char retbuf[1024]; struct tradebot_info *bot;
if ( remoteaddr == 0 )
{
if ( (exchange= exchanges777_info(exchangestr,1,json,remoteaddr)) != 0 )
{
if ( (bot= tradebot_create(myinfo,exchange,base,rel,1,price,fabs(volume),duration)) != 0 )
{
sprintf(retbuf,"{\"result\":\"tradebot created\",\"botid\":\"%s\"}",bot->name);
return(clonestr(retbuf));
}
else return(clonestr("{\"error\":\"couldnt create exchange tradebot\"}"));
} else return(clonestr("{\"error\":\"couldnt find/create exchange info\"}"));
} else return(clonestr("{\"error\":\"tradebots only local usage!\"}"));
}
char *tradebot_control(struct supernet_info *myinfo,char *exchangestr,char *botid,int32_t control,char *remoteaddr,cJSON *json)
{
struct tradebot_info *bot; struct exchange_info *exchange;
if ( remoteaddr == 0 )
{
if ( (exchange= exchanges777_info(exchangestr,1,json,remoteaddr)) != 0 )
{
if ( (bot= tradebot_find(myinfo,exchange,botid,0,0,0)) != 0 )
{
if ( control > 1 )
bot->pause = (uint32_t)time(NULL);
else if ( control == 0 )
bot->pause = 0;
else bot->dead = (uint32_t)time(NULL);
return(clonestr("{\"result\":\"ask bot to pause\"}"));
} else return(clonestr("{\"error\":\"cant find tradebot\"}"));
}
else return(clonestr("{\"error\":\"couldnt find/create exchange info\"}"));
} else return(clonestr("{\"error\":\"tradebots only local usage!\"}"));
}
#include "../includes/iguana_apidefs.h"
#include "../includes/iguana_apideclares.h"
#include "../includes/iguana_apideclares2.h"
HASH_ARRAY_STRING(tradebot,liquidity,hash,vals,targetcoin)
{
tradebot_liquidity_command(myinfo,targetcoin,hash,vals);
return(clonestr("{\"result\":\"targetcoin updated\"}"));
}
STRING_ARG(tradebot,amlp,blocktrail)
{
myinfo->IAMLP = 1;
if ( blocktrail != 0 )
safecopy(myinfo->blocktrail_apikey,blocktrail,sizeof(myinfo->blocktrail_apikey));
return(clonestr("{\"result\":\"liquidity provider active\"}"));
}
ZERO_ARGS(tradebot,notlp)
{
myinfo->IAMLP = 0;
return(clonestr("{\"result\":\"not liquidity provider\"}"));
}
THREE_STRINGS_AND_DOUBLE(tradebot,monitor,exchange,base,rel,commission)
{
int32_t allfields = 1,depth = 50; struct exchange_info *ptr;
if ( remoteaddr == 0 )
{
if ( (ptr= exchanges777_info(exchange,1,json,remoteaddr)) != 0 )
return(exchanges777_Qprices(ptr,base,rel,30,allfields,depth,json,1,commission * .01));
else return(clonestr("{\"error\":\"couldnt find/create exchange info\"}"));
} else return(clonestr("{\"error\":\"tradebots only local usage!\"}"));
}
STRING_AND_DOUBLE(tradebot,monitorall,exchange,commission)
{
int32_t i,n,allfields = 1,depth = 50; cJSON *arg,*array,*item; char *base,*rel,*str,*str2; struct exchange_info *ptr;
if ( remoteaddr == 0 )
{
if ( (ptr= exchanges777_info(exchange,1,json,remoteaddr)) != 0 )
{
if ( (str= InstantDEX_allpairs(myinfo,0,json,remoteaddr,exchange)) != 0 )
{
if ( (arg= cJSON_Parse(str)) != 0 )
{
if ( (array= jarray(&n,arg,"result")) != 0 )
{
for (i=0; i<n; i++)
{
item = jitem(array,i);
if ( is_cJSON_Array(item) != 0 && cJSON_GetArraySize(item) == 2 )
{
base = jstr(jitem(item,0),0);
rel = jstr(jitem(item,1),0);
if ( base != 0 && rel != 0 && (str2= exchanges777_Qprices(ptr,base,rel,30,allfields,depth,json,1,commission * .01)) != 0 )
{
printf("%s/%s ",base,rel);
free(str2);
}
}
}
}
free_json(arg);
}
free(str);
}
return(clonestr("{\"result\":\"monitorall started\"}"));
}
else return(clonestr("{\"error\":\"couldnt find/create exchange info\"}"));
} else return(clonestr("{\"error\":\"tradebots only local usage!\"}"));
}
THREE_STRINGS(tradebot,unmonitor,exchange,base,rel)
{
struct exchange_info *ptr;
if ( remoteaddr == 0 )
{
if ( (ptr= exchanges777_info(exchange,1,json,remoteaddr)) != 0 )
return(exchanges777_unmonitor(ptr,base,rel));
else return(clonestr("{\"error\":\"couldnt find/create exchange info\"}"));
} else return(clonestr("{\"error\":\"tradebots only local usage!\"}"));
}
THREE_STRINGS_AND_THREE_DOUBLES(tradebot,accumulate,exchange,base,rel,price,volume,duration)
{
return(tradebot_launch(myinfo,exchange,base,rel,1,price,volume,duration,remoteaddr,json));
}
THREE_STRINGS_AND_THREE_DOUBLES(tradebot,divest,exchange,base,rel,price,volume,duration)
{
return(tradebot_launch(myinfo,exchange,base,rel,-1,price,volume,duration,remoteaddr,json));
}
TWO_STRINGS(tradebot,status,exchange,botid)
{
struct tradebot_info *bot; struct exchange_info *ptr;
if ( remoteaddr == 0 )
{
if ( (ptr= exchanges777_info(exchange,1,json,remoteaddr)) != 0 )
{
if ( (bot= tradebot_find(myinfo,ptr,botid,0,0,0)) != 0 )
return(jprint(tradebot_json(myinfo,ptr,bot),1));
}
}
return(clonestr("{\"error\":\"tradebots only local usage!\"}"));
}
STRING_ARG(tradebot,activebots,exchange)
{
struct exchange_info *ptr; cJSON *retjson,*array = cJSON_CreateArray();
if ( remoteaddr == 0 )
{
if ( (ptr= exchanges777_info(exchange,1,json,remoteaddr)) != 0 )
{
tradebot_find(myinfo,ptr,0,array,0,0);
retjson = cJSON_CreateObject();
jaddstr(retjson,"result","success");
jadd(retjson,"tradebots",array);
return(jprint(retjson,1));
}
}
return(clonestr("{\"error\":\"tradebots only local usage!\"}"));
}
TWO_STRINGS(tradebot,pause,exchange,botid)
{
return(tradebot_control(myinfo,exchange,botid,1,remoteaddr,json));
}
TWO_STRINGS(tradebot,stop,exchange,botid)
{
return(tradebot_control(myinfo,exchange,botid,-1,remoteaddr,json));
}
TWO_STRINGS(tradebot,resume,exchange,botid)
{
return(tradebot_control(myinfo,exchange,botid,0,remoteaddr,json));
}
#include "../includes/iguana_apiundefs.h"