/******************************************************************************
 * 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 "SuperNET.h"

//struct iguana_txid { bits256 txid; uint32_t txidind,firstvout,firstvin,locktime,version,timestamp; uint16_t numvouts,numvins; } __attribute__((packed));

//struct iguana_msgvin { bits256 prev_hash; uint8_t *script; uint32_t prev_vout,scriptlen,sequence; } __attribute__((packed));

//struct iguana_spend { uint32_t spendtxidind; int16_t prevout; uint16_t tbd:14,external:1,diffsequence:1; } __attribute__((packed));

void iguana_vinset(struct iguana_info *coin,int32_t height,struct iguana_msgvin *vin,struct iguana_txid *tx,int32_t i)
{
    struct iguana_spend *s,*S; uint32_t spendind,unspentind; struct iguana_bundle *bp;
    struct iguana_ramchaindata *rdata; struct iguana_txid *T; bits256 *X;
    memset(vin,0,sizeof(*vin));
    if ( height >= 0 && height < coin->chain->bundlesize*coin->bundlescount && (bp= coin->bundles[height / coin->chain->bundlesize]) != 0 && (rdata= bp->ramchain.H.data) != 0 )
    {
        S = (void *)(long)((long)rdata + rdata->Soffset);
        X = (void *)(long)((long)rdata + rdata->Xoffset);
        T = (void *)(long)((long)rdata + rdata->Toffset);
        spendind = (tx->firstvin + i);
        s = &S[spendind];
        if ( s->sequenceid == 1 )
            vin->sequence = 0xffffffff;
        else if ( s->sequenceid == 2 )
            vin->sequence = 0xfffffffe;
        else vin->sequence = 0;
        vin->prev_vout = s->prevout;
        iguana_ramchain_spendtxid(coin,&unspentind,&vin->prev_hash,T,rdata->numtxids,X,rdata->numexternaltxids,s);
    }
}

int32_t iguana_voutset(struct iguana_info *coin,uint8_t *scriptspace,char *asmstr,int32_t height,struct iguana_msgvout *vout,struct iguana_txid *tx,int32_t i)
{
    struct iguana_ramchaindata *rdata; uint32_t unspentind,scriptlen = 0; struct iguana_bundle *bp;
    struct iguana_unspent *u,*U; char coinaddr[65]; struct iguana_pkhash *P,*p; struct vin_info V;
    memset(vout,0,sizeof(*vout));
    if ( height >= 0 && height < coin->chain->bundlesize*coin->bundlescount && (bp= coin->bundles[height / coin->chain->bundlesize]) != 0  && (rdata= bp->ramchain.H.data) != 0 && i < tx->numvouts )
    {
        U = (void *)(long)((long)rdata + rdata->Uoffset);
        P = (void *)(long)((long)rdata + rdata->Poffset);
        unspentind = (tx->firstvout + i);
        u = &U[unspentind];
        if ( u->txidind != tx->txidind || u->vout != i || u->hdrsi != height / coin->chain->bundlesize )
            printf("iguana_voutset: txidind mismatch %d vs %d || %d vs %d || (%d vs %d)\n",u->txidind,u->txidind,u->vout,i,u->hdrsi,height / coin->chain->bundlesize);
        p = &P[u->pkind];
        vout->value = u->value;
        vout->pk_script = scriptspace;
        memset(&V,0,sizeof(V));
        scriptlen = iguana_scriptgen(coin,&V.M,&V.N,coinaddr,scriptspace,asmstr,p->rmd160,u->type,(const struct vin_info *)&V,i);
    }
    vout->pk_scriptlen = scriptlen;
    return(scriptlen);
}

struct iguana_txid *iguana_blocktx(struct iguana_info *coin,struct iguana_txid *tx,struct iguana_block *block,int32_t i)
{
    struct iguana_bundle *bp; uint32_t txidind;
    if ( i >= 0 && i < block->RO.txn_count )
    {
        if ( block->height >= 0 ) //
        {
            if ( (bp= coin->bundles[block->hdrsi]) != 0 )
            {
                if ( (txidind= block->RO.firsttxidind) > 0 )//bp->firsttxidinds[block->bundlei]) > 0 )
                {
                    if ( iguana_bundletx(coin,bp,block->bundlei,tx,txidind+i) == tx )
                        return(tx);
                    printf("error getting txidind.%d + i.%d from hdrsi.%d\n",txidind,i,block->hdrsi);
                    return(0);
                } else printf("iguana_blocktx null txidind\n");
            } else printf("iguana_blocktx no bp\n");
        }
    } else printf("i.%d vs txn_count.%d\n",i,block->RO.txn_count);
    return(0);
}

cJSON *iguana_blockjson(struct iguana_info *coin,struct iguana_block *block,int32_t txidsflag)
{
    char str[65],hexstr[1024]; int32_t i,len; struct iguana_txid *tx,T; struct iguana_msgblock msg;
    bits256 hash2; uint8_t serialized[1024]; cJSON *array,*json = cJSON_CreateObject();
    jaddstr(json,"result","success");
    jaddstr(json,"blockhash",bits256_str(str,block->RO.hash2));
    jaddnum(json,"height",block->height);
    //jaddnum(json,"ipbits",block->fpipbits);
    jaddstr(json,"merkle_root",bits256_str(str,block->RO.merkle_root));
    jaddstr(json,"prev_block",bits256_str(str,block->RO.prev_block));
    jaddnum(json,"timestamp",block->RO.timestamp);
    jaddstr(json,"utc",utc_str(str,block->RO.timestamp));
    jaddnum(json,"nonce",block->RO.nonce);
    jaddnum(json,"version",block->RO.version);
    jaddnum(json,"numvouts",block->RO.numvouts);
    jaddnum(json,"numvins",block->RO.numvins);
    jaddnum(json,"recvlen",block->RO.recvlen);
    jaddnum(json,"hdrsi",block->hdrsi);
    jaddnum(json,"PoW",block->PoW);
    jaddnum(json,"bundlei",block->bundlei);
    jaddnum(json,"mainchain",block->mainchain);
    jaddnum(json,"valid",block->valid);
    jaddnum(json,"txn_count",block->RO.txn_count);
    
    jaddnum(json,"nBits",block->RO.bits);
    serialized[0] = ((uint8_t *)&block->RO.bits)[3];
    serialized[1] = ((uint8_t *)&block->RO.bits)[2];
    serialized[2] = ((uint8_t *)&block->RO.bits)[1];
    serialized[3] = ((uint8_t *)&block->RO.bits)[0];
    init_hexbytes_noT(hexstr,serialized,sizeof(uint32_t));
    jaddstr(json,"nBitshex",hexstr);
    memset(&msg,0,sizeof(msg));
    msg.H.version = block->RO.version;
    msg.H.merkle_root = block->RO.merkle_root;
    msg.H.timestamp = block->RO.timestamp;
    msg.H.bits = block->RO.bits;
    msg.H.nonce = block->RO.nonce;
    msg.txn_count = 0;//block->RO.txn_count;
    len = iguana_rwblock(1,&hash2,serialized,&msg);
    init_hexbytes_noT(hexstr,serialized,len);
    jaddstr(json,"blockheader",hexstr);
    if ( txidsflag != 0 )
    {
        array = cJSON_CreateArray();
        for (i=0; i<block->RO.txn_count; i++)
        {
            if ( (tx= iguana_blocktx(coin,&T,block,i)) != 0 )
                jaddistr(array,bits256_str(str,tx->txid));
        }
        jadd(json,"txids",array);
        //printf("add txids[%d]\n",block->txn_count);
    }
    return(json);
}


/*
 //char *hashstr,*txidstr,*coinaddr,*txbytes,rmd160str[41],str[65]; int32_t len,height,i,n,valid = 0;
 //cJSON *addrs,*retjson,*retitem; uint8_t rmd160[20],addrtype; bits256 hash2,checktxid;
 //memset(&hash2,0,sizeof(hash2)); struct iguana_txid *tx,T; struct iguana_block *block = 0;
 
 if ( (coinaddr= jstr(json,"address")) != 0 )
 {
 if ( btc_addr2univ(&addrtype,rmd160,coinaddr) == 0 )
 {
 if ( addrtype == coin->chain->pubval || addrtype == coin->chain->p2shval )
 valid = 1;
 else return(clonestr("{\"error\":\"invalid addrtype\"}"));
 } else return(clonestr("{\"error\":\"cant convert address to rmd160\"}"));
 }
 if ( strcmp(method,"block") == 0 )
 {
 height = -1;
 if ( ((hashstr= jstr(json,"blockhash")) != 0 || (hashstr= jstr(json,"hash")) != 0) && strlen(hashstr) == sizeof(bits256)*2 )
 decode_hex(hash2.bytes,sizeof(hash2),hashstr);
 else
 {
 height = juint(json,"height");
 hash2 = iguana_blockhash(coin,height);
 }
 retitem = cJSON_CreateObject();
 if ( (block= iguana_blockfind(coin,hash2)) != 0 )
 {
 if ( (height >= 0 && block->height == height) || memcmp(hash2.bytes,block->RO.hash2.bytes,sizeof(hash2)) == 0 )
 {
 char str[65],str2[65]; printf("hash2.(%s) -> %s\n",bits256_str(str,hash2),bits256_str(str2,block->RO.hash2));
 return(jprint(iguana_blockjson(coin,block,juint(json,"txids")),1));
 }
 }
 else return(clonestr("{\"error\":\"cant find block\"}"));
 }
 else if ( strcmp(method,"tx") == 0 )
 {
 if ( ((txidstr= jstr(json,"txid")) != 0 || (txidstr= jstr(json,"hash")) != 0) && strlen(txidstr) == sizeof(bits256)*2 )
 {
 retitem = cJSON_CreateObject();
 decode_hex(hash2.bytes,sizeof(hash2),txidstr);
 if ( (tx= iguana_txidfind(coin,&height,&T,hash2)) != 0 )
 {
 jadd(retitem,"tx",iguana_txjson(coin,tx,height));
 return(jprint(retitem,1));
 }
 return(clonestr("{\"error\":\"cant find txid\"}"));
 }
 else return(clonestr("{\"error\":\"invalid txid\"}"));
 }
 else if ( strcmp(method,"rawtx") == 0 )
 {
 if ( ((txidstr= jstr(json,"txid")) != 0 || (txidstr= jstr(json,"hash")) != 0) && strlen(txidstr) == sizeof(bits256)*2 )
 {
 decode_hex(hash2.bytes,sizeof(hash2),txidstr);
 if ( (tx= iguana_txidfind(coin,&height,&T,hash2)) != 0 )
 {
 if ( (len= iguana_txbytes(coin,coin->blockspace,sizeof(coin->blockspace),&checktxid,tx,height,0,0)) > 0 )
 {
 txbytes = mycalloc('x',1,len*2+1);
 init_hexbytes_noT(txbytes,coin->blockspace,len*2+1);
 retitem = cJSON_CreateObject();
 jaddstr(retitem,"txid",bits256_str(str,hash2));
 jaddnum(retitem,"height",height);
 jaddstr(retitem,"rawtx",txbytes);
 myfree(txbytes,len*2+1);
 return(jprint(retitem,1));
 } else return(clonestr("{\"error\":\"couldnt generate txbytes\"}"));
 }
 return(clonestr("{\"error\":\"cant find txid\"}"));
 }
 else return(clonestr("{\"error\":\"invalid txid\"}"));
 }
 else if ( strcmp(method,"txs") == 0 )
 {
 if ( ((hashstr= jstr(json,"block")) != 0 || (hashstr= jstr(json,"blockhash")) != 0) && strlen(hashstr) == sizeof(bits256)*2 )
 {
 decode_hex(hash2.bytes,sizeof(hash2),hashstr);
 if ( (block= iguana_blockfind(coin,hash2)) == 0 )
 return(clonestr("{\"error\":\"cant find blockhash\"}"));
 }
 else if ( jobj(json,"height") != 0 )
 {
 height = juint(json,"height");
 hash2 = iguana_blockhash(coin,height);
 if ( (block= iguana_blockfind(coin,hash2)) == 0 )
 return(clonestr("{\"error\":\"cant find block at height\"}"));
 }
 else if ( valid == 0 )
 return(clonestr("{\"error\":\"txs needs blockhash or height or address\"}"));
 retitem = cJSON_CreateArray();
 if ( block != 0 )
 {
 for (i=0; i<block->RO.txn_count; i++)
 {
 if ( (tx= iguana_blocktx(coin,&T,block,i)) != 0 )
 jaddi(retitem,iguana_txjson(coin,tx,-1));
 }
 }
 else
 {
 init_hexbytes_noT(rmd160str,rmd160,20);
 jaddnum(retitem,"addrtype",addrtype);
 jaddstr(retitem,"rmd160",rmd160str);
 jaddstr(retitem,"txlist","get list of all tx for this address");
 }
 return(jprint(retitem,1));
 }
 
 else
 {
 n = 0;
 if ( valid == 0 )
 {
 if ( (addrs= jarray(&n,json,"addrs")) == 0 )
 return(clonestr("{\"error\":\"need address or addrs\"}"));
 }
 for (i=0; i<=n; i++)
 {
 retitem = cJSON_CreateObject();
 if ( i > 0 )
 retjson = cJSON_CreateArray();
 if ( i > 0 )
 {
 if ( (coinaddr= jstr(jitem(addrs,i-1),0)) == 0 )
 return(clonestr("{\"error\":\"missing address in addrs\"}"));
 if ( btc_addr2univ(&addrtype,rmd160,coinaddr) < 0 )
 {
 free_json(retjson);
 return(clonestr("{\"error\":\"illegal address in addrs\"}"));
 }
 if ( addrtype != coin->chain->pubval && addrtype != coin->chain->p2shval )
 return(clonestr("{\"error\":\"invalid addrtype in addrs\"}"));
 }
 if ( strcmp(method,"utxo") == 0 )
 {
 jaddstr(retitem,"utxo","utxo entry");
 }
 else if ( strcmp(method,"unconfirmed") == 0 )
 {
 jaddstr(retitem,"unconfirmed","unconfirmed entry");
 }
 else if ( strcmp(method,"balance") == 0 )
 {
 jaddstr(retitem,"balance","balance entry");
 }
 else if ( strcmp(method,"totalreceived") == 0 )
 {
 jaddstr(retitem,"totalreceived","totalreceived entry");
 }
 else if ( strcmp(method,"totalsent") == 0 )
 {
 jaddstr(retitem,"totalsent","totalsent entry");
 }
 else if ( strcmp(method,"validateaddress") == 0 )
 {
 jaddstr(retitem,"validate",coinaddr);
 }
 if ( n == 0 )
 return(jprint(retitem,1));
 else jaddi(retjson,retitem);
 }
 return(jprint(retjson,1));
 }
*/

/*
 char *iguana_listsinceblock(struct supernet_info *myinfo,struct iguana_info *coin,bits256 blockhash,int32_t target)
 {
 cJSON *retitem = cJSON_CreateObject();
 return(jprint(retitem,1));
 }
 
 char *iguana_getinfo(struct supernet_info *myinfo,struct iguana_info *coin)
 {
 cJSON *retitem = cJSON_CreateObject();
 jaddstr(retitem,"result",coin->statusstr);
 return(jprint(retitem,1));
 }
 
 char *iguana_getbestblockhash(struct supernet_info *myinfo,struct iguana_info *coin)
 {
 cJSON *retitem = cJSON_CreateObject();
 char str[65]; jaddstr(retitem,"result",bits256_str(str,coin->blocks.hwmchain.RO.hash2));
 return(jprint(retitem,1));
 }
 
 char *iguana_getblockcount(struct supernet_info *myinfo,struct iguana_info *coin)
 {
 cJSON *retitem = cJSON_CreateObject();
 jaddnum(retitem,"result",coin->blocks.hwmchain.height);
 return(jprint(retitem,1));
 }*/