无法通过JSON-RPC将CREATERAWTRANSACTION发布到Bitcoin Core

时间:2018-12-03 22:49:42

标签: c# json json.net bitcoin json-rpc

我正在尝试通过SELECT case_staging_table.CASE_NUMBER, case_staging_table.CASE_DETAILS, attachment_staging_table.ATTACHMENT_NAME FROM case_staging_table LEFT JOIN attachment_staging_table ON case_staging_table.PARENT_ID_NUMBER = attachment_staging_table.PARENT_ID_NUMBER WHERE case_staging_table.PROCESSED_FLAG IS NULL AND case_staging_table.PARENT_ID_NUMBER not in ( select x.PARENT_ID_NUMBER from attachment_staging_table x where x.ATTACHMENT_SENT_FLAG = 'N' ) 发布到本地比特币完整节点,但是服务器出现错误。

按照此处的文档进行操作: https://bitcoincore.org/en/doc/0.17.0/rpc/rawtransactions/createrawtransaction/

我可以看到json-rpc请求的以下示例结构:

createrawtransaction

我的代码创建了以下结构,该结构似乎与bitcoincore.org中示例的结构相匹配:

{"jsonrpc": "1.0", "id":"curltest", "method": "createrawtransaction", "params": ["[{\"txid\":\"myid\",\"vout\":0}]", "[{\"address\":0.01}]"] }

但是它给出了一个错误:

{"jsonrpc":"1.0","id":"1","method":"createrawtransaction","params":["[{\"txid\":\"1a43a1f27c5837d5319a45217aa948a4d39c1d89faf497ce59de5bd570a64a26\",\"vout\":1}]","[{\"2NAZpRsvj9BstxxCDkKoe1FVjmPPxdmvqKj\":0.01}]"]}

下面是我用来发出RPC请求的方法,该方法是从此处的API参考中获得的:

https://en.bitcoin.it/wiki/API_reference_(JSON-RPC)#.NET_.28C.23.29

System.Net.WebException
  HResult=0x80131509
  Message=The remote server returned an error: (500) Internal Server Error.
  Source=RawTransactions
  StackTrace:
   at RawTransactions.Form1.RequestServer(String methodName, List`1 parameters) in C:\Users\userthree\Documents\Visual Studio 2017\Projects\RawTransactions\RawTransactions\Form1.cs:line 132
   at RawTransactions.Form1.button1_Click(Object sender, EventArgs e) in C:\Users\userthree\Documents\Visual Studio 2017\Projects\RawTransactions\RawTransactions\Form1.cs:line 77
   at System.Windows.Forms.Control.OnClick(EventArgs e)
   at System.Windows.Forms.Button.OnClick(EventArgs e)
   at System.Windows.Forms.Button.OnMouseUp(MouseEventArgs mevent)
   at System.Windows.Forms.Control.WmMouseUp(Message& m, MouseButtons button, Int32 clicks)
   at System.Windows.Forms.Control.WndProc(Message& m)
   at System.Windows.Forms.ButtonBase.WndProc(Message& m)
   at System.Windows.Forms.Button.WndProc(Message& m)
   at System.Windows.Forms.Control.ControlNativeWindow.OnMessage(Message& m)
   at System.Windows.Forms.Control.ControlNativeWindow.WndProc(Message& m)
   at System.Windows.Forms.NativeWindow.DebuggableCallback(IntPtr hWnd, Int32 msg, IntPtr wparam, IntPtr lparam)
   at System.Windows.Forms.UnsafeNativeMethods.DispatchMessageW(MSG& msg)
   at System.Windows.Forms.Application.ComponentManager.System.Windows.Forms.UnsafeNativeMethods.IMsoComponentManager.FPushMessageLoop(IntPtr dwComponentID, Int32 reason, Int32 pvLoopData)
   at System.Windows.Forms.Application.ThreadContext.RunMessageLoopInner(Int32 reason, ApplicationContext context)
   at System.Windows.Forms.Application.ThreadContext.RunMessageLoop(Int32 reason, ApplicationContext context)
   at System.Windows.Forms.Application.Run(Form mainForm)
   at RawTransactions.Program.Main() in C:\Users\userthree\Documents\Visual Studio 2017\Projects\RawTransactions\RawTransactions\Program.cs:line 19

这是我尝试使用上述方法的方法:

public static string RequestServer(string methodName, List<string> parameters)
{
    string ServerIp = "http://localhost:18332";
    string UserName = "USERNAME";
    string Password = "PASSWORD";

    HttpWebRequest webRequest = (HttpWebRequest)WebRequest.Create(ServerIp);
    webRequest.Credentials = new NetworkCredential(UserName, Password);

    webRequest.ContentType = "application/json-rpc";
    webRequest.Method = "POST";

    string respVal = string.Empty;

    JObject joe = new JObject();
    joe.Add(new JProperty("jsonrpc", "1.0"));
    joe.Add(new JProperty("id", "1"));
    joe.Add(new JProperty("method", methodName));

    JArray props = new JArray();
    foreach (var parameter in parameters)
    {
        props.Add(parameter);
    }

    joe.Add(new JProperty("params", props));

    // serialize json for the request
    string s = JsonConvert.SerializeObject(joe);
    byte[] byteArray = Encoding.UTF8.GetBytes(s);
    webRequest.ContentLength = byteArray.Length;

    Stream dataStream = webRequest.GetRequestStream();
    dataStream.Write(byteArray, 0, byteArray.Length);
    dataStream.Close();

    StreamReader streamReader = null;
    try
    {
        WebResponse webResponse = webRequest.GetResponse();

        streamReader = new StreamReader(webResponse.GetResponseStream(), true);

        respVal = streamReader.ReadToEnd();
        var data = JsonConvert.DeserializeObject(respVal).ToString();
        return data;
    }
    catch (Exception exp)
    {
        throw (exp);
    }
    finally
    {
        if (streamReader != null)
        {
            streamReader.Close();
        }
    }

    return string.Empty;
}

其他类似的命令也可以工作:

private void button1_Click(object sender, EventArgs e)
{
    StringBuilder sb1 = new StringBuilder();

    sb1.Append("[{\"");
    sb1.Append("txid");
    sb1.Append("\":\"");
    sb1.Append(Convert.ToString(data["result"][Convert.ToInt32(txtFromJSON.Text)]["txid"]));
    sb1.Append("\",\"");
    sb1.Append("vout");
    sb1.Append("\":");
    sb1.Append(Convert.ToString(data["result"][Convert.ToInt32(txtFromJSON.Text)]["vout"]));
    sb1.Append("}]");

    StringBuilder sb2 = new StringBuilder();

    sb2.Append("[{\"");
    sb2.Append(Convert.ToString(data["result"][Convert.ToInt32(txtToJSON.Text)]["address"]));
    sb2.Append("\":");
    sb2.Append(txtAmountToSpend.Text);
    sb2.Append("}]");

    // {"jsonrpc":"1.0","id":"1","method":"createrawtransaction","params":["[{\"txid\":\"1a43a1f27c5837d5319a45217aa948a4d39c1d89faf497ce59de5bd570a64a26\",\"vout\":1}]","[{\"2NAZpRsvj9BstxxCDkKoe1FVjmPPxdmvqKj\":0.01}]"]}
    data = JObject.Parse(RequestServer("createrawtransaction", new List<string>() { Convert.ToString(sb1), Convert.ToString(sb2) }));

    MessageBox.Show(Convert.ToString(data));
}

这也有效:

// {"jsonrpc":"1.0","id":"1","method":"sendtoaddress","params":["2N8hwP1WmJrFF5QWABn38y63uYLhnJYJYTF","0.1"]}
data = JObject.Parse(RequestServer("sendtoaddress", new List<string>() { "2N8hwP1WmJrFF5QWABn38y63uYLhnJYJYTF", Convert.ToString(0.1) } ));

我的问题:

// {"jsonrpc":"1.0","id":"1","method":"listunspent","params":[]} data = JObject.Parse(RequestServer("listunspent", new List<String>() { })); 我做了什么错事?


更新1:

如评论中所建议,我已经更改了createrawtransaction,现在使用的是对象,然后使用Newtonsoft.Json序列化了对象。

这是我第二次尝试使用https://en.bitcoin.it/wiki/API_reference_(JSON-RPC)#.NET_.28C.23.29中的API参考代码:

StringBuilder

这是新的序列化JSON:

private void button1_Click(object sender, EventArgs e)
{
    JContainer jArray = new JArray();

    JObject jFromTx = new JObject
    {
        { "txid", data["result"][Convert.ToInt32(txtFromJSON.Text)]["txid"] },
        { "vout", data["result"][Convert.ToInt32(txtFromJSON.Text)]["vout"] }
    };

    jArray.Add(jFromTx);

    JObject jToTx = new JObject
    {
        { Convert.ToString(data["result"][Convert.ToInt32(txtToJSON.Text)]["address"]), Convert.ToDouble(txtAmountToSpend.Text) }
    };

    JContainer jArray2 = new JArray
    {
        jToTx
    };

    string strFrom = JsonConvert.SerializeObject(jArray);
    string strTo = JsonConvert.SerializeObject(jArray2);

    data = JObject.Parse(RequestServer("createrawtransaction", new List<string>() { strFrom, strTo }));
}

与第一次尝试时的旧StringBuilder JSON相比:

{"jsonrpc":"1.0","id":"1","method":"createrawtransaction","params":["[{\"txid\":\"1a43a1f27c5837d5319a45217aa948a4d39c1d89faf497ce59de5bd570a64a26\",\"vout\":1}]","[{\"2NAZpRsvj9BstxxCDkKoe1FVjmPPxdmvqKj\":0.01}]"]}

我仍然收到与以前相同的错误消息(见上文)。

1 个答案:

答案 0 :(得分:4)

认为问题是您的params数组正在进行双序列化,因此服务器不知道如何解释请求。我确实意识到您的JSON与示例相同,因此我在这里很可能是错误的。我绝对不是使用Bitcoin Core的API的专家。但是,我确实查看了third-party library的源代码,该源代码应该与Bitcoin Core兼容(也许您应该使用它?),并且它似乎并没有对{{ 1}}请求。这使我相信问题是双序列化参数。

要修复,请尝试以下操作:

  1. 为此更改现有createrawtransation方法的方法签名:

    RequestServer

    对此:

    public static string RequestServer(string methodName, List<string> parameters)
    
  2. 使用旧签名创建public static string RequestServer(string methodName, List<JToken> parameters) 方法的新重载,该旧签名将调用刚刚更改的现有签名。这将使您的其他已经使用的方法(例如RequestServersendtoaddress)保持不变。

    listunspent
  3. 最后,更改您的public static string RequestServer(string methodName, List<string> parameters) { return RequestServer(methodName, parameters.Select(p => new JValue(p)).ToList<JToken>()); } 方法中的代码,以使其不序列化button1_ClickjArray,而是将它们在jArray2中传递给{ {1}}。换句话说,更改此代码:

    List<JToken>

    对此:

    RequestServer

有了这些更改,string strFrom = JsonConvert.SerializeObject(jArray); string strTo = JsonConvert.SerializeObject(jArray2); data = JObject.Parse(RequestServer("createrawtransaction", new List<string>() { strFrom, strTo })); 的RPC JSON应该最终看起来像这样:

data = JObject.Parse(RequestServer("createrawtransaction", new List<JToken>() { jArray, jArray2 }));

请注意,createrawtransaction数组中多余的引号和反斜杠已消失。与以前相比:

{"jsonrpc":"1.0","id":"1","method":"createrawtransaction","params":[[{"txid":"1a43a1f27c5837d5319a45217aa948a4d39c1d89faf497ce59de5bd570a64a26","vout":1}],[{"2NAZpRsvj9BstxxCDkKoe1FVjmPPxdmvqKj":0.01}]]}

让我知道这是否适合您。如果没有,我将删除此答案。