Skip to content

Commit

Permalink
ScenarioRunner: enrich incomplete transactions
Browse files Browse the repository at this point in the history
fixes #11352

CHANGELOG_BEGIN
CHANGELOG_END
  • Loading branch information
remyhaemmerle-da committed Oct 25, 2021
1 parent 5365d68 commit 1ff2c42
Show file tree
Hide file tree
Showing 5 changed files with 75 additions and 56 deletions.
Original file line number Diff line number Diff line change
Expand Up @@ -7,7 +7,13 @@ package engine
import com.daml.lf.data.Ref.{Identifier, Name, PackageId}
import com.daml.lf.language.{Ast, LookupError}
import com.daml.lf.transaction.Node.{GenNode, KeyWithMaintainers}
import com.daml.lf.transaction.{Node, NodeId, VersionedTransaction}
import com.daml.lf.transaction.{
IncompleteTransaction,
GenTransaction,
Node,
NodeId,
VersionedTransaction,
}
import com.daml.lf.value.Value
import com.daml.lf.value.Value.VersionedValue
import com.daml.lf.speedy.SValue
Expand Down Expand Up @@ -169,7 +175,7 @@ final class ValueEnricher(
} yield exe.copy(chosenValue = choiceArg, exerciseResult = result, key = key)
}

def enrichTransaction(tx: VersionedTransaction): Result[VersionedTransaction] = {
def enrichTransaction(tx: GenTransaction): Result[GenTransaction] =
for {
normalizedNodes <-
tx.nodes.foldLeft[Result[Map[NodeId, GenNode]]](ResultDone(Map.empty)) {
Expand All @@ -179,11 +185,19 @@ final class ValueEnricher(
normalizedNode <- enrichNode(node)
} yield nodes.updated(nid, normalizedNode)
}
} yield VersionedTransaction(
version = tx.version,
} yield GenTransaction(
nodes = normalizedNodes,
roots = tx.roots,
)
}

def enrichTransaction(versionedTx: VersionedTransaction): Result[VersionedTransaction] =
enrichTransaction(GenTransaction(versionedTx.nodes, versionedTx.roots)).map {
case GenTransaction(nodes, roots) =>
VersionedTransaction(versionedTx.version, nodes, roots)
}

def enrichTransaction(incompleteTx: IncompleteTransaction): Result[IncompleteTransaction] =
enrichTransaction(incompleteTx.transaction).map(transaction =>
incompleteTx.copy(transaction = transaction)
)
}
Original file line number Diff line number Diff line change
Expand Up @@ -37,14 +37,8 @@ private[lf] object PartialTransaction {
type Node = Node.GenNode
type LeafNode = Node.LeafOnlyActionNode

private type TX = GenTransaction
private type ExerciseNode = Node.NodeExercises

private final case class IncompleteTxImpl(
val transaction: TX,
val locationInfo: Map[NodeId, Location],
) extends transaction.IncompleteTransaction

sealed abstract class ContextInfo {
val actionChildSeed: Int => crypto.Hash
def authorizers: Set[Party]
Expand Down Expand Up @@ -395,7 +389,7 @@ private[speedy] case class PartialTransaction(

val ptx = unwind()

IncompleteTxImpl(
transaction.IncompleteTransaction(
GenTransaction(
ptx.nodes,
ptx.context.children.toImmArray.toSeq.sortBy(_.index).toImmArray,
Expand Down
5 changes: 4 additions & 1 deletion daml-lf/scenario-interpreter/BUILD.bazel
Original file line number Diff line number Diff line change
Expand Up @@ -20,7 +20,10 @@ da_scala_library(
name = "scenario-interpreter",
srcs = glob(["src/main/**/*.scala"]),
main_class = "com.daml.lf.speedy.Main",
scala_deps = ["@maven//:org_typelevel_paiges_core"],
scala_deps = [
"@maven//:org_typelevel_paiges_core",
"@maven//:org_scalaz_scalaz_core",
],
scalacopts = lf_scalacopts_stricter,
tags = ["maven_coordinates=com.daml:daml-lf-scenario-interpreter:__VERSION__"],
versioned_scala_deps = {"2.12": ["@maven//:org_scalaz_scalaz_core"]},
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -374,6 +374,44 @@ object ScenarioRunner {
}
}

private[this] abstract class Enricher {
def enrich(tx: SubmittedTransaction): SubmittedTransaction
def enrich(tx: IncompleteTransaction): IncompleteTransaction
}

private[this] object NoEnricher extends Enricher {
override def enrich(tx: SubmittedTransaction): SubmittedTransaction = tx
override def enrich(tx: IncompleteTransaction): IncompleteTransaction = tx
}

private[this] class EnricherImpl(compiledPackages: CompiledPackages) extends Enricher {
val config = Engine.DevEngine().config
val valueTranslator =
new ValueTranslator(
interface = compiledPackages.interface,
forbidV0ContractId = config.forbidV0ContractId,
requireV1ContractIdSuffix = config.requireSuffixedGlobalContractId,
)
def translateValue(typ: Ast.Type, value: Value): Result[SValue] =
valueTranslator.translateValue(typ, value) match {
case Left(err) => ResultError(err)
case Right(sv) => ResultDone(sv)
}
def loadPackage(pkgId: PackageId, context: language.Reference): Result[Unit] = {
crash(LookupError.MissingPackage.pretty(pkgId, context))
}
val enricher = new ValueEnricher(compiledPackages, translateValue, loadPackage)
def consume[V](res: Result[V]): V =
res match {
case ResultDone(x) => x
case x => crash(s"unexpected Result when enriching value: $x")
}
override def enrich(tx: SubmittedTransaction): SubmittedTransaction =
SubmittedTransaction(consume(enricher.enrichTransaction(tx)))
override def enrich(tx: IncompleteTransaction): IncompleteTransaction =
consume(enricher.enrichTransaction(tx))
}

def submit[R](
compiledPackages: CompiledPackages,
ledger: LedgerApi[R],
Expand All @@ -398,53 +436,29 @@ object ScenarioRunner {
commitLocation = location,
)
val onLedger = ledgerMachine.withOnLedger(NameOf.qualifiedNameOfCurrentFunc)(identity)

def enrich(tx: SubmittedTransaction): SubmittedTransaction = {
val config = Engine.DevEngine().config
val valueTranslator =
new ValueTranslator(
interface = compiledPackages.interface,
forbidV0ContractId = config.forbidV0ContractId,
requireV1ContractIdSuffix = config.requireSuffixedGlobalContractId,
)
def translateValue(typ: Ast.Type, value: Value): Result[SValue] =
valueTranslator.translateValue(typ, value) match {
case Left(err) => ResultError(err)
case Right(sv) => ResultDone(sv)
}
def loadPackage(pkgId: PackageId, context: language.Reference): Result[Unit] = {
crash(LookupError.MissingPackage.pretty(pkgId, context))
}
val enricher = new ValueEnricher(compiledPackages, translateValue, loadPackage)
def consume[V](res: Result[V]): V =
res match {
case ResultDone(x) => x
case x => crash(s"unexpected Result when enriching value: $x")
}
SubmittedTransaction(consume(enricher.enrichTransaction(tx)))
}
val enricher = if (doEnrichment) new EnricherImpl(compiledPackages) else NoEnricher
import enricher._

@tailrec
def go(): SubmissionResult[R] = {
ledgerMachine.run() match {
case SResult.SResultFinalValue(resultValue) =>
onLedger.ptxInternal.finish match {
case PartialTransaction.CompleteTransaction(tx0, locationInfo, _) =>
val tx = if (doEnrichment) enrich(tx0) else tx0
ledger.commit(committers, readAs, location, tx, locationInfo) match {
case PartialTransaction.CompleteTransaction(tx, locationInfo, _) =>
ledger.commit(committers, readAs, location, enrich(tx), locationInfo) match {
case Left(err) =>
SubmissionError(err, onLedger.incompleteTransaction)
SubmissionError(err, enrich(onLedger.incompleteTransaction))
case Right(r) =>
Commit(r, resultValue, onLedger.incompleteTransaction)
Commit(r, resultValue, enrich(onLedger.incompleteTransaction))
}
case PartialTransaction.IncompleteTransaction(ptx) =>
throw new RuntimeException(s"Unexpected abort: $ptx")
}
case SResultError(err) =>
SubmissionError(Error.RunnerException(err), onLedger.incompleteTransaction)
SubmissionError(Error.RunnerException(err), enrich(onLedger.incompleteTransaction))
case SResultNeedContract(coid, tid @ _, committers, callback) =>
ledger.lookupContract(coid, committers, readAs, callback) match {
case Left(err) => SubmissionError(err, onLedger.incompleteTransaction)
case Left(err) => SubmissionError(err, enrich(onLedger.incompleteTransaction))
case Right(_) => go()
}
case SResultNeedKey(keyWithMaintainers, committers, callback) =>
Expand All @@ -455,7 +469,7 @@ object ScenarioRunner {
readAs,
callback,
) match {
case Left(err) => SubmissionError(err, onLedger.incompleteTransaction)
case Left(err) => SubmissionError(err, enrich(onLedger.incompleteTransaction))
case Right(_) => go()
}
case SResultNeedTime(callback) =>
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -6,13 +6,7 @@ package transaction

import com.daml.lf.data.Ref.Location

trait IncompleteTransaction {

type Nid = NodeId
type TX = GenTransaction
type ExerciseNode = Node.NodeExercises

def transaction: TX

def locationInfo: Map[Nid, Location]
}
final case class IncompleteTransaction(
transaction: GenTransaction,
locationInfo: Map[NodeId, Location],
)

0 comments on commit 1ff2c42

Please sign in to comment.