mirror of
https://github.com/mimblewimble/grin.git
synced 2025-01-22 03:51:08 +03:00
86d7628562
* add request method name to the result to allow matching onthe client side * update tests for change in rpc response api
163 lines
5.6 KiB
Rust
163 lines
5.6 KiB
Rust
// Copyright 2018 The Grin Developers
|
|
//
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
// you may not use this file except in compliance with the License.
|
|
// You may obtain a copy of the License at
|
|
//
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
//
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
// See the License for the specific language governing permissions and
|
|
// limitations under the License.
|
|
|
|
extern crate grin_api as api;
|
|
extern crate grin_chain as chain;
|
|
extern crate grin_core as core;
|
|
extern crate grin_servers as servers;
|
|
extern crate grin_p2p as p2p;
|
|
extern crate grin_pow as pow;
|
|
extern crate grin_util as util;
|
|
extern crate grin_wallet as wallet;
|
|
|
|
extern crate bufstream;
|
|
extern crate serde_json;
|
|
|
|
mod framework;
|
|
|
|
use std::io::prelude::*;
|
|
use std::net::TcpStream;
|
|
use bufstream::BufStream;
|
|
use serde_json::Value;
|
|
|
|
use std::thread;
|
|
use std::time;
|
|
|
|
use core::global;
|
|
use core::global::ChainTypes;
|
|
|
|
use framework::{config, miner_config};
|
|
|
|
// Create a grin server, and a stratum server.
|
|
// Simulate a few JSONRpc requests and verify the results.
|
|
// Validate disconnected workers
|
|
// Validate broadcasting new jobs
|
|
#[test]
|
|
fn basic_stratum_server() {
|
|
util::init_test_logger();
|
|
global::set_mining_mode(ChainTypes::AutomatedTesting);
|
|
|
|
let test_name_dir = "stratum_server";
|
|
framework::clean_all_output(test_name_dir);
|
|
|
|
// Create a server
|
|
let s = servers::Server::new(config(4000, test_name_dir, 0)).unwrap();
|
|
|
|
// Get mining config with stratumserver enabled
|
|
let mut miner_cfg = miner_config();
|
|
miner_cfg.enable_mining = false;
|
|
miner_cfg.attempt_time_per_block = 999;
|
|
miner_cfg.enable_stratum_server = Some(true);
|
|
miner_cfg.stratum_server_addr = Some(String::from("127.0.0.1:11101"));
|
|
|
|
// Start stratum server
|
|
s.start_stratum_server(miner_cfg);
|
|
|
|
// Wait for stratum server to start and
|
|
// Verify stratum server accepts connections
|
|
loop {
|
|
if let Ok(_stream) = TcpStream::connect("127.0.0.1:11101") {
|
|
break;
|
|
} else {
|
|
thread::sleep(time::Duration::from_millis(500));
|
|
}
|
|
// As this stream falls out of scope it will be disconnected
|
|
}
|
|
|
|
// Create a few new worker connections
|
|
let mut workers = vec![];
|
|
for _n in 0..5 {
|
|
let w = TcpStream::connect("127.0.0.1:11101").unwrap();
|
|
w.set_nonblocking(true)
|
|
.expect("Failed to set TcpStream to non-blocking");
|
|
let stream = BufStream::new(w);
|
|
workers.push(stream);
|
|
}
|
|
assert!(workers.len() == 5);
|
|
|
|
// Simulate a worker lost connection
|
|
workers.remove(4);
|
|
|
|
// Swallow the genesis block
|
|
thread::sleep(time::Duration::from_secs(1)); // Wait for the server to broadcast
|
|
let mut response = String::new();
|
|
for n in 0..workers.len() {
|
|
let _result = workers[n].read_line(&mut response);
|
|
}
|
|
|
|
// Verify a few stratum JSONRpc commands
|
|
// getjobtemplate - expected block template result
|
|
let mut response = String::new();
|
|
let job_req = "{\"id\": \"Stratum\", \"jsonrpc\": \"2.0\", \"method\": \"getjobtemplate\"}\n";
|
|
workers[2].write(job_req.as_bytes()).unwrap();
|
|
workers[2].flush().unwrap();
|
|
thread::sleep(time::Duration::from_secs(1)); // Wait for the server to reply
|
|
match workers[2].read_line(&mut response) {
|
|
Ok(_) => {
|
|
let r: Value = serde_json::from_str(&response).unwrap();
|
|
assert_eq!(r["error"], serde_json::Value::Null);
|
|
assert_ne!(r["result"], serde_json::Value::Null);
|
|
}
|
|
Err(_e) => {
|
|
assert!(false);
|
|
}
|
|
}
|
|
|
|
// keepalive - expected "ok" result
|
|
let mut response = String::new();
|
|
let job_req = "{\"id\":\"3\",\"jsonrpc\":\"2.0\",\"method\":\"keepalive\"}\n";
|
|
let ok_resp = "{\"id\":\"3\",\"jsonrpc\":\"2.0\",\"method\":\"keepalive\",\"result\":\"ok\",\"error\":null}\n";
|
|
workers[2].write(job_req.as_bytes()).unwrap();
|
|
workers[2].flush().unwrap();
|
|
thread::sleep(time::Duration::from_secs(1)); // Wait for the server to reply
|
|
let _st = workers[2].read_line(&mut response);
|
|
assert_eq!(response.as_str(), ok_resp);
|
|
|
|
// "doesnotexist" - error expected
|
|
let mut response = String::new();
|
|
let job_req = "{\"id\":\"4\",\"jsonrpc\":\"2.0\",\"method\":\"doesnotexist\"}\n";
|
|
let ok_resp = "{\"id\":\"4\",\"jsonrpc\":\"2.0\",\"method\":\"doesnotexist\",\"result\":null,\"error\":{\"code\":-32601,\"message\":\"Method not found\"}}\n";
|
|
workers[3].write(job_req.as_bytes()).unwrap();
|
|
workers[3].flush().unwrap();
|
|
thread::sleep(time::Duration::from_secs(1)); // Wait for the server to reply
|
|
let _st = workers[3].read_line(&mut response);
|
|
assert_eq!(response.as_str(), ok_resp);
|
|
|
|
// Verify stratum server and worker stats
|
|
let stats = s.get_server_stats().unwrap();
|
|
assert_eq!(stats.stratum_stats.block_height, 1); // just 1 genesis block
|
|
assert_eq!(stats.stratum_stats.num_workers, 4); // 5 - 1 = 4
|
|
assert_eq!(stats.stratum_stats.worker_stats[5].is_connected, false); // worker was removed
|
|
assert_eq!(stats.stratum_stats.worker_stats[1].is_connected, true);
|
|
|
|
// Start mining blocks
|
|
s.start_miner(miner_config());
|
|
|
|
// Simulate a worker lost connection
|
|
workers.remove(1);
|
|
|
|
// Verify blocks are being broadcast to workers
|
|
let expected = String::from("job");
|
|
thread::sleep(time::Duration::from_secs(3)); // Wait for a few mined blocks
|
|
let mut jobtemplate = String::new();
|
|
let _st = workers[2].read_line(&mut jobtemplate);
|
|
let job_template: Value = serde_json::from_str(&jobtemplate).unwrap();
|
|
assert_eq!(job_template["method"], expected);
|
|
|
|
// Verify stratum server and worker stats
|
|
let stats = s.get_server_stats().unwrap();
|
|
assert_eq!(stats.stratum_stats.num_workers, 3); // 5 - 2 = 3
|
|
assert_eq!(stats.stratum_stats.worker_stats[2].is_connected, false); // worker was removed
|
|
assert_ne!(stats.stratum_stats.block_height, 1);
|
|
}
|