Skip to content

Latest commit

 

History

History
135 lines (96 loc) · 6.18 KB

README.md

File metadata and controls

135 lines (96 loc) · 6.18 KB

TheOffchainResolver.sol

TOR — a trustless universal hybrid off-chain ENS and DNS resolver contract and protocol.

TOR Protocol

  • bytes requestData = calldata from CCIP-Read
  • bytes responseData = the answer to that request
  • bytes32 requestHash = keccak256 of requestData
  • bytes32 responseHash = keccak256 of responseData
  • uint64 expires = expiration time in seconds (Ethereum time, since 1/1/1970)
  • address resolver = TOR contract address
    • use different endpoints to service multiple resolvers (main vs test, DNS vs ENS)
  • bytes signedData = abi.encoded (resolver, expires, requestHash, responseHash)
  • bytes32 signedHash = keccak256 of signedData
  • bytes signature = signature of signedHash with private key
  • bytes data = abi.encoded (signature, expires, signedData)
  • reply with data

Implementations

Context Format

CONTEXT = ${SIGNER} ${ENDPOINT}

  • SIGNER = 0x-prefixed public address of signing key
    • ethers.computeAddress(new ethers.SigningKey("..."))
  • ENDPOINT = URL of your CCIP-Read server

Setup

  • "I have a DNS name"
    • DNS TXT = ENS1 ${TheOffchainResolver} ${CONTEXT}
      • Mainnet Example: ezccip.raffy.xyz
      • Context: 0xd00d726b2aD6C81E894DC6B87BE6Ce9c5572D2cd https://raffy.xyz/ezccip/
      • For full wildcard coverage, set @ (basename) and * (descendents)
  • "I have an ENS name"
    • ENS.setResolver() = ${TheOffchainResolver}
    • setText("ccip.context") = CONTEXT
      • Sepolia Example: ezccip.eth
      • Context: 0xd00d726b2aD6C81E894DC6B87BE6Ce9c5572D2cd https://raffy.xyz/ezccip/s

That's it! 🎉️

Features

  • If all requested records are available on-chain, the off-chain server is not invoked.
  • Supports resolve(multicall([...]))
  • Supports TOR Lensing
    • Prefix calldata with 0x000000FF to only get off-chain values (no on-chain values are accessed)
    • Prefix calldata with 0xFFFFFF00 to only get on-chain values (off-chain server is not invoked)
  • Supports text(), addr(), contenthash(), pubkey()
  • Uses adraffy/TinyKV.sol to minimize storage costs.

Fallbacks

  • Set addr(C) where C = keccak256("fallback")
    • 20 bytes — use another Resolver (on-chain)
    • 32 bytes — use another Namehash (on-chain)
    • 0 bytes Default — use _.[name]
    • Anything else — disabled
  • Priority: TOR > Fallback > Off-chain

On-chain Only

  • Use toggleOnchain(bytes32 node) to enable
  • When onchain(bytes32 node) is true, the name will never query the off-chain server for additional records and treat unset records as null.
  • Fallbacks still functional

DNSTORWithENSProtocol.sol

DNS part of TOR but uses standard ENS signing protocol that is compatible with official ENS Offchain Resolver.


eXclusivelyOnchainResolver.sol

XOR — a trustless on-chain wildcard resolver contract that translates ENSIP-10 calls into non-ENSIP-10 calls.

Usage

Append .onchain.eth to any ENS name and resolve!


OffchainTunnel.sol

An on-chain function registry for arbitrary CCIP-Read functions.

Function Registry

When selector is called with CCIP-Read, the calldata is forwarded to the endpoint and the response must be signed by signer. The function selector is associated with an (owner, index)–pair which points to a (endpoint, signer)–pair. The CCIP-Read exchange follows the TOR protocol.

  • claimAndSetContext(bytes4 selector, address signer, string calldata endpoint, uint96 index)
  • claim(bytes4 selector, uint256 index) + setContext(address signer, string calldata endpoint, uint96 index)

Gasless Debugging

call(address signer, string memory endpoint, bytes memory request) does the same thing as above, except the (signer, endpoint)–pair is provided.

Testing

All contracts have end-to-end adraffy/blocksmith tests.

  1. foundryup
  2. forge install
  3. npm i
  4. npm run test