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.

186 lines
6.7 KiB

package controllers
import com.alexitc.playsonify.PublicErrorRenderer
import com.alexitc.playsonify.core.{ApplicationResult, FutureApplicationResult}
import com.alexitc.playsonify.models.{Count, PaginatedQuery, PaginatedResult}
import com.xsn.explorer.data.TransactionBlockingDataHandler
import com.xsn.explorer.errors.AddressFormatError
import com.xsn.explorer.helpers.{DataHelper, DummyXSNService}
import com.xsn.explorer.models._
import com.xsn.explorer.models.rpc.AddressBalance
import com.xsn.explorer.services.XSNService
import controllers.common.MyAPISpec
import org.scalactic.{Good, One, Or}
import play.api.inject.bind
import play.api.libs.json.{JsValue, Json}
import play.api.test.Helpers._
import scala.concurrent.Future
class AddressesControllerSpec extends MyAPISpec {
import DataHelper._
val addressEmpty = createAddressDetails(0, 0, List())
val addressFilled = createAddressDetails(
100,
200,
List(
createTransactionId("0834641a7d30d8a2d2b451617599670445ee94ed7736e146c13be260c576c641"),
createTransactionId("024aba1d535cfe5dd3ea465d46a828a57b00e1df012d7a2d158e0f7484173f7c")
)
)
val addressForUtxos = DataHelper.createAddress("XeNEPsgeWqNbrEGEN5vqv4wYcC3qQrqNyp")
val utxosResponse =
"""
|[
| {
| "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 addressForTransactions = createAddress("XxQ7j37LfuXgsLd5DZAwFKhT3s2ZMkW86F")
val addressTransaction = TransactionWithValues(
createTransactionId("92c51e4fe89466faa734d6207a7ef6115fa1dd33f7156b006fafc6bb85a79eb8"),
createBlockhash("ad22f0dcea2fdaa357aac6eab00695cf07b487e34113598909f625c24629c981"),
12312312L,
Size(1000),
sent = 50,
received = 200)
val customXSNService = new DummyXSNService {
val map = Map(
"Xi3sQfMQsy2CzMZTrnKW6HFGp1VqFThdLw" -> addressEmpty,
"XnH3bC9NruJ4wnu4Dgi8F3wemmJtcxpKp6" -> addressFilled
)
override def getAddressBalance(address: Address): FutureApplicationResult[AddressBalance] = {
val maybe = map.get(address.string).map(_.balance)
val result = Or.from(maybe, One(AddressFormatError))
Future.successful(result)
}
override def getTransactions(address: Address): FutureApplicationResult[List[TransactionId]] = {
val maybe = map.get(address.string).map(_.transactions)
val result = Or.from(maybe, One(AddressFormatError))
Future.successful(result)
}
override def getUnspentOutputs(address: Address): FutureApplicationResult[JsValue] = {
if (address == addressForUtxos) {
val result = Good(Json.parse(utxosResponse))
Future.successful(result)
} else {
super.getUnspentOutputs(address)
}
}
}
private val customTransactionDataHandler = new TransactionBlockingDataHandler {
override def upsert(transaction: Transaction): ApplicationResult[Transaction] = ???
override def delete(transactionId: TransactionId): ApplicationResult[Transaction] = ???
override def deleteBy(blockhash: Blockhash): ApplicationResult[List[Transaction]] = ???
override def getBy(address: Address, paginatedQuery: PaginatedQuery): ApplicationResult[PaginatedResult[TransactionWithValues]] = {
if (address == addressForTransactions) {
Good(PaginatedResult(paginatedQuery.offset, paginatedQuery.limit, Count(1), List(addressTransaction)))
} else {
Good(PaginatedResult(paginatedQuery.offset, paginatedQuery.limit, Count(0), List.empty))
}
}
}
override val application = guiceApplicationBuilder
.overrides(bind[XSNService].to(customXSNService))
.overrides(bind[TransactionBlockingDataHandler].to(customTransactionDataHandler))
.build()
"GET /addresses/:address" should {
def url(address: String) = s"/addresses/$address"
"retrieve address information" in {
val address = addressFilled
val response = GET(url("XnH3bC9NruJ4wnu4Dgi8F3wemmJtcxpKp6"))
status(response) mustEqual OK
val json = contentAsJson(response)
(json \ "balance").as[BigDecimal] mustEqual address.balance.balance
(json \ "received").as[BigDecimal] mustEqual address.balance.received
(json \ "transactions").as[List[String]].size mustEqual address.transactions.size
}
"fail on bad address format" in {
val address = "XnH3bC9NruJ4wnu4Dgi8F3wemmJtcxpKp"
val response = GET(url(address))
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 "address"
}
}
"GET /addresses/:address/utxos" should {
def url(address: String) = s"/addresses/$address/utxos"
"return an array with the result" in {
val response = GET(url(addressForUtxos.string))
status(response) mustEqual OK
contentAsJson(response) mustEqual Json.parse(utxosResponse)
}
}
"GET /addresses/:address/transactions" should {
def url(address: String, offset: Int, limit: Int) = s"/addresses/$address/transactions?offset=$offset&limit=$limit"
"return the transactions where the address was involved" in {
val offset = 0
val limit = 5
val response = GET(url(addressForTransactions.string, offset, limit))
status(response) mustEqual OK
val json = contentAsJson(response)
(json \ "offset").as[Int] mustEqual offset
(json \ "limit").as[Int] mustEqual limit
(json \ "total").as[Int] mustEqual 1
val data = (json \ "data").as[List[JsValue]]
data.size mustEqual 1
val item = data.head
(item \ "id").as[String] mustEqual addressTransaction.id.string
(item \ "blockhash").as[String] mustEqual addressTransaction.blockhash.string
(item \ "time").as[Long] mustEqual addressTransaction.time
(item \ "size").as[Int] mustEqual addressTransaction.size.int
(item \ "sent").as[BigDecimal] mustEqual addressTransaction.sent
(item \ "received").as[BigDecimal] mustEqual addressTransaction.received
}
}
}