feat(rpc): transaction detail and bulk fetch endpoints
Co-Authored-By: Charon <charon@lethean.io> Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
This commit is contained in:
parent
e008dc5cfe
commit
91e124bc94
2 changed files with 169 additions and 0 deletions
47
rpc/transactions.go
Normal file
47
rpc/transactions.go
Normal file
|
|
@ -0,0 +1,47 @@
|
||||||
|
// Copyright (c) 2017-2026 Lethean (https://lt.hn)
|
||||||
|
//
|
||||||
|
// Licensed under the European Union Public Licence (EUPL) version 1.2.
|
||||||
|
// SPDX-License-Identifier: EUPL-1.2
|
||||||
|
|
||||||
|
package rpc
|
||||||
|
|
||||||
|
import "fmt"
|
||||||
|
|
||||||
|
// GetTxDetails returns detailed information about a transaction.
|
||||||
|
func (c *Client) GetTxDetails(txHash string) (*TxInfo, error) {
|
||||||
|
params := struct {
|
||||||
|
TxHash string `json:"tx_hash"`
|
||||||
|
}{TxHash: txHash}
|
||||||
|
var resp struct {
|
||||||
|
TxInfo TxInfo `json:"tx_info"`
|
||||||
|
Status string `json:"status"`
|
||||||
|
}
|
||||||
|
if err := c.call("get_tx_details", params, &resp); err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
if resp.Status != "OK" {
|
||||||
|
return nil, fmt.Errorf("get_tx_details: status %q", resp.Status)
|
||||||
|
}
|
||||||
|
return &resp.TxInfo, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// GetTransactions fetches transactions by hash.
|
||||||
|
// Uses the legacy /gettransactions endpoint (not available via /json_rpc).
|
||||||
|
// Returns hex-encoded transaction blobs and any missed (not found) hashes.
|
||||||
|
func (c *Client) GetTransactions(hashes []string) (txsHex []string, missed []string, err error) {
|
||||||
|
params := struct {
|
||||||
|
TxsHashes []string `json:"txs_hashes"`
|
||||||
|
}{TxsHashes: hashes}
|
||||||
|
var resp struct {
|
||||||
|
TxsAsHex []string `json:"txs_as_hex"`
|
||||||
|
MissedTx []string `json:"missed_tx"`
|
||||||
|
Status string `json:"status"`
|
||||||
|
}
|
||||||
|
if err := c.legacyCall("/gettransactions", params, &resp); err != nil {
|
||||||
|
return nil, nil, err
|
||||||
|
}
|
||||||
|
if resp.Status != "OK" {
|
||||||
|
return nil, nil, fmt.Errorf("gettransactions: status %q", resp.Status)
|
||||||
|
}
|
||||||
|
return resp.TxsAsHex, resp.MissedTx, nil
|
||||||
|
}
|
||||||
122
rpc/transactions_test.go
Normal file
122
rpc/transactions_test.go
Normal file
|
|
@ -0,0 +1,122 @@
|
||||||
|
// Copyright (c) 2017-2026 Lethean (https://lt.hn)
|
||||||
|
//
|
||||||
|
// Licensed under the European Union Public Licence (EUPL) version 1.2.
|
||||||
|
// SPDX-License-Identifier: EUPL-1.2
|
||||||
|
|
||||||
|
package rpc
|
||||||
|
|
||||||
|
import (
|
||||||
|
"encoding/json"
|
||||||
|
"net/http"
|
||||||
|
"net/http/httptest"
|
||||||
|
"testing"
|
||||||
|
)
|
||||||
|
|
||||||
|
func TestGetTxDetails_Good(t *testing.T) {
|
||||||
|
srv := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
||||||
|
w.Header().Set("Content-Type", "application/json")
|
||||||
|
json.NewEncoder(w).Encode(jsonRPCResponse{
|
||||||
|
JSONRPC: "2.0",
|
||||||
|
ID: json.RawMessage(`"0"`),
|
||||||
|
Result: json.RawMessage(`{
|
||||||
|
"status": "OK",
|
||||||
|
"tx_info": {
|
||||||
|
"id": "a6e8da986858e6825fce7a192097e6afae4e889cabe853a9c29b964985b23da8",
|
||||||
|
"blob_size": 6794,
|
||||||
|
"fee": 1000000000,
|
||||||
|
"amount": 18999000000000,
|
||||||
|
"timestamp": 1557345925,
|
||||||
|
"keeper_block": 51,
|
||||||
|
"blob": "ARMB..."
|
||||||
|
}
|
||||||
|
}`),
|
||||||
|
})
|
||||||
|
}))
|
||||||
|
defer srv.Close()
|
||||||
|
|
||||||
|
c := NewClient(srv.URL)
|
||||||
|
tx, err := c.GetTxDetails("a6e8da986858e6825fce7a192097e6afae4e889cabe853a9c29b964985b23da8")
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("GetTxDetails: %v", err)
|
||||||
|
}
|
||||||
|
if tx.ID != "a6e8da986858e6825fce7a192097e6afae4e889cabe853a9c29b964985b23da8" {
|
||||||
|
t.Errorf("id: got %q", tx.ID)
|
||||||
|
}
|
||||||
|
if tx.Fee != 1000000000 {
|
||||||
|
t.Errorf("fee: got %d, want 1000000000", tx.Fee)
|
||||||
|
}
|
||||||
|
if tx.KeeperBlock != 51 {
|
||||||
|
t.Errorf("keeper_block: got %d, want 51", tx.KeeperBlock)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestGetTxDetails_Bad_NotFound(t *testing.T) {
|
||||||
|
srv := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
||||||
|
w.Header().Set("Content-Type", "application/json")
|
||||||
|
json.NewEncoder(w).Encode(jsonRPCResponse{
|
||||||
|
JSONRPC: "2.0",
|
||||||
|
ID: json.RawMessage(`"0"`),
|
||||||
|
Error: &jsonRPCError{Code: -14, Message: "NOT_FOUND"},
|
||||||
|
})
|
||||||
|
}))
|
||||||
|
defer srv.Close()
|
||||||
|
|
||||||
|
c := NewClient(srv.URL)
|
||||||
|
_, err := c.GetTxDetails("0000000000000000000000000000000000000000000000000000000000000000")
|
||||||
|
if err == nil {
|
||||||
|
t.Fatal("expected error")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestGetTransactions_Good(t *testing.T) {
|
||||||
|
srv := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
||||||
|
if r.URL.Path != "/gettransactions" {
|
||||||
|
t.Errorf("path: got %s, want /gettransactions", r.URL.Path)
|
||||||
|
}
|
||||||
|
w.Header().Set("Content-Type", "application/json")
|
||||||
|
w.Write([]byte(`{
|
||||||
|
"txs_as_hex": ["01020304"],
|
||||||
|
"missed_tx": ["abcd1234"],
|
||||||
|
"status": "OK"
|
||||||
|
}`))
|
||||||
|
}))
|
||||||
|
defer srv.Close()
|
||||||
|
|
||||||
|
c := NewClient(srv.URL)
|
||||||
|
found, missed, err := c.GetTransactions([]string{"deadbeef", "abcd1234"})
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("GetTransactions: %v", err)
|
||||||
|
}
|
||||||
|
if len(found) != 1 {
|
||||||
|
t.Fatalf("found: got %d, want 1", len(found))
|
||||||
|
}
|
||||||
|
if found[0] != "01020304" {
|
||||||
|
t.Errorf("found[0]: got %q, want %q", found[0], "01020304")
|
||||||
|
}
|
||||||
|
if len(missed) != 1 {
|
||||||
|
t.Fatalf("missed: got %d, want 1", len(missed))
|
||||||
|
}
|
||||||
|
if missed[0] != "abcd1234" {
|
||||||
|
t.Errorf("missed[0]: got %q, want %q", missed[0], "abcd1234")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestGetTransactions_Good_AllFound(t *testing.T) {
|
||||||
|
srv := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
||||||
|
w.Header().Set("Content-Type", "application/json")
|
||||||
|
w.Write([]byte(`{"txs_as_hex":["aa","bb"],"missed_tx":[],"status":"OK"}`))
|
||||||
|
}))
|
||||||
|
defer srv.Close()
|
||||||
|
|
||||||
|
c := NewClient(srv.URL)
|
||||||
|
found, missed, err := c.GetTransactions([]string{"hash1", "hash2"})
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("GetTransactions: %v", err)
|
||||||
|
}
|
||||||
|
if len(found) != 2 {
|
||||||
|
t.Errorf("found: got %d, want 2", len(found))
|
||||||
|
}
|
||||||
|
if len(missed) != 0 {
|
||||||
|
t.Errorf("missed: got %d, want 0", len(missed))
|
||||||
|
}
|
||||||
|
}
|
||||||
Loading…
Add table
Reference in a new issue