Paths are now typed. Paths in the storage domain have type StoragePath
, in the private domain PrivatePath
, and in the public domain PublicPath
. PrivatePath
and PublicPath
are subtypes of CapabilityPath
. Both StoragePath
and CapabilityPath
are subtypes of Path
.
Path | |||
CapabilityPath | StoragePath | ||
PrivatePath | PublicPath |
With paths being typed, it was possible to make the Storage API type-safer and easier to use: It is now statically checked if the correct type of path is given to a function, instead of at run-time, and therefore capability return types can now be non-optional.
The changes are as follows:
For PublicAccount
:
-
old:
fun getCapability<T>(_ path: Path): Capability<T>?
new:fun getCapability<T>(_ path: PublicPath): Capability<T>
-
old:
fun getLinkTarget(_ path: Path): Path?
new:fun getLinkTarget(_ path: CapabilityPath): Path?
For AuthAccount
:
-
old:
fun save<T>(_ value: T, to: Path)
new:fun save<T>(_ value: T, to: StoragePath)
-
old:
fun load<T>(from: Path): T?
new:fun load<T>(from: StoragePath): T?
-
old:
fun copy<T: AnyStruct>(from: Path): T?
new:fun copy<T: AnyStruct>(from: StoragePath): T?
-
old:
fun borrow<T: &Any>(from: Path): T?
new:fun borrow<T: &Any>(from: StoragePath): T?
-
old:
fun link<T: &Any>(_ newCapabilityPath: Path, target: Path): Capability<T>?
new:fun link<T: &Any>(_ newCapabilityPath: CapabilityPath, target: Path): Capability<T>?
-
old:
fun getCapability<T>(_ path: Path): Capability<T>?
new:fun getCapability<T>(_ path: CapabilityPath): Capability<T>
-
old:
fun getLinkTarget(_ path: Path): Path?
new:fun getLinkTarget(_ path: CapabilityPath): Path?
-
old:
fun unlink(_ path: Path)
new:fun unlink(_ path: CapabilityPath)
-
Add a hash function to the crypto contract (#379)
-
Added npm packages for components of Cadence. This eases the development of developer tools for Cadence:
cadence-language-server
: The Cadence Language Servermonaco-languageclient-cadence
: Language Server Protocol client for the the Monaco editorcadence-parser
: The Cadence parser
In addition, there are also examples for the language server and the parser that demonstrate the use of the packages.
-
Add a command to the language server that allows getting the entry point (transaction or script) parameters (#406)
- Allow references to be returned from from scripts (#400)
- Panic with a dedicated error for out of bounds array index (#396)
- Document resource identifiers (#394)
- Document iteration over dictionary entries (#399)
- The changes to the CBOR library have been merged, so the
replace
statement that was necessary in the last release must be removed.
This release adds support for deploying multiple contracts per account. The API for deploying has changed:
-
The functions
AuthAccount.setCode
andAuthAccount.unsafeNotInitializingSetCode
were removed (#390). -
A new contract management API has been added, which allows adding, updating, and removing multiple contracts per account (#333, #352).
See the updated documentation for details and examples.
This release adds support for enumerations (#344).
For example, a days-of-the-week enum can be declared though:
enum Day: UInt8 {
case Monday
case Tuesday
case Wednesday
case Thursday
case Friday
case Saturday
case Sunday
}
See the documentation for further details and examples.
This release adds support for switch statements (#365).
fun describe(number: Int): String {
switch number {
case 1:
return "one"
case 2:
return "two"
default:
return "other"
}
}
See the documentation for further details and examples.
Development of a code formatter has started, in form of a plugin for Prettier (#348). If you would like to contribute, please let us know!
- Limitations on data structure complexity are now enforced when decoding and also when encoding storage data, e.g number of array elements, number of dictionary entries, etc. (#370)
- Using the force-unwrap operator on a non-optional is no longer an error, but a hint is reported suggesting the removal of the unnecessary operation (#347)
- Language Server: The features requiring a Flow client connection are now optional. This allows using the language server in editors other than Visual Studio Code, such as Emacs, Vim, etc. (#303)
- Fixed the encoding of bignums in storage (#370)
- Fixed the timing of recording composite-type explicit-conformances (#356)
- Added support for exporting and JSON encoding/decoding of type values and capabilities (#374)
- Added support for exporting/importing and JSON encoding/decoding of path values (#319)
- Fixed the handling of empty integer literals in the checker (#354)
- Fixed the non-storable fields error (#350)
- Added a roadmap describing planned features and ideas (#367)
- Various documentation improvements (#385). Thanks to @andrejtokarcik for contributing this!
- Copy values on return (#355)
- Validate script argument count (#316)
This release focuses on improvements, bug fixes, and bringing the documentation up-to-date.
-
Improved support on Windows: Treat carriage returns as space (#304)
-
Improved JSON marshalling for more AST elements (#292, #286)
-
Recursively check if assignment target expression is valid, don't copy returned values (#288)
-
Consider existing prefix when suggesting completion items (#285)
-
Include available declarations in "not declared" error (#280)
-
Enforce the requirement for types to be storable in more places:
- Transaction/script parameter types (#305)
- Script return type (#291)
- Arguments passed to the account
load
andstore
functions (#251)
Previously, using non-storable failed at run-time, now the programs are rejected statically.
- Add a version constant (#289)
- Language Server: Include error notes as related information in diagnostic (#276)
- Don't return an error when a location type is not supported (#290)
- Handle incomplete types in checker (#284)
- Fix the suggestion in the error when an interface is used as a type (#281)
- Brought the documentation up-to-date, include all new language features and API changes (#275, #277, #279)
-
Arguments passed to
cadence.Value#WithType
are now typed as pointers (*cadence.Type
) rather than values (cadence.Type
)Example:
myEventType := cadence.EventType{...}) // this π myEvent := cadence.NewEvent(...).WithType(myEventType) // changes to this π myEvent := cadence.NewEvent(...).WithType(&myEventType)
This release contains a lot of improvements to the language server, which improve the development experience in the Visual Studio Code extension, and will also soon be integrated into the Flow Playground.
-
Added member completion to the language server (#257)
- Insert the argument list when completing functions (#268)
- Offer keyword snippets (#268)
-
Enabled running the language server in the browser by compiling it to WebAssembly (#242)
-
Added support for docstrings to the parser (#246, #255)
-
Added support for pragmas to the parser (#239)
-
Started a source compatibility suite (#226, #270).
Cadence will be regularly checked for compatibility with existing code to ensure it does not regress in source compatibility and performance. As a start, the Fungible Token standard and Non-Fungible Token standard repositories are checked.
Consider requesting your repository to be included in the compatibility suite!
-
Added a simple minifier (#252)
- Made embedding the language server easier (#274, #262)
- Extended the JSON serialization for Cadence values and types (#260, #264, #265, #266, #271, #273)
- Improved conformance checking (#245). Made return types covariant, i.e. allow subtypes for return types of functions to satisfy interface conformance.
- Made strings immutable (#269)
- Moved the Visual Studio Code extension to a new repository: https://github.com/onflow/vscode-flow (#244)
- Added a magic prefix to stored data (#236). This allows versioning the data. Existing data without a prefix is migrated automatically
- Changed the order in which fields in values and types are exported to declaration order (#272)
- Removed the service account from the list of usable accounts in the language server (#261)
- Fixed a parsing ambiguity for return types (#267)
- Fixed non-composite reference subtyping (#253)
- Removed the old parser (#249)
This is a small release with some bug fixes, internal improvements, and one breaking change for code that embeds Cadence.
-
Removed the unused
controller
parameter from the storage interface methods:-
func GetValue(owner, controller, key []byte) (value []byte, err error)
βfunc GetValue(owner, key []byte) (value []byte, err error)
-
SetValue(owner, controller, key, value []byte) (err error)
βSetValue(owner, key, value []byte) (err error)
-
ValueExists(owner, controller, key []byte) (exists bool, err error)
βValueExists(owner, key []byte) (exists bool, err error)
This is only a breaking change in the implementation of Cadence, i.e for code that embeds Cadence, not in the Cadence language, i.e. for users of Cadence.
-
-
Added an optional callback for writes with high-level values to the interface:
type HighLevelStorage interface { // HighLevelStorageEnabled should return true // if the functions of HighLevelStorage should be called, // e.g. SetCadenceValue HighLevelStorageEnabled() bool // SetCadenceValue sets a value for the given key in the storage, owned by the given account. SetCadenceValue(owner Address, key string, value cadence.Value) (err error) }
This is a feature in the implementation of Cadence, i.e for code that embeds Cadence, not in the Cadence language, i.e. for users of Cadence.
- Don't report an error for a restriction with an invalid type
- Record the occurrences of types. This enables the "Go to definition" feature for types in the language server
- Parse member expressions without a name. This allows the checker to run. This will enable adding code completion support or members in the future.
- Fixed a crash when checking the invocation of a function on an undeclared variable
- Fixed handling of functions in composite values in the JSON-CDC encoding
- Fixed a potential stack overflow when checking member storability
A new built-in contract Crypto
was added for performing cryptographic operations.
The contract can be imported using import Crypto
.
This first iteration provides support for validating signatures, with an API that supports multiple signatures and weighted keys.
For example, to verify two signatures with equal weights for some signed data:
let keyList = Crypto.KeyList()
let publicKeyA = Crypto.PublicKey(
publicKey:
"db04940e18ec414664ccfd31d5d2d4ece3985acb8cb17a2025b2f1673427267968e52e2bbf3599059649d4b2cce98fdb8a3048e68abf5abe3e710129e90696ca".decodeHex(),
signatureAlgorithm: Crypto.ECDSA_P256
)
keyList.add(
publicKeyA,
hashAlgorithm: Crypto.SHA3_256,
weight: 0.5
)
let publicKeyB = Crypto.PublicKey(
publicKey:
"df9609ee588dd4a6f7789df8d56f03f545d4516f0c99b200d73b9a3afafc14de5d21a4fc7a2a2015719dc95c9e756cfa44f2a445151aaf42479e7120d83df956".decodeHex(),
signatureAlgorithm: Crypto.ECDSA_P256
)
keyList.add(
publicKeyB,
hashAlgorithm: Crypto.SHA3_256,
weight: 0.5
)
let signatureSet = [
Crypto.KeyListSignature(
keyIndex: 0,
signature:
"8870a8cbe6f44932ba59e0d15a706214cc4ad2538deb12c0cf718d86f32c47765462a92ce2da15d4a29eb4e2b6fa05d08c7db5d5b2a2cd8c2cb98ded73da31f6".decodeHex()
),
Crypto.KeyListSignature(
keyIndex: 1,
signature:
"bbdc5591c3f937a730d4f6c0a6fde61a0a6ceaa531ccb367c3559335ab9734f4f2b9da8adbe371f1f7da913b5a3fdd96a871e04f078928ca89a83d841c72fadf".decodeHex()
)
]
// "foo", encoded as UTF-8, in hex representation
let signedData = "666f6f".decodeHex()
let isValid = keyList.isValid(
signatureSet: signatureSet,
signedData: signedData
)
The type Type
was added to represent types at run-time.
To create a type value, use the constructor function Type<T>()
, which accepts the static type as a type argument.
This is similar to e.g. T.self
in Swift, T::class
in Kotlin, and T.class
in Java.
For example, to represent the type Int
at run-time:
let intType: Type = Type<Int>()
This works for both built-in and user-defined types. For example, to get the type value for a resource:
resource Collectible {}
let collectibleType = Type<@Collectible>()
The function fun isInstance(_ type: Type): Bool
can be used to check if a value has a certain type:
let collectible <- create Collectible()
let collectibleType = Type<@Collectible>()
let result = collectible.isInstance(collectibleType)
For example, this allows implementing a marketplace sale resource:
pub resource SimpleSale {
pub var objectForSale: @AnyResource?
pub let priceForObject: UFix64
pub let requiredCurrency: Type
pub let paymentReceiver: Capability<&{FungibleToken.Receiver}>
init(
objectForSale: @AnyResource,
priceForObject: UFix64,
requiredCurrency: Type,
paymentReceiver: Capability<&{FungibleToken.Receiver}>
) {
self.objectForSale <- objectForSale
self.priceForObject = priceForObject
self.requiredCurrency = requiredCurrency
self.paymentReceiver = paymentReceiver
}
destroy() {
destroy self.objectForSale
}
pub fun buyObject(purchaseAmount: @FungibleToken.Vault): @AnyResource {
pre {
self.objectForSale != nil
purchaseAmount.balance >= self.priceForObject
purchaseAmount.isInstance(self.requiredCurrency)
}
let receiver = self.paymentReceiver.borrow()
?? panic("failed to borrow payment receiver capability")
receiver.deposit(from: <-purchaseAmount)
let objectForSale <- self.objectForSale <- nil
return <-objectForSale
}
}
The existing parser was replaced by a new implementation, completely written from scratch, producing the same result as the old parser.
It is significantly faster than the old parser. For example, the following benchmark shows the difference for parsing all fungible and non-fungible token contracts and example transactions:
$ go run ./cmd/parse -bench ../../flow-{n,}ft/{contracts,transactions}/*
flow-nft/contracts/ExampleNFT.cdc: [old] 9 111116110 ns/op
flow-nft/contracts/ExampleNFT.cdc: [new] 2712 463483 ns/op
flow-nft/contracts/NonFungibleToken.cdc: [old] 393 3097172 ns/op
flow-nft/contracts/NonFungibleToken.cdc: [new] 3489 348496 ns/op
flow-nft/transactions/mint_nft.cdc: [old] 700 1574730 ns/op
flow-nft/transactions/mint_nft.cdc: [new] 12770 94070 ns/op
flow-nft/transactions/read_nft_data.cdc: [old] 994 1222887 ns/op
flow-nft/transactions/read_nft_data.cdc: [new] 15242 79295 ns/op
flow-nft/transactions/setup_account.cdc: [old] 1281 879751 ns/op
flow-nft/transactions/setup_account.cdc: [new] 16675 71759 ns/op
flow-nft/transactions/transfer_nft.cdc: [old] 72 16417568 ns/op
flow-nft/transactions/transfer_nft.cdc: [new] 10000 109380 ns/op
flow-ft/contracts/CustodialDeposit.cdc: [old] 18 64938763 ns/op
flow-ft/contracts/CustodialDeposit.cdc: [new] 3482 354662 ns/op
flow-ft/contracts/FlowToken.cdc: [old] 7 177111544 ns/op
flow-ft/contracts/FlowToken.cdc: [new] 1920 640557 ns/op
flow-ft/contracts/FungibleToken.cdc: [old] 232 5324962 ns/op
flow-ft/contracts/FungibleToken.cdc: [new] 2947 419529 ns/op
flow-ft/contracts/TokenForwarding.cdc: [old] 44 25136749 ns/op
flow-ft/contracts/TokenForwarding.cdc: [new] 7183 172917 ns/op
flow-ft/transactions/burn_tokens.cdc: [old] 37 31475393 ns/op
flow-ft/transactions/burn_tokens.cdc: [new] 11361 105932 ns/op
flow-ft/transactions/create_forwarder.cdc: [old] 733 1636347 ns/op
flow-ft/transactions/create_forwarder.cdc: [new] 8127 147520 ns/op
flow-ft/transactions/create_minter.cdc: [old] 1306 923201 ns/op
flow-ft/transactions/create_minter.cdc: [new] 15240 77666 ns/op
flow-ft/transactions/custodial_deposit.cdc: [old] 69 16504795 ns/op
flow-ft/transactions/custodial_deposit.cdc: [new] 7940 144228 ns/op
flow-ft/transactions/get_balance.cdc: [old] 1094 1111272 ns/op
flow-ft/transactions/get_balance.cdc: [new] 18741 65745 ns/op
flow-ft/transactions/get_supply.cdc: [old] 1392 740989 ns/op
flow-ft/transactions/get_supply.cdc: [new] 46008 26719 ns/op
flow-ft/transactions/mint_tokens.cdc: [old] 72 17435128 ns/op
flow-ft/transactions/mint_tokens.cdc: [new] 8841 124117 ns/op
flow-ft/transactions/setup_account.cdc: [old] 1219 1089357 ns/op
flow-ft/transactions/setup_account.cdc: [new] 13797 84948 ns/op
flow-ft/transactions/transfer_tokens.cdc: [old] 74 17011751 ns/op
flow-ft/transactions/transfer_tokens.cdc: [new] 9578 125829 ns/op
The new parser also provides better error messages and will allow us to provide even better error messages in the future.
The new parser is enabled by default β if you discover any problems or regressions, please report them!
Capabilities now accept an optional type argument, the reference type the capability should be borrowed as.
If a type argument is provided, it will be used for borrow
and check
calls, so there is no need to provide a type argument for the calls anymore.
The function getCapability
now also accepts an optional type argument, and returns a typed capability if one is provided.
For example, the following two functions have the same behaviour:
fun cap1(): &Something? {
// The type annotation for `cap` is only added for demonstration purposes,
// it can also be omitted, because it can be inferred from the value
let cap: Capability = getAccount(0x1).getCapability(/public/something)!
return cap.borrow<&Something>()
}
fun cap2(): &Something? {
// The type annotation for `cap` is only added for demonstration purposes,
// it can also be omitted, because it can be inferred from the value
let cap: Capability<&Something> =
getAccount(0x1).getCapability<&Something>(/public/something)!
return cap.borrow()
}
Prefer a typed capability over a non-typed one to reduce uses / borrows to a simple .borrow()
, instead of having to repeatedly mention the borrowed type.
Just like transactions, the main
functions of scripts can now have parameters.
For example, a script that can be passed an integer and a string, and which logs both values, can be written as:
pub fun main(x: Int, y: String) {
log(x)
log(y)
}
The function fun toString(): String
was added to all number types and addresses. It returns the textual representation of the type. A future version of Cadence will add this function to more types.
The function fun toBytes(): [UInt8]
was added to Address
. It returns the byte representation of the address.
The function fun toBigEndianBytes(): [UInt8]
was added to all number types. It returns the big-endian byte representation of the number value.
- Fixed the checking of return statements: A missing return value is now properly reported
- Fixed the checking of function invocations: Resources are temporarily moved into the invoked function
- Disabled caching of top-level programs (transactions, scripts)
- Fixed the comparison of optional values
- Fixed parsing of unpadded fixed point numbers in the JSON encoding of Cadence values (JSON-CDC)
Fields which have a non-storable type are now rejected:
Non-storable types are:
- Functions (e.g.
((Int): Bool)
) - Accounts (
AuthAccount
/PublicAccount
) - Transactions
Void
A future version will also make references non-storable. Instead of storing a reference, store a capability and borrow it to acquire a reference.
Cadence now represents all byte arrays as [UInt8]
rather than [Int]
. This affects the following functions:
String.decodeHex(): [Int]
=>String.decodeHex(): [UInt8]
AuthAccount.addPublicKey(publicKey: [Int])
=>AuthAccount.addPublicKey(publicKey: [UInt8])
AuthAccount.setCode(code: [Int])
=>AuthAccount.setCode(code: [UInt8])
However, array literals such as the following will still be inferred as [Int]
, meaning that they can't be used directly:
myAccount.addPublicKey([1, 2, 3])
Consider using String.decodeHex()
for now until type inference has been improved.
- The
AuthAccount
constructor now requires apayer
parameter, from which the account creation fee is deducted. ThepublicKeys
andcode
parameters were removed.
-
Added the function
unsafeNotInitializingSetCode
toAuthAccount
which, likesetCode
, updates the account's code but does not create a new contract object or invoke its initializer. This function can be used to upgrade account code without overwriting the existing stored contract object.β οΈ This is potentially unsafe because no data migration is performed. For example, changing a field's type results in undefined behaviour (i.e. could lead to the execution being aborted).
- Fixed variable shadowing for-in identifier to shadow outer scope (#72)
- Fixed order of declaring conformances and usages
- Fixed detection of resource losses in invocation of optional chaining result
- Dictionary values which are resources are now stored in separate storage keys. This is transparent to programs, so is not breaking on a program level, but on the integration level.
- The state encoding was switched from Gob to CBOR. This is transparent to programs, so is not breaking on a program level, but on the integration level.
- The size of account addresses was reduced from 160 bits to 64 bits
- Added bitwise operations
- Added support for caching of program ASTs
- Added metrics
- Block info can be queried now
- Unnecessary writes are now avoided by tracking modifications
- Improved access modifier checks
- Improved the REPL
- Decode and validate transaction arguments
- Work on a new parser has started and is almost complete
- Fixed account code updated
- Fixed the destruction of arrays
- Fixed type requirements
- Implemented modulo for
[U]Fix64
, fix modulo with negative operands - Implemented division for
[U]Fix64
- Fixed the
flow.AccountCodeUpdated
event - Fixed number conversion
- Fixed function return checking