Browse Source

server: BlockService uses the cache for loading headers

master
Alexis Hernandez 6 years ago
parent
commit
b9ee34d97d
  1. 27
      server/app/com/xsn/explorer/services/BlockService.scala
  2. 5
      server/test/com/xsn/explorer/services/LedgerSynchronizerServiceSpec.scala

27
server/app/com/xsn/explorer/services/BlockService.scala

@ -2,11 +2,14 @@ package com.xsn.explorer.services
import com.alexitc.playsonify.core.FutureApplicationResult import com.alexitc.playsonify.core.FutureApplicationResult
import com.alexitc.playsonify.core.FutureOr.Implicits.{FutureOps, OrOps} import com.alexitc.playsonify.core.FutureOr.Implicits.{FutureOps, OrOps}
import com.alexitc.playsonify.models.ordering.OrderingCondition
import com.alexitc.playsonify.models.pagination.{Limit, Offset, PaginatedQuery} import com.alexitc.playsonify.models.pagination.{Limit, Offset, PaginatedQuery}
import com.alexitc.playsonify.validators.PaginatedQueryValidator import com.alexitc.playsonify.validators.PaginatedQueryValidator
import com.xsn.explorer.cache.BlockHeaderCache
import com.xsn.explorer.data.async.BlockFutureDataHandler import com.xsn.explorer.data.async.BlockFutureDataHandler
import com.xsn.explorer.errors.{BlockRewardsNotFoundError, BlockhashFormatError} import com.xsn.explorer.errors.{BlockRewardsNotFoundError, BlockhashFormatError}
import com.xsn.explorer.models._ import com.xsn.explorer.models._
import com.xsn.explorer.models.persisted.BlockHeader
import com.xsn.explorer.models.rpc.{Block, TransactionVIN} import com.xsn.explorer.models.rpc.{Block, TransactionVIN}
import com.xsn.explorer.models.values.{Blockhash, Height} import com.xsn.explorer.models.values.{Blockhash, Height}
import com.xsn.explorer.parsers.OrderingConditionParser import com.xsn.explorer.parsers.OrderingConditionParser
@ -14,7 +17,7 @@ import com.xsn.explorer.services.logic.{BlockLogic, TransactionLogic}
import com.xsn.explorer.util.Extensions.FutureOrExt import com.xsn.explorer.util.Extensions.FutureOrExt
import javax.inject.Inject import javax.inject.Inject
import org.scalactic.{Bad, Good, One, Or} import org.scalactic.{Bad, Good, One, Or}
import play.api.libs.json.JsValue import play.api.libs.json.{JsValue, Writes}
import scala.concurrent.{ExecutionContext, Future} import scala.concurrent.{ExecutionContext, Future}
@ -24,7 +27,8 @@ class BlockService @Inject() (
paginatedQueryValidator: PaginatedQueryValidator, paginatedQueryValidator: PaginatedQueryValidator,
blockLogic: BlockLogic, blockLogic: BlockLogic,
transactionLogic: TransactionLogic, transactionLogic: TransactionLogic,
orderingConditionParser: OrderingConditionParser)( orderingConditionParser: OrderingConditionParser,
blockHeaderCache: BlockHeaderCache)(
implicit ec: ExecutionContext) { implicit ec: ExecutionContext) {
private val maxHeadersPerQuery = 1000 private val maxHeadersPerQuery = 1000
@ -32,7 +36,8 @@ class BlockService @Inject() (
def getBlockHeaders( def getBlockHeaders(
limit: Limit, limit: Limit,
lastSeenHashString: Option[String], lastSeenHashString: Option[String],
orderingConditionString: String): FutureApplicationResult[WrappedResult[List[persisted.BlockHeader]]] = { orderingConditionString: String)(
implicit writes: Writes[BlockHeader]): FutureApplicationResult[JsValue] = {
val result = for { val result = for {
lastSeenHash <- { lastSeenHash <- {
@ -46,11 +51,27 @@ class BlockService @Inject() (
_ <- paginatedQueryValidator.validate(PaginatedQuery(Offset(0), limit), maxHeadersPerQuery).toFutureOr _ <- paginatedQueryValidator.validate(PaginatedQuery(Offset(0), limit), maxHeadersPerQuery).toFutureOr
orderingCondition <- orderingConditionParser.parseReuslt(orderingConditionString).toFutureOr orderingCondition <- orderingConditionParser.parseReuslt(orderingConditionString).toFutureOr
json <- getCacheableHeaders(limit, orderingCondition, lastSeenHash).toFutureOr
} yield json
result.toFuture
}
private def getCacheableHeaders(
limit: Limit,
orderingCondition: OrderingCondition,
lastSeenHash: Option[Blockhash])(
implicit writes: Writes[BlockHeader]) = {
val cacheKey = BlockHeaderCache.Key(limit, orderingCondition, lastSeenHash)
blockHeaderCache.getOrSet(cacheKey) {
val result = for {
headers <-blockDataHandler.getHeaders(limit, orderingCondition, lastSeenHash).toFutureOr headers <-blockDataHandler.getHeaders(limit, orderingCondition, lastSeenHash).toFutureOr
} yield WrappedResult(headers) } yield WrappedResult(headers)
result.toFuture result.toFuture
} }
}
def getRawBlock(blockhashString: String): FutureApplicationResult[JsValue] = { def getRawBlock(blockhashString: String): FutureApplicationResult[JsValue] = {
val result = for { val result = for {

5
server/test/com/xsn/explorer/services/LedgerSynchronizerServiceSpec.scala

@ -2,6 +2,7 @@ package com.xsn.explorer.services
import com.alexitc.playsonify.core.FutureApplicationResult import com.alexitc.playsonify.core.FutureApplicationResult
import com.alexitc.playsonify.validators.PaginatedQueryValidator import com.alexitc.playsonify.validators.PaginatedQueryValidator
import com.xsn.explorer.cache.BlockHeaderCache
import com.xsn.explorer.data.async.{BlockFutureDataHandler, LedgerFutureDataHandler, TransactionFutureDataHandler} import com.xsn.explorer.data.async.{BlockFutureDataHandler, LedgerFutureDataHandler, TransactionFutureDataHandler}
import com.xsn.explorer.data.common.PostgresDataHandlerSpec import com.xsn.explorer.data.common.PostgresDataHandlerSpec
import com.xsn.explorer.errors.BlockNotFoundError import com.xsn.explorer.errors.BlockNotFoundError
@ -222,7 +223,9 @@ class LedgerSynchronizerServiceSpec extends PostgresDataHandlerSpec with BeforeA
new PaginatedQueryValidator, new PaginatedQueryValidator,
new BlockLogic, new BlockLogic,
new TransactionLogic, new TransactionLogic,
new OrderingConditionParser) new OrderingConditionParser,
BlockHeaderCache.default
)
val transactionRPCService = new TransactionRPCService(xsnService) val transactionRPCService = new TransactionRPCService(xsnService)
new LedgerSynchronizerService( new LedgerSynchronizerService(
xsnService, xsnService,

Loading…
Cancel
Save