LibContext
Conventions for working with context as a calling contract. All of this functionality is OPTIONAL but probably useful for the majority of use cases. By building and authenticating onchain, caller provided and signed contexts all in a standard way the overall usability of context is greatly improved for expression authors and readers. Any calling contract that can match the context expectations of an existing expression is one large step closer to compatibility and portability, inheriting network effects of what has already been authored elsewhere.
Functions
base
The base context is the msg.sender
and address of the calling contract.
As the interpreter itself is called via an external interface and may be
statically calling itself, it MAY NOT have any ability to inspect either
of these values. Even if this were not the case the calling contract
cannot assume the existence of some opcode(s) in the interpreter that
inspect the caller, so providing these two values as context is
sufficient to decouple the calling contract from the interpreter. It is
STRONGLY RECOMMENDED that even if the calling contract has "no context"
that it still provides this base to every eval
.
Calling contracts DO NOT need to call this directly. It is built and
merged automatically into the standard context built by build
.
function base() internal view returns (uint256[] memory);
Returns
Name | Type | Description |
---|---|---|
<none> | uint256[] | The msg.sender and address of the calling contract using this library, as a context-compatible array. |
hash
Standard hashing process over a single SignedContextV1
. Notably used
to hash a list as SignedContextV1[]
but could also be used to hash a
single SignedContextV1
in isolation. Avoids allocating memory by
hashing each struct field in sequence within the memory scratch space.
function hash(SignedContextV1 memory signedContext) internal pure returns (bytes32 hashed);
Parameters
Name | Type | Description |
---|---|---|
signedContext | SignedContextV1 | The signed context to hash. |
hash
Standard hashing process over a list of signed contexts. Situationally
useful if the calling contract wants to record that it has seen a set of
signed data then later compare it against some input (e.g. to ensure that
many calls of some function all share the same input values). Note that
unlike the internals of build
, this hashes over the signer and the
signature, to ensure that some data cannot be re-signed and used under
a different provenance later.
function hash(SignedContextV1[] memory signedContexts) internal pure returns (bytes32 hashed);
Parameters
Name | Type | Description |
---|---|---|
signedContexts | SignedContextV1[] | The list of signed contexts to hash over. |
Returns
Name | Type | Description |
---|---|---|
hashed | bytes32 | The hash of the signed contexts. |
build
Builds a standard 2-dimensional context array from base, calling and
signed contexts. Note that "columns" of a context array refer to each
uint256[]
and each item within a uint256[]
is a "row".
function build(uint256[][] memory baseContext, SignedContextV1[] memory signedContexts)
internal
view
returns (uint256[][] memory);
Parameters
Name | Type | Description |
---|---|---|
baseContext | uint256[][] | Anything the calling contract can provide which MAY include input from the msg.sender of the calling contract. The default base context from LibContext.base() DOES NOT need to be provided by the caller, this matrix MAY be empty and will be simply merged into the final context. The base context matrix MUST contain a consistent number of columns from the calling contract so that the expression can always predict how many unsigned columns there will be when it runs. |
signedContexts | SignedContextV1[] | Signed contexts are provided by the msg.sender but signed by a third party. The expression (author) defines who may sign and the calling contract authenticates the signature over the signed data. Technically build handles all the authentication inline for the calling contract so if some context builds it can be treated as authentic. The builder WILL REVERT if any of the signatures are invalid. Note two things about the structure of the final built context re: signed contexts: - The first column is a list of the signers in order of what they signed - The msg.sender can provide an arbitrary number of signed contexts so expressions DO NOT know exactly how many columns there are. The expression is responsible for defining e.g. a domain separator in a position that would force signed context to be provided in the "correct" order, rather than relying on the msg.sender to honestly present data in any particular structure/order. |