|
|
|
@ -10,6 +10,7 @@
|
|
|
|
|
#include "config.h" |
|
|
|
|
|
|
|
|
|
#include <sys/socket.h> |
|
|
|
|
#include <jansson.h> |
|
|
|
|
#include <string.h> |
|
|
|
|
#include <unistd.h> |
|
|
|
|
|
|
|
|
@ -18,6 +19,29 @@
|
|
|
|
|
#include "generator.h" |
|
|
|
|
#include "bitcoin.h" |
|
|
|
|
|
|
|
|
|
/* Per proxied pool instance data */ |
|
|
|
|
struct proxy_instance { |
|
|
|
|
char *auth; |
|
|
|
|
char *pass; |
|
|
|
|
|
|
|
|
|
char *enonce1; |
|
|
|
|
char *enonce1bin; |
|
|
|
|
char *sessionid; |
|
|
|
|
|
|
|
|
|
tv_t last_message; |
|
|
|
|
|
|
|
|
|
double diff; |
|
|
|
|
int absolute_shares; |
|
|
|
|
int diff_shares; |
|
|
|
|
tv_t last_share; |
|
|
|
|
|
|
|
|
|
int id; /* Message id for sending stratum messages */ |
|
|
|
|
bool no_sessionid; /* Doesn't support session id resume on subscribe */ |
|
|
|
|
bool no_params; /* Doesn't want any parameters on subscribe */ |
|
|
|
|
}; |
|
|
|
|
|
|
|
|
|
typedef struct proxy_instance proxy_instance_t; |
|
|
|
|
|
|
|
|
|
static int gen_loop(proc_instance_t *pi, connsock_t *cs) |
|
|
|
|
{ |
|
|
|
|
unixsock_t *us = &pi->us; |
|
|
|
@ -101,17 +125,226 @@ out:
|
|
|
|
|
return ret; |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
static int server_mode(ckpool_t *ckp, proc_instance_t *pi, connsock_t *cs, |
|
|
|
|
const char *auth, const char *pass) |
|
|
|
|
{ |
|
|
|
|
char *userpass = NULL; |
|
|
|
|
gbtbase_t gbt; |
|
|
|
|
int ret = 1; |
|
|
|
|
|
|
|
|
|
memset(&gbt, 0, sizeof(gbt)); |
|
|
|
|
|
|
|
|
|
userpass = strdup(auth); |
|
|
|
|
realloc_strcat(&userpass, ":"); |
|
|
|
|
realloc_strcat(&userpass, pass); |
|
|
|
|
cs->auth = http_base64(userpass); |
|
|
|
|
if (!cs->auth) { |
|
|
|
|
LOGWARNING("Failed to create base64 auth from %s", userpass); |
|
|
|
|
goto out; |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
cs->fd = connect_socket(cs->url, cs->port); |
|
|
|
|
if (cs->fd < 0) { |
|
|
|
|
LOGWARNING("FATAL: Failed to connect socket to %s:%s !", cs->url, cs->port); |
|
|
|
|
goto out; |
|
|
|
|
} |
|
|
|
|
keep_sockalive(cs->fd); |
|
|
|
|
/* Test we can connect, authorise and get a block template */ |
|
|
|
|
if (!gen_gbtbase(cs, &gbt)) { |
|
|
|
|
LOGWARNING("FATAL: Failed to get test block template from %s:%s auth %s !", |
|
|
|
|
cs->url, cs->port, userpass); |
|
|
|
|
goto out; |
|
|
|
|
} |
|
|
|
|
clear_gbtbase(&gbt); |
|
|
|
|
if (!validate_address(cs, ckp->btcaddress)) { |
|
|
|
|
LOGWARNING("FATAL: Invalid btcaddress: %s !", ckp->btcaddress); |
|
|
|
|
goto out; |
|
|
|
|
} |
|
|
|
|
ret = gen_loop(pi, cs); |
|
|
|
|
out: |
|
|
|
|
close(cs->fd); |
|
|
|
|
dealloc(userpass); |
|
|
|
|
return ret; |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
static bool send_json_msg(connsock_t *cs, json_t *json_msg) |
|
|
|
|
{ |
|
|
|
|
int len, sent; |
|
|
|
|
char *s; |
|
|
|
|
|
|
|
|
|
s = json_dumps(json_msg, JSON_ESCAPE_SLASH); |
|
|
|
|
realloc_strcat(&s, "\n"); |
|
|
|
|
len = strlen(s); |
|
|
|
|
sent = write_socket(cs->fd, s, len); |
|
|
|
|
dealloc(s); |
|
|
|
|
if (sent != len) { |
|
|
|
|
LOGWARNING("Failed to send %d bytes in send_json_msg", len); |
|
|
|
|
return false; |
|
|
|
|
} |
|
|
|
|
return true; |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
static bool connect_proxy(connsock_t *cs) |
|
|
|
|
{ |
|
|
|
|
cs->fd = connect_socket(cs->url, cs->port); |
|
|
|
|
if (cs->fd < 0) { |
|
|
|
|
LOGWARNING("Failed to connect socket to %s:%s in connect_proxy", |
|
|
|
|
cs->url, cs->port); |
|
|
|
|
return false; |
|
|
|
|
} |
|
|
|
|
keep_sockalive(cs->fd); |
|
|
|
|
return true; |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
static bool parse_subscribe(connsock_t *cs, proxy_instance_t *proxi) |
|
|
|
|
{ |
|
|
|
|
bool ret = false; |
|
|
|
|
int size; |
|
|
|
|
|
|
|
|
|
size = read_socket_line(cs); |
|
|
|
|
if (size < 1) { |
|
|
|
|
LOGWARNING("Failed to receive line in parse_subscribe"); |
|
|
|
|
goto out; |
|
|
|
|
} |
|
|
|
|
LOGWARNING("Got message: %s", cs->buf); |
|
|
|
|
ret = true; |
|
|
|
|
|
|
|
|
|
out: |
|
|
|
|
return ret; |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
static bool subscribe_stratum(connsock_t *cs, proxy_instance_t *proxi) |
|
|
|
|
{ |
|
|
|
|
json_t *req; |
|
|
|
|
bool ret = false; |
|
|
|
|
|
|
|
|
|
retry: |
|
|
|
|
/* Attempt to reconnect if the pool supports resuming */ |
|
|
|
|
if (proxi->sessionid) { |
|
|
|
|
req = json_pack("{s:i,s:s,s:[s,s]}", |
|
|
|
|
"id", proxi->id++, |
|
|
|
|
"method", "mining.subscribe", |
|
|
|
|
"params", "ckproxy", proxi->sessionid); |
|
|
|
|
/* Then attempt to connect with just the client description */ |
|
|
|
|
} else if (!proxi->no_params) { |
|
|
|
|
req = json_pack("{s:i,s:s,s:[s]}", |
|
|
|
|
"id", proxi->id++, |
|
|
|
|
"method", "mining.subscribe", |
|
|
|
|
"params", "ckproxy"); |
|
|
|
|
/* Then try without any parameters */ |
|
|
|
|
} else { |
|
|
|
|
req = json_pack("{s:i,s:s,s:[]}", |
|
|
|
|
"id", proxi->id++, |
|
|
|
|
"method", "mining.subscribe", |
|
|
|
|
"params"); |
|
|
|
|
} |
|
|
|
|
ret = send_json_msg(cs, req); |
|
|
|
|
json_decref(req); |
|
|
|
|
if (!ret) { |
|
|
|
|
LOGWARNING("Failed to send message in subscribe_stratum"); |
|
|
|
|
close(cs->fd); |
|
|
|
|
goto out; |
|
|
|
|
} |
|
|
|
|
ret = parse_subscribe(cs, proxi); |
|
|
|
|
if (ret) |
|
|
|
|
goto out; |
|
|
|
|
|
|
|
|
|
close(cs->fd); |
|
|
|
|
if (proxi->no_params) { |
|
|
|
|
LOGWARNING("Failed all subscription options in subscribe_stratum"); |
|
|
|
|
goto out; |
|
|
|
|
} |
|
|
|
|
if (proxi->sessionid) { |
|
|
|
|
LOGNOTICE("Failed sessionid reconnect in subscribe_stratum, retrying without"); |
|
|
|
|
proxi->no_sessionid = true; |
|
|
|
|
dealloc(proxi->sessionid); |
|
|
|
|
} else { |
|
|
|
|
LOGNOTICE("Failed connecting with parameters in subscribe_stratum, retrying without"); |
|
|
|
|
proxi->no_params = true; |
|
|
|
|
} |
|
|
|
|
ret = connect_proxy(cs); |
|
|
|
|
if (!ret) { |
|
|
|
|
LOGWARNING("Failed to reconnect in subscribe_stratum"); |
|
|
|
|
goto out; |
|
|
|
|
} |
|
|
|
|
goto retry; |
|
|
|
|
|
|
|
|
|
out: |
|
|
|
|
return ret; |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
static bool auth_stratum(connsock_t *cs, proxy_instance_t *proxi, const char *auth, |
|
|
|
|
const char *pass) |
|
|
|
|
{ |
|
|
|
|
json_t *req; |
|
|
|
|
bool ret; |
|
|
|
|
|
|
|
|
|
req = json_pack("{s:i,s:s,s:[s,s]}", |
|
|
|
|
"id", proxi->id++, |
|
|
|
|
"method", "mining.authorize", |
|
|
|
|
"params", auth, pass); |
|
|
|
|
ret = send_json_msg(cs, req); |
|
|
|
|
json_decref(req); |
|
|
|
|
if (!ret) { |
|
|
|
|
LOGWARNING("Failed to send message in auth_stratum"); |
|
|
|
|
close(cs->fd); |
|
|
|
|
goto out; |
|
|
|
|
} |
|
|
|
|
out: |
|
|
|
|
return ret; |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
static int proxy_loop(proc_instance_t *pi, connsock_t *cs) |
|
|
|
|
{ |
|
|
|
|
return 0; |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
static int proxy_mode(ckpool_t *ckp, proc_instance_t *pi, connsock_t *cs, |
|
|
|
|
const char *auth, const char *pass) |
|
|
|
|
{ |
|
|
|
|
proxy_instance_t proxi; |
|
|
|
|
int ret; |
|
|
|
|
|
|
|
|
|
memset(&proxi, 0, sizeof(proxi)); |
|
|
|
|
if (!connect_proxy(cs)) { |
|
|
|
|
LOGWARNING("FATAL: Failed to connect to %s:%s in proxy_mode!", |
|
|
|
|
cs->url, cs->port); |
|
|
|
|
goto out; |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
/* Test we can connect, authorise and get stratum information */ |
|
|
|
|
if (!subscribe_stratum(cs, &proxi)) { |
|
|
|
|
LOGWARNING("FATAL: Failed initial subscribe to %s:%s !", |
|
|
|
|
cs->url, cs->port); |
|
|
|
|
goto out; |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
if (!auth_stratum(cs, &proxi, auth, pass)) { |
|
|
|
|
LOGWARNING("FATAL: Failed initial authorise to %s:%s with %s:%s !", |
|
|
|
|
cs->url, cs->port, auth, pass); |
|
|
|
|
goto out; |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
ret = proxy_loop(pi, cs); |
|
|
|
|
out: |
|
|
|
|
close(cs->fd); |
|
|
|
|
free(proxi.enonce1); |
|
|
|
|
free(proxi.enonce1bin); |
|
|
|
|
free(proxi.sessionid); |
|
|
|
|
|
|
|
|
|
return ret; |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* FIXME: Hard wired to just use config 0 for now */ |
|
|
|
|
int generator(proc_instance_t *pi) |
|
|
|
|
{ |
|
|
|
|
char *url, *auth, *pass, *userpass = NULL; |
|
|
|
|
ckpool_t *ckp = pi->ckp; |
|
|
|
|
gbtbase_t gbt; |
|
|
|
|
connsock_t cs; |
|
|
|
|
int ret = 1; |
|
|
|
|
|
|
|
|
|
memset(&cs, 0, sizeof(cs)); |
|
|
|
|
memset(&gbt, 0, sizeof(gbt)); |
|
|
|
|
|
|
|
|
|
if (!ckp->proxy) { |
|
|
|
|
url = ckp->btcdurl[0]; |
|
|
|
@ -126,35 +359,11 @@ int generator(proc_instance_t *pi)
|
|
|
|
|
LOGWARNING("Failed to extract address from %s", url); |
|
|
|
|
goto out; |
|
|
|
|
} |
|
|
|
|
userpass = strdup(auth); |
|
|
|
|
realloc_strcat(&userpass, ":"); |
|
|
|
|
realloc_strcat(&userpass, pass); |
|
|
|
|
cs.auth = http_base64(userpass); |
|
|
|
|
if (!cs.auth) { |
|
|
|
|
LOGWARNING("Failed to create base64 auth from %s", userpass); |
|
|
|
|
goto out; |
|
|
|
|
} |
|
|
|
|
dealloc(userpass); |
|
|
|
|
cs.fd = connect_socket(cs.url, cs.port); |
|
|
|
|
if (cs.fd < 0) { |
|
|
|
|
LOGWARNING("Failed to connect socket to %s:%s", cs.url, cs.port); |
|
|
|
|
goto out; |
|
|
|
|
} |
|
|
|
|
keep_sockalive(cs.fd); |
|
|
|
|
|
|
|
|
|
/* Test we can connect, authorise and get a block template */ |
|
|
|
|
if (!gen_gbtbase(&cs, &gbt)) { |
|
|
|
|
LOGWARNING("Failed to get test block template from %s:%s auth %s", |
|
|
|
|
cs.url, cs.port, userpass); |
|
|
|
|
goto out; |
|
|
|
|
} |
|
|
|
|
clear_gbtbase(&gbt); |
|
|
|
|
if (!validate_address(&cs, ckp->btcaddress)) { |
|
|
|
|
LOGWARNING("Invalid btcaddress: %s, unable to start!", ckp->btcaddress); |
|
|
|
|
goto out; |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
ret = gen_loop(pi, &cs); |
|
|
|
|
if (!ckp->proxy) |
|
|
|
|
ret = server_mode(ckp, pi, &cs, auth, pass); |
|
|
|
|
else |
|
|
|
|
ret = proxy_mode(ckp, pi, &cs, auth, pass); |
|
|
|
|
out: |
|
|
|
|
/* Clean up here */ |
|
|
|
|
dealloc(cs.url); |
|
|
|
|