-
Notifications
You must be signed in to change notification settings - Fork 23
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Change member access semantics #89
Merged
Merged
Changes from all commits
Commits
Show all changes
8 commits
Select commit
Hold shift + click to select a range
2d5e032
Add FLIP for changing member access semantics
SupunS 37df62b
Fix link
SupunS 459b3b1
Update example to use resources
SupunS dd2825b
Merge branch 'supun/member-access-semantics' of https://github.com/on…
SupunS 179c6b1
Add example for structs
SupunS 883764e
Add reference to the 'vision'
SupunS b2f3996
accept
turbolent 94d8273
Merge branch 'main' into supun/member-access-semantics
turbolent File filter
Filter by extension
Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,243 @@ | ||
--- | ||
status: accepted | ||
flip: 89 | ||
authors: Supun Setunga ([email protected]) | ||
sponsor: Supun Setunga ([email protected]) | ||
updated: 2023-07-05 | ||
--- | ||
|
||
# FLIP 89: Change Member Access Semantics | ||
|
||
## Objective | ||
|
||
This proposal suggest to return a reference when accessing a member of a container (field of a composite value, | ||
key of a map, index of an array), if the parent value is also a reference and the accessed member is a container type. | ||
|
||
## Motivation | ||
|
||
A previous version of Cadence ("Secure Cadence") restricted the potential foot-gun of mutating container-typed | ||
`let` fields/variables via the | ||
[Cadence mutability restrictions FLIP](https://github.com/onflow/flips/blob/main/cadence/20211129-cadence-mutability-restrictions.md). | ||
However, this was a partial solution, and was only applicable for arrays and dictionaries, but did not solve it for | ||
user-defined types. | ||
|
||
There are still ways to mutate such fields by: | ||
- Directly mutating a nested composite typed field. | ||
- Mutating a field by calling a mutating function on the field. | ||
- Mutating the field via a reference. | ||
|
||
More details on this problem is described in the [FLIP for improving mutability restrictions](https://github.com/onflow/flips/pull/58). | ||
|
||
This proposal is aimed at moving one step closer to solving this problem by returning references for member-access, | ||
so that [Entitlements](https://github.com/onflow/flips/pull/54) can be used to control who can perform mutating | ||
operations via references. | ||
|
||
The [Mutability Restrictions Vision](https://github.com/onflow/flips/pull/97) explains how this proposed change | ||
contributes to solving the aforementioned problems, and how the final solution looks like, with examples. | ||
|
||
## User Benefit | ||
|
||
As mentioned in the previous section, this change enables using Entitlements to prevent accidental mutations via references. | ||
|
||
## Design Proposal | ||
|
||
This proposal suggests field-access (i.e. `foo.bar`) and index-access (`foo[bar]`) to | ||
return a reference to the targeting (field or element) value, instead of the value itself, if: | ||
- The parent (i.e. `foo`) is a reference to a container-typed value. | ||
- The accessed field/element is again container-typed. | ||
|
||
where a "container" is any of: | ||
- Dictionary | ||
- Array | ||
- Composite | ||
- An optional of above | ||
|
||
### Resource Example: | ||
|
||
For example, consider the below `Collection` resource which has two fields: one (`id`) is String-typed, | ||
and the other (`ownedNFTs`) is dictionary-typed. | ||
|
||
```cadence | ||
pub resource Collection { | ||
|
||
// Primitive-typed field | ||
pub var id: String | ||
|
||
// Dictionary typed field | ||
pub var ownedNFTs: @{UInt64: NFT} | ||
} | ||
``` | ||
|
||
#### Case I: Code owns the container value | ||
|
||
Assume `collection` is of type `Collection`. i.e. the code owns the value. | ||
Then, there is no change to the current behavior. | ||
|
||
```cadence | ||
var collection: @Collection <- ... | ||
|
||
// `collection.ownedNFTs` would return the concrete value, which is of type `@{UInt64: NFT}`. | ||
// This is an error because it is not possible to move nested resource. | ||
// This is same as existing semantics. | ||
var ownedNFTs: @{UInt64: NFT} <- collection.ownedNFTs | ||
|
||
// `collection.id` would return the concrete string value. | ||
// This is same as existing semantics. | ||
var id: String = collection.id | ||
``` | ||
|
||
#### Case II: Code only has a reference to the container value | ||
|
||
Assume a reference to the collection `collectionRef` is available, and is of type `&Collection`. | ||
i.e. code doesn't own the value, but has only a reference to the value. | ||
|
||
```cadence | ||
var collectionRef: &Collection = ... | ||
|
||
// `collectionRef.ownedNFTs` would now return a reference of type `&{UInt64: NFT}`. | ||
var ownedNFTsRef: &{UInt64: NFT} = collectionRef.ownedNFTs | ||
|
||
// However, `collectionRef.id` would return the value, since it is a primitive type. | ||
// This is same as existing semantics. | ||
var id: String = collectionRef.id | ||
``` | ||
|
||
It is also important to note that, the returned reference has no entitlements assigned to them. | ||
i.e: they are **non-**`auth` references. | ||
|
||
#### Nested resources | ||
|
||
Assume a nested resource collection as bellow. | ||
|
||
```cadence | ||
pub resource MasterCollection { | ||
pub var kittyCollection: @Collection | ||
pub var topshotCollection: @Collection | ||
} | ||
|
||
|
||
pub resource Collection { | ||
pub var ownedNFTs: @{UInt64: NonFungibleToken.NFT} | ||
SupunS marked this conversation as resolved.
Show resolved
Hide resolved
|
||
|
||
access(Withdraw) fun withdraw(withdrawID: UInt64): @NonFungibleToken.NFT { | ||
let token <- self.ownedNFTs.remove(key: withdrawID) ?? panic("missing NFT") | ||
emit Withdraw(id: token.id, from: self.owner?.address) | ||
return <-token | ||
} | ||
} | ||
``` | ||
|
||
Also, suppose a reference to the `MasterCollection` without any entitlements (i.e: non-auth) is available/borrowed. | ||
|
||
```cadence | ||
var masterCollectionRef: @MasterCollection = ... | ||
``` | ||
|
||
Previously, it was possible to withdraw from the inner collection, despite the `masterCollectionRef` not having the | ||
`Withdraw` entitlement. | ||
|
||
```cadence | ||
masterCollectionRef.kittyCollection.withdraw(24) // OK | ||
``` | ||
|
||
With the proposed changes, this will be an error, since `masterCollectionRef.kittyCollection` is now return a | ||
non-auth reference, and calling `withdraw` is prohibited because that reference doesn't have the `Withdraw` entitlement. | ||
|
||
```cadence | ||
masterCollectionRef.kittyCollection.withdraw(24) // Static Error | ||
``` | ||
|
||
### Struct Example: | ||
|
||
Similarly, consider the below `NFTView` struct which has two fields: one (`id`) is UInt64-typed, | ||
and the other (`royalties`) is array-typed. | ||
|
||
```cadence | ||
pub struct NFTView { | ||
|
||
// Primitive-typed field | ||
pub var id: UInt64 | ||
|
||
// Array typed field | ||
pub var royalties: [Royalty] | ||
} | ||
``` | ||
|
||
#### Case I: Code owns the container value | ||
|
||
Assume `nftView` is of type `NFTView`. i.e. the code owns the value. | ||
Then, there is no change to the current behavior. | ||
|
||
```cadence | ||
var nftView: NFTView = ... | ||
|
||
// `nftView.royalties` would return the concrete value, which is of type `[Royalty]`. | ||
// This is same as existing semantics. | ||
var royalties: [Royalty] = nftView.royalties | ||
|
||
// `nftView.id` would return the concrete `UInt64` value. | ||
// This is same as existing semantics. | ||
var id: UInt64 = collection.id | ||
``` | ||
|
||
#### Case II: Code only has a reference to the container value | ||
|
||
Assume a reference to the nftView `nftViewRef` is available, and is of type `&NFTView`. | ||
i.e. code doesn't own the value, but has only a reference to the value. | ||
|
||
```cadence | ||
var nftViewRef: &Collection = ... | ||
|
||
// `nftViewRef.royalties` would now return a reference of type `&[Royalty]`. | ||
var royaltiesRef: &[Royalty] = nftViewRef.royalties | ||
|
||
// However, `nftViewRef.id` would return the value, since it is a primitive type. | ||
// This is same as existing semantics. | ||
var id: String = nftViewRef.id | ||
``` | ||
|
||
Similar to resources, the returned reference has no entitlements assigned to them. | ||
i.e: they are **non-**`auth` references. | ||
|
||
### Drawbacks | ||
|
||
None | ||
|
||
### Alternatives Considered | ||
|
||
None | ||
|
||
### Performance Implications | ||
|
||
There is no direct performance impact by this change. | ||
|
||
However, since accessing a nested objects through a reference would now be returning reference, | ||
it will avoid the copying of nested objects, which may increase the performance for certain use-cases. | ||
|
||
### Dependencies | ||
|
||
None | ||
|
||
### Engineering Impact | ||
|
||
This change has medium complexity in the implementation. | ||
|
||
### Compatibility | ||
|
||
This is a backward incompatible change. | ||
|
||
### User Impact | ||
|
||
Many deployed contracts might be affected by this change. | ||
|
||
## Related Issues | ||
|
||
None | ||
|
||
## Prior Art | ||
|
||
None | ||
|
||
## Questions and Discussion Topics | ||
|
||
None |
Add this suggestion to a batch that can be applied as a single commit.
This suggestion is invalid because no changes were made to the code.
Suggestions cannot be applied while the pull request is closed.
Suggestions cannot be applied while viewing a subset of changes.
Only one suggestion per line can be applied in a batch.
Add this suggestion to a batch that can be applied as a single commit.
Applying suggestions on deleted lines is not supported.
You must change the existing code in this line in order to create a valid suggestion.
Outdated suggestions cannot be applied.
This suggestion has been applied or marked resolved.
Suggestions cannot be applied from pending reviews.
Suggestions cannot be applied on multi-line comments.
Suggestions cannot be applied while the pull request is queued to merge.
Suggestion cannot be applied right now. Please check back later.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
A strong misson statement here, and a good reason to do it. love it.