From 917c913f8d4e205be0ac7c1fcf5f74d9c6dbe986 Mon Sep 17 00:00:00 2001 From: Daniel McGregor Date: Thu, 7 Nov 2024 15:42:52 +0800 Subject: [PATCH] doc: consolidate existing documentation on calling other applications --- docs/language-guide.md | 1 + docs/lg-arc4.md | 66 ------------------- docs/lg-calling-apps.md | 142 ++++++++++++++++++++++++++++++++++++++++ docs/lg-compile.md | 58 ---------------- docs/lg-transactions.md | 39 ----------- 5 files changed, 143 insertions(+), 163 deletions(-) create mode 100644 docs/lg-calling-apps.md diff --git a/docs/language-guide.md b/docs/language-guide.md index 7c3d7cc585..ef035576da 100644 --- a/docs/language-guide.md +++ b/docs/language-guide.md @@ -80,6 +80,7 @@ lg-ops lg-opcode-budget lg-arc4 lg-arc28 +lg-calling-apps lg-compile lg-unsupported-python-features ``` diff --git a/docs/lg-arc4.md b/docs/lg-arc4.md index f2022d061d..a51377fea6 100644 --- a/docs/lg-arc4.md +++ b/docs/lg-arc4.md @@ -221,69 +221,3 @@ Which means if one reference is mutated, the other references would not see the In order to keep the semantics the same, the compiler forces the addition of `.copy()` each time a new reference to the same object to match what will happen on the AVM. Struct types can be indicated as `frozen` which will eliminate the need for a `.copy()` as long as the struct also contains no mutable fields (such as arrays or another mutable struct) - -## Typed clients - -[`arc4.abi_call`](#algopy.arc4.abi_call) can be used to do type safe calls to an ABI method of another contract, these -calls can be expressed in a few ways. - -### ARC4Contract method - -An ARC4Contract method written in Algorand Python can be referenced directly e.g. - -```python -from algopy import arc4, subroutine - -class HelloWorldContract(arc4.ARC4Contract): - - def hello(self, name: arc4.String) -> arc4.String: ... # implementation omitted - -@subroutine -def call_another_contract() -> None: - result, txn = arc4.abi_call(HelloWorldContract.hello, arc4.String("World"), app=...) - assert result == "Hello, World" -``` - -### ARC4Client method - -A ARC4Client client represents the ARC4 abimethods of a smart contract and can be used to call abimethods in a type safe way - -ARC4Client's can be produced by using `puyapy --output-client=True` when compiling a smart contract -(this would be useful if you wanted to publish a client for consumption by other smart contracts) -An ARC4Client can also be be generated from an ARC-32 application.json using `puyapy-clientgen` -e.g. `puyapy-clientgen examples/hello_world_arc4/out/HelloWorldContract.arc32.json`, this would be -the recommended approach for calling another smart contract that is not written in Algorand Python or does not provide the source - -```python -from algopy import arc4, subroutine - -class HelloWorldClient(arc4.ARC4Client): - - def hello(self, name: arc4.String) -> arc4.String: ... - -@subroutine -def call_another_contract() -> None: - # can reference another algopy contract method - result, txn = arc4.abi_call(HelloWorldClient.hello, arc4.String("World"), app=...) - assert result == "Hello, World" -``` - -### Method signature or name - -An ARC4 method selector can be used e.g. `"hello(string)string` along with a type index to specify the return type. -Additionally just a name can be provided and the method signature will be inferred e.g. - -```python -from algopy import arc4, subroutine - - -@subroutine -def call_another_contract() -> None: - # can reference a method selector - result, txn = arc4.abi_call[arc4.String]("hello(string)string", arc4.String("Algo"), app=...) - assert result == "Hello, Algo" - - # can reference a method name, the method selector is inferred from arguments and return type - result, txn = arc4.abi_call[arc4.String]("hello", "There", app=...) - assert result == "Hello, There" -``` diff --git a/docs/lg-calling-apps.md b/docs/lg-calling-apps.md new file mode 100644 index 0000000000..966ffe1650 --- /dev/null +++ b/docs/lg-calling-apps.md @@ -0,0 +1,142 @@ +# Calling other applications + +The preferred way to call other smart contracts is using [`algopy.arc4.abi_call`](#algopyarc4abi_call), [`algopy.arc4.arc4_create`](#algopyarc4arc4_create) or +[`algopy.arc4.arc4_update`](#algopyarc4arc4_update). These methods support type checking and encoding of arguments, decoding of results, group transactions, +and in the case of `arc4_create` and `arc4_update` automatic inclusion of approval and clear state programs. + +## `algopy.arc4.abi_call` + +[`algopy.arc4.abi_call`](#algopy.arc4.abi_call) can be used to call other ARC4 contracts, the first argument should refer to +an ARC4 method either by referencing an Algorand Python [`algopy.arc4.ARC4Contract`](#algopy.arc4.ARC4Contract) method, +an [`algopy.arc4.ARC4Client`](#algopy.arc4.ARC4Client) method generated from an ARC-32 app spec, or a string representing +the ARC4 method signature or name. +The following arguments should then be the arguments required for the call, these arguments will be type checked and converted where appropriate. +Any other related transaction parameters such as `app_id`, `fee` etc. can also be provided as keyword arguments. + +If the ARC4 method returns an ARC4 result then the result will be a tuple of the ARC4 result and the inner transaction. +If the ARC4 method does not return a result, or if the result type is not fully qualified then just the inner transaction is returned. + +```python +from algopy import Application, ARC4Contract, String, arc4, subroutine + +class HelloWorld(ARC4Contract): + + @arc4.abimethod() + def greet(self, name: String) -> String: + return "Hello " + name + +@subroutine +def call_existing_application(app: Application) -> None: + greeting, greet_txn = arc4.abi_call(HelloWorld.greet, "there", app_id=app) + + assert greeting == "Hello there" + assert greet_txn.app_id == 1234 +``` + + +### Alternative ways to use `arc4.abi_call` + +#### ARC4Client method + +A ARC4Client client represents the ARC4 abimethods of a smart contract and can be used to call abimethods in a type safe way + +ARC4Client's can be produced by using `puyapy --output-client=True` when compiling a smart contract +(this would be useful if you wanted to publish a client for consumption by other smart contracts) +An ARC4Client can also be be generated from an ARC-32 application.json using `puyapy-clientgen` +e.g. `puyapy-clientgen examples/hello_world_arc4/out/HelloWorldContract.arc32.json`, this would be +the recommended approach for calling another smart contract that is not written in Algorand Python or does not provide the source + +```python +from algopy import arc4, subroutine + +class HelloWorldClient(arc4.ARC4Client): + + def hello(self, name: arc4.String) -> arc4.String: ... + +@subroutine +def call_another_contract() -> None: + # can reference another algopy contract method + result, txn = arc4.abi_call(HelloWorldClient.hello, arc4.String("World"), app=...) + assert result == "Hello, World" +``` + +#### Method signature or name + +An ARC4 method selector can be used e.g. `"hello(string)string` along with a type index to specify the return type. +Additionally just a name can be provided and the method signature will be inferred e.g. + +```python +from algopy import arc4, subroutine + + +@subroutine +def call_another_contract() -> None: + # can reference a method selector + result, txn = arc4.abi_call[arc4.String]("hello(string)string", arc4.String("Algo"), app=...) + assert result == "Hello, Algo" + + # can reference a method name, the method selector is inferred from arguments and return type + result, txn = arc4.abi_call[arc4.String]("hello", "There", app=...) + assert result == "Hello, There" +``` + + +## `algopy.arc4.arc4_create` + +[`algopy.arc4.arc4_create`](#algopy.arc4.arc4_create) can be used to create ARC4 applications, and will automatically populate required fields for app creation (such as approval program, clear state program, and global/local state allocation). + +Like [`algopy.arc4.abi_call`](lg-transactions.md#arc4-application-calls) it handles ARC4 arguments and provides ARC4 return values. + +If the compiled programs and state allocation fields need to be customized (for example due to [template variables](#within-other-contracts)), +this can be done by passing a [`algopy.CompiledContract`](#algopy.CompiledContract) via the `compiled` keyword argument. + +```python +from algopy import ARC4Contract, String, arc4, subroutine + +class HelloWorld(ARC4Contract): + + @arc4.abimethod() + def greet(self, name: String) -> String: + return "Hello " + name + +@subroutine +def create_new_application() -> None: + hello_world_app = arc4.arc4_create(HelloWorld).created_app + + greeting, _txn = arc4.abi_call(HelloWorld.greet, "there", app_id=hello_world_app) + + assert greeting == "Hello there" +``` + +## `algopy.arc4.arc4_update` + +[`algopy.arc4.arc4_update`](#algopy.arc4.arc4_update) is used to update an existing ARC4 application and will automatically populate the required approval and clear state program fields. + +Like [`algopy.arc4.abi_call`](lg-transactions.md#arc4-application-calls) it handles ARC4 arguments and provides ARC4 return values. + +If the compiled programs need to be customized (for example due to (for example due to [template variables](#within-other-contracts)), +this can be done by passing a [`algopy.CompiledContract`](#algopy.CompiledContract) via the `compiled` keyword argument. + +```python +from algopy import Application, ARC4Contract, String, arc4, subroutine + +class NewApp(ARC4Contract): + + @arc4.abimethod() + def greet(self, name: String) -> String: + return "Hello " + name + +@subroutine +def update_existing_application(existing_app: Application) -> None: + hello_world_app = arc4.arc4_update(NewApp, app_id=existing_app) + + greeting, _txn = arc4.abi_call(NewApp.greet, "there", app_id=hello_world_app) + + assert greeting == "Hello there" +``` + +## Using `itxn.ApplicationCall` + +If the application being called is not an ARC4 contract, or an application specification is not available, +then [`algopy.itxn.ApplicationCall`](#algopy.itxn.ApplicationCall) can be used. This approach is generally more verbose +than the above approaches, so should only be used if required. See [here](./lg-transactions.md#create-an-arc4-application-and-then-call-it) for an example diff --git a/docs/lg-compile.md b/docs/lg-compile.md index e6182e127a..6062823fa8 100644 --- a/docs/lg-compile.md +++ b/docs/lg-compile.md @@ -16,64 +16,6 @@ This compiled contract can then be used to create an [`algopy.itxn.ApplicationCa The [`compile_logicsig`](#algopy.compile_logicsig) takes an Algorand Python logic signature and returns a [`CompiledLogicSig`](#algopy.CompiledLogicSig), which can be used to verify if a transaction has been signed by a particular logic signature. -## ARC4 contracts - -Additional functions are available for [creating](lg-compile.md#create) and [updating](lg-compile.md#update) ARC4 applications on-chain via an inner transaction. - -### Create - -[`algopy.arc4.arc4_create`](#algopy.arc4.arc4_create) can be used to create ARC4 applications, and will automatically populate required fields for app creation (such as approval program, clear state program, and global/local state allocation). - -Like [`algopy.arc4.abi_call`](lg-transactions.md#arc4-application-calls) it handles ARC4 arguments and provides ARC4 return values. - -If the compiled programs and state allocation fields need to be customized (for example due to [template variables](#within-other-contracts)), -this can be done by passing a [`algopy.CompiledContract`](#algopy.CompiledContract) via the `compiled` keyword argument. - -```python -from algopy import ARC4Contract, String, arc4, subroutine - -class HelloWorld(ARC4Contract): - - @arc4.abimethod() - def greet(self, name: String) -> String: - return "Hello " + name - -@subroutine -def create_new_application() -> None: - hello_world_app = arc4.arc4_create(HelloWorld).created_app - - greeting, _txn = arc4.abi_call(HelloWorld.greet, "there", app_id=hello_world_app) - - assert greeting == "Hello there" -``` - -### Update - -[`algopy.arc4.arc4_update`](#algopy.arc4.arc4_update) is used to update an existing ARC4 application and will automatically populate the required approval and clear state program fields. - -Like [`algopy.arc4.abi_call`](lg-transactions.md#arc4-application-calls) it handles ARC4 arguments and provides ARC4 return values. - -If the compiled programs need to be customized (for example due to (for example due to [template variables](#within-other-contracts)), -this can be done by passing a [`algopy.CompiledContract`](#algopy.CompiledContract) via the `compiled` keyword argument. - -```python -from algopy import Application, ARC4Contract, String, arc4, subroutine - -class NewApp(ARC4Contract): - - @arc4.abimethod() - def greet(self, name: String) -> String: - return "Hello " + name - -@subroutine -def update_existing_application(existing_app: Application) -> None: - hello_world_app = arc4.arc4_update(NewApp, app_id=existing_app) - - greeting, _txn = arc4.abi_call(NewApp.greet, "there", app_id=hello_world_app) - - assert greeting == "Hello there" -``` - ## Template variables Algorand Python supports defining [`algopy.TemplateVar`](#algopy.TemplateVar) variables that can be substituted during compilation. diff --git a/docs/lg-transactions.md b/docs/lg-transactions.md index 4d24a95637..362ac1744d 100644 --- a/docs/lg-transactions.md +++ b/docs/lg-transactions.md @@ -149,14 +149,6 @@ def example() -> None: ).submit() # extract result hello_world_result = arc4.String.from_log(call_txn.last_log) - - # OR, call it automatic ARC4 encoding, type validation and result handling - hello_world_result, call_txn = arc4.abi_call[arc4.String]( # declare return type - "hello(string)string", # method signature to call - "again", # abi method arguments - fee=0, # other transaction parameters - app_id=app - ) ``` #### Create and submit transactions in a loop @@ -175,37 +167,6 @@ def example(receivers: tuple[Account, Account, Account]) -> None: ).submit() ``` -### ARC4 Application calls - -#### `algopy.arc4.abi_call` - -[`algopy.arc4.abi_call`](#algopy.arc4.abi_call) can be used to call other ARC4 contracts, the first argument should refer to -an ARC4 method either by referencing an Algorand Python [`algopy.arc4.ARC4Contract`](#algopy.arc4.ARC4Contract) method, -an [`algopy.arc4.ARC4Client`](#algopy.arc4.ARC4Client) method generated from an ARC-32 app spec, or a string representing -the ARC4 method signature or name. -The following arguments should then be the arguments required for the call, these arguments will be type checked and converted where appropriate. -Any other related transaction parameters such as `app_id`, `fee` etc. can also be provided as keyword arguments. - -If the ARC4 method returns an ARC4 result then the result will be a tuple of the ARC4 result and the inner transaction. -If the ARC4 method does not return a result, or if the result type is not fully qualified then just the inner transaction is returned. - -```python -from algopy import Application, ARC4Contract, String, arc4, subroutine - -class HelloWorld(ARC4Contract): - - @arc4.abimethod() - def greet(self, name: String) -> String: - return "Hello " + name - -@subroutine -def call_existing_application(app: Application) -> None: - greeting, greet_txn = arc4.abi_call(HelloWorld.greet, "there", app_id=app) - - assert greeting == "Hello there" - assert greet_txn.app_id == 1234 -``` - ### Limitations Inner transactions are powerful, but currently do have some restrictions in how they are used.