Advanced
Signatures

Signatures

The Safe supports different types of signatures. All signatures are combined into a single bytes and transmitted to the contract when a transaction should be executed.

Encoding

Each signature has a constant length of 65 bytes. If more data is necessary it can be appended to the end of concatenated constant data of all signatures. The position is encoded into the constant length data.

Constant part per signature: {(max) 64-bytes signature data}{1-byte signature type}

All the signatures are sorted by the signer address and concatenated.

ECDSA signature

31 > signature type > 26

To be able to have the ECDSA signature without the need of additional data we use the signature type byte to encode v.

Constant part:

{32-bytes r}{32-bytes s}{1-byte v}

r, s and v are the required parts of the ECDSA signature to recover the signer.

eth_sign signature

signature type > 30

To be able to use eth_sign we need to take the parameters r, s and v from calling eth_sign and set v = v + 4

Constant part:

{32-bytes r}{32-bytes s}{1-byte v}

r, s and vare the required parts of the ECDSA signature to recover the signer. v will be subtracted by 4 to calculate the signature.

Contract signature (EIP-1271)

signature type == 0

Constant part:

{32-bytes signature verifier}{32-bytes data position}{1-byte signature type}

Signature verifier - Padded address of the contract that implements the EIP-1271 interface to verify the signature

Data position - Position of the start of the signature data (offset relative to the beginning of the signature data)

Signature type - 0

Dynamic part (solidity bytes):

{32-bytes signature length}{bytes signature data}

Signature data - Signature bytes that are verified by the signature verifier

The method signMessage can be used to mark a message as signed on-chain.

Pre-validated signatures

signature type == 1

Constant Part:

{32-bytes hash validator}{32-bytes ignored}{1-byte signature type}

Hash validator - Padded address of the account that pre-validated the hash that should be validated. The Safe keeps track of all hashes that have been pre-validated. This is done with a mapping address to mapping of bytes32 to boolean where it's possible to set a hash as validated by a certain address (hash validator). To add an entry to this mapping use approveHash. Also if the validator is the sender of transaction that executed the Safe transaction it's not required to use approveHash to add an entry to the mapping. (This can be seen in the Team Edition tests (opens in a new tab))

Signature type - 1

Examples

Assuming that three signatures are required to confirm a transaction where one signer uses an EOA to generate a ECDSA signature, another a contract signature and the last a pre-validated signature:

We assume that the following addresses generate the following signatures:

  1. 0x3 (EOA address) -> bde0b9f486b1960454e326375d0b1680243e031fd4fb3f070d9a3ef9871ccfd5 (r) + 7d1a653cffb6321f889169f08e548684e005f2b0c3a6c06fba4c4a68f5e00624 (s) + 1c (v)
  2. 0x1 (EIP-1271 validator contract address) -> 0000000000000000000000000000000000000000000000000000000000000001 (address) + 00000000000000000000000000000000000000000000000000000000000000c3 (dynamic position) + 00 (signature type)
    • The contract takes the following bytes (dynamic part) for verification 00000000000000000000000000000000000000000000000000000000deadbeef
  3. 0x2 (Validator address) -> 0000000000000000000000000000000000000000000000000000000000000002 (address) +0000000000000000000000000000000000000000000000000000000000000000 (padding - not used) + 01 (signature type)

The constant parts need to be sorted so that the recovered signers are sorted ascending (natural order) by address (not checksummed).

The signatures bytes used for execTransaction would therefore be the following:


_10
"0x" +
_10
"000000000000000000000000000000000000000000000000000000000000000100000000000000000000000000000000000000000000000000000000000000c300" + // encoded EIP-1271 signature
_10
"0000000000000000000000000000000000000000000000000000000000000002000000000000000000000000000000000000000000000000000000000000000001" + // encoded pre-validated signature
_10
"bde0b9f486b1960454e326375d0b1680243e031fd4fb3f070d9a3ef9871ccfd57d1a653cffb6321f889169f08e548684e005f2b0c3a6c06fba4c4a68f5e006241c" + // encoded ECDSA signature
_10
"0000000000000000000000000000000000000000000000000000000000000004deadbeef" // length of bytes + data of bytes

Was this page helpful?