]> Git Repo - VerusCoin.git/blame - src/komodo_bitcoind.h
test
[VerusCoin.git] / src / komodo_bitcoind.h
CommitLineData
d019c447 1/******************************************************************************
37fed603 2 * Copyright © 2014-2017 The SuperNET Developers. *
d019c447 3 * *
4 * See the AUTHORS, DEVELOPER-AGREEMENT and LICENSE files at *
5 * the top-level directory of this distribution for the individual copyright *
6 * holder information and the developer policies on copyright and licensing. *
7 * *
8 * Unless otherwise agreed in a custom licensing agreement, no part of the *
9 * SuperNET software, including this file may be copied, modified, propagated *
10 * or distributed except according to the terms contained in the LICENSE file *
11 * *
12 * Removal or modification of this copyright notice is prohibited. *
13 * *
14 ******************************************************************************/
15
3eea72f2 16// komodo functions that interact with bitcoind C++
17
9b0e1808 18#ifdef _WIN32
19#include <curl.h>
20#include <easy.h>
21#else
22#include <curl/curl.h>
23#include <curl/easy.h>
24#endif
25
fba6038c 26#define issue_curl(cmdstr) bitcoind_RPC(0,(char *)"curl",(char *)"http://127.0.0.1:7776",0,0,(char *)(cmdstr))
77be6cd9 27
9b0e1808 28struct MemoryStruct { char *memory; size_t size; };
c2b50ce0 29struct return_string { char *ptr; size_t len; };
30
31// return data from the server
32#define CURL_GLOBAL_ALL (CURL_GLOBAL_SSL|CURL_GLOBAL_WIN32)
33#define CURL_GLOBAL_SSL (1<<0)
34#define CURL_GLOBAL_WIN32 (1<<1)
35
36
4ec67e74 37/************************************************************************
38 *
39 * Initialize the string handler so that it is thread safe
40 *
41 ************************************************************************/
42
43void init_string(struct return_string *s)
44{
45 s->len = 0;
46 s->ptr = (char *)calloc(1,s->len+1);
47 if ( s->ptr == NULL )
48 {
49 fprintf(stderr,"init_string malloc() failed\n");
50 exit(-1);
51 }
52 s->ptr[0] = '\0';
53}
c2b50ce0 54
4ec67e74 55/************************************************************************
56 *
57 * Use the "writer" to accumulate text until done
58 *
59 ************************************************************************/
60
61size_t accumulatebytes(void *ptr,size_t size,size_t nmemb,struct return_string *s)
62{
63 size_t new_len = s->len + size*nmemb;
64 s->ptr = (char *)realloc(s->ptr,new_len+1);
65 if ( s->ptr == NULL )
66 {
67 fprintf(stderr, "accumulate realloc() failed\n");
68 exit(-1);
69 }
70 memcpy(s->ptr+s->len,ptr,size*nmemb);
71 s->ptr[new_len] = '\0';
72 s->len = new_len;
73 return(size * nmemb);
74}
c2b50ce0 75
76/************************************************************************
77 *
78 * return the current system time in milliseconds
79 *
80 ************************************************************************/
81
82#define EXTRACT_BITCOIND_RESULT // if defined, ensures error is null and returns the "result" field
83#ifdef EXTRACT_BITCOIND_RESULT
84
85/************************************************************************
86 *
87 * perform post processing of the results
88 *
89 ************************************************************************/
90
91char *post_process_bitcoind_RPC(char *debugstr,char *command,char *rpcstr,char *params)
92{
93 long i,j,len; char *retstr = 0; cJSON *json,*result,*error;
94 //printf("<<<<<<<<<<< bitcoind_RPC: %s post_process_bitcoind_RPC.%s.[%s]\n",debugstr,command,rpcstr);
95 if ( command == 0 || rpcstr == 0 || rpcstr[0] == 0 )
96 {
97 if ( strcmp(command,"signrawtransaction") != 0 )
98 printf("<<<<<<<<<<< bitcoind_RPC: %s post_process_bitcoind_RPC.%s.[%s]\n",debugstr,command,rpcstr);
99 return(rpcstr);
100 }
101 json = cJSON_Parse(rpcstr);
102 if ( json == 0 )
103 {
104 printf("<<<<<<<<<<< bitcoind_RPC: %s post_process_bitcoind_RPC.%s can't parse.(%s) params.(%s)\n",debugstr,command,rpcstr,params);
105 free(rpcstr);
106 return(0);
107 }
108 result = cJSON_GetObjectItem(json,"result");
109 error = cJSON_GetObjectItem(json,"error");
110 if ( error != 0 && result != 0 )
111 {
112 if ( (error->type&0xff) == cJSON_NULL && (result->type&0xff) != cJSON_NULL )
113 {
114 retstr = cJSON_Print(result);
115 len = strlen(retstr);
116 if ( retstr[0] == '"' && retstr[len-1] == '"' )
117 {
118 for (i=1,j=0; i<len-1; i++,j++)
119 retstr[j] = retstr[i];
120 retstr[j] = 0;
121 }
122 }
123 else if ( (error->type&0xff) != cJSON_NULL || (result->type&0xff) != cJSON_NULL )
124 {
125 if ( strcmp(command,"signrawtransaction") != 0 )
126 printf("<<<<<<<<<<< bitcoind_RPC: %s post_process_bitcoind_RPC (%s) error.%s\n",debugstr,command,rpcstr);
127 }
128 free(rpcstr);
129 } else retstr = rpcstr;
130 free_json(json);
131 //fprintf(stderr,"<<<<<<<<<<< bitcoind_RPC: postprocess returns.(%s)\n",retstr);
132 return(retstr);
133}
134#endif
135
136/************************************************************************
137 *
138 * perform the query
139 *
140 ************************************************************************/
141
c2b50ce0 142char *bitcoind_RPC(char **retstrp,char *debugstr,char *url,char *userpass,char *command,char *params)
143{
144 static int didinit,count,count2; static double elapsedsum,elapsedsum2;
145 struct curl_slist *headers = NULL; struct return_string s; CURLcode res; CURL *curl_handle;
146 char *bracket0,*bracket1,*databuf = 0; long len; int32_t specialcase,numretries; double starttime;
147 if ( didinit == 0 )
148 {
149 didinit = 1;
150 curl_global_init(CURL_GLOBAL_ALL); //init the curl session
151 }
152 numretries = 0;
153 if ( debugstr != 0 && strcmp(debugstr,"BTCD") == 0 && command != 0 && strcmp(command,"SuperNET") == 0 )
154 specialcase = 1;
155 else specialcase = 0;
156 if ( url[0] == 0 )
157 strcpy(url,"http://127.0.0.1:7876/nxt");
158 if ( specialcase != 0 && 0 )
159 printf("<<<<<<<<<<< bitcoind_RPC: debug.(%s) url.(%s) command.(%s) params.(%s)\n",debugstr,url,command,params);
160try_again:
161 if ( retstrp != 0 )
162 *retstrp = 0;
163 starttime = OS_milliseconds();
164 curl_handle = curl_easy_init();
165 init_string(&s);
166 headers = curl_slist_append(0,"Expect:");
167
168 curl_easy_setopt(curl_handle,CURLOPT_USERAGENT,"mozilla/4.0");//"Mozilla/4.0 (compatible; )");
169 curl_easy_setopt(curl_handle,CURLOPT_HTTPHEADER, headers);
170 curl_easy_setopt(curl_handle,CURLOPT_URL, url);
4ec67e74 171 curl_easy_setopt(curl_handle,CURLOPT_WRITEFUNCTION, (void *)accumulatebytes); // send all data to this function
c2b50ce0 172 curl_easy_setopt(curl_handle,CURLOPT_WRITEDATA, &s); // we pass our 's' struct to the callback
173 curl_easy_setopt(curl_handle,CURLOPT_NOSIGNAL, 1L); // supposed to fix "Alarm clock" and long jump crash
174 curl_easy_setopt(curl_handle,CURLOPT_NOPROGRESS, 1L); // no progress callback
175 if ( strncmp(url,"https",5) == 0 )
176 {
177 curl_easy_setopt(curl_handle,CURLOPT_SSL_VERIFYPEER,0);
178 curl_easy_setopt(curl_handle,CURLOPT_SSL_VERIFYHOST,0);
179 }
180 if ( userpass != 0 )
181 curl_easy_setopt(curl_handle,CURLOPT_USERPWD, userpass);
182 databuf = 0;
183 if ( params != 0 )
184 {
185 if ( command != 0 && specialcase == 0 )
186 {
187 len = strlen(params);
188 if ( len > 0 && params[0] == '[' && params[len-1] == ']' ) {
189 bracket0 = bracket1 = (char *)"";
190 }
191 else
192 {
193 bracket0 = (char *)"[";
194 bracket1 = (char *)"]";
195 }
196
197 databuf = (char *)malloc(256 + strlen(command) + strlen(params));
198 sprintf(databuf,"{\"id\":\"jl777\",\"method\":\"%s\",\"params\":%s%s%s}",command,bracket0,params,bracket1);
199 //printf("url.(%s) userpass.(%s) databuf.(%s)\n",url,userpass,databuf);
200 //
201 } //else if ( specialcase != 0 ) fprintf(stderr,"databuf.(%s)\n",params);
202 curl_easy_setopt(curl_handle,CURLOPT_POST,1L);
203 if ( databuf != 0 )
204 curl_easy_setopt(curl_handle,CURLOPT_POSTFIELDS,databuf);
205 else curl_easy_setopt(curl_handle,CURLOPT_POSTFIELDS,params);
206 }
207 //laststart = milliseconds();
208 res = curl_easy_perform(curl_handle);
209 curl_slist_free_all(headers);
210 curl_easy_cleanup(curl_handle);
211 if ( databuf != 0 ) // clean up temporary buffer
212 {
213 free(databuf);
214 databuf = 0;
215 }
216 if ( res != CURLE_OK )
217 {
218 numretries++;
219 if ( specialcase != 0 )
220 {
221 printf("<<<<<<<<<<< bitcoind_RPC.(%s): BTCD.%s timeout params.(%s) s.ptr.(%s) err.%d\n",url,command,params,s.ptr,res);
222 free(s.ptr);
223 return(0);
224 }
c59a3beb 225 else if ( numretries >= 1 )
c2b50ce0 226 {
bb7f7473 227 //printf("Maximum number of retries exceeded!\n");
c2b50ce0 228 free(s.ptr);
229 return(0);
230 }
32323ea1 231 if ( (rand() % 1000) == 0 )
232 printf( "curl_easy_perform() failed: %s %s.(%s %s), retries: %d\n",curl_easy_strerror(res),debugstr,url,command,numretries);
c2b50ce0 233 free(s.ptr);
234 sleep((1<<numretries));
235 goto try_again;
236
237 }
238 else
239 {
240 if ( command != 0 && specialcase == 0 )
241 {
242 count++;
243 elapsedsum += (OS_milliseconds() - starttime);
32323ea1 244 if ( (count % 1000000) == 0)
c2b50ce0 245 printf("%d: ave %9.6f | elapsed %.3f millis | bitcoind_RPC.(%s) url.(%s)\n",count,elapsedsum/count,(OS_milliseconds() - starttime),command,url);
246 if ( retstrp != 0 )
247 {
248 *retstrp = s.ptr;
249 return(s.ptr);
250 }
251 return(post_process_bitcoind_RPC(debugstr,command,s.ptr,params));
252 }
253 else
254 {
255 if ( 0 && specialcase != 0 )
256 fprintf(stderr,"<<<<<<<<<<< bitcoind_RPC: BTCD.(%s) -> (%s)\n",params,s.ptr);
257 count2++;
258 elapsedsum2 += (OS_milliseconds() - starttime);
259 if ( (count2 % 10000) == 0)
260 printf("%d: ave %9.6f | elapsed %.3f millis | NXT calls.(%s) cmd.(%s)\n",count2,elapsedsum2/count2,(double)(OS_milliseconds() - starttime),url,command);
261 return(s.ptr);
262 }
263 }
264 printf("bitcoind_RPC: impossible case\n");
265 free(s.ptr);
266 return(0);
267}
9b0e1808 268
269static size_t WriteMemoryCallback(void *ptr,size_t size,size_t nmemb,void *data)
270{
271 size_t realsize = (size * nmemb);
272 struct MemoryStruct *mem = (struct MemoryStruct *)data;
b858fa74 273 mem->memory = (char *)((ptr != 0) ? realloc(mem->memory,mem->size + realsize + 1) : malloc(mem->size + realsize + 1));
9b0e1808 274 if ( mem->memory != 0 )
275 {
276 if ( ptr != 0 )
277 memcpy(&(mem->memory[mem->size]),ptr,realsize);
278 mem->size += realsize;
279 mem->memory[mem->size] = 0;
280 }
281 //printf("got %d bytes\n",(int32_t)(size*nmemb));
282 return(realsize);
283}
284
05d3d5ff 285char *curl_post(CURL **cHandlep,char *url,char *userpass,char *postfields,char *hdr0,char *hdr1,char *hdr2,char *hdr3)
9b0e1808 286{
287 struct MemoryStruct chunk; CURL *cHandle; long code; struct curl_slist *headers = 0;
288 if ( (cHandle= *cHandlep) == NULL )
289 *cHandlep = cHandle = curl_easy_init();
290 else curl_easy_reset(cHandle);
291 //#ifdef DEBUG
292 //curl_easy_setopt(cHandle,CURLOPT_VERBOSE, 1);
293 //#endif
294 curl_easy_setopt(cHandle,CURLOPT_USERAGENT,"mozilla/4.0");//"Mozilla/4.0 (compatible; )");
295 curl_easy_setopt(cHandle,CURLOPT_SSL_VERIFYPEER,0);
296 //curl_easy_setopt(cHandle,CURLOPT_SSLVERSION,1);
297 curl_easy_setopt(cHandle,CURLOPT_URL,url);
298 curl_easy_setopt(cHandle,CURLOPT_CONNECTTIMEOUT,10);
299 if ( userpass != 0 && userpass[0] != 0 )
300 curl_easy_setopt(cHandle,CURLOPT_USERPWD,userpass);
301 if ( postfields != 0 && postfields[0] != 0 )
302 {
303 curl_easy_setopt(cHandle,CURLOPT_POST,1);
304 curl_easy_setopt(cHandle,CURLOPT_POSTFIELDS,postfields);
305 }
306 if ( hdr0 != NULL && hdr0[0] != 0 )
307 {
308 //printf("HDR0.(%s) HDR1.(%s) HDR2.(%s) HDR3.(%s)\n",hdr0!=0?hdr0:"",hdr1!=0?hdr1:"",hdr2!=0?hdr2:"",hdr3!=0?hdr3:"");
309 headers = curl_slist_append(headers,hdr0);
310 if ( hdr1 != 0 && hdr1[0] != 0 )
311 headers = curl_slist_append(headers,hdr1);
312 if ( hdr2 != 0 && hdr2[0] != 0 )
313 headers = curl_slist_append(headers,hdr2);
314 if ( hdr3 != 0 && hdr3[0] != 0 )
315 headers = curl_slist_append(headers,hdr3);
316 } //headers = curl_slist_append(0,"Expect:");
317 if ( headers != 0 )
318 curl_easy_setopt(cHandle,CURLOPT_HTTPHEADER,headers);
319 //res = curl_easy_perform(cHandle);
320 memset(&chunk,0,sizeof(chunk));
321 curl_easy_setopt(cHandle,CURLOPT_WRITEFUNCTION,WriteMemoryCallback);
322 curl_easy_setopt(cHandle,CURLOPT_WRITEDATA,(void *)&chunk);
323 curl_easy_perform(cHandle);
324 curl_easy_getinfo(cHandle,CURLINFO_RESPONSE_CODE,&code);
325 if ( headers != 0 )
326 curl_slist_free_all(headers);
327 if ( code != 200 )
328 printf("(%s) server responded with code %ld (%s)\n",url,code,chunk.memory);
329 return(chunk.memory);
330}
331
0ded57c8 332char *komodo_issuemethod(char *userpass,char *method,char *params,uint16_t port)
9b0e1808 333{
4ac6cd92 334 //static void *cHandle;
a9132082 335 char url[512],*retstr=0,*retstr2=0,postdata[8192];
365378b5 336 if ( params == 0 || params[0] == 0 )
337 params = (char *)"[]";
9b0e1808 338 if ( strlen(params) < sizeof(postdata)-128 )
339 {
3c0f5d94 340 sprintf(url,(char *)"http://127.0.0.1:%u",port);
9b0e1808 341 sprintf(postdata,"{\"method\":\"%s\",\"params\":%s}",method,params);
f38345e9 342 //printf("postdata.(%s) USERPASS.(%s)\n",postdata,KMDUSERPASS);
0ded57c8 343 retstr2 = bitcoind_RPC(&retstr,(char *)"debug",url,userpass,method,params);
c2b50ce0 344 //retstr = curl_post(&cHandle,url,USERPASS,postdata,0,0,0,0);
9b0e1808 345 }
1ccb1cf0 346 return(retstr2);
9b0e1808 347}
348
b55c2367 349int32_t komodo_verifynotarizedscript(uint8_t *script,int32_t len,uint256 NOTARIZED_HASH)
350{
351 int32_t i; uint256 hash; char params[256];
352 for (i=0; i<32; i++)
353 printf("%02x",((uint8_t *)&NOTARIZED_HASH)[i]);
354 printf(" notarized, ");
355 for (i=0; i<32; i++)
356 {
357 ((uint8_t *)&hash)[i] = script[31+2-i];
358 printf("%02x",script[31+2-i]);
359 }
360 printf(" <- opreturn?\n");
361 if ( hash == NOTARIZED_HASH )
362 return(0);
363 else return(-1);
364}
365
366int32_t komodo_verifynotarization(char *symbol,char *dest,int32_t height,int32_t NOTARIZED_HEIGHT,uint256 NOTARIZED_HASH,uint256 NOTARIZED_DESTTXID)
367{
731f2c84 368 char params[256],*jsonstr,*hexstr; uint8_t script[8192]; int32_t i,n,len,retval = -1; cJSON *txjson,*vouts,*vout,*skey;
eee28662 369 params[0] = '[';
370 params[1] = '"';
371 for (i=0; i<32; i++)
372 sprintf(&params[i*2 + 2],"%02x",((uint8_t *)&NOTARIZED_DESTTXID)[i]);
373 strcat(params,"\", 1]");
374 //sprintf(params,"[\"%s\", 1]",NOTARIZED_DESTTXID.ToString().c_str());
de28264f 375 printf("src.%s dest.%s params.[%s] ht.%d notarized.%d\n",symbol,dest,params,height,NOTARIZED_HEIGHT);
b55c2367 376 if ( strcmp(dest,"KMD") == 0 )
b148643e 377 {
378 if ( KMDUSERPASS[0] != 0 )
379 jsonstr = komodo_issuemethod(KMDUSERPASS,(char *)"getrawtransaction",params,7771);
380 //else jsonstr = _dex_getrawtransaction();
48d0e5df 381 else return(0); // need universal way to issue DEX* API, since notaries mine most blocks, this ok
b148643e 382 }
b55c2367 383 else if ( strcmp(dest,"BTC") == 0 )
b148643e 384 {
385 if ( BTCUSERPASS[0] != 0 )
48d0e5df 386 {
eee28662 387 //printf("BTCUSERPASS.(%s)\n",BTCUSERPASS);
b148643e 388 jsonstr = komodo_issuemethod(BTCUSERPASS,(char *)"getrawtransaction",params,8332);
48d0e5df 389 }
b148643e 390 //else jsonstr = _dex_getrawtransaction();
48d0e5df 391 else return(0);
b148643e 392 }
b55c2367 393 else return(-1);
394 if ( jsonstr != 0 )
395 {
48d0e5df 396 printf("GOT.(%s)\n",jsonstr);
b55c2367 397 if ( (txjson= cJSON_Parse(jsonstr)) != 0 )
398 {
523a5dc7 399 if ( (vouts= jarray(&n,txjson,(char *)"vout")) > 0 )
b55c2367 400 {
401 vout = jitem(vouts,n-1);
48d0e5df 402 printf("vout.(%s)\n",jprint(vout,0));
523a5dc7 403 if ( (skey= jobj(vout,(char *)"scriptPubKey")) != 0 )
b55c2367 404 {
523a5dc7 405 if ( (hexstr= jstr(skey,(char *)"hex")) != 0 )
b55c2367 406 {
48d0e5df 407 printf("HEX.(%s)\n",hexstr);
b55c2367 408 len = strlen(hexstr) >> 1;
409 decode_hex(script,len,hexstr);
410 retval = komodo_verifynotarizedscript(script,len,NOTARIZED_HASH);
411 }
412 }
413 }
414 free_json(txjson);
415 }
416 free(jsonstr);
417 }
418 return(retval);
419}
420
23256cca 421uint256 komodo_getblockhash(int32_t height)
422{
dffceecf 423 uint256 hash; char params[128],*hexstr,*jsonstr; cJSON *result; int32_t i; uint8_t revbuf[32];
23256cca 424 memset(&hash,0,sizeof(hash));
425 sprintf(params,"[%d]",height);
ccf895cf 426 if ( (jsonstr= komodo_issuemethod(KMDUSERPASS,(char *)"getblockhash",params,7771)) != 0 )
23256cca 427 {
ccf895cf 428 if ( (result= cJSON_Parse(jsonstr)) != 0 )
429 {
25391719 430 if ( (hexstr= jstr(result,(char *)"result")) != 0 )
ccf895cf 431 {
432 if ( is_hexstr(hexstr,0) == 64 )
e9f49d50 433 {
434 decode_hex(revbuf,32,hexstr);
435 for (i=0; i<32; i++)
436 ((uint8_t *)&hash)[i] = revbuf[31-i];
437 }
ccf895cf 438 }
439 free_json(result);
440 }
5f1f19fd 441 printf("KMD hash.%d (%s) %x\n",height,jsonstr,*(uint32_t *)&hash);
1fd57718 442 free(jsonstr);
23256cca 443 }
444 return(hash);
445}
446
6b0e06b5 447uint256 _komodo_getblockhash(int32_t height);
448
23256cca 449uint64_t komodo_seed(int32_t height)
450{
39b462c7 451 uint64_t seed = 0;
8aef9e40 452 if ( 0 ) // problem during init time, seeds are needed for loading blockindex, so null seeds...
453 {
39b462c7 454 uint256 hash,zero; CBlockIndex *pindex;
8aef9e40 455 memset(&hash,0,sizeof(hash));
456 memset(&zero,0,sizeof(zero));
457 if ( height > 10 )
458 height -= 10;
459 if ( ASSETCHAINS_SYMBOL[0] == 0 )
460 hash = _komodo_getblockhash(height);
461 if ( memcmp(&hash,&zero,sizeof(hash)) == 0 )
462 hash = komodo_getblockhash(height);
463 int32_t i;
464 for (i=0; i<32; i++)
465 printf("%02x",((uint8_t *)&hash)[i]);
466 printf(" seed.%d\n",height);
8c072b0b 467 seed = arith_uint256(hash.GetHex()).GetLow64();
8aef9e40 468 }
469 else
470 {
93fb251c 471 seed = (height << 13) ^ (height << 2);
472 seed <<= 21;
8aef9e40 473 seed |= (height & 0xffffffff);
93fb251c 474 seed ^= (seed << 17) ^ (seed << 1);
8aef9e40 475 }
23256cca 476 return(seed);
477}
478
3eea72f2 479uint32_t komodo_txtime(uint256 hash)
480{
481 CTransaction tx;
482 uint256 hashBlock;
7637aa7f 483 if (!GetTransaction(hash, tx,
484#ifndef KOMODO_ZCASH
485 Params().GetConsensus(),
486#endif
487 hashBlock, true))
3eea72f2 488 {
489 //printf("null GetTransaction\n");
490 return(tx.nLockTime);
491 }
492 return(0);
493}
494
495void komodo_disconnect(CBlockIndex *pindex,CBlock& block)
496{
b46eae72 497 char symbol[16],dest[16]; struct komodo_state *sp;
c93dc546 498 komodo_init(pindex->nHeight);
f3a1de3a 499 if ( (sp= komodo_stateptr(symbol,dest)) != 0 )
7a183152 500 {
9c432d66 501 //sp->rewinding = pindex->nHeight;
502 //fprintf(stderr,"-%d ",pindex->nHeight);
7a183152 503 } else printf("komodo_disconnect: ht.%d cant get komodo_state.(%s)\n",pindex->nHeight,ASSETCHAINS_SYMBOL);
3eea72f2 504}
505
f24b36ca 506int32_t komodo_is_notarytx(const CTransaction& tx)
507{
508 uint8_t *ptr,crypto777[33];
509#ifdef KOMODO_ZCASH
ed652d8b 510 ptr = (uint8_t *)tx.vout[0].scriptPubKey.data();
f24b36ca 511#else
ed652d8b 512 ptr = (uint8_t *)&tx.vout[0].scriptPubKey[0];
f24b36ca 513#endif
514 decode_hex(crypto777,33,(char *)CRYPTO777_PUBSECPSTR);
515 if ( memcmp(ptr+1,crypto777,33) == 0 )
516 {
0e2acef5 517 //printf("found notarytx\n");
f24b36ca 518 return(1);
519 }
520 else return(0);
521}
522
3eea72f2 523int32_t komodo_block2height(CBlock *block)
524{
189d9dee 525 int32_t i,n,height = 0; uint8_t *ptr;
189d9dee 526#ifdef KOMODO_ZCASH
527 ptr = (uint8_t *)block->vtx[0].vin[0].scriptSig.data();
528#else
529 ptr = (uint8_t *)&block->vtx[0].vin[0].scriptSig[0];
530#endif
3eea72f2 531 if ( block->vtx[0].vin[0].scriptSig.size() > 5 )
532 {
533 //for (i=0; i<6; i++)
534 // printf("%02x",ptr[i]);
535 n = ptr[0];
536 for (i=0; i<n; i++)
537 {
538 //03bb81000101(bb 187) (81 48001) (00 12288256) <- coinbase.6 ht.12288256
539 height += ((uint32_t)ptr[i+1] << (i*8));
540 //printf("(%02x %x %d) ",ptr[i+1],((uint32_t)ptr[i+1] << (i*8)),height);
541 }
542 //printf(" <- coinbase.%d ht.%d\n",(int32_t)block->vtx[0].vin[0].scriptSig.size(),height);
543 }
bc9f2e03 544 //komodo_init(height);
3eea72f2 545 return(height);
546}
547
548void komodo_block2pubkey33(uint8_t *pubkey33,CBlock& block)
549{
2dbf06e8 550 int32_t n;
da00f2dd 551#ifdef KOMODO_ZCASH
484f8777 552 uint8_t *ptr = (uint8_t *)block.vtx[0].vout[0].scriptPubKey.data();
da00f2dd 553#else
554 uint8_t *ptr = (uint8_t *)&block.vtx[0].vout[0].scriptPubKey[0];
555#endif
bc9f2e03 556 //komodo_init(0);
18594cf7 557 n = block.vtx[0].vout[0].scriptPubKey.size();
558 if ( n == 35 )
559 memcpy(pubkey33,ptr+1,33);
560 else memset(pubkey33,0,33);
3eea72f2 561}
562
d7093b33 563int32_t komodo_blockload(CBlock& block,CBlockIndex *pindex)
875cf683 564{
565 block.SetNull();
566 // Open history file to read
567 CAutoFile filein(OpenBlockFile(pindex->GetBlockPos(),true),SER_DISK,CLIENT_VERSION);
568 if (filein.IsNull())
569 return(-1);
570 // Read block
571 try { filein >> block; }
572 catch (const std::exception& e)
573 {
574 fprintf(stderr,"readblockfromdisk err B\n");
575 return(-1);
576 }
577 return(0);
578}
579
3eea72f2 580void komodo_index2pubkey33(uint8_t *pubkey33,CBlockIndex *pindex,int32_t height)
581{
582 CBlock block;
bc9f2e03 583 //komodo_init(height);
3eea72f2 584 memset(pubkey33,0,33);
585 if ( pindex != 0 )
586 {
a92123ef 587 if ( komodo_blockload(block,pindex) == 0 )
588 komodo_block2pubkey33(pubkey33,block);
3eea72f2 589 }
590 else
591 {
592 // height -> pubkey33
593 //printf("extending chaintip komodo_index2pubkey33 height.%d need to get pubkey33\n",height);
594 }
595}
596
875cf683 597void komodo_connectpindex(CBlockIndex *pindex)
598{
599 CBlock block;
a92123ef 600 if ( komodo_blockload(block,pindex) == 0 )
601 komodo_connectblock(pindex,block);
875cf683 602}
603
6034c671 604int32_t komodo_notaries(uint8_t pubkeys[64][33],int32_t height);
605
4a4e912b 606int8_t komodo_minerid(int32_t height)
607{
18594cf7 608 int32_t num,i; CBlockIndex *pindex; uint8_t pubkey33[33],pubkeys[64][33];
fc318ffe 609 if ( (pindex= chainActive[height]) != 0 )
610 {
18594cf7 611 komodo_index2pubkey33(pubkey33,pindex,height);
612 if ( (num= komodo_notaries(pubkeys,height)) > 0 )
e9ed4986 613 {
18594cf7 614 for (i=0; i<num; i++)
615 if ( memcmp(pubkeys[i],pubkey33,33) == 0 )
616 return(i);
fc318ffe 617 }
618 }
619 return(-1);
4a4e912b 620}
621
dbaf1154 622int32_t komodo_minerids(uint8_t *minerids,int32_t height)
623{
624 int32_t i,n=0;
625 for (i=0; i<1000; i++,n++)
056447a6 626 {
627 if ( height-i <= 0 )
628 break;
dbaf1154 629 minerids[i] = komodo_minerid(height - i);
056447a6 630 }
dbaf1154 631 return(n);
632}
633
4a4e912b 634int32_t komodo_is_special(int32_t height,uint8_t pubkey33[33])
635{
28bc6208 636 int32_t i,notaryid,minerid,limit;
4a4e912b 637 komodo_chosennotary(&notaryid,height,pubkey33);
638 if ( height >= 34000 && notaryid >= 0 )
639 {
031b1129 640 if ( height < 79693 )
641 limit = 64;
642 else if ( height < 82000 )
3fe63a7d 643 limit = 8;
28bc6208 644 else limit = 66;
1b3ce02c 645 for (i=1; i<limit; i++)
4a4e912b 646 {
e21ffa1d 647 if ( komodo_minerid(height-i) == notaryid )
66e3c532 648 {
6a645f7b 649 //fprintf(stderr,"ht.%d notaryid.%d already mined -i.%d\n",height,notaryid,i);
4a4e912b 650 return(-1);
66e3c532 651 }
4a4e912b 652 }
6a645f7b 653 //fprintf(stderr,"special notaryid.%d ht.%d limit.%d\n",notaryid,height,limit);
4a4e912b 654 return(1);
655 }
656 return(0);
657}
658
b62d7030 659int32_t komodo_checkpoint(int32_t *notarized_heightp,int32_t nHeight,uint256 hash)
660{
d82623d2 661 int32_t notarized_height; uint256 notarized_hash,notarized_desttxid; CBlockIndex *notary; CBlockIndex *pindex;
662 if ( (pindex= chainActive.Tip()) == 0 )
663 return(-1);
664 notarized_height = komodo_notarizeddata(pindex->nHeight,&notarized_hash,&notarized_desttxid);
b62d7030 665 *notarized_heightp = notarized_height;
aba9ea3c 666 if ( notarized_height >= 0 && notarized_height <= pindex->nHeight && (notary= mapBlockIndex[notarized_hash]) != 0 )
b62d7030 667 {
d82623d2 668 //printf("nHeight.%d -> (%d %s)\n",pindex->Tip()->nHeight,notarized_height,notarized_hash.ToString().c_str());
b62d7030 669 if ( notary->nHeight == notarized_height ) // if notarized_hash not in chain, reorg
670 {
671 if ( nHeight < notarized_height )
672 {
673 fprintf(stderr,"nHeight.%d < NOTARIZED_HEIGHT.%d\n",nHeight,notarized_height);
674 return(-1);
675 }
676 else if ( nHeight == notarized_height && memcmp(&hash,&notarized_hash,sizeof(hash)) != 0 )
677 {
678 fprintf(stderr,"nHeight.%d == NOTARIZED_HEIGHT.%d, diff hash\n",nHeight,notarized_height);
679 return(-1);
680 }
681 } else fprintf(stderr,"unexpected error notary_hash %s ht.%d at ht.%d\n",notarized_hash.ToString().c_str(),notarized_height,notary->nHeight);
e21ffa1d 682 } else if ( notarized_height > 0 && notarized_height != 73880 )
d82623d2 683 fprintf(stderr,"[%s] couldnt find notarized.(%s %d) ht.%d\n",ASSETCHAINS_SYMBOL,notarized_hash.ToString().c_str(),notarized_height,pindex->nHeight);
b62d7030 684 return(0);
685}
485e48ca 686
c1c95e36 687uint32_t komodo_interest_args(int32_t *txheightp,uint32_t *tiptimep,uint64_t *valuep,uint256 hash,int32_t n)
688{
689 LOCK(cs_main);
690 CTransaction tx; uint256 hashBlock; CBlockIndex *pindex,*tipindex;
691 if ( !GetTransaction(hash,tx,hashBlock,true) )
692 return(0);
693 uint32_t locktime = 0;
694 if ( n < tx.vout.size() )
695 {
696 if ( (pindex= mapBlockIndex[hashBlock]) != 0 && (tipindex= chainActive.Tip()) != 0 )
697 {
698 *valuep = tx.vout[n].nValue;
699 *txheightp = pindex->nHeight;
700 *tiptimep = tipindex->nTime;
701 locktime = tx.nLockTime;
1f52f946 702 //fprintf(stderr,"tx locktime.%u %.8f height.%d | tiptime.%u\n",locktime,(double)*valuep/COIN,*txheightp,*tiptimep);
c1c95e36 703 }
704 }
705 return(locktime);
706}
707
708uint64_t komodo_interest(int32_t txheight,uint64_t nValue,uint32_t nLockTime,uint32_t tiptime);
798f28c7 709uint64_t komodo_accrued_interest(int32_t *txheightp,uint32_t *locktimep,uint256 hash,int32_t n,int32_t checkheight,uint64_t checkvalue)
c1c95e36 710{
798f28c7 711 uint64_t value; uint32_t tiptime;
712 if ( (*locktimep= komodo_interest_args(txheightp,&tiptime,&value,hash,n)) != 0 )
c1c95e36 713 {
798f28c7 714 if ( (checkvalue == 0 || value == checkvalue) && (checkheight == 0 || *txheightp == checkheight) )
715 return(komodo_interest(*txheightp,value,*locktimep,tiptime));
c1c95e36 716 //fprintf(stderr,"nValue %llu lock.%u:%u nTime.%u -> %llu\n",(long long)coins.vout[n].nValue,coins.nLockTime,timestamp,pindex->nTime,(long long)interest);
1c2f0c49 717 else fprintf(stderr,"komodo_accrued_interest value mismatch %llu vs %llu or height mismatch %d vs %d\n",(long long)value,(long long)checkvalue,*txheightp,checkheight);
718 }
c1c95e36 719 return(0);
720}
8a7f7063 721
722int32_t komodo_isrealtime(int32_t *kmdheightp)
723{
d82623d2 724 struct komodo_state *sp; CBlockIndex *pindex;
8a7c9241 725 if ( (sp= komodo_stateptrget((char *)"KMD")) != 0 )
ffbc2f00 726 *kmdheightp = sp->CURRENT_HEIGHT;
727 else *kmdheightp = 0;
d82623d2 728 if ( (pindex= chainActive.Tip()) != 0 && pindex->nHeight == (int32_t)komodo_longestchain() )
8a7f7063 729 return(1);
730 else return(0);
731}
This page took 0.244065 seconds and 4 git commands to generate.