326 lines
13 KiB
326 lines
13 KiB
9 years ago
|
/******************************************************************************
|
||
|
* 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 "peggy.h"
|
||
|
|
||
|
#define accts777_getaddrkv(accts,type) ((accts != 0) ? (accts)->addrkvs[type] : 0)
|
||
|
|
||
|
bits256 PAX_hashiter(bits256 privkey,bits256 pubkey,int32_t lockdays,uint8_t chainlen)
|
||
|
{
|
||
|
uint16_t lockseed,signlen = 0; uint8_t signbuf[16]; bits256 shared,lockhash;
|
||
|
lockseed = (chainlen & 0x7f) | (lockdays << 7);
|
||
|
signlen = 0, signbuf[signlen++] = lockseed & 0xff, signbuf[signlen++] = (lockseed >> 8) & 0xff;
|
||
|
|
||
|
privkey.bytes[0] &= 248, privkey.bytes[31] &= 127, privkey.bytes[31] |= 64;
|
||
|
shared = curve25519(privkey,pubkey);
|
||
|
vcalc_sha256cat(lockhash.bytes,shared.bytes,sizeof(shared),signbuf,signlen);
|
||
|
return(lockhash);
|
||
|
}
|
||
|
|
||
|
bits256 PAX_lockhash(bits256 pubkey,int32_t lockdays,uint8_t chainlen)
|
||
|
{
|
||
|
bits256 lockhash = GENESIS_PRIVKEY;
|
||
|
while ( chainlen > 0 )
|
||
|
lockhash = PAX_hashiter(lockhash,pubkey,lockdays,chainlen--);
|
||
|
return(lockhash);
|
||
|
}
|
||
|
|
||
|
bits256 PAX_invoicehash(bits256 *invoicehash,uint16_t lockdays,uint8_t chainlen)
|
||
|
{
|
||
|
int32_t i; bits256 lockhash,privkey;
|
||
|
OS_randombytes(privkey.bytes,sizeof(privkey)); // both privkey and pubkey are sensitive. pubkey allows verification, privkey proves owner
|
||
|
lockhash = privkey;
|
||
|
for (i=0; i<chainlen; i++)
|
||
|
lockhash = PAX_hashiter(lockhash,GENESIS_PUBKEY,chainlen - i,lockdays);
|
||
|
*invoicehash = lockhash;
|
||
|
return(privkey);
|
||
|
}
|
||
|
|
||
|
uint64_t PAX_sign(struct PAX_sig *sig,bits256 privkey,bits256 otherpubkey,uint32_t timestamp,uint8_t *data,int32_t datalen)
|
||
|
{
|
||
|
int32_t i; bits256 shared; uint8_t buf[sizeof(shared) + sizeof(timestamp)]; uint32_t t = timestamp;
|
||
|
for (i=0; i<sizeof(t); i++,t>>=8)
|
||
|
buf[i] = (t & 0xff);
|
||
|
sig->timestamp = timestamp;
|
||
|
shared = curve25519(privkey,otherpubkey);
|
||
|
memcpy(&buf[sizeof(timestamp)],shared.bytes,sizeof(shared));
|
||
|
vcalc_sha256cat(sig->sigbits.bytes,buf,sizeof(buf),data,datalen);
|
||
|
sig->pubkey = acct777_pubkey(privkey), sig->signer64bits = acct777_nxt64bits(sig->pubkey);
|
||
|
//printf(" calcsig.%llx pubkey.%llx signer.%llu | t%u crc.%08x len.%d shared.%llx <- %llx * %llx\n",(long long)sig->sigbits.txid,(long long)sig->pubkey.txid,(long long)sig->signer64bits,timestamp,_crc32(0,data,datalen),datalen,(long long)shared.txid,(long long)privkey.txid,(long long)otherpubkey.txid);
|
||
|
return(sig->signer64bits);
|
||
|
}
|
||
|
|
||
|
uint64_t PAX_validate(struct PAX_sig *sig,uint32_t timestamp,uint8_t *data,int32_t datalen)
|
||
|
{
|
||
|
struct PAX_sig checksig;
|
||
|
PAX_sign(&checksig,GENESIS_PRIVKEY,sig->pubkey,timestamp,data,datalen);
|
||
|
if ( memcmp(checksig.sigbits.bytes,sig->sigbits.bytes,sizeof(checksig.sigbits)) != 0 )
|
||
|
{
|
||
|
printf("sig compare error using sig->pub from %llu\n",(long long)acct777_nxt64bits(sig->pubkey));
|
||
|
return(0);
|
||
|
}
|
||
|
return(acct777_nxt64bits(sig->pubkey));
|
||
|
}
|
||
|
|
||
|
uint64_t PAX_signtx(struct PAX_sig *sig,bits256 privkey,uint32_t timestamp,uint8_t *data,int32_t datalen)
|
||
|
{
|
||
|
return(PAX_sign(sig,privkey,GENESIS_PUBKEY,timestamp,data,datalen));
|
||
|
}
|
||
|
|
||
|
uint64_t PAX_swaptx(bits256 privkey,struct PAX_sig *sig,uint32_t timestamp,uint8_t *data,int32_t datalen)
|
||
|
{
|
||
|
uint64_t othernxt;
|
||
|
if ( (othernxt= PAX_validate(sig,timestamp,data,datalen)) != sig->signer64bits )
|
||
|
return(0);
|
||
|
return(PAX_sign(sig,privkey,GENESIS_PUBKEY,timestamp,data,datalen));
|
||
|
}
|
||
|
|
||
|
struct accts777_info *accts777_init(char *dirname,struct txinds777_info *txinds)
|
||
|
{
|
||
|
struct accts777_info *accts = calloc(1,sizeof(*accts));
|
||
|
accts->hashaddrs = ramkv777_init(accts->numkvs++,"hashaddrs",sizeof(bits256),1);
|
||
|
accts->coinaddrs = ramkv777_init(accts->numkvs++,"coinaddrs",BTCDADDRSIZE,1);
|
||
|
accts->nxtaddrs = ramkv777_init(accts->numkvs++,"nxtaddrs",sizeof(uint64_t),1);
|
||
|
accts->SaMaddrs = ramkv777_init(accts->numkvs++,"SaMaddrs",sizeof(bits384),1);
|
||
|
accts->bets = ramkv777_init(accts->numkvs++,"bets",BTCDADDRSIZE,1);
|
||
|
accts->pricefeeds = ramkv777_init(accts->numkvs++,"pricefeeds",sizeof(uint32_t) * 2,1);//, accts->pricefeeds->dispflag = 0;
|
||
|
if ( accts->numkvs > ACCTS777_MAXRAMKVS )
|
||
|
{
|
||
|
printf("too many ramkvs for accts %d vs %d\n",accts->numkvs,ACCTS777_MAXRAMKVS);
|
||
8 years ago
|
iguana_exit(0,0);
|
||
9 years ago
|
}
|
||
|
accts->addrkvs[PEGGY_ADDRFUNDING] = accts->addrkvs[PEGGY_ADDRBTCD] = accts->coinaddrs;
|
||
|
accts->addrkvs[PEGGY_ADDR777] = accts->SaMaddrs;
|
||
|
accts->addrkvs[PEGGY_ADDRNXT] = accts->nxtaddrs;
|
||
|
accts->addrkvs[PEGGY_ADDRCREATE] = accts->addrkvs[PEGGY_ADDRUNIT] = accts->addrkvs[PEGGY_ADDRPUBKEY] = accts->hashaddrs;
|
||
|
if ( (accts->txinds= txinds) == 0 )
|
||
|
accts->txinds = txinds777_init(dirname,"txinds");
|
||
|
return(accts);
|
||
|
}
|
||
|
|
||
|
void accts777_free(struct accts777_info *accts)
|
||
|
{
|
||
|
int32_t i;
|
||
|
queue_free(&accts->PaymentsQ);
|
||
|
for (i=0; i<sizeof(accts->addrkvs)/sizeof(*accts->addrkvs); i++)
|
||
|
if ( accts->addrkvs[i] != 0 )
|
||
|
ramkv777_free(accts->addrkvs[i]);
|
||
|
free(accts);
|
||
|
}
|
||
|
|
||
|
struct accts777_info *accts777_clone(char *path,struct accts777_info *accts)
|
||
|
{
|
||
|
struct accts777_info *clone;
|
||
|
clone = accts777_init(path,accts->txinds);
|
||
|
queue_clone(&clone->PaymentsQ,&accts->PaymentsQ,sizeof(struct opreturn_payment));
|
||
|
if ( accts->numunits > 0 && accts->units != 0 )
|
||
|
{
|
||
|
clone->units = calloc(accts->numunits,sizeof(*accts->units));
|
||
|
memcpy(clone->units,accts->units,accts->numunits * sizeof(*accts->units));
|
||
|
}
|
||
|
clone->peggyhash = accts->peggyhash;
|
||
|
ramkv777_clone(clone->bets,accts->bets);
|
||
|
ramkv777_clone(clone->pricefeeds,accts->pricefeeds);
|
||
|
ramkv777_clone(clone->hashaddrs,accts->hashaddrs);
|
||
|
ramkv777_clone(clone->coinaddrs,accts->coinaddrs);
|
||
|
ramkv777_clone(clone->SaMaddrs,accts->SaMaddrs);
|
||
|
ramkv777_clone(clone->nxtaddrs,accts->nxtaddrs);
|
||
|
return(clone);
|
||
|
}
|
||
|
|
||
|
void *accts777_key(union peggy_addr *addr,int32_t type)
|
||
|
{
|
||
|
void *key;
|
||
|
switch ( type )
|
||
|
{
|
||
|
case PEGGY_ADDRFUNDING: case PEGGY_ADDRBTCD: key = &addr->coinaddr; break;
|
||
|
case PEGGY_ADDRNXT: key = &addr->nxt64bits; break;
|
||
|
case PEGGY_ADDR777: key = &addr->SaMbits; break;
|
||
|
case PEGGY_ADDRCREATE: key = &addr->newunit.sha256; break;
|
||
|
case PEGGY_ADDRUNIT: key = &addr->sha256; break;
|
||
|
case PEGGY_ADDRPUBKEY: key = &addr->sha256; break;
|
||
|
default: key = 0; break;
|
||
|
}
|
||
|
return(key);
|
||
|
}
|
||
|
|
||
|
struct acct777 *accts777_find(int32_t *valuesizep,struct accts777_info *accts,union peggy_addr *addr,int32_t type)
|
||
|
{
|
||
|
void *key;
|
||
|
if ( (key= accts777_key(addr,type)) != 0 )
|
||
|
return(ramkv777_read(valuesizep,accts->addrkvs[type],key));
|
||
|
else
|
||
|
{
|
||
|
if ( valuesizep != 0 )
|
||
|
*valuesizep = 0;
|
||
|
return(0);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
struct acct777 *accts777_create(struct accts777_info *accts,union peggy_addr *addr,int32_t type,uint32_t blocknum,uint32_t blocktimestamp)
|
||
|
{
|
||
|
struct acct777 *acct,A;
|
||
|
if ( (acct= accts777_find(0,accts,addr,type)) == 0 )
|
||
|
{
|
||
|
memset(&A,0,sizeof(A));
|
||
|
A.firstblocknum = blocknum, A.firsttimestamp = blocktimestamp;
|
||
|
acct = ramkv777_write(accts->addrkvs[type],accts777_key(addr,type),&A,sizeof(A));
|
||
|
}
|
||
|
else if ( blocknum < acct->firstblocknum || blocktimestamp < acct->firsttimestamp )
|
||
|
{
|
||
|
printf("accts777_create: already exists but with an earlier block/timestamp? %u:%u vs %u:%u\n",blocknum,acct->firstblocknum,blocktimestamp,acct->firsttimestamp);
|
||
|
return(0);
|
||
|
}
|
||
|
return(acct);
|
||
|
}
|
||
|
|
||
|
void peggy_delete(struct accts777_info *accts,struct peggy_unit *U,int32_t reason)
|
||
|
{
|
||
|
memcpy(U,&accts->units[--accts->numunits],sizeof(struct peggy_unit));
|
||
|
//U->redeemed = (uint32_t)time(NULL);
|
||
|
//if ( PEGS->lockhashes != 0 )
|
||
|
// kv777_delete(PEGS->lockhashes,U->lockPeriodHash,HASH_SIZE);
|
||
|
}
|
||
|
|
||
|
int32_t peggy_addunit(struct accts777_info *accts,struct peggy_unit *U,bits256 lockhash)
|
||
|
{
|
||
|
U->lockhash = lockhash;
|
||
|
accts->units = realloc(accts->units,sizeof(*accts->units) * (accts->numunits + 1));
|
||
|
accts->units[accts->numunits] = *U;
|
||
|
//if ( PEGS->lockhashes != 0 )
|
||
|
// kv777_write(PEGS->lockhashes,lockPeriodHash,HASH_SIZE,U,sizeof(*U));
|
||
|
return(accts->numunits++);
|
||
|
}
|
||
|
|
||
|
struct peggy_unit *peggy_match(struct accts777_info *accts,int32_t peg,uint64_t nxt64bits,bits256 lockhash,uint16_t lockdays)
|
||
|
{
|
||
|
int32_t i,size; struct peggy_unit *U;
|
||
|
if ( accts->hashaddrs == 0 )
|
||
|
{
|
||
|
for (i=0,U=&accts->units[0]; i<accts->numunits; i++,U++)
|
||
|
{
|
||
|
//if ( U->nxt64bits == 0 || U->nxt64bits == nxt64bits )
|
||
|
{
|
||
|
if ( U->lock.peg == peg && lockdays >= U->lock.minlockdays && lockdays <= U->lock.maxlockdays )
|
||
|
{
|
||
|
if ( memcmp(lockhash.bytes,U->lockhash.bytes,sizeof(lockhash)) == 0 )
|
||
|
return(U);
|
||
|
return(0);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
size = sizeof(*U);
|
||
|
if ( (U= ramkv777_read(&size,accts->hashaddrs,lockhash.bytes)) != 0 && size == sizeof(U) )
|
||
|
{
|
||
|
//if ( U->nxt64bits == 0 || U->nxt64bits == nxt64bits )
|
||
|
{
|
||
|
if ( U->lock.peg == peg && lockdays >= U->lock.minlockdays && lockdays <= U->lock.maxlockdays )
|
||
|
return(U);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
return(0);
|
||
|
}
|
||
|
|
||
|
int32_t peggy_swap(struct accts777_info *accts,uint64_t signerA,uint64_t signerB,bits256 hashA,bits256 hashB)
|
||
|
{
|
||
|
struct peggy_unit *U,*U2; int32_t size; uint64_t nxtA,nxtB;
|
||
|
size = sizeof(*U);
|
||
|
if ( (U= ramkv777_read(&size,accts->hashaddrs,hashA.bytes)) != 0 && size == sizeof(U) )
|
||
|
{
|
||
|
if ( (U2= ramkv777_read(&size,accts->hashaddrs,hashB.bytes)) != 0 && size == sizeof(U2) )
|
||
|
{
|
||
|
nxtA = acct777_nxt64bits(hashA), nxtB = acct777_nxt64bits(hashB);
|
||
|
if ( (nxtA == signerA && nxtB == signerB) || (nxtA == signerB && nxtB == signerA) )
|
||
|
{
|
||
|
// need to verify ownership
|
||
|
U2->lockhash = hashA, U->lockhash = hashB;
|
||
|
return(0);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
return(-1);
|
||
|
}
|
||
|
|
||
|
int32_t acct777_pay(struct accts777_info *accts,struct acct777 *srcacct,struct acct777 *acct,int64_t value,uint32_t blocknum,uint32_t blocktimestamp)
|
||
|
{
|
||
|
if ( srcacct != 0 )
|
||
|
{
|
||
|
if ( srcacct->balance < value )
|
||
|
return(-1);
|
||
|
srcacct->balance -= value;
|
||
|
}
|
||
|
acct->balance += value;
|
||
|
return(0);
|
||
|
}
|
||
|
|
||
|
int64_t acct777_balance(struct accts777_info *accts,uint32_t blocknum,uint32_t blocktimestamp,union peggy_addr *addr,int32_t type)
|
||
|
{
|
||
|
int64_t balance = 0;
|
||
|
struct acct777 *acct;
|
||
|
if ( (acct= accts777_find(0,accts,addr,type)) != 0 )
|
||
|
balance = acct->balance;
|
||
|
return(balance);
|
||
|
}
|
||
|
|
||
|
int32_t peggy_flush(void *_PEGS,uint32_t currentblocknum,uint32_t blocknum,uint32_t blocktimestamp)
|
||
|
{
|
||
|
struct peggy_info *PEGS = _PEGS;
|
||
|
if ( PEGS != 0 && PEGS->accts != 0 )
|
||
|
return(txinds777_flush(PEGS->accts->txinds,blocknum,blocktimestamp));
|
||
|
else return(-1);
|
||
|
}
|
||
|
|
||
|
int32_t peggy_payments(queue_t *PaymentsQ,struct opreturn_payment *payments,int32_t max,uint32_t currentblocknum,uint32_t blocknum,uint32_t blocktimestamp)
|
||
|
{
|
||
|
struct opreturn_payment *payment; int32_t n = 0;
|
||
|
while ( max > 0 && (payment= queue_dequeue(PaymentsQ,0)) != 0 )
|
||
|
{
|
||
|
if ( payment->value != 0 && payment->coinaddr[0] != 0 )
|
||
|
*payments++ = *payment;
|
||
|
free(payment);
|
||
|
n++;
|
||
|
}
|
||
|
return(n);
|
||
|
}
|
||
|
|
||
|
int32_t peggy_emit(void *_PEGS,uint8_t opreturndata[MAX_OPRETURNSIZE],struct opreturn_payment *payments,int32_t max,uint32_t currentblocknum,uint32_t blocknum,uint32_t blocktimestamp)
|
||
|
{
|
||
|
char *opreturnstr; int32_t nonz,len = 0; struct peggy_info *PEGS = _PEGS;
|
||
|
if ( payments != 0 && max > 1 && PEGS->accts != 0 && peggy_payments(&PEGS->accts->PaymentsQ,payments,max,currentblocknum,blocknum,blocktimestamp) < 0 )
|
||
|
return(-1);
|
||
|
if ( opreturndata != 0 && (opreturnstr= peggy_emitprices(&nonz,PEGS,blocktimestamp,0)) != 0 )
|
||
|
{
|
||
|
memset(opreturndata,0,MAX_OPRETURNSIZE);
|
||
|
len = (int32_t)strlen(opreturnstr) / 2;
|
||
|
decode_hex(opreturndata,len,opreturnstr);
|
||
|
free(opreturnstr);
|
||
|
}
|
||
|
return(len);
|
||
|
}
|
||
|
|
||
|
uint32_t peggy_clone(char *path,void *dest,void *src)
|
||
|
{
|
||
|
struct peggy_info *destPEGS,*srcPEGS;
|
||
|
printf("inside peggy_clone sizeof peggy_info %d %d %d\n",(int32_t)sizeof(*destPEGS),(int32_t)sizeof(destPEGS->pricedpegs),(int32_t)sizeof(destPEGS->pairedpegs));//, getchar();
|
||
|
destPEGS = dest, srcPEGS = src;
|
||
|
*destPEGS = *srcPEGS;
|
||
|
destPEGS->accts = accts777_clone(path,srcPEGS->accts);
|
||
|
return(0);
|
||
|
}
|