Browse Source

server: Add lite-version config

Enabling the lite-version allows to prune block details
to allow syncing fast until what the light wallet needs.
develop
Alexis Hernandez 6 years ago
parent
commit
5527b545b6
  1. 17
      server/app/com/xsn/explorer/config/ExplorerConfig.scala
  2. 14
      server/app/com/xsn/explorer/data/anorm/dao/AddressTransactionDetailsPostgresDAO.scala
  3. 11
      server/app/com/xsn/explorer/data/anorm/dao/TransactionInputPostgresDAO.scala
  4. 12
      server/app/com/xsn/explorer/data/anorm/dao/TransactionOutputPostgresDAO.scala
  5. 6
      server/app/com/xsn/explorer/data/anorm/dao/TransactionPostgresDAO.scala
  6. 2
      server/app/com/xsn/explorer/modules/ConfigModule.scala
  7. 9
      server/app/com/xsn/explorer/services/LedgerSynchronizerService.scala
  8. 17
      server/conf/application.conf
  9. 13
      server/test/com/xsn/explorer/helpers/Config.scala
  10. 9
      server/test/com/xsn/explorer/helpers/DataHandlerObjects.scala
  11. 1
      server/test/com/xsn/explorer/services/LedgerSynchronizerServiceSpec.scala
  12. 2
      server/test/com/xsn/explorer/services/XSNServiceRPCImplSpec.scala

17
server/app/com/xsn/explorer/config/ExplorerConfig.scala

@ -7,9 +7,15 @@ import play.api.Configuration
trait ExplorerConfig { trait ExplorerConfig {
def genesisBlock: Blockhash def genesisBlock: Blockhash
def liteVersionConfig: ExplorerConfig.LiteVersionConfig
} }
class PlayExplorerConfig @Inject() (config: Configuration) extends ExplorerConfig { object ExplorerConfig {
case class LiteVersionConfig(enabled: Boolean, syncTransactionsFromBlock: Int)
class Play @Inject() (config: Configuration) extends ExplorerConfig {
override val genesisBlock: Blockhash = { override val genesisBlock: Blockhash = {
Blockhash Blockhash
@ -17,5 +23,12 @@ class PlayExplorerConfig @Inject() (config: Configuration) extends ExplorerConfi
.getOrElse(throw new RuntimeException("The given genesisBlock is incorrect")) .getOrElse(throw new RuntimeException("The given genesisBlock is incorrect"))
} }
} override def liteVersionConfig: LiteVersionConfig = {
val inner = config.get[Configuration]("explorer.liteVersion")
val enabled = inner.get[Boolean]("enabled")
val syncTransactionsFromBlock = inner.get[Int]("syncTransactionsFromBlock")
LiteVersionConfig(enabled, syncTransactionsFromBlock)
}
}
}

14
server/app/com/xsn/explorer/data/anorm/dao/AddressTransactionDetailsPostgresDAO.scala

@ -3,11 +3,16 @@ package com.xsn.explorer.data.anorm.dao
import java.sql.Connection import java.sql.Connection
import anorm._ import anorm._
import com.xsn.explorer.config.ExplorerConfig
import com.xsn.explorer.data.anorm.parsers.TransactionParsers._ import com.xsn.explorer.data.anorm.parsers.TransactionParsers._
import com.xsn.explorer.models.persisted.{AddressTransactionDetails, Transaction} import com.xsn.explorer.models.persisted.{AddressTransactionDetails, Transaction}
import com.xsn.explorer.models.values.TransactionId import com.xsn.explorer.models.values.TransactionId
import javax.inject.Inject
import org.slf4j.LoggerFactory
class AddressTransactionDetailsPostgresDAO { class AddressTransactionDetailsPostgresDAO @Inject() (explorerConfig: ExplorerConfig) {
private val logger = LoggerFactory.getLogger(this.getClass)
def batchInsertDetails(transaction: Transaction.HasIO)(implicit conn: Connection): Option[Unit] = { def batchInsertDetails(transaction: Transaction.HasIO)(implicit conn: Connection): Option[Unit] = {
val outputAddressValueList = for { val outputAddressValueList = for {
@ -66,9 +71,12 @@ class AddressTransactionDetailsPostgresDAO {
params.tail: _* params.tail: _*
) )
val success = batch.execute().forall(_ == 1) val result = batch.execute()
val success = result.forall(_ == 1)
if (success ||
explorerConfig.liteVersionConfig.enabled) {
if (success) {
Some(()) Some(())
} else { } else {
None None

11
server/app/com/xsn/explorer/data/anorm/dao/TransactionInputPostgresDAO.scala

@ -3,12 +3,14 @@ package com.xsn.explorer.data.anorm.dao
import java.sql.Connection import java.sql.Connection
import anorm._ import anorm._
import com.xsn.explorer.config.ExplorerConfig
import com.xsn.explorer.data.anorm.parsers.TransactionParsers._ import com.xsn.explorer.data.anorm.parsers.TransactionParsers._
import com.xsn.explorer.models.persisted.Transaction import com.xsn.explorer.models.persisted.Transaction
import com.xsn.explorer.models.values.{Address, TransactionId} import com.xsn.explorer.models.values.{Address, TransactionId}
import javax.inject.Inject
import org.slf4j.LoggerFactory import org.slf4j.LoggerFactory
class TransactionInputPostgresDAO { class TransactionInputPostgresDAO @Inject() (explorerConfig: ExplorerConfig) {
private val logger = LoggerFactory.getLogger(this.getClass) private val logger = LoggerFactory.getLogger(this.getClass)
@ -41,8 +43,11 @@ class TransactionInputPostgresDAO {
params.tail: _* params.tail: _*
) )
val success = batch.execute().forall(_ == 1) val result = batch.execute()
if (success) { val success = result.forall(_ == 1)
if (success ||
explorerConfig.liteVersionConfig.enabled) {
Some(inputs) Some(inputs)
} else { } else {
None None

12
server/app/com/xsn/explorer/data/anorm/dao/TransactionOutputPostgresDAO.scala

@ -3,12 +3,14 @@ package com.xsn.explorer.data.anorm.dao
import java.sql.Connection import java.sql.Connection
import anorm._ import anorm._
import com.xsn.explorer.config.ExplorerConfig
import com.xsn.explorer.data.anorm.parsers.TransactionParsers._ import com.xsn.explorer.data.anorm.parsers.TransactionParsers._
import com.xsn.explorer.models.persisted.Transaction import com.xsn.explorer.models.persisted.Transaction
import com.xsn.explorer.models.values.{Address, TransactionId} import com.xsn.explorer.models.values.{Address, TransactionId}
import javax.inject.Inject
import org.slf4j.LoggerFactory import org.slf4j.LoggerFactory
class TransactionOutputPostgresDAO { class TransactionOutputPostgresDAO @Inject() (explorerConfig: ExplorerConfig) {
private val logger = LoggerFactory.getLogger(this.getClass) private val logger = LoggerFactory.getLogger(this.getClass)
@ -68,7 +70,9 @@ class TransactionOutputPostgresDAO {
) )
val success = batch.execute().forall(_ == 1) val success = batch.execute().forall(_ == 1)
if (success) { if (success ||
explorerConfig.liteVersionConfig.enabled) {
Some(outputs) Some(outputs)
} else{ } else{
None None
@ -145,7 +149,9 @@ class TransactionOutputPostgresDAO {
""".stripMargin """.stripMargin
).executeUpdate() ).executeUpdate()
if (result == inputs.size) { if (result == inputs.size ||
explorerConfig.liteVersionConfig.enabled) {
Option(()) Option(())
} else { } else {
None None

6
server/app/com/xsn/explorer/data/anorm/dao/TransactionPostgresDAO.scala

@ -6,6 +6,7 @@ import anorm._
import com.alexitc.playsonify.models.ordering.{FieldOrdering, OrderingCondition} import com.alexitc.playsonify.models.ordering.{FieldOrdering, OrderingCondition}
import com.alexitc.playsonify.models.pagination.{Count, Limit, PaginatedQuery} import com.alexitc.playsonify.models.pagination.{Count, Limit, PaginatedQuery}
import com.alexitc.playsonify.sql.FieldOrderingSQLInterpreter import com.alexitc.playsonify.sql.FieldOrderingSQLInterpreter
import com.xsn.explorer.config.ExplorerConfig
import com.xsn.explorer.data.anorm.parsers.TransactionParsers._ import com.xsn.explorer.data.anorm.parsers.TransactionParsers._
import com.xsn.explorer.models._ import com.xsn.explorer.models._
import com.xsn.explorer.models.fields.TransactionField import com.xsn.explorer.models.fields.TransactionField
@ -15,6 +16,7 @@ import javax.inject.Inject
import org.slf4j.LoggerFactory import org.slf4j.LoggerFactory
class TransactionPostgresDAO @Inject() ( class TransactionPostgresDAO @Inject() (
explorerConfig: ExplorerConfig,
transactionInputDAO: TransactionInputPostgresDAO, transactionInputDAO: TransactionInputPostgresDAO,
transactionOutputDAO: TransactionOutputPostgresDAO, transactionOutputDAO: TransactionOutputPostgresDAO,
tposContractDAO: TPoSContractDAO, tposContractDAO: TPoSContractDAO,
@ -65,8 +67,12 @@ class TransactionPostgresDAO @Inject() (
private def spend(transactions: List[Transaction.HasIO])(implicit conn: Connection): Unit = { private def spend(transactions: List[Transaction.HasIO])(implicit conn: Connection): Unit = {
val spendResult = transactions.map { tx => transactionOutputDAO.batchSpend(tx.id, tx.inputs) } val spendResult = transactions.map { tx => transactionOutputDAO.batchSpend(tx.id, tx.inputs) }
if (explorerConfig.liteVersionConfig.enabled) {
()
} else {
assert(spendResult.forall(_.isDefined), "Spending inputs batch failed") assert(spendResult.forall(_.isDefined), "Spending inputs batch failed")
} }
}
private def closeContracts(transactions: List[Transaction.HasIO])(implicit conn: Connection): Unit = { private def closeContracts(transactions: List[Transaction.HasIO])(implicit conn: Connection): Unit = {
for { for {

2
server/app/com/xsn/explorer/modules/ConfigModule.scala

@ -8,6 +8,6 @@ class ConfigModule extends AbstractModule {
override def configure(): Unit = { override def configure(): Unit = {
bind(classOf[RPCConfig]).to(classOf[PlayRPCConfig]) bind(classOf[RPCConfig]).to(classOf[PlayRPCConfig])
bind(classOf[LedgerSynchronizerConfig]).to(classOf[LedgerSynchronizerPlayConfig]) bind(classOf[LedgerSynchronizerConfig]).to(classOf[LedgerSynchronizerPlayConfig])
bind(classOf[ExplorerConfig]).to(classOf[PlayExplorerConfig]) bind(classOf[ExplorerConfig]).to(classOf[ExplorerConfig.Play])
} }
} }

9
server/app/com/xsn/explorer/services/LedgerSynchronizerService.scala

@ -2,6 +2,7 @@ package com.xsn.explorer.services
import com.alexitc.playsonify.core.FutureOr.Implicits.{FutureOps, OptionOps} import com.alexitc.playsonify.core.FutureOr.Implicits.{FutureOps, OptionOps}
import com.alexitc.playsonify.core.{ApplicationResult, FutureApplicationResult} import com.alexitc.playsonify.core.{ApplicationResult, FutureApplicationResult}
import com.xsn.explorer.config.ExplorerConfig
import com.xsn.explorer.data.async.{BlockFutureDataHandler, LedgerFutureDataHandler} import com.xsn.explorer.data.async.{BlockFutureDataHandler, LedgerFutureDataHandler}
import com.xsn.explorer.errors.BlockNotFoundError import com.xsn.explorer.errors.BlockNotFoundError
import com.xsn.explorer.models._ import com.xsn.explorer.models._
@ -16,6 +17,7 @@ import org.slf4j.LoggerFactory
import scala.concurrent.{ExecutionContext, Future} import scala.concurrent.{ExecutionContext, Future}
class LedgerSynchronizerService @Inject() ( class LedgerSynchronizerService @Inject() (
explorerConfig: ExplorerConfig,
xsnService: XSNService, xsnService: XSNService,
blockService: BlockService, blockService: BlockService,
transactionCollectorService: TransactionCollectorService, transactionCollectorService: TransactionCollectorService,
@ -172,8 +174,15 @@ class LedgerSynchronizerService @Inject() (
val result = for { val result = for {
rpcBlock <- xsnService.getBlock(blockhash).toFutureOr rpcBlock <- xsnService.getBlock(blockhash).toFutureOr
} yield { } yield {
if (explorerConfig.liteVersionConfig.enabled &&
rpcBlock.height.int < explorerConfig.liteVersionConfig.syncTransactionsFromBlock) {
// lite version, ignore transactions
rpcBlock.copy(transactions = List.empty)
} else {
rpcBlock rpcBlock
} }
}
result.toFuture result.toFuture
} }

17
server/conf/application.conf

@ -42,7 +42,7 @@ rpc {
synchronizer { synchronizer {
enabled = true enabled = true
initialDelay = "10 seconds" initialDelay = "10 seconds"
interval = "1 minute" interval = "1 minutes"
enabled = ${?XSN_SYNCHRONIZER_ENABLED} enabled = ${?XSN_SYNCHRONIZER_ENABLED}
} }
@ -110,4 +110,19 @@ externalService.dispatcher {
explorer { explorer {
genesisBlock = "00000c822abdbb23e28f79a49d29b41429737c6c7e15df40d1b1f1b35907ae34" genesisBlock = "00000c822abdbb23e28f79a49d29b41429737c6c7e15df40d1b1f1b35907ae34"
genesisBlock = ${?EXPLORER_GENESIS_BLOCK} genesisBlock = ${?EXPLORER_GENESIS_BLOCK}
liteVersion {
# Enabling the lite explorer, starts syncing the block details only, the transactions and
# its related details are synced only on blocks having height >= syncTransactionsFromBlock
#
# This is useful to support our light wallet, assuming that all its addresses are new (hence, old details aren't important).
#
# This also implies that some consistency checks are removed, you are required to drop some foreign keys manually.
#
# Updating these values on an existing explorer is likely to need to rebuilt the database from scratch.
enabled = false
enabled = ${?EXPLORER_LITE_VERSION_ENABLED}
syncTransactionsFromBlock = 0
}
} }

13
server/test/com/xsn/explorer/helpers/Config.scala

@ -0,0 +1,13 @@
package com.xsn.explorer.helpers
import com.xsn.explorer.config.ExplorerConfig
import com.xsn.explorer.models.values.Blockhash
object Config {
val explorerConfig = new ExplorerConfig {
override def genesisBlock: Blockhash = Blockhash.from("00000c822abdbb23e28f79a49d29b41429737c6c7e15df40d1b1f1b35907ae34").get
override def liteVersionConfig: ExplorerConfig.LiteVersionConfig = ExplorerConfig.LiteVersionConfig(enabled = false, 0)
}
}

9
server/test/com/xsn/explorer/helpers/DataHandlerObjects.scala

@ -7,12 +7,15 @@ import play.api.db.Database
trait DataHandlerObjects { trait DataHandlerObjects {
import Config.explorerConfig
lazy val fieldOrderingSQLInterpreter = new FieldOrderingSQLInterpreter lazy val fieldOrderingSQLInterpreter = new FieldOrderingSQLInterpreter
lazy val transactionInputDAO = new TransactionInputPostgresDAO lazy val transactionInputDAO = new TransactionInputPostgresDAO(explorerConfig)
lazy val transactionOutputDAO = new TransactionOutputPostgresDAO lazy val transactionOutputDAO = new TransactionOutputPostgresDAO(explorerConfig)
lazy val addressTransactionDetailsDAO = new AddressTransactionDetailsPostgresDAO lazy val addressTransactionDetailsDAO = new AddressTransactionDetailsPostgresDAO(explorerConfig)
lazy val tposContractDAO = new TPoSContractDAO lazy val tposContractDAO = new TPoSContractDAO
lazy val transactionPostgresDAO = new TransactionPostgresDAO( lazy val transactionPostgresDAO = new TransactionPostgresDAO(
explorerConfig,
transactionInputDAO, transactionInputDAO,
transactionOutputDAO, transactionOutputDAO,
tposContractDAO, tposContractDAO,

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

@ -227,6 +227,7 @@ class LedgerSynchronizerServiceSpec extends PostgresDataHandlerSpec with BeforeA
new TransactionFutureDataHandler(transactionDataHandler)(Executors.databaseEC) new TransactionFutureDataHandler(transactionDataHandler)(Executors.databaseEC)
) )
new LedgerSynchronizerService( new LedgerSynchronizerService(
Config.explorerConfig,
xsnService, xsnService,
blockService, blockService,
transactionCollectorService, transactionCollectorService,

2
server/test/com/xsn/explorer/services/XSNServiceRPCImplSpec.scala

@ -33,6 +33,8 @@ class XSNServiceRPCImplSpec extends WordSpec {
val explorerConfig = new ExplorerConfig { val explorerConfig = new ExplorerConfig {
override def genesisBlock: Blockhash = Blockhash.from("00000c822abdbb23e28f79a49d29b41429737c6c7e15df40d1b1f1b35907ae34").get override def genesisBlock: Blockhash = Blockhash.from("00000c822abdbb23e28f79a49d29b41429737c6c7e15df40d1b1f1b35907ae34").get
override def liteVersionConfig: ExplorerConfig.LiteVersionConfig = ???
} }
val request = mock[WSRequest] val request = mock[WSRequest]

Loading…
Cancel
Save