package com.xsn.explorer.services import com.xsn.explorer.config.{ExplorerConfig, RPCConfig} import com.xsn.explorer.errors._ import com.xsn.explorer.helpers.{BlockLoader, DataHelper, Executors, TransactionLoader} import com.xsn.explorer.models.rpc.Masternode import com.xsn.explorer.models.values._ import org.mockito.ArgumentMatchers._ import org.mockito.Mockito.{mock => _, _} import org.scalactic.{Bad, Good} import org.scalatest.MustMatchers._ import org.scalatest.WordSpec import org.scalatest.concurrent.PatienceConfiguration.Timeout import org.scalatest.concurrent.ScalaFutures._ import org.scalatest.mockito.MockitoSugar._ import org.scalatest.time.{Seconds, Span} import play.api.libs.json.{JsNull, JsString, JsValue, Json} import play.api.libs.ws.{WSClient, WSRequest, WSResponse} import scala.concurrent.Future class XSNServiceRPCImplSpec extends WordSpec { import DataHelper._ val ws = mock[WSClient] val ec = Executors.externalServiceEC val rpcConfig = new RPCConfig { override def password: RPCConfig.Password = RPCConfig.Password("pass") override def host: RPCConfig.Host = RPCConfig.Host("localhost") override def username: RPCConfig.Username = RPCConfig.Username("user") } val explorerConfig = new ExplorerConfig { override def genesisBlock: Blockhash = Blockhash.from("00000c822abdbb23e28f79a49d29b41429737c6c7e15df40d1b1f1b35907ae34").get override def liteVersionConfig: ExplorerConfig.LiteVersionConfig = ??? } val request = mock[WSRequest] val response = mock[WSResponse] when(ws.url(anyString)).thenReturn(request) when(request.withAuth(anyString(), anyString(), any())).thenReturn(request) when(request.withHttpHeaders(any())).thenReturn(request) val service = new XSNServiceRPCImpl(ws, rpcConfig, explorerConfig)(ec) def createRPCSuccessfulResponse(result: JsValue): String = { s""" |{ | "result": ${Json.prettyPrint(result)}, | "id": null, | "error": null |} """.stripMargin } def createRPCErrorResponse(errorCode: Int, message: String): String = { s""" |{ | "result": null, | "id": null, | "error": { | "code": $errorCode, | "message": "$message" | } |} """.stripMargin } "getTransaction" should { "handle coinbase" in { val txid = createTransactionId("024aba1d535cfe5dd3ea465d46a828a57b00e1df012d7a2d158e0f7484173f7c") val responseBody = createRPCSuccessfulResponse(TransactionLoader.json(txid.string)) val json = Json.parse(responseBody) mockRequest(request, response)(200, json) whenReady(service.getTransaction(txid)) { result => result.isGood mustEqual true val tx = result.get tx.id.string mustEqual "024aba1d535cfe5dd3ea465d46a828a57b00e1df012d7a2d158e0f7484173f7c" tx.vin.isEmpty mustEqual true tx.vout.size mustEqual 1 } } "handle non-coinbase result" in { val txid = createTransactionId("0834641a7d30d8a2d2b451617599670445ee94ed7736e146c13be260c576c641") val responseBody = createRPCSuccessfulResponse(TransactionLoader.json(txid.string)) val json = Json.parse(responseBody) mockRequest(request, response)(200, json) whenReady(service.getTransaction(txid)) { result => result.isGood mustEqual true val tx = result.get tx.id.string mustEqual txid.string tx.vin.size mustEqual 1 tx.vin.head.txid.string mustEqual "585cec5009c8ca19e83e33d282a6a8de65eb2ca007b54d6572167703768967d9" tx.vout.size mustEqual 3 } } "handle transaction having no blocktime, nor time" in { // TODO: Remove this test when https://github.com/X9Developers/XSN/issues/72 is fixed. val txid = createTransactionId("f24cd135c34ebb9032f8bc5b45599f1424980d34583df2847c4a4db584c94e97") val responseBody = createRPCSuccessfulResponse(TransactionLoader.json(txid.string)) val json = Json.parse(responseBody) mockRequest(request, response)(200, json) whenReady(service.getTransaction(txid)) { result => result.isGood mustEqual true val tx = result.get tx.id mustEqual txid } } "handle transaction not found" in { val txid = createTransactionId("0834641a7d30d8a2d2b451617599670445ee94ed7736e146c13be260c576c641") val responseBody = createRPCErrorResponse(-5, "No information available about transaction") val json = Json.parse(responseBody) mockRequest(request, response)(500, json) whenReady(service.getTransaction(txid)) { result => result mustEqual Bad(TransactionNotFoundError).accumulating } } "handle error with message" in { val errorMessage = "Params must be an array" val txid = createTransactionId("0834641a7d30d8a2d2b451617599670445ee94ed7736e146c13be260c576c641") val responseBody = createRPCErrorResponse(-32600, errorMessage) val json = Json.parse(responseBody) mockRequest(request, response)(200, json) whenReady(service.getTransaction(txid)) { result => val error = XSNMessageError(errorMessage) result mustEqual Bad(error).accumulating } } "handle non successful status" in { val txid = createTransactionId("0834641a7d30d8a2d2b451617599670445ee94ed7736e146c13be260c576c641") mockRequest(request, response)(403, JsNull) whenReady(service.getTransaction(txid)) { result => result mustEqual Bad(XSNUnexpectedResponseError).accumulating } } "handle unexpected error" in { val txid = createTransactionId("0834641a7d30d8a2d2b451617599670445ee94ed7736e146c13be260c576c641") val responseBody = """{"result":null,"error":{},"id":null}""" val json = Json.parse(responseBody) mockRequest(request, response)(200, json) whenReady(service.getTransaction(txid)) { result => result mustEqual Bad(XSNUnexpectedResponseError).accumulating } } "handle work queue depth exceeded" in { val txid = createTransactionId("0834641a7d30d8a2d2b451617599670445ee94ed7736e146c13be260c576c641") val responseBody = createRPCErrorResponse(-1, "Work queue depth exceeded") val json = Json.parse(responseBody) mockRequest(request, response)(200, json) whenReady(service.getTransaction(txid)) { result => result mustEqual Bad(XSNWorkQueueDepthExceeded).accumulating } } "handle work queue depth exceeded (no json)" in { val txid = createTransactionId("0834641a7d30d8a2d2b451617599670445ee94ed7736e146c13be260c576c641") val responseBody = "Work queue depth exceeded" mockRequestString(request, response)(500, responseBody) val timeout = Timeout(Span(5, Seconds)) whenReady(service.getTransaction(txid), timeout) { result => result mustEqual Bad(XSNWorkQueueDepthExceeded).accumulating } } } "getRawTransaction" should { "retrieve the raw transaction" in { val txid = createTransactionId("024aba1d535cfe5dd3ea465d46a828a57b00e1df012d7a2d158e0f7484173f7c") val expected = TransactionLoader.json(txid.string) val responseBody = createRPCSuccessfulResponse(TransactionLoader.json(txid.string)) val json = Json.parse(responseBody) mockRequest(request, response)(200, json) whenReady(service.getRawTransaction(txid)) { result => result mustEqual Good(expected) } } } "getAddressBalance" should { "return the balance" in { val responseBody = """ |{ | "result": { | "balance": 2465010000000000, | "received": 1060950100000000 | }, | "error": null, | "id": null |} """.stripMargin.trim val address = Address.from("Xi3sQfMQsy2CzMZTrnKW6HFGp1VqFThdLw").get val json = Json.parse(responseBody) mockRequest(request, response)(200, json) whenReady(service.getAddressBalance(address)) { result => result.isGood mustEqual true val balance = result.get balance.balance mustEqual BigDecimal("24650100.00000000") balance.received mustEqual BigDecimal("10609501.00000000") } } "fail on invalid address" in { val responseBody = createRPCErrorResponse(-5, "Invalid address") val address = Address.from("Xi3sQfMQsy2CzMZTrnKW6HFGp1VqFThdLW").get val json = Json.parse(responseBody) mockRequest(request, response)(200, json) whenReady(service.getAddressBalance(address)) { result => result mustEqual Bad(AddressFormatError).accumulating } } } "getTransactions" should { "return the transactions" in { val responseBody = """ |{ | "result": [ | "3963203e8ff99c0effbc7c90ef1b534f7e60d9d4d1d131375bc73eb6af8b62d0", | "56eff1fc3ec29277a039944a10826e1bd24685bec5e5c46c946846cb859dc14b", | "d5718b83b6cec30e075c20dc61005a25a6eb707f14b92e89c2a9c4bc39635b5d", | "9cd1d22e786b7b8722ce51f551c3c5af2053a52bd7694b9ef79e0a5d95053b19", | "1dbf0277891ed39f8175fa08844eadbb6ed4b28464fbac0ba88464001192d79e", | "2520ec229a76db8efa8e3b384582ac5c1969224a97f32475b957151f0c8cdfa7", | "46d2f51afeab7aeb7adab821c374c7348ae0ff4edb7d0c9af995360630194cc8", | "1a91406280e2a77dc0baf8a13491a977cba2d2dae6a8ba93fc6bbd3a7aeec4e5", | "def0ae8bbfa45dca177f9c9f169e362bd25dee460a8ddc8c662e92e6968cd6d8" | ], | "error": null, | "id": null |} """.stripMargin.trim val address = Address.from("Xi3sQfMQsy2CzMZTrnKW6HFGp1VqFThdLw").get val json = Json.parse(responseBody) mockRequest(request, response)(200, json) whenReady(service.getTransactions(address)) { result => result.isGood mustEqual true result.get.size mustEqual 9 } } "fail on invalid address" in { val responseBody = """{"result":null,"error":{"code":-5,"message":"Invalid address"},"id":null}""" val address = Address.from("Xi3sQfMQsy2CzMZTrnKW6HFGp1VqFThdLW").get val json = Json.parse(responseBody) mockRequest(request, response)(200, json) whenReady(service.getTransactions(address)) { result => result mustEqual Bad(AddressFormatError).accumulating } } } "getBlock" should { "return the genesis block" in { val block = BlockLoader.json("00000c822abdbb23e28f79a49d29b41429737c6c7e15df40d1b1f1b35907ae34") val responseBody = createRPCSuccessfulResponse(block) val blockhash = Blockhash.from("00000c822abdbb23e28f79a49d29b41429737c6c7e15df40d1b1f1b35907ae34").get val json = Json.parse(responseBody) mockRequest(request, response)(200, json) whenReady(service.getBlock(blockhash)) { result => result.isGood mustEqual true val block = result.get block.hash mustEqual blockhash block.transactions mustEqual List.empty } } "return a block" in { val block = BlockLoader.json("b72dd1655408e9307ef5874be20422ee71029333283e2360975bc6073bdb2b81") val responseBody = createRPCSuccessfulResponse(block) val blockhash = Blockhash.from("b72dd1655408e9307ef5874be20422ee71029333283e2360975bc6073bdb2b81").get val json = Json.parse(responseBody) mockRequest(request, response)(200, json) whenReady(service.getBlock(blockhash)) { result => result.isGood mustEqual true val block = result.get block.hash.string mustEqual "b72dd1655408e9307ef5874be20422ee71029333283e2360975bc6073bdb2b81" block.transactions.size mustEqual 2 } } "return a TPoS block" in { val block = BlockLoader.json("a3a9fb111a3f85c3d920c2dc58ce14d541a65763834247ef958aa3b4d665ef9c") val responseBody = createRPCSuccessfulResponse(block) val blockhash = Blockhash.from("a3a9fb111a3f85c3d920c2dc58ce14d541a65763834247ef958aa3b4d665ef9c").get val json = Json.parse(responseBody) mockRequest(request, response)(200, json) whenReady(service.getBlock(blockhash)) { result => result.isGood mustEqual true val block = result.get block.hash.string mustEqual blockhash.string block.transactions.size mustEqual 2 } } "fail on unknown block" in { val responseBody = """{"result":null,"error":{"code":-5,"message":"Block not found"},"id":null}""" val blockhash = Blockhash.from("b72dd1655408e9307ef5874be20422ee71029333283e2360975bc6073bdb2b80").get val json = Json.parse(responseBody) mockRequest(request, response)(200, json) whenReady(service.getBlock(blockhash)) { result => result mustEqual Bad(BlockNotFoundError).accumulating } } } "getRawBlock" should { "return a block" in { val block = BlockLoader.json("b72dd1655408e9307ef5874be20422ee71029333283e2360975bc6073bdb2b81") val responseBody = createRPCSuccessfulResponse(block) val blockhash = Blockhash.from("b72dd1655408e9307ef5874be20422ee71029333283e2360975bc6073bdb2b81").get val json = Json.parse(responseBody) mockRequest(request, response)(200, json) whenReady(service.getRawBlock(blockhash)) { result => result mustEqual Good(block) } } } "getBlockhash" should { "return the blockhash" in { val blockhash = Blockhash.from("00000766115b26ecbc09cd3a3db6870fdaf2f049d65a910eb2f2b48b566ca7bd").get val responseBody = createRPCSuccessfulResponse(JsString(blockhash.string)) val json = Json.parse(responseBody) mockRequest(request, response)(200, json) whenReady(service.getBlockhash(Height(3))) { result => result mustEqual Good(blockhash) } } "fail on unknown block" in { val responseBody = createRPCErrorResponse(-8, "Block height out of range") val json = Json.parse(responseBody) mockRequest(request, response)(200, json) whenReady(service.getBlockhash(Height(-1))) { result => result mustEqual Bad(BlockNotFoundError).accumulating } } } "getServerStatistics" should { "return the statistics" in { val content = """ |{ | "height": 45204, | "bestblock": "60da3eccf50f10254dcc35c9a25006e129bc5f0d101f83bad5ce008cc4b47c75", | "transactions": 93047, | "txouts": 142721, | "hash_serialized_2": "1f439c1d43753b9935abeb8a8de9f9010b96f7533ccfdde4432de3648a6f20de", | "disk_size": 7105097, | "total_amount": 77634169.93285364 |} """.stripMargin val responseBody = createRPCSuccessfulResponse(Json.parse(content)) val json = Json.parse(responseBody) mockRequest(request, response)(200, json) whenReady(service.getServerStatistics()) { result => result.isGood mustEqual true val stats = result.get stats.height mustEqual Height(45204) stats.transactions mustEqual 93047 stats.totalSupply mustEqual BigDecimal("77634169.93285364") } } } "getMasternodeCount" should { "return the count" in { val content = "10" val responseBody = createRPCSuccessfulResponse(Json.parse(content)) val json = Json.parse(responseBody) mockRequest(request, response)(200, json) whenReady(service.getMasternodeCount()) { result => result mustEqual Good(10) } } } "getDifficulty" should { "return the difficulty" in { val content = "129.1827211827212" val responseBody = createRPCSuccessfulResponse(Json.parse(content)) val json = Json.parse(responseBody) mockRequest(request, response)(200, json) whenReady(service.getDifficulty()) { result => result mustEqual Good(129.1827211827212) } } } "getMasternodes" should { "return the masternodes" in { val content = """ |{ | "COutPoint(c3efb8b60bda863a3a963d340901dc2b870e6ea51a34276a8f306d47ffb94f01, 0)": " WATCHDOG_EXPIRED 70209 XqdmM7rop8Sdgn8UjyNh3Povc3rhNSXYw2 1532897292 0 0 0 45.77.136.212:62583", | "COutPoint(b02f99d87194c9400ab147c070bf621770684906dedfbbe9ba5f3a35c26b8d01, 1)": " ENABLED 70209 XdNDRAiMUC9KiVRzhCTg9w44jQRdCpCRe3 1532905050 6010 1532790407 202522 45.32.148.13:62583" |} """.stripMargin val expected = List( Masternode( txid = TransactionId.from("c3efb8b60bda863a3a963d340901dc2b870e6ea51a34276a8f306d47ffb94f01").get, ip = "45.77.136.212:62583", protocol = "70209", status = "WATCHDOG_EXPIRED", activeSeconds = 0, lastSeen = 1532897292, Address.from("XqdmM7rop8Sdgn8UjyNh3Povc3rhNSXYw2").get), Masternode( txid = TransactionId.from("b02f99d87194c9400ab147c070bf621770684906dedfbbe9ba5f3a35c26b8d01").get, ip = "45.32.148.13:62583", protocol = "70209", status = "ENABLED", activeSeconds = 6010, lastSeen = 1532905050, Address.from("XdNDRAiMUC9KiVRzhCTg9w44jQRdCpCRe3").get) ) val responseBody = createRPCSuccessfulResponse(Json.parse(content)) val json = Json.parse(responseBody) mockRequest(request, response)(200, json) whenReady(service.getMasternodes()) { result => result.isGood mustEqual true val masternodes = result.get masternodes mustEqual expected } } } "getMasternode" should { "return the masternode" in { val content = """ |{ | "COutPoint(b02f99d87194c9400ab147c070bf621770684906dedfbbe9ba5f3a35c26b8d01, 1)": " ENABLED 70209 XdNDRAiMUC9KiVRzhCTg9w44jQRdCpCRe3 1532905050 6010 1532790407 202522 45.32.148.13:62583" |} """.stripMargin val expected = Masternode( txid = TransactionId.from("b02f99d87194c9400ab147c070bf621770684906dedfbbe9ba5f3a35c26b8d01").get, ip = "45.32.148.13:62583", protocol = "70209", status = "ENABLED", activeSeconds = 6010, lastSeen = 1532905050, Address.from("XdNDRAiMUC9KiVRzhCTg9w44jQRdCpCRe3").get) val responseBody = createRPCSuccessfulResponse(Json.parse(content)) val json = Json.parse(responseBody) mockRequest(request, response)(200, json) val ip = IPAddress.from("45.32.148.13").get whenReady(service.getMasternode(ip)) { result => result mustEqual Good(expected) } } "fail when the masternode is not found" in { val content = """ |{} """.stripMargin val responseBody = createRPCSuccessfulResponse(Json.parse(content)) val json = Json.parse(responseBody) mockRequest(request, response)(200, json) val ip = IPAddress.from("45.32.148.13").get whenReady(service.getMasternode(ip)) { result => result mustEqual Bad(MasternodeNotFoundError).accumulating } } } "getUnspentOutputs" should { "get the results" in { val content = """ |[ | { | "address": "XeNEPsgeWqNbrEGEN5vqv4wYcC3qQrqNyp", | "height": 22451, | "outputIndex": 0, | "satoshis": 1500000000000, | "script": "76a914285b6f1ccacea0059ff5393cb4eb2f0569e2b3e988ac", | "txid": "ea837f2011974b6a1a2fa077dc33684932c514a4ec6febc10e1a19ebe1336539" | }, | { | "address": "XeNEPsgeWqNbrEGEN5vqv4wYcC3qQrqNyp", | "height": 25093, | "outputIndex": 3, | "satoshis": 2250000000, | "script": "76a914285b6f1ccacea0059ff5393cb4eb2f0569e2b3e988ac", | "txid": "96a06b802d1c15818a42aa9b46dd2e236cde746000d35f74d3eb940ab9d5694d" | } |] """.stripMargin val responseBody = createRPCSuccessfulResponse(Json.parse(content)) val json = Json.parse(responseBody) mockRequest(request, response)(200, json) val address = DataHelper.createAddress("XeNEPsgeWqNbrEGEN5vqv4wYcC3qQrqNyp") whenReady(service.getUnspentOutputs(address)) { result => result mustEqual Good(Json.parse(content)) } } } "isTPoSContract" should { "return true when the contract is valid" in { val txid = createTransactionId("b02f99d87194c9400ab147c070bf621770684906dedfbbe9ba5f3a35c26b8d01") val content = "Contract is valid" val responseBody = createRPCSuccessfulResponse(JsString(content)) val json = Json.parse(responseBody) mockRequest(request, response)(200, json) whenReady(service.isTPoSContract(txid)) { result => result mustEqual Good(true) } } "return false when the contract is not valid" in { val txid = createTransactionId("b02f99d87194c9400ab147c070bf621770684906dedfbbe9ba5f3a35c26b8d01") val content = "Contract invalid, error: Signature invalid" val responseBody = createRPCSuccessfulResponse(JsString(content)) val json = Json.parse(responseBody) mockRequest(request, response)(200, json) whenReady(service.isTPoSContract(txid)) { result => result mustEqual Good(false) } } } "estimateSmartFee" should { "return the result" in { val content = """{"feerate":0.00001,"blocks":2}""" val responseBody = createRPCSuccessfulResponse(Json.parse(content)) val json = Json.parse(responseBody) mockRequest(request, response)(200, json) whenReady(service.estimateSmartFee(1)) { result => result mustEqual Good(Json.parse(content)) } } } private def mockRequest(request: WSRequest, response: WSResponse)(status: Int, body: JsValue) = { when(response.status).thenReturn(status) when(response.json).thenReturn(body) when(response.body).thenReturn(body.toString()) when(request.post[AnyRef](any())(any())).thenReturn(Future.successful(response)) } private def mockRequestString(request: WSRequest, response: WSResponse)(status: Int, body: String) = { when(response.status).thenReturn(status) when(response.body).thenReturn(body) when(request.post[String](anyString())(any())).thenReturn(Future.successful(response)) } }