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.
262 lines
9.8 KiB
262 lines
9.8 KiB
package controllers
|
|
|
|
import com.alexitc.playsonify.PublicErrorRenderer
|
|
import com.alexitc.playsonify.core.{ApplicationResult, FutureApplicationResult}
|
|
import com.xsn.explorer.data.TransactionBlockingDataHandler
|
|
import com.xsn.explorer.errors.TransactionNotFoundError
|
|
import com.xsn.explorer.helpers.{DataHelper, DummyXSNService, TransactionDummyDataHandler, TransactionLoader}
|
|
import com.xsn.explorer.models._
|
|
import com.xsn.explorer.models.rpc.Transaction
|
|
import com.xsn.explorer.services.XSNService
|
|
import controllers.common.MyAPISpec
|
|
import org.scalactic.{Bad, Every, Good}
|
|
import play.api.inject.bind
|
|
import play.api.libs.json.{JsValue, Json}
|
|
import play.api.test.Helpers._
|
|
|
|
import scala.concurrent.Future
|
|
|
|
class TransactionsControllerSpec extends MyAPISpec {
|
|
|
|
import DataHelper._
|
|
|
|
val coinbaseTx = TransactionLoader.get("024aba1d535cfe5dd3ea465d46a828a57b00e1df012d7a2d158e0f7484173f7c")
|
|
val nonCoinbaseTx = TransactionLoader.get("0834641a7d30d8a2d2b451617599670445ee94ed7736e146c13be260c576c641")
|
|
val nonCoinbasePreviousTx = TransactionLoader.get("585cec5009c8ca19e83e33d282a6a8de65eb2ca007b54d6572167703768967d9")
|
|
val severalInputsTx = TransactionLoader.get("a3c43d22bbba31a6e5c00f565cb9c5a1a365407df4cc90efa8a865656b52c0eb")
|
|
val firstAddress = createAddress("fygsydgfygsdyfgsdyg")
|
|
val secondAddress = createAddress("56wedf5wedweedw")
|
|
val firstTxId = DataHelper.createTransactionId("a3c43d223658a8656a31a6e5c407df4bbb0f565cb9c5a1acc90efa056b52c0eb")
|
|
val secondTxId = DataHelper.createTransactionId("8a865656b5a3c43d22b00f565cb9c5a1a3bba31a6e5c65407df4cc90efa2c0eb")
|
|
|
|
val customXSNService = new DummyXSNService {
|
|
val map = Map(
|
|
coinbaseTx.id -> coinbaseTx,
|
|
nonCoinbaseTx.id -> nonCoinbaseTx,
|
|
nonCoinbasePreviousTx.id -> nonCoinbasePreviousTx,
|
|
severalInputsTx.id -> severalInputsTx
|
|
)
|
|
|
|
override def getTransaction(txid: TransactionId): FutureApplicationResult[Transaction] = {
|
|
val result = map.get(txid)
|
|
.map(Good(_))
|
|
.getOrElse {
|
|
Bad(TransactionNotFoundError).accumulating
|
|
}
|
|
|
|
Future.successful(result)
|
|
}
|
|
|
|
override def getRawTransaction(txid: TransactionId): FutureApplicationResult[JsValue] = {
|
|
val result = map.get(txid)
|
|
.map { _ => TransactionLoader.json(txid.string) }
|
|
.map(Good(_))
|
|
.getOrElse {
|
|
Bad(TransactionNotFoundError).accumulating
|
|
}
|
|
|
|
Future.successful(result)
|
|
}
|
|
}
|
|
|
|
val transactionDataHandler = new TransactionDummyDataHandler {
|
|
|
|
val map: Map[Address, TransactionId] = Map(firstAddress -> firstTxId, secondAddress -> secondTxId)
|
|
|
|
override def getLatestTransactionBy(addresses: Every[Address]): ApplicationResult[Map[String, String]] = {
|
|
val result = map
|
|
.filterKeys(addresses contains _)
|
|
.map( x => x._1.string -> x._2.string )
|
|
|
|
Good(result)
|
|
}
|
|
}
|
|
|
|
override val application = guiceApplicationBuilder
|
|
.overrides(bind[XSNService].to(customXSNService))
|
|
.overrides(bind[TransactionBlockingDataHandler].to(transactionDataHandler))
|
|
.build()
|
|
|
|
"GET /transactions/:txid" should {
|
|
def url(txid: String) = s"/transactions/$txid"
|
|
|
|
"return coinbase transaction" in {
|
|
val tx = coinbaseTx
|
|
val response = GET(url(tx.id.string))
|
|
|
|
status(response) mustEqual OK
|
|
val json = contentAsJson(response)
|
|
(json \ "id").as[String] mustEqual tx.id.string
|
|
(json \ "blockhash").as[String] mustEqual tx.blockhash.string
|
|
(json \ "size").as[Size] mustEqual tx.size
|
|
(json \ "time").as[Long] mustEqual tx.time
|
|
(json \ "blocktime").as[Long] mustEqual tx.blocktime
|
|
(json \ "confirmations").as[Confirmations] mustEqual tx.confirmations
|
|
|
|
val outputJsonList = (json \ "output").as[List[JsValue]]
|
|
outputJsonList.size mustEqual 1
|
|
|
|
val outputJson = outputJsonList.head
|
|
(outputJson \ "address").as[String] mustEqual tx.vout.head.address.get.string
|
|
(outputJson \ "value").as[BigDecimal] mustEqual tx.vout.head.value
|
|
}
|
|
|
|
"return non-coinbase transaction" in {
|
|
val tx = nonCoinbaseTx
|
|
val input = List(
|
|
TransactionValue(
|
|
createAddress("XgEGH3y7RfeKEdn2hkYEvBnrnmGBr7zvjL"),
|
|
BigDecimal("2343749.965625"))
|
|
)
|
|
val details = TransactionDetails.from(nonCoinbaseTx, input)
|
|
val response = GET(url(tx.id.string))
|
|
|
|
status(response) mustEqual OK
|
|
val json = contentAsJson(response)
|
|
(json \ "id").as[String] mustEqual tx.id.string
|
|
(json \ "blockhash").as[String] mustEqual tx.blockhash.string
|
|
(json \ "size").as[Size] mustEqual tx.size
|
|
(json \ "time").as[Long] mustEqual tx.time
|
|
(json \ "blocktime").as[Long] mustEqual tx.blocktime
|
|
(json \ "confirmations").as[Confirmations] mustEqual tx.confirmations
|
|
|
|
val inputJsonList = (json \ "input").as[List[JsValue]]
|
|
inputJsonList.size mustEqual 1
|
|
|
|
val inputJson = inputJsonList.head
|
|
(inputJson \ "address").as[String] mustEqual details.input.head.address.string
|
|
(inputJson \ "value").as[BigDecimal] mustEqual details.input.head.value
|
|
|
|
val outputJsonList = (json \ "output").as[List[JsValue]]
|
|
outputJsonList.size mustEqual 2
|
|
|
|
val outputJson = outputJsonList.head
|
|
(outputJson \ "address").as[String] mustEqual details.output.head.address.string
|
|
(outputJson \ "value").as[BigDecimal] mustEqual details.output.head.value
|
|
|
|
val outputJson2 = outputJsonList.drop(1).head
|
|
(outputJson2 \ "address").as[String] mustEqual details.output.drop(1).head.address.string
|
|
(outputJson2 \ "value").as[BigDecimal] mustEqual details.output.drop(1).head.value
|
|
}
|
|
|
|
"return a transaction with several inputs" in {
|
|
val tx = severalInputsTx
|
|
val inputValue = TransactionValue(
|
|
createAddress("XgEGH3y7RfeKEdn2hkYEvBnrnmGBr7zvjL"),
|
|
BigDecimal("73242.18642578"))
|
|
|
|
val response = GET(url(tx.id.string))
|
|
|
|
status(response) mustEqual OK
|
|
val json = contentAsJson(response)
|
|
(json \ "id").as[String] mustEqual tx.id.string
|
|
(json \ "blockhash").as[String] mustEqual tx.blockhash.string
|
|
(json \ "size").as[Size] mustEqual tx.size
|
|
(json \ "time").as[Long] mustEqual tx.time
|
|
(json \ "blocktime").as[Long] mustEqual tx.blocktime
|
|
(json \ "confirmations").as[Confirmations] mustEqual tx.confirmations
|
|
|
|
val inputJsonList = (json \ "input").as[List[JsValue]]
|
|
inputJsonList.size mustEqual 11
|
|
|
|
inputJsonList.foreach { inputJson =>
|
|
(inputJson \ "address").as[String] mustEqual inputValue.address.string
|
|
(inputJson \ "value").as[BigDecimal] mustEqual inputValue.value
|
|
}
|
|
|
|
val outputJsonList = (json \ "output").as[List[JsValue]]
|
|
outputJsonList.size mustEqual 2
|
|
|
|
val outputJson = outputJsonList.head
|
|
(outputJson \ "address").as[String] mustEqual "XcmqLX4qptMgAigXTVH4SJRVb6ZKmq8rjH"
|
|
(outputJson \ "value").as[BigDecimal] mustEqual BigDecimal("55664.05066658")
|
|
|
|
val outputJson2 = outputJsonList.drop(1).head
|
|
(outputJson2 \ "address").as[String] mustEqual "XvUAd4vQtFtZ7v2Uo8e6aSsiRLAwyq1jwb"
|
|
(outputJson2 \ "value").as[BigDecimal] mustEqual BigDecimal("750000.00000000")
|
|
}
|
|
|
|
"fail on wrong transaction format" in {
|
|
// 63 characters
|
|
val txid = "000001d535cfe5dd3ea465d46a828a57b00e1df012d7a2d158e0f7484173f7c"
|
|
val response = GET(url(txid))
|
|
|
|
status(response) mustEqual BAD_REQUEST
|
|
val json = contentAsJson(response)
|
|
val errorList = (json \ "errors").as[List[JsValue]]
|
|
|
|
errorList.size mustEqual 1
|
|
val error = errorList.head
|
|
|
|
(error \ "type").as[String] mustEqual PublicErrorRenderer.FieldValidationErrorType
|
|
(error \ "field").as[String] mustEqual "transactionId"
|
|
(error \ "message").as[String].nonEmpty mustEqual true
|
|
}
|
|
|
|
"fail on unknown transaction" in {
|
|
val txid = "0000001d535cfe5dd3ea465d46a828a57b00e1df012d7a2d158e0f7484173f7c"
|
|
val response = GET(url(txid))
|
|
|
|
status(response) mustEqual BAD_REQUEST
|
|
val json = contentAsJson(response)
|
|
val errorList = (json \ "errors").as[List[JsValue]]
|
|
|
|
errorList.size mustEqual 1
|
|
val error = errorList.head
|
|
|
|
(error \ "type").as[String] mustEqual PublicErrorRenderer.FieldValidationErrorType
|
|
(error \ "field").as[String] mustEqual "transactionId"
|
|
(error \ "message").as[String].nonEmpty mustEqual true
|
|
}
|
|
}
|
|
|
|
"GET transactions/:txid/raw" should {
|
|
def url(txid: String) = s"/transactions/$txid/raw"
|
|
|
|
"retrieve the raw transaction" in {
|
|
val tx = coinbaseTx
|
|
val expected = TransactionLoader.json(tx.id.string)
|
|
val response = GET(url(tx.id.string))
|
|
|
|
status(response) mustEqual OK
|
|
val json = contentAsJson(response)
|
|
|
|
json mustEqual expected
|
|
}
|
|
}
|
|
|
|
"POST /transactions/latest" should {
|
|
def url = s"/transactions/latest"
|
|
|
|
"return the latest transactions for the addresses" in {
|
|
|
|
val addresses = List(firstAddress.string, secondAddress.string, "3rdaddress")
|
|
.map(x => s""" "$x" """)
|
|
.mkString("[", ",", "]")
|
|
val body = s"""{ "addresses": $addresses }"""
|
|
val expected = Json.obj(firstAddress.string -> firstTxId, secondAddress.string -> secondTxId)
|
|
|
|
val response = POST(url, Some(body))
|
|
status(response) mustEqual OK
|
|
val json = contentAsJson(response)
|
|
json mustEqual expected
|
|
}
|
|
|
|
"fail while passing no addresses" in {
|
|
val body = """{ "addresses": [] }"""
|
|
val response = POST(url, Some(body))
|
|
|
|
status(response) mustEqual BAD_REQUEST
|
|
|
|
val json = contentAsJson(response)
|
|
val errorList = (json \ "errors").as[List[JsValue]]
|
|
|
|
errorList.size mustEqual 1
|
|
val error = errorList.head
|
|
|
|
(error \ "type").as[String] mustEqual PublicErrorRenderer.FieldValidationErrorType
|
|
(error \ "field").as[String] mustEqual "addresses"
|
|
(error \ "message").as[String].nonEmpty mustEqual true
|
|
}
|
|
}
|
|
}
|
|
|