Browse Source

Fix mining notify json and add basic authorise mechanism

master
Con Kolivas 11 years ago
parent
commit
db5aece42a
  1. 66
      src/stratifier.c

66
src/stratifier.c

@ -118,7 +118,9 @@ struct stratum_instance {
char enonce1[20]; char enonce1[20];
double diff; double diff;
bool authorised; bool authorised;
char *name; char *useragent;
char *workername;
int user_id;
}; };
typedef struct stratum_instance stratum_instance_t; typedef struct stratum_instance stratum_instance_t;
@ -465,13 +467,15 @@ static json_t *parse_subscribe(int client_id, json_t *params_val)
LOGERR("Failed to find client id %d in hashtable!", client_id); LOGERR("Failed to find client id %d in hashtable!", client_id);
return NULL; return NULL;
} }
ck_runlock(&instance_lock);
arr_size = json_array_size(params_val); arr_size = json_array_size(params_val);
if (arr_size > 0) { if (arr_size > 0) {
const char *buf; const char *buf;
buf = json_string_value(json_array_get(params_val, 0)); buf = json_string_value(json_array_get(params_val, 0));
if (buf && strlen(buf)) if (buf && strlen(buf))
client->name = strdup(buf); client->useragent = strdup(buf);
if (arr_size > 1) { if (arr_size > 1) {
/* This would be the session id for reconnect */ /* This would be the session id for reconnect */
buf = json_string_value(json_array_get(params_val, 1)); buf = json_string_value(json_array_get(params_val, 1));
@ -480,14 +484,13 @@ static json_t *parse_subscribe(int client_id, json_t *params_val)
} }
} }
enonce1 = strdup(client->enonce1); enonce1 = strdup(client->enonce1);
ck_runlock(&instance_lock);
ck_rlock(&workbase_lock); ck_rlock(&workbase_lock);
if (likely(workbases)) if (likely(workbases))
n2len = workbases->enonce2varlen; n2len = workbases->enonce2varlen;
else else
n2len = 8; n2len = 8;
ret = json_pack("[[s,s],s,i]", "mining.notify", enonce1, enonce1, n2len); ret = json_pack("[[[s,s]],s,i]", "mining.notify", enonce1, enonce1, n2len);
ck_runlock(&workbase_lock); ck_runlock(&workbase_lock);
free(enonce1); free(enonce1);
@ -495,9 +498,53 @@ static json_t *parse_subscribe(int client_id, json_t *params_val)
return ret; return ret;
} }
static int authorise_user(const char __maybe_unused *workername)
{
/* Talk to database here and return user_id or -1 if invalid */
return 1;
}
static json_t *parse_authorize(stratum_instance_t *client, json_t *params_val, json_t **err_val)
{
int arr_size, user_id;
bool ret = false;
const char *buf;
if (unlikely(!json_is_array(params_val))) {
*err_val = json_string("params not an array");
goto out;
}
arr_size = json_array_size(params_val);
if (unlikely(arr_size < 1)) {
*err_val = json_string("params missing array entries");
goto out;
}
buf = json_string_value(json_array_get(params_val, 0));
if (!buf) {
*err_val = json_string("Invalid workername parameter");
goto out;
}
if (!strlen(buf)) {
*err_val = json_string("Empty workername parameter");
goto out;
}
user_id = authorise_user(buf);
if (user_id < 0) {
*err_val = json_string("User not found");
goto out;
}
client->workername = strdup(buf);
client->user_id = user_id;
client->authorised = true;
ret = true;
out:
return json_boolean(ret);
}
/* We should have already determined all the values passed to this are valid /* We should have already determined all the values passed to this are valid
* by now. Set update if we should also send the latest stratum parameters */ * by now. Set update if we should also send the latest stratum parameters */
static json_t *gen_json_result(int client_id, json_t *method_val, json_t *params_val, bool *update) static json_t *gen_json_result(int client_id, json_t *method_val, json_t *params_val,
json_t **err_val, bool *update)
{ {
stratum_instance_t *client = NULL; stratum_instance_t *client = NULL;
const char *method; const char *method;
@ -508,7 +555,6 @@ static json_t *gen_json_result(int client_id, json_t *method_val, json_t *params
return parse_subscribe(client_id, params_val); return parse_subscribe(client_id, params_val);
} }
/* We should only accept authorised requests from here on */
ck_rlock(&instance_lock); ck_rlock(&instance_lock);
client = __instance_by_id(client_id); client = __instance_by_id(client_id);
if (unlikely(!client)) { if (unlikely(!client)) {
@ -518,6 +564,10 @@ static json_t *gen_json_result(int client_id, json_t *method_val, json_t *params
} }
ck_runlock(&instance_lock); ck_runlock(&instance_lock);
if (!strncasecmp(method, "mining.authorize", 16))
return parse_authorize(client, params_val, err_val);
/* We should only accept authorised requests from here on */
if (!client->authorised) if (!client->authorised)
return json_string("Unauthorised"); return json_string("Unauthorised");
@ -533,6 +583,7 @@ static void stratum_send_update(int client_id, bool clean)
val = json_pack("{s:[sss[o]sssb],s:o,s:s}", val = json_pack("{s:[sss[o]sssb],s:o,s:s}",
"params", "params",
current_workbase->idstring, current_workbase->idstring,
current_workbase->prevhash,
current_workbase->coinb1, current_workbase->coinb1,
current_workbase->coinb2, current_workbase->coinb2,
json_copy(current_workbase->merkle_array), json_copy(current_workbase->merkle_array),
@ -578,8 +629,9 @@ static void parse_instance_msg(int client_id, json_t *msg)
err_val = json_string("-1:params not found"); err_val = json_string("-1:params not found");
goto out; goto out;
} }
result_val = gen_json_result(client_id, method, params, &err_val, &update);
if (!err_val)
err_val = json_null(); err_val = json_null();
result_val = gen_json_result(client_id, method, params, &update);
out: out:
json_object_set_nocheck(json_msg, "id", id_val); json_object_set_nocheck(json_msg, "id", id_val);
json_object_set_nocheck(json_msg, "error", err_val); json_object_set_nocheck(json_msg, "error", err_val);

Loading…
Cancel
Save