diff --git a/src/pages/smart-contracts/clarity-values-guide.md b/src/pages/smart-contracts/clarity-values-guide.md index f0b4cc04..bf270ef0 100644 --- a/src/pages/smart-contracts/clarity-values-guide.md +++ b/src/pages/smart-contracts/clarity-values-guide.md @@ -8,7 +8,7 @@ tags: ## Introduction The Clarity language makes use of a strong static [type system](https://docs.blockstack.org/references/language-clarity#clarity-type-system). This simply means that every function defined in Clarity expects arguments of specific types, and that a failure to provide properly typed arguments will result in your code failing to compile, or your `contract-call` transaction failing prior to execution. -In order to build web applications that interact with Clarity contracts, we must include type information as a part of our function calls. The [@stacks/transactions](https://github.com/blockstack/stacks.js/tree/master/packages/transactions) library makes this easy, as we will demonstrate below. +In order to build web applications that interact with Clarity contracts, you will need to learn how to construct and use `ClarityValues`. The [@stacks/transactions](https://github.com/blockstack/stacks.js/tree/master/packages/transactions) library makes this easy, as we will demonstrate below. ## Clarity Types Values in Clarity can have the following types: @@ -25,105 +25,224 @@ Values in Clarity can have the following types: - `int` - signed 128-bit integer - `uint` - unsigned 128-bit integer -## Constructing Clarity Values -Clarity values can be constructed with functions provided by the **@stacks/transactions** library. These function simply output javascript objects that contain a value and a numerical representation of the Clarity type information. These Clarity value objects can then be easily serialized into `contract-call` transactions for interacting with Clarity contracts. +## Constructing Clarity Values and accessing their data +Clarity values can be constructed with functions provided by the **@stacks/transactions** library. These functions simply output javascript objects that contain a value and a numerical representation of the Clarity type information. -Here are examples of how to construct each type of Clarity value: +```typescript +export enum ClarityType { + Int = 0x00, + UInt = 0x01, + Buffer = 0x02, + BoolTrue = 0x03, + BoolFalse = 0x04, + PrincipalStandard = 0x05, + PrincipalContract = 0x06, + ResponseOk = 0x07, + ResponseErr = 0x08, + OptionalNone = 0x09, + OptionalSome = 0x0a, + List = 0x0b, + Tuple = 0x0c, + StringASCII = 0x0d, + StringUTF8 = 0x0e, +} +``` + +`ClarityValue` objects can be serialized and included in transactions that interact with published Clarity contracts. + +Here are examples of how to construct each type of Clarity value, and how to access its data if it has any: ### Booleans -```javascript +```typescript const t = trueCV(); const f = falseCV(); ``` + +Boolean Clarity Values don't contain any underlying data. They are simply objects with `type` information. + ### Optional Values -```javascript -const nothing = noneCV(); -const something = someCV(t); +```typescript +const nothing: NoneCV = noneCV(); +const something: SomeCV = someCV(trueCV()); + +console.log(something.value); +// { type: ClarityType.BoolTrue } +``` + +Optional Clarity Values can either be nothing (an empty type that has no data), or something (a wrapped value). + +If you are dealing with a function or contract function that returns an `OptionalCV`, you should always check what type it is before trying to access its value. + +```typescript +const maybeVal: OptionalCV = await callReadOnlyFunction(...); + +if (maybeVal.type === ClarityType.OptionalSome) { + console.log(maybeVal.value); +} else if (maybeVal.type === ClarityType.OptionalNone) { + // deal with `none` value +} ``` ### Buffers -```javascript +```typescript const buffer = Buffer.from('foo'); -const bufCV = bufferCV(buffer); +const bufCV: BufferCV = bufferCV(buffer); + +console.log(bufCV.buffer); +// ``` ### Integers -```javascript -const i = intCV(-10); -const u = uintCV(10); +```typescript +const i: IntCV = intCV(-10); +const u: UIntCV = uintCV(10); + +console.log(i.value); +// -10 + +console.log(u.value); +// 10 ``` +Clarity supports both integers and unsigned integers. + ### Strings -```javascript -const ascii = stringAsciiCV('hello world'); -const utf8 = stringUtf8CV('hello 🌾'); +```typescript +const ascii: StringAsciiCV = stringAsciiCV('hello world'); +const utf8: StringUtf8CV = stringUtf8CV('hello 🌾'); + +console.log(ascii.data); +// 'hello world' + +console.log(utf8.data); +// 'hello 🌾' ``` +Clarity supports both ascii and utf8 strings. + ### Principals -```javascript +```typescript const address = 'SP2JXKMSH007NPYAQHKJPQMAQYAD90NQGTVJVQ02B'; const contractName = 'contract-name'; const spCV = standardPrincipalCV(address); const cpCV = contractPrincipalCV(address, contractName); + +console.log(spCV.address); +// { +// type: ClarityType.PrincipalStandard +// address: { +// type: StacksMessageType.Address, +// version: AddressVersion.MainnetSingleSig, +// hash160: "SP2JXKMSH007NPYAQHKJPQMAQYAD90NQGTVJVQ02B" +// } +// } + +console.log(cpCV.address); +// { +// type: ClarityType.PrincipalContract,j +// contractName: { +// type: StacksMessageType.LengthPrefixedString, +// content: 'contract-name', +// lengthPrefixBytes: 1, +// maxLengthBytes: 128, +// }, +// address: { +// type: StacksMessageType.Address, +// version: AddressVersion.MainnetSingleSig, +// hash160: "SP2JXKMSH007NPYAQHKJPQMAQYAD90NQGTVJVQ02B" +// } +// } ``` +Both kinds of Clarity principal values contain `type` information and an `address` object. Contract principals also contain a `contractName`. + ### Response Values -```javascript +```typescript const errCV = responseErrorCV(trueCV()); const okCV = responseOkCV(falseCV()); + +console.log(errCV.value); +// { type: ClarityType.BoolTrue } + +console.log(okCV.value); +// { type: ClarityType.BoolFalse } ``` +Response Clarity Values will either have the type `ClarityType.ResponseOk` or `ClarityType.ResponseErr`. They both contain a Clarity Value. Often this value will be an integer error code if the response is an `Error`. + ### Tuples -```javascript +```typescript const tupCV = tupleCV({ 'a': intCV(1), 'b': trueCV(), 'c': falseCV() }) + +console.log(tupCV.data['b']); +// { type: ClarityType.BoolTrue } ``` +Tuples in Clarity are typed and contain named fields. The tuple above, for example, contains three fields with the names **a**, **b** and **c**, and the types of their values are `Int`, `Boolean` and `Boolean`, respectively. + +Clarity tuples are represented in JavaScript as objects and a tuple's data can be accessed by its `data` field, where the underlying JS object is stored. + ### Lists -```javascript +```typescript const l = listCV([trueCV(), falseCV()]) + +console.log(l.list[0]); +// { type: ClarityType.BoolTrue } ``` +Lists, in Clarity, are homogeneous, meaning they can only contain elements of a singular (Clarity) type. Make sure to avoid constructing lists that have elements of multiple types. + +A Clarity lists underlying data can be accessed via its `list` field. + ## Using Clarity Values -Now that you know how to construct Clarity values, you can use them to build `contract-call` transactions that call smart contract functions. +Now that you know how to construct _and_ deconstruct Clarity values, you can use them to build `contract-call` transactions that call smart contract functions. -Here's an example of how this is done: +This is covered [here](https://docs.blockstack.org/stacks-blockchain/transactions#construction). -```javascript -import { - makeContractCall, - uintCV, - stringAsciiCV, - StacksMainnet, - broadcastTransaction -} from '@stacks/transactions'; - -const network = new StacksMainnet(); - -const txOptions = { - contractAddress: 'SPBMRFRPPGCDE3F384WCJPK8PQJGZ8K9QKK7F59X', - contractName: 'contract_name', - functionName: 'contract_function', - functionArgs: [uintCV(10), stringAsciiCV('test')], - senderKey: 'b244296d5907de9864c0b0d51f98a13c52890be0404e83f273144cd5b9960eed01', - validateWithAbi: true, +## Utilizing Clarity Values from Transaction Responses +`Read-only` Clarity functions can return Clarity values as a response. These `read-only` functions can be accessed easily in JavaScript via the [`callReadOnlyFunction()`](https://github.com/blockstack/stacks.js/tree/master/packages/transactions#calling-read-only-contract-function) function included in `@stacks/transactions`, which returns a `ClarityValue`. + +As mentioned above, `ClarityValues` are simply javascript objects containing a value and its associated Clarity type information. These object types are defined [here](https://github.com/blockstack/stacks.js/tree/1f2b5fd8bdf1c2b5866e8171163594d7708a8c7a/packages/transactions/src/clarity/types). + +When you are calling a `read-only` contract function, you will always know what type the function will return, since functions in Clarity are strongly typed. + +It is common for Clarity functions to return values wrapped in a `Response`, in order to indicate if there was success or an error. + +Since every `ClarityValue` has a `type` field, the type of the result of a `read-only` function call can be checked and acted upon like so: + +```typescript +const contractAddress = 'ST3KC0MTNW34S1ZXD36JYKFD3JJMWA01M55DSJ4JE'; +const contractName = 'kv-store'; +const functionName = 'get-value'; +const buffer = bufferCVFromString('foo'); +const network = new StacksTestnet(); +const senderAddress = 'ST2F4BK4GZH6YFBNHYDDGN4T1RKBA7DA1BJZPJEJJ'; + +const options = { + contractAddress, + contractName, + functionName, + functionArgs: [buffer], network, + senderAddress, }; -const transaction = await makeContractCall(txOptions); +// make a read-only call to a contract function that +// returns a Response +const result: ResponseCV = await callReadOnlyFunction(options); -broadcastTransaction(transaction, network); +if (result.type === ClarityType.ResponseOk) { + console.log(cvToString(result.value)); +} else if (result.type === ClarityType.ResponseErr) { + throw new Error(`kv-store contract error: ${result.value.data}`); +} ``` -## Utilizing Clarity Values from Transaction Responses -If you successfully submit a `contract-call` transaction that calls a Clarity function, you may receive a resulting Clarity value in response. - -If using the transaction library's `callReadOnlyFunction()` method, the result will be a `ClarityValue`. As mentioned above, `ClarityValues` are simply javascript objects containing a value and its associated Clarity type information. These object types are defined [here](https://github.com/blockstack/stacks.js/tree/1f2b5fd8bdf1c2b5866e8171163594d7708a8c7a/packages/transactions/src/clarity/types). - ### Deserializing Clarity Values from Hex If you receive a response from a transaction in the form of a Hex string, you can deserialize it into a Clarity value like so: @@ -133,11 +252,27 @@ import { hexToCV } from '@stacks/transactions'; let cv = hexToCV('hex_string'); ``` -## Debugging with Clarity Values -Sometimes you might receive as a response a Clarity value that you were not expecting. Logging the value to your console won't always prove to be useful, unless you have memorized the Clarity value type encodings. In order to figure out what kind of value you are dealing with, you can use the `cvToString()` function to convert the Clarity value to a more easily readable string. +## Debugging Clarity Values +Sometimes you might receive a Clarity value that you were not expecting. Logging the value to your console won't always prove to be useful, unless you have memorized the Clarity value type enum values. + +In order to figure out what kind of value you are dealing with, you can use the `cvToString()` function to convert the Clarity value to a more easily readable string. -For example, calling `cvToString()` on a `tuple` might yield something like: +For example, calling `cvToString()` on a large `tuple` might yield something like: ``` -(tuple (key "value")) +(tuple + (a -1) + (b u1) + (c 0x74657374) + (d true) + (e (some true)) + (f none) + (g SP2JXKMSH007NPYAQHKJPQMAQYAD90NQGTVJVQ02B) + (h SP2JXKMSH007NPYAQHKJPQMAQYAD90NQGTVJVQ02B.test) + (i (ok true)) + (j (err false)) + (k (list true false)) + (l (tuple (a true) (b false))) + (m "hello world") + (n u"hello \u{1234}"))` ```