|  |  | @ -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); | 
			
		
	
	
		
		
			
				
					|  |  | 
 |