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.

190 lines
7.4 KiB

package controllers
import com.alexitc.playsonify.core.ApplicationResult
import com.alexitc.playsonify.play.PublicErrorRenderer
import com.xsn.explorer.data.TransactionBlockingDataHandler
import com.xsn.explorer.errors.TransactionNotFoundError
import com.xsn.explorer.helpers.{DataHelper, FileBasedXSNService, TransactionDummyDataHandler, TransactionLoader}
import com.xsn.explorer.models._
import com.xsn.explorer.models.values._
import com.xsn.explorer.services.XSNService
import controllers.common.MyAPISpec
import org.scalactic.Bad
import play.api.inject.bind
import play.api.libs.json.JsValue
import play.api.test.Helpers._
class TransactionsControllerSpec extends MyAPISpec {
import DataHelper._
private val coinbaseTx = TransactionLoader.get("024aba1d535cfe5dd3ea465d46a828a57b00e1df012d7a2d158e0f7484173f7c")
private val nonCoinbaseTx = TransactionLoader.get("0834641a7d30d8a2d2b451617599670445ee94ed7736e146c13be260c576c641")
private val severalInputsTx = TransactionLoader.get("a3c43d22bbba31a6e5c00f565cb9c5a1a365407df4cc90efa8a865656b52c0eb")
private val customXSNService = new FileBasedXSNService
private val transactionDataHandler = new TransactionDummyDataHandler {
override def getOutput(txid: TransactionId, index: Int): ApplicationResult[persisted.Transaction.Output] = {
Bad(TransactionNotFoundError).accumulating
}
}
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.addresses.flatMap(_.headOption).get.string
(outputJson \ "value").as[BigDecimal] mustEqual tx.vout.head.value
}
"return non-coinbase transaction" in {
val input = List(
TransactionValue(
createAddress("XgEGH3y7RfeKEdn2hkYEvBnrnmGBr7zvjL"),
BigDecimal("2343749.965625")))
.map { v =>
rpc.TransactionVIN.HasValues(createTransactionId("024aba1d535cfe5dd3ea465d46a828a57b00e1df012d7a2d158e0f7484173f7c"), 0, v.value, v.addresses)
}
val tx = nonCoinbaseTx.copy(vin = input)
val details = TransactionDetails.from(tx)
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.map(_.string).getOrElse("")
(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.map(_.string).getOrElse("")
(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.map(_.string).getOrElse("")
(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.map(_.string).getOrElse("")
(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
}
}
}