package com.xsn.explorer.services import com.alexitc.playsonify.core.FutureOr.Implicits.{FutureListOps, FutureOps, OrOps} import com.alexitc.playsonify.core.{FutureApplicationResult, FutureOr, FuturePaginatedResult} import com.alexitc.playsonify.models.ordering.{OrderingCondition, OrderingError, OrderingQuery} import com.alexitc.playsonify.models.pagination.{Limit, Offset, PaginatedQuery} import com.alexitc.playsonify.validators.PaginatedQueryValidator import com.xsn.explorer.data.async.TransactionFutureDataHandler import com.xsn.explorer.errors._ import com.xsn.explorer.models._ import com.xsn.explorer.models.rpc.TransactionVIN import com.xsn.explorer.parsers.TransactionOrderingParser import com.xsn.explorer.util.Extensions.FutureOrExt import io.scalaland.chimney.dsl._ import javax.inject.Inject import org.scalactic._ import org.slf4j.LoggerFactory import play.api.libs.json.{JsObject, JsString, JsValue} import scala.concurrent.{ExecutionContext, Future} import scala.util.control.NonFatal class TransactionService @Inject() ( paginatedQueryValidator: PaginatedQueryValidator, transactionOrderingParser: TransactionOrderingParser, xsnService: XSNService, transactionFutureDataHandler: TransactionFutureDataHandler)( implicit ec: ExecutionContext) { private val logger = LoggerFactory.getLogger(this.getClass) private val maxTransactionsPerQuery = 100 def getRawTransaction(txidString: String): FutureApplicationResult[JsValue] = { val result = for { txid <- { val maybe = TransactionId.from(txidString) Or.from(maybe, One(TransactionFormatError)).toFutureOr } transaction <- xsnService.getRawTransaction(txid).toFutureOr } yield transaction result.toFuture } def getTransactionDetails(txidString: String): FutureApplicationResult[TransactionDetails] = { val result = for { txid <- { val maybe = TransactionId.from(txidString) Or.from(maybe, One(TransactionFormatError)).toFutureOr } transaction <- xsnService.getTransaction(txid).toFutureOr input <- transaction .vin .map(getTransactionValue) .toFutureOr } yield TransactionDetails.from(transaction, input) result.toFuture } def getTransaction(txid: TransactionId): FutureApplicationResult[Transaction] = { val result = for { tx <- xsnService.getTransaction(txid).toFutureOr transactionVIN <- getTransactionVIN(tx.vin).toFutureOr rpcTransaction = tx.copy(vin = transactionVIN) } yield Transaction.fromRPC(rpcTransaction) result.toFuture } private def getTransactionVIN(list: List[TransactionVIN]): FutureApplicationResult[List[TransactionVIN]] = { def getVIN(vin: TransactionVIN) = { getTransactionValue(vin) .map { case Good(transactionValue) => val newVIN = vin.copy(address = Some(transactionValue.address), value = Some(transactionValue.value)) Good(newVIN) case Bad(e) => Bad(e) } } def loadVINSequentially(pending: List[TransactionVIN]): FutureOr[List[TransactionVIN]] = pending match { case x :: xs => for { tx <- getVIN(x).toFutureOr next <- loadVINSequentially(xs) } yield tx :: next case _ => Future.successful(Good(List.empty)).toFutureOr } list .map(getVIN) .toFutureOr .toFuture .recoverWith { case NonFatal(ex) => logger.warn(s"Failed to load VIN, trying sequentially, error = ${ex.getMessage}") loadVINSequentially(list).toFuture } } def getTransactions(ids: List[TransactionId]): FutureApplicationResult[List[Transaction]] = { def loadTransactionsSlowly(pending: List[TransactionId]): FutureOr[List[Transaction]] = pending match { case x :: xs => for { tx <- getTransaction(x).toFutureOr next <- loadTransactionsSlowly(xs) } yield tx :: next case _ => Future.successful(Good(List.empty)).toFutureOr } ids .map(getTransaction) .toFutureOr .recoverWith(XSNWorkQueueDepthExceeded) { logger.warn("Unable to load transaction due to server overload, loading them slowly") loadTransactionsSlowly(ids) } .toFuture .recoverWith { case NonFatal(ex) => logger.warn(s"Unable to load transactions due to server error, loading them sequentially, error = ${ex.getMessage}") loadTransactionsSlowly(ids).toFuture } } def getTransactions( addressString: String, paginatedQuery: PaginatedQuery, orderingQuery: OrderingQuery): FuturePaginatedResult[TransactionWithValues] = { val result = for { address <- { val maybe = Address.from(addressString) Or.from(maybe, One(AddressFormatError)).toFutureOr } paginatedQuery <- paginatedQueryValidator.validate(paginatedQuery, maxTransactionsPerQuery).toFutureOr ordering <- transactionOrderingParser.from(orderingQuery).toFutureOr transactions <- transactionFutureDataHandler.getBy(address, paginatedQuery, ordering).toFutureOr } yield transactions result.toFuture } def getLightWalletTransactions( addressString: String, limit: Limit, lastSeenTxidString: Option[String], orderingConditionString: String): FutureApplicationResult[WrappedResult[List[LightWalletTransaction]]] = { val result = for { address <- { val maybe = Address.from(addressString) Or.from(maybe, One(AddressFormatError)).toFutureOr } _ <- paginatedQueryValidator.validate(PaginatedQuery(Offset(0), limit), maxTransactionsPerQuery).toFutureOr orderingCondition <- getOrderingConditionResult(orderingConditionString).toFutureOr lastSeenTxid <- { lastSeenTxidString .map(TransactionId.from) .map { txid => Or.from(txid, One(TransactionFormatError)).map(Option.apply) } .getOrElse(Good(Option.empty)) .toFutureOr } transactions <- transactionFutureDataHandler.getBy(address, limit, lastSeenTxid, orderingCondition).toFutureOr } yield { val lightTxs = transactions.map { tx => val inputs = tx.inputs.map { input => input .into[LightWalletTransaction.Input] .withFieldRenamed(_.fromOutputIndex, _.index) .withFieldRenamed(_.fromTxid, _.txid) .transform } val outputs = tx.outputs.map { output => output.into[LightWalletTransaction.Output].transform } tx .into[LightWalletTransaction] .withFieldConst(_.inputs, inputs) .withFieldConst(_.outputs, outputs) .transform } WrappedResult(lightTxs) } result.toFuture } def sendRawTransaction(hexString: String): FutureApplicationResult[JsValue] = { val result = for { hex <- Or.from(HexString.from(hexString), One(InvalidRawTransactionError)).toFutureOr _ <- xsnService.sendRawTransaction(hex).toFutureOr } yield JsObject.empty + ("hex" -> JsString(hex.string)) result.toFuture } def getByBlockhash(blockhashString: String, paginatedQuery: PaginatedQuery, orderingQuery: OrderingQuery): FuturePaginatedResult[TransactionWithValues] = { val result = for { blockhash <- Or.from(Blockhash.from(blockhashString), One(BlockhashFormatError)).toFutureOr validatedQuery <- paginatedQueryValidator.validate(paginatedQuery, maxTransactionsPerQuery).toFutureOr order <- transactionOrderingParser.from(orderingQuery).toFutureOr r <- transactionFutureDataHandler.getByBlockhash(blockhash, validatedQuery, order).toFutureOr } yield r result.toFuture } def getByBlockhash(blockhashString: String, limit: Limit, lastSeenTxidString: Option[String]): FutureApplicationResult[WrappedResult[List[TransactionWithValues]]] = { val result = for { blockhash <- Or.from(Blockhash.from(blockhashString), One(BlockhashFormatError)).toFutureOr _ <- paginatedQueryValidator.validate(PaginatedQuery(Offset(0), limit), maxTransactionsPerQuery).toFutureOr lastSeenTxid <- { lastSeenTxidString .map(TransactionId.from) .map { txid => Or.from(txid, One(TransactionFormatError)).map(Option.apply) } .getOrElse(Good(Option.empty)) .toFutureOr } r <- transactionFutureDataHandler.getByBlockhash(blockhash, limit, lastSeenTxid).toFutureOr } yield WrappedResult(r) result.toFuture } private def getTransactionValue(vin: TransactionVIN): FutureApplicationResult[TransactionValue] = { val valueMaybe = for { value <- vin.value address <- vin.address } yield TransactionValue(address, value) valueMaybe .map(Good(_)) .map(Future.successful) .getOrElse { val txid = vin.txid val result = for { tx <- xsnService.getTransaction(txid).toFutureOr r <- { val maybe = tx .vout .find(_.n == vin.voutIndex) .flatMap(TransactionValue.from) Or.from(maybe, One(TransactionNotFoundError)).toFutureOr } } yield r result.toFuture } } /** TODO: Move to another file */ private def getOrderingConditionResult(unsafeOrderingCondition: String) = { val maybe = parseOrderingCondition(unsafeOrderingCondition) Or.from(maybe, One(OrderingError.InvalidCondition)) } /** TODO: Move to another file */ private def parseOrderingCondition(unsafeOrderingCondition: String): Option[OrderingCondition] = unsafeOrderingCondition.toLowerCase match { case "asc" => Some(OrderingCondition.AscendingOrder) case "desc" => Some(OrderingCondition.DescendingOrder) case _ => None } }