derohe-miniblock-mod/walletapi/rpcserver/rpc_websocket_server.go

339 lines
11 KiB
Go
Raw Normal View History

2021-12-04 16:42:11 +00:00
// Copyright 2017-2021 DERO Project. All rights reserved.
// Use of this source code in any form is governed by RESEARCH license.
// license can be found in the LICENSE file.
// GPG: 0F39 E425 8C65 3947 702A 8234 08B2 0360 A03A 9DE8
//
//
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND ANY
// EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
// MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL
// THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
// PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
// INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
// STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF
// THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
package rpcserver
import "io"
import "io/ioutil"
import "net"
import "fmt"
import "net/http"
import "time"
import "sync"
import "sync/atomic"
import "context"
import "strings"
import "runtime/debug"
import "encoding/json"
import "github.com/deroproject/derohe/config"
import "github.com/deroproject/derohe/globals"
import "github.com/deroproject/derohe/walletapi"
import "github.com/deroproject/derohe/rpc"
import "github.com/deroproject/derohe/glue/rwc"
import "github.com/gorilla/websocket"
import "github.com/go-logr/logr"
import "github.com/creachadair/jrpc2"
import "github.com/creachadair/jrpc2/handler"
import "github.com/creachadair/jrpc2/channel"
//import "github.com/creachadair/jrpc2/server"
import "github.com/creachadair/jrpc2/jhttp"
/* this file implements the rpcserver api, so as wallet and block explorer tools can work without migration */
// all components requiring access to blockchain must use , this struct to communicate
// this structure must be update while mutex
type RPCServer struct {
srv *http.Server
mux *http.ServeMux
logger logr.Logger
2022-02-06 07:06:32 +00:00
user string
password string
2021-12-04 16:42:11 +00:00
Exit_Event chan bool // blockchain is shutting down and we must quit ASAP
sync.RWMutex
}
var client_connections sync.Map
func RPCServer_Start(wallet *walletapi.Wallet_Disk, title string) (*RPCServer, error) {
var r RPCServer
r.logger = globals.Logger.WithName(title) // all components must use this logger
r.Exit_Event = make(chan bool)
2022-02-06 07:06:32 +00:00
if globals.Arguments["--rpc-login"] != nil { // this was verified at startup
userpass := globals.Arguments["--rpc-login"].(string)
parts := strings.SplitN(userpass, ":", 2)
r.user = parts[0]
r.password = parts[1]
}
2021-12-04 16:42:11 +00:00
go r.Run(wallet)
atomic.AddUint32(&globals.Subsystem_Active, 1) // increment subsystem
return &r, nil
}
// shutdown the rpc server component
func (r *RPCServer) RPCServer_Stop() {
r.Lock()
defer r.Unlock()
close(r.Exit_Event) // send signal to all connections to exit
if r.srv != nil {
r.srv.Shutdown(context.Background()) // shutdown the server
}
// TODO we must wait for connections to kill themselves
time.Sleep(1 * time.Second)
r.logger.Info("RPC Shutdown")
atomic.AddUint32(&globals.Subsystem_Active, ^uint32(0)) // this decrement 1 fom subsystem
}
2022-02-06 07:06:32 +00:00
// check basic authrizaion
func hasbasicauthfailed(rpcserver *RPCServer, w http.ResponseWriter, r *http.Request) bool {
if rpcserver.user == "" {
return false
}
u, p, ok := r.BasicAuth()
if !ok {
w.WriteHeader(401)
io.WriteString(w, "Authorization Required")
return true
}
if u != rpcserver.user || p != rpcserver.password {
w.WriteHeader(401)
io.WriteString(w, "Authorization Required")
return true
}
if globals.Arguments["--allow-rpc-password-change"] != nil && globals.Arguments["--allow-rpc-password-change"].(bool) == true {
if r.Header.Get("Pass") != "" {
rpcserver.password = r.Header.Get("Pass")
}
}
return false
}
2021-12-04 16:42:11 +00:00
// setup handlers
func (rpcserver *RPCServer) Run(wallet *walletapi.Wallet_Disk) {
var wallet_apis WALLET_CONTEXT
wallet_apis.logger = rpcserver.logger
wallet_apis.wallet = wallet
var options = &jrpc2.ServerOptions{AllowPush: true, NewContext: func() context.Context { return context.WithValue(context.Background(), "wallet_context", &wallet_apis) }}
// create a new mux
rpcserver.mux = http.NewServeMux()
default_address := "127.0.0.1:" + fmt.Sprintf("%d", config.Mainnet.Wallet_RPC_Default_Port)
if !globals.IsMainnet() {
default_address = "127.0.0.1:" + fmt.Sprintf("%d", config.Testnet.Wallet_RPC_Default_Port)
}
if _, ok := globals.Arguments["--rpc-bind"]; ok && globals.Arguments["--rpc-bind"] != nil {
addr, err := net.ResolveTCPAddr("tcp", globals.Arguments["--rpc-bind"].(string))
if err != nil {
rpcserver.logger.Error(err, "--rpc-bind address is invalid")
} else {
if addr.Port == 0 {
rpcserver.logger.Info("RPC server is disabled, No ports will be opened for RPC")
return
} else {
default_address = addr.String()
}
}
}
rpcserver.logger.Info("Wallet RPC/Websocket server starting", "address", default_address)
rpcserver.Lock()
rpcserver.srv = &http.Server{Addr: default_address, Handler: rpcserver.mux}
rpcserver.Unlock()
// Bridge HTTP to the JSON-RPC server.
var bridge = jhttp.NewBridge(wallet_handler, &jhttp.BridgeOptions{Server: options})
translate_http_to_jsonrpc_and_vice_versa := func(w http.ResponseWriter, r *http.Request) {
2022-02-06 07:06:32 +00:00
if hasbasicauthfailed(rpcserver, w, r) {
return
}
2021-12-04 16:42:11 +00:00
bridge.ServeHTTP(w, r)
}
ws_handler := func(w http.ResponseWriter, r *http.Request) {
var ws_server *jrpc2.Server
defer func() {
if r := recover(); r != nil { // safety so if anything wrong happens, verification fails
rpcserver.logger.V(1).Error(nil, "Recovered while processing websocket request", "r", r, "stack", debug.Stack())
}
if ws_server != nil {
client_connections.Delete(ws_server)
}
}()
2022-02-06 07:06:32 +00:00
if hasbasicauthfailed(rpcserver, w, r) {
return
}
2021-12-04 16:42:11 +00:00
c, err := upgrader.Upgrade(w, r, nil)
if err != nil {
rpcserver.logger.V(1).Error(err, "upgrade:")
return
}
defer c.Close()
input_output := rwc.New(c)
ws_server = jrpc2.NewServer(servicemux, options).Start(channel.RawJSON(input_output, input_output))
client_connections.Store(ws_server, 1)
ws_server.Wait()
}
rpcserver.mux.HandleFunc("/json_rpc", translate_http_to_jsonrpc_and_vice_versa)
rpcserver.mux.HandleFunc("/ws", ws_handler)
rpcserver.mux.HandleFunc("/", hello)
// handle SC installer, // this will install an sc an
rpcserver.mux.HandleFunc("/install_sc", func(w http.ResponseWriter, req *http.Request) { // translate call internally, how to do it using a single json request
var p rpc.Transfer_Params
2022-02-06 07:06:32 +00:00
if hasbasicauthfailed(rpcserver, w, req) {
return
}
2021-12-04 16:42:11 +00:00
b, err := ioutil.ReadAll(req.Body)
defer req.Body.Close()
if err != nil {
http.Error(w, err.Error(), 500)
return
}
p.SC_Code = string(b) // encode as base64
p.Ringsize = 2 // experts need not use this, they have direct call to do it
if result, err := Transfer(context.WithValue(context.Background(), "wallet_context", &wallet_apis), p); err != nil {
fmt.Fprintf(w, err.Error())
return
} else {
if err := json.NewEncoder(w).Encode(result); err != nil {
fmt.Fprintf(w, err.Error())
return
}
}
})
// handle nasty http requests
//r.mux.HandleFunc("/getheight", getheight)
//if DEBUG_MODE {
// r.mux.HandleFunc("/debug/pprof/", pprof.Index)
// Register pprof handlers individually if required
/* r.mux.HandleFunc("/debug/pprof/", pprof.Index)
r.mux.HandleFunc("/debug/pprof/cmdline", pprof.Cmdline)
r.mux.HandleFunc("/debug/pprof/profile", pprof.Profile)
r.mux.HandleFunc("/debug/pprof/symbol", pprof.Symbol)
r.mux.HandleFunc("/debug/pprof/trace", pprof.Trace)
*/
/*
// Register pprof handlers individually if required
r.mux.HandleFunc("/cdebug/pprof/", pprof.Index)
r.mux.HandleFunc("/cdebug/pprof/cmdline", pprof.Cmdline)
r.mux.HandleFunc("/cdebug/pprof/profile", pprof.Profile)
r.mux.HandleFunc("/cdebug/pprof/symbol", pprof.Symbol)
r.mux.HandleFunc("/cdebug/pprof/trace", pprof.Trace)
*/
// register metrics handler
// r.mux.HandleFunc("/metrics", prometheus.InstrumentHandler("dero", promhttp.HandlerFor(metrics.Registry, promhttp.HandlerOpts{})))
//}
//r.mux.HandleFunc("/json_rpc/debug", mr.ServeDebug)
if err := rpcserver.srv.ListenAndServe(); err != http.ErrServerClosed {
rpcserver.logger.Error(err, "ListenAndServe failed")
}
}
func hello(w http.ResponseWriter, r *http.Request) {
io.WriteString(w, "DERO BLOCKCHAIN Hello world!")
}
var upgrader = websocket.Upgrader{CheckOrigin: func(r *http.Request) bool { return true }} // use default options
type WALLET_CONTEXT struct {
r *RPCServer
logger logr.Logger
wallet *walletapi.Wallet_Disk
} // exports daemon status and other RPC apis
func WalletEcho(ctx context.Context, args []string) string {
return "WALLET " + strings.Join(args, " ")
}
// used to verify whether the connection is alive
func Ping(ctx context.Context) string {
return "Pong "
}
func Echo(ctx context.Context, args []string) string {
return "DERO " + strings.Join(args, " ")
}
var wallet_handler = handler.Map{
"Echo": handler.New(WalletEcho),
"getaddress": handler.New(GetAddress),
"GetAddress": handler.New(GetAddress),
"getbalance": handler.New(GetBalance),
"GetBalance": handler.New(GetBalance),
"getheight": handler.New(GetHeight),
"GetHeight": handler.New(GetHeight),
"get_transfer_by_txid": handler.New(GetTransferbyTXID),
"GetTransferbyTXID": handler.New(GetTransferbyTXID),
"get_transfers": handler.New(GetTransfers),
"GetTransfers": handler.New(GetTransfers),
"make_integrated_address": handler.New(MakeIntegratedAddress),
"MakeIntegratedAddress": handler.New(MakeIntegratedAddress),
"split_integrated_address": handler.New(SplitIntegratedAddress),
"SplitIntegratedAddress": handler.New(SplitIntegratedAddress),
"query_key": handler.New(QueryKey),
"QueryKey": handler.New(QueryKey),
"transfer": handler.New(Transfer),
"Transfer": handler.New(Transfer),
"transfer_split": handler.New(Transfer),
"scinvoke": handler.New(ScInvoke),
}
var servicemux = handler.ServiceMap{
"DERO": handler.Map{
"Echo": handler.New(Echo),
"Ping": handler.New(Ping),
},
"WALLET": wallet_handler,
}
func fromContext(ctx context.Context) *WALLET_CONTEXT {
u, ok := ctx.Value("wallet_context").(*WALLET_CONTEXT)
if !ok {
panic("cannot find wallet context")
}
return u
}