|
|
|
/******************************************************************************
|
|
|
|
* Copyright © 2014-2016 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 "iguana777.h"
|
|
|
|
#include "../includes/tweetnacl.h"
|
|
|
|
|
|
|
|
bits256 SuperNET_sharedseed(bits256 privkey,bits256 otherpub)
|
|
|
|
{
|
|
|
|
bits256 seed2,seed;
|
|
|
|
seed = curve25519_shared(privkey,otherpub);
|
|
|
|
vcalc_sha256(0,seed2.bytes,seed.bytes,sizeof(bits256));
|
|
|
|
return(seed2);
|
|
|
|
}
|
|
|
|
|
|
|
|
int32_t SuperNET_delaymillis(struct supernet_info *myinfo,int32_t maxdelay)
|
|
|
|
{
|
|
|
|
maxdelay += myinfo->maxdelay;
|
|
|
|
if ( maxdelay > SUPERNET_MAXDELAY )
|
|
|
|
maxdelay = SUPERNET_MAXDELAY;
|
|
|
|
if ( maxdelay == 0 )
|
|
|
|
return(0);
|
|
|
|
return(rand() % maxdelay);
|
|
|
|
}
|
|
|
|
|
|
|
|
void SuperNET_remotepeer(struct supernet_info *myinfo,struct iguana_info *coin,char *symbol,char *ipaddr,int32_t supernetflag)
|
|
|
|
{
|
|
|
|
uint64_t ipbits; struct iguana_peer *addr;
|
|
|
|
ipbits = calc_ipbits(ipaddr);
|
|
|
|
printf("got %s remotepeer.(%s) supernet.%d\n",symbol,ipaddr,supernetflag);
|
|
|
|
if ( supernetflag != 0 && (uint32_t)myinfo->myaddr.selfipbits != (uint32_t)ipbits )
|
|
|
|
{
|
|
|
|
if ( (addr= iguana_peerslot(coin,ipbits)) != 0 )
|
|
|
|
{
|
|
|
|
printf("launch startconnection to supernet peer.(%s)\n",ipaddr);
|
|
|
|
iguana_launch(coin,"connection",iguana_startconnection,addr,IGUANA_CONNTHREAD);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
iguana_possible_peer(coin,ipaddr);
|
|
|
|
}
|
|
|
|
|
|
|
|
int32_t SuperNET_confirmip(struct supernet_info *myinfo,uint32_t ipbits)
|
|
|
|
{
|
|
|
|
int32_t i,j,total = 0; uint32_t x;
|
|
|
|
for (i=0; i<IGUANA_MAXCOINS; i++)
|
|
|
|
{
|
|
|
|
if ( Coins[i] != 0 )
|
|
|
|
{
|
|
|
|
for (j=0; j<IGUANA_MAXPEERS; j++)
|
|
|
|
{
|
|
|
|
if ( (x= Coins[i]->peers.active[j].myipbits) != 0 )
|
|
|
|
{
|
|
|
|
if ( x == ipbits )
|
|
|
|
total++;
|
|
|
|
else total--;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return(total);
|
|
|
|
}
|
|
|
|
|
|
|
|
void SuperNET_myipaddr(struct supernet_info *myinfo,struct iguana_info *coin,struct iguana_peer *addr,char *myipaddr,char *remoteaddr)
|
|
|
|
{
|
|
|
|
uint32_t myipbits = (uint32_t)calc_ipbits(myipaddr);
|
|
|
|
if ( addr->myipbits == 0 )
|
|
|
|
addr->myipbits = myipbits;
|
|
|
|
else if ( addr->myipbits != myipbits )
|
|
|
|
{
|
|
|
|
printf("%s: myipaddr conflict %x != %x?\n",addr->ipaddr,addr->myipbits,myipbits);
|
|
|
|
addr->myipbits = 0;
|
|
|
|
}
|
|
|
|
if ( addr->myipbits != 0 && myinfo->myaddr.myipbits == 0 )
|
|
|
|
myinfo->myaddr.myipbits = addr->myipbits;
|
|
|
|
if ( addr->myipbits == myinfo->myaddr.myipbits )
|
|
|
|
myinfo->myaddr.confirmed++;
|
|
|
|
else myinfo->myaddr.confirmed--;
|
|
|
|
if ( (myinfo->myaddr.totalconfirmed= SuperNET_confirmip(myinfo,addr->myipbits)) >= coin->peers.numranked )
|
|
|
|
myinfo->myaddr.selfipbits = addr->myipbits;
|
|
|
|
if ( myinfo->myaddr.selfipbits == myinfo->myaddr.myipbits )
|
|
|
|
{
|
|
|
|
expand_ipbits(myinfo->ipaddr,myinfo->myaddr.selfipbits);
|
|
|
|
vcalc_sha256(0,myinfo->myaddr.iphash.bytes,(uint8_t *)&myinfo->myaddr.selfipbits,sizeof(myinfo->myaddr.selfipbits));
|
|
|
|
}
|
|
|
|
//printf("myipaddr.%s self.%x your.%x\n",myinfo->ipaddr,myinfo->myaddr.selfipbits,myinfo->myaddr.myipbits);
|
|
|
|
}
|
|
|
|
|
|
|
|
int32_t _SuperNET_cipher(uint8_t nonce[crypto_box_NONCEBYTES],uint8_t *cipher,uint8_t *message,int32_t len,bits256 destpub,bits256 srcpriv,uint8_t *buf)
|
|
|
|
{
|
|
|
|
memset(cipher,0,len+crypto_box_ZEROBYTES);
|
|
|
|
memset(buf,0,crypto_box_ZEROBYTES);
|
|
|
|
memcpy(buf+crypto_box_ZEROBYTES,message,len);
|
|
|
|
crypto_box(cipher,buf,len+crypto_box_ZEROBYTES,nonce,destpub.bytes,srcpriv.bytes);
|
|
|
|
return(len + crypto_box_ZEROBYTES);
|
|
|
|
}
|
|
|
|
|
|
|
|
uint8_t *_SuperNET_decipher(uint8_t nonce[crypto_box_NONCEBYTES],uint8_t *cipher,uint8_t *message,int32_t len,bits256 srcpub,bits256 mypriv)
|
|
|
|
{
|
|
|
|
int32_t err;
|
|
|
|
if ( (err= crypto_box_open(message,cipher,len,nonce,srcpub.bytes,mypriv.bytes)) == 0 )
|
|
|
|
{
|
|
|
|
message += crypto_box_ZEROBYTES;
|
|
|
|
len -= crypto_box_ZEROBYTES;
|
|
|
|
return(message);
|
|
|
|
}
|
|
|
|
return(0);
|
|
|
|
}
|
|
|
|
|
|
|
|
void *SuperNET_deciphercalc(void **ptrp,int32_t *msglenp,bits256 privkey,bits256 srcpubkey,uint8_t *cipher,int32_t cipherlen,uint8_t *buf,int32_t bufsize)
|
|
|
|
{
|
|
|
|
uint8_t *origptr,*nonce,*message; void *retptr;
|
|
|
|
if ( bits256_nonz(privkey) == 0 )
|
|
|
|
privkey = GENESIS_PRIVKEY;
|
|
|
|
*ptrp = 0;
|
|
|
|
if ( cipherlen > bufsize )
|
|
|
|
{
|
|
|
|
message = calloc(1,cipherlen);
|
|
|
|
*ptrp = (void *)message;
|
|
|
|
}
|
|
|
|
else message = buf;
|
|
|
|
origptr = cipher;
|
|
|
|
if ( bits256_nonz(srcpubkey) == 0 )
|
|
|
|
{
|
|
|
|
memcpy(srcpubkey.bytes,cipher,sizeof(srcpubkey));
|
|
|
|
//char str[65]; printf("use attached pubkey.(%s)\n",bits256_str(str,srcpubkey));
|
|
|
|
cipher += sizeof(srcpubkey);
|
|
|
|
cipherlen -= sizeof(srcpubkey);
|
|
|
|
}
|
|
|
|
nonce = cipher;
|
|
|
|
cipher += crypto_box_NONCEBYTES, cipherlen -= crypto_box_NONCEBYTES;
|
|
|
|
*msglenp = cipherlen - crypto_box_ZEROBYTES;
|
|
|
|
if ( (retptr= _SuperNET_decipher(nonce,cipher,message,cipherlen,srcpubkey,privkey)) == 0 )
|
|
|
|
{
|
|
|
|
*msglenp = -1;
|
|
|
|
free(*ptrp);
|
|
|
|
}
|
|
|
|
return(retptr);
|
|
|
|
}
|
|
|
|
|
|
|
|
uint8_t *SuperNET_ciphercalc(void **ptrp,int32_t *cipherlenp,bits256 *privkeyp,bits256 *destpubkeyp,uint8_t *data,int32_t datalen,uint8_t *space2,int32_t space2size)
|
|
|
|
{
|
|
|
|
bits256 mypubkey; uint8_t *buf,*nonce,*cipher,*origptr,space[8192]; int32_t onetimeflag=0,allocsize;
|
|
|
|
*ptrp = 0;
|
|
|
|
allocsize = (datalen + crypto_box_NONCEBYTES + crypto_box_ZEROBYTES);
|
|
|
|
if ( bits256_nonz(*destpubkeyp) == 0 || memcmp(destpubkeyp->bytes,GENESIS_PUBKEY.bytes,sizeof(*destpubkeyp)) == 0 )
|
|
|
|
{
|
|
|
|
*destpubkeyp = GENESIS_PUBKEY;
|
|
|
|
onetimeflag = 2; // prevent any possible leakage of privkey by encrypting to known destpub
|
|
|
|
}
|
|
|
|
if ( bits256_nonz(*privkeyp) == 0 )
|
|
|
|
onetimeflag = 1;
|
|
|
|
if ( onetimeflag != 0 )
|
|
|
|
{
|
|
|
|
crypto_box_keypair(mypubkey.bytes,privkeyp->bytes);
|
|
|
|
allocsize += sizeof(bits256);
|
|
|
|
}
|
|
|
|
if ( allocsize > sizeof(space) )
|
|
|
|
buf = calloc(1,allocsize);
|
|
|
|
else buf = space;
|
|
|
|
if ( allocsize+sizeof(struct iguana_msghdr) > space2size )
|
|
|
|
{
|
|
|
|
cipher = calloc(1,allocsize + sizeof(struct iguana_msghdr));
|
|
|
|
*ptrp = (void *)cipher;
|
|
|
|
} else cipher = space2;
|
|
|
|
cipher = &cipher[sizeof(struct iguana_msghdr)];
|
|
|
|
origptr = nonce = cipher;
|
|
|
|
if ( onetimeflag != 0 )
|
|
|
|
{
|
|
|
|
memcpy(cipher,mypubkey.bytes,sizeof(mypubkey));
|
|
|
|
nonce = &cipher[sizeof(mypubkey)];
|
|
|
|
}
|
|
|
|
OS_randombytes(nonce,crypto_box_NONCEBYTES);
|
|
|
|
cipher = &nonce[crypto_box_NONCEBYTES];
|
|
|
|
_SuperNET_cipher(nonce,cipher,(void *)data,datalen,*destpubkeyp,*privkeyp,buf);
|
|
|
|
if ( buf != space )
|
|
|
|
free(buf);
|
|
|
|
*cipherlenp = allocsize;
|
|
|
|
return(origptr);
|
|
|
|
}
|
|
|
|
|
|
|
|
int32_t SuperNET_copybits(int32_t reverse,uint8_t *dest,uint8_t *src,int32_t len)
|
|
|
|
{
|
|
|
|
int32_t i; uint8_t *tmp;
|
|
|
|
if ( reverse != 0 )
|
|
|
|
{
|
|
|
|
tmp = dest;
|
|
|
|
dest = src;
|
|
|
|
src = tmp;
|
|
|
|
}
|
|
|
|
//printf("src.%p dest.%p len.%d\n",src,dest,len);
|
|
|
|
//for (i=0; i<len; i++)
|
|
|
|
// dest[i] = 0;
|
|
|
|
memset(dest,0,len);
|
|
|
|
len <<= 3;
|
|
|
|
for (i=0; i<len; i++)
|
|
|
|
if ( GETBIT(src,i) != 0 )
|
|
|
|
SETBIT(dest,i);
|
|
|
|
return(len >> 3);
|
|
|
|
}
|
|
|
|
|
|
|
|
uint16_t SuperNET_checkc(bits256 privkey,bits256 otherpub,uint64_t tag)
|
|
|
|
{
|
|
|
|
uint8_t buf[40]; bits256 check,seed,seed2;
|
|
|
|
seed = curve25519_shared(privkey,otherpub);
|
|
|
|
vcalc_sha256(0,seed2.bytes,seed.bytes,sizeof(seed));
|
|
|
|
memcpy(buf,seed2.bytes,sizeof(seed));
|
|
|
|
iguana_rwnum(1,&buf[sizeof(seed)],sizeof(tag),&tag);
|
|
|
|
vcalc_sha256(0,check.bytes,buf,sizeof(buf));
|
|
|
|
return(check.ushorts[0]);
|
|
|
|
}
|
|
|
|
|
|
|
|
int32_t SuperNET_json2bits(char *myipaddr,bits256 privkey,bits256 mypubkey,uint8_t *serialized,int32_t maxsize,char *destip,cJSON *json,bits256 destpub,int16_t othervalid)
|
|
|
|
{
|
|
|
|
uint16_t apinum,checkc; uint32_t ipbits,crc; char *agent,*method; uint64_t tag; char *hexmsg;
|
|
|
|
int32_t n,len = sizeof(uint32_t);
|
|
|
|
if ( (tag= j64bits(json,"tag")) == 0 )
|
|
|
|
OS_randombytes((uint8_t *)&tag,sizeof(tag));
|
|
|
|
agent = jstr(json,"agent"), method = jstr(json,"method");
|
|
|
|
if ( agent != 0 && method != 0 && strcmp(agent,"SuperNET") == 0 && strcmp(method,"json2bits") == 0 )
|
|
|
|
{
|
|
|
|
agent = jstr(json,"destagent");
|
|
|
|
method = jstr(json,"destmethod");
|
|
|
|
}
|
|
|
|
if ( (apinum= SuperNET_API2num(agent,method)) == 0xffff )
|
|
|
|
{
|
|
|
|
printf("agent.(%s) method.(%s) is not found\n",agent,method);
|
|
|
|
return(-1);
|
|
|
|
}
|
|
|
|
checkc = SuperNET_checkc(privkey,destpub,tag);
|
|
|
|
ipbits = (uint32_t)calc_ipbits(destip);
|
|
|
|
len += iguana_rwnum(1,&serialized[len],sizeof(uint32_t),&ipbits);
|
|
|
|
ipbits = (uint32_t)calc_ipbits(myipaddr);
|
|
|
|
len += iguana_rwnum(1,&serialized[len],sizeof(uint32_t),&ipbits);
|
|
|
|
len += iguana_rwnum(1,&serialized[len],sizeof(checkc),&checkc);
|
|
|
|
len += iguana_rwnum(1,&serialized[len],sizeof(apinum),&apinum);
|
|
|
|
len += iguana_rwnum(1,&serialized[len],sizeof(tag),&tag);
|
|
|
|
len += iguana_rwbignum(1,&serialized[len],sizeof(mypubkey),mypubkey.bytes);
|
|
|
|
len += iguana_rwnum(1,&serialized[len],sizeof(othervalid),&othervalid);
|
|
|
|
if ( (hexmsg= jstr(json,"message")) != 0 )
|
|
|
|
{
|
|
|
|
n = (int32_t)strlen(hexmsg);
|
|
|
|
if ( (n & 1) == 0 && is_hexstr(hexmsg,n) > 0 )
|
|
|
|
{
|
|
|
|
n >>= 1;
|
|
|
|
decode_hex(&serialized[len],n,hexmsg);
|
|
|
|
len += n;
|
|
|
|
} else return(-1);
|
|
|
|
}
|
|
|
|
crc = calc_crc32(0,&serialized[sizeof(crc)],len - sizeof(crc));
|
|
|
|
char str[65]; printf("crc.%u ip.(%s %s) tag.%llx checkc.%x apinum.%d >>>>>>>>>>>>>>>> mypub.%s\n",crc,destip,myipaddr,(long long)tag,checkc,apinum,bits256_str(str,mypubkey));
|
|
|
|
iguana_rwnum(1,serialized,sizeof(crc),&crc);
|
|
|
|
//int32_t i; for (i=0; i<len; i++)
|
|
|
|
// printf("%02x ",serialized[i]);
|
|
|
|
//printf("SEND[%d]\n",len);
|
|
|
|
return(len);
|
|
|
|
}
|
|
|
|
|
|
|
|
cJSON *SuperNET_bits2json(struct iguana_peer *addr,uint8_t *serialized,int32_t datalen)
|
|
|
|
{
|
|
|
|
char destip[64],method[64],checkstr[5],agent[64],myipaddr[64],str[65],*hexmsg; uint64_t tag;
|
|
|
|
uint16_t apinum,checkc; int16_t othervalid; uint32_t destipbits,myipbits; bits256 senderpub;
|
|
|
|
int32_t len = 0; uint32_t crc; cJSON *json = cJSON_CreateObject();
|
|
|
|
len += iguana_rwnum(0,&serialized[len],sizeof(uint32_t),&crc);
|
|
|
|
len += iguana_rwnum(0,&serialized[len],sizeof(uint32_t),&destipbits);
|
|
|
|
len += iguana_rwnum(0,&serialized[len],sizeof(uint32_t),&myipbits);
|
|
|
|
len += iguana_rwnum(0,&serialized[len],sizeof(checkc),&checkc);
|
|
|
|
len += iguana_rwnum(0,&serialized[len],sizeof(apinum),&apinum);
|
|
|
|
len += iguana_rwnum(0,&serialized[len],sizeof(tag),&tag);
|
|
|
|
len += iguana_rwbignum(0,&serialized[len],sizeof(bits256),senderpub.bytes);
|
|
|
|
len += iguana_rwnum(0,&serialized[len],sizeof(othervalid),&othervalid);
|
|
|
|
printf("<<<<<<<<<<<<<<<< crc.%u ipbits.(%x %x) tag.%llx checkc.%x apinum.%d valid.%d other.%d\n",crc,destipbits,myipbits,(long long)tag,checkc,apinum,addr->validpub,othervalid);
|
|
|
|
if ( SuperNET_num2API(agent,method,apinum) >= 0 )
|
|
|
|
{
|
|
|
|
jaddstr(json,"agent",agent);
|
|
|
|
jaddstr(json,"method",method);
|
|
|
|
expand_ipbits(destip,destipbits), jaddstr(json,"yourip",destip);
|
|
|
|
expand_ipbits(myipaddr,myipbits), jaddstr(json,"myip",myipaddr);
|
|
|
|
jaddstr(json,"mypub",bits256_str(str,senderpub));
|
|
|
|
jadd64bits(json,"tag",tag);
|
|
|
|
init_hexbytes_noT(checkstr,(void *)&checkc,sizeof(checkc));
|
|
|
|
jaddstr(json,"check",checkstr);
|
|
|
|
jaddnum(json,"ov",othervalid);
|
|
|
|
if ( len < datalen )
|
|
|
|
{
|
|
|
|
printf("len %d vs %d datalen\n",len,datalen);
|
|
|
|
hexmsg = malloc(((datalen - len)<<1) + 1);
|
|
|
|
init_hexbytes_noT(hexmsg,&serialized[len],datalen - len);
|
|
|
|
printf("hex.(%s)\n",hexmsg);
|
|
|
|
jaddstr(json,"message",hexmsg);
|
|
|
|
free(hexmsg);
|
|
|
|
}
|
|
|
|
//printf("bits2json.(%s)\n",jprint(json,0));
|
|
|
|
return(json);
|
|
|
|
} else printf("cant decode apinum.%d (%d.%d)\n",apinum,apinum>>5,apinum%0x1f);
|
|
|
|
return(0);
|
|
|
|
}
|
|
|
|
|
|
|
|
void iguana_setkeys(struct supernet_info *myinfo,struct iguana_peer *addr,bits256 *myprivp,bits256 *mypubp,bits256 *destpubp,bits256 *nextprivp,bits256 *nextpubp,bits256 *nextdestpubp)
|
|
|
|
{
|
|
|
|
*nextprivp = myinfo->privkey;
|
|
|
|
*nextpubp = myinfo->myaddr.pubkey;
|
|
|
|
*nextdestpubp = addr->pubkey;
|
|
|
|
if ( addr->validpub < 3 || addr->othervalid < 3 )
|
|
|
|
*myprivp = GENESIS_PRIVKEY, *destpubp = *mypubp = GENESIS_PUBKEY;
|
|
|
|
else *myprivp = *nextprivp, *mypubp = *nextpubp, *destpubp = *nextdestpubp;
|
|
|
|
char str[65]; printf("(priv.%llx pub.%llx) -> destpub.%s\n",(long long)myprivp->txid,(long long)mypubp->txid,bits256_str(str,*destpubp));
|
|
|
|
}
|
|
|
|
|
|
|
|
bits256 iguana_actualpubkey(int32_t *offsetp,uint8_t *cipher,int32_t cipherlen,bits256 destpubkey)
|
|
|
|
{
|
|
|
|
int32_t i;
|
|
|
|
*offsetp = 0;
|
|
|
|
if ( cipherlen < 56+16 )
|
|
|
|
return(destpubkey);
|
|
|
|
for (i=56; i<56+16; i++)
|
|
|
|
if ( cipher[i] != 0 )
|
|
|
|
break;
|
|
|
|
if ( i == 56+16 )
|
|
|
|
{
|
|
|
|
*offsetp = sizeof(destpubkey);
|
|
|
|
memcpy(destpubkey.bytes,cipher,sizeof(destpubkey));
|
|
|
|
char str[65]; printf("extracted destpubkey.(%s)\n",bits256_str(str,destpubkey));
|
|
|
|
}
|
|
|
|
return(destpubkey);
|
|
|
|
}
|
|
|
|
|
|
|
|
int32_t iguana_send_supernet(struct iguana_info *coin,struct iguana_peer *addr,char *jsonstr,int32_t delaymillis)
|
|
|
|
{
|
|
|
|
int32_t datalen,cipherlen,qlen = -1; uint8_t *serialized,space2[8192],*cipher; cJSON *json;
|
|
|
|
struct supernet_info *myinfo; bits256 destpub,privkey,pubkey,nextprivkey,nextpubkey,nextdestpub; void *ptr = 0;
|
|
|
|
myinfo = SuperNET_MYINFO(0);
|
|
|
|
if ( (json= cJSON_Parse(jsonstr)) != 0 )
|
|
|
|
{
|
|
|
|
iguana_setkeys(myinfo,addr,&privkey,&pubkey,&destpub,&nextprivkey,&nextpubkey,&nextdestpub);
|
|
|
|
if ( juint(json,"plaintext") == 0 && memcmp(destpub.bytes,GENESIS_PUBKEY.bytes,sizeof(pubkey)) == 0 )
|
|
|
|
{
|
|
|
|
printf("reject broadcasting non-plaintext! (%s)\n",jsonstr);
|
|
|
|
free_json(json);
|
|
|
|
return(-1);
|
|
|
|
}
|
|
|
|
serialized = malloc(sizeof(struct iguana_msghdr) + IGUANA_MAXPACKETSIZE);
|
|
|
|
if ( (datalen= SuperNET_json2bits(myinfo->ipaddr,nextprivkey,nextpubkey,&serialized[sizeof(struct iguana_msghdr)],IGUANA_MAXPACKETSIZE,addr->ipaddr,json,nextdestpub,addr->validpub)) > 0 )
|
|
|
|
{
|
|
|
|
printf("SUPERSEND.(%s) -> (%s) delaymillis.%d datalen.%d\n",jsonstr,addr->ipaddr,delaymillis,datalen);
|
|
|
|
if ( 0 && memcmp(destpub.bytes,GENESIS_PUBKEY.bytes,sizeof(destpub)) == 0 )
|
|
|
|
qlen = iguana_queue_send(coin,addr,delaymillis,serialized,"SuperNET",datalen,0,0);
|
|
|
|
else
|
|
|
|
{
|
|
|
|
//int32_t i; for (i=0; i<datalen; i++)
|
|
|
|
// printf("%02x ",serialized[sizeof(struct iguana_msghdr) + i]);
|
|
|
|
//printf("data.%d\n",datalen);
|
|
|
|
printf("encrypt mypriv.%llx destpub.%llx\n",(long long)privkey.txid,(long long)destpub.txid);
|
|
|
|
if ( (cipher= SuperNET_ciphercalc(&ptr,&cipherlen,&privkey,&destpub,&serialized[sizeof(struct iguana_msghdr)],datalen,space2,sizeof(space2))) != 0 )
|
|
|
|
{
|
|
|
|
/*void *msgbits; int32_t msglen; uint8_t space[65536]; void *ptr2;
|
|
|
|
destpub = iguana_actualpubkey(&offset,cipher,cipherlen,destpub);
|
|
|
|
if ( (msgbits= SuperNET_deciphercalc(&ptr2,&msglen,testpriv,destpub,&cipher[offset],cipherlen-offset,space,sizeof(space))) == 0 )
|
|
|
|
{
|
|
|
|
int32_t i; for (i=0; i<cipherlen; i++)
|
|
|
|
printf("%02x",cipher[i]);
|
|
|
|
printf(" cant decrypt cipherlen.%d otherpriv.%llx pub.%llx\n",cipherlen,(long long)testpriv.txid,(long long)pubkey.txid);
|
|
|
|
printf("encrypted mypriv.%llx destpub.%llx\n",(long long)privkey.txid,(long long)destpub.txid);
|
|
|
|
} else printf("decrypted\n");*/
|
|
|
|
qlen = iguana_queue_send(coin,addr,delaymillis,&cipher[-sizeof(struct iguana_msghdr)],"SuperNETb",cipherlen,0,0);
|
|
|
|
if ( ptr != 0 )
|
|
|
|
free(ptr);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
free(serialized);
|
|
|
|
} else printf("cant parse.(%s)\n",jsonstr);
|
|
|
|
return(qlen);
|
|
|
|
}
|
|
|
|
|
|
|
|
int32_t DHT_dist(bits256 desthash,bits256 hash)
|
|
|
|
{
|
|
|
|
int32_t i,dist = 0;
|
|
|
|
for (i=0; i<4; i++)
|
|
|
|
dist += bitweight(desthash.ulongs[i] ^ hash.ulongs[i]);
|
|
|
|
printf("(dist.%d) ",dist);
|
|
|
|
return(dist*0);
|
|
|
|
}
|
|
|
|
|
|
|
|
char *SuperNET_DHTsend(struct supernet_info *myinfo,bits256 routehash,char *hexmsg,int32_t maxdelay,int32_t broadcastflag)
|
|
|
|
{
|
|
|
|
static int lastpurge; static uint64_t Packetcache[1024];
|
|
|
|
bits256 packethash; char retbuf[512]; int32_t mydist,i,j,datalen,firstz,iter,n = 0; char *jsonstr=0;
|
|
|
|
struct iguana_peer *addr; cJSON *json;
|
|
|
|
if ( myinfo == 0 )
|
|
|
|
return(clonestr("{\"error\":\"no supernet_info\"}"));
|
|
|
|
datalen = (int32_t)strlen(hexmsg) + 1;
|
|
|
|
json = cJSON_CreateObject();
|
|
|
|
jaddstr(json,"agent","SuperNET");
|
|
|
|
jaddstr(json,"method","DHT");
|
|
|
|
jaddstr(json,"message",hexmsg);
|
|
|
|
jsonstr = jprint(json,1);
|
|
|
|
vcalc_sha256(0,packethash.bytes,(void *)hexmsg,datalen);
|
|
|
|
firstz = -1;
|
|
|
|
for (i=broadcastflag!=0; i<sizeof(Packetcache)/sizeof(*Packetcache); i++)
|
|
|
|
{
|
|
|
|
if ( Packetcache[i] == 0 )
|
|
|
|
{
|
|
|
|
Packetcache[i] = packethash.txid;
|
|
|
|
printf("add.%llx packetcache(%s)\n",(long long)packethash.txid,hexmsg);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
else if ( Packetcache[i] == packethash.txid )
|
|
|
|
{
|
|
|
|
printf("SuperNET_DHTsend reject repeated packet.%llx (%s)\n",(long long)packethash.txid,hexmsg);
|
|
|
|
return(clonestr("{\"error\":\"duplicate packet rejected\"}"));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if ( i == sizeof(Packetcache)/sizeof(*Packetcache) )
|
|
|
|
{
|
|
|
|
printf("purge slot[%d]\n",lastpurge);
|
|
|
|
Packetcache[lastpurge++] = packethash.txid;
|
|
|
|
if ( lastpurge >= sizeof(Packetcache)/sizeof(*Packetcache) )
|
|
|
|
lastpurge = 0;
|
|
|
|
}
|
|
|
|
mydist = DHT_dist(packethash,myinfo->myaddr.iphash);
|
|
|
|
for (iter=broadcastflag!=0; iter<2; iter++)
|
|
|
|
{
|
|
|
|
for (i=0; i<IGUANA_MAXCOINS; i++)
|
|
|
|
{
|
|
|
|
if ( Coins[i] != 0 )
|
|
|
|
{
|
|
|
|
for (j=0; j<IGUANA_MAXPEERS; j++)
|
|
|
|
{
|
|
|
|
addr = &Coins[i]->peers.active[j];
|
|
|
|
if ( addr->usock >= 0 )
|
|
|
|
{
|
|
|
|
if ( iter == 0 && memcmp(addr->iphash.bytes,routehash.bytes,sizeof(addr->iphash)) == 0 )
|
|
|
|
{
|
|
|
|
printf("DHT send\n");
|
|
|
|
iguana_send_supernet(Coins[i],addr,jsonstr,maxdelay==0?0:(rand()%maxdelay));
|
|
|
|
return(clonestr("{\"result\":\"packet sent directly to destip\"}"));
|
|
|
|
}
|
|
|
|
else if ( iter == 1 )
|
|
|
|
{
|
|
|
|
if ( DHT_dist(packethash,addr->iphash) <= mydist )
|
|
|
|
{
|
|
|
|
iguana_send_supernet(Coins[i],addr,jsonstr,maxdelay==0?0:(rand()%maxdelay));
|
|
|
|
n++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if ( jsonstr != 0 )
|
|
|
|
free(jsonstr);
|
|
|
|
if ( n > 0 )
|
|
|
|
sprintf(retbuf,"{\"result\":\"packet forwarded to superDHT\",\"branches\":%d}",n);
|
|
|
|
else sprintf(retbuf,"{\"error\":\"no nodes to forward packet to\"}");
|
|
|
|
return(clonestr(retbuf));
|
|
|
|
}
|
|
|
|
|
|
|
|
char *SuperNET_DHTencode(struct supernet_info *myinfo,char *destip,bits256 destpub,char *hexmsg,int32_t maxdelay,int32_t broadcastflag)
|
|
|
|
{
|
|
|
|
uint32_t destipbits; bits256 routehash; char *retstr; cJSON *msgjson = cJSON_CreateObject();
|
|
|
|
if ( destip == 0 || destip[0] == 0 || strncmp(destip,"127.0.0.1",strlen("127.0.0.1")) == 0 )
|
|
|
|
{
|
|
|
|
routehash = destpub;
|
|
|
|
jaddbits256(msgjson,"destpub",destpub);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
destipbits = (uint32_t)calc_ipbits(destip);
|
|
|
|
vcalc_sha256(0,routehash.bytes,(uint8_t *)&destipbits,sizeof(destipbits));
|
|
|
|
jaddstr(msgjson,"destip",destip);
|
|
|
|
}
|
|
|
|
retstr = SuperNET_DHTsend(myinfo,routehash,hexmsg,maxdelay,broadcastflag);
|
|
|
|
return(retstr);
|
|
|
|
}
|
|
|
|
|
|
|
|
char *SuperNET_forward(struct supernet_info *myinfo,char *hexmsg,uint32_t destipbits,bits256 destpub,int32_t maxdelay,int32_t broadcastflag)
|
|
|
|
{
|
|
|
|
bits256 routehash;
|
|
|
|
if ( destipbits != 0 )
|
|
|
|
vcalc_sha256(0,routehash.bytes,(uint8_t *)&destipbits,sizeof(destipbits));
|
|
|
|
else routehash = destpub;
|
|
|
|
return(SuperNET_DHTsend(myinfo,routehash,hexmsg,maxdelay,broadcastflag));
|
|
|
|
}
|
|
|
|
|
|
|
|
int32_t SuperNET_destination(struct supernet_info *myinfo,uint32_t *destipbitsp,bits256 *destpubp,int32_t *maxdelayp,cJSON *json,char *remoteaddr)
|
|
|
|
{
|
|
|
|
char *destip; int32_t destflag = 0;
|
|
|
|
if ( (destip= jstr(json,"destip")) != 0 )
|
|
|
|
*destipbitsp = (uint32_t)calc_ipbits(destip);
|
|
|
|
else *destipbitsp = 0;
|
|
|
|
*maxdelayp = juint(json,"delay");
|
|
|
|
*destpubp = jbits256(json,"destpub");
|
|
|
|
if ( *destipbitsp != 0 )
|
|
|
|
{
|
|
|
|
if ( *destipbitsp == myinfo->myaddr.selfipbits )
|
|
|
|
destflag |= SUPERNET_ISMINE;
|
|
|
|
else destflag |= SUPERNET_FORWARD;
|
|
|
|
}
|
|
|
|
else if ( bits256_nonz(*destpubp) > 0 )
|
|
|
|
{
|
|
|
|
if ( memcmp(destpubp,myinfo->myaddr.pubkey.bytes,sizeof(*destpubp)) == 0 )
|
|
|
|
destflag |= SUPERNET_ISMINE;
|
|
|
|
else destflag |= SUPERNET_FORWARD;
|
|
|
|
}
|
|
|
|
else if ( remoteaddr == 0 || remoteaddr[0] == 0 )
|
|
|
|
destflag |= SUPERNET_ISMINE;
|
|
|
|
return(destflag);
|
|
|
|
}
|
|
|
|
|
|
|
|
char *SuperNET_JSON(struct supernet_info *myinfo,cJSON *json,char *remoteaddr)
|
|
|
|
{
|
|
|
|
int32_t destflag,maxdelay; bits256 destpub; uint32_t destipbits; cJSON *retjson;
|
|
|
|
char *forwardstr=0,*retstr=0,*agent=0,*method=0,*message,*jsonstr=0;
|
|
|
|
if ( remoteaddr != 0 && strcmp(remoteaddr,"127.0.0.1") == 0 )
|
|
|
|
remoteaddr = 0;
|
|
|
|
//printf("SuperNET_JSON.(%s) remote.(%s)\n",jprint(json,0),remoteaddr!=0?remoteaddr:"");
|
|
|
|
destflag = SuperNET_destination(myinfo,&destipbits,&destpub,&maxdelay,json,remoteaddr);
|
|
|
|
//printf("destflag.%d\n",destflag);
|
|
|
|
if ( (destflag & SUPERNET_FORWARD) != 0 )
|
|
|
|
{
|
|
|
|
if ( (message= jstr(json,"message")) == 0 )
|
|
|
|
{
|
|
|
|
jsonstr = jprint(json,0);
|
|
|
|
message = jsonstr;
|
|
|
|
}
|
|
|
|
forwardstr = SuperNET_forward(myinfo,message,destipbits,destpub,maxdelay,juint(json,"broadcast"));
|
|
|
|
}
|
|
|
|
if ( (destflag & SUPERNET_ISMINE) && (agent= jstr(json,"agent")) != 0 && (method= jstr(json,"method")) != 0 )
|
|
|
|
{
|
|
|
|
if ( (retstr= SuperNET_processJSON(myinfo,json,remoteaddr)) != 0 )
|
|
|
|
{
|
|
|
|
//printf("retstr.(%s)\n",retstr);
|
|
|
|
if ( remoteaddr != 0 && (retjson= cJSON_Parse(retstr)) != 0 )
|
|
|
|
{
|
|
|
|
if ( jobj(retjson,"result") != 0 || jobj(retjson,"error") != 0 || jobj(retjson,"method") == 0 )
|
|
|
|
{
|
|
|
|
//printf("it is a result, dont return\n");
|
|
|
|
free(retstr);
|
|
|
|
retstr = 0;
|
|
|
|
}
|
|
|
|
free_json(retjson);
|
|
|
|
}
|
|
|
|
} else printf("null retstr from SuperNET_JSON\n");
|
|
|
|
}
|
|
|
|
if ( retstr == 0 )
|
|
|
|
retstr = forwardstr, forwardstr = 0;
|
|
|
|
if ( forwardstr != 0 )
|
|
|
|
free(forwardstr);
|
|
|
|
if ( jsonstr != 0 )
|
|
|
|
free(jsonstr);
|
|
|
|
return(retstr);
|
|
|
|
}
|
|
|
|
|
|
|
|
char *SuperNET_p2p(struct iguana_info *coin,struct iguana_peer *addr,int32_t *delaymillisp,char *ipaddr,uint8_t *data,int32_t datalen,int32_t compressed)
|
|
|
|
{
|
|
|
|
struct supernet_info *myinfo; cJSON *json; char *myipaddr,*method,*retstr,*checkstr; void *ptr=0;
|
|
|
|
bits256 senderpub,privkey,pubkey,nextprivkey,nextpubkey,nextdestpub; uint16_t checkc,othercheckc;
|
|
|
|
int32_t offset,maxdelay,msglen = datalen; uint8_t space[8192],*msgbits = 0;
|
|
|
|
myinfo = SuperNET_MYINFO(0);
|
|
|
|
retstr = 0;
|
|
|
|
*delaymillisp = 0;
|
|
|
|
if ( compressed != 0 )
|
|
|
|
{
|
|
|
|
//int32_t i; for (i=0; i<datalen; i++)
|
|
|
|
// printf("%02x ",data[i]);
|
|
|
|
//printf("DECRYPT %d\n",datalen);
|
|
|
|
iguana_setkeys(myinfo,addr,&privkey,&pubkey,&senderpub,&nextprivkey,&nextpubkey,&nextdestpub);
|
|
|
|
senderpub = iguana_actualpubkey(&offset,data,datalen,senderpub);
|
|
|
|
if ( (msgbits= SuperNET_deciphercalc(&ptr,&msglen,privkey,senderpub,data+offset,datalen-offset,space,sizeof(space))) == 0 )
|
|
|
|
{
|
|
|
|
if ( (msgbits= SuperNET_deciphercalc(&ptr,&msglen,GENESIS_PRIVKEY,senderpub,data+offset,datalen-offset,space,sizeof(space))) == 0 )
|
|
|
|
{
|
|
|
|
int32_t i; for (i=0; i<datalen; i++)
|
|
|
|
printf("%02x ",data[i]);
|
|
|
|
printf("error decryptint %d\n",datalen);
|
|
|
|
addr->validpub = 0;
|
|
|
|
//return(clonestr("{\"error\":\"couldnt decrypt p2p packet\"}"));
|
|
|
|
return(0);
|
|
|
|
} else printf("GENESIS recv\n");
|
|
|
|
} else printf("decrypted mypriv.%llx senderpub.%llx\n",(long long)privkey.txid,(long long)senderpub.txid);
|
|
|
|
//for (i=0; i<msglen; i++)
|
|
|
|
// printf("%02x ",msgbits[i]);
|
|
|
|
//printf("DECRYPTED %d\n",msglen);
|
|
|
|
} else msgbits = data;
|
|
|
|
if ( (json= SuperNET_bits2json(addr,msgbits,msglen)) != 0 )
|
|
|
|
{
|
|
|
|
senderpub = jbits256(json,"mypub");
|
|
|
|
if ( (checkstr= jstr(json,"check")) != 0 )
|
|
|
|
{
|
|
|
|
decode_hex((uint8_t *)&othercheckc,sizeof(othercheckc),checkstr);
|
|
|
|
checkc = SuperNET_checkc(myinfo->privkey,senderpub,j64bits(json,"tag"));
|
|
|
|
if ( checkc == othercheckc )
|
|
|
|
addr->validpub++;
|
|
|
|
else if ( addr->validpub > 0 )
|
|
|
|
addr->validpub >>= 1;
|
|
|
|
else addr->validpub--;
|
|
|
|
printf("validpub.%d: %x vs %x shared.%llx\n",addr->validpub,checkc,othercheckc,(long long)addr->sharedseed.txid);
|
|
|
|
}
|
|
|
|
/*if ( addr->validpub > 3 && bits256_nonz(addr->sharedseed) == 0 )
|
|
|
|
addr->sharedseed = SuperNET_sharedseed(myinfo->privkey,senderpub);
|
|
|
|
else if ( addr->validpub < -2 )
|
|
|
|
memset(addr->sharedseed.bytes,0,sizeof(addr->sharedseed));*/
|
|
|
|
maxdelay = juint(json,"maxdelay");
|
|
|
|
printf("GOT >>>>>>>> SUPERNET P2P.(%s) from.%s valid.%d:%d\n",jprint(json,0),coin->symbol,addr->validpub,addr->othervalid);
|
|
|
|
if ( (myipaddr= jstr(json,"yourip")) != 0 )
|
|
|
|
SuperNET_myipaddr(SuperNET_MYINFO(0),coin,addr,myipaddr,ipaddr);
|
|
|
|
jaddstr(json,"fromp2p",coin->symbol);
|
|
|
|
method = jstr(json,"method");
|
|
|
|
if ( method != 0 && strcmp(method,"stop") == 0 )
|
|
|
|
{
|
|
|
|
addr->dead = (uint32_t)time(NULL);
|
|
|
|
free_json(json);
|
|
|
|
if ( ptr != 0 )
|
|
|
|
free(ptr);
|
|
|
|
//return(clonestr("{\"result\":\"peer marked as dead\"}"));
|
|
|
|
return(0);
|
|
|
|
}
|
|
|
|
retstr = SuperNET_JSON(myinfo,json,ipaddr);
|
|
|
|
//printf("p2pret.(%s)\n",retstr);
|
|
|
|
*delaymillisp = SuperNET_delaymillis(myinfo,maxdelay);
|
|
|
|
senderpub = jbits256(json,"mypub");
|
|
|
|
//if ( memcmp(senderpub.bytes,addr->pubkey.bytes,sizeof(senderpub)) != 0 )
|
|
|
|
// addr->pubkey = senderpub;
|
|
|
|
addr->othervalid = (int32_t)jdouble(json,"ov");
|
|
|
|
addr->pubkey = senderpub;
|
|
|
|
free_json(json);
|
|
|
|
}
|
|
|
|
if ( ptr != 0 )
|
|
|
|
free(ptr);
|
|
|
|
return(retstr);
|
|
|
|
}
|
|
|
|
|
|
|
|
cJSON *SuperNET_peerarray(struct iguana_info *coin,int32_t max,int32_t supernetflag)
|
|
|
|
{
|
|
|
|
int32_t i,r,j,n = 0; struct iguana_peer *addr; cJSON *array = cJSON_CreateArray();
|
|
|
|
r = rand();
|
|
|
|
for (j=0; j<IGUANA_MAXPEERS; j++)
|
|
|
|
{
|
|
|
|
i = (r + j) % IGUANA_MAXPEERS;
|
|
|
|
addr = &coin->peers.active[i];
|
|
|
|
if ( addr->usock >= 0 && supernetflag == (addr->supernet != 0) )
|
|
|
|
{
|
|
|
|
jaddistr(array,addr->ipaddr);
|
|
|
|
if ( ++n >= max )
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if ( n == 0 )
|
|
|
|
{
|
|
|
|
free_json(array);
|
|
|
|
return(0);
|
|
|
|
}
|
|
|
|
return(array);
|
|
|
|
}
|
|
|
|
|
|
|
|
int32_t SuperNET_coinpeers(struct iguana_info *coin,cJSON *SNjson,cJSON *rawjson,int32_t max)
|
|
|
|
{
|
|
|
|
cJSON *array,*item;
|
|
|
|
if ( (array= SuperNET_peerarray(coin,max,1)) != 0 )
|
|
|
|
{
|
|
|
|
max -= cJSON_GetArraySize(array);
|
|
|
|
item = cJSON_CreateObject();
|
|
|
|
jaddstr(item,"coin",coin->symbol);
|
|
|
|
jadd(item,"peers",array);
|
|
|
|
jaddi(SNjson,item);
|
|
|
|
}
|
|
|
|
if ( max > 0 && (array= SuperNET_peerarray(coin,max,0)) != 0 )
|
|
|
|
{
|
|
|
|
max -= cJSON_GetArraySize(array);
|
|
|
|
item = cJSON_CreateObject();
|
|
|
|
jaddstr(item,"coin",coin->symbol);
|
|
|
|
jadd(item,"peers",array);
|
|
|
|
jaddi(rawjson,item);
|
|
|
|
}
|
|
|
|
return(max);
|
|
|
|
}
|
|
|
|
|
|
|
|
void SuperNET_parsepeers(struct supernet_info *myinfo,cJSON *array,int32_t n,int32_t supernetflag)
|
|
|
|
{
|
|
|
|
int32_t i,j,m; cJSON *coinarray,*item; char *symbol,*ipaddr; struct iguana_info *ptr;
|
|
|
|
if ( array != 0 && n > 0 )
|
|
|
|
{
|
|
|
|
for (i=0; i<n; i++)
|
|
|
|
{
|
|
|
|
if ( (item= jitem(array,i)) != 0 && (symbol= jstr(item,"coin")) != 0 )
|
|
|
|
{
|
|
|
|
ptr = iguana_coinfind(symbol);
|
|
|
|
if ( (coinarray= jarray(&m,item,"peers")) != 0 )
|
|
|
|
{
|
|
|
|
for (j=0; j<m; j++)
|
|
|
|
{
|
|
|
|
if ( (ipaddr= jstr(jitem(coinarray,j),0)) != 0 )
|
|
|
|
SuperNET_remotepeer(myinfo,ptr,symbol,ipaddr,supernetflag);
|
|
|
|
else printf("no ipaddr[%d] of %d\n",j,m);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
printf("parsed.%d %s.peers supernet.%d\n",m,symbol,supernetflag);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#include "../includes/iguana_apidefs.h"
|
|
|
|
|
|
|
|
HASH_ARG(SuperNET,priv2pub,privkey)
|
|
|
|
{
|
|
|
|
cJSON *retjson = cJSON_CreateObject(); bits256 pubkey;
|
|
|
|
crypto_box_priv2pub(pubkey.bytes,privkey.bytes);
|
|
|
|
jaddbits256(retjson,"result",pubkey);
|
|
|
|
return(jprint(retjson,1));
|
|
|
|
}
|
|
|
|
|
|
|
|
ZERO_ARGS(SuperNET,keypair)
|
|
|
|
{
|
|
|
|
cJSON *retjson = cJSON_CreateObject(); bits256 pubkey,privkey;
|
|
|
|
crypto_box_keypair(pubkey.bytes,privkey.bytes);
|
|
|
|
jaddstr(retjson,"result","generated keypair");
|
|
|
|
jaddbits256(retjson,"privkey",privkey);
|
|
|
|
jaddbits256(retjson,"pubkey",pubkey);
|
|
|
|
return(jprint(retjson,1));
|
|
|
|
}
|
|
|
|
|
|
|
|
TWOHASHES_AND_STRING(SuperNET,decipher,privkey,srcpubkey,cipherstr)
|
|
|
|
{
|
|
|
|
int32_t cipherlen,msglen; char *retstr; cJSON *retjson; void *ptr = 0; uint8_t *cipher,*message,space[8192];
|
|
|
|
cipherlen = (int32_t)strlen(cipherstr) >> 1;
|
|
|
|
if ( cipherlen < crypto_box_NONCEBYTES )
|
|
|
|
return(clonestr("{\"error\":\"cipher is too short\"}"));
|
|
|
|
cipher = calloc(1,cipherlen);
|
|
|
|
decode_hex(cipher,cipherlen,cipherstr);
|
|
|
|
if ( (message= SuperNET_deciphercalc(&ptr,&msglen,privkey,srcpubkey,cipher,cipherlen,space,sizeof(space))) != 0 )
|
|
|
|
{
|
|
|
|
message[msglen] = 0;
|
|
|
|
retjson = cJSON_CreateObject();
|
|
|
|
jaddstr(retjson,"result","deciphered message");
|
|
|
|
jaddstr(retjson,"message",(char *)message);
|
|
|
|
retstr = jprint(retjson,1);
|
|
|
|
if ( ptr != 0 )
|
|
|
|
free(ptr);
|
|
|
|
} else retstr = clonestr("{\"error\":\"couldnt decipher message\"}");
|
|
|
|
return(retstr);
|
|
|
|
}
|
|
|
|
|
|
|
|
TWOHASHES_AND_STRING(SuperNET,cipher,privkey,destpubkey,message)
|
|
|
|
{
|
|
|
|
cJSON *retjson; char *retstr,*hexstr,space[8129]; uint8_t space2[8129];
|
|
|
|
uint8_t *cipher; int32_t cipherlen,onetimeflag; bits256 origprivkey; void *ptr = 0;
|
|
|
|
if ( (cipher= SuperNET_ciphercalc(&ptr,&cipherlen,&privkey,&destpubkey,(uint8_t *)message,(int32_t)strlen(message)+1,space2,sizeof(space2))) != 0 )
|
|
|
|
{
|
|
|
|
if ( cipherlen > sizeof(space)/2 )
|
|
|
|
hexstr = calloc(1,(cipherlen<<1)+1);
|
|
|
|
else hexstr = (void *)space;
|
|
|
|
init_hexbytes_noT(hexstr,cipher,cipherlen);
|
|
|
|
retjson = cJSON_CreateObject();
|
|
|
|
jaddstr(retjson,"result",hexstr);
|
|
|
|
//jaddstr(retjson,"message",message);
|
|
|
|
//jaddstr(retjson,"cipher",hexstr);
|
|
|
|
onetimeflag = memcmp(origprivkey.bytes,privkey.bytes,sizeof(privkey));
|
|
|
|
if ( onetimeflag != 0 )
|
|
|
|
{
|
|
|
|
jaddbits256(retjson,"onetime_privkey",privkey);
|
|
|
|
jaddbits256(retjson,"onetime_pubkey",destpubkey);
|
|
|
|
if ( onetimeflag == 2 )
|
|
|
|
jaddstr(retjson,"warning","onetime keypair was used to broadcast");
|
|
|
|
}
|
|
|
|
retstr = jprint(retjson,1);
|
|
|
|
if ( hexstr != (void *)space )
|
|
|
|
free(hexstr);
|
|
|
|
if ( ptr != 0 )
|
|
|
|
free(ptr);
|
|
|
|
return(retstr);
|
|
|
|
} else return(clonestr("{\"error\":\"cant encrypt message\"}"));
|
|
|
|
}
|
|
|
|
|
|
|
|
bits256 SuperNET_pindecipher(IGUANA_ARGS,char *pin,char *privcipher)
|
|
|
|
{
|
|
|
|
cJSON *testjson; char *mstr,*cstr; bits256 privkey,pinpriv,pinpub;
|
|
|
|
conv_NXTpassword(pinpriv.bytes,pinpub.bytes,(uint8_t *)pin,(int32_t)strlen(pin));
|
|
|
|
if ( (cstr= SuperNET_decipher(IGUANA_CALLARGS,pinpriv,pinpub,privcipher)) != 0 )
|
|
|
|
{
|
|
|
|
if ( (testjson= cJSON_Parse(cstr)) != 0 )
|
|
|
|
{
|
|
|
|
if ( (mstr= jstr(testjson,"message")) != 0 && strlen(mstr) == sizeof(bits256)*2 )
|
|
|
|
{
|
|
|
|
decode_hex(privkey.bytes,sizeof(privkey),mstr);
|
|
|
|
} //else jaddstr(retjson,"error","invalid return from deciphering privcipher");
|
|
|
|
free_json(testjson);
|
|
|
|
}
|
|
|
|
free(cstr);
|
|
|
|
} //else jaddstr(retjson,"error","null return from deciphering privcipher");
|
|
|
|
return(privkey);
|
|
|
|
}
|
|
|
|
|
|
|
|
THREE_STRINGS(SuperNET,rosetta,passphrase,pin,showprivkey)
|
|
|
|
{
|
|
|
|
uint8_t rmd160[20],pub[33],flag = 0; uint64_t nxt64bits; bits256 check,privkey,pubkey,pinpriv,pinpub;
|
|
|
|
char str2[41],wifbuf[64],addr[64],str[128],privcipher[512],*privcipherstr,*cstr; cJSON *retjson;
|
|
|
|
nxt64bits = conv_NXTpassword(privkey.bytes,pubkey.bytes,(uint8_t *)passphrase,(int32_t)strlen(passphrase));
|
|
|
|
if ( showprivkey != 0 && strcmp(showprivkey,"yes") == 0 )
|
|
|
|
flag = 1;
|
|
|
|
privcipher[0] = 0;
|
|
|
|
conv_NXTpassword(pinpriv.bytes,pinpub.bytes,(uint8_t *)pin,(int32_t)strlen(pin));
|
|
|
|
if ( (cstr= SuperNET_cipher(IGUANA_CALLARGS,pinpriv,pinpub,bits256_str(str,privkey))) != 0 )
|
|
|
|
{
|
|
|
|
if ( (retjson= cJSON_Parse(cstr)) != 0 )
|
|
|
|
{
|
|
|
|
if ( (privcipherstr= jstr(retjson,"cipher")) != 0 )
|
|
|
|
strcpy(privcipher,privcipherstr);
|
|
|
|
free_json(retjson);
|
|
|
|
}
|
|
|
|
free(cstr);
|
|
|
|
}
|
|
|
|
retjson = cJSON_CreateObject();
|
|
|
|
jaddstr(retjson,"privcipher",privcipher);
|
|
|
|
jaddbits256(retjson,"pubkey",pubkey);
|
|
|
|
RS_encode(str,nxt64bits);
|
|
|
|
jaddstr(retjson,"RS",str);
|
|
|
|
jadd64bits(retjson,"NXT",nxt64bits);
|
|
|
|
btc_priv2pub(pub,privkey.bytes);
|
|
|
|
init_hexbytes_noT(str,pub,33);
|
|
|
|
jaddstr(retjson,"btcpubkey",str);
|
|
|
|
calc_OP_HASH160(str2,rmd160,str);
|
|
|
|
jaddstr(retjson,"rmd160",str2);
|
|
|
|
if ( btc_coinaddr(addr,0,str) == 0 )
|
|
|
|
{
|
|
|
|
jaddstr(retjson,"BTC",addr);
|
|
|
|
btc_priv2wip(wifbuf,privkey.bytes,0x80);
|
|
|
|
if ( flag != 0 )
|
|
|
|
jaddstr(retjson,"BTCwif",wifbuf);
|
|
|
|
}
|
|
|
|
if ( btc_coinaddr(addr,60,str) == 0 )
|
|
|
|
{
|
|
|
|
jaddstr(retjson,"BTCD",addr);
|
|
|
|
btc_priv2wip(wifbuf,privkey.bytes,0xbc);
|
|
|
|
if ( flag != 0 )
|
|
|
|
jaddstr(retjson,"BTCDwif",wifbuf);
|
|
|
|
}
|
|
|
|
if ( flag != 0 )
|
|
|
|
jaddbits256(retjson,"privkey",privkey);
|
|
|
|
check = SuperNET_pindecipher(IGUANA_CALLARGS,pin,privcipher);
|
|
|
|
if ( memcmp(check.bytes,privkey.bytes,sizeof(check)) != 0 )
|
|
|
|
{
|
|
|
|
jaddbits256(retjson,"deciphered",check);
|
|
|
|
jaddstr(retjson,"error","cant recreate privkey from (pin + privcipher)");
|
|
|
|
}
|
|
|
|
else if ( flag != 0 )
|
|
|
|
jaddbits256(retjson,"deciphered",check);
|
|
|
|
if ( jobj(retjson,"error") == 0 )
|
|
|
|
jaddstr(retjson,"result","use pin and privcipher to access wallet");
|
|
|
|
return(jprint(retjson,1));
|
|
|
|
}
|
|
|
|
|
|
|
|
STRING_ARG(SuperNET,broadcastcipher,message)
|
|
|
|
{
|
|
|
|
bits256 zero;
|
|
|
|
memset(zero.bytes,0,sizeof(zero));
|
|
|
|
return(SuperNET_cipher(IGUANA_CALLARGS,zero,zero,message));
|
|
|
|
}
|
|
|
|
|
|
|
|
STRING_ARG(SuperNET,broadcastdecipher,message)
|
|
|
|
{
|
|
|
|
bits256 zero;
|
|
|
|
memset(zero.bytes,0,sizeof(zero));
|
|
|
|
return(SuperNET_decipher(IGUANA_CALLARGS,zero,zero,message));
|
|
|
|
}
|
|
|
|
|
|
|
|
HASH_AND_STRING(SuperNET,multicastcipher,pubkey,message)
|
|
|
|
{
|
|
|
|
bits256 zero;
|
|
|
|
memset(zero.bytes,0,sizeof(zero));
|
|
|
|
return(SuperNET_cipher(IGUANA_CALLARGS,zero,pubkey,message));
|
|
|
|
}
|
|
|
|
|
|
|
|
HASH_AND_STRING(SuperNET,multicastdecipher,privkey,cipherstr)
|
|
|
|
{
|
|
|
|
bits256 zero;
|
|
|
|
memset(zero.bytes,0,sizeof(zero));
|
|
|
|
return(SuperNET_decipher(IGUANA_CALLARGS,privkey,zero,cipherstr));
|
|
|
|
}
|
|
|
|
|
|
|
|
ZERO_ARGS(SuperNET,stop)
|
|
|
|
{
|
|
|
|
if ( remoteaddr == 0 || strncmp(remoteaddr,"127.0.0.1",strlen("127.0.0.1")) == 0 )
|
|
|
|
{
|
|
|
|
iguana_exit();
|
|
|
|
return(clonestr("{\"result\":\"exit started\"}"));
|
|
|
|
} else return(clonestr("{\"error\":\"cant do a remote stop of this node\"}"));
|
|
|
|
}
|
|
|
|
|
|
|
|
TWO_ARRAYS(SuperNET,mypeers,supernet,rawpeers)
|
|
|
|
{
|
|
|
|
SuperNET_parsepeers(myinfo,supernet,cJSON_GetArraySize(supernet),1);
|
|
|
|
SuperNET_parsepeers(myinfo,rawpeers,cJSON_GetArraySize(rawpeers),0);
|
|
|
|
return(clonestr("{\"result\":\"peers parsed\"}"));
|
|
|
|
}
|
|
|
|
|
|
|
|
STRING_ARG(SuperNET,getpeers,activecoin)
|
|
|
|
{
|
|
|
|
int32_t i,max = 64;
|
|
|
|
cJSON *SNjson,*rawjson,*retjson = cJSON_CreateObject();
|
|
|
|
SNjson = cJSON_CreateArray();
|
|
|
|
rawjson = cJSON_CreateArray();
|
|
|
|
if ( coin != 0 )
|
|
|
|
max = SuperNET_coinpeers(coin,SNjson,rawjson,max);
|
|
|
|
else
|
|
|
|
{
|
|
|
|
for (i=0; i<IGUANA_MAXCOINS&&max>0; i++)
|
|
|
|
if ( Coins[i] != 0 )
|
|
|
|
max = SuperNET_coinpeers(Coins[i],SNjson,rawjson,max);
|
|
|
|
}
|
|
|
|
if ( max != 64 )
|
|
|
|
{
|
|
|
|
jaddstr(retjson,"agent","SuperNET");
|
|
|
|
jaddstr(retjson,"method","mypeers");
|
|
|
|
jadd(retjson,"supernet",SNjson);
|
|
|
|
jadd(retjson,"rawpeers",rawjson);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
jaddstr(retjson,"error","no peers");
|
|
|
|
free_json(SNjson);
|
|
|
|
free_json(rawjson);
|
|
|
|
}
|
|
|
|
return(jprint(retjson,1));
|
|
|
|
}
|
|
|
|
|
|
|
|
TWOSTRINGS_AND_HASH_AND_TWOINTS(SuperNET,DHT,hexmsg,destip,destpub,maxdelay,broadcast)
|
|
|
|
{
|
|
|
|
if ( remoteaddr != 0 )
|
|
|
|
return(clonestr("{\"error\":\"cant remote DHT\"}"));
|
|
|
|
else if ( is_hexstr(hexmsg,(int32_t)strlen(hexmsg)) <= 0 )
|
|
|
|
return(clonestr("{\"error\":\"message must be in hex\"}"));
|
|
|
|
return(SuperNET_DHTencode(myinfo,destip,destpub,hexmsg,maxdelay,broadcast));
|
|
|
|
}
|
|
|
|
|
|
|
|
HASH_AND_STRING(SuperNET,saveconf,wallethash,confjsonstr)
|
|
|
|
{
|
|
|
|
return(clonestr("{\"result\":\"saveconf here\"}"));
|
|
|
|
}
|
|
|
|
|
|
|
|
HASH_ARRAY_STRING(SuperNET,layer,mypriv,otherpubs,str)
|
|
|
|
{
|
|
|
|
return(clonestr("{\"result\":\"layer encrypt here\"}"));
|
|
|
|
}
|
|
|
|
|
|
|
|
THREE_STRINGS(SuperNET,announce,category,subcategory,message)
|
|
|
|
{
|
|
|
|
cJSON *argjson = cJSON_CreateObject(); int32_t len; char *hexmsg=0,*retstr = 0;
|
|
|
|
//SuperNET_ann(myinfo,category,subcategory,message);
|
|
|
|
if ( remoteaddr == 0 )
|
|
|
|
{
|
|
|
|
len = (int32_t)strlen(message);
|
|
|
|
if ( is_hexstr(message,len) == 0 )
|
|
|
|
{
|
|
|
|
}
|
|
|
|
else hexmsg = message;
|
|
|
|
retstr = SuperNET_DHTsend(myinfo,GENESIS_PUBKEY,hexmsg,0,1);
|
|
|
|
if ( hexmsg != message)
|
|
|
|
free(hexmsg);
|
|
|
|
}
|
|
|
|
return(jprint(argjson,1));
|
|
|
|
}
|
|
|
|
|
|
|
|
THREE_STRINGS(SuperNET,survey,category,subcategory,message)
|
|
|
|
{
|
|
|
|
return(clonestr("{\"result\":\"layer encrypt here\"}"));
|
|
|
|
}
|
|
|
|
#include "../includes/iguana_apiundefs.h"
|