// 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 main //import "os" import "fmt" import "time" import "crypto/sha1" import "github.com/romana/rlog" import "etcd.io/bbolt" import "github.com/deroproject/derohe/rpc" import "github.com/deroproject/derohe/walletapi" import "github.com/ybbus/jsonrpc" const PLUGIN_NAME = "pong_server" const DEST_PORT = uint64(0x1234567812345678) var expected_arguments = rpc.Arguments{ {rpc.RPC_DESTINATION_PORT, rpc.DataUint64, DEST_PORT}, // { rpc.RPC_EXPIRY , rpc.DataTime, time.Now().Add(time.Hour).UTC()}, {rpc.RPC_COMMENT, rpc.DataString, "Purchase PONG"}, //{"float64", rpc.DataFloat64, float64(0.12345)}, // in atomic units {rpc.RPC_VALUE_TRANSFER, rpc.DataUint64, uint64(12345)}, // in atomic units } // currently the interpreter seems to have a glitch if this gets initialized within the code // see limitations github.com/traefik/yaegi var response = rpc.Arguments{ {rpc.RPC_DESTINATION_PORT, rpc.DataUint64, uint64(0)}, {rpc.RPC_SOURCE_PORT, rpc.DataUint64, DEST_PORT}, {rpc.RPC_COMMENT, rpc.DataString, "Successfully purchased pong (this could be serial/license key or download link or further)"}, } var rpcClient = jsonrpc.NewClient("http://127.0.0.1:40403/json_rpc") // empty place holder func main() { var err error fmt.Printf("Pong Server to demonstrate RPC over dero chain.\n") var addr *rpc.Address var addr_result rpc.GetAddress_Result err = rpcClient.CallFor(&addr_result, "GetAddress") if err != nil || addr_result.Address == "" { fmt.Printf("Could not obtain address from wallet err %s\n", err) return } if addr, err = rpc.NewAddress(addr_result.Address); err != nil { fmt.Printf("address could not be parsed: addr:%s err:%s\n", addr_result.Address, err) return } shasum := fmt.Sprintf("%x", sha1.Sum([]byte(addr.String()))) db_name := fmt.Sprintf("%s_%s.bbolt.db", PLUGIN_NAME, shasum) db, err := bbolt.Open(db_name, 0600, nil) if err != nil { fmt.Printf("could not open db err:%s\n", err) return } //defer db.Close() err = db.Update(func(tx *bbolt.Tx) error { _, err := tx.CreateBucketIfNotExists([]byte("SALE")) return err }) if err != nil { fmt.Printf("err creating bucket. err %s\n", err) } fmt.Printf("Persistant store created in '%s'\n", db_name) fmt.Printf("Wallet Address: %s\n", addr) service_address_without_amount := addr.Clone() service_address_without_amount.Arguments = expected_arguments[:len(expected_arguments)-1] fmt.Printf("Integrated address to activate '%s', (without hardcoded amount) service: \n%s\n", PLUGIN_NAME, service_address_without_amount.String()) // service address can be created client side for now service_address := addr.Clone() service_address.Arguments = expected_arguments fmt.Printf("Integrated address to activate '%s', service: \n%s\n", PLUGIN_NAME, service_address.String()) processing_thread(db) // rkeep processing //time.Sleep(time.Second) //return } func processing_thread(db *bbolt.DB) { var err error for { // currently we traverse entire history time.Sleep(time.Second) var transfers rpc.Get_Transfers_Result err = rpcClient.CallFor(&transfers, "GetTransfers", rpc.Get_Transfers_Params{In: true, DestinationPort: DEST_PORT}) if err != nil { rlog.Warnf("Could not obtain gettransfers from wallet err %s\n", err) continue } for _, e := range transfers.Entries { if e.Coinbase || !e.Incoming { // skip coinbase or outgoing, self generated transactions continue } // check whether the entry has been processed before, if yes skip it var already_processed bool db.View(func(tx *bbolt.Tx) error { if b := tx.Bucket([]byte("SALE")); b != nil { if ok := b.Get([]byte(e.TXID)); ok != nil { // if existing in bucket already_processed = true } } return nil }) if already_processed { // if already processed skip it continue } // check whether this service should handle the transfer if !e.Payload_RPC.Has(rpc.RPC_DESTINATION_PORT, rpc.DataUint64) || DEST_PORT != e.Payload_RPC.Value(rpc.RPC_DESTINATION_PORT, rpc.DataUint64).(uint64) { // this service is expecting value to be specfic continue } rlog.Infof("tx should be processed %s\n", e.TXID) if expected_arguments.Has(rpc.RPC_VALUE_TRANSFER, rpc.DataUint64) { // this service is expecting value to be specfic value_expected := expected_arguments.Value(rpc.RPC_VALUE_TRANSFER, rpc.DataUint64).(uint64) if e.Amount != value_expected { // TODO we should mark it as faulty rlog.Warnf("user transferred %d, we were expecting %d. so we will not do anything\n", e.Amount, value_expected) // this is an unexpected situation continue } // value received is what we are expecting, so time for response response[0].Value = e.SourcePort // source port now becomes destination port, similar to TCP response[2].Value = fmt.Sprintf("Sucessfully purchased pong (could be serial, license or download link or anything).You sent %s at height %d", walletapi.FormatMoney(e.Amount), e.Height) //_, err := response.CheckPack(transaction.PAYLOAD0_LIMIT)) // we only have 144 bytes for RPC // sender of ping now becomes destination var str string tparams := rpc.Transfer_Params{Transfers: []rpc.Transfer{{Destination: e.Sender, Amount: uint64(1), Payload_RPC: response}}} err = rpcClient.CallFor(&str, "Transfer", tparams) if err != nil { rlog.Warnf("sending reply tx err %s\n", err) continue } err = db.Update(func(tx *bbolt.Tx) error { b := tx.Bucket([]byte("SALE")) return b.Put([]byte(e.TXID), []byte("done")) }) if err != nil { rlog.Warnf("err updating db to err %s\n", err) } else { rlog.Infof("ping replied successfully with pong") } } } } }