Transform Data Using Choices¶
In the example in Contract Keys the accountant party wanted to change some data on a contract. They did so by archiving the contract and re-creating it with the updated data. That works because the accountant is the sole signatory on the Account contract defined there.
But what if the accountant wanted to allow the bank to change their own telephone number? Or what if the owner of a CashBalance should be able to transfer ownership to someone else?
In this section you will learn about how to define simple data transformations using choices and how to delegate the right to exercise these choices to other parties.
Hint
Remember that you can load all the code for this section into a folder called intro4 by running daml new intro4 --template daml-intro-4
Choices as Methods¶
If you think of templates as classes and contracts as objects, where are the methods?
Take as an example a Contact contract on which the contact owner wants to be able to change the telephone number, just like on the Account in Contract Keys. Rather than requiring them to manually look up the contract, archive the old one and create a new one, you can provide them a convenience method on Contact:
template Contact
  with
    owner : Party
    party : Party
    address : Text
    telephone : Text
  where
    signatory owner
    observer party
    choice UpdateTelephone
      : ContractId Contact
      with
        newTelephone : Text
      controller owner
      do
        create this with
          telephone = newTelephone
The above defines a choice called UpdateTelephone. Choices are part of a contract template. They’re permissioned functions that result in an Update. Using choices, authority can be passed around, allowing the construction of complex transactions.
Let’s unpack the code snippet above:
- The first line, - choice UpdateTelephoneindicates a choice definition,- UpdateTelephoneis the name of the choice. It starts a new block in which that choice is defined.
- : ContractId Contactis the return type of the choice.- This particular choice archives the current - Contact, and creates a new one. What it returns is a reference to the new contract, in the form of a- ContractId Contact
- The following - withblock is that of a record. Just like with templates, in the background, a new record type is declared:- data UpdateTelephone = UpdateTelephone with
- The line - controller ownersays that this choice is controlled by- owner, meaning- owneris the only party that is allowed to exercise them.
- The - dostarts a block defining the action the choice should perform when exercised. In this case a new- Contactis created.
- The new - Contactis created using- this with.- thisis a special value available within the- whereblock of templates and takes the value of the current contract’s arguments.
There is nothing here explicitly saying that the current Contact should be archived. That’s because choices are consuming by default. That means when the above choice is exercised on a contract, that contract is archived.
As mentioned in Data Types, within a choice we use create
instead of createCmd. Whereas createCmd builds up a list of
commands to be sent to the ledger, create builds up a more
flexible Update that is executed directly by the ledger. You might
have noticed that create returns an Update (ContractId
Contact), not a ContractId Contact. As a do block always
returns the value of the last statement within it, the whole do
block returns an Update, but the return type on the choice is just
a ContractId Contact. This is a convenience. Choices always
return an Update so for readability it’s omitted on the type
declaration of a choice.
Now to exercise the new choice in a script:
choice_test = do
  owner <- allocateParty "Alice"
  party <- allocateParty "Bob"
  contactCid <- submit owner do
     createCmd Contact with
      owner
      party
      address = "1 Bobstreet"
      telephone = "012 345 6789"
  -- Bob can't change his own telephone number as Alice controls
  -- that choice.
  submitMustFail party do
    exerciseCmd contactCid UpdateTelephone with
      newTelephone = "098 7654 321"
  newContactCid <- submit owner do
    exerciseCmd contactCid UpdateTelephone with
      newTelephone = "098 7654 321"
  Some newContact <- queryContractId owner newContactCid
  assert (newContact.telephone == "098 7654 321")
You exercise choices using the exercise function, which takes a ContractId a, and a value of type c, where c is a choice on template a. Since c is just a record, you can also just fill in the choice parameters using the with syntax you are already familiar with.
exerciseCmd returns a Commands r where r is the return
type specified on the choice, allowing the new ContractId Contact
to be stored in the variable newContactCid. Just like for
createCmd and create, there is also exerciseCmd and
exercise. The versions with the cmd suffix is always used on
the client side to build up the list of commands on the ledger. The
versions without the suffix are used within choices and are executed
directly on the server.
There is also createAndExerciseCmd and createAndExercise which
we have seen in the previous section. This allows you to create a new
contract with the given arguments and immediately exercise a choice on
it. For a consuming choice, this archives the contract so the contract
is created and archived within the same transaction.
Choices as Delegation¶
Up to this point all the contracts only involved one party. party may have been stored as Party field in the above, which suggests they are actors on the ledger, but they couldn’t see the contracts, nor change them in any way. It would be reasonable for the party for which a Contact is stored to be able to update their own address and telephone number. In other words, the owner of a Contact should be able to delegate the right to perform a certain kind of data transformation to party.
The below demonstrates this using an UpdateAddress choice and corresponding extension of the script:
    choice UpdateAddress
      : ContractId Contact
      with
        newAddress : Text
      controller party
      do
        create this with
          address = newAddress
  newContactCid <- submit party do
    exerciseCmd newContactCid UpdateAddress with
      newAddress = "1-10 Bobstreet"
  Some newContact <- queryContractId owner newContactCid
  assert (newContact.address == "1-10 Bobstreet")
If you open the script view in the IDE, you will notice that Bob sees the Contact. This is because party is specified as an observer in the template, and in this case Bob is the party. More on observers later, but in short, they get to see any changes to the contract.
Choices In the Ledger Model¶
In Basic Contracts you learned about the high-level structure of a Daml ledger. With choices and the exercise function, you have the next important ingredient to understand the structure of the ledger and transactions.
A transaction is a list of actions, and there are just four kinds of action: create, exercise, fetch and key assertion.
- A createaction creates a new contract with the given arguments and sets its status to active.
- A fetchaction checks the existence and activeness of a contract.
- An exerciseaction exercises a choice on a contract resulting in a transaction (list of sub-actions) called the consequences. Exercises come in two kinds calledconsumingandnonconsuming.consumingis the default kind and changes the contract’s status from active to archived.
- A key assertionrecords the assertion that the given contract key (see Contract Keys) is not assigned to any active contract on the ledger.
Each action can be visualized as a tree, where the action is the root node, and its children are its consequences. Every consequence may have further consequences. As fetch, create and key assertion actions have no consequences, they are always leaf nodes. You can see the actions and their consequences in the transaction view of the above script:
Transactions:
  TX 0 1970-01-01T00:00:00Z (Contact:46:17)
  #0:0
  │   consumed by: #2:0
  │   referenced by #2:0
  │   disclosed to (since): 'Alice' (0), 'Bob' (0)
  └─> create Contact:Contact
      with
        owner = 'Alice'; party = 'Bob'; address = "1 Bobstreet"; telephone = "012 345 6789"
  TX 1 1970-01-01T00:00:00Z
    mustFailAt actAs: {'Bob'} readAs: {} (Contact:55:3)
  TX 2 1970-01-01T00:00:00Z (Contact:59:20)
  #2:0
  │   disclosed to (since): 'Alice' (2), 'Bob' (2)
  └─> 'Alice' exercises UpdateTelephone on #0:0 (Contact:Contact)
              with
                newTelephone = "098 7654 321"
      children:
      #2:1
      │   consumed by: #3:0
      │   referenced by #3:0
      │   disclosed to (since): 'Alice' (2), 'Bob' (2)
      └─> create Contact:Contact
          with
            owner = 'Alice'; party = 'Bob'; address = "1 Bobstreet"; telephone = "098 7654 321"
  TX 3 1970-01-01T00:00:00Z (Contact:69:20)
  #3:0
  │   disclosed to (since): 'Alice' (3), 'Bob' (3)
  └─> 'Bob' exercises UpdateAddress on #2:1 (Contact:Contact)
            with
              newAddress = "1-10 Bobstreet"
      children:
      #3:1
      │   disclosed to (since): 'Alice' (3), 'Bob' (3)
      └─> create Contact:Contact
          with
            owner = 'Alice';
            party = 'Bob';
            address = "1-10 Bobstreet";
            telephone = "098 7654 321"
Active contracts:  #3:1
Return value: {}
There are four commits corresponding to the four submit statements in the script. Within each commit, we see that it’s actually actions that have IDs of the form #commit_number:action_number. Contract IDs are just the ID of their create action.
So commits #2 and #3 contain exercise actions with IDs #2:0 and #3:0. The create actions of the updated Contact contracts,  #2:1 and #3:1, are indented and found below a line reading children:, making the tree structure apparent.
The Archive Choice¶
You may have noticed that there is no archive action. That’s because archive cid is just shorthand for exercise cid Archive, where Archive is a choice implicitly added to every template, with the signatories as controllers.
A Simple Cash Model¶
With the power of choices, you can build your first interesting model: issuance of cash IOUs (I owe you). The model presented here is simpler than the one in Data Types as it’s not concerned with the location of the physical cash, but merely with liabilities:
-- Copyright (c) 2022 Digital Asset (Switzerland) GmbH and/or its affiliates. All rights reserved.
-- SPDX-License-Identifier: Apache-2.0
module SimpleIou where
import Daml.Script
data Cash = Cash with
  currency : Text
  amount : Decimal
    deriving (Eq, Show)
template SimpleIou
  with
    issuer : Party
    owner : Party
    cash : Cash
  where
    signatory issuer
    observer owner
    choice Transfer
      : ContractId SimpleIou
      with
        newOwner : Party
      controller owner
      do
        create this with owner = newOwner
test_iou = script do
  alice <- allocateParty "Alice"
  bob <- allocateParty "Bob"
  charlie <- allocateParty "Charlie"
  dora <- allocateParty "Dora"
  -- Dora issues an Iou for $100 to Alice.
  iou <- submit dora do
    createCmd SimpleIou with
      issuer = dora
      owner = alice
      cash = Cash with
        amount = 100.0
        currency = "USD"
  -- Alice transfers it to Bob.
  iou2 <- submit alice do
    exerciseCmd iou Transfer with
      newOwner = bob
  -- Bob transfers it to Charlie.
  submit bob do
    exerciseCmd iou2 Transfer with
      newOwner = charlie
The above model is fine as long as everyone trusts Dora. Dora could revoke the SimpleIou at any point by archiving it. However, the provenance of all transactions would be on the ledger so the owner could prove that Dora was dishonest and cancelled her debt.
Next Up¶
You can now store and transform data on the ledger, even giving other parties specific write access through choices.
In Add Constraints to a Contract, you will learn how to restrict data and transformations further. In that context, you will also learn about time on Daml ledgers, do blocks and <- notation within those.