Browse Source

server: Store the block filter while synchronizing the ledger

master
Alexis Hernandez 6 years ago
parent
commit
52ae52e3c4
  1. 16
      server/app/com/xsn/explorer/data/anorm/LedgerPostgresDataHandler.scala
  2. 43
      server/app/com/xsn/explorer/data/anorm/dao/BlockFilterPostgresDAO.scala
  3. 26
      server/app/com/xsn/explorer/data/anorm/parsers/BlockFilterParsers.scala
  4. 2
      server/test/com/xsn/explorer/data/common/PostgresDataHandlerSpec.scala
  5. 2
      server/test/com/xsn/explorer/helpers/DataHandlerObjects.scala

16
server/app/com/xsn/explorer/data/anorm/LedgerPostgresDataHandler.scala

@ -5,8 +5,9 @@ import java.sql.Connection
import com.alexitc.playsonify.core.ApplicationResult
import com.alexitc.playsonify.models.ApplicationError
import com.xsn.explorer.data.LedgerBlockingDataHandler
import com.xsn.explorer.data.anorm.dao.{AggregatedAmountPostgresDAO, BalancePostgresDAO, BlockPostgresDAO, TransactionPostgresDAO}
import com.xsn.explorer.data.anorm.dao._
import com.xsn.explorer.errors.{PostgresForeignKeyViolationError, PreviousBlockMissingError, RepeatedBlockHeightError}
import com.xsn.explorer.gcs.{GolombCodedSet, GolombEncoding}
import com.xsn.explorer.models.persisted.{Balance, Block, Transaction}
import com.xsn.explorer.models.values.Address
import com.xsn.explorer.util.Extensions.ListOptionExt
@ -18,6 +19,7 @@ import play.api.db.Database
class LedgerPostgresDataHandler @Inject() (
override val database: Database,
blockPostgresDAO: BlockPostgresDAO,
blockFilterPostgresDAO: BlockFilterPostgresDAO,
transactionPostgresDAO: TransactionPostgresDAO,
balancePostgresDAO: BalancePostgresDAO,
aggregatedAmountPostgresDAO: AggregatedAmountPostgresDAO)
@ -33,9 +35,11 @@ class LedgerPostgresDataHandler @Inject() (
override def push(
block: Block.HasTransactions): ApplicationResult[Unit] = {
// the filter is computed outside the transaction to avoid unnecessary locking
val filter = GolombEncoding.encode(block)
val result = withTransaction { implicit conn =>
val result = for {
_ <- upsertBlockCascade(block.asTip)
_ <- upsertBlockCascade(block.asTip, filter)
} yield ()
result
@ -66,11 +70,16 @@ class LedgerPostgresDataHandler @Inject() (
.getOrElse(throw new RuntimeException("Unable to pop block"))
}
private def upsertBlockCascade(block: Block.HasTransactions)(implicit conn: Connection): Option[Unit] = {
private def upsertBlockCascade(
block: Block.HasTransactions,
filter: Option[GolombCodedSet])(
implicit conn: Connection): Option[Unit] = {
val result = for {
// block
_ <- deleteBlockCascade(block.block).orElse(Some(()))
_ <- blockPostgresDAO.insert(block.block)
_ = filter.foreach { f => blockFilterPostgresDAO.insert(block.hash, f) }
// batch insert
_ <- transactionPostgresDAO.insert(block.transactions)
@ -96,6 +105,7 @@ class LedgerPostgresDataHandler @Inject() (
private def deleteBlockCascade(block: Block)(implicit conn: Connection): Option[Unit] = {
// transactions
val deletedTransactions = transactionPostgresDAO.deleteBy(block.hash)
val _ = blockFilterPostgresDAO.delete(block.hash)
for {
// block
_ <- blockPostgresDAO.delete(block.hash)

43
server/app/com/xsn/explorer/data/anorm/dao/BlockFilterPostgresDAO.scala

@ -0,0 +1,43 @@
package com.xsn.explorer.data.anorm.dao
import java.sql.Connection
import anorm._
import com.xsn.explorer.data.anorm.parsers.BlockFilterParsers
import com.xsn.explorer.gcs.GolombCodedSet
import com.xsn.explorer.models.values.Blockhash
class BlockFilterPostgresDAO {
import BlockFilterParsers._
def insert(blockhash: Blockhash, filter: GolombCodedSet)(implicit conn: Connection): GolombCodedSet = {
SQL(
"""
|INSERT INTO block_address_gcs
| (blockhash, m, n, p, hex)
|VALUES
| ({blockhash}, {m}, {n}, {p}, {hex})
|RETURNING blockhash, m, n, p, hex
""".stripMargin
).on(
'blockhash -> blockhash.string,
'm -> filter.m,
'n -> filter.n,
'p -> filter.p,
'hex -> filter.hex.string
).as(parseFilter.single)
}
def delete(blockhash: Blockhash)(implicit conn: Connection): Option[GolombCodedSet] = {
SQL(
"""
|DELETE FROM block_address_gcs
|WHERE blockhash = {blockhash}
|RETURNING blockhash, m, n, p, hex
""".stripMargin
).on(
'blockhash -> blockhash.string
).as(parseFilter.singleOpt)
}
}

26
server/app/com/xsn/explorer/data/anorm/parsers/BlockFilterParsers.scala

@ -0,0 +1,26 @@
package com.xsn.explorer.data.anorm.parsers
import anorm.SqlParser._
import anorm._
import com.xsn.explorer.gcs.GolombCodedSet
import com.xsn.explorer.models.values.HexString
object BlockFilterParsers {
val parseP = int("p")
val parseM = int("m")
val parseN = int("n")
val parseHex = str("hex")
.map(HexString.from)
.map { _.getOrElse(throw new RuntimeException("corrupted hex")) }
val parseFilter = (parseN ~ parseM ~ parseP ~ parseHex).map {
case n ~ m ~ p ~ hex =>
new GolombCodedSet(
n = n,
m = m,
p = p,
hex = hex
)
}
}

2
server/test/com/xsn/explorer/data/common/PostgresDataHandlerSpec.scala

@ -64,10 +64,10 @@ trait PostgresDataHandlerSpec
_root_.anorm.SQL("""DELETE FROM transaction_outputs""").execute()
_root_.anorm.SQL("""DELETE FROM address_transaction_details""").execute()
_root_.anorm.SQL("""DELETE FROM transactions""").execute()
_root_.anorm.SQL("""DELETE FROM block_address_gcs""").execute()
_root_.anorm.SQL("""DELETE FROM blocks""").execute()
_root_.anorm.SQL("""DELETE FROM balances""").execute()
_root_.anorm.SQL("""DELETE FROM hidden_addresses""").execute()
}
}
}

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

@ -17,6 +17,7 @@ trait DataHandlerObjects {
addressTransactionDetailsDAO,
fieldOrderingSQLInterpreter)
lazy val blockPostgresDAO = new BlockPostgresDAO(fieldOrderingSQLInterpreter)
lazy val blockFilterPostgresDAO = new BlockFilterPostgresDAO
lazy val balancePostgresDAO = new BalancePostgresDAO(fieldOrderingSQLInterpreter)
lazy val aggregatedAmountPostgresDAO = new AggregatedAmountPostgresDAO
@ -24,6 +25,7 @@ trait DataHandlerObjects {
new LedgerPostgresDataHandler(
database,
blockPostgresDAO,
blockFilterPostgresDAO,
transactionPostgresDAO,
balancePostgresDAO,
aggregatedAmountPostgresDAO)

Loading…
Cancel
Save