From 966b9b7326e0a86d4e97329d5c707f6cb1a9c9f8 Mon Sep 17 00:00:00 2001 From: kanoi Date: Fri, 13 Mar 2015 15:52:08 +1100 Subject: [PATCH 1/5] ckdb - pass all worker diffs back to auth requests --- src/ckdb.h | 3 +- src/ckdb_cmd.c | 98 ++++++++++++++++++++++++++++++++++++++----------- src/ckdb_data.c | 20 ++++++++++ 3 files changed, 98 insertions(+), 23 deletions(-) diff --git a/src/ckdb.h b/src/ckdb.h index ecddeebc..830a7e9d 100644 --- a/src/ckdb.h +++ b/src/ckdb.h @@ -55,7 +55,7 @@ #define DB_VLOCK "1" #define DB_VERSION "1.0.0" -#define CKDB_VERSION DB_VERSION"-1.023" +#define CKDB_VERSION DB_VERSION"-1.030" #define WHERE_FFL " - from %s %s() line %d" #define WHERE_FFL_HERE __FILE__, __func__, __LINE__ @@ -1726,6 +1726,7 @@ extern cmp_t cmp_useratts(K_ITEM *a, K_ITEM *b); extern K_ITEM *find_useratts(int64_t userid, char *attname); extern cmp_t cmp_workers(K_ITEM *a, K_ITEM *b); extern K_ITEM *find_workers(int64_t userid, char *workername); +extern K_ITEM *first_workers(int64_t userid, K_TREE_CTX *ctx); extern K_ITEM *new_worker(PGconn *conn, bool update, int64_t userid, char *workername, char *diffdef, char *idlenotificationenabled, char *idlenotificationtime, char *by, diff --git a/src/ckdb_cmd.c b/src/ckdb_cmd.c index 80b1f9d1..3e73553c 100644 --- a/src/ckdb_cmd.c +++ b/src/ckdb_cmd.c @@ -2308,15 +2308,18 @@ static char *cmd_auth_do(PGconn *conn, char *cmd, char *id, char *by, char *code, char *inet, tv_t *cd, K_TREE *trf_root) { + K_TREE_CTX ctx[1]; char reply[1024] = ""; size_t siz = sizeof(reply); K_ITEM *i_poolinstance, *i_username, *i_workername, *i_clientid; - K_ITEM *i_enonce1, *i_useragent, *i_preauth, *u_item, *oc_item; + K_ITEM *i_enonce1, *i_useragent, *i_preauth, *u_item, *oc_item, *w_item; USERS *users = NULL; char *username; WORKERS *workers = NULL; OPTIONCONTROL *optioncontrol; - bool ok; + size_t len, off; + char *buf; + bool ok, first; LOGDEBUG("%s(): cmd '%s'", __func__, cmd); @@ -2378,18 +2381,42 @@ static char *cmd_auth_do(PGconn *conn, char *cmd, char *id, char *by, if (!ok) { LOGDEBUG("%s() %s.failed.DBE", __func__, id); return strdup("failed.DBE"); - } else { - // Only flag a successful auth - ck_wlock(&last_lock); - setnow(&last_auth); - ck_wunlock(&last_lock); } + + // Only flag a successful auth + ck_wlock(&last_lock); + setnow(&last_auth); + ck_wunlock(&last_lock); + + APPEND_REALLOC_INIT(buf, off, len); snprintf(reply, siz, "ok.authorise={\"secondaryuserid\":\"%s\"," - "\"difficultydefault\":%d}", - users->secondaryuserid, workers->difficultydefault); - LOGDEBUG("%s.%s", id, reply); - return strdup(reply); + "\"workers\":[", + users->secondaryuserid); + APPEND_REALLOC(buf, off, len, reply); + first = true; + K_RLOCK(workers_free); + w_item = first_workers(users->userid, ctx); + DATA_WORKERS_NULL(workers, w_item); + while (w_item && workers->userid == users->userid) { + if (CURRENT(&(workers->expirydate))) { + snprintf(reply, siz, + "%s{\"workername\":\"%s\"," + "\"difficultydefault\":%"PRId32"}", + first ? EMPTY : ",", + workers->workername, + workers->difficultydefault); + APPEND_REALLOC(buf, off, len, reply); + first = false; + } + w_item = next_in_ktree(ctx); + DATA_WORKERS_NULL(workers, w_item); + } + K_RUNLOCK(workers_free); + APPEND_REALLOC(buf, off, len, "]}"); + + LOGDEBUG("%s.%s", id, buf); + return buf; } static char *cmd_auth(PGconn *conn, char *cmd, char *id, @@ -2404,13 +2431,16 @@ static char *cmd_addrauth_do(PGconn *conn, char *cmd, char *id, char *by, char *code, char *inet, tv_t *cd, K_TREE *trf_root) { + K_TREE_CTX ctx[1]; char reply[1024] = ""; size_t siz = sizeof(reply); K_ITEM *i_poolinstance, *i_username, *i_workername, *i_clientid; - K_ITEM *i_enonce1, *i_useragent, *i_preauth; + K_ITEM *i_enonce1, *i_useragent, *i_preauth, *w_item; USERS *users = NULL; WORKERS *workers = NULL; - bool ok; + size_t len, off; + char *buf; + bool ok, first; LOGDEBUG("%s(): cmd '%s'", __func__, cmd); @@ -2456,18 +2486,42 @@ static char *cmd_addrauth_do(PGconn *conn, char *cmd, char *id, char *by, if (!ok) { LOGDEBUG("%s() %s.failed.DBE", __func__, id); return strdup("failed.DBE"); - } else { - // Only flag a successful auth - ck_wlock(&last_lock); - setnow(&last_auth); - ck_wunlock(&last_lock); } + + // Only flag a successful auth + ck_wlock(&last_lock); + setnow(&last_auth); + ck_wunlock(&last_lock); + + APPEND_REALLOC_INIT(buf, off, len); snprintf(reply, siz, "ok.addrauth={\"secondaryuserid\":\"%s\"," - "\"difficultydefault\":%d}", - users->secondaryuserid, workers->difficultydefault); - LOGDEBUG("%s.%s", id, reply); - return strdup(reply); + "\"workers\":[", + users->secondaryuserid); + APPEND_REALLOC(buf, off, len, reply); + first = true; + K_RLOCK(workers_free); + w_item = first_workers(users->userid, ctx); + DATA_WORKERS_NULL(workers, w_item); + while (w_item && workers->userid == users->userid) { + if (CURRENT(&(workers->expirydate))) { + snprintf(reply, siz, + "%s{\"workername\":\"%s\"," + "\"difficultydefault\":%"PRId32"}", + first ? EMPTY : ",", + workers->workername, + workers->difficultydefault); + APPEND_REALLOC(buf, off, len, reply); + first = false; + } + w_item = next_in_ktree(ctx); + DATA_WORKERS_NULL(workers, w_item); + } + K_RUNLOCK(workers_free); + APPEND_REALLOC(buf, off, len, "]}"); + + LOGDEBUG("%s.%s", id, buf); + return buf; } static char *cmd_addrauth(PGconn *conn, char *cmd, char *id, diff --git a/src/ckdb_data.c b/src/ckdb_data.c index a4227e7e..f1303265 100644 --- a/src/ckdb_data.c +++ b/src/ckdb_data.c @@ -1113,6 +1113,26 @@ K_ITEM *find_workers(int64_t userid, char *workername) return find_in_ktree(workers_root, &look, cmp_workers, ctx); } +K_ITEM *first_workers(int64_t userid, K_TREE_CTX *ctx) +{ + WORKERS workers; + K_TREE_CTX ctx0[1]; + K_ITEM look; + + if (ctx == NULL) + ctx = ctx0; + + workers.userid = userid; + workers.workername[0] = '\0'; + workers.expirydate.tv_sec = 0L; + workers.expirydate.tv_usec = 0L; + + INIT_WORKERS(&look); + look.data = (void *)(&workers); + // Caller needs to check userid/expirydate if the result != NULL + return find_after_in_ktree(workers_root, &look, cmp_workers, ctx); +} + K_ITEM *new_worker(PGconn *conn, bool update, int64_t userid, char *workername, char *diffdef, char *idlenotificationenabled, char *idlenotificationtime, char *by, From 4f327ee6eded45e7beb3c24f0212fcf0992d8de1 Mon Sep 17 00:00:00 2001 From: kanoi Date: Fri, 13 Mar 2015 20:58:20 +1100 Subject: [PATCH 2/5] ckdb - switch_state to allow code upgrades while running --- src/ckdb.c | 2 ++ src/ckdb.h | 23 ++++++++++++++++++++++- src/ckdb_cmd.c | 18 ++++++++++++++++++ src/ckdb_dbio.c | 24 +++++++++++++++++++++++- 4 files changed, 65 insertions(+), 2 deletions(-) diff --git a/src/ckdb.c b/src/ckdb.c index ac39b470..ecf4af24 100644 --- a/src/ckdb.c +++ b/src/ckdb.c @@ -151,6 +151,8 @@ static char *restorefrom; // Only accessed in here static bool markersummary_auto; +int switch_state = SWITCH_STATE_ALL; + // disallow: '/' '.' '_' and FLDSEP const char *userpatt = "^[^/\\._"FLDSEPSTR"]*$"; const char *mailpatt = "^[A-Za-z0-9_-][A-Za-z0-9_\\.-]*@[A-Za-z0-9][A-Za-z0-9\\.-]*[A-Za-z0-9]$"; diff --git a/src/ckdb.h b/src/ckdb.h index 830a7e9d..6893ecfa 100644 --- a/src/ckdb.h +++ b/src/ckdb.h @@ -55,7 +55,7 @@ #define DB_VLOCK "1" #define DB_VERSION "1.0.0" -#define CKDB_VERSION DB_VERSION"-1.030" +#define CKDB_VERSION DB_VERSION"-1.031" #define WHERE_FFL " - from %s %s() line %d" #define WHERE_FFL_HERE __FILE__, __func__, __LINE__ @@ -81,6 +81,27 @@ #define TRUE_CHR 'Y' #define FALSE_CHR 'N' +/* Set by cmd_setopts() and used by whatever code needs it + * It's loaded during startup but set to SWITCH_STATE_ALL if it's missing, + * meaning all switches are active + * The idea is that if you need to manually switch code over from one version + * up to the next version at an indeterminate time then you can do that by + * coding a switch_state test and then externally switch the code over via + * the cmd_setopts() socket interface + * It's not for coding runtime options into the code since that can be done + * using optioncontrol directly + * It's stored in optioncontrol so that it's value is permanent and + * activationdate and activationheight have their values overridden to + * disable using them + * N.B. optioncontrol_item_add() intercepts the change by name and updates + * switch_state but ONLY if the DB update succeeds */ +extern int switch_state; +#define SWITCH_STATE_NAME "SwitchState" +/* Each switch state must be higher than all previous + * so that future states don't undo old changes */ +#define SWITCH_STATE_AUTHWORKERS 1 +#define SWITCH_STATE_ALL 666666 + extern char *EMPTY; extern const char *userpatt; diff --git a/src/ckdb_cmd.c b/src/ckdb_cmd.c index 3e73553c..93c42a69 100644 --- a/src/ckdb_cmd.c +++ b/src/ckdb_cmd.c @@ -2388,6 +2388,15 @@ static char *cmd_auth_do(PGconn *conn, char *cmd, char *id, char *by, setnow(&last_auth); ck_wunlock(&last_lock); + if (switch_state < SWITCH_STATE_AUTHWORKERS) { + snprintf(reply, siz, + "ok.authorise={\"secondaryuserid\":\"%s\"," + "\"difficultydefault\":%d}", + users->secondaryuserid, workers->difficultydefault); + LOGDEBUG("%s.%s", id, reply); + return strdup(reply); + } + APPEND_REALLOC_INIT(buf, off, len); snprintf(reply, siz, "ok.authorise={\"secondaryuserid\":\"%s\"," @@ -2493,6 +2502,15 @@ static char *cmd_addrauth_do(PGconn *conn, char *cmd, char *id, char *by, setnow(&last_auth); ck_wunlock(&last_lock); + if (switch_state < SWITCH_STATE_AUTHWORKERS) { + snprintf(reply, siz, + "ok.addrauth={\"secondaryuserid\":\"%s\"," + "\"difficultydefault\":%d}", + users->secondaryuserid, workers->difficultydefault); + LOGDEBUG("%s.%s", id, reply); + return strdup(reply); + } + APPEND_REALLOC_INIT(buf, off, len); snprintf(reply, siz, "ok.addrauth={\"secondaryuserid\":\"%s\"," diff --git a/src/ckdb_dbio.c b/src/ckdb_dbio.c index 91e43fd8..5b3c1544 100644 --- a/src/ckdb_dbio.c +++ b/src/ckdb_dbio.c @@ -2183,6 +2183,13 @@ K_ITEM *optioncontrol_item_add(PGconn *conn, K_ITEM *oc_item, tv_t *cd, bool beg DATA_OPTIONCONTROL(row, oc_item); + // Enforce the rule that switch_state isn't date/height controlled + if (strcmp(row->optionname, SWITCH_STATE_NAME) == 0) { + row->activationdate.tv_sec = date_begin.tv_sec; + row->activationdate.tv_usec = date_begin.tv_usec; + row->activationheight = OPTIONCONTROL_HEIGHT; + } + INIT_OPTIONCONTROL(&look); look.data = (void *)row; K_RLOCK(optioncontrol_free); @@ -2282,6 +2289,11 @@ nostart: } optioncontrol_root = add_to_ktree(optioncontrol_root, oc_item, cmp_optioncontrol); k_add_head(optioncontrol_store, oc_item); + if (strcmp(row->optionname, SWITCH_STATE_NAME) == 0) { + switch_state = atoi(row->optionvalue); + LOGWARNING("%s() set switch_state to %d", + __func__, switch_state); + } } K_WUNLOCK(optioncontrol_free); @@ -2320,7 +2332,7 @@ K_ITEM *optioncontrol_add(PGconn *conn, char *optionname, char *optionvalue, TXT_TO_INT("activationheight", activationheight, row->activationheight); } else - row->activationheight = 1; + row->activationheight = OPTIONCONTROL_HEIGHT; HISTORYDATEINIT(row, cd, by, code, inet); HISTORYDATETRANSFER(trf_root, row); @@ -2406,6 +2418,14 @@ bool optioncontrol_fill(PGconn *conn) optioncontrol_root = add_to_ktree(optioncontrol_root, item, cmp_optioncontrol); k_add_head(optioncontrol_store, item); + + // There should only be one CURRENT version of switch_state + if (CURRENT(&(row->expirydate)) && + strcmp(row->optionname, SWITCH_STATE_NAME) == 0) { + switch_state = atoi(row->optionvalue); + LOGWARNING("%s() set switch_state to %d", + __func__, switch_state); + } } if (!ok) { FREENULL(row->optionvalue); @@ -2418,6 +2438,8 @@ bool optioncontrol_fill(PGconn *conn) if (ok) { LOGDEBUG("%s(): built", __func__); LOGWARNING("%s(): loaded %d optioncontrol records", __func__, n); + LOGWARNING("%s() switch_state initially %d", + __func__, switch_state); } return ok; From f2d80c7e959fef2450f06400ff051527b3f5204d Mon Sep 17 00:00:00 2001 From: kanoi Date: Fri, 13 Mar 2015 22:39:06 +1100 Subject: [PATCH 3/5] php - more constructive ckdb restart message --- pool/base.php | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/pool/base.php b/pool/base.php index a109a23c..66eb014d 100644 --- a/pool/base.php +++ b/pool/base.php @@ -259,7 +259,7 @@ function safetext($txt, $len = 1024) # function dbd($data, $user) { - return "
Web site is currently down
"; + return "
Database is reloading, mining is all OK
"; } # function dbdown() From 8091b77c7ce363b4116d2e73aa6db15e8aee1b2a Mon Sep 17 00:00:00 2001 From: Con Kolivas Date: Fri, 13 Mar 2015 22:46:41 +1100 Subject: [PATCH 4/5] Allow arbitrary length responses from ckdb in send_recv_auth to allow for hundreds of workers --- src/stratifier.c | 7 +++++-- 1 file changed, 5 insertions(+), 2 deletions(-) diff --git a/src/stratifier.c b/src/stratifier.c index 2f38a53d..33dafa07 100644 --- a/src/stratifier.c +++ b/src/stratifier.c @@ -2390,13 +2390,16 @@ static int send_recv_auth(stratum_instance_t *client) free(json_msg); if (likely(buf)) { + char *cmd = NULL, *secondaryuserid = NULL, *response; worker_instance_t *worker = client->worker_instance; - char *cmd = NULL, *secondaryuserid = NULL; - char response[PAGESIZE] = {}; json_error_t err_val; + size_t responselen; json_t *val = NULL; LOGINFO("Got ckdb response: %s", buf); + responselen = strlen(buf); /* Leave ample room for response based on buf length */ + response = alloca(responselen); + memset(response, 0, responselen); if (unlikely(sscanf(buf, "id.%*d.%s", response) < 1 || strlen(response) < 1 || !strchr(response, '='))) { if (cmdmatch(response, "failed")) goto out; From b780dc006153b4a6c6672a71846c8bc92192a42e Mon Sep 17 00:00:00 2001 From: Con Kolivas Date: Sat, 14 Mar 2015 10:56:11 +1100 Subject: [PATCH 5/5] Demote message verbosity for no auth response when we know ckdb is offline --- src/stratifier.c | 8 ++++++-- 1 file changed, 6 insertions(+), 2 deletions(-) diff --git a/src/stratifier.c b/src/stratifier.c index 33dafa07..0c732c72 100644 --- a/src/stratifier.c +++ b/src/stratifier.c @@ -2437,8 +2437,12 @@ static int send_recv_auth(stratum_instance_t *client) } if (contended) LOGWARNING("Prolonged lock contention for ckdb while trying to authorise"); - else - LOGWARNING("Got no auth response from ckdb :("); + else { + if (!sdata->ckdb_offline) + LOGWARNING("Got no auth response from ckdb :("); + else + LOGNOTICE("No auth response for %s from offline ckdb", user->username); + } out_fail: ret = -1; out: