🎫Connect The Graph Phat Contract to EVM Consumer Contract

In the previous section we described how The Graph Starter Kit works. Let's revisit the diagram and description below.

Overview

The diagram above displays the user journey of (1) Alice requesting a trust score for Eve from the EVM Consumer Contract. When (2) a new action request is added to the queue, (3) the Phala Network Phat Contract will pop the action off the queue and perform the off-chain work to compute a trust score.

First (4) the Phat Contract will create a batchHttpRequest to query 3 separate subgraph endpoints to determine if Eve has any ERC-721 NFTs, a NounsDAO NFT, an ENS Domain, and if any accounts delegate votes to Eve on Snapshot. Once this data is returned then (5) the Phat Contract will compute a score based on some scoring criteria. You can view the code here. Lastly, (6) the trust score for Eve has been returned to the EVM Consumer Contract and Eve's score is set in the Consumer Contract's storage for anyone to query.

Getting Started

If you have not setup The Graph code repo locally, go back to the Quick Start section and follow the initial setup steps.

Before Deployment

Before deploying, you will need to either export your Phala Account via polkadot.js extension or store your private key in the .env (optionally you can manually enter during deployment as well).

Option 1: Export Polkadot account as json file

Go to your browser and click on the polkadot.js extension. Select your account and click "Export Account". Next, you will be prompted for your password before saving the file to your project directory. Note this is what will be set to POLKADOT_WALLET_ACCOUNT_PASSPHRASE. Make sure to save the file as polkadot-account.json in the root of your project directory.

Option 2: Set mnemonic phrase to POLKADOT_WALLET_SURI

After creating your Phala Profile, set your .env variable POLKADOT_WALLET_SURI to the mnemonic phrase from generating the new Polkadot Account.

Here is a screenshot of how to set POLKADOT_WALLET_SURI:

Deployments (Local, Testnet, Mainnet)

This guide will be separated into 3 tabs including:

  • Local: Local Testnet Deployment

  • Testnet: PoC6 Testnet & EVM Chain Testnet Deployment

  • Mainnet: Phala Mainnet & EVM Chain Mainnet Deployment

Secrets (What are Secrets?):

  • apiUrl - The endpoint base URL to the separate subgraph endpoints hosted on The Graph

  • apiKey - an API key created on The Graph

{
    "apiUrl": "https://gateway.thegraph.com/api/",
    "apiKey": "cd22a01e5b7f9828cddcb52caf03ee79"
}

In the previous Quick Start section, we installed the dependencies and ran 2 separate tests locally, but these tests were not run against a live local testnet.

This section will describe the process of:

  • Start up a local hardhat node

  • Deploy the EVM Consumer Contract to the local testnet

  • Run the @phala/fn watch command to run a local instance of The Graph Phat Contract

  • Simulate a sample request by executing npm run localhost-push-request

  • See the The Graph Phat Contract reply with a result to the EVM Consumer Contract

Testing Locally

First step is to install the package dependencies with the following command:

npm install

With all the dependencies installed, we are ready to build The Graph Phat Contract.

npx @phala/fn build

To simulate the expected result locally, run the Phat Contract script now with this command:

Use decode and encode playground at https://playground.ethers.org.

npx @phala/fn run dist/index.js -a 0x0000000000000000000000000000000000000000000000000000000000000001000000000000000000000000de1683287529b9b4c3132af8aad210644b259cfd '{"apiUrl": "https://gateway.thegraph.com/api/", "apiKey": "cd22a01e5b7f9828cddcb52caf03ee79"}'

Here is the expected output of this call where the encoded call will request a trust score result for the address hashwarlock.eth. The result is 16.

npx @phala/fn run dist/index.js -a 0x0000000000000000000000000000000000000000000000000000000000000001000000000000000000000000de1683287529b9b4c3132af8aad210644b259cfd '{"apiUrl": "https://gateway.thegraph.com/api/", "apiKey": "cd22a01e5b7f9828cddcb52caf03ee79"}'
handle req: 0x0000000000000000000000000000000000000000000000000000000000000001000000000000000000000000de1683287529b9b4c3132af8aad210644b259cfd
[1]: 0xdE1683287529B9B4C3132af8AaD210644B259CfD
Request received for profile 0xdE1683287529B9B4C3132af8AaD210644B259CfD
{"data":{"account":{"id":"0xde1683287529b9b4c3132af8aad210644b259cfd","ERC721tokens":[{"id":"0x57f1887a8bf19b14fc0df6fd9b2acc9af147ea85/0x1c1f4a45a4e02789c110c3771dac92f92a9498f8016c8a6fefcf4d117603d277","identifier":"12720044626168998947918556575205935844278720363082869735772187331921734521463","contract":{"id":"0x57f1887a8bf19b14fc0df6fd9b2acc9af147ea85"}},{"id":"0x57f1887a8bf19b14fc0df6fd9b2acc9af147ea85/0x8078323181367af7deac0d322698034faf460e340e8bcfde9469655682e637b5","identifier":"58108412688393927589019354909227047790661963669020827084996334376817871173557","contract":{"id":"0x57f1887a8bf19b14fc0df6fd9b2acc9af147ea85"}},{"id":"0x57f1887a8bf19b14fc0df6fd9b2acc9af147ea85/0x94286edc2c3c238c35cdeb1775c329c554fdd9df064c9ce86dad277d74a1667","identifier":"4188358787553715944519630366137388336862205236752970957509174576540310705767","contract":{"id":"0x57f1887a8bf19b14fc0df6fd9b2acc9af147ea85"}},{"id":"0x57f1887a8bf19b14fc0df6fd9b2acc9af147ea85/0xba4f236216fb3975855ea0572713af6335f992a110b7ea520a05f06440768b05","identifier":"84270014960222169810810437893688928763911475898905132261856796581223088950021","contract":{"id":"0x57f1887a8bf19b14fc0df6fd9b2acc9af147ea85"}}]}}}
ERC-721 NFTs owned on ETH Check... Result [4]
{"data":{"account":{"id":"0xde1683287529b9b4c3132af8aad210644b259cfd","ERC721tokens":[]}}}
Has a NounsDAO NFT Check... Result [4]
{"data":{"account":{"domains":[{"name":"warlox.eth"},{"name":"hashwarlock.eth"},{"name":"[4df3c10b7d9a8cfdbb386728d398389b2dfbcd0f66e4c1ee612a7c82f649ac0a].addr.reverse"}]}}}
Has ENS Domains Check... Result [16]
{"data":{"delegations":[]}}
Has Delegated Votes on Snapshot Check... Result [16]
response: 0,1,16
{"output":"0x000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000010000000000000000000000000000000000000000000000000000000000000010"}

Deploy a Local Hardhat Node

Now that we have a simple understanding of the expected functionality of the Phat Contract, we can now take our tests to a local testnet. Here we will use Hardhat to deploy the EVM Consumer Contract then listen from new action requests and reply with built Phat Contract script.

First we will start a local hardhat node.

npm run localhost-node

With our hardhat node running locally, we can now deploy the OracleConsumerContract.sol contract to the local hardhat network.

npm run localhost-deploy
npm run localhost-deploy
> the-graph-phat-contract@1.0.0 localhost-deploy
> hardhat run --network localhost ./scripts/localhost/deploy.ts

Compiled 18 Solidity files successfully (evm target: london).
Deploying...
Deployed { consumer: '0x5FbDB2315678afecb367f032d93F642f64180aa3' }

Make sure to copy the deployed contract address when you deploy your own contract locally. Note you contract address will be different than 0x5FbDB2315678afecb367f032d93F642f64180aa3. We will now start watching the hardhat node deployed contract for any new requests from The Graph Phat Contract.

npx @phala/fn watch 0x5FbDB2315678afecb367f032d93F642f64180aa3 artifacts/contracts/OracleConsumerContract.sol/OracleConsumerContract.json dist/index.js -a '{"apiUrl": "https://gateway.thegraph.com/api/", "apiKey": "cd22a01e5b7f9828cddcb52caf03ee79"}'
npx @phala/fn watch 0x5FbDB2315678afecb367f032d93F642f64180aa3 artifacts/contracts/OracleConsumerContract.sol/OracleConsumerContract.json dist/index.js -a '{"apiUrl": "https://gateway.thegraph.com/api/", "apiKey": "cd22a01e5b7f9828cddcb52caf03ee79"}'
Listening for OracleConsumerContract MessageQueued events...

Let’s now make a new request and see what happens with the listener’s output. In separate tab, you will push a request with the following.

Note: The file can be edited here where you can change the target address.

LOCALHOST_CONSUMER_CONTRACT_ADDRESS=0x5FbDB2315678afecb367f032d93F642f64180aa3 npm run localhost-push-request
LOCALHOST_CONSUMER_CONTRACT_ADDRESS=0x5FbDB2315678afecb367f032d93F642f64180aa3 npm run localhost-push-request
> the-graph-phat-contract@1.0.0 localhost-push-request
> hardhat run --network localhost ./scripts/localhost/push-request.ts

Pushing a request...
Received event [ResponseReceived]: {
  reqId: BigNumber { value: "1" },
  target: '0x011c23b3AadAf3D4991f3aBeE262A34d18e9fdb5',
  value: BigNumber { value: "70" }
}

If you check back the tab where the Phat Contract is listening for new requests, the console log may look similar to below:

Received event [MessageQueued]: {
  tail: 0n,
  data: '0x0000000000000000000000000000000000000000000000000000000000000001000000000000000000000000011c23b3aadaf3d4991f3abee262a34d18e9fdb5'
}
handle req: 0x0000000000000000000000000000000000000000000000000000000000000001000000000000000000000000011c23b3aadaf3d4991f3abee262a34d18e9fdb5
[1]: 0x011c23b3AadAf3D4991f3aBeE262A34d18e9fdb5
Request received for profile 0x011c23b3AadAf3D4991f3aBeE262A34d18e9fdb5
{"data":{"account":{"id":"0x011c23b3aadaf3d4991f3abee262a34d18e9fdb5","ERC721tokens":[{"id":"0x00703f9b11f2ac02d391a11e7b97c6ee80cd8563/0x2bd","identifier":"701","contract":{"id":"0x00703f9b11f2ac02d391a11e7b97c6ee80cd8563"}},{"id":"0x008f5a13d37db25d1bf7e7115747450e12e471b9/0x13fc","identifier":"5116","contract":{"id":"0x008f5a13d37db25d1bf7e7115747450e12e471b9"}},{"id":"0x0144ecf966096108b03148d0071df6c70c051a52/0x149f","identifier":"5279","contract":{"id":"0x0144ecf966096108b03148d0071df6c70c051a52"}},{"id":"0x015fcab6a246cfc0679c33ef0b9d9ef947d0bde4/0x1782","identifier":"6018","contract":{"id":"0x015fcab6a246cfc0679c33ef0b9d9ef947d0bde4"}},{"id":"0x0191c41dbceb20a612b25137133ca719e84f7933/0x11f","identifier":"287","contract":{"id":"0x0191c41dbceb20a612b25137133ca719e84f7933"}},{"id":"0x0191c41dbceb20a612b25137133ca719e84f7933/0x120","identifier":"288","contract":{"id":"0x0191c41dbceb20a612b25137133ca719e84f7933"}},{"id":"0x0191c41dbceb20a612b25137133ca719e84f7933/0xd8e","identifier":"3470","contract":{"id":"0x0191c41dbceb20a612b25137133ca719e84f7933"}},{"id":"0x0208517aa68e7c72769af76f4cfdeea9fa4ef4b9/0x213","identifier":"531","contract":{"id":"0x0208517aa68e7c72769af76f4cfdeea9fa4ef4b9"}},{"id":"0x026dce20bf77e08ca8aceb6b239cc54bb9d638ac/0x237","identifier":"567","contract":{"id":"0x026dce20bf77e08ca8aceb6b239cc54bb9d638ac"}},{"id":"0x032d96756697af7ec02ce03d39001b39f7a5d849/0x1a4","identifier":"420","contract":{"id":"0x032d96756697af7ec02ce03d39001b39f7a5d849"}},{"id":"0x038cc0f103c380400482d87be0d3abcc4d9b2225/0x2ed","identifier":"749","contract":{"id":"0x038cc0f103c380400482d87be0d3abcc4d9b2225"}},{"id":"0x039483c56aad5ee68a92eff1a1b666f2893c623e/0xf9d","identifier":"3997","contract":{"id":"0x039483c56aad5ee68a92eff1a1b666f2893c623e"}},{"id":"0x03ef30e1aee25abd320ad961b8cd31aa1a011c97/0x17a1","identifier":"6049","contract":{"id":"0x03ef30e1aee25abd320ad961b8cd31aa1a011c97"}},{"id":"0x03ef30e1aee25abd320ad961b8cd31aa1a011c97/0xcf1","identifier":"3313","contract":{"id":"0x03ef30e1aee25abd320ad961b8cd31aa1a011c97"}},{"id":"0x04b4786c3bb42387235a63628b7a4cb178817429/0x1000","identifier":"4096","contract":{"id":"0x04b4786c3bb42387235a63628b7a4cb178817429"}},{"id":"0x04b4786c3bb42387235a63628b7a4cb178817429/0x100b","identifier":"4107","contract":{"id":"0x04b4786c3bb42387235a63628b7a4cb178817429"}},{"id":"0x04b4786c3bb42387235a63628b7a4cb178817429/0x107d","identifier":"4221","contract":{"id":"0x04b4786c3bb42387235a63628b7a4cb178817429"}},{"id":"0x04b4786c3bb42387235a63628b7a4cb178817429/0x10de","identifier":"4318","contract":{"id":"0x04b4786c3bb42387235a63628b7a4cb178817429"}},{"id":"0x04b4786c3bb42387235a63628b7a4cb178817429/0x11aa","identifier":"4522","contract":{"id":"0x04b4786c3bb42387235a63628b7a4cb178817429"}},{"id":"0x04b4786c3bb42387235a63628b7a4cb178817429/0x12c5","identifier":"4805","contract":{"id":"0x04b4786c3bb42387235a63628b7a4cb178817429"}},{"id":"0x04b4786c3bb42387235a63628b7a4cb178817429/0x139a","identifier":"5018","contract":{"id":"0x04b4786c3bb42387235a63628b7a4cb178817429"}},{"id":"0x04b4786c3bb42387235a63628b7a4cb178817429/0x144","identifier":"324","contract":{"id":"0x04b4786c3bb42387235a63628b7a4cb178817429"}},{"id":"0x04b4786c3bb42387235a63628b7a4cb178817429/0x144e","identifier":"5198","contract":{"id":"0x04b4786c3bb42387235a63628b7a4cb178817429"}},{"id":"0x04b4786c3bb42387235a63628b7a4cb178817429/0x14a9","identifier":"5289","contract":{"id":"0x04b4786c3bb42387235a63628b7a4cb178817429"}},{"id":"0x04b4786c3bb42387235a63628b7a4cb178817429/0x14c0","identifier":"5312","contract":{"id":"0x04b4786c3bb42387235a63628b7a4cb178817429"}},{"id":"0x04b4786c3bb42387235a63628b7a4cb178817429/0x1531","identifier":"5425","contract":{"id":"0x04b4786c3bb42387235a63628b7a4cb178817429"}},{"id":"0x04b4786c3bb42387235a63628b7a4cb178817429/0x153e","identifier":"5438","contract":{"id":"0x04b4786c3bb42387235a63628b7a4cb178817429"}},{"id":"0x04b4786c3bb42387235a63628b7a4cb178817429/0x1569","identifier":"5481","contract":{"id":"0x04b4786c3bb42387235a63628b7a4cb178817429"}},{"id":"0x04b4786c3bb42387235a63628b7a4cb178817429/0x158c","identifier":"5516","contract":{"id":"0x04b4786c3bb42387235a63628b7a4cb178817429"}},{"id":"0x04b4786c3bb42387235a63628b7a4cb178817429/0x15ee","identifier":"5614","contract":{"id":"0x04b4786c3bb42387235a63628b7a4cb178817429"}},{"id":"0x04b4786c3bb42387235a63628b7a4cb178817429/0x161f","identifier":"5663","contract":{"id":"0x04b4786c3bb42387235a63628b7a4cb178817429"}},{"id":"0x04b4786c3bb42387235a63628b7a4cb178817429/0x164d","identifier":"5709","contract":{"id":"0x04b4786c3bb42387235a63628b7a4cb178817429"}},{"id":"0x04b4786c3bb42387235a63628b7a4cb178817429/0x16db","identifier":"5851","contract":{"id":"0x04b4786c3bb42387235a63628b7a4cb178817429"}},{"id":"0x04b4786c3bb42387235a63628b7a4cb178817429/0x1735","identifier":"5941","contract":{"id":"0x04b4786c3bb42387235a63628b7a4cb178817429"}},{"id":"0x04b4786c3bb42387235a63628b7a4cb178817429/0x1742","identifier":"5954","contract":{"id":"0x04b4786c3bb42387235a63628b7a4cb178817429"}},{"id":"0x04b4786c3bb42387235a63628b7a4cb178817429/0x1764","identifier":"5988","contract":{"id":"0x04b4786c3bb42387235a63628b7a4cb178817429"}},{"id":"0x04b4786c3bb42387235a63628b7a4cb178817429/0x1793","identifier":"6035","contract":{"id":"0x04b4786c3bb42387235a63628b7a4cb178817429"}},{"id":"0x04b4786c3bb42387235a63628b7a4cb178817429/0x18b7","identifier":"6327","contract":{"id":"0x04b4786c3bb42387235a63628b7a4cb178817429"}},{"id":"0x04b4786c3bb42387235a63628b7a4cb178817429/0x1926","identifier":"6438","contract":{"id":"0x04b4786c3bb42387235a63628b7a4cb178817429"}},{"id":"0x04b4786c3bb42387235a63628b7a4cb178817429/0x1940","identifier":"6464","contract":{"id":"0x04b4786c3bb42387235a63628b7a4cb178817429"}},{"id":"0x04b4786c3bb42387235a63628b7a4cb178817429/0x1988","identifier":"6536","contract":{"id":"0x04b4786c3bb42387235a63628b7a4cb178817429"}},{"id":"0x04b4786c3bb42387235a63628b7a4cb178817429/0x199a","identifier":"6554","contract":{"id":"0x04b4786c3bb42387235a63628b7a4cb178817429"}},{"id":"0x04b4786c3bb42387235a63628b7a4cb178817429/0x1a8e","identifier":"6798","contract":{"id":"0x04b4786c3bb42387235a63628b7a4cb178817429"}},{"id":"0x04b4786c3bb42387235a63628b7a4cb178817429/0x1b4","identifier":"436","contract":{"id":"0x04b4786c3bb42387235a63628b7a4cb178817429"}},{"id":"0x04b4786c3bb42387235a63628b7a4cb178817429/0x1be3","identifier":"7139","contract":{"id":"0x04b4786c3bb42387235a63628b7a4cb178817429"}},{"id":"0x04b4786c3bb42387235a63628b7a4cb178817429/0x1beb","identifier":"7147","contract":{"id":"0x04b4786c3bb42387235a63628b7a4cb178817429"}},{"id":"0x04b4786c3bb42387235a63628b7a4cb178817429/0x1d83","identifier":"7555","contract":{"id":"0x04b4786c3bb42387235a63628b7a4cb178817429"}},{"id":"0x04b4786c3bb42387235a63628b7a4cb178817429/0x1dc0","identifier":"7616","contract":{"id":"0x04b4786c3bb42387235a63628b7a4cb178817429"}},{"id":"0x04b4786c3bb42387235a63628b7a4cb178817429/0x1dd8","identifier":"7640","contract":{"id":"0x04b4786c3bb42387235a63628b7a4cb178817429"}},{"id":"0x04b4786c3bb42387235a63628b7a4cb178817429/0x1e45","identifier":"7749","contract":{"id":"0x04b4786c3bb42387235a63628b7a4cb178817429"}}]}}}
ERC-721 NFTs owned on ETH Check... Result [50]
{"data":{"account":{"id":"0x011c23b3aadaf3d4991f3abee262a34d18e9fdb5","ERC721tokens":[]}}}
Has a NounsDAO NFT Check... Result [50]
{"data":{"account":{"domains":[{"name":"sissitian.eth"},{"name":"ariahuang.eth"},{"name":"hugovault.eth"},{"name":"hugosu.eth"},{"name":"dominichuang.eth"}]}}}
Has ENS Domains Check... Result [70]
{"data":{"delegations":[]}}
Has Delegated Votes on Snapshot Check... Result [70]
response: 0,1,70
JS Execution output: 0x000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000010000000000000000000000000000000000000000000000000000000000000046

🎉 Congratulations!

You've completed deploying and testing The Graph Phat Contract successfully in a local testnet. Now let's move to deploying to an EVM Testnet and connecting a deployed Phat Contract on PoC6 Testnet to visualize how this works autonomously.

Last updated

Logo

COPYRIGHT © 2024 PHALA.LTD ALL RIGHTS RESERVED. May Phala be with you!