From 4c68daf4d0ac3812be2e729d1a6dc129d3c125da Mon Sep 17 00:00:00 2001 From: Builddddder <147791546+Builddddder@users.noreply.github.com> Date: Tue, 24 Oct 2023 02:29:09 +0800 Subject: [PATCH 1/3] feat: As a user, I want to have an example ERC-1271 module (#314) --- .../src/examples/modules/ERC1271Module.sol | 76 +++++++++++++++++++ contracts/test/example/ERC1271Module.t.sol | 76 +++++++++++++++++++ 2 files changed, 152 insertions(+) create mode 100644 contracts/src/examples/modules/ERC1271Module.sol create mode 100644 contracts/test/example/ERC1271Module.t.sol diff --git a/contracts/src/examples/modules/ERC1271Module.sol b/contracts/src/examples/modules/ERC1271Module.sol new file mode 100644 index 00000000..185303c1 --- /dev/null +++ b/contracts/src/examples/modules/ERC1271Module.sol @@ -0,0 +1,76 @@ +// SPDX-License-Identifier: MIT +pragma solidity 0.8.21; + +import { AbstractModule } from "../../interface/AbstractModule.sol"; +import { AttestationPayload } from "../../types/Structs.sol"; + +contract ERC1271Module is AbstractModule { + address public owner; + + error InvalidSignature(); + error WrongSender(); + + constructor(address _owner) { + owner = _owner; + } + + /** + * @notice Recover the signer of hash, assuming it's an EOA account + * @dev Only for EthSign signatures + * @param _hash Hash of message that was signed + * @param _signature Signature encoded as (bytes32 r, bytes32 s, uint8 v) + */ + function recoverSigner(bytes32 _hash, bytes memory _signature) internal pure returns (address signer) { + require(_signature.length == 65, "Invalid signature length"); + bytes32 r; + bytes32 s; + uint8 v; + assembly { + r := mload(add(_signature, 0x20)) + s := mload(add(_signature, 0x40)) + v := byte(0, mload(add(_signature, 0x60))) + } + if (uint256(s) > 0x7FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF5D576E7357A4501DDFE92F46681B20A0) { + revert("SignatureValidator#recoverSigner: invalid signature 's' value"); + } + if (v != 27 && v != 28) { + revert("SignatureValidator#recoverSigner: invalid signature 'v' value"); + } + + signer = ecrecover(_hash, v, r, s); + // Prevent signer from being 0x0 + require(signer != address(0x0), "SignatureValidator#recoverSigner: INVALID_SIGNER"); + return signer; + } + + /** + * @notice Verifies that the signer is the owner of the signing contract. + */ + function isValidSignature(bytes32 _hash, bytes memory _signature) internal view returns (bytes4) { + // Validate signatures + address signer = recoverSigner(_hash, _signature); + if (signer == owner) { + return 0x1626ba7e; + } else { + return 0xffffffff; + } + } + + function run( + AttestationPayload memory attestationPayload, + bytes memory validationPayload, + address txSender, + uint256 /*value*/ + ) public view override { + if (txSender != owner) { + revert WrongSender(); + } + address signee = abi.decode(attestationPayload.subject, (address)); + uint256 nonce = abi.decode(attestationPayload.attestationData, (uint256)); + bytes memory message = abi.encodePacked(signee, nonce); + bytes32 digest = keccak256(abi.encodePacked("\x19Ethereum Signed Message:\n", message)); + if (isValidSignature(digest, validationPayload) != 0x1626ba7e) { + revert InvalidSignature(); + } + } +} diff --git a/contracts/test/example/ERC1271Module.t.sol b/contracts/test/example/ERC1271Module.t.sol new file mode 100644 index 00000000..60161c9b --- /dev/null +++ b/contracts/test/example/ERC1271Module.t.sol @@ -0,0 +1,76 @@ +// SPDX-License-Identifier: MIT +pragma solidity 0.8.21; + +import { Test } from "forge-std/Test.sol"; +import { AbstractModule } from "../../src/interface/AbstractModule.sol"; +import { ERC1271Module } from "../../src/examples/modules/ERC1271Module.sol"; +import { AttestationPayload } from "../../src/types/Structs.sol"; + +contract ERC1271ModuleTest is Test { + ERC1271Module private erc1271Module; + uint256 internal signerPrivateKey; + address internal signerAddress; + + event ModuleRegistered(string name, string description, address moduleAddress); + + function setUp() public { + (signerAddress, signerPrivateKey) = makeAddrAndKey("veraxUser"); + erc1271Module = new ERC1271Module(signerAddress); + vm.deal(signerAddress, 1 ether); + } + + function test_ERC1271Module_verifySignature() public { + address user = makeAddr("user"); + uint256 nonce = 1234567; + AttestationPayload memory attestationPayload = AttestationPayload( + bytes32(uint256(1234)), + 0, + abi.encode(user), + abi.encode(nonce) + ); + + bytes32 hash = keccak256(abi.encodePacked("\x19Ethereum Signed Message:\n", abi.encodePacked(user, nonce))); + (uint8 v, bytes32 r, bytes32 s) = vm.sign(signerPrivateKey, hash); + + bytes memory signature = abi.encodePacked(r, s, v); + + erc1271Module.run(attestationPayload, signature, signerAddress, 0); + } + + function test_ERC1271Module_InvalidSignature() public { + address user = makeAddr("user"); + uint256 nonce = 1234567; + AttestationPayload memory attestationPayload = AttestationPayload( + bytes32(uint256(1234)), + 0, + abi.encode(user), + abi.encode(nonce) + ); + + bytes32 hash = keccak256(abi.encodePacked("\x19Not a correct SignMessage:\n", abi.encodePacked(user, nonce))); + (uint8 v, bytes32 r, bytes32 s) = vm.sign(signerPrivateKey, hash); + + bytes memory signature = abi.encodePacked(r, s, v); + + vm.expectRevert(ERC1271Module.InvalidSignature.selector); + erc1271Module.run(attestationPayload, signature, signerAddress, 0); + } + + function test_ERC1271Module_WrongSender() public { + address user = makeAddr("NotASigner"); + AttestationPayload memory attestationPayload = AttestationPayload( + bytes32(uint256(1234)), + 0, + bytes("subject"), + new bytes(1) + ); + + vm.expectRevert(ERC1271Module.WrongSender.selector); + erc1271Module.run(attestationPayload, bytes("0"), user, 0); + } + + function test_EcRecoverModule_supportsInterface() public { + bool isAbstractModuleSupported = erc1271Module.supportsInterface(type(AbstractModule).interfaceId); + assertEq(isAbstractModuleSupported, true); + } +} From b69c80becf96c42dede6f023b12a88871d106ad3 Mon Sep 17 00:00:00 2001 From: Alain Nicolas Date: Mon, 23 Oct 2023 22:22:31 +0200 Subject: [PATCH 2/3] chore: Add unit test to the subgraph (#269) --- .github/workflows/subgraph.yml | 64 ++++++++ .gitignore | 3 + pnpm-lock.yaml | 48 ++++-- subgraph/package.json | 11 +- subgraph/src/module-registry.ts | 12 +- subgraph/src/portal-registry.ts | 6 +- subgraph/src/schema-registry.ts | 14 +- subgraph/tests/attestation-registry.test.ts | 154 ++++++++++++++++++++ subgraph/tests/module-registry.test.ts | 72 +++++++++ subgraph/tests/portal-registry.test.ts | 104 +++++++++++++ subgraph/tests/schema-registry.test.ts | 94 ++++++++++++ subgraph/tsconfig.json | 3 +- 12 files changed, 548 insertions(+), 37 deletions(-) create mode 100644 .github/workflows/subgraph.yml create mode 100644 subgraph/tests/attestation-registry.test.ts create mode 100644 subgraph/tests/module-registry.test.ts create mode 100644 subgraph/tests/portal-registry.test.ts create mode 100644 subgraph/tests/schema-registry.test.ts diff --git a/.github/workflows/subgraph.yml b/.github/workflows/subgraph.yml new file mode 100644 index 00000000..3ea20e8c --- /dev/null +++ b/.github/workflows/subgraph.yml @@ -0,0 +1,64 @@ +name: Subgraph + +on: + pull_request: + branches: + - main + - dev + - release/* + push: + branches: + - main + - dev + - release/* + +jobs: + test: + runs-on: ubuntu-latest + + defaults: + run: + working-directory: subgraph + + steps: + - name: Check out the repo + uses: actions/checkout@v3 + + - name: Install Pnpm + uses: pnpm/action-setup@v2 + with: + version: 8 + run_install: false + + - name: Install Node.js + uses: actions/setup-node@v3 + with: + node-version: 18 + cache: pnpm + + - name: Get pnpm store directory + shell: bash + run: | + echo "STORE_PATH=$(pnpm store path --silent)" >> $GITHUB_ENV + + - uses: actions/cache@v3 + name: Setup pnpm cache + with: + path: ${{ env.STORE_PATH }} + key: ${{ runner.os }}-pnpm-store-${{ hashFiles('**/pnpm-lock.yaml') }} + restore-keys: | + ${{ runner.os }}-pnpm-store- + + - name: Install dependencies + run: pnpm install --frozen-lockfile + + - name: Build the subgraph + run: pnpm run build:goerli + + - name: Run the unit tests + run: pnpm run test + + - name: Add test summary + run: | + echo "## Unit tests result" >> $GITHUB_STEP_SUMMARY + echo "✅ Passed" >> $GITHUB_STEP_SUMMARY diff --git a/.gitignore b/.gitignore index 157884c0..89c5cbdd 100644 --- a/.gitignore +++ b/.gitignore @@ -25,6 +25,9 @@ typechain-types build generated subgraph.yaml +.bin +.latest.json +.docker # Misc .DS_Store diff --git a/pnpm-lock.yaml b/pnpm-lock.yaml index 79428e9c..43d5701a 100644 --- a/pnpm-lock.yaml +++ b/pnpm-lock.yaml @@ -1,9 +1,5 @@ lockfileVersion: '6.0' -settings: - autoInstallPeers: true - excludeLinksFromLockfile: false - overrides: flat@<5.0.1: '>=5.0.1' ejs@<3.1.7: '>=3.1.7' @@ -140,11 +136,17 @@ importers: subgraph: devDependencies: '@graphprotocol/graph-cli': - specifier: 0.58.0 - version: 0.58.0(@types/node@20.8.0)(node-fetch@3.3.2)(typescript@5.2.2) + specifier: 0.60.0 + version: 0.60.0(@types/node@20.8.0)(node-fetch@3.3.2)(typescript@5.2.2) '@graphprotocol/graph-ts': specifier: 0.31.0 version: 0.31.0 + assemblyscript: + specifier: 0.19.10 + version: 0.19.10 + matchstick-as: + specifier: 0.6.0 + version: 0.6.0 packages: @@ -2256,8 +2258,8 @@ packages: tslib: 2.6.2 dev: false - /@graphprotocol/graph-cli@0.58.0(@types/node@20.8.0)(node-fetch@3.3.2)(typescript@5.2.2): - resolution: {integrity: sha512-EbdL5LZFmIMAuItQXv7LXgd7cqYQ3BdIJR2jxNr+LRL0juBAxmEz6zVvYnIUmgXoa5SB5rxE9ZT6pfe+fhbD6Q==} + /@graphprotocol/graph-cli@0.60.0(@types/node@20.8.0)(node-fetch@3.3.2)(typescript@5.2.2): + resolution: {integrity: sha512-8tGaQJ0EzAPtkDXCAijFGoVdJXM+pKFlGxjiU31TdG5bS4cIUoSB6yWojVsFFod0yETAwf+giel/0/8sudYsDw==} engines: {node: '>=14'} hasBin: true dependencies: @@ -5211,7 +5213,7 @@ packages: /@types/cli-progress@3.11.2: resolution: {integrity: sha512-Yt/8rEJalfa9ve2SbfQnwFHrc9QF52JIZYHW3FDaTMpkCvnns26ueKiPHDxyJ0CS//IqjMINTx7R5Xa7k7uFHQ==} dependencies: - '@types/node': 20.5.7 + '@types/node': 20.8.0 dev: true /@types/concat-stream@1.6.1: @@ -5223,7 +5225,7 @@ packages: /@types/connect@3.4.36: resolution: {integrity: sha512-P63Zd/JUGq+PdrM1lv0Wv5SBYeA2+CORvbrXbngriYY0jzLUWfQMQQxOhjONEz/wlHOAxOdY7CY65rgQdTjq2w==} dependencies: - '@types/node': 20.5.7 + '@types/node': 20.8.0 dev: true /@types/form-data@0.0.33: @@ -5352,7 +5354,7 @@ packages: /@types/ws@7.4.7: resolution: {integrity: sha512-JQbbmxZTZehdc2iszGKs5oC3NFnjeay7mtAWrdt7qNtAVK0g19muApzAy4bm9byz79xa2ZnO/BOBC2R8RC5Lww==} dependencies: - '@types/node': 20.5.7 + '@types/node': 20.8.0 dev: true /@types/ws@8.5.6: @@ -8426,7 +8428,7 @@ packages: dependencies: inflight: 1.0.6 inherits: 2.0.4 - minimatch: 9.0.3 + minimatch: 3.1.2 once: 1.4.0 path-is-absolute: 1.0.1 dev: true @@ -8437,7 +8439,7 @@ packages: fs.realpath: 1.0.0 inflight: 1.0.6 inherits: 2.0.4 - minimatch: 9.0.3 + minimatch: 3.1.2 once: 1.4.0 path-is-absolute: 1.0.1 dev: true @@ -10622,6 +10624,12 @@ packages: resolution: {integrity: sha512-1RUZVgQlpJSPWYbFSpmudq5nHY1doEIv89gBtF0s4gW1GF2XorxcA/70M5vq7rLv0a6mhOUccRsqkwhwLCIQ2Q==} dev: true + /matchstick-as@0.6.0: + resolution: {integrity: sha512-E36fWsC1AbCkBFt05VsDDRoFvGSdcZg6oZJrtIe/YDBbuFh8SKbR5FcoqDhNWqSN+F7bN/iS2u8Md0SM+4pUpw==} + dependencies: + wabt: 1.0.24 + dev: true + /mcl-wasm@0.7.9: resolution: {integrity: sha512-iJIUcQWA88IJB/5L15GnJVnSQJmf/YaxxV6zRavv83HILHaJQb6y0iFyDMdDO0gN8X37tdxmAOrH/P8B6RB8sQ==} engines: {node: '>=8.9.0'} @@ -11062,6 +11070,7 @@ packages: engines: {node: '>=16 || 14 >=14.17'} dependencies: brace-expansion: 2.0.1 + dev: false /minimist@1.2.8: resolution: {integrity: sha512-2yyAR8qBkN3YuheJanUpWC5U3bb5osDywNB8RzDVlDwDHbocAJveqqj1u8+SVD7jkWT4yvsHCpWqqWqAxb0zCA==} @@ -11265,7 +11274,7 @@ packages: resolution: {integrity: sha512-tmPX422rYgofd4epzrNoOXiE8XFZYOcCq1vD7MAXCDO+O+zndlA2ztdKKMa+EeuBG5tHETpr4ml4RGgpqDCCAg==} engines: {node: '>= 0.10.5'} dependencies: - minimatch: 9.0.3 + minimatch: 3.1.2 dev: false /node-domexception@1.0.0: @@ -13139,7 +13148,7 @@ packages: dependencies: '@istanbuljs/schema': 0.1.3 glob: 7.2.3 - minimatch: 9.0.3 + minimatch: 3.1.2 dev: true /text-table@0.2.0: @@ -13758,6 +13767,11 @@ packages: resolution: {integrity: sha512-gQpnTgkubC6hQgdIcRdYGDSDc+SaujOdyesZQMv6JlfQee/9Mp0Qhnys6WxDWvQnL5WZdT7o2Ul187aSt0Rq+w==} dev: false + /wabt@1.0.24: + resolution: {integrity: sha512-8l7sIOd3i5GWfTWciPL0+ff/FK/deVK2Q6FN+MPz4vfUcD78i2M/49XJTwF6aml91uIiuXJEsLKWMB2cw/mtKg==} + hasBin: true + dev: true + /walker@1.0.8: resolution: {integrity: sha512-ts/8E8l5b7kY0vlWLewOkDXMmPdLcVV4GmOQLyxuSswIJsweeFZtAsMF7k1Nszz+TYBQrlYRmzOnr398y1JemQ==} dependencies: @@ -14149,3 +14163,7 @@ packages: /zod@3.22.4: resolution: {integrity: sha512-iC+8Io04lddc+mVqQ9AZ7OQ2MrUKGN+oIQyq1vemgt46jwCwLfhq7/pwnBnNXXXZb8VTVLKwp9EDkx+ryxIWmg==} dev: false + +settings: + autoInstallPeers: true + excludeLinksFromLockfile: false diff --git a/subgraph/package.json b/subgraph/package.json index 0b35af66..5ef1d646 100644 --- a/subgraph/package.json +++ b/subgraph/package.json @@ -24,10 +24,15 @@ "deploy": "source .env && cp subgraph.mainnet.yaml subgraph.yaml && pnpm run build && graph deploy --network linea-mainnet --node $DEPLOY_ENDPOINT_MAINNET --headers \"{\\\"Authorization\\\": \\\"Basic $IPFS_IDENTIFIERS\\\"}\" --ipfs $IPFS_ENDPOINT --version-label v0.0.1 Consensys/linea-attestation-registry", "deploy:goerli": "source .env && cp subgraph.goerli.yaml subgraph.yaml && pnpm run build:goerli && graph deploy --network linea-goerli --node $DEPLOY_ENDPOINT_GOERLI --headers \"{\\\"Authorization\\\": \\\"Basic $IPFS_IDENTIFIERS\\\"}\" --ipfs $IPFS_ENDPOINT --version-label v0.0.5 Consensys/linea-attestation-registry", "remove": "source .env && graph remove --node $DEPLOY_ENDPOINT_MAINNET Consensys/linea-attestation-registry", - "remove:goerli": "source .env && graph remove --node $DEPLOY_ENDPOINT_GOERLI Consensys/linea-attestation-registry" + "remove:goerli": "source .env && graph remove --node $DEPLOY_ENDPOINT_GOERLI Consensys/linea-attestation-registry", + "test": "graph test", + "test:coverage": "graph test -c", + "test:docker": "graph test -d" }, "devDependencies": { - "@graphprotocol/graph-cli": "0.58.0", - "@graphprotocol/graph-ts": "0.31.0" + "@graphprotocol/graph-cli": "0.60.0", + "@graphprotocol/graph-ts": "0.31.0", + "assemblyscript": "0.19.10", + "matchstick-as": "0.6.0" } } diff --git a/subgraph/src/module-registry.ts b/subgraph/src/module-registry.ts index 2b87823b..612dd9f6 100644 --- a/subgraph/src/module-registry.ts +++ b/subgraph/src/module-registry.ts @@ -1,16 +1,14 @@ -import { ModuleRegistered as ModuleRegisteredEvent, ModuleRegistry } from "../generated/ModuleRegistry/ModuleRegistry"; +import { ModuleRegistered as ModuleRegisteredEvent } from "../generated/ModuleRegistry/ModuleRegistry"; import { Counter, Module } from "../generated/schema"; export function handleModuleRegistered(event: ModuleRegisteredEvent): void { - const contract = ModuleRegistry.bind(event.address); - const moduleData = contract.modules(event.params.moduleAddress); - const module = new Module(event.params.moduleAddress.toHex()); + const module = new Module(event.params.moduleAddress.toHexString()); incrementModulesCount(); - module.moduleAddress = moduleData.getModuleAddress(); - module.name = moduleData.getName(); - module.description = moduleData.getDescription(); + module.moduleAddress = event.params.moduleAddress; + module.name = event.params.name; + module.description = event.params.description; module.save(); } diff --git a/subgraph/src/portal-registry.ts b/subgraph/src/portal-registry.ts index f8b3ae22..811700c3 100644 --- a/subgraph/src/portal-registry.ts +++ b/subgraph/src/portal-registry.ts @@ -5,15 +5,15 @@ import { Counter, Portal } from "../generated/schema"; export function handlePortalRegistered(event: PortalRegisteredEvent): void { const contract = PortalRegistry.bind(event.address); const portalData = contract.getPortalByAddress(event.params.portalAddress); - const portal = new Portal(event.params.portalAddress.toHex()); + const portal = new Portal(event.params.portalAddress.toHexString()); incrementPortalsCount(); + portal.name = event.params.name; + portal.description = event.params.description; portal.ownerAddress = portalData.ownerAddress; portal.modules = changetype(portalData.modules); portal.isRevocable = portalData.isRevocable; - portal.name = portalData.name; - portal.description = portalData.description; portal.ownerName = portalData.ownerName; portal.save(); diff --git a/subgraph/src/schema-registry.ts b/subgraph/src/schema-registry.ts index 816e5e6a..cc2069bd 100644 --- a/subgraph/src/schema-registry.ts +++ b/subgraph/src/schema-registry.ts @@ -1,17 +1,15 @@ -import { SchemaCreated as SchemaCreatedEvent, SchemaRegistry } from "../generated/SchemaRegistry/SchemaRegistry"; +import { SchemaCreated as SchemaCreatedEvent } from "../generated/SchemaRegistry/SchemaRegistry"; import { Counter, Schema } from "../generated/schema"; export function handleSchemaCreated(event: SchemaCreatedEvent): void { - const contract = SchemaRegistry.bind(event.address); - const schemaData = contract.getSchema(event.params.id); - const schema = new Schema(event.params.id.toHex()); + const schema = new Schema(event.params.id.toHexString()); incrementSchemasCount(); - schema.name = schemaData.name; - schema.description = schemaData.description; - schema.context = schemaData.context; - schema.schema = schemaData.schema; + schema.name = event.params.name; + schema.description = event.params.description; + schema.context = event.params.context; + schema.schema = event.params.schemaString; schema.save(); } diff --git a/subgraph/tests/attestation-registry.test.ts b/subgraph/tests/attestation-registry.test.ts new file mode 100644 index 00000000..43b65010 --- /dev/null +++ b/subgraph/tests/attestation-registry.test.ts @@ -0,0 +1,154 @@ +import { afterEach, assert, beforeAll, clearStore, createMockedFunction, describe, test } from "matchstick-as"; +import { Address, BigInt, Bytes, ethereum } from "@graphprotocol/graph-ts"; +import { + AttestationRegistered as AttestationRegisteredEvent, + AttestationRegistry, +} from "../generated/AttestationRegistry/AttestationRegistry"; +import { newTypedMockEvent } from "matchstick-as/assembly/defaults"; +import { handleAttestationRegistered } from "../src/attestation-registry"; +import { Schema } from "../generated/schema"; + +const attestationRegistryAddress = Address.fromString("C765F28096F6121C2F2b82D35A4346280164428b"); +const attestationId = Bytes.fromHexString("0x00000000000000000000000000000000000000000000000000000000000010f5"); +const schemaId = Bytes.fromHexString("0x7930a5ebfabdd4ef76bbb8cdcbc2225b6256d9511d9cf5ff0d6514c1bdb4d7dc"); +const replacedBy = Bytes.fromHexString("0x00000000000000000000000000000000000000000000000000000000000010f6"); +const attester = Address.fromString("e02bd7a6c8aa401189aebb5bad755c2610940a73"); +const portal = Address.fromString("f75be6f9418710fd516fa82afb3aad07e11a0f1b"); +const attestedDate = BigInt.fromString("1234"); +const expirationDate = BigInt.fromString("1234"); +const revocationDate = BigInt.fromString("0"); +const version = BigInt.fromString("4"); +const revoked = false; +const subject = Bytes.fromHexString("0xC765F28096F6121C2F2b82D35A4346280164428b"); +const attestationData = Bytes.fromHexString("0x0000000000000000000000000000000000000000000000000000000000000001"); + +describe("handleAttestationRegistered()", () => { + beforeAll(() => { + const tupleArray: Array = [ + ethereum.Value.fromFixedBytes(attestationId), + ethereum.Value.fromFixedBytes(schemaId), + ethereum.Value.fromFixedBytes(replacedBy), + ethereum.Value.fromAddress(attester), + ethereum.Value.fromAddress(portal), + ethereum.Value.fromUnsignedBigInt(attestedDate), + ethereum.Value.fromUnsignedBigInt(expirationDate), + ethereum.Value.fromUnsignedBigInt(revocationDate), + ethereum.Value.fromUnsignedBigInt(version), + ethereum.Value.fromBoolean(revoked), + ethereum.Value.fromBytes(subject), + ethereum.Value.fromBytes(attestationData), + ]; + + // Convert it to the Tuple type + const tuple = changetype(tupleArray); + + // Create a tuple Value + const tupleValue = ethereum.Value.fromTuple(tuple); + + createMockedFunction( + attestationRegistryAddress, + "getAttestation", + "getAttestation(bytes32):((bytes32,bytes32,bytes32,address,address,uint64,uint64,uint64,uint16,bool,bytes,bytes))", + ) + .withArgs([ethereum.Value.fromFixedBytes(attestationId)]) + .returns([tupleValue]); + }); + + afterEach(() => { + clearStore(); + }); + + test("Should mock the call to the AttestationRegistry", () => { + const contract = AttestationRegistry.bind(attestationRegistryAddress); + const result = contract.getAttestation(attestationId); + + assert.bytesEquals(result.attestationId, attestationId); + assert.bytesEquals(result.schemaId, schemaId); + assert.bytesEquals(result.replacedBy, replacedBy); + assert.addressEquals(result.attester, attester); + assert.addressEquals(result.portal, portal); + assert.bigIntEquals(result.attestedDate, attestedDate); + assert.bigIntEquals(result.expirationDate, expirationDate); + assert.bigIntEquals(result.revocationDate, revocationDate); + assert.i32Equals(result.version, version.toI32()); + assert.booleanEquals(result.revoked, revoked); + assert.bytesEquals(result.subject, subject); + assert.bytesEquals(result.attestationData, attestationData); + }); + + test("Should create a new Attestation entity", () => { + assert.entityCount("Attestation", 0); + + const attestationRegisteredEvent = createAttestationRegisteredEvent(attestationId); + + handleAttestationRegistered(attestationRegisteredEvent); + + assert.entityCount("Attestation", 1); + + assert.fieldEquals("Attestation", attestationId.toHexString(), "id", attestationId.toHexString()); + assert.fieldEquals("Attestation", attestationId.toHexString(), "schemaId", schemaId.toHexString()); + assert.fieldEquals("Attestation", attestationId.toHexString(), "replacedBy", replacedBy.toHexString()); + assert.fieldEquals("Attestation", attestationId.toHexString(), "attester", attester.toHexString()); + assert.fieldEquals("Attestation", attestationId.toHexString(), "portal", portal.toHexString()); + assert.fieldEquals("Attestation", attestationId.toHexString(), "attestedDate", attestedDate.toU64().toString()); + assert.fieldEquals("Attestation", attestationId.toHexString(), "expirationDate", expirationDate.toU64().toString()); + assert.fieldEquals("Attestation", attestationId.toHexString(), "revocationDate", revocationDate.toU64().toString()); + assert.fieldEquals("Attestation", attestationId.toHexString(), "version", version.toI32().toString()); + assert.fieldEquals("Attestation", attestationId.toHexString(), "revoked", revoked.toString()); + assert.fieldEquals("Attestation", attestationId.toHexString(), "subject", subject.toHexString()); + assert.fieldEquals("Attestation", attestationId.toHexString(), "attestationData", attestationData.toHexString()); + }); + + test("Should increment the attestations Counter", () => { + assert.entityCount("Attestation", 0); + assert.entityCount("Counter", 0); + + const attestationRegisteredEvent1 = createAttestationRegisteredEvent(attestationId); + + handleAttestationRegistered(attestationRegisteredEvent1); + + assert.entityCount("Attestation", 1); + assert.fieldEquals("Counter", "counter", "attestations", "1"); + }); + + test("Should decode a basic attestation data", () => { + const schema = new Schema(schemaId.toHexString()); + schema.name = "name"; + schema.description = "description"; + schema.context = "context"; + schema.schema = "bool isOk"; + schema.save(); + + const attestationRegisteredEvent = createAttestationRegisteredEvent(attestationId); + + handleAttestationRegistered(attestationRegisteredEvent); + + assert.entityCount("Attestation", 1); + + assert.fieldEquals("Attestation", attestationId.toHexString(), "id", attestationId.toHexString()); + assert.fieldEquals("Attestation", attestationId.toHexString(), "schemaId", schemaId.toHexString()); + assert.fieldEquals("Attestation", attestationId.toHexString(), "replacedBy", replacedBy.toHexString()); + assert.fieldEquals("Attestation", attestationId.toHexString(), "attester", attester.toHexString()); + assert.fieldEquals("Attestation", attestationId.toHexString(), "portal", portal.toHexString()); + assert.fieldEquals("Attestation", attestationId.toHexString(), "attestedDate", attestedDate.toU64().toString()); + assert.fieldEquals("Attestation", attestationId.toHexString(), "expirationDate", expirationDate.toU64().toString()); + assert.fieldEquals("Attestation", attestationId.toHexString(), "revocationDate", revocationDate.toU64().toString()); + assert.fieldEquals("Attestation", attestationId.toHexString(), "version", version.toI32().toString()); + assert.fieldEquals("Attestation", attestationId.toHexString(), "revoked", revoked.toString()); + assert.fieldEquals("Attestation", attestationId.toHexString(), "subject", subject.toHexString()); + assert.fieldEquals("Attestation", attestationId.toHexString(), "attestationData", attestationData.toHexString()); + assert.fieldEquals("Attestation", attestationId.toHexString(), "schemaString", "bool"); + assert.fieldEquals("Attestation", attestationId.toHexString(), "decodedData", "[true]"); + }); +}); + +function createAttestationRegisteredEvent(attestationId: Bytes): AttestationRegisteredEvent { + const attestationRegisteredEvent = newTypedMockEvent(); + attestationRegisteredEvent.address = attestationRegistryAddress; + + attestationRegisteredEvent.parameters.push( + new ethereum.EventParam("attestationId", ethereum.Value.fromBytes(attestationId)), + ); + + return attestationRegisteredEvent; +} diff --git a/subgraph/tests/module-registry.test.ts b/subgraph/tests/module-registry.test.ts new file mode 100644 index 00000000..3ca7796a --- /dev/null +++ b/subgraph/tests/module-registry.test.ts @@ -0,0 +1,72 @@ +import { afterEach, assert, clearStore, describe, test } from "matchstick-as"; +import { + ModuleRegistered as ModuleRegisteredEvent, + ModuleRegistered, +} from "../generated/ModuleRegistry/ModuleRegistry"; +import { Address, ethereum } from "@graphprotocol/graph-ts"; +import { newTypedMockEvent } from "matchstick-as/assembly/defaults"; +import { handleModuleRegistered } from "../src/module-registry"; + +describe("handleModuleRegistered()", () => { + const moduleAddress = "f75be6f9418710fd516fa82afb3aad07e11a0f1b"; + const moduleName = "module name"; + const moduleDescription = "module description"; + + afterEach(() => { + clearStore(); + }); + + test("Should create a new Module entity", () => { + assert.entityCount("Module", 0); + + const moduleRegisteredEvent = createModuleRegisteredEvent(moduleAddress, moduleName, moduleDescription); + + handleModuleRegistered(moduleRegisteredEvent); + + assert.entityCount("Module", 1); + + assert.fieldEquals("Module", "0x" + moduleAddress, "id", "0x" + moduleAddress); + assert.fieldEquals("Module", "0x" + moduleAddress, "name", moduleName); + assert.fieldEquals("Module", "0x" + moduleAddress, "description", moduleDescription); + assert.fieldEquals("Module", "0x" + moduleAddress, "moduleAddress", "0x" + moduleAddress); + }); + + test("Should increment the modules Counter", () => { + assert.entityCount("Module", 0); + + const moduleAddress1 = "f75be6f9418710fd516fa82afb3aad07e11a0f1b"; + const moduleAddress2 = "e75be6f9418710fd516fa82afb3aad07e11a0f1b"; + + const moduleRegisteredEvent1 = createModuleRegisteredEvent(moduleAddress1, moduleName, moduleDescription); + + handleModuleRegistered(moduleRegisteredEvent1); + + assert.entityCount("Module", 1); + assert.fieldEquals("Counter", "counter", "modules", "1"); + + const moduleRegisteredEvent2 = createModuleRegisteredEvent(moduleAddress2, moduleName, moduleDescription); + + handleModuleRegistered(moduleRegisteredEvent2); + + assert.entityCount("Module", 2); + assert.fieldEquals("Counter", "counter", "modules", "2"); + }); +}); + +function createModuleRegisteredEvent( + moduleAddress: string, + moduleName: string, + moduleDescription: string, +): ModuleRegistered { + const moduleRegisteredEvent = newTypedMockEvent(); + + moduleRegisteredEvent.parameters.push(new ethereum.EventParam("name", ethereum.Value.fromString(moduleName))); + moduleRegisteredEvent.parameters.push( + new ethereum.EventParam("description", ethereum.Value.fromString(moduleDescription)), + ); + moduleRegisteredEvent.parameters.push( + new ethereum.EventParam("moduleAddress", ethereum.Value.fromAddress(Address.fromString(moduleAddress))), + ); + + return moduleRegisteredEvent; +} diff --git a/subgraph/tests/portal-registry.test.ts b/subgraph/tests/portal-registry.test.ts new file mode 100644 index 00000000..1249e64a --- /dev/null +++ b/subgraph/tests/portal-registry.test.ts @@ -0,0 +1,104 @@ +import { afterEach, assert, beforeAll, clearStore, createMockedFunction, describe, test } from "matchstick-as"; +import { Address, ethereum } from "@graphprotocol/graph-ts"; +import { PortalRegistered as PortalRegisteredEvent, PortalRegistry } from "../generated/PortalRegistry/PortalRegistry"; +import { newTypedMockEvent } from "matchstick-as/assembly/defaults"; +import { handlePortalRegistered } from "../src/portal-registry"; + +const portalRegistryAddress = Address.fromString("506f88a5Ca8D5F001f2909b029738A40042e42a6"); +const portalAddress = Address.fromString("f75be6f9418710fd516fa82afb3aad07e11a0f1b"); +const ownerAddress = Address.fromString("e75be6f9418710fd516fa82afb3aad07e11a0f1b"); +const modules = [Address.zero()]; +const isRevocable = true; +const name = "portal name"; +const description = "portal description"; +const ownerName = "Verax"; + +describe("handlePortalRegistered()", () => { + beforeAll(() => { + const tupleArray: Array = [ + ethereum.Value.fromAddress(portalAddress), + ethereum.Value.fromAddress(ownerAddress), + ethereum.Value.fromAddressArray(modules), + ethereum.Value.fromBoolean(isRevocable), + ethereum.Value.fromString(name), + ethereum.Value.fromString(description), + ethereum.Value.fromString(ownerName), + ]; + + // Convert it to the Tuple type + const tuple = changetype(tupleArray); + + // Create a tuple Value + const tupleValue = ethereum.Value.fromTuple(tuple); + + createMockedFunction( + portalRegistryAddress, + "getPortalByAddress", + "getPortalByAddress(address):((address,address,address[],bool,string,string,string))", + ) + .withArgs([ethereum.Value.fromAddress(Address.fromString("f75be6f9418710fd516fa82afb3aad07e11a0f1b"))]) + .returns([tupleValue]); + }); + + afterEach(() => { + clearStore(); + }); + + test("Should mock the call to the PortalRegistry", () => { + const contract = PortalRegistry.bind(portalRegistryAddress); + const result = contract.getPortalByAddress(portalAddress); + + assert.addressEquals(result.id, portalAddress); + assert.addressEquals(result.ownerAddress, ownerAddress); + assert.addressEquals(result.modules.at(0), modules.at(0)); + assert.booleanEquals(result.isRevocable, isRevocable); + assert.stringEquals(result.name, name); + assert.stringEquals(result.description, description); + assert.stringEquals(result.ownerName, ownerName); + }); + + test("Should create a new Portal entity", () => { + assert.entityCount("Portal", 0); + + const portalRegisteredEvent = createPortalRegisteredEvent(portalAddress, name, description); + + handlePortalRegistered(portalRegisteredEvent); + + assert.entityCount("Portal", 1); + + assert.fieldEquals("Portal", portalAddress.toHexString(), "id", portalAddress.toHexString()); + assert.fieldEquals("Portal", portalAddress.toHexString(), "name", name); + assert.fieldEquals("Portal", portalAddress.toHexString(), "description", description); + }); + + test("Should increment the portals Counter", () => { + assert.entityCount("Portal", 0); + assert.entityCount("Counter", 0); + + const portalRegisteredEvent1 = createPortalRegisteredEvent(portalAddress, name, description); + + handlePortalRegistered(portalRegisteredEvent1); + + assert.entityCount("Portal", 1); + assert.fieldEquals("Counter", "counter", "portals", "1"); + }); +}); + +function createPortalRegisteredEvent( + portalAddress: Address, + portalName: string, + portalDescription: string, +): PortalRegisteredEvent { + const portalRegisteredEvent = newTypedMockEvent(); + portalRegisteredEvent.address = portalRegistryAddress; + + portalRegisteredEvent.parameters.push(new ethereum.EventParam("name", ethereum.Value.fromString(portalName))); + portalRegisteredEvent.parameters.push( + new ethereum.EventParam("description", ethereum.Value.fromString(portalDescription)), + ); + portalRegisteredEvent.parameters.push( + new ethereum.EventParam("portalAddress", ethereum.Value.fromAddress(portalAddress)), + ); + + return portalRegisteredEvent; +} diff --git a/subgraph/tests/schema-registry.test.ts b/subgraph/tests/schema-registry.test.ts new file mode 100644 index 00000000..7362c056 --- /dev/null +++ b/subgraph/tests/schema-registry.test.ts @@ -0,0 +1,94 @@ +import { afterEach, assert, clearStore, describe, test } from "matchstick-as"; +import { Bytes, ethereum } from "@graphprotocol/graph-ts"; +import { newTypedMockEvent } from "matchstick-as/assembly/defaults"; +import { handleSchemaCreated } from "../src/schema-registry"; +import { SchemaCreated, SchemaCreated as SchemaCreatedEvent } from "../generated/SchemaRegistry/SchemaRegistry"; + +describe("handleSchemaCreated()", () => { + const schemaId = "0x7930a5ebfabdd4ef76bbb8cdcbc2225b6256d9511d9cf5ff0d6514c1bdb4d7dc"; + const schemaName = "module name"; + const schemaDescription = "module description"; + const schemaContext = "schema context"; + const schemaString = "(bool isBuilder)"; + + afterEach(() => { + clearStore(); + }); + + test("Should create a new Schema entity", () => { + assert.entityCount("Schema", 0); + + const schemaCreatedEvent = createSchemaCreatedEvent( + schemaId, + schemaName, + schemaDescription, + schemaContext, + schemaString, + ); + + handleSchemaCreated(schemaCreatedEvent); + + assert.entityCount("Schema", 1); + + assert.fieldEquals("Schema", schemaId, "id", schemaId); + assert.fieldEquals("Schema", schemaId, "name", schemaName); + assert.fieldEquals("Schema", schemaId, "description", schemaDescription); + assert.fieldEquals("Schema", schemaId, "context", schemaContext); + assert.fieldEquals("Schema", schemaId, "schema", schemaString); + }); + + test("Should increment the schemas Counter", () => { + assert.entityCount("Schema", 0); + + const schemaId1 = "0x7930a5ebfabdd4ef76bbb8cdcbc2225b6256d9511d9cf5ff0d6514c1bdb4d7dc"; + const schemaId2 = "0x8930a5ebfabdd4ef76bbb8cdcbc2225b6256d9511d9cf5ff0d6514c1bdb4d7dc"; + + const schemaCreatedEvent1 = createSchemaCreatedEvent( + schemaId1, + schemaName, + schemaDescription, + schemaContext, + schemaString, + ); + + handleSchemaCreated(schemaCreatedEvent1); + + assert.entityCount("Schema", 1); + assert.fieldEquals("Counter", "counter", "schemas", "1"); + + const schemaCreatedEvent2 = createSchemaCreatedEvent( + schemaId2, + schemaName, + schemaDescription, + schemaContext, + schemaString, + ); + + handleSchemaCreated(schemaCreatedEvent2); + + assert.entityCount("Schema", 2); + assert.fieldEquals("Counter", "counter", "schemas", "2"); + }); +}); + +function createSchemaCreatedEvent( + schemaId: string, + schemaName: string, + schemaDescription: string, + schemaContext: string, + schemaString: string, +): SchemaCreated { + const schemaCreatedEvent = newTypedMockEvent(); + + schemaCreatedEvent.parameters.push( + new ethereum.EventParam("id", ethereum.Value.fromBytes(Bytes.fromHexString(schemaId))), + ); + schemaCreatedEvent.parameters.push(new ethereum.EventParam("name", ethereum.Value.fromString(schemaName))); + schemaCreatedEvent.parameters.push( + new ethereum.EventParam("description", ethereum.Value.fromString(schemaDescription)), + ); + schemaCreatedEvent.parameters.push(new ethereum.EventParam("context", ethereum.Value.fromString(schemaContext))); + schemaCreatedEvent.parameters.push(new ethereum.EventParam("schemaString", ethereum.Value.fromString(schemaString))); + + return schemaCreatedEvent; +} diff --git a/subgraph/tsconfig.json b/subgraph/tsconfig.json index a6d2ae82..4e866720 100644 --- a/subgraph/tsconfig.json +++ b/subgraph/tsconfig.json @@ -1,3 +1,4 @@ { - "extends": "@graphprotocol/graph-ts/types/tsconfig.base.json" + "extends": "@graphprotocol/graph-ts/types/tsconfig.base.json", + "include": ["src", "tests"] } From 484c24d63bca28406b87f82c28f4c2ba321a9aab Mon Sep 17 00:00:00 2001 From: Alain Nicolas Date: Tue, 24 Oct 2023 00:13:46 +0200 Subject: [PATCH 3/3] fix: Fix SDK tests following new version of the subgraph (#315) --- sdk/test/integration/Attestation.integration.test.ts | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/sdk/test/integration/Attestation.integration.test.ts b/sdk/test/integration/Attestation.integration.test.ts index 9be6f4c9..ad5a7a11 100644 --- a/sdk/test/integration/Attestation.integration.test.ts +++ b/sdk/test/integration/Attestation.integration.test.ts @@ -26,7 +26,7 @@ describe("AttestationDataMapper", () => { expect(result.revoked).toBeFalsy(); expect(result.subject).toEqual("0x000000000000000000000000cb859f99f84ab770a50380680be94ad9331bcec5"); expect(result.attestationData).toEqual("0x0000000000000000000000000000000000000000000000000000000000000004"); - expect(result.schemaString).toEqual("uint8 rating"); + expect(result.schemaString).toEqual("uint8"); expect(result.decodedData).toEqual(["0x4"]); }); });