status | flip | authors | sponsor | updated |
---|---|---|---|---|
implemented |
54 |
Daniel Sainati ([email protected]) |
Daniel Sainati ([email protected]) |
2023-05-10 |
This FLIP proposes two major changes to the language:
- The addition of new, bespoke access modifiers on fields and functions in composites and interfaces in the form of entitlements
- A change to the behavior of reference types to add granular authority restrictions to specific interfaces, and allow safe downcasting
Currently in Cadence, when users are in possession of a reference value, unless that reference is declared
with an auth
type, users are unable to downcast that reference to a more specific type. This restriction
exists for security reasons; it should not be possible for someone with a &{Balance}
reference to downcast that
to a &Vault
. Cadence's current access control model relies on exposing limited interfaces to grant others capabilities
to resources they own.
However, this can often result in restrictions that are unnecessary and onerous to users. In the above example, a user
with a &{Balance}
reference would not be able to cast that reference to a &{Receiver}
reference, despite the fact
that there would be no safety concerns in doing so; it should be possible to call deposit
regardless of what kind of
reference a user possesses.
The proposed change is designed to overcome this limitation by allowing developers to directly mark which fields and functions should be access-limited using these new bespoke modifiers, and which should be generally available to anyone.
This feature would make interacting with references significantly easier, as users would no longer be restricted from
accessing methods like deposit
that are safe for anyone to use, just because they do not possess a reference of the specific type,
or because they are trying to write a generic method that works, say, on any NFT
type.
It would also increase safety, as we would be able to do additional checking on the values and types to which users create capabilities.
For example, it is currently possible for a user to (presumably accidentally) create a public Capability directly to a Vault
object,
rather than a Capability to a &{Balance, Receiver}
restricted type as is intended. With these changes, such a Capability still would
not be subject to having its funds withdraw
n, as this Vault
Capability would not be auth(Provider)
. In order to have access to an auth
method like withdraw
, the user would need to explicitly create the Capability with an auth(Provider)
reference, and we could enforce statically
(or warn) that auth
reference capabilies should not be stored publicly.
The first part of this FLIP proposes to add a new declaration type to Cadence: entitlement
s. entitlement
declarations are simple,
just the keyword entitlement
and the name; we only require that they be pre-declared to guard against typos and other minor errors.
A sample entitlement
may look like this:
entitlement E
In the future, this is easily extensible to allow creating entitlements that are constructed from others via operators like &
or |
.
To go with these entitlement
declarations, this FLIP proposes to add a new access control modifier to field and function declarations in composite types:
access(E)
, which allows access to either the immediate owner of the resource (i.e. anybody who has the actual resource value),
or someone with an auth(E)
reference to the type on which the member is defined. The X
here can be the qualified name of any entitlement, e.g.:
entitlement X
resource R {
access(E) fun foo(a: Int) { }
access(E) let bar: String
}
A single member definition can include multiple entitlements, using either a |
or a ,
separator when defining the list.
An entitlement list defined using a |
functions like a disjunction (or an "or"); it is accessible to any auth
reference with any of those entitlements.
An entitlement list defined using a ,
functions like a conjunction set (or an "and"); it is accessible only to an auth
reference with all of those entitlements.
Note that these operators cannot be mixed within a single list; any entitlement access modifier may use either |
or ,
, but not both.
So, for example, in
entitlement E
entitlement F
resource R {
access(E, F) foo() {}
access(E | F) bar() {}
}
foo
is only calleable on a reference to R
that is auth
for both E
and F
, while bar
is calleable on any auth
reference that is auth
for either E
or F
(or both).
Like access(contract)
and access(account)
, this new modifier sits exactly between pub
and priv
(or equivalently access(self)
)
in permissiveness; it allows less access than pub
, but strictly more than priv
, as an access(E)
field or function can be used anywhere
in the implementation of the composite. To see why, consider that self
is necessarily of the same type as the composite, meaning
that the access rules defined above allow any access(I)
members to be accessed on it. A table summarizing the new access modifiers is included here:
Modifier(s) | Visibility |
---|---|
access(self) (priv ) |
Methods defined within the same type object. |
access(contract) |
Methods defined within the same smart contract object. |
access(account) |
Methods defined in a contract deployed to the same account. |
access(E) |
Code holding an auth(E) reference for some defined entitlement E . |
access(all) (pub ) |
Any code with any reference to the object. |
As specified above, note that actual ownership of an object grants access to any access(E)
member for any E
.
As such, the following would be prohibited statically:
pub entitlement E
pub resource R {
access(E) fun foo() { ... }
}
let r: &R = // ...
r.foo()
while all of these would be permitted:
pub entitlement E
pub resource R {
access(E) fun foo() { ... }
pub fun bar() {
self.foo()
}
}
let r: @R = // ...
r.foo()
let ref: auth(E) &R = &r
ref.foo()
Note also that while the normal interface implementation subtyping rules would allow a composite to implement an access(E)
interface member with a pub
composite member, as this is less restrictive, in order to prevent users from accidentally surrendering authority and security this way, we prevent this statically.
As such, the below code would not typecheck.
pub entitlement E
pub resource interface I {
access(E) fun foo()
}
pub resource R: I {
pub fun foo() {} // must also be access(E)
}
If users would like to expose an access-limited function to pub
users, they can do so by wrapping the access(E)
function in a pub
function.
As with normal subtyping, this would also be statically rejected:
pub entitlement E
pub resource interface I {
pub fun foo()
}
pub resource R: I {
access(E) fun foo() {}
}
since if this were to typecheck, anybody with a &R
reference could upcast it to &{I}
and thus gain the ability to call foo
.
When multiple interfaces declare the same function with different entitlements, a composite implementing both interfaces must use the |
union of the function's entitlement
sets as the access modifier for that function. E.g.
pub entitlement E
pub resource interface I {
access(E) fun foo()
}
pub entitlement F
pub resource interface G {
access(F) fun foo()
}
pub resource R: I, G {
access(E | F) fun foo() {} // valid, access(E | F) permits both access(E) and access(F)
}
pub resource S: I, G {
access(E) fun foo() {} // invalid, does not match definition in G
}
pub resource T: I, G {
access(F) fun foo() {} // invalid, does not match definition in I
}
The second, more complex part of this proposal, is a change to the behavior of references to allow the auth
modifier not to
refer to the entire referenced value, but to the specific set of interfaces to which the referenced value has auth
permissions (the reference's entitlements).
To express this, the auth
keyword can now be used with similar syntax to that of restricted types: auth(E1, E2, ...)
, where the E
s
in the parentheses denote the entitlements. This permits these references to access access(E)
members for any E
entitlement they possess.
So, for example, given two entitlement definitions and a composite definition:
pub entitlement A
pub entitlement B
pub resource R {
access(A) fun foo() { ... }
access(B) fun bar() { ... }
pub fun baz() { ... }
}
a value of type auth(A) &R
would be able to access foo
, because the reference has the A
entitlement, but not bar
, because
it does not have an entitlement for B
. However, baz
would be accessible on this reference, since it has pub
access.
The other part of this change is to remove the limitations on resource downcasting that used to exist. Prior to this change,
non-auth
references could not be downcast at all, since the sole purpose of the auth
keyword was to indicate that references could be
downcast. With the proposed change, all reference types can be downcast or upcast the same way any other type would be. So, for example
&{I} as! &R
would be valid, as would &AnyResource as? &{I}
or &{I} as? &{J}
, using the hierarchy defined above (for any J
).
However, the auth
-ness (and the reference's set of entitlements) would not change on downcasting, nor would that set be expandable via casting.
In fact, the set of entitlements to which a reference is authorized is purely a static construct, and entitlements do not exist as a concept at runtime.
In particular, what this means that it is not ever possible to downcast a reference to a type with a more restrictive set of entitlements.
As such, while auth(A, B) &R
would be statically upcastable to auth(A) &R
, since this decreases the permissions on the
reference, it would not be possible to go from auth(A) &R
to auth(A, B) &R
.
The subtyping rules for auth
references is that auth (U1, U2, ... ) &X <: auth (T1, T2, ... ) &X
whenever {U1, U2, ...}
is a superset of {T1, T2, ...}
, or equivalently ∀T ∈ {T1, T2, ...}, ∃U ∈ {U1, U2, ...}, T = U
. Of course, all auth
reference types
would remain subtypes of all non-auth
reference types as before.
In addition to the ,
-separated list of entitlements (which defines a conjunction/"and" set for auth
modifiers similarly to its behavior for access
modifiers), it is also possible,
although very rarely necessary, to define |
-separated entitlement lists in auth
modifers for references, like so: auth(E1 | E2 | ...) &T
. In this case, the type denotes that the
reference to T
is authorized for at least one of the entitled specified, not that it is auth
for all of them. This means, for example, that an auth(A | B) &R
reference
would not be permitted to call an access(A)
function on R
, because we only know that the reference is authorized for one of A
or B
, and cannot guarantee that it is permitted to
call an A
-entitled function. However, an auth(A | B) &R
reference could call an access(A | B)
function on R
, because the function requires one of A
or B
, and we know
that our reference has at least one of these entitlements.
entitlement E
entitlement F
resource R {
access(E | F) fun foo() { ... }
access(E, F) fun bar() { ... }
access(E) fun baz() { ... }
access(F) fun qux() { ... }
}
fun test(ref: auth(E | F) &R) {
ref.foo() // allowed because `foo` requires either `E` or `F`
ref.bar() // not allowed because the reference may not have `E` and `F`
ref.baz() // not allowed because the reference may not have `E`
ref.qux() // not allowed because the reference may not have `F`
}
The subtyping rules for |
-separated entitlement lists allow lists to expand on supertyping. I.e., auth(A | B) &R <: auth(A | B | C) &R
, as this decreases the information we have about
the reference's entitlements and thus permits fewer operations. In general, auth (U1 | U2 | ... ) &X <: auth (T1 | T2 | ... ) &X
whenever {U1, U2, ...}
is a subset of {T1, T2, ...}
, or equivalently ∀U ∈ {U1, U2, ...}, ∃T ∈ {T1, T2, ...}, T = U
.
,
-separated entitlement lists subtype |
-separated ones as long as the two sets are not disjoint; that is, as long as there is an entitlement in the subtype set that is also in the supertype set.
This is because we are casting from a type that is known to possess all of the listed entitlements to a type that is only guaranted to possess one. More specifically,
auth (U1, U2, ... ) &X <: auth (T1 | T2 | ... ) &X
whenever {U1, U2, ...}
is not disjoint from {T1, T2, ...}
, or equivalently ∃U ∈ {U1, U2, ...}, ∃T ∈ {T1, T2, ...}, T = U
.
In practice |
-separated entitlement lists never subtype ,
-separated ones except in the trivial case. This is because we are attempting to upcast (change the type without gaining any new
specificity) from a reference where we only know we possess at least one of the listed entitlement to one where we know we possess all of them. This is only possible when all of the references
in both lists are equal. More specifically, auth (U1 | U2 | ... ) &X <: auth (T1, T2, ... ) &X
whenever ∀U ∈ {U1, U2, ...}, ∀T ∈ {T1, T2, ...}, T = U
.
As one can see, this is only possible when every U
and T
are the same entitlement, or when the two entitlement lists each only have a single equivalent element.
When objects have fields that are child objects, it can often be valuable to have different views of that reference depending on the entitlements one has on the reference to the parent object. Consider the following example:
entitlement OuterEntitlement
entitlement SubEntitlement
resource SubResource {
access(all) fun foo() { ... }
access(SubEntitlement) fun bar() { ... }
}
resource OuterResource {
access(self) let childResource: @SubResource
access(all) fun getPubRef(): &SubResource {
return &self.childResource as &SubResource
}
access(OuterEntitlement) fun getEntitledRef(): auth(SubEntitlement) &SubResource {
return &self.childResource as auth(SubEntitlement) &SubResource
}
init(r: @SubResource) {
self.childResource <- r
}
}
With this pattern, we can store to a SubResource
on an OuterResource
value, and create different ways to access that nested resource depending on the entitlement oneposseses.
Somoneone with only an unauthorized reference to OuterResource
can only call the getPubRef
function, and thus can only get an unauthorized reference to SubResource
that lets them call foo
.
However, someone with a OuterEntitlement
-authorized refererence to the OuterResource
can call the getEntitledRef
function, giving them a SubEntitlement
-authorized reference to SubResource
that
allows them to call bar
.
This pattern is functional, but it is unfortunate that we are forced to "duplicate" the accessors to SubResource
, duplicating the code and storing two functions on the object, essentially creating
two different views to the same object that are stored as different functions. To avoid necessitating this duplication, we add support to the language for "entitlement mappings", a way to declare
statically how entitlements are propagated from parents to child objects in a nesting hierarchy. So, the above example could be equivalently written as:
entitlement OuterEntitlement
entitlement SubEntitlement
// specify a mapping for entitlements called `Map`, which defines a function
// from an input set of entitlements (called the domain) to an output set (called the image)
entitlement mapping Map {
OuterEntitlement -> SubEntitlement
}
resource SubResource {
access(all) fun foo() { ... }
access(SubEntitlement) fun bar() { ... }
}
resource OuterResource {
access(self) let childResource: @SubResource
// by referering to `Map` here, we declare that the entitlements we receive when accessing the `getRef` function on this resource
// will depend on the entitlements we possess to the resource during the access.
access(Map) fun getRef(): auth(Map) &SubResource {
return &self.childResource as auth(Map) &SubResource
}
init(r: @SubResource) {
self.childResource = r
}
}
// given some value `r` of type `@OuterResource`
let pubRef = &r as &OuterResource
let pubSubRef = r.getRef() // has type `&SubResource`
let entitledRef = &r as auth(OuterEntitlement) &OuterResource
let entiteldSubRef = r.getRef() // `OuterEntitlement` is defined to map to `SubEntitlement`, so this access yields a value of type `auth(SubEntitlement) &SubResource`
Entitlement mappings do not need to be 1:1; it is perfectly valid to define a mapping like so:
entitlement mapping M {
A -> C
B -> C
A -> D
}
If this mapping were used to define the auth
access of a nested resource reference, one could obtain a C
-entitled reference to that nested resource with either an
A
or a B
entitled outer reference. Conversely, an A
-entitled outer reference would yield a nested reference entitled for both C
and D
. Specifically, if the
field with the nested reference were called foo
, then we'd get these different outputs for differently typed ref
inputs:
- if
ref
was typed as&Outer
, thenref.foo
would just be an&Inner
- if
ref
was typed asauth(A) Outer
, thenref.foo
would beauth(C, D) &Inner
, sinceA
maps to both of these outputs - if
ref
was typed asauth(B) Outer
, thenref.foo
would beauth(C) &Inner
, sinceB
maps only toC
- if
ref
was typed asauth(A | B) Outer
, thenref.foo
would beauth(C) &Inner
, sinceC
is mapped to by both of these inputs
Note, however, that because the |
and ,
typed entitlement sets cannot be mixed (this is a restriction for simplicity more than anything else), it is not possible to use non 1:1
mappings in situations where their output would be an unrepresentable entitlement set. So with this mapping:
entitlement mapping M {
E -> A
E -> B
F -> C
F -> D
}
We would have the following access relationship:
- if
ref
was typed as&auth(E) Outer
, thenref.foo
would beauth(A, B) &Inner
- if
ref
was typed as&auth(F) Outer
, thenref.foo
would beauth(C, D) &Inner
- if
ref
was typed as&auth(E, F) Outer
, thenref.foo
would beauth(A, B, C, D) &Inner
- However, if
ref
was typed as&auth(E | F) Outer
, thenref.foo
would result in a static error, as this would beauth((A, B) | (C, D)) &Inner
, which is not representable in Cadence
It is also important to note that when using a mapping, when the mapped field is initialized, it must be initialized with a concrete reference value that is fully-entitled to the output of the mapping. So, given the following mapping:
entitlement mapping Map {
A -> B
C -> D
C -> E
}
resource SubResource { ... }
resource OuterResource {
access(Map) let ref: auth(Map) &SubResource
init(ref: auth(B) &SubResource) {
self.ref = ref // this would fail, as `ref` is only entitled to `B`
}
}
If this were to succeed, then a user could create an OuterResource
with only a B
-entitled reference to SubResource
, and use the mapping on it to get a D
-entitled
reference to the SubResource
, since the owner of the OuterResource
can get any entitled reference to it. In order to be a valid initial value for the self.ref
field here,
the ref
argument to the constructor would need to have type auth(B, D, E) &SubResource
. This is most easily achievable if the creator of the OuterResource
is also
the owner of the inner resource.
Attachments would interact with entitlements and access-limited members in a nuanced but intuitive manner, using the entitlement mapping feature described above. Instead
of requiring that all attachment declarations are pub
, they can additionally be declared with an access(E)
modifier, where E
is the name of an entitlement mapping.
When declared with an entitlement mapping access, the attachment's entitlements are propagated from the entitlements of its base
value according to the mapping. Attachments would remain pub
accessible, but would permit the declaration of access
-limited members. So, for example, given some declarations like:
entitlement E
entitlement F
entitlement mapping M {
E -> F
}
access(M) attachment A for R {}
given a reference to r
called ref
, when ref
has type &R
then ref[A]
has type &A?
, while when ref
has type auth(E) &R
then ref[A]
has type auth(F) &A?
. Additionally, as
owned values are considered fully entitled, accessing A
directly off of an @R
-typed value r
will yield a reference to A
that is fully entitled, that is, an A
reference that
is authorized for the entire image of M
.
Within the declaration of the attachment itself, the members of the attachments can use any entitlements that exist in the image of M
, but cannot use other entitlements as the
attachment access semantics would make it impossible to obtain a reference to the attachment with entitlements not in the image of M
.
Within A
's member functions, the self
value is an &A
reference that is fully entitled to the image of M
. This results in similar behavior to regular composite declarations,
where self
is fully entitled because it is an owned value.
Meanwhile, the base
value would have slightly more complex semantics. In the default case, the base
value is an unentitled reference, which will prevent
attachment authors from using the base
to access restricted functionality on the base resource. However, if the author of an attachment needs a specific entitlement on the base
in order to implement their functionality, they can explicitly require it in the declaration of the attachment using a new require entitlement X
syntax. This will require that
the attachment explicitly be given an entitlement to X
on creation, and thus makes X
an available entitlement on the base
in the declaration. So in the following example:
entitlement E
entitlement F
entitlement X
entitlement Y
entitlement mapping M {
E -> F
X -> Y
}
access(M) attachment A for R {
require entitlement X
// ...
}
base
will be considered to possess an X
entitlement within all the members of A
.
Creating attachments that require certain entitlements can be done with an extension to the original attach
expression: attach A() to v with X, Y, ...
. Here the
A()
is an attachment constructor call as before, while v
is the value to which the attachment is to be attached. However, after this the user may optionally
provide a with
followed by a list of entitlements which they wish to give to A
. This list must superset the list of entitlements required by A
's declaration,
or the attach expression will fail to typecheck. This way the user must be explicit about what permissions they are giving to each attachment they create.
Putting all of these rules together, we can see that given the following declaration:
entitlement Withdraw
entitlement ConvertAndWithdraw
entitlement mapping ConverterMap {
Withdraw -> ConvertAndWithdraw
}
interface Provider {
access(Withdraw) fun withdraw(_ amount: UFix64): @Vault
}
access(ConverterMap) attachment CurrencyConverter for Provider {
require entitlement Withdraw
access(all) fun convert(_ amount: UFix64): UFix64 {
// ...
}
access(all) fun convertVault(_ vault: @Vault): @Vault {
vault.balance = self.convert(vault.balance)
return <-vault
}
access(ConvertAndWithdraw) fun withdraw(_ amount: UFix64): @Vault {
let convertedAmount = self.convert(amount)
// this is permitted because the attachment declaration explicitly requires an entitlement to `Withdraw`
return <-base.withdraw(amount: amount)
}
access(all) fun deposit (from: @Vault) {
// cast is permissable under the new reference casting rules
let baseReceiverOptional = base as? &{Receiver}
if let baseReceiver = baseReceiverOptional {
let convertedVault <- self.convertVault(<-from)
// this is ok because `deposit` has `all` access
baseReceiver.deposit(from: <-convertedVault)
}
}
}
let vault <- attach CurrencyConverter() to <-create Vault(balance: /*...*/) with Withdraw
let authVaultReference = &vault as auth(Withdraw) &Vault
let converterRef = authVaultReference[CurrencyConverter]! // has type auth(ConvertAndWithdraw) &CurrencyConverter, can call `withdraw`
This dramatically increases the complexity of references and capabilities by requiring users to think not only about which types they are creating the references with, but also which entitlements they are providing on each reference they create. It also increases the implementation complexity of the type system significantly by adding an additional dimension to the reference type's subtype heirarchy.
This would encourage users to use the new auth
access modifier to restrict access on their contracts and resources; and use different
entitlement sets to control access.
In prior versions of Cadence, the Vault
heirarchy might be written like so:
access(all) entitlement Withdraw
access(all) resource interface Provider {
access(all) fun withdraw(amount: UFix64): @Vault {
// ...
}
}
access(all) resource interface Receiver {
access(all) fun deposit(from: @Vault) {
// ...
}
}
access(all) resource interface Balance {
access(all) var balance: UFix64
}
access(all) resource Vault: Provider, Receiver, Balance {
access(all) fun withdraw(amount: UFix64): @Vault {
// ...
}
access(all) fun deposit(from: @Vault) {
// ...
}
access(all) var balance: UFix64
}
Here, the access control to the Vault
is determined by the type of reference issued to it:
someone with a &{Provider}
reference can call withdraw
, but someone with a &{Receiver}
cannot.
With this new design, the Vault
heirarchy might be written like so:
access(all) entitlement Withdraw
access(all) resource interface Provider {
access(Withdraw) fun withdraw(amount: UFix64): @Vault {
// ...
}
}
access(all) resource interface Receiver {
access(all) fun deposit(from: @Vault) {
// ...
}
}
access(all) resource interface Balance {
access(all) var balance: UFix64
}
access(all) resource Vault: Provider, Receiver, Balance {
access(Withdraw) fun withdraw(amount: UFix64): @Vault {
// ...
}
access(all) fun deposit(from: @Vault) {
// ...
}
access(all) var balance: UFix64
}
Note how the primary difference here is that withdraw
now requires a Withdraw
entitlement.
Then, someone with a &{Balance}
reference would be able to cast this to a &{Receiver}
and call deposit
on it.
They would also be able to cast this to a &Vault
reference, but because this reference is not entitled to Withdraw
,
they would be unable to call the withdraw
function. However, if a user possessed a auth(Withdraw) &{Balance}
reference,
they would be able to cast this to auth(Withdraw) &Vault
and call withdraw
.
This would not be backwards compatible with existing code, and thus would need to be part of the Stable Cadence release.
Most contracts would need to be audited and rewritten to use the new access modifiers, as they would immediately become
vulnerable to having pub
fields accessed from now-downcastable references.
There is a huge user impact to rolling out this change; every single contract would become invalid, and would need to be manually audited
by the authors in order to be used. This is because all code previously written using the old model of access control (wherein giving someone
a &{Balance}
would prevent them from calling pub
functions on Provider
like withdraw
), would become invalidated, allowing everyone
to call pub
members like withdraw
unless those methods were updated to be access(Withdraw)
.
There are two cases that need handling to migrate to this new paradigm: contracts and data.
Existing references in storage (i.e. Capability
values) would need to be migrated, as they would no longer be semantically valid under the new system.
The simplest way to do this would be to ensure existing capabilities stay usable by converting existing capabilities and links' reference types to auth
references, along with creating
entitlements that map 1:1 with existing interfaces, e.g. Capability<&Vault{Withdraw}>
-> Capability<auth(Withdraw) &Vault>
. Specifically, all existing
references would become auth
with regard to their entire borrow type, as this does not add any additional power to these Capability
values that did not previously exist.
For example, &Vault{Provider}
previously had the ability to call withdraw
, which was pub
on Provider
. After this change, it will still have the ability to call withdraw
,
as it is now access(Withdraw)
on Provider
, but the reference now has auth(Withdraw)
access to that type.
However, this does not handle the previously mentioned problem wherein existing contracts become vulnerable to exploitation, as all their pub
functions would become
accessible to anybody with any kind of reference to a contract or resource.
To handle this case we woukld "freeze" all existing contracts, preventing calling any code defined in them or interacting with their data until they are updated at least once after the release of this feature (it's also possible that given the large number of breaking changes being released with Cadence 1.0, this restriction would happen automatically and would not need special handling). Developers would be encouraged to give their contracts and resources the proper access modifiers.
A previous version of this idea involved having the implementors of concrete types like Vault
specify in their interface conformance list
which interfaces were auth
for that concrete type. So, for example, one would write something like (using strawman syntax):
pub resource interface A {
pub fun foo()
}
pub resource interface B {
pub fun bar()
}
pub resource R: auth A, B {
pub fun foo() {}
pub fun bar() {}
}
Then, given a reference of type &R
, only the functions and fields defined on B
would be accessible, since the reference is non-auth
. In
order to access the fields and functions on A
, one would need an auth
reference to R
, since A
was declared as auth
in R
's conformances.
The upside of this proposal was that there was a single decision point, in that the implementor of the concrete type needed to decide which interfaces
would be auth
and which would be not, and the creators of the interfaces and the users of the concrete types would have no decisions to make other than
whether or not the reference they are creating should be auth
or not.
The drawback of this compared to the proposal in this FLIP is that it lacks granularity; an entire interface must be made auth
or not, whereas in the
FLIP's proposal individual functions can be declared with auth
access or pub
access. This allows the creator of the interface (and then later, the
creator of the reference type) to exert more fine-grained control over what is accessible. By contrast, the implementor of the concrete type has little
to no say in the access-control on their value beyond selecting which interfaces they wish to conform to.
entitlement
s function similarly to facets in the Object Capabilities model.
-
One potential change this unlocks would be to restrict the creation of Capabilities based on the entitlements they contain. We could restrict the
public
domain to be only for non-entitled Capabilities, while theprivate
domain would be only forauth
-references, which would prevent accidentally allowing anybody to get access to yourwithdraw
function, for example. -
We've discussed the value of being able to declare entitlements in terms of existing entitlements. I.e. if I have some entitlement
A
andB
, it would be useful to be able to declareC
as the intersection ofA
andB
or the union. Additionally it would be valuable to be able to declareC
as a subset ofA
orB
's functions to reduce code repetition. However, this is not a necessary part of the initial implementation of this FLIP, as it can be added later in a non-breaking way. As such this feature is out of scope for this FLIP.