From 0bba409d0e2074e83bde8cc8a03f8b5c62d717ce Mon Sep 17 00:00:00 2001 From: Remy Date: Wed, 26 Oct 2022 20:13:38 +0200 Subject: [PATCH] [LF] Make ApiCommand.Exercise work directly with type constructor (#15360) CHANGELOG_BEGIN CHANGELOG_END --- .../scala/com/daml/TemplateOrInterface.scala | 8 ------- .../preprocessing/CommandPreprocessor.scala | 23 +++++++------------ .../engine/preprocessing/Preprocessor.scala | 4 ++-- .../engine/ApiCommandPreprocessorSpec.scala | 12 ++++------ .../daml/lf/engine/AuthPropagationSpec.scala | 14 +++++------ .../daml/lf/engine/EngineTest.scala | 20 ++++++++-------- .../daml/lf/engine/InterfacesTest.scala | 18 +++++++-------- .../daml/lf/engine/LargeTransactionTest.scala | 8 +++---- .../com/digitalasset/daml/lf/ApiCommand.scala | 12 +++++----- .../daml/lf/engine/script/Converter.scala | 2 +- .../ledgerinteraction/GrpcLedgerClient.scala | 2 +- .../ledgerinteraction/JsonLedgerClient.scala | 8 +++---- .../api/validation/CommandsValidator.scala | 4 +--- security-evidence.md | 6 ++--- 14 files changed, 60 insertions(+), 81 deletions(-) diff --git a/daml-lf/data/src/main/scala/com/daml/TemplateOrInterface.scala b/daml-lf/data/src/main/scala/com/daml/TemplateOrInterface.scala index 7753fba6d9b4..cb24dacccfe7 100644 --- a/daml-lf/data/src/main/scala/com/daml/TemplateOrInterface.scala +++ b/daml-lf/data/src/main/scala/com/daml/TemplateOrInterface.scala @@ -10,12 +10,4 @@ sealed abstract class TemplateOrInterface[+T, +I] extends Product with Serializa object TemplateOrInterface { final case class Template[+T](value: T) extends TemplateOrInterface[T, Nothing] final case class Interface[+I](value: I) extends TemplateOrInterface[Nothing, I] - - implicit class MergeableTemplateOrInterface[A](private val x: TemplateOrInterface[A, A]) - extends AnyVal { - def merge: A = x match { - case Template(a) => a - case Interface(a) => a - } - } } diff --git a/daml-lf/engine/src/main/scala/com/digitalasset/daml/lf/engine/preprocessing/CommandPreprocessor.scala b/daml-lf/engine/src/main/scala/com/digitalasset/daml/lf/engine/preprocessing/CommandPreprocessor.scala index 4eea6ab1dbb0..dbb1a43994d4 100644 --- a/daml-lf/engine/src/main/scala/com/digitalasset/daml/lf/engine/preprocessing/CommandPreprocessor.scala +++ b/daml-lf/engine/src/main/scala/com/digitalasset/daml/lf/engine/preprocessing/CommandPreprocessor.scala @@ -53,24 +53,17 @@ private[lf] final class CommandPreprocessor( } def unsafePreprocessExercise( - typeId: data.TemplateOrInterface[Ref.Identifier, Ref.Identifier], + typeId: Ref.Identifier, contractId: Value.ContractId, choiceId: Ref.ChoiceName, argument: Value, - ): speedy.Command = typeId match { - // TODO: https://github.com/digital-asset/daml/issues/14747 - // In order to split the issue in several PRs, we allow abusing the templateId case as an interface. - // We will change once we have added the interface_id field to the legder API Exercise command - case TemplateOrInterface.Template(templateId) => - handleLookup(pkgInterface.lookupTemplateOrInterface(templateId)) match { - case TemplateOrInterface.Template(_) => - unsafePreprocessExerciseTemplate(templateId, contractId, choiceId, argument) - case TemplateOrInterface.Interface(_) => - unsafePreprocessExerciseInterface(templateId, contractId, choiceId, argument) - } - case TemplateOrInterface.Interface(ifaceId) => - unsafePreprocessExerciseInterface(ifaceId, contractId, choiceId, argument) - } + ): speedy.Command = + handleLookup(pkgInterface.lookupTemplateOrInterface(typeId)) match { + case TemplateOrInterface.Template(_) => + unsafePreprocessExerciseTemplate(typeId, contractId, choiceId, argument) + case TemplateOrInterface.Interface(_) => + unsafePreprocessExerciseInterface(typeId, contractId, choiceId, argument) + } def unsafePreprocessExerciseTemplate( templateId: Ref.Identifier, diff --git a/daml-lf/engine/src/main/scala/com/digitalasset/daml/lf/engine/preprocessing/Preprocessor.scala b/daml-lf/engine/src/main/scala/com/digitalasset/daml/lf/engine/preprocessing/Preprocessor.scala index 336afca4a351..57f4fb6e29c3 100644 --- a/daml-lf/engine/src/main/scala/com/digitalasset/daml/lf/engine/preprocessing/Preprocessor.scala +++ b/daml-lf/engine/src/main/scala/com/digitalasset/daml/lf/engine/preprocessing/Preprocessor.scala @@ -137,7 +137,7 @@ private[engine] final class Preprocessor( private[engine] def preprocessApiCommand( cmd: command.ApiCommand ): Result[speedy.Command] = - safelyRun(pullTemplatePackage(List(cmd.typeId.merge))) { + safelyRun(pullTemplatePackage(List(cmd.typeId))) { commandPreprocessor.unsafePreprocessApiCommand(cmd) } @@ -146,7 +146,7 @@ private[engine] final class Preprocessor( def preprocessApiCommands( cmds: data.ImmArray[command.ApiCommand] ): Result[ImmArray[speedy.Command]] = - safelyRun(pullTemplatePackage(cmds.toSeq.view.map(_.typeId.merge))) { + safelyRun(pullTemplatePackage(cmds.toSeq.view.map(_.typeId))) { commandPreprocessor.unsafePreprocessApiCommands(cmds) } diff --git a/daml-lf/engine/src/test/scala/com/digitalasset/daml/lf/engine/ApiCommandPreprocessorSpec.scala b/daml-lf/engine/src/test/scala/com/digitalasset/daml/lf/engine/ApiCommandPreprocessorSpec.scala index 1ccbe0c33e0d..56f97ab520b5 100644 --- a/daml-lf/engine/src/test/scala/com/digitalasset/daml/lf/engine/ApiCommandPreprocessorSpec.scala +++ b/daml-lf/engine/src/test/scala/com/digitalasset/daml/lf/engine/ApiCommandPreprocessorSpec.scala @@ -119,7 +119,7 @@ class ApiCommandPreprocessorSpec ) // TEST_EVIDENCE: Integrity: well formed exercise API command is accepted val validExeTemplate = ApiCommand.Exercise( - TemplateOrInterface.Template("Mod:Record"), + "Mod:Record", newCid, "Transfer", ValueRecord("", ImmArray("content" -> ValueList(FrontStack(ValueParty("Clara"))))), @@ -133,7 +133,7 @@ class ApiCommandPreprocessorSpec ) // TEST_EVIDENCE: Integrity: well formed exercise-by-interface command is accepted val validExeInterface = ApiCommand.Exercise( - TemplateOrInterface.Interface("Mod:Iface"), + "Mod:Iface", newCid, "IfaceChoice", ValueUnit, @@ -162,9 +162,7 @@ class ApiCommandPreprocessorSpec validCreate.copy(argument = ValueRecord("", ImmArray("content" -> ValueInt64(42)))) -> a[Error.Preprocessing.TypeMismatch], // TEST_EVIDENCE: Integrity: ill-formed exercise API command is rejected - validExeTemplate.copy(typeId = TemplateOrInterface.Template("Mod:Undefined")) -> - a[Error.Preprocessing.Lookup], - validExeTemplate.copy(typeId = TemplateOrInterface.Interface("Mod:Undefined")) -> + validExeTemplate.copy(typeId = "Mod:Undefined") -> a[Error.Preprocessing.Lookup], validExeTemplate.copy(choiceId = "Undefined") -> a[Error.Preprocessing.Lookup], @@ -217,13 +215,13 @@ class ApiCommandPreprocessorSpec ValueRecord("", ImmArray("" -> valueParties, "" -> ValueContractId(culpritCid))), ), ApiCommand.Exercise( - TemplateOrInterface.Template("Mod:RecordRef"), + "Mod:RecordRef", innocentCid, "Change", ValueContractId(culpritCid), ), ApiCommand.Exercise( - TemplateOrInterface.Template("Mod:RecordRef"), + "Mod:RecordRef", culpritCid, "Change", ValueContractId(innocentCid), diff --git a/daml-lf/engine/src/test/scala/com/digitalasset/daml/lf/engine/AuthPropagationSpec.scala b/daml-lf/engine/src/test/scala/com/digitalasset/daml/lf/engine/AuthPropagationSpec.scala index 9c06f8228df9..311fdc3bc900 100644 --- a/daml-lf/engine/src/test/scala/com/digitalasset/daml/lf/engine/AuthPropagationSpec.scala +++ b/daml-lf/engine/src/test/scala/com/digitalasset/daml/lf/engine/AuthPropagationSpec.scala @@ -7,7 +7,7 @@ package engine import com.daml.bazeltools.BazelRunfiles import com.daml.lf.archive.UniversalArchiveDecoder import com.daml.lf.command.{ApiCommand, ApiCommands} -import com.daml.lf.data.{Bytes, FrontStack, ImmArray, TemplateOrInterface, Time} +import com.daml.lf.data.{Bytes, FrontStack, ImmArray, Time} import com.daml.lf.data.Ref.{Identifier, Name, PackageId, ParticipantId, Party, QualifiedName} import com.daml.lf.language.Ast.Package import com.daml.lf.ledger.FailedAuthorization.{ @@ -220,7 +220,7 @@ class AuthPropagationSpec extends AnyFreeSpec with Matchers with Inside with Baz "ok (Alice signed contract; Bob exercised Choice)" in { val command: ApiCommand = ApiCommand.Exercise( - TemplateOrInterface.Template("T1"), + "T1", toContractId("t1a"), "Choice1", ValueRecord( @@ -242,7 +242,7 @@ class AuthPropagationSpec extends AnyFreeSpec with Matchers with Inside with Baz "fail: ExerciseMissingAuthorization" in { val command: ApiCommand = ApiCommand.Exercise( - TemplateOrInterface.Template("T1"), + "T1", toContractId("t1a"), "Choice1", ValueRecord( @@ -274,7 +274,7 @@ class AuthPropagationSpec extends AnyFreeSpec with Matchers with Inside with Baz "fail: CreateMissingAuthorization" in { val command: ApiCommand = ApiCommand.Exercise( - TemplateOrInterface.Template("T1"), + "T1", toContractId("t1a"), "Choice1", ValueRecord( @@ -305,7 +305,7 @@ class AuthPropagationSpec extends AnyFreeSpec with Matchers with Inside with Baz "ok (Bob signed contract; Alice exercised Choice)" in { val command: ApiCommand = ApiCommand.Exercise( - TemplateOrInterface.Template("T1"), + "T1", toContractId("t1b"), "Choice1", ValueRecord( @@ -333,7 +333,7 @@ class AuthPropagationSpec extends AnyFreeSpec with Matchers with Inside with Baz "fail (no implicit authority from outer exercise's contract's signatories)" in { val command: ApiCommand = ApiCommand.Exercise( - TemplateOrInterface.Template("X1"), + "X1", toContractId("x1b"), "ChoiceA", ValueRecord( @@ -376,7 +376,7 @@ class AuthPropagationSpec extends AnyFreeSpec with Matchers with Inside with Baz "ok" in { val command: ApiCommand = ApiCommand.Exercise( - TemplateOrInterface.Template("X1"), + "X1", toContractId("x1b"), "ChoiceA", ValueRecord( diff --git a/daml-lf/engine/src/test/scala/com/digitalasset/daml/lf/engine/EngineTest.scala b/daml-lf/engine/src/test/scala/com/digitalasset/daml/lf/engine/EngineTest.scala index bf602b1e408c..3313efec07ed 100644 --- a/daml-lf/engine/src/test/scala/com/digitalasset/daml/lf/engine/EngineTest.scala +++ b/daml-lf/engine/src/test/scala/com/digitalasset/daml/lf/engine/EngineTest.scala @@ -267,7 +267,7 @@ class EngineTest val cid = toContractId("BasicTests:Simple:1") val command = ApiCommand.Exercise( - TemplateOrInterface.Template(templateId), + templateId, cid, "Hello", ValueRecord(Some(hello), ImmArray.Empty), @@ -1048,7 +1048,7 @@ class EngineTest // we need to fix time as cid are depending on it val let = Time.Timestamp.assertFromString("1969-07-20T20:17:00Z") val command = ApiCommand.Exercise( - TemplateOrInterface.Template(templateId), + templateId, originalCoid, "Transfer", ValueRecord(None, ImmArray((Some[Name]("newReceiver"), ValueParty(clara)))), @@ -1231,7 +1231,7 @@ class EngineTest def runExample(cid: ContractId, exerciseActor: Party) = { val command = ApiCommand.Exercise( - TemplateOrInterface.Template(fetcherTid), + fetcherTid, cid, "DoFetch", ValueRecord(None, ImmArray((Some[Name]("cid"), ValueContractId(fetchedCid)))), @@ -1395,7 +1395,7 @@ class EngineTest "mark all lookupByKey nodes as byKey" in { val exerciseCmd = ApiCommand.Exercise( - TemplateOrInterface.Template(lookerUpTemplateId), + lookerUpTemplateId, lookerUpCid, "Lookup", ValueRecord(None, ImmArray((Some[Name]("n"), ValueInt64(42)))), @@ -1427,7 +1427,7 @@ class EngineTest "be reinterpreted to the same node when lookup finds a contract" in { val exerciseCmd = ApiCommand.Exercise( - TemplateOrInterface.Template(lookerUpTemplateId), + lookerUpTemplateId, lookerUpCid, "Lookup", ValueRecord(None, ImmArray((Some[Name]("n"), ValueInt64(42)))), @@ -1470,7 +1470,7 @@ class EngineTest "be reinterpreted to the same node when lookup doesn't find a contract" in { val exerciseCmd = ApiCommand.Exercise( - TemplateOrInterface.Template(lookerUpTemplateId), + lookerUpTemplateId, lookerUpCid, "Lookup", ValueRecord(None, ImmArray((Some[Name]("n"), ValueInt64(57)))), @@ -1747,7 +1747,7 @@ class EngineTest .preprocessApiCommands( ImmArray( ApiCommand.Exercise( - TemplateOrInterface.Template(fetcherTemplateId), + fetcherTemplateId, fetcherCid, "Fetch", ValueRecord(None, ImmArray((Some[Name]("n"), ValueInt64(42)))), @@ -1814,21 +1814,21 @@ class EngineTest val lookupContract = contracts.get _ val correctCommand = ApiCommand.Exercise( - TemplateOrInterface.Template(withKeyId), + withKeyId, cid, "SumToK", ValueRecord(None, ImmArray((None, ValueInt64(42)))), ) val incorrectCommand = ApiCommand.Exercise( - TemplateOrInterface.Template(simpleId), + simpleId, cid, "Hello", ValueRecord(None, ImmArray.Empty), ) val incorrectFetch = ApiCommand.Exercise( - TemplateOrInterface.Template(fetcherId), + fetcherId, fetcherCid, "DoFetch", ValueRecord(None, ImmArray((None, ValueContractId(cid)))), diff --git a/daml-lf/engine/src/test/scala/com/digitalasset/daml/lf/engine/InterfacesTest.scala b/daml-lf/engine/src/test/scala/com/digitalasset/daml/lf/engine/InterfacesTest.scala index 0834b9617693..e006defe092a 100644 --- a/daml-lf/engine/src/test/scala/com/digitalasset/daml/lf/engine/InterfacesTest.scala +++ b/daml-lf/engine/src/test/scala/com/digitalasset/daml/lf/engine/InterfacesTest.scala @@ -117,7 +117,7 @@ class InterfacesTest /* generic exercise tests */ "be able to exercise interface I1 on a T1 contract" in { val command = ApiCommand.Exercise( - TemplateOrInterface.Interface(idI1), + idI1, cid1, "C1", ValueRecord(None, ImmArray.empty), @@ -126,7 +126,7 @@ class InterfacesTest } "be able to exercise interface I1 on a T2 contract" in { val command = ApiCommand.Exercise( - TemplateOrInterface.Interface(idI1), + idI1, cid2, "C1", ValueRecord(None, ImmArray.empty), @@ -135,7 +135,7 @@ class InterfacesTest } "be able to exercise interface I2 on a T2 contract" in { val command = ApiCommand.Exercise( - TemplateOrInterface.Interface(idI2), + idI2, cid2, "C2", ValueRecord(None, ImmArray.empty), @@ -144,7 +144,7 @@ class InterfacesTest } "be unable to exercise interface I2 on a T1 contract" in { val command = ApiCommand.Exercise( - TemplateOrInterface.Interface(idI2), + idI2, cid1, "C2", ValueRecord(None, ImmArray.empty), @@ -157,7 +157,7 @@ class InterfacesTest } "be able to exercise T1 by interface I1" in { val command = ApiCommand.Exercise( - TemplateOrInterface.Interface(idI1), + idI1, cid1, "C1", ValueRecord(None, ImmArray.empty), @@ -166,7 +166,7 @@ class InterfacesTest } "be able to exercise T2 by interface I1" in { val command = ApiCommand.Exercise( - TemplateOrInterface.Interface(idI1), + idI1, cid2, "C1", ValueRecord(None, ImmArray.empty), @@ -175,7 +175,7 @@ class InterfacesTest } "be able to exercise T2 by interface I2" in { val command = ApiCommand.Exercise( - TemplateOrInterface.Interface(idI2), + idI2, cid2, "C2", ValueRecord(None, ImmArray.empty), @@ -184,7 +184,7 @@ class InterfacesTest } "be unable to exercise T1 by interface I2 (stopped in preprocessor)" in { val command = ApiCommand.Exercise( - TemplateOrInterface.Interface(idT1), + idT1, cid1, "C2", ValueRecord(None, ImmArray.empty), @@ -194,7 +194,7 @@ class InterfacesTest "be able to exercise T2 by interface I5 and usedPackages should include I5's packageId" in { val command = ApiCommand.Exercise( - TemplateOrInterface.Interface(idI5), + idI5, cid2, "C5", ValueRecord(None, ImmArray.empty), diff --git a/daml-lf/engine/src/test/scala/com/digitalasset/daml/lf/engine/LargeTransactionTest.scala b/daml-lf/engine/src/test/scala/com/digitalasset/daml/lf/engine/LargeTransactionTest.scala index c3b17d86873f..b156fe34b76e 100644 --- a/daml-lf/engine/src/test/scala/com/digitalasset/daml/lf/engine/LargeTransactionTest.scala +++ b/daml-lf/engine/src/test/scala/com/digitalasset/daml/lf/engine/LargeTransactionTest.scala @@ -8,7 +8,7 @@ import java.io.File import com.daml.bazeltools.BazelRunfiles import com.daml.lf.archive.UniversalArchiveDecoder import com.daml.lf.data.Ref._ -import com.daml.lf.data.{FrontStack, ImmArray, Ref, TemplateOrInterface, Time} +import com.daml.lf.data.{FrontStack, ImmArray, Ref, Time} import com.daml.lf.language.Ast import com.daml.lf.scenario.ScenarioLedger import com.daml.lf.transaction.{Node, SubmittedTransaction, VersionedTransaction} @@ -321,7 +321,7 @@ class LargeTransactionTest extends AnyWordSpec with Matchers with BazelRunfiles ): ApiCommand.Exercise = { val choice = "ToListContainer" val emptyArgs = ValueRecord(None, ImmArray.Empty) - ApiCommand.Exercise(TemplateOrInterface.Template(templateId), contractId, choice, (emptyArgs)) + ApiCommand.Exercise(templateId, contractId, choice, (emptyArgs)) } private def toListOfIntContainers( @@ -330,7 +330,7 @@ class LargeTransactionTest extends AnyWordSpec with Matchers with BazelRunfiles ): ApiCommand.Exercise = { val choice = "ToListOfIntContainers" val emptyArgs = ValueRecord(None, ImmArray.Empty) - ApiCommand.Exercise(TemplateOrInterface.Template(templateId), contractId, choice, (emptyArgs)) + ApiCommand.Exercise(templateId, contractId, choice, (emptyArgs)) } private def listUtilCreateCmd(templateId: Identifier): ApiCommand.Create = { @@ -345,7 +345,7 @@ class LargeTransactionTest extends AnyWordSpec with Matchers with BazelRunfiles val choiceDefRef = Identifier(templateId.packageId, qn(s"LargeTransaction:$choice")) val damlList = ValueList(List.range(0L, size.toLong).map(ValueInt64).to(FrontStack)) val choiceArgs = ValueRecord(Some(choiceDefRef), ImmArray((None, damlList))) - ApiCommand.Exercise(TemplateOrInterface.Template(templateId), contractId, choice, choiceArgs) + ApiCommand.Exercise(templateId, contractId, choice, choiceArgs) } private def assertSizeExerciseTransaction( diff --git a/daml-lf/transaction/src/main/scala/com/digitalasset/daml/lf/ApiCommand.scala b/daml-lf/transaction/src/main/scala/com/digitalasset/daml/lf/ApiCommand.scala index a1424c0c32d6..e73a3cb9a74a 100644 --- a/daml-lf/transaction/src/main/scala/com/digitalasset/daml/lf/ApiCommand.scala +++ b/daml-lf/transaction/src/main/scala/com/digitalasset/daml/lf/ApiCommand.scala @@ -6,11 +6,11 @@ package command import com.daml.lf.data.Ref._ import com.daml.lf.value.Value -import com.daml.lf.data.{ImmArray, TemplateOrInterface, Time} +import com.daml.lf.data.{ImmArray, Time} /** Accepted commands coming from API */ sealed abstract class ApiCommand extends Product with Serializable { - def typeId: TemplateOrInterface[TypeConName, TypeConName] + def typeId: TypeConName } object ApiCommand { @@ -21,7 +21,7 @@ object ApiCommand { * @param argument value passed to the template */ final case class Create(templateId: TypeConName, argument: Value) extends ApiCommand { - def typeId: TemplateOrInterface.Template[TypeConName] = TemplateOrInterface.Template(templateId) + override def typeId: TypeConName = templateId } /** Command for exercising a choice on an existing contract @@ -32,7 +32,7 @@ object ApiCommand { * @param argument value passed for the choice */ final case class Exercise( - typeId: TemplateOrInterface[TypeConName, TypeConName], + typeId: TypeConName, contractId: Value.ContractId, choiceId: ChoiceName, argument: Value, @@ -51,7 +51,7 @@ object ApiCommand { choiceId: ChoiceName, argument: Value, ) extends ApiCommand { - def typeId: TemplateOrInterface.Template[TypeConName] = TemplateOrInterface.Template(templateId) + override def typeId: TypeConName = templateId } /** Command for creating a contract and exercising a choice @@ -68,7 +68,7 @@ object ApiCommand { choiceId: ChoiceName, choiceArgument: Value, ) extends ApiCommand { - def typeId: TemplateOrInterface.Template[TypeConName] = TemplateOrInterface.Template(templateId) + override def typeId: TypeConName = templateId } } diff --git a/daml-script/runner/src/main/scala/com/digitalasset/daml/lf/engine/script/Converter.scala b/daml-script/runner/src/main/scala/com/digitalasset/daml/lf/engine/script/Converter.scala index 96366145054c..685ef888a779 100644 --- a/daml-script/runner/src/main/scala/com/digitalasset/daml/lf/engine/script/Converter.scala +++ b/daml-script/runner/src/main/scala/com/digitalasset/daml/lf/engine/script/Converter.scala @@ -255,7 +255,7 @@ object Converter { cid <- toContractId(vals.get(1)) anyChoice <- toAnyChoice(vals.get(2)) } yield command.ApiCommand.Exercise( - typeId = TemplateOrInterface.Template(typeId), + typeId = typeId, contractId = cid, choiceId = anyChoice.name, argument = anyChoice.arg.toUnnormalizedValue, diff --git a/daml-script/runner/src/main/scala/com/digitalasset/daml/lf/engine/script/ledgerinteraction/GrpcLedgerClient.scala b/daml-script/runner/src/main/scala/com/digitalasset/daml/lf/engine/script/ledgerinteraction/GrpcLedgerClient.scala index 3fc3a59682bd..29198aa62008 100644 --- a/daml-script/runner/src/main/scala/com/digitalasset/daml/lf/engine/script/ledgerinteraction/GrpcLedgerClient.scala +++ b/daml-script/runner/src/main/scala/com/digitalasset/daml/lf/engine/script/ledgerinteraction/GrpcLedgerClient.scala @@ -313,7 +313,7 @@ class GrpcLedgerClient(val grpcClient: LedgerClient, val applicationId: Applicat } yield Command().withExercise( // TODO: https://github.com/digital-asset/daml/issues/14747 // Fix once the new field interface_id have been added to the API Exercise Command - ExerciseCommand(Some(toApiIdentifier(typeId.merge)), contractId.coid, choice, Some(arg)) + ExerciseCommand(Some(toApiIdentifier(typeId)), contractId.coid, choice, Some(arg)) ) case command.ExerciseByKeyCommand(templateId, key, choice, argument) => for { diff --git a/daml-script/runner/src/main/scala/com/digitalasset/daml/lf/engine/script/ledgerinteraction/JsonLedgerClient.scala b/daml-script/runner/src/main/scala/com/digitalasset/daml/lf/engine/script/ledgerinteraction/JsonLedgerClient.scala index 1e5b5d4e11c8..5230d1980c29 100644 --- a/daml-script/runner/src/main/scala/com/digitalasset/daml/lf/engine/script/ledgerinteraction/JsonLedgerClient.scala +++ b/daml-script/runner/src/main/scala/com/digitalasset/daml/lf/engine/script/ledgerinteraction/JsonLedgerClient.scala @@ -260,9 +260,7 @@ class JsonLedgerClient( case command.CreateCommand(tplId, argument) => create(tplId, argument, partySets) case command.ExerciseCommand(typeId, cid, choice, argument) => - // TODO: https://github.com/digital-asset/daml/issues/14747 - // Fix once Json API distinguish between template and interfaceId within Exercise Command - exercise(typeId.merge, cid, choice, argument, partySets) + exercise(typeId, cid, choice, argument, partySets) case command.ExerciseByKeyCommand(tplId, key, choice, argument) => exerciseByKey(tplId, key, choice, argument, partySets) case command.CreateAndExerciseCommand(tplId, template, choice, argument) => @@ -283,10 +281,10 @@ class JsonLedgerClient( commands: List[command.ApiCommand], optLocation: Option[Location], )(implicit ec: ExecutionContext, mat: Materializer) = { - submit(actAs, readAs, commands, optLocation).map({ + submit(actAs, readAs, commands, optLocation).map { case Right(_) => Left(()) case Left(_) => Right(()) - }) + } } override def submitTree( diff --git a/ledger/ledger-api-common/src/main/scala/com/digitalasset/ledger/api/validation/CommandsValidator.scala b/ledger/ledger-api-common/src/main/scala/com/digitalasset/ledger/api/validation/CommandsValidator.scala index 843707b41072..629c4cc98181 100644 --- a/ledger/ledger-api-common/src/main/scala/com/digitalasset/ledger/api/validation/CommandsValidator.scala +++ b/ledger/ledger-api-common/src/main/scala/com/digitalasset/ledger/api/validation/CommandsValidator.scala @@ -158,9 +158,7 @@ final class CommandsValidator( value <- requirePresence(e.value.choiceArgument, "value") validatedValue <- validateValue(value) } yield ApiCommand.Exercise( - // TODO: https://github.com/digital-asset/daml/issues/14747 - // Fix once the new field interface_id have been added to the API Exercise Command - typeId = TemplateOrInterface.Template(validatedTemplateId), + typeId = validatedTemplateId, contractId = contractId, choiceId = choice, argument = validatedValue, diff --git a/security-evidence.md b/security-evidence.md index 92383d235c94..399cda9acbd4 100644 --- a/security-evidence.md +++ b/security-evidence.md @@ -220,15 +220,15 @@ - contract keys should be evaluated only when executing create: [ContractKeySpec.scala](daml-lf/engine/src/test/scala/com/digitalasset/daml/lf/engine/ContractKeySpec.scala#L149) - ensure builtin operators have the correct type: [TypingSpec.scala](daml-lf/validation/src/test/scala/com/digitalasset/daml/lf/validation/TypingSpec.scala#L48) - ensure expression forms have the correct type: [TypingSpec.scala](daml-lf/validation/src/test/scala/com/digitalasset/daml/lf/validation/TypingSpec.scala#L108) -- exercise-by-interface command is rejected for a: [ApiCommandPreprocessorSpec.scala](daml-lf/engine/src/test/scala/com/digitalasset/daml/lf/engine/ApiCommandPreprocessorSpec.scala#L173) +- exercise-by-interface command is rejected for a: [ApiCommandPreprocessorSpec.scala](daml-lf/engine/src/test/scala/com/digitalasset/daml/lf/engine/ApiCommandPreprocessorSpec.scala#L171) - exercise_interface with a contract instance that does not implement the interface fails.: [EvaluationOrderTest.scala](daml-lf/interpreter/src/test/scala/com/digitalasset/daml/lf/speedy/EvaluationOrderTest.scala#L1638) - ill-formed create API command is rejected: [ApiCommandPreprocessorSpec.scala](daml-lf/engine/src/test/scala/com/digitalasset/daml/lf/engine/ApiCommandPreprocessorSpec.scala#L159) - ill-formed create replay command is rejected: [ReplayCommandPreprocessorSpec.scala](daml-lf/engine/src/test/scala/com/digitalasset/daml/lf/engine/ReplayCommandPreprocessorSpec.scala#L109) -- ill-formed create-and-exercise API command is rejected: [ApiCommandPreprocessorSpec.scala](daml-lf/engine/src/test/scala/com/digitalasset/daml/lf/engine/ApiCommandPreprocessorSpec.scala#L186) +- ill-formed create-and-exercise API command is rejected: [ApiCommandPreprocessorSpec.scala](daml-lf/engine/src/test/scala/com/digitalasset/daml/lf/engine/ApiCommandPreprocessorSpec.scala#L184) - ill-formed exception definitions are rejected: [TypingSpec.scala](daml-lf/validation/src/test/scala/com/digitalasset/daml/lf/validation/TypingSpec.scala#L1816) - ill-formed exercise API command is rejected: [ApiCommandPreprocessorSpec.scala](daml-lf/engine/src/test/scala/com/digitalasset/daml/lf/engine/ApiCommandPreprocessorSpec.scala#L164) - ill-formed exercise replay command is rejected: [ReplayCommandPreprocessorSpec.scala](daml-lf/engine/src/test/scala/com/digitalasset/daml/lf/engine/ReplayCommandPreprocessorSpec.scala#L114) -- ill-formed exercise-by-key API command is rejected: [ApiCommandPreprocessorSpec.scala](daml-lf/engine/src/test/scala/com/digitalasset/daml/lf/engine/ApiCommandPreprocessorSpec.scala#L177) +- ill-formed exercise-by-key API command is rejected: [ApiCommandPreprocessorSpec.scala](daml-lf/engine/src/test/scala/com/digitalasset/daml/lf/engine/ApiCommandPreprocessorSpec.scala#L175) - ill-formed exercise-by-key replay command is rejected: [ReplayCommandPreprocessorSpec.scala](daml-lf/engine/src/test/scala/com/digitalasset/daml/lf/engine/ReplayCommandPreprocessorSpec.scala#L121) - ill-formed expressions are rejected: [TypingSpec.scala](daml-lf/validation/src/test/scala/com/digitalasset/daml/lf/validation/TypingSpec.scala#L450) - ill-formed fetch command is rejected: [ReplayCommandPreprocessorSpec.scala](daml-lf/engine/src/test/scala/com/digitalasset/daml/lf/engine/ReplayCommandPreprocessorSpec.scala#L168)