1497 lines
62 KiB
1497 lines
62 KiB
9 years ago
|
/******************************************************************************
|
||
|
* 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. *
|
||
|
* *
|
||
|
******************************************************************************/
|
||
|
|
||
|
#ifndef xcode_subatomic_h
|
||
|
#define xcode_subatomic_h
|
||
|
|
||
|
//https://bitcointalk.org/index.php?topic=1172153.0
|
||
|
|
||
|
#include <stdbool.h>
|
||
|
|
||
|
struct bp_key { void *k; };
|
||
|
typedef struct cstring {
|
||
|
char *str; // string data, incl. NUL
|
||
|
size_t len; // length of string, not including NUL
|
||
|
size_t alloc; // total allocated buffer length
|
||
|
} cstring;
|
||
|
|
||
|
extern bool bp_key_init(struct bp_key *key);
|
||
|
extern void bp_key_free(struct bp_key *key);
|
||
|
extern bool bp_key_generate(struct bp_key *key);
|
||
|
extern bool bp_privkey_set(struct bp_key *key, const void *privkey, size_t pk_len);
|
||
|
extern bool bp_pubkey_set(struct bp_key *key, const void *pubkey, size_t pk_len);
|
||
|
extern bool bp_key_secret_set(struct bp_key *key, const void *privkey_, size_t pk_len);
|
||
|
extern bool bp_privkey_get(const struct bp_key *key, void **privkey, size_t *pk_len);
|
||
|
extern bool bp_pubkey_get(const struct bp_key *key, void **pubkey, size_t *pk_len);
|
||
|
extern bool bp_key_secret_get(void *p, size_t len, const struct bp_key *key);
|
||
|
extern bool bp_sign(const struct bp_key *key, const void *data, size_t data_len,void **sig_, size_t *sig_len_);
|
||
|
extern bool bp_verify(const struct bp_key *key, const void *data, size_t data_len,const void *sig, size_t sig_len);
|
||
|
|
||
|
void cstr_free(cstring *s, bool free_buf);
|
||
|
cstring *base58_encode_check(unsigned char addrtype,bool have_addrtype,const void *data,size_t data_len);
|
||
|
cstring *base58_decode_check(unsigned char *addrtype, const char *s_in);
|
||
|
int32_t btc_setprivkey(struct bp_key *key,char *privkeystr);
|
||
|
int32_t btc_getpubkey(char pubkeystr[67],uint8_t pubkeybuf[33],struct bp_key *key);
|
||
|
void btc_freekey(void *key);
|
||
|
|
||
|
struct btcaddr
|
||
|
{
|
||
|
struct bp_key key;
|
||
|
uint8_t *pubkey; uint16_t p2sh;
|
||
|
char addr[36],coin[8];
|
||
|
uint8_t privkey[280];
|
||
|
};
|
||
|
|
||
|
#define SCRIPT_OP_IF 0x63
|
||
|
#define SCRIPT_OP_ELSE 0x67
|
||
|
#define SCRIPT_OP_DUP 0x76
|
||
|
#define SCRIPT_OP_ENDIF 0x68
|
||
|
#define SCRIPT_OP_TRUE 0x51
|
||
|
#define SCRIPT_OP_2 0x52
|
||
|
#define SCRIPT_OP_3 0x53
|
||
|
#define SCRIPT_OP_EQUALVERIFY 0x88
|
||
|
#define SCRIPT_OP_HASH160 0xa9
|
||
|
#define SCRIPT_OP_EQUAL 0x87
|
||
|
#define SCRIPT_OP_CHECKSIG 0xac
|
||
|
#define SCRIPT_OP_CHECKMULTISIG 0xae
|
||
|
#define SCRIPT_OP_CHECKMULTISIGVERIFY 0xaf
|
||
|
|
||
|
char *create_atomictx_scripts(uint8_t addrtype,char *scriptPubKey,char *p2shaddr,char *pubkeyA,char *pubkeyB,char *hash160str)
|
||
|
{
|
||
|
// if ( refund ) OP_HASH160 <2of2 multisig hash> OP_EQUAL // standard multisig
|
||
|
// else OP_DUP OP_HASH160 <hash160> OP_EQUALVERIFY OP_CHECKSIG // standard spend
|
||
|
cstring *btc_addr; char *retstr; uint8_t pubkeyAbytes[33],pubkeyBbytes[33],hash160[20],tmpbuf[24],hex[4096]; int32_t i,n = 0;
|
||
|
decode_hex(pubkeyAbytes,33,pubkeyA);
|
||
|
decode_hex(pubkeyBbytes,33,pubkeyB);
|
||
|
decode_hex(hash160,20,hash160str);
|
||
|
hex[n++] = SCRIPT_OP_IF;
|
||
|
hex[n++] = SCRIPT_OP_2;
|
||
|
hex[n++] = 33, memcpy(&hex[n],pubkeyAbytes,33), n += 33;
|
||
|
hex[n++] = 33, memcpy(&hex[n],pubkeyBbytes,33), n += 33;
|
||
|
hex[n++] = SCRIPT_OP_2;
|
||
|
hex[n++] = SCRIPT_OP_CHECKMULTISIG;
|
||
|
hex[n++] = SCRIPT_OP_ELSE;
|
||
|
hex[n++] = SCRIPT_OP_DUP;
|
||
|
hex[n++] = SCRIPT_OP_HASH160;
|
||
|
hex[n++] = 20; memcpy(&hex[n],hash160,20); n += 20;
|
||
|
hex[n++] = SCRIPT_OP_EQUALVERIFY;
|
||
|
hex[n++] = SCRIPT_OP_CHECKSIG;
|
||
|
hex[n++] = SCRIPT_OP_ENDIF;
|
||
|
if ( (retstr= calloc(1,n*2+16)) == 0 )
|
||
|
return(0);
|
||
|
//printf("pubkeyA.(%s) pubkeyB.(%s) hash160.(%s) ->\n",pubkeyA,pubkeyB,hash160str);
|
||
|
//strcpy(retstr,"01");
|
||
|
//sprintf(retstr+2,"%02x",n);
|
||
|
for (i=0; i<n; i++)
|
||
|
{
|
||
|
retstr[i*2] = hexbyte((hex[i]>>4) & 0xf);
|
||
|
retstr[i*2 + 1] = hexbyte(hex[i] & 0xf);
|
||
|
//printf("%02x",hex[i]);
|
||
|
}
|
||
|
retstr[n*2] = 0;
|
||
|
calc_OP_HASH160(scriptPubKey,tmpbuf+2,retstr);
|
||
|
tmpbuf[0] = SCRIPT_OP_HASH160;
|
||
|
tmpbuf[1] = 20;
|
||
|
tmpbuf[22] = SCRIPT_OP_EQUAL;
|
||
|
init_hexbytes_noT(scriptPubKey,tmpbuf,23);
|
||
|
if ( p2shaddr != 0 )
|
||
|
{
|
||
|
p2shaddr[0] = 0;
|
||
|
if ( (btc_addr= base58_encode_check(addrtype,true,tmpbuf+2,20)) != 0 )
|
||
|
{
|
||
|
if ( strlen(btc_addr->str) < 36 )
|
||
|
strcpy(p2shaddr,btc_addr->str);
|
||
|
cstr_free(btc_addr,true);
|
||
|
}
|
||
|
}
|
||
|
return(retstr);
|
||
|
}
|
||
|
|
||
|
#ifdef noiguana
|
||
|
int32_t create_MofN(uint8_t addrtype,char *redeemScript,char *scriptPubKey,char *p2shaddr,char *pubkeys[],int32_t M,int32_t N)
|
||
|
{
|
||
|
cstring *btc_addr; uint8_t pubkey[33],tmpbuf[24],hex[4096]; int32_t i,n = 0;
|
||
|
hex[n++] = 0x50 + M;
|
||
|
for (i=0; i<N; i++)
|
||
|
{
|
||
|
decode_hex(pubkey,33,pubkeys[i]);
|
||
|
hex[n++] = 33;
|
||
|
memcpy(&hex[n],pubkey,33);
|
||
|
n += 33;
|
||
|
}
|
||
|
hex[n++] = 0x50 + N;
|
||
|
hex[n++] = SCRIPT_OP_CHECKMULTISIG;
|
||
|
for (i=0; i<n; i++)
|
||
|
{
|
||
|
redeemScript[i*2] = hexbyte((hex[i]>>4) & 0xf);
|
||
|
redeemScript[i*2 + 1] = hexbyte(hex[i] & 0xf);
|
||
|
//fprintf(stderr,"%02x",hex[i]);
|
||
|
}
|
||
|
//fprintf(stderr," n.%d\n",n);
|
||
|
redeemScript[n*2] = 0;
|
||
|
calc_OP_HASH160(0,tmpbuf+2,redeemScript);
|
||
|
//printf("op160.(%s)\n",redeemScript);
|
||
|
tmpbuf[0] = SCRIPT_OP_HASH160;
|
||
|
tmpbuf[1] = 20;
|
||
|
tmpbuf[22] = SCRIPT_OP_EQUAL;
|
||
|
init_hexbytes_noT(scriptPubKey,tmpbuf,23);
|
||
|
p2shaddr[0] = 0;
|
||
|
if ( (btc_addr= base58_encode_check(addrtype,true,tmpbuf+2,20)) != 0 )
|
||
|
{
|
||
|
if ( strlen(btc_addr->str) < 36 )
|
||
|
strcpy(p2shaddr,btc_addr->str);
|
||
|
cstr_free(btc_addr,true);
|
||
|
}
|
||
|
return(n);
|
||
|
}
|
||
|
|
||
|
struct btcaddr *btcaddr_new(char *coinstr,char *p2sh_script)
|
||
|
{
|
||
|
uint8_t script[8192],md160[20]; char pubkeystr[512],privkeystr[512],hashstr[41]; struct coin777 *coin;
|
||
|
void *privkey=0,*pubkey=0; int32_t n; size_t len,slen; cstring *btc_addr; struct btcaddr *btc;
|
||
|
if ( (btc= calloc(1,sizeof(*btc))) == 0 || (coin = coin777_find(coinstr,1)) == 0 )
|
||
|
{
|
||
|
if ( btc != 0 )
|
||
|
free(btc);
|
||
|
return(0);
|
||
|
}
|
||
|
strncpy(btc->coin,coin->name,sizeof(btc->coin)-1);
|
||
|
if ( p2sh_script != 0 )
|
||
|
{
|
||
|
calc_OP_HASH160(0,md160,p2sh_script);
|
||
|
btc->p2sh = n = (int32_t)strlen(p2sh_script) >> 1;
|
||
|
decode_hex(script,n,p2sh_script);
|
||
|
if ( (btc_addr= base58_encode_check(coin->p2shtype,true,md160,sizeof(md160))) != 0 )
|
||
|
{
|
||
|
if ( n > sizeof(btc->privkey)-23 )
|
||
|
{
|
||
|
printf("script.(%s) len.%d is too big\n",p2sh_script,n);
|
||
|
free(btc);
|
||
|
return(0);
|
||
|
}
|
||
|
strcpy(btc->addr,btc_addr->str);
|
||
|
memcpy(btc->privkey,script,n);
|
||
|
btc->pubkey = &btc->privkey[sizeof(btc->privkey) - 23];
|
||
|
btc->pubkey[0] = SCRIPT_OP_HASH160;
|
||
|
btc->pubkey[2] = 20;
|
||
|
memcpy(&btc->pubkey[2],md160,20);
|
||
|
btc->pubkey[22] = SCRIPT_OP_EQUAL;
|
||
|
init_hexbytes_noT(privkeystr,script,n);
|
||
|
printf("type.%u btcaddr.%ld addr.(%s) %ld p2sh.(%s) %d\n",coin->p2shtype,(long)sizeof(struct btcaddr),btc->addr,(long)strlen(btc->addr),privkeystr,n);
|
||
|
cstr_free(btc_addr,true);
|
||
|
} else free(btc), btc = 0;
|
||
|
return(btc);
|
||
|
}
|
||
|
else if ( bp_key_init(&btc->key) != 0 && bp_key_generate(&btc->key) != 0 && bp_pubkey_get(&btc->key,&pubkey,&len) != 0 && bp_privkey_get(&btc->key,&privkey,&slen) != 0 )
|
||
|
{
|
||
|
if ( len == 33 && slen == 214 && memcmp((void *)((long)privkey + slen - 33),pubkey,33) == 0 )
|
||
|
{
|
||
|
init_hexbytes_noT(pubkeystr,pubkey,len);
|
||
|
init_hexbytes_noT(privkeystr,privkey,slen);
|
||
|
calc_OP_HASH160(hashstr,md160,pubkeystr);
|
||
|
if ( (btc_addr= base58_encode_check(coin->addrtype,true,md160,sizeof(md160))) != 0 )
|
||
|
{
|
||
|
strcpy(btc->addr,btc_addr->str);
|
||
|
memcpy(btc->privkey,privkey,slen);
|
||
|
btc->pubkey = &btc->privkey[slen - len];
|
||
|
printf("type.%u btcaddr.%ld rmd160.(%s) addr.(%s) %ld pubkey.(%s) %d privkey.(%s) %d\n",coin->addrtype,(long)sizeof(struct btcaddr),hashstr,btc->addr,(long)strlen(btc->addr),pubkeystr,(int32_t)len,privkeystr,(int32_t)slen);
|
||
|
cstr_free(btc_addr,true);
|
||
|
}
|
||
|
else free(btc), btc = 0;
|
||
|
} else free(btc), btc = 0;
|
||
|
}
|
||
|
return(btc);
|
||
|
}
|
||
|
|
||
|
int32_t btc_getpubkey(char pubkeystr[67],uint8_t pubkeybuf[33],struct bp_key *key)
|
||
|
{
|
||
|
void *pubkey = 0; size_t len = 0;
|
||
|
bp_pubkey_get(key,&pubkey,&len);
|
||
|
if ( pubkey != 0 )
|
||
|
{
|
||
|
if ( pubkeystr != 0 )
|
||
|
{
|
||
|
if ( len < 34 )
|
||
|
{
|
||
|
init_hexbytes_noT(pubkeystr,pubkey,(int32_t)len);
|
||
|
memcpy(pubkeybuf,pubkey,len);
|
||
|
}
|
||
|
else printf("btc_getpubkey error len.%d\n",(int32_t)len), len = -1;
|
||
|
}
|
||
|
//printf("btc_getpubkey len.%ld (%s).%p\n",len,pubkeystr,pubkeystr);
|
||
|
} else len = -1;
|
||
|
return((int32_t)len);
|
||
|
}
|
||
|
|
||
|
int32_t btc_convrmd160(char *coinaddr,uint8_t addrtype,uint8_t md160[20])
|
||
|
{
|
||
|
cstring *btc_addr;
|
||
|
if ( (btc_addr= base58_encode_check(addrtype,true,md160,20)) != 0 )
|
||
|
{
|
||
|
strcpy(coinaddr,btc_addr->str);
|
||
|
cstr_free(btc_addr,true);
|
||
|
return(0);
|
||
|
}
|
||
|
return(-1);
|
||
|
}
|
||
|
|
||
|
int32_t btc_coinaddr(char *coinaddr,uint8_t addrtype,char *pubkeystr)
|
||
|
{
|
||
|
uint8_t rmd160[20]; char hashstr[41];
|
||
|
calc_OP_HASH160(hashstr,rmd160,pubkeystr);
|
||
|
return(btc_convrmd160(coinaddr,addrtype,rmd160));
|
||
|
}
|
||
|
|
||
|
int32_t btc_convaddr(char *hexaddr,char *addr58)
|
||
|
{
|
||
|
uint8_t addrtype; cstring *cstr;
|
||
|
if ( (cstr= base58_decode_check(&addrtype,(const char *)addr58)) != 0 )
|
||
|
{
|
||
|
sprintf(hexaddr,"%02x",addrtype);
|
||
|
init_hexbytes_noT(hexaddr+2,(void *)cstr->str,cstr->len);
|
||
|
cstr_free(cstr,true);
|
||
|
return(0);
|
||
|
}
|
||
|
return(-1);
|
||
|
}
|
||
|
|
||
|
int32_t btc_priv2wip(char *wipstr,uint8_t privkey[32],uint8_t addrtype)
|
||
|
{
|
||
|
uint8_t tmp[128]; char hexstr[67]; cstring *btc_addr;
|
||
|
memcpy(tmp,privkey,32);
|
||
|
tmp[32] = 1;
|
||
|
init_hexbytes_noT(hexstr,tmp,32);
|
||
|
if ( (btc_addr= base58_encode_check(addrtype,true,tmp,33)) != 0 )
|
||
|
{
|
||
|
strcpy(wipstr,btc_addr->str);
|
||
|
cstr_free(btc_addr,true);
|
||
|
}
|
||
|
printf("-> (%s) -> wip.(%s) addrtype.%02x\n",hexstr,wipstr,addrtype);
|
||
|
return(0);
|
||
|
}
|
||
|
|
||
|
int32_t btc_wip2priv(uint8_t privkey[32],char *wipstr)
|
||
|
{
|
||
|
uint8_t addrtype; cstring *cstr; int32_t len = -1;
|
||
|
if ( (cstr= base58_decode_check(&addrtype,(const char *)wipstr)) != 0 )
|
||
|
{
|
||
|
init_hexbytes_noT((void *)privkey,(void *)cstr->str,cstr->len);
|
||
|
if ( cstr->str[cstr->len-1] == 0x01 )
|
||
|
cstr->len--;
|
||
|
memcpy(privkey,cstr->str,cstr->len);
|
||
|
len = (int32_t)cstr->len;
|
||
|
char tmp[138];
|
||
|
btc_priv2wip(tmp,privkey,addrtype);
|
||
|
printf("addrtype.%02x wipstr.(%llx) len.%d\n",addrtype,*(long long *)privkey,len);
|
||
|
cstr_free(cstr,true);
|
||
|
}
|
||
|
return(len);
|
||
|
}
|
||
|
|
||
|
int32_t btc_setprivkey(struct bp_key *key,char *privkeystr)
|
||
|
{
|
||
|
uint8_t privkey[512]; int32_t len = btc_wip2priv(privkey,privkeystr);
|
||
|
if ( len < 0 || bp_key_init(key) == 0 || bp_key_secret_set(key,privkey,len) == 0 )
|
||
|
{
|
||
|
printf("error setting privkey\n");
|
||
|
return(-1);
|
||
|
}
|
||
|
return(0);
|
||
|
}
|
||
|
|
||
|
void jumblr_freekey(void *key)
|
||
|
{
|
||
|
bp_key_free(key);
|
||
|
free(key);
|
||
|
}
|
||
|
|
||
|
int32_t btc_priv2pub(uint8_t pubkey[33],uint8_t privkey[32])
|
||
|
{
|
||
|
size_t len; void *pub = 0; int32_t retval = -1;
|
||
|
struct bp_key *key = calloc(1,sizeof(*key));
|
||
|
if ( key != 0 && bp_key_init(key) != 0 && bp_key_secret_set(key,privkey,32) != 0 )
|
||
|
{
|
||
|
bp_pubkey_get(key,&pub,&len);
|
||
|
bp_key_free(key);
|
||
|
if ( len == 33 )
|
||
|
memcpy(pubkey,pub,33);
|
||
|
if ( pub != 0 )
|
||
|
free(pub);
|
||
|
return(retval);
|
||
|
}
|
||
|
if ( key != 0 )
|
||
|
bp_key_free(key);
|
||
|
return(retval);
|
||
|
}
|
||
|
|
||
|
int32_t btc_pub2rmd(uint8_t rmd160[20],uint8_t pubkey[33])
|
||
|
{
|
||
|
char pubkeystr[67],hashstr[41];
|
||
|
init_hexbytes_noT(pubkeystr,pubkey,33);
|
||
|
calc_OP_HASH160(hashstr,rmd160,pubkeystr);
|
||
|
return(0);
|
||
|
}
|
||
|
|
||
|
void *jumblr_bpkey(char *pubP,struct coin777 *coin,char *coinaddr)
|
||
|
{
|
||
|
uint8_t buf[2048]; char *privkey; struct bp_key *key = 0;
|
||
|
//printf("coin.%s (%s)\n",coin->name,coinaddr);
|
||
|
if ( (privkey = dumpprivkey(coin->name,coin->serverport,coin->userpass,coinaddr)) != 0 )
|
||
|
{
|
||
|
//printf("privkey.(%s)\n",privkey);
|
||
|
key = calloc(1,sizeof(*key));
|
||
|
if ( key != 0 && btc_setprivkey(key,privkey) == 0 && btc_getpubkey(pubP,buf,key) > 0 )
|
||
|
return(key);
|
||
|
btc_freekey(key);
|
||
|
}
|
||
|
return(0);
|
||
|
}
|
||
|
|
||
|
void set_spendscript(char *spendscript,char *coinaddr)
|
||
|
{
|
||
|
char hexaddr[128];
|
||
|
btc_convaddr(hexaddr,coinaddr);
|
||
|
sprintf(spendscript,"76a914%s88ac",hexaddr+2);
|
||
|
}
|
||
|
|
||
|
int32_t script_coinaddr(char *coinaddr,cJSON *scriptobj)
|
||
|
{
|
||
|
struct destbuf buf; cJSON *addresses;
|
||
|
coinaddr[0] = 0;
|
||
|
if ( scriptobj == 0 )
|
||
|
return(-1);
|
||
|
if ( (addresses= cJSON_GetObjectItem(scriptobj,"addresses")) != 0 )
|
||
|
{
|
||
|
copy_cJSON(&buf,jitem(addresses,0));
|
||
|
strcpy(coinaddr,buf.buf);
|
||
|
return(0);
|
||
|
}
|
||
|
return(-1);
|
||
|
}
|
||
|
|
||
|
char *pangea_signp2sh(int32_t oldtx_format,struct cointx_info *refT,int32_t redeemi,char *redeemscript,char sigs[][256],int32_t n,uint8_t privkey[32],int32_t privkeyind)
|
||
|
{
|
||
|
char hexstr[16384]; bits256 hash2; uint8_t data[4096],sigbuf[512]; struct bp_key key;
|
||
|
struct cointx_info *T; int32_t i,len; void *sig = NULL; size_t siglen = 0; struct cointx_input *vin;
|
||
|
if ( bp_key_init(&key) != 0 && bp_key_secret_set(&key,privkey,32) != 0 )
|
||
|
{
|
||
|
if ( (T= calloc(1,sizeof(*T))) == 0 )
|
||
|
return(0);
|
||
|
*T = *refT; vin = &T->inputs[redeemi];
|
||
|
for (i=0; i<T->numinputs; i++)
|
||
|
strcpy(T->inputs[i].sigs,"00");
|
||
|
strcpy(vin->sigs,redeemscript);
|
||
|
vin->sequence = (uint32_t)-1;
|
||
|
T->nlocktime = 0;
|
||
|
//disp_cointx(&T);
|
||
|
emit_cointx(&hash2,data,sizeof(data),T,oldtx_format,SIGHASH_ALL);
|
||
|
//printf("HASH2.(%llx)\n",(long long)hash2.txid);
|
||
|
if ( bp_sign(&key,hash2.bytes,sizeof(hash2),&sig,&siglen) != 0 )
|
||
|
{
|
||
|
memcpy(sigbuf,sig,siglen);
|
||
|
sigbuf[siglen++] = SIGHASH_ALL;
|
||
|
init_hexbytes_noT(sigs[privkeyind],sigbuf,(int32_t)siglen);
|
||
|
strcpy(vin->sigs,"00");
|
||
|
for (i=0; i<n; i++)
|
||
|
{
|
||
|
if ( sigs[i][0] != 0 )
|
||
|
{
|
||
|
sprintf(vin->sigs + strlen(vin->sigs),"%02x%s",(int32_t)strlen(sigs[i])>>1,sigs[i]);
|
||
|
//printf("(%s).%ld ",sigs[i],strlen(sigs[i]));
|
||
|
}
|
||
|
}
|
||
|
len = (int32_t)(strlen(redeemscript)/2);
|
||
|
if ( len >= 0xfd )
|
||
|
sprintf(&vin->sigs[strlen(vin->sigs)],"4d%02x%02x",len & 0xff,(len >> 8) & 0xff);
|
||
|
else sprintf(&vin->sigs[strlen(vin->sigs)],"4c%02x",len);
|
||
|
sprintf(&vin->sigs[strlen(vin->sigs)],"%s",redeemscript);
|
||
|
//printf("after A.(%s) othersig.(%s) siglen.%02lx -> (%s)\n",hexstr,othersig != 0 ? othersig : "",siglen,vin->sigs);
|
||
|
//printf("vinsigs.(%s) %ld\n",vin->sigs,strlen(vin->sigs));
|
||
|
_emit_cointx(hexstr,sizeof(hexstr),T,oldtx_format);
|
||
|
//disp_cointx(&T);
|
||
|
free(T);
|
||
|
return(clonestr(hexstr));
|
||
|
}
|
||
|
else printf("error signing\n");
|
||
|
free(T);
|
||
|
}
|
||
|
return(0);
|
||
|
}
|
||
|
|
||
|
uint64_t jumblr_getcoinaddr(char *coinaddr,struct destbuf *scriptPubKey,struct coin777 *coin,char *txid,int32_t vout)
|
||
|
{
|
||
|
char *rawtransaction,*txidstr,*asmstr; uint64_t value = 0; int32_t n,m,len,reqSigs; cJSON *json,*scriptobj,*array,*item,*hexobj;
|
||
|
scriptPubKey->buf[0] = 0;
|
||
|
if ( (rawtransaction= _get_transaction(coin->name,coin->serverport,coin->userpass,txid)) == 0 )
|
||
|
{
|
||
|
printf("jumblr_getprivkey: error getting (%s)\n",txid);
|
||
|
return(0);
|
||
|
}
|
||
|
if ( (json= cJSON_Parse(rawtransaction)) != 0 )
|
||
|
{
|
||
|
if ( (txidstr= jstr(json,"txid")) == 0 || strcmp(txidstr,txid) != 0 )
|
||
|
{
|
||
|
printf("jumblr_getcoinaddr no txid or mismatch\n");
|
||
|
free_json(json);
|
||
|
free(rawtransaction);
|
||
|
return(0);
|
||
|
}
|
||
|
if ( (array= jarray(&n,json,"vout")) != 0 && vout < n && (item= jitem(array,vout)) != 0 )
|
||
|
{
|
||
|
reqSigs = (int32_t)get_cJSON_int(item,"reqSigs");
|
||
|
value = conv_cJSON_float(item,"value");
|
||
|
scriptobj = cJSON_GetObjectItem(item,"scriptPubKey");
|
||
|
printf("ITEM.(%s)\n",jprint(item,0));
|
||
|
if ( scriptobj != 0 )
|
||
|
{
|
||
|
printf("script.(%s)\n",jprint(scriptobj,0));
|
||
|
script_coinaddr(coinaddr,scriptobj);
|
||
|
hexobj = cJSON_GetObjectItem(scriptobj,"hex");
|
||
|
if ( scriptPubKey != 0 && hexobj != 0 )
|
||
|
copy_cJSON(scriptPubKey,hexobj);
|
||
|
else
|
||
|
{
|
||
|
// OP_DUP OP_HASH160 f563e867027dedd109c9bb5f3354c3cc41dc7c7f OP_EQUALVERIFY OP_CHECKSIG
|
||
|
// 0318d4f6cdcbe6c822b979fc318dbe4ad58287223c8fb57b7bec0c88cd58a4b16a OP_CHECKSIG
|
||
|
if ( (asmstr= jstr(scriptobj,"asm")) != 0 )
|
||
|
{
|
||
|
len = (int32_t)strlen(asmstr);
|
||
|
m = (int32_t)strlen(" OP_EQUALVERIFY OP_CHECKSIG");
|
||
|
if ( strncmp(asmstr,"OP_DUP OP_HASH160 ",strlen("OP_DUP OP_HASH160 ")) == 0 && strcmp(&asmstr[len - m]," OP_EQUALVERIFY OP_CHECKSIG") == 0 )
|
||
|
set_spendscript(scriptPubKey->buf,coinaddr);
|
||
|
else
|
||
|
{
|
||
|
printf("nonstandard.(%s)\n",&asmstr[len - m]);
|
||
|
m = (int32_t)strlen(" OP_CHECKSIG");
|
||
|
if ( strcmp(&asmstr[len - m]," OP_CHECKSIG") == 0 )
|
||
|
{
|
||
|
printf("key sig (%s)\n",asmstr);
|
||
|
sprintf(scriptPubKey->buf,"%02x",(len-m)/2);
|
||
|
memcpy(&scriptPubKey->buf[2],asmstr,(len - m));
|
||
|
scriptPubKey->buf[2 + (len - m)] = 0;
|
||
|
strcat(scriptPubKey->buf,"ac");
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
} else printf("null scriptobj.%p (%s)\n",scriptobj,coinaddr);
|
||
|
}
|
||
|
free_json(json);
|
||
|
}
|
||
|
free(rawtransaction);
|
||
|
return(value);
|
||
|
}
|
||
|
|
||
|
char *jumblr_getprivkey(uint64_t *valuep,struct destbuf *scriptPubKey,uint32_t *locktimep,struct coin777 *coin,char *txid,int32_t vout)
|
||
|
{
|
||
|
char *rawtransaction,*txidstr,*privkey=0,coinaddr[64]; uint64_t value = 0; int32_t n,reqSigs; cJSON *json,*scriptobj,*array,*item,*hexobj;
|
||
|
*locktimep = -1;
|
||
|
scriptPubKey->buf[0] = 0;
|
||
|
if ( (rawtransaction= _get_transaction(coin->name,coin->serverport,coin->userpass,txid)) == 0 )
|
||
|
{
|
||
|
printf("jumblr_getprivkey: error getting (%s)\n",txid);
|
||
|
return(0);
|
||
|
}
|
||
|
if ( (json= cJSON_Parse(rawtransaction)) != 0 )//get_decoderaw_json(coin,rawtransaction)) != 0 )
|
||
|
{
|
||
|
*locktimep = (int32_t)get_cJSON_int(json,"locktime");
|
||
|
if ( (txidstr= jstr(json,"txid")) == 0 || strcmp(txidstr,txid) != 0 )
|
||
|
{
|
||
|
printf("jumblr_getprivkey no txid or mismatch\n");
|
||
|
free_json(json);
|
||
|
free(rawtransaction);
|
||
|
return(0);
|
||
|
}
|
||
|
//printf("txidstr.(%s) vout.%d\n",txidstr,vout);
|
||
|
if ( (array= jarray(&n,json,"vout")) != 0 && (item= jitem(array,vout)) != 0 )
|
||
|
{
|
||
|
scriptobj = cJSON_GetObjectItem(item,"scriptPubKey");
|
||
|
if ( scriptobj != 0 && script_coinaddr(coinaddr,scriptobj) == 0 )
|
||
|
{
|
||
|
reqSigs = (int32_t)get_cJSON_int(item,"reqSigs");
|
||
|
value = conv_cJSON_float(item,"value");
|
||
|
hexobj = cJSON_GetObjectItem(scriptobj,"hex");
|
||
|
if ( scriptPubKey != 0 && hexobj != 0 )
|
||
|
copy_cJSON(scriptPubKey,hexobj);
|
||
|
privkey = dumpprivkey(coin->name,coin->serverport,coin->userpass,coinaddr);
|
||
|
} else printf("null scriptobj.%p (%s)\n",scriptobj,coinaddr);
|
||
|
}
|
||
|
free_json(json);
|
||
|
}
|
||
|
free(rawtransaction);
|
||
|
if ( valuep != 0 )
|
||
|
*valuep = value;
|
||
|
return(privkey);
|
||
|
}
|
||
|
|
||
|
cJSON *cointx_vins_json_params(struct coin777 *coin,char *rawbytes)
|
||
|
{
|
||
|
int32_t i; cJSON *json,*array; char coinaddr[128]; struct destbuf scriptPubKey; struct cointx_info *cointx;
|
||
|
array = cJSON_CreateArray();
|
||
|
printf("convert.(%s)\n",rawbytes);
|
||
|
if ( (cointx= _decode_rawtransaction(rawbytes,coin->mgw.oldtx_format)) != 0 )
|
||
|
{
|
||
|
disp_cointx(cointx);
|
||
|
for (i=0; i<cointx->numinputs; i++)
|
||
|
{
|
||
|
json = cJSON_CreateObject();
|
||
|
jaddstr(json,"txid",cointx->inputs[i].tx.txidstr);
|
||
|
jaddnum(json,"vout",cointx->inputs[i].tx.vout);
|
||
|
if ( cointx->inputs[i].sigs[0] != 0 )
|
||
|
jaddstr(json,"scriptPubKey",cointx->inputs[i].sigs);
|
||
|
else
|
||
|
{
|
||
|
jumblr_getcoinaddr(coinaddr,&scriptPubKey,coin,cointx->inputs[i].tx.txidstr,cointx->inputs[i].tx.vout);
|
||
|
jaddstr(json,"scriptPubKey",scriptPubKey.buf);
|
||
|
}
|
||
|
cJSON_AddItemToArray(array,json);
|
||
|
}
|
||
|
free(cointx);
|
||
|
}
|
||
|
return(array);
|
||
|
}
|
||
|
|
||
|
char *jumblr_signraw_json_params(struct coin777 *coin,char *rawbytes)
|
||
|
{
|
||
|
char *paramstr = 0; cJSON *array,*rawobj,*vinsobj;//,*keysobj;char *coinaddrs[MAX_SUBATOMIC_INPUTS+1],
|
||
|
if ( (rawobj= cJSON_CreateString(rawbytes)) != 0 )
|
||
|
{
|
||
|
if ( (vinsobj= cointx_vins_json_params(coin,rawbytes)) != 0 )
|
||
|
{
|
||
|
array = cJSON_CreateArray();
|
||
|
jaddi(array,rawobj);
|
||
|
jaddi(array,vinsobj);
|
||
|
//cJSON_AddItemToArray(array,keysobj);
|
||
|
paramstr = jprint(array,1);
|
||
|
}
|
||
|
else free_json(rawobj);
|
||
|
}
|
||
|
return(paramstr);
|
||
|
}
|
||
|
|
||
|
int32_t jumblr_signtx(char *signedtx,unsigned long destsize,struct coin777 *coin,char *signparams)
|
||
|
{
|
||
|
cJSON *json,*compobj; char *retstr,*deststr; uint32_t completed = 0;
|
||
|
signedtx[0] = 0;
|
||
|
//printf("cp.%d vs %d: subatomic_signtx rawbytes.(%s)\n",cp->coinid,coinid,rawbytes);
|
||
|
if ( coin != 0 && signparams != 0 )
|
||
|
{
|
||
|
_stripwhite(signparams,' ');
|
||
|
printf("got signparams.(%s)\n",signparams);
|
||
|
if ( (retstr= bitcoind_RPC(0,coin->name,coin->serverport,coin->userpass,"signrawtransaction",signparams)) != 0 )
|
||
|
{
|
||
|
//printf("got retstr.(%s)\n",retstr);
|
||
|
if ( (json= cJSON_Parse(retstr)) != 0 )
|
||
|
{
|
||
|
if ( (deststr= jstr(json,"hex")) != 0 )
|
||
|
{
|
||
|
compobj = cJSON_GetObjectItem(json,"complete");
|
||
|
if ( compobj != 0 )
|
||
|
completed = ((compobj->type&0xff) == cJSON_True);
|
||
|
if ( strlen(deststr) > destsize )
|
||
|
printf("sign_rawtransaction: strlen(deststr) %ld > %ld destize\n",(long)strlen(deststr),destsize);
|
||
|
else strcpy(signedtx,deststr);
|
||
|
} else printf("cant get hex from.(%s)\n",retstr);
|
||
|
free_json(json);
|
||
|
} else printf("json parse error.(%s)\n",retstr);
|
||
|
free(retstr);
|
||
|
} else printf("error signing rawtx\n");
|
||
|
} else printf("error generating signparams\n");
|
||
|
return(completed);
|
||
|
}
|
||
|
|
||
|
char *jumblr_signvin(char *sigstr,struct coin777 *coin,char *signedtx,int32_t bufsize,void *bpkey,char *pubP,struct cointx_info *refT,int32_t redeemi,char *rawtx)
|
||
|
{
|
||
|
// signrawtransaction <hex string> [{"txid":txid,"vout":n,"scriptPubKey":hex},...] [<privatekey1>,...]
|
||
|
char hexstr[4096],redeem[2048]; bits256 hash2; uint8_t *data,sigbuf[1024];
|
||
|
struct cointx_info *T; int32_t i; void *sig = NULL; size_t siglen = 0; struct cointx_input *vin;
|
||
|
sigstr[0] = 0;
|
||
|
if ( 1 )
|
||
|
{
|
||
|
char *paramstr; cJSON *vinarray,*item,*array = cJSON_CreateArray();
|
||
|
vinarray = cJSON_CreateArray();
|
||
|
jaddistr(array,rawtx);
|
||
|
for (i=0; i<refT->numinputs; i++)
|
||
|
{
|
||
|
vin = &refT->inputs[i];
|
||
|
item = cJSON_CreateObject();
|
||
|
jaddstr(item,"txid",vin->tx.txidstr);
|
||
|
jaddnum(item,"vout",vin->tx.vout);
|
||
|
jaddstr(item,"scriptPubKey",vin->sigs);
|
||
|
jaddi(vinarray,item);
|
||
|
}
|
||
|
jaddi(array,vinarray);
|
||
|
paramstr = jprint(array,1);
|
||
|
if ( jumblr_signtx(signedtx,bufsize,coin,paramstr) > 0 )
|
||
|
printf("SIGS completed\n");
|
||
|
if ( signedtx[0] != 0 )
|
||
|
{
|
||
|
if ( (T= _decode_rawtransaction(signedtx,coin->mgw.oldtx_format)) != 0 )
|
||
|
{
|
||
|
strcpy(sigstr,T->inputs[redeemi].sigs);
|
||
|
free(T);
|
||
|
return(sigstr);
|
||
|
}
|
||
|
}
|
||
|
return(0);
|
||
|
}
|
||
|
if ( (T = calloc(1,sizeof(*T))) == 0 )
|
||
|
{
|
||
|
printf("unexpected out of mem in jumblr_signvin\n");
|
||
|
return(0);
|
||
|
}
|
||
|
|
||
|
*T = *refT;
|
||
|
vin = &T->inputs[redeemi];
|
||
|
safecopy(redeem,vin->sigs,sizeof(redeem));
|
||
|
fprintf(stderr,"redeemi.%d numinputs.%d\n",redeemi,T->numinputs);
|
||
|
for (i=0; i<T->numinputs; i++)
|
||
|
if ( i != redeemi )
|
||
|
strcpy(T->inputs[i].sigs,"00");
|
||
|
vin->sequence = (uint32_t)-1;
|
||
|
T->nlocktime = 0;
|
||
|
data = malloc(65536);
|
||
|
disp_cointx(T);
|
||
|
emit_cointx(&hash2,data,sizeof(data),T,coin->mgw.oldtx_format,SIGHASH_ALL);
|
||
|
free(data);
|
||
|
if ( bp_sign(bpkey,hash2.bytes,sizeof(hash2),&sig,&siglen) != 0 && sig != 0 )
|
||
|
{
|
||
|
memcpy(sigbuf,sig,siglen);
|
||
|
free(sig);
|
||
|
sigbuf[siglen++] = SIGHASH_ALL;
|
||
|
init_hexbytes_noT(hexstr,sigbuf,(int32_t)siglen);
|
||
|
sprintf(vin->sigs,"%02x%s%02x%s",(uint32_t)siglen,hexstr,(uint32_t)strlen(pubP)/2,pubP);
|
||
|
strcpy(sigstr,vin->sigs);
|
||
|
printf("after P.(%s) siglen.%02x -> %s pubP.(%s)\n",sigstr,(uint32_t)siglen,vin->sigs,pubP);
|
||
|
}
|
||
|
free(T);
|
||
|
if ( sigstr[0] != 0 )
|
||
|
return(sigstr);
|
||
|
else return(0);
|
||
|
}
|
||
|
|
||
|
int32_t script_has_coinaddr(cJSON *scriptobj,char *coinaddr)
|
||
|
{
|
||
|
int32_t i,n; struct destbuf buf; cJSON *addresses,*addrobj;
|
||
|
if ( scriptobj == 0 )
|
||
|
return(0);
|
||
|
addresses = cJSON_GetObjectItem(scriptobj,"addresses");
|
||
|
if ( addresses != 0 )
|
||
|
{
|
||
|
n = cJSON_GetArraySize(addresses);
|
||
|
for (i=0; i<n; i++)
|
||
|
{
|
||
|
addrobj = cJSON_GetArrayItem(addresses,i);
|
||
|
copy_cJSON(&buf,addrobj);
|
||
|
if ( strcmp(buf.buf,coinaddr) == 0 )
|
||
|
return(1);
|
||
|
}
|
||
|
}
|
||
|
return(0);
|
||
|
}
|
||
|
|
||
|
cJSON *get_decoderaw_json(struct coin777 *coin,char *rawtransaction)
|
||
|
{
|
||
|
char *str,*retstr; cJSON *json = 0;
|
||
|
str = malloc(strlen(rawtransaction)+4);
|
||
|
//printf("got rawtransaction.(%s)\n",rawtransaction);
|
||
|
sprintf(str,"\"%s\"",rawtransaction);
|
||
|
if ( (retstr= bitcoind_passthru(coin->name,coin->serverport,coin->userpass,"decoderawtransaction",str)) != 0 && retstr[0] != 0 )
|
||
|
{
|
||
|
//printf("got decodetransaction.(%s)\n",retstr);
|
||
|
json = cJSON_Parse(retstr);
|
||
|
} else printf("error decoding.(%s)\n",str);
|
||
|
if ( retstr != 0 )
|
||
|
free(retstr);
|
||
|
free(str);
|
||
|
return(json);
|
||
|
}
|
||
|
|
||
|
char *subatomic_decodetxid(int64_t *valuep,struct destbuf *scriptPubKey,uint32_t *locktimep,struct coin777 *coin,char *rawtransaction,char *mycoinaddr)
|
||
|
{
|
||
|
char *txidstr,checkasmstr[1024],*asmstr,*txid = 0; uint64_t value = 0; int32_t i,n,nval,reqSigs; cJSON *json,*scriptobj,*array,*item,*hexobj;
|
||
|
*locktimep = -1;
|
||
|
if ( (json= get_decoderaw_json(coin,rawtransaction)) != 0 )
|
||
|
{
|
||
|
*locktimep = (int32_t)get_cJSON_int(json,"locktime");
|
||
|
if ( (txidstr= jstr(json,"txid")) == 0 )
|
||
|
{
|
||
|
printf("subatomic_decodetxid no txid\n");
|
||
|
return(0);
|
||
|
}
|
||
|
txid = clonestr(txidstr);
|
||
|
array = cJSON_GetObjectItem(json,"vout");
|
||
|
if ( mycoinaddr != 0 && is_cJSON_Array(array) != 0 )
|
||
|
{
|
||
|
n = cJSON_GetArraySize(array);
|
||
|
for (i=0; i<n; i++)
|
||
|
{
|
||
|
item = cJSON_GetArrayItem(array,i);
|
||
|
hexobj = 0;
|
||
|
scriptobj = cJSON_GetObjectItem(item,"scriptPubKey");
|
||
|
if ( mycoinaddr != 0 && scriptobj != 0 && script_has_coinaddr(scriptobj,mycoinaddr) != 0 )
|
||
|
{
|
||
|
nval = (int32_t)get_cJSON_int(item,"n");
|
||
|
if ( nval == i )
|
||
|
{
|
||
|
reqSigs = (int32_t)get_cJSON_int(item,"reqSigs");
|
||
|
value = conv_cJSON_float(item,"value");
|
||
|
hexobj = cJSON_GetObjectItem(scriptobj,"hex");
|
||
|
if ( scriptPubKey != 0 && hexobj != 0 )
|
||
|
copy_cJSON(scriptPubKey,hexobj);
|
||
|
if ( reqSigs == 1 && hexobj != 0 )
|
||
|
{
|
||
|
sprintf(checkasmstr,"OP_DUP OP_HASH160 %s OP_EQUALVERIFY OP_CHECKSIG","need to figure out how ot gen magic number");
|
||
|
if ( (asmstr= jstr(scriptobj,"asm")) != 0 && strcmp(asmstr,checkasmstr) != 0 )
|
||
|
printf("warning: (%s) != check.(%s)\n",asmstr,checkasmstr);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
if ( valuep != 0 )
|
||
|
*valuep = value;
|
||
|
return(txid);
|
||
|
}
|
||
|
|
||
|
cJSON *subatomic_vins_json_params(struct coin777 *coin,struct subatomic_rawtransaction *rp)
|
||
|
{
|
||
|
int32_t i; cJSON *json,*array; struct subatomic_unspent_tx *up;
|
||
|
array = cJSON_CreateArray();
|
||
|
for (i=0; i<rp->numinputs; i++)
|
||
|
{
|
||
|
up = &rp->inputs[i];
|
||
|
json = cJSON_CreateObject();
|
||
|
jaddstr(json,"txid",up->txid.buf);
|
||
|
jaddnum(json,"vout",up->vout);
|
||
|
if ( up->scriptPubKey.buf[0] != 0 )
|
||
|
jaddstr(json,"scriptPubKey",up->scriptPubKey.buf);
|
||
|
if ( up->redeemScript.buf[0] != 0 )
|
||
|
jaddstr(json,"redeemScript",up->redeemScript.buf);
|
||
|
cJSON_AddItemToArray(array,json);
|
||
|
}
|
||
|
return(array);
|
||
|
}
|
||
|
|
||
|
cJSON *subatomic_privkeys_json_params(struct coin777 *coin,char **coinaddrs,int32_t n)
|
||
|
{
|
||
|
int32_t i; char *privkey; cJSON *array = cJSON_CreateArray();
|
||
|
//sprintf(walletkey,"[\"%s\",%d]",Global_subatomic->NXTADDR,BITCOIN_WALLET_UNLOCKSECONDS);
|
||
|
// locking first avoids error, hacky but no time for wallet fiddling now
|
||
|
//bitcoind_RPC(0,coin->name,coin->serverport,coin->userpass,"walletlock",0);
|
||
|
//bitcoind_RPC(0,coin->name,coin->serverport,coin->userpass,"walletpassphrase",walletkey);
|
||
|
for (i=0; i<n; i++)
|
||
|
{
|
||
|
if ( coinaddrs[i][0] != 0 )
|
||
|
{
|
||
|
printf("privkeys.(%s)\n",coinaddrs[i]);
|
||
|
if ( (privkey= dumpprivkey(coin->name,coin->serverport,coin->userpass,coinaddrs[i])) != 0 )
|
||
|
{
|
||
|
jaddistr(array,privkey);
|
||
|
free(privkey);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
return(array);
|
||
|
}
|
||
|
|
||
|
char *subatomic_signraw_json_params(char *skipaddr,char *coinaddr,struct coin777 *coin,struct subatomic_rawtransaction *rp,char *rawbytes)
|
||
|
{
|
||
|
int32_t i,j,flag; char *coinaddrs[MAX_SUBATOMIC_INPUTS+1],*paramstr = 0; cJSON *array,*rawobj,*vinsobj,*keysobj;
|
||
|
if ( (rawobj= cJSON_CreateString(rawbytes)) != 0 )
|
||
|
{
|
||
|
if ( (vinsobj= subatomic_vins_json_params(coin,rp)) != 0 )
|
||
|
{
|
||
|
// printf("add %d inputs skipaddr.%s coinaddr.%s\n",rp->numinputs,skipaddr,coinaddr);
|
||
|
for (i=flag=j=0; i<rp->numinputs; i++)
|
||
|
{
|
||
|
if ( skipaddr == 0 || strcmp(rp->inputs[i].address.buf,skipaddr) != 0 )
|
||
|
{
|
||
|
printf("i.%d j.%d flag.%d %s\n",i,j,flag,rp->inputs[i].address.buf);
|
||
|
coinaddrs[j] = rp->inputs[i].address.buf;
|
||
|
if ( coinaddr != 0 && strcmp(coinaddrs[j],coinaddr) == 0 )
|
||
|
flag++;
|
||
|
j++;
|
||
|
}
|
||
|
}
|
||
|
//printf("i.%d j.%d flag.%d\n",i,j,flag);
|
||
|
//if ( coinaddr != 0 && flag == 0 )
|
||
|
//coinaddrs[j++] = coinaddr;
|
||
|
coinaddrs[j] = 0;
|
||
|
keysobj = subatomic_privkeys_json_params(coin,coinaddrs,j);
|
||
|
if ( keysobj != 0 )
|
||
|
{
|
||
|
array = cJSON_CreateArray();
|
||
|
cJSON_AddItemToArray(array,rawobj);
|
||
|
cJSON_AddItemToArray(array,vinsobj);
|
||
|
cJSON_AddItemToArray(array,keysobj);
|
||
|
paramstr = cJSON_Print(array);
|
||
|
free_json(array);
|
||
|
}
|
||
|
else free_json(vinsobj);
|
||
|
}
|
||
|
else free_json(rawobj);
|
||
|
}
|
||
|
return(paramstr);
|
||
|
}
|
||
|
|
||
|
char *subatomic_signtx(char *skipaddr,uint32_t *lockedblockp,int64_t *valuep,char *coinaddr,char *signedtx,unsigned long destsize,struct coin777 *coin,struct subatomic_rawtransaction *rp,char *rawbytes)
|
||
|
{
|
||
|
cJSON *json,*compobj; char *retstr,*deststr,*signparams,*txid = 0; uint32_t locktime = 0;
|
||
|
rp->txid[0] = signedtx[0] = 0;
|
||
|
rp->completed = -1;
|
||
|
//printf("cp.%d vs %d: subatomic_signtx rawbytes.(%s)\n",cp->coinid,coinid,rawbytes);
|
||
|
if ( coin != 0 && (signparams= subatomic_signraw_json_params(skipaddr,coinaddr,coin,rp,rawbytes)) != 0 )
|
||
|
{
|
||
|
_stripwhite(signparams,' ');
|
||
|
//printf("got signparams.(%s)\n",signparams);
|
||
|
if ( (retstr= bitcoind_RPC(0,coin->name,coin->serverport,coin->userpass,"signrawtransaction",signparams)) != 0 )
|
||
|
{
|
||
|
//printf("got retstr.(%s)\n",retstr);
|
||
|
if ( (json= cJSON_Parse(retstr)) != 0 )
|
||
|
{
|
||
|
if ( (deststr= jstr(json,"hex")) != 0 )
|
||
|
{
|
||
|
compobj = cJSON_GetObjectItem(json,"complete");
|
||
|
if ( compobj != 0 )
|
||
|
rp->completed = ((compobj->type&0xff) == cJSON_True);
|
||
|
if ( strlen(deststr) > destsize )
|
||
|
printf("sign_rawtransaction: strlen(deststr) %ld > %ld destize\n",(long)strlen(deststr),destsize);
|
||
|
else
|
||
|
{
|
||
|
strcpy(signedtx,deststr);
|
||
|
txid = subatomic_decodetxid(valuep,0,&locktime,coin,deststr,coinaddr);
|
||
|
if ( txid != 0 )
|
||
|
{
|
||
|
safecopy(rp->txid,txid,sizeof(rp->txid));
|
||
|
free(txid);
|
||
|
txid = rp->txid;
|
||
|
}
|
||
|
// printf("got signedtransaction -> txid.(%s) %.8f\n",rp->txid,dstr(valuep!=0?*valuep:0));
|
||
|
}
|
||
|
} else printf("cant get hex from.(%s)\n",retstr);
|
||
|
free_json(json);
|
||
|
} else printf("json parse error.(%s)\n",retstr);
|
||
|
free(retstr);
|
||
|
} else printf("error signing rawtx\n");
|
||
|
free(signparams);
|
||
|
} else printf("error generating signparams\n");
|
||
|
if ( lockedblockp != 0 )
|
||
|
*lockedblockp = locktime;
|
||
|
return(txid);
|
||
|
}
|
||
|
|
||
|
cJSON *subatomic_vouts_json_params(struct subatomic_rawtransaction *rp)
|
||
|
{
|
||
|
int32_t i; cJSON *json,*obj;
|
||
|
json = cJSON_CreateObject();
|
||
|
for (i=0; i<rp->numoutputs; i++)
|
||
|
{
|
||
|
obj = cJSON_CreateNumber((double)rp->destamounts[i]/SATOSHIDEN);
|
||
|
cJSON_AddItemToObject(json,rp->destaddrs[i],obj);
|
||
|
}
|
||
|
// printf("numdests.%d (%s)\n",rp->numoutputs,cJSON_Print(json));
|
||
|
return(json);
|
||
|
}
|
||
|
|
||
|
char *subatomic_rawtxid_json(struct coin777 *coin,struct subatomic_rawtransaction *rp)
|
||
|
{
|
||
|
char *paramstr = 0; cJSON *array,*vinsobj,*voutsobj;
|
||
|
if ( (vinsobj= subatomic_vins_json_params(coin,rp)) != 0 )
|
||
|
{
|
||
|
if ( (voutsobj= subatomic_vouts_json_params(rp)) != 0 )
|
||
|
{
|
||
|
array = cJSON_CreateArray();
|
||
|
cJSON_AddItemToArray(array,vinsobj);
|
||
|
cJSON_AddItemToArray(array,voutsobj);
|
||
|
paramstr = cJSON_Print(array);
|
||
|
free_json(array); // this frees both vinsobj and voutsobj
|
||
|
}
|
||
|
else free_json(vinsobj);
|
||
|
}
|
||
|
// printf("subatomic_rawtxid_json.%s\n",paramstr);
|
||
|
return(paramstr);
|
||
|
}
|
||
|
|
||
|
uint64_t subatomic_donation(struct coin777 *coin,uint64_t amount)
|
||
|
{
|
||
|
uint64_t donation = 0;
|
||
|
if ( coin->donationaddress[0] != 0 )
|
||
|
{
|
||
|
donation = amount >> 11;
|
||
|
if ( donation < coin->mgw.txfee )
|
||
|
donation = coin->mgw.txfee;
|
||
|
}
|
||
|
return(donation);
|
||
|
}
|
||
|
|
||
|
char *gather_account_addresses(struct coin777 *coin,char *account)
|
||
|
{
|
||
|
cJSON *array,*retarray,*subarray,*item; int32_t i,j,m,n; char *acct;
|
||
|
//printf("call listaddressgroupings\n");
|
||
|
if ( (array= _get_localaddresses(coin->name,coin->serverport,coin->userpass)) != 0 )
|
||
|
{
|
||
|
retarray = cJSON_CreateArray();
|
||
|
n = cJSON_GetArraySize(array);
|
||
|
for (i=0; i<n; i++)
|
||
|
{
|
||
|
if ( (subarray= jitem(array,i)) != 0 )
|
||
|
{
|
||
|
//printf("%d of %d: %s\n",i,n,jprint(subarray,0));
|
||
|
if ( is_cJSON_Array(subarray) != 0 && (m= cJSON_GetArraySize(subarray)) > 0 )
|
||
|
{
|
||
|
for (j=0; j<m; j++)
|
||
|
{
|
||
|
if ( (item= jitem(subarray,j)) != 0 && is_cJSON_Array(item) != 0 && cJSON_GetArraySize(item) > 2 )
|
||
|
{
|
||
|
if ( (acct= jstr(jitem(item,2),0)) != 0 && strcmp(acct,account) == 0 )
|
||
|
{
|
||
|
//printf("gather.(%s) %s\n",jstr(jitem(item,0),0),account);
|
||
|
jaddistr(retarray,jstr(jitem(item,0),0));
|
||
|
}
|
||
|
} //else printf("skip item.%p, %d %d\n",item,is_cJSON_Array(item),cJSON_GetArraySize(item));
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
free_json(array);
|
||
|
if ( cJSON_GetArraySize(retarray) == 0 )
|
||
|
{
|
||
|
free_json(retarray);
|
||
|
return(0);
|
||
|
}
|
||
|
else return(jprint(retarray,1));
|
||
|
}
|
||
|
else return(0);
|
||
|
}
|
||
|
|
||
|
struct subatomic_unspent_tx *gather_unspents(uint64_t *totalp,int32_t *nump,struct coin777 *coin,char *account)
|
||
|
{
|
||
|
int32_t i,j,num; struct subatomic_unspent_tx *ups = 0; char *params,*addrs,*retstr; cJSON *json,*item;
|
||
|
/*{
|
||
|
"txid" : "1ccd2a9d0f8d690ed13b6768fc6c041972362f5531922b6b152ed2c98d3fe113",
|
||
|
"vout" : 1,
|
||
|
"address" : "DK3nxu6GshBcQNDMqc66ARcwqDZ1B5TJe5",
|
||
|
"scriptPubKey" : "76a9149891029995222077889b36c77e2b85690878df9088ac",
|
||
|
"amount" : 2.00000000,
|
||
|
"confirmations" : 72505
|
||
|
},*/
|
||
|
*totalp = *nump = 0;
|
||
|
if ( account != 0 && account[0] != 0 )
|
||
|
{
|
||
|
if ( (addrs= gather_account_addresses(coin,account)) != 0 )
|
||
|
{
|
||
|
if ( (params = calloc(1,strlen(addrs) + 128)) == 0 )
|
||
|
{
|
||
|
free(addrs);
|
||
|
return(0);
|
||
|
}
|
||
|
addrs[strlen(addrs)-1] = 0;
|
||
|
sprintf(params,"[%d, 99999999, [%s]]",coin->minconfirms,addrs+1);
|
||
|
free(addrs);
|
||
|
} else return(0);
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
if ( (params = calloc(1,128)) == 0 )
|
||
|
return(0);
|
||
|
sprintf(params,"%d, 99999999",coin->minconfirms);
|
||
|
}
|
||
|
//printf("issue listunspent.(%s)\n",params);
|
||
|
if ( (retstr= bitcoind_passthru(coin->name,coin->serverport,coin->userpass,"listunspent",params)) != 0 )
|
||
|
{
|
||
|
//printf("unspents (%s)\n",retstr);
|
||
|
if ( (json= cJSON_Parse(retstr)) != 0 )
|
||
|
{
|
||
|
if ( is_cJSON_Array(json) != 0 && (num= cJSON_GetArraySize(json)) > 0 )
|
||
|
{
|
||
|
ups = calloc(num,sizeof(struct subatomic_unspent_tx));
|
||
|
for (i=j=0; i<num; i++)
|
||
|
{
|
||
|
item = cJSON_GetArrayItem(json,i);
|
||
|
copy_cJSON(&ups[j].address,cJSON_GetObjectItem(item,"address"));
|
||
|
//if ( skipcoinaddr == 0 || strcmp(skipcoinaddr,ups[j].address.buf) != 0 )
|
||
|
{
|
||
|
copy_cJSON(&ups[j].txid,cJSON_GetObjectItem(item,"txid"));
|
||
|
copy_cJSON(&ups[j].scriptPubKey,cJSON_GetObjectItem(item,"scriptPubKey"));
|
||
|
ups[j].vout = (int32_t)get_cJSON_int(item,"vout");
|
||
|
ups[j].amount = conv_cJSON_float(item,"amount");
|
||
|
ups[j].confirmations = (int32_t)get_cJSON_int(item,"confirmations");
|
||
|
*totalp += ups[j].amount;
|
||
|
j++;
|
||
|
}
|
||
|
}
|
||
|
*nump = j;
|
||
|
if ( j > 0 )
|
||
|
{
|
||
|
int _decreasing_signedint64(const void *a,const void *b);
|
||
|
if ( j > 1 )
|
||
|
qsort(ups,j,sizeof(*ups),_decreasing_signedint64);
|
||
|
if ( coin->changeaddr[0] == 0 )
|
||
|
strcpy(coin->changeaddr,ups[0].address.buf);
|
||
|
//for (i=0; i<j; i++)
|
||
|
//printf("%s/v%-3d %13.6f %s confs.%-6d | total %.6f\n",ups[i].txid.buf,ups[i].vout,dstr(ups[i].amount),ups[i].address.buf,ups[i].confirmations,dstr(*totalp));
|
||
|
}
|
||
|
}
|
||
|
free_json(json);
|
||
|
}
|
||
|
free(retstr);
|
||
|
}
|
||
|
free(params);
|
||
|
if ( *nump == 0 )
|
||
|
printf("no (%s) unspents for (%s)\n",coin->name,account != 0 ? account : "");
|
||
|
return(ups);
|
||
|
}
|
||
|
|
||
|
struct subatomic_unspent_tx *subatomic_bestfit(struct coin777 *coin,struct subatomic_unspent_tx *unspents,int32_t numunspents,uint64_t value,int32_t mode)
|
||
|
{
|
||
|
int32_t i; uint64_t above,below,gap,atx_value; struct subatomic_unspent_tx *vin,*abovevin,*belowvin;
|
||
|
abovevin = belowvin = 0;
|
||
|
for (above=below=i=0; i<numunspents; i++)
|
||
|
{
|
||
|
vin = &unspents[i];
|
||
|
atx_value = vin->amount;
|
||
|
//printf("(%.8f vs %.8f)\n",dstr(atx_value),dstr(value));
|
||
|
if ( atx_value == value )
|
||
|
return(vin);
|
||
|
else if ( atx_value > value )
|
||
|
{
|
||
|
gap = (atx_value - value);
|
||
|
if ( above == 0 || gap < above )
|
||
|
{
|
||
|
above = gap;
|
||
|
abovevin = vin;
|
||
|
}
|
||
|
}
|
||
|
else if ( mode == 0 )
|
||
|
{
|
||
|
gap = (value - atx_value);
|
||
|
if ( below == 0 || gap < below )
|
||
|
{
|
||
|
below = gap;
|
||
|
belowvin = vin;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
if ( (vin= (abovevin != 0) ? abovevin : belowvin) == 0 && mode == 1 )
|
||
|
vin = unspents;
|
||
|
return(vin);
|
||
|
}
|
||
|
|
||
|
int64_t subatomic_calc_rawinputs(struct coin777 *coin,struct subatomic_rawtransaction *rp,uint64_t amount,struct subatomic_unspent_tx *ups,int32_t num,uint64_t donation)
|
||
|
{
|
||
|
uint64_t sum = 0; struct subatomic_unspent_tx *up; int32_t i;
|
||
|
rp->inputsum = rp->numinputs = 0;
|
||
|
printf("unspent num %d, amount %.8f vs donation %.8f txfee %.8f\n",num,dstr(amount),dstr(donation),dstr(coin->mgw.txfee));
|
||
|
if ( coin == 0 || num == 0 ) // (donation + coin->mgw.txfee) > amount ||
|
||
|
return(0);
|
||
|
amount += coin->mgw.txfee + donation;
|
||
|
for (i=0; i<num&&i<((int32_t)(sizeof(rp->inputs)/sizeof(*rp->inputs))); i++)
|
||
|
{
|
||
|
if ( (up= subatomic_bestfit(coin,ups,num,amount,0)) != 0 )
|
||
|
{
|
||
|
sum += up->amount;
|
||
|
rp->inputs[rp->numinputs++] = *up;
|
||
|
if ( sum >= amount )
|
||
|
{
|
||
|
rp->amount = (amount - coin->mgw.txfee - donation);
|
||
|
rp->change = (sum - amount);
|
||
|
rp->inputsum = sum;
|
||
|
printf("numinputs %d sum %.8f vs amount %.8f change %.8f -> txfee %.8f\n",rp->numinputs,dstr(rp->inputsum),dstr(amount),dstr(rp->change),dstr(sum - rp->change - rp->amount));
|
||
|
return(rp->inputsum);
|
||
|
}
|
||
|
}
|
||
|
printf("error getting bestfit unspent\n");
|
||
|
break;
|
||
|
}
|
||
|
printf("i.%d error numinputs %d sum %.8f\n",i,rp->numinputs,dstr(rp->inputsum));
|
||
|
return(0);
|
||
|
}
|
||
|
|
||
|
char *subatomic_gen_rawtransaction(char *skipaddr,struct coin777 *coin,struct subatomic_rawtransaction *rp,char *signcoinaddr,uint32_t locktime,uint32_t vin0sequenceid,char *redeem0script)
|
||
|
{
|
||
|
char *rawparams,*retstr,*txid=0; int64_t value; long len; struct cointx_info *cointx;
|
||
|
if ( (rawparams= subatomic_rawtxid_json(coin,rp)) != 0 )
|
||
|
{
|
||
|
_stripwhite(rawparams,' ');
|
||
|
//printf("create.(%s)\n",rawparams);
|
||
|
if ( (retstr= bitcoind_RPC(0,coin->name,coin->serverport,coin->userpass,"createrawtransaction",rawparams)) != 0 )
|
||
|
{
|
||
|
if ( retstr[0] != 0 )
|
||
|
{
|
||
|
// printf("calc_rawtransaction retstr.(%s)\n",retstr);
|
||
|
safecopy(rp->rawtransaction,retstr,sizeof(rp->rawtransaction));
|
||
|
len = strlen(rp->rawtransaction);
|
||
|
if ( len < 8 )
|
||
|
{
|
||
|
printf("funny rawtransactionlen %ld??\n",len);
|
||
|
free(rawparams);
|
||
|
return(0);
|
||
|
}
|
||
|
if ( locktime != 0 || redeem0script != 0 )
|
||
|
{
|
||
|
if ( (cointx= _decode_rawtransaction(rp->rawtransaction,coin->mgw.oldtx_format)) != 0 )
|
||
|
{
|
||
|
//printf("%s\n->\n",rp->rawtransaction);
|
||
|
cointx->nlocktime = locktime;
|
||
|
cointx->inputs[0].sequence = vin0sequenceid;
|
||
|
if ( redeem0script != 0 )
|
||
|
safecopy(cointx->outputs[0].script,redeem0script,sizeof(cointx->outputs[0].script));
|
||
|
_emit_cointx(rp->rawtransaction,sizeof(rp->rawtransaction),cointx,coin->mgw.oldtx_format);
|
||
|
_validate_decoderawtransaction(rp->rawtransaction,cointx,coin->mgw.oldtx_format);
|
||
|
//printf("spliced tx.(%s)\n",rp->rawtransaction);
|
||
|
free(cointx);
|
||
|
}
|
||
|
printf("locktime.%d sequenceid.%d signcoinaddr.(%s)\n",locktime,vin0sequenceid,signcoinaddr!=0?signcoinaddr:"");
|
||
|
}
|
||
|
if ( signcoinaddr != 0 )
|
||
|
{
|
||
|
txid = subatomic_signtx(skipaddr,0,&value,signcoinaddr,rp->signedtransaction,sizeof(rp->signedtransaction),coin,rp,rp->rawtransaction);
|
||
|
printf("signedtxid.%s\n",txid);
|
||
|
}
|
||
|
}
|
||
|
free(retstr);
|
||
|
} else printf("error creating rawtransaction from.(%s)\n",rawparams);
|
||
|
free(rawparams);
|
||
|
} else printf("error creating rawparams\n");
|
||
|
return(txid);
|
||
|
}
|
||
|
|
||
|
char *subatomic_signp2sh(char *sigstr,struct coin777 *coin,struct cointx_info *refT,int32_t msigflag,int32_t lockblocks,int32_t redeemi,char *redeemscript,int32_t p2shflag,char *privkeystr,int32_t privkeyind,char *othersig,char *otherpubkey,char *checkprivkey)
|
||
|
{
|
||
|
char hexstr[1024],pubP[128],*sig0,*sig1; bits256 hash2; uint8_t data[4096],sigbuf[512]; struct bp_key key,keyV;
|
||
|
struct cointx_info *T; int32_t i,n; void *sig = NULL; size_t siglen = 0; struct cointx_input *vin;
|
||
|
if ( (T= calloc(1,sizeof(*T))) == 0 )
|
||
|
return(0);
|
||
|
if ( privkeystr != 0 )
|
||
|
btc_setprivkey(&key,privkeystr);
|
||
|
*T = *refT; vin = &T->inputs[redeemi];
|
||
|
for (i=0; i<T->numinputs; i++)
|
||
|
strcpy(T->inputs[i].sigs,"00");
|
||
|
strcpy(vin->sigs,redeemscript);
|
||
|
if ( msigflag == 0 )
|
||
|
{
|
||
|
vin->sequence = (uint32_t)-1;
|
||
|
T->nlocktime = 0;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
if ( vin->sequence == 0 )
|
||
|
vin->sequence = (uint32_t)time(NULL);
|
||
|
if ( T->nlocktime == 0 && lockblocks != 0 )
|
||
|
{
|
||
|
if ( lockblocks != 0 )
|
||
|
{
|
||
|
coin->ramchain.RTblocknum = _get_RTheight(&coin->ramchain.lastgetinfo,coin->name,coin->serverport,coin->userpass,coin->ramchain.RTblocknum);
|
||
|
if ( coin->ramchain.RTblocknum == 0 )
|
||
|
{
|
||
|
printf("cant get RTblocknum for %s\n",coin->name);
|
||
|
free(T);
|
||
|
return(0);
|
||
|
}
|
||
|
lockblocks += coin->ramchain.RTblocknum;
|
||
|
}
|
||
|
T->nlocktime = lockblocks;
|
||
|
}
|
||
|
}
|
||
|
//disp_cointx(&T);
|
||
|
emit_cointx(&hash2,data,sizeof(data),T,coin->mgw.oldtx_format,SIGHASH_ALL);
|
||
|
//printf("HASH2.(%llx)\n",(long long)hash2.txid);
|
||
|
if ( msigflag != 0 )
|
||
|
{
|
||
|
if ( othersig != 0 )
|
||
|
{
|
||
|
n = (int32_t)strlen(otherpubkey) >> 1;
|
||
|
decode_hex(data,n,otherpubkey);
|
||
|
if ( bp_key_init(&keyV) == 0 || bp_pubkey_set(&keyV,data,n) == 0 )
|
||
|
{
|
||
|
printf("cant set pubkey\n");
|
||
|
free(T);
|
||
|
return(0);
|
||
|
}
|
||
|
n = (int32_t)strlen(othersig) >> 1;
|
||
|
decode_hex(data,n,othersig);
|
||
|
if ( data[n-1] != SIGHASH_ALL )
|
||
|
{
|
||
|
printf("othersig.(%s) hash type mismatch %d != %d\n",othersig,data[n-1],SIGHASH_ALL);
|
||
|
free(T);
|
||
|
return(0);
|
||
|
}
|
||
|
if ( bp_verify(&keyV,hash2.bytes,sizeof(hash2),data,n-1) == 0 )
|
||
|
{
|
||
|
hexstr[0] = 0;
|
||
|
if ( checkprivkey != 0 )
|
||
|
{
|
||
|
//printf("checkprivkey.(%s)\n",checkprivkey);
|
||
|
btc_setprivkey(&keyV,checkprivkey);
|
||
|
void *dispkey; size_t slen;
|
||
|
bp_privkey_get(&keyV,&dispkey,&slen);
|
||
|
//for (i=0; i<slen; i++)
|
||
|
// printf("%02x",((uint8_t *)dispkey)[i]);
|
||
|
//printf(" checkkey\n");
|
||
|
if ( bp_sign(&keyV,hash2.bytes,sizeof(hash2),&sig,&siglen) != 0 )
|
||
|
init_hexbytes_noT(hexstr,sigbuf,(int32_t)siglen);
|
||
|
}
|
||
|
printf("othersig.(%s) doesnt verify vs (%s)\n",othersig,hexstr);
|
||
|
//return(0);
|
||
|
} else printf("SIG.%d VERIFIED\n",privkeyind ^ 1);
|
||
|
}
|
||
|
if ( privkeystr != 0 )
|
||
|
{
|
||
|
void *dispkey; size_t slen;
|
||
|
bp_privkey_get(&key,&dispkey,&slen);
|
||
|
//for (i=0; i<slen; i++)
|
||
|
// printf("%02x",((uint8_t *)dispkey)[i]);
|
||
|
//printf(" dispkey.(%s)\n",privkeystr);
|
||
|
if ( bp_sign(&key,hash2.bytes,sizeof(hash2),&sig,&siglen) != 0 )
|
||
|
{
|
||
|
memcpy(sigbuf,sig,siglen);
|
||
|
sigbuf[siglen++] = SIGHASH_ALL;
|
||
|
init_hexbytes_noT(hexstr,sigbuf,(int32_t)siglen);
|
||
|
if ( sigstr != 0 )
|
||
|
strcpy(sigstr,hexstr);
|
||
|
if ( privkeyind == 0 )
|
||
|
sig0 = hexstr, sig1 = othersig != 0 ? othersig : "";
|
||
|
else sig1 = hexstr, sig0 = othersig != 0 ? othersig : "";
|
||
|
sprintf(vin->sigs,"00%02x%s%02x%s51",(int32_t)strlen(sig0)>>1,sig0,(int32_t)strlen(sig1)>>1,sig1);
|
||
|
//printf("after A.(%s) othersig.(%s) siglen.%02lx -> (%s)\n",hexstr,othersig != 0 ? othersig : "",siglen,vin->sigs);
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
printf("error signing\n");
|
||
|
free(T);
|
||
|
return(0);
|
||
|
}
|
||
|
}
|
||
|
else vin->sigs[0] = 0;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
if ( bp_sign(&key,hash2.bytes,sizeof(hash2),&sig,&siglen) != 0 && btc_getpubkey(pubP,data,&key) > 0 )
|
||
|
{
|
||
|
memcpy(sigbuf,sig,siglen);
|
||
|
sigbuf[siglen++] = SIGHASH_ALL;
|
||
|
init_hexbytes_noT(hexstr,sigbuf,(int32_t)siglen);
|
||
|
sprintf(vin->sigs,"%02x%s%02x%s00",(int32_t)siglen,hexstr,(int32_t)strlen(pubP)/2,pubP);
|
||
|
//printf("after P.(%s) siglen.%02lx\n",vin->sigs,siglen);
|
||
|
}
|
||
|
}
|
||
|
if ( vin->sigs[0] != 0 )
|
||
|
{
|
||
|
if ( p2shflag != 0 )
|
||
|
sprintf(&vin->sigs[strlen(vin->sigs)],"4c%02x",(int32_t)strlen(redeemscript)/2);
|
||
|
sprintf(&vin->sigs[strlen(vin->sigs)],"%s",redeemscript);
|
||
|
}
|
||
|
//printf("scriptSig.(%s)\n",vin->sigs);
|
||
|
_emit_cointx(hexstr,sizeof(hexstr),T,coin->mgw.oldtx_format);
|
||
|
//disp_cointx(&T);
|
||
|
free(T);
|
||
|
return(clonestr(hexstr));
|
||
|
//printf("T.msigredeem %d -> (%s)\n",msigflag,hexstr);
|
||
|
}
|
||
|
|
||
|
char *subatomic_fundingtx(char *refredeemscript,struct subatomic_rawtransaction *funding,struct coin777 *coin,char *mypubkey,char *otherpubkey,char *pkhash,uint64_t amount,int32_t lockblocks)
|
||
|
{
|
||
|
char scriptPubKey[128],mycoinaddr[64],p2shaddr[64],sigstr[512],*refundtx=0,*redeemscript,*txid=0; struct subatomic_unspent_tx *utx;
|
||
|
uint64_t total,donation; int32_t num,n=0,lockblock = 0; struct cointx_info *refT; uint8_t rmd160[20];
|
||
|
memset(funding,0,sizeof(*funding));
|
||
|
refredeemscript[0] = 0;
|
||
|
if ( (redeemscript= create_atomictx_scripts(coin->p2shtype,scriptPubKey,p2shaddr,mypubkey,otherpubkey,pkhash)) != 0 )
|
||
|
{
|
||
|
strcpy(refredeemscript,redeemscript);
|
||
|
if ( btc_coinaddr(mycoinaddr,coin->addrtype,mypubkey) != 0 && (utx= gather_unspents(&total,&num,coin,0)) != 0 )
|
||
|
{
|
||
|
donation = subatomic_donation(coin,amount);
|
||
|
//printf("CREATE FUNDING TX.(%s) [%s %s %s] for %.8f -> %s locktime.%u donation %.8f\n",coin->name,mypubkey,otherpubkey,pkhash,dstr(amount),p2shaddr,lockblock,dstr(donation));
|
||
|
if ( subatomic_calc_rawinputs(coin,funding,amount,utx,num,donation) >= amount )
|
||
|
{
|
||
|
if ( funding->amount == amount && funding->change == (funding->inputsum - amount - coin->mgw.txfee - donation) )
|
||
|
{
|
||
|
safecopy(funding->destaddrs[n],p2shaddr,sizeof(funding->destaddrs[n]));
|
||
|
funding->destamounts[n] = amount;
|
||
|
n++;
|
||
|
}
|
||
|
if ( donation != 0 )
|
||
|
{
|
||
|
if ( coin->donationaddress[0] != 0 )
|
||
|
{
|
||
|
safecopy(funding->destaddrs[n],coin->donationaddress,sizeof(funding->destaddrs[n]));
|
||
|
funding->destamounts[n] = donation;
|
||
|
n++;
|
||
|
} else funding->change += donation;
|
||
|
}
|
||
|
if ( funding->change != 0 )
|
||
|
{
|
||
|
if ( coin->changeaddr[0] == 0 )
|
||
|
{
|
||
|
printf("no changeaddress for (%s)\n",coin->name);
|
||
|
return(0);
|
||
|
}
|
||
|
safecopy(funding->destaddrs[n],coin->changeaddr,sizeof(funding->destaddrs[n]));
|
||
|
funding->destamounts[n] = funding->change;
|
||
|
n++;
|
||
|
}
|
||
|
funding->numoutputs = n;
|
||
|
if ( (txid= subatomic_gen_rawtransaction(0,coin,funding,p2shaddr,lockblock,lockblock==0?0xffffffff:(uint32_t)time(NULL),coin->usep2sh!=0?0:redeemscript)) == 0 )
|
||
|
printf("error creating tx\n");
|
||
|
else
|
||
|
{
|
||
|
if ( (refT= calloc(1,sizeof(*refT))) == 0 )
|
||
|
return(0);
|
||
|
refT->version = 1;
|
||
|
refT->timestamp = (uint32_t)time(NULL);
|
||
|
strcpy(refT->inputs[0].tx.txidstr,txid);
|
||
|
refT->inputs[0].tx.vout = 0;
|
||
|
refT->numinputs = 1;
|
||
|
strcpy(scriptPubKey,"76a914");
|
||
|
calc_OP_HASH160(scriptPubKey+6,rmd160,mypubkey);
|
||
|
strcat(scriptPubKey,"88ac");
|
||
|
if ( mycoinaddr[0] != 0 )
|
||
|
{
|
||
|
strcpy(refT->outputs[0].coinaddr,mycoinaddr);
|
||
|
strcpy(refT->outputs[0].script,scriptPubKey);
|
||
|
refT->outputs[0].value = funding->destamounts[0] - coin->mgw.txfee;
|
||
|
refT->numoutputs = 1;
|
||
|
if ( lockblocks == 0 )
|
||
|
lockblocks = 10;
|
||
|
refundtx = subatomic_signp2sh(sigstr,coin,refT,1,lockblocks,0,redeemscript,coin->usep2sh,0,0,0,0,0);
|
||
|
free(refT);
|
||
|
} else printf("cant get %s addr from (%s)\n",coin->name,mypubkey);
|
||
|
}
|
||
|
} else printf("error: probably not enough funds\n");
|
||
|
} else printf("error: btc_coinaddr.(%s)\n",mycoinaddr);
|
||
|
free(redeemscript);
|
||
|
} else printf("subatomic_fundingtx: cant create redeemscript\n");
|
||
|
return(refundtx);
|
||
|
}
|
||
|
|
||
|
char *subatomic_spendtx(struct destbuf *spendtxid,char *vintxid,char *refundsig,struct coin777 *coin,char *otherpubkey,char *mypubkey,char *onetimepubkey,uint64_t amount,char *refundtx,char *refredeemscript)
|
||
|
{
|
||
|
char scriptPubKey[128],p2shaddr[64],rmdstr[41],onetimecoinaddr[64],msigcoinaddr[64],sigstr[512]; cJSON *json;
|
||
|
char *redeemscript,*signedtx,*spendtx=0,*mprivkey,*oprivkey; uint8_t rmd160[20]; long diff=0; struct cointx_info *refundT=0;
|
||
|
refundsig[0] = onetimecoinaddr[0] = msigcoinaddr[0] = spendtxid->buf[0] = vintxid[0] = 0;
|
||
|
if ( btc_coinaddr(onetimecoinaddr,coin->addrtype,onetimepubkey) != 0 && btc_coinaddr(msigcoinaddr,coin->addrtype,mypubkey) != 0 )
|
||
|
{
|
||
|
//printf("mypubkey.(%s) -> (%s)\n",mypubkey,msigcoinaddr);
|
||
|
calc_OP_HASH160(rmdstr,rmd160,onetimepubkey);
|
||
|
amount -= coin->mgw.txfee;
|
||
|
coin->ramchain.RTblocknum = _get_RTheight(&coin->ramchain.lastgetinfo,coin->name,coin->serverport,coin->userpass,coin->ramchain.RTblocknum);
|
||
|
if ( (refundT= _decode_rawtransaction(refundtx,coin->mgw.oldtx_format)) != 0 && refundT->inputs[0].sequence != 0xffffffff && refundT->nlocktime != 0 && (diff= ((long)refundT->nlocktime - coin->ramchain.RTblocknum)) > 1 && diff < 1000 )
|
||
|
{
|
||
|
strcpy(vintxid,refundT->inputs[0].tx.txidstr);
|
||
|
if ( (redeemscript= create_atomictx_scripts(coin->p2shtype,scriptPubKey,p2shaddr,otherpubkey,mypubkey,rmdstr)) != 0 )
|
||
|
{
|
||
|
if ( refundT->outputs[0].value == amount && strcmp(refredeemscript,redeemscript) == 0 && refundT->numinputs == 1 && refundT->numoutputs == 1 )
|
||
|
{
|
||
|
if ( (mprivkey= dumpprivkey(coin->name,coin->serverport,coin->userpass,msigcoinaddr)) != 0 && (oprivkey= dumpprivkey(coin->name,coin->serverport,coin->userpass,onetimecoinaddr)) != 0 )
|
||
|
{
|
||
|
//printf("mprivkey.(%s)\n",mprivkey);
|
||
|
if ( (signedtx= subatomic_signp2sh(refundsig,coin,refundT,1,0,0,redeemscript,coin->usep2sh,mprivkey,1,0,0,0)) != 0 )
|
||
|
{
|
||
|
//printf("one sig.(%s)\n",signedtx);
|
||
|
free(signedtx);
|
||
|
strcpy(refundT->outputs[0].coinaddr,onetimecoinaddr);
|
||
|
sprintf(scriptPubKey,"76a914%s88ac",rmdstr);
|
||
|
strcpy(refundT->outputs[0].script,scriptPubKey);
|
||
|
spendtx = subatomic_signp2sh(sigstr,coin,refundT,0,0,0,redeemscript,coin->usep2sh,oprivkey,0,0,0,0);
|
||
|
if ( (json= get_decoderaw_json(coin,spendtx)) != 0 )
|
||
|
{
|
||
|
copy_cJSON(spendtxid,jobj(json,"txid"));
|
||
|
free_json(json);
|
||
|
}
|
||
|
} else printf("Error signing\n");
|
||
|
free(mprivkey);
|
||
|
free(oprivkey);
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
if ( mprivkey != 0 )
|
||
|
free(mprivkey);
|
||
|
printf("error getting privkeys M.(%s) onetime.(%s)\n",msigcoinaddr,onetimecoinaddr);
|
||
|
}
|
||
|
} else printf("error (%.8f vs %.8f) comparing redeemscript.(%s) vs (%s) io.(%d %d)\n",dstr(refundT->outputs[0].value),dstr(amount),refredeemscript,redeemscript,refundT->numinputs,refundT->numoutputs);
|
||
|
free(redeemscript);
|
||
|
} else printf("error creating redeemscript\n");
|
||
|
free(refundT);
|
||
|
} else printf("error decoding refundT.%p or diff %ld too big (%u %u)\n",refundT,diff,refundT->nlocktime,coin->ramchain.RTblocknum);
|
||
|
} else printf("error getting addresses (%s) (%s)\n",msigcoinaddr,onetimecoinaddr);
|
||
|
return(spendtx);
|
||
|
}
|
||
|
|
||
|
char *subatomic_validate(struct coin777 *coin,char *pubA,char *pubB,char *pkhash,char *refundtx,char *refundsig)
|
||
|
{
|
||
|
char scriptPubKey[512],mycoinaddr[64],p2shaddr[128],mysig[512],*redeemscript,*privkeystr,*signedrefund=0;
|
||
|
struct cointx_info *refundT;
|
||
|
if ( (refundT= _decode_rawtransaction(refundtx,coin->mgw.oldtx_format)) != 0 && btc_coinaddr(mycoinaddr,coin->addrtype,pubA) != 0 )
|
||
|
{
|
||
|
if ( (privkeystr= dumpprivkey(coin->name,coin->serverport,coin->userpass,mycoinaddr)) != 0 )
|
||
|
{
|
||
|
if ( (redeemscript= create_atomictx_scripts(coin->p2shtype,scriptPubKey,p2shaddr,pubA,pubB,pkhash)) != 0 )
|
||
|
{
|
||
|
if ( (signedrefund= subatomic_signp2sh(mysig,coin,refundT,1,0,0,redeemscript,1,privkeystr,0,refundsig,pubB,0)) != 0 )
|
||
|
{
|
||
|
//printf("SIGNEDREFUND.(%s)\n",signedrefund);
|
||
|
}
|
||
|
free(redeemscript);
|
||
|
}
|
||
|
free(privkeystr);
|
||
|
}
|
||
|
free(refundT);
|
||
|
}
|
||
|
return(signedrefund);
|
||
|
}
|
||
|
|
||
|
void test_subatomic()
|
||
|
{
|
||
|
char pkhash[8192],pubA[67],pubB[67],pubP[67]; uint8_t tmpbuf[512]; struct coin777 *coin;
|
||
|
struct subatomic_rawtransaction funding; char refredeemscript[4096],vintxid[128],swapacct[64],othercoinaddr[64],mycoinaddr[64],onetimeaddr[64],refundsig[512],*signedrefund,*refundtx=0,*spendtx=0;
|
||
|
uint64_t amount; struct destbuf pubkey; struct destbuf spendtxid;
|
||
|
coin = coin777_find("BTCD",1);
|
||
|
if ( strcmp(coin->name,"BTC") == 0 )
|
||
|
coin->mgw.oldtx_format = 1;
|
||
|
//coin->usep2sh = 0;
|
||
|
strcpy(mycoinaddr,coin->atomicsend),get_pubkey(&pubkey,coin->name,coin->serverport,coin->userpass,mycoinaddr), strcpy(pubA,pubkey.buf);
|
||
|
strcpy(othercoinaddr,coin->atomicrecv),get_pubkey(&pubkey,coin->name,coin->serverport,coin->userpass,othercoinaddr), strcpy(pubB,pubkey.buf);
|
||
|
sprintf(swapacct,"%u",777);
|
||
|
if ( get_acct_coinaddr(onetimeaddr,coin->name,coin->serverport,coin->userpass,swapacct) != 0 )
|
||
|
{
|
||
|
get_pubkey(&pubkey,coin->name,coin->serverport,coin->userpass,onetimeaddr);
|
||
|
strcpy(pubP,pubkey.buf);
|
||
|
printf("onetimeadddr.(%s) pubkey.(%s)\n",onetimeaddr,pubP);
|
||
|
}
|
||
|
calc_OP_HASH160(pkhash,tmpbuf,pubP);
|
||
|
amount = 20000;
|
||
|
printf("pkhash.(%s)\n",pkhash);
|
||
|
if ( (refundtx= subatomic_fundingtx(refredeemscript,&funding,coin,pubA,pubB,pkhash,20000,10)) != 0 )
|
||
|
{
|
||
|
printf("FUNDING.(%s) unsignedrefund.(%s)\n",funding.signedtransaction,refundtx);
|
||
|
if ( (spendtx= subatomic_spendtx(&spendtxid,vintxid,refundsig,coin,pubA,pubB,pubP,amount,refundtx,refredeemscript)) != 0 )
|
||
|
{
|
||
|
printf("vin.%s SPENDTX.(%s) %s refundsig.(%s)\n",vintxid,spendtx,spendtxid.buf,refundsig);
|
||
|
if ( (signedrefund= subatomic_validate(coin,pubA,pubB,pkhash,refundtx,refundsig)) != 0 )
|
||
|
{
|
||
|
printf("SIGNEDREFUND.(%s)\n",signedrefund);
|
||
|
free(signedrefund);
|
||
|
} else printf("null signedrefund\n");
|
||
|
} else printf("null spendtx\n");
|
||
|
free(refundtx);
|
||
|
}
|
||
|
getchar();
|
||
|
}
|
||
|
#endif
|
||
|
|
||
|
#endif
|