381 lines
15 KiB
381 lines
15 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. *
|
|
* *
|
|
******************************************************************************/
|
|
#ifdef later
|
|
|
|
#ifdef DEFINES_ONLY
|
|
#ifndef tourney777_h
|
|
#define tourney777_h
|
|
|
|
// nonplaying nodeA
|
|
|
|
#include <stdio.h>
|
|
#include <stdint.h>
|
|
#include <stdlib.h>
|
|
#include <memory.h>
|
|
#include "../iguana777.h"
|
|
|
|
struct tourney777_table { uint64_t tableid; union hostnet777 *hn[CARDS777_MAXPLAYERS]; int32_t N,numactive; };
|
|
|
|
struct tourney777
|
|
{
|
|
struct tourney777_table tables[1024]; char transport[16],ipaddr[64],name[128]; uint32_t started,finished,numtables; uint16_t port;
|
|
union hostnet777 hn[];
|
|
} *Tournament;
|
|
|
|
#define TOURNEY777_TABLESIZE 9
|
|
#define TOURNEY777_MINPLAYERS (TOURNEY777_TABLESIZE - 2)
|
|
|
|
#endif
|
|
#else
|
|
#ifndef tourney777_c
|
|
#define tourney777_c
|
|
|
|
#ifndef tourney777_h
|
|
#define DEFINES_ONLY
|
|
#include "tourney777.c"
|
|
#undef DEFINES_ONLY
|
|
#endif
|
|
|
|
void tourney777_jointable(struct tourney777 *tp,struct tourney777_table *table,union hostnet777 *player)
|
|
{
|
|
player->client->tableid = table->tableid;
|
|
printf("jointable.%llu tableid.%llu\n",(long long)player->client->H.nxt64bits,(long long)table->tableid);
|
|
}
|
|
|
|
void tourney777_leavetable(struct tourney777 *tp,struct tourney777_table *table,union hostnet777 *player)
|
|
{
|
|
printf("leavetable.%llu tableid.%llu\n",(long long)player->client->H.nxt64bits,(long long)table->tableid);
|
|
player->client->tableid = 0;
|
|
}
|
|
|
|
void tourney777_createtable(struct tourney777 *tp,union hostnet777 *players[],int32_t num)
|
|
{
|
|
int32_t i; uint64_t tableid = rand();
|
|
for (i=0; i<num; i++)
|
|
players[i]->client->tableid = tableid;
|
|
printf("createtable.%llu numplayers.%d\n",(long long)tableid,num);
|
|
}
|
|
|
|
void tourney777_freetable(struct tourney777 *tp,struct tourney777_table *table)
|
|
{
|
|
printf("freetable.%llu numplayers.%d numactives.%d\n",(long long)table->tableid,table->N,table->numactive);
|
|
}
|
|
|
|
struct tourney777_table *tourney777_findtable(struct tourney777 *tp,uint64_t tableid,struct tourney777_table *avoid)
|
|
{
|
|
struct tourney777_table *mintable = 0; int32_t i,minplayers = CARDS777_MAXPLAYERS + 1;
|
|
if ( tp->numtables > 0 )
|
|
{
|
|
for (i=0; i<tp->numtables; i++)
|
|
{
|
|
if ( tableid == 0 )
|
|
{
|
|
if ( (avoid == 0 || avoid != &tp->tables[i]) && tp->tables[i].numactive != TOURNEY777_TABLESIZE && tp->tables[i].numactive < minplayers )
|
|
{
|
|
minplayers = tp->tables[i].numactive;
|
|
mintable = &tp->tables[i];
|
|
}
|
|
}
|
|
else if ( tp->tables[i].tableid == tableid )
|
|
return(&tp->tables[i]);
|
|
}
|
|
}
|
|
return(mintable);
|
|
}
|
|
|
|
union hostnet777 *tourney777_nextblind(struct tourney777_table *table)
|
|
{
|
|
int32_t i,ind,j = (rand() % table->N);
|
|
for (i=0; i<table->N; i++)
|
|
{
|
|
ind = (j + i) % table->N;
|
|
if ( table->hn[ind] != 0 && table->hn[ind]->client->balance != 0 )
|
|
return(table->hn[ind]);
|
|
}
|
|
return(0);
|
|
}
|
|
|
|
void tourney777_rebalance(struct tourney777 *tp,int32_t delta)
|
|
{
|
|
int32_t i,j,num,n,pertable,flag,threshold,needs_table = 0; struct hostnet777_server *srv;
|
|
union hostnet777 *player,*newtable[TOURNEY777_TABLESIZE]; struct tourney777_table *table,*t;
|
|
srv = tp->hn[0].server;
|
|
if ( delta < 0 )
|
|
{
|
|
if ( (table= tourney777_findtable(tp,0,0)) != 0 )
|
|
{
|
|
if ( table->numactive < TOURNEY777_MINPLAYERS )
|
|
{
|
|
if ( tp->numtables >= table->numactive )
|
|
{
|
|
for (j=0; j<table->N; j++)
|
|
tourney777_leavetable(tp,table,table->hn[j]);
|
|
for (j=0; j<table->N; j++)
|
|
{
|
|
if ( (t= tourney777_findtable(tp,0,table)) != 0 )
|
|
tourney777_jointable(tp,t,table->hn[j]);
|
|
else printf("tourney777_rebalance: cant find table with slot\n");
|
|
}
|
|
tourney777_freetable(tp,table);
|
|
}
|
|
else
|
|
{
|
|
printf("tourney777_rebalance: imbalance tableid.%llu numactive.%d vs total tables.%d\n",(long long)table->tableid,table->numactive,tp->numtables);
|
|
flag = 0;
|
|
for (threshold=TOURNEY777_TABLESIZE; threshold>=TOURNEY777_MINPLAYERS+1; threshold--)
|
|
{
|
|
for (i=0; i<tp->numtables; i++)
|
|
{
|
|
if ( tp->tables[i].numactive == threshold )
|
|
{
|
|
if ( (player= tourney777_nextblind(&tp->tables[i])) != 0 )
|
|
{
|
|
tourney777_leavetable(tp,&tp->tables[i],player);
|
|
tourney777_jointable(tp,table,player);
|
|
flag = 1;
|
|
break;
|
|
}
|
|
else printf("tourney777_nextblind: cant find nextblind\n");
|
|
}
|
|
}
|
|
if ( flag != 0 )
|
|
break;
|
|
}
|
|
if ( flag == 0 )
|
|
{
|
|
printf("tourney777_rebalance: couldnt find donor tableid.%llu numactive.%d vs total tables.%d\n",(long long)table->tableid,table->numactive,tp->numtables);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
for (i=1; i<srv->num; i++)
|
|
{
|
|
player = &tp->hn[i];
|
|
if ( player != 0 && player->client->tableid == 0 && player->client->balance != 0 )
|
|
{
|
|
if ( (table= tourney777_findtable(tp,0,0)) != 0 )
|
|
tourney777_jointable(tp,table,player);
|
|
else needs_table++;
|
|
}
|
|
}
|
|
if ( needs_table >= TOURNEY777_TABLESIZE )
|
|
{
|
|
num = (needs_table / TOURNEY777_TABLESIZE);
|
|
if ( (needs_table % TOURNEY777_TABLESIZE) != 0 )
|
|
num++;
|
|
pertable = (needs_table / num);
|
|
printf("needs_table.%d pertable.%d num.%d\n",needs_table,pertable,num);
|
|
for (j=0; j<num; j++)
|
|
{
|
|
memset(newtable,0,sizeof(newtable));
|
|
// find nodes with endpoints and verify connections
|
|
for (i=1,n=0; i<srv->num; i++)
|
|
{
|
|
player = &tp->hn[i];
|
|
if ( player != 0 && player->client->tableid == 0 && player->client->balance != 0 )
|
|
{
|
|
newtable[n++] = player;
|
|
if ( n >= pertable )
|
|
break;
|
|
}
|
|
}
|
|
tourney777_createtable(tp,newtable,n);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void tourney777_newhand(union hostnet777 *hn,uint64_t tableid,cJSON *json,uint8_t *data,int32_t datalen)
|
|
{
|
|
struct pangea_info *pangea_find(uint64_t tableid,int32_t threadid);
|
|
char *pangea_dispsummary(struct pangea_info *sp,int32_t verbose,uint8_t *summary,int32_t summarysize,uint64_t tableid,int32_t handid,int32_t numplayers);
|
|
char *handhist; int32_t i,j,handid,numplayers,n,m,busted,rebuy; cJSON *handjson,*array,*balances,*item; struct tourney777_table *table;
|
|
handid = juint(json,"handid");
|
|
numplayers = juint(json,"numplayers");
|
|
if ( (handhist= pangea_dispsummary(pangea_find(tableid,0),1,data,datalen,tableid,handid,numplayers)) != 0 )
|
|
{
|
|
printf("GOT HANDHIST.(%s)\n",handhist);
|
|
if ( (handjson= cJSON_Parse(handhist)) != 0 )
|
|
{
|
|
if ( (array= jarray(&n,handjson,"hand")) != 0 )
|
|
{
|
|
for (i=0; i<n; i++)
|
|
{
|
|
item = jitem(array,i);
|
|
rebuy = juint(item,"rebuy");
|
|
busted = juint(item,"busted");
|
|
if ( (balances= jarray(&m,item,"balances")) != 0 && (table= tourney777_findtable(Tournament,tableid,0)) != 0 )
|
|
{
|
|
for (j=0; j<m; j++)
|
|
table->hn[j]->client->balance = j64bits(jitem(balances,j),0);
|
|
}
|
|
if ( busted != 0 )
|
|
tourney777_rebalance(Tournament,-1);
|
|
}
|
|
}
|
|
free_json(handjson);
|
|
}
|
|
free(handhist);
|
|
}
|
|
}
|
|
|
|
void tourney777_poll(union hostnet777 *hn)
|
|
{
|
|
char *jsonstr; uint64_t senderbits,tableid; uint8_t *buf=0; int32_t maxlen,len,senderind; uint32_t timestamp; char *cmdstr,*hexstr; cJSON *json;
|
|
maxlen = 65536;
|
|
if ( (buf= malloc(maxlen)) == 0 )
|
|
{
|
|
printf("tourney777_poll: cant allocate buf\n");
|
|
return;
|
|
}
|
|
if ( (jsonstr= queue_dequeue(&hn->server->H.Q,1)) != 0 )
|
|
{
|
|
printf("tourney slot.%d GOT.(%s)\n",hn->client->H.slot,jsonstr);
|
|
if ( (json= cJSON_Parse(jsonstr)) != 0 )
|
|
{
|
|
tableid = j64bits(json,"tableid");
|
|
if ( tourney777_findtable(Tournament,tableid,0) == 0 )
|
|
{
|
|
free(buf);
|
|
return;
|
|
}
|
|
senderbits = j64bits(json,"sender");
|
|
if ( (senderind= juint(json,"myind")) < 0 || senderind >= hn->server->num )
|
|
{
|
|
printf("pangea_poll: illegal senderind.%d cardi.%d turni.%d\n",senderind,juint(json,"cardi"),juint(json,"turni"));
|
|
goto cleanup;
|
|
}
|
|
timestamp = juint(json,"timestamp");
|
|
hn->client->H.state = juint(json,"state");
|
|
len = juint(json,"n");
|
|
cmdstr = jstr(json,"cmd");
|
|
if ( (hexstr= jstr(json,"data")) != 0 && strlen(hexstr) == (len<<1) )
|
|
{
|
|
if ( len > maxlen )
|
|
{
|
|
printf("len too big for tourney777_poll\n");
|
|
goto cleanup;
|
|
}
|
|
decode_hex(buf,len,hexstr);
|
|
} else if ( hexstr != 0 )
|
|
printf("len.%d vs hexlen.%d (%s)\n",len,(int32_t)strlen(hexstr)>>1,hexstr);
|
|
if ( cmdstr != 0 )
|
|
{
|
|
if ( strcmp(cmdstr,"newhand") == 0 )
|
|
tourney777_newhand(hn,tableid,json,buf,len);
|
|
}
|
|
cleanup:
|
|
free_json(json);
|
|
}
|
|
free_queueitem(jsonstr);
|
|
}
|
|
free(buf);
|
|
}
|
|
|
|
char *tourney777_start(char *name,cJSON *json)
|
|
{
|
|
if ( Tournament != 0 && strcmp(Tournament->name,name) == 0 )
|
|
{
|
|
Tournament->started = (uint32_t)time(NULL);
|
|
tourney777_rebalance(Tournament,1);
|
|
return(clonestr("{\"result\":\"tournament started\"}"));
|
|
} else return(clonestr("{\"error\":\"no matching tournament\"}"));
|
|
}
|
|
|
|
char *tourney777_register(char *name,bits256 pubkey,cJSON *json)
|
|
{
|
|
int32_t i; struct hostnet777_server *srv;
|
|
if ( Tournament != 0 && strcmp(Tournament->name,name) == 0 && (srv= Tournament->hn[0].server) != 0 )
|
|
{
|
|
if ( Tournament->started != 0 )
|
|
return(clonestr("{\"error\":\"tournament already started\"}"));
|
|
else
|
|
{
|
|
for (i=0; i<srv->num; i++)
|
|
if ( memcmp(pubkey.bytes,srv->clients[i].pubkey.bytes,sizeof(bits256)) == 0 )
|
|
return(clonestr("{\"error\":\"already registered in tournament\"}"));
|
|
srv->clients[srv->num].lastcontact = (uint32_t)time(NULL);
|
|
// get endpoint
|
|
srv->num++;
|
|
return(clonestr("{\"result\":\"registered in tournament\"}"));
|
|
}
|
|
} else return(clonestr("{\"error\":\"no matching tournament\"}"));
|
|
}
|
|
|
|
char *tourney777_deregister(char *name,bits256 pubkey,cJSON *json)
|
|
{
|
|
int32_t i; struct hostnet777_server *srv;
|
|
if ( Tournament != 0 && strcmp(Tournament->name,name) == 0 && (srv= Tournament->hn[0].server) != 0 )
|
|
{
|
|
if ( Tournament->started != 0 )
|
|
return(clonestr("{\"error\":\"tournament already started\"}"));
|
|
else
|
|
{
|
|
for (i=0; i<srv->num; i++)
|
|
if ( memcmp(pubkey.bytes,srv->clients[i].pubkey.bytes,sizeof(bits256)) == 0 )
|
|
{
|
|
memset(&srv->clients[i],0,sizeof(srv->clients[i]));
|
|
if ( i == srv->num-1 )
|
|
srv->num--;
|
|
return(clonestr("{\"result\":\"deregistered from tournament\"}"));
|
|
}
|
|
return(clonestr("{\"error\":\"not registered in tournament\"}"));
|
|
}
|
|
} else return(clonestr("{\"error\":\"no matching tournament\"}"));
|
|
}
|
|
|
|
struct tourney777 *tourney777_init(char *name,bits256 privkey,char *transport,char *ipaddr,uint16_t port,int32_t maxplayers,cJSON *json)
|
|
{
|
|
struct hostnet777_server *srv; bits256 pubkey; struct tourney777 *tourney; int32_t i; char endpoint[128];
|
|
tourney = calloc(1,sizeof(*tourney) + sizeof(*tourney->hn)*maxplayers);
|
|
pubkey = acct777_pubkey(privkey);
|
|
safecopy(tourney->name,name,sizeof(tourney->name)-1);
|
|
if ( transport == 0 || transport[0] == 0 )
|
|
transport = "tcp";
|
|
strcpy(tourney->transport,transport);
|
|
if ( ipaddr == 0 || ipaddr[0] == 0 )
|
|
ipaddr = "127.0.0.1";
|
|
strcpy(tourney->ipaddr,ipaddr);
|
|
if ( port == 0 )
|
|
port = 8897;
|
|
tourney->port = port;
|
|
if ( (srv= hostnet777_server(privkey,pubkey,tourney->transport,tourney->ipaddr,tourney->port,maxplayers)) == 0 )
|
|
{
|
|
printf("tourney777_init: cant create hostnet777 server\n");
|
|
return(0);
|
|
}
|
|
srv->H.privkey = privkey, srv->H.pubkey = pubkey;
|
|
for (i=0; i<maxplayers; i++)
|
|
{
|
|
sprintf(endpoint,"%s://%s:%u",srv->ep.transport,srv->ep.ipaddr,srv->ep.port + i + 1);
|
|
srv->clients[i].pmsock = nn_createsocket(endpoint,1,"NN_PULL",NN_PULL,srv->ep.port + i + 1,10,10);
|
|
}
|
|
srv->H.pollfunc = tourney777_poll;
|
|
tourney->hn[0].server = srv;
|
|
// set tournament parameters
|
|
if ( portable_thread_create((void *)hostnet777_idler,&tourney->hn[0]) == 0 )
|
|
printf("error launching server thread\n");
|
|
Tournament = tourney;
|
|
return(tourney);
|
|
}
|
|
|
|
|
|
#endif
|
|
#endif
|
|
|
|
#endif
|
|
|