You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
561 lines
19 KiB
561 lines
19 KiB
package com.xsn.explorer.services
|
|
|
|
import com.xsn.explorer.config.RPCConfig
|
|
import com.xsn.explorer.errors._
|
|
import com.xsn.explorer.helpers.{BlockLoader, DataHelper, Executors, TransactionLoader}
|
|
import com.xsn.explorer.models._
|
|
import com.xsn.explorer.models.rpc.Masternode
|
|
import org.mockito.ArgumentMatchers._
|
|
import org.mockito.Mockito._
|
|
import org.scalactic.{Bad, Good}
|
|
import org.scalatest.concurrent.ScalaFutures
|
|
import org.scalatest.mockito.MockitoSugar
|
|
import org.scalatest.{MustMatchers, OptionValues, WordSpec}
|
|
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 with MustMatchers with ScalaFutures with MockitoSugar with OptionValues {
|
|
|
|
import DataHelper._
|
|
|
|
val ws = mock[WSClient]
|
|
val ec = Executors.externalServiceEC
|
|
val config = 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 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, config)(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
|
|
}
|
|
}
|
|
}
|
|
|
|
"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)
|
|
}
|
|
}
|
|
}
|
|
|
|
"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))
|
|
}
|
|
}
|
|
}
|
|
|
|
private def mockRequest(request: WSRequest, response: WSResponse)(status: Int, body: JsValue) = {
|
|
when(response.status).thenReturn(200)
|
|
when(response.json).thenReturn(body)
|
|
when(request.post[String](anyString())(any())).thenReturn(Future.successful(response))
|
|
}
|
|
}
|
|
|