Skip to content

Commit

Permalink
Verify Multiple P-256 Verifiers Are Supported (#307)
Browse files Browse the repository at this point in the history
Partially addresses #285

This PR adds new tests that verify multiple P-256 verifiers are
supported by our code. The `webauthn` shim code now moved to the
`passkey` package and is imported by the `4337` package, as this made
more sense to me. @mmv08 moved E2E tests for Passkey+4337 to the
`passkey` project, so once they both merge, the dependency can be
removed.

Note that for now, we use a `TestWebAuthnSignerFactor` contract just for
implementing the tests, but it should be switched to using the canonical
signing factory from #306 once merged.

The additional verifier that is being used is the one from
[Daimo-eth](https://github.com/daimo-eth/p256-verifier). The artifact
was vendored into the project, as it is a Foundry project, and this is
the easiest way to include the dependency without building it (giving us
identical bytecode to what is deployed on-chain).

Adding a test for using the precompile will be done in a separate PR.

---

Note that this PR contains some unrelated changes to the deployment and
E2E setup. This was a result of rebasing onto changes introduced in #306
to get things working as well as some nits leftover from the
aforementioned PR.
  • Loading branch information
nlordell authored Mar 7, 2024
1 parent a2914d2 commit 92e5bc0
Show file tree
Hide file tree
Showing 21 changed files with 325 additions and 116 deletions.
1 change: 1 addition & 0 deletions .eslintrc.js
Original file line number Diff line number Diff line change
@@ -1,4 +1,5 @@
module.exports = {
root: true,
env: {
browser: true,
es2021: true,
Expand Down
2 changes: 1 addition & 1 deletion .github/workflows/ci_passkey.yml
Original file line number Diff line number Diff line change
Expand Up @@ -46,4 +46,4 @@ jobs:
cache-dependency-path: package-lock.json
- run: |
npm ci
npm run test:e2e -w modules/passkey
npm run test:4337 -w modules/passkey
2 changes: 0 additions & 2 deletions modules/passkey/.env.sample

This file was deleted.

3 changes: 3 additions & 0 deletions modules/passkey/.eslintignore
Original file line number Diff line number Diff line change
@@ -0,0 +1,3 @@
coverage/
dist/
typechain-types/
1 change: 0 additions & 1 deletion modules/passkey/contracts/test/TestP256VerifierLib.sol
Original file line number Diff line number Diff line change
@@ -1,5 +1,4 @@
// SPDX-License-Identifier: LGPL-3.0-only
/* solhint-disable payable-fallback */
pragma solidity ^0.8.0;

import {IP256Verifier, P256VerifierLib} from "../verifiers/IP256Verifier.sol";
Expand Down
86 changes: 86 additions & 0 deletions modules/passkey/contracts/test/TestWebAuthnSignerFactory.sol
Original file line number Diff line number Diff line change
@@ -0,0 +1,86 @@
// SPDX-License-Identifier: LGPL-3.0-only
/* solhint-disable one-contract-per-file */
pragma solidity ^0.8.0;

import {IP256Verifier, P256VerifierLib} from "../verifiers/IP256Verifier.sol";

contract TestWebAuthnSignerFactory {
function createSigner(address verifier, uint256 x, uint256 y) external returns (TestWebAuthnSigner signer) {
signer = new TestWebAuthnSigner{salt: 0}(verifier, x, y);
}
}

contract TestWebAuthnSigner {
using P256VerifierLib for IP256Verifier;

struct SignatureData {
bytes authenticatorData;
bytes clientDataFields;
uint256 r;
uint256 s;
}

string private constant _BASE64URL = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789-_";

address private immutable _VERIFIER;
uint256 private immutable _X;
uint256 private immutable _Y;

constructor(address verifier, uint256 x, uint256 y) {
_VERIFIER = verifier;
_X = x;
_Y = y;
}

function isValidSignature(bytes32 hash, bytes calldata signature) external view returns (bytes4 magicValue) {
SignatureData calldata sig;
// solhint-disable-next-line no-inline-assembly
assembly ("memory-safe") {
sig := signature.offset
}

if (
IP256Verifier(_VERIFIER).verifySignatureAllowMalleability(
_signingMessage(hash, sig.authenticatorData, sig.clientDataFields),
sig.r,
sig.s,
_X,
_Y
)
) {
magicValue = this.isValidSignature.selector;
}
}

function _signingMessage(
bytes32 challenge,
bytes calldata authenticatorData,
bytes calldata clientDataFields
) internal pure returns (bytes32 message) {
/* solhint-disable quotes */
bytes memory clientDataJson = abi.encodePacked(
'{"type":"webauthn.get","challenge":"AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA",',
clientDataFields,
"}"
);
/* solhint-enable quotes */

string memory alphabet = _BASE64URL;
// solhint-disable-next-line no-inline-assembly
assembly ("memory-safe") {
let lut := add(alphabet, 1)
let ptr := add(clientDataJson, 68)

for {
let i := 0
} lt(i, 42) {
i := add(i, 1)
} {
mstore8(add(ptr, i), mload(add(lut, and(0x3f, shr(sub(250, mul(6, i)), challenge)))))
}
mstore8(add(ptr, 42), mload(add(lut, and(0x3f, shl(2, challenge)))))
}

message = sha256(abi.encodePacked(authenticatorData, sha256(clientDataJson)));
}
}
4 changes: 1 addition & 3 deletions modules/passkey/docker-compose.yaml
Original file line number Diff line number Diff line change
Expand Up @@ -18,8 +18,6 @@ services:
build:
context: .
dockerfile: docker/bundler/Dockerfile
args:
TAG: 26e4f4c433916a6a2ea8ecc91ba8a56cd904f27f
restart: always
command: ['--auto', '--network=http://geth:8545']
ports:
Expand All @@ -30,7 +28,7 @@ services:
context: .
dockerfile: docker/bundler/Dockerfile
args:
TAG: 26e4f4c433916a6a2ea8ecc91ba8a56cd904f27f
TAG: main
restart: always
command: ['--auto', '--network=http://geth:8545']
ports:
Expand Down
4 changes: 1 addition & 3 deletions modules/passkey/hardhat.config.ts
Original file line number Diff line number Diff line change
Expand Up @@ -3,7 +3,6 @@ import dotenv from 'dotenv'
import type { HardhatUserConfig } from 'hardhat/config'
import 'hardhat-deploy'

// Load environment variables.
dotenv.config()

const config: HardhatUserConfig = {
Expand All @@ -16,10 +15,9 @@ const config: HardhatUserConfig = {
networks: {
localhost: {
url: 'http://localhost:8545',
tags: ['dev', 'safe'],
tags: ['dev'],
},
hardhat: {
gasPrice: 10000000000,
tags: ['test'],
},
},
Expand Down
7 changes: 4 additions & 3 deletions modules/passkey/package.json
Original file line number Diff line number Diff line change
Expand Up @@ -35,15 +35,16 @@
"fmt:check": "prettier --check .",
"lint": "npm run lint:sol && npm run lint:ts",
"lint:sol": "solhint 'contracts/**/*.sol'",
"lint:ts": "eslint --fix ./src && eslint --fix ./test",
"lint:ts": "eslint .",
"test": "hardhat test",
"test:e2e": "./test/4337-e2e/run.sh"
"test:4337": "./test/4337/run.sh"
},
"devDependencies": {
"@account-abstraction/contracts": "^0.7.0",
"@noble/curves": "^1.3.0",
"@nomicfoundation/hardhat-toolbox": "^4.0.0",
"@account-abstraction/contracts": "^0.7.0",
"@safe-global/safe-erc4337": "^0.3.0",
"@simplewebauthn/server": "^9.0.3",
"cbor": "^9.0.2",
"dotenv": "^16.4.5",
"hardhat": "^2.21.0",
Expand Down
1 change: 0 additions & 1 deletion modules/passkey/src/constants.ts

This file was deleted.

16 changes: 5 additions & 11 deletions modules/passkey/src/deploy/launchpad.ts
Original file line number Diff line number Diff line change
@@ -1,25 +1,19 @@
import { DeployFunction } from 'hardhat-deploy/types'
import { LAUNCHPAD_DEPLOYMENT_ENTRY_POINT_ADDRESS } from '../constants'

const deploy: DeployFunction = async ({ deployments, getNamedAccounts, network }) => {
if (!network.tags.dev && !network.tags.test) {
return
}

const deploy: DeployFunction = async ({ deployments, getNamedAccounts }) => {
const { deployer } = await getNamedAccounts()
const { deploy } = deployments

const entryPoint = (await deployments.getOrNull('EntryPoint').then((d) => d?.address)) || LAUNCHPAD_DEPLOYMENT_ENTRY_POINT_ADDRESS
if (!entryPoint) {
throw new Error('Entry point contract should be deployed or set in LAUNCHPAD_DEPLOYMENT_ENTRY_POINT_ADDRESS')
}
const entryPoint = await deployments.get('EntryPoint')

await deploy('Safe256BitECSignerLaunchpad', {
from: deployer,
args: [entryPoint],
args: [entryPoint.address],
log: true,
deterministicDeployment: true,
})
}

deploy.dependencies = ['entrypoint']

export default deploy
13 changes: 5 additions & 8 deletions modules/passkey/src/deploy/safe.ts
Original file line number Diff line number Diff line change
Expand Up @@ -4,21 +4,17 @@ import SafeL2 from '@safe-global/safe-contracts/build/artifacts/contracts/SafeL2
import Safe4337Module from '@safe-global/safe-erc4337/build/artifacts/contracts/Safe4337Module.sol/Safe4337Module.json'
import SafeModuleSetup from '@safe-global/safe-erc4337/build/artifacts/contracts/SafeModuleSetup.sol/SafeModuleSetup.json'
import { DeployFunction } from 'hardhat-deploy/types'
import { LAUNCHPAD_DEPLOYMENT_ENTRY_POINT_ADDRESS } from '../constants'

const deploy: DeployFunction = async ({ deployments, getNamedAccounts, network }) => {
if (!network.tags.safe && !network.tags.test) {
if (!network.tags.dev && !network.tags.test) {
return
}

const entryPoint = (await deployments.getOrNull('EntryPoint').then((d) => d?.address)) || LAUNCHPAD_DEPLOYMENT_ENTRY_POINT_ADDRESS
if (!entryPoint) {
throw new Error('Entry point contract should be deployed or set in LAUNCHPAD_DEPLOYMENT_ENTRY_POINT_ADDRESS')
}

const { deployer } = await getNamedAccounts()
const { deploy } = deployments

const entryPoint = await deployments.get('EntryPoint')

await deploy('MultiSend', {
from: deployer,
contract: MultiSend,
Expand Down Expand Up @@ -50,12 +46,13 @@ const deploy: DeployFunction = async ({ deployments, getNamedAccounts, network }
await deploy('Safe4337Module', {
from: deployer,
contract: Safe4337Module,
args: [entryPoint],
args: [entryPoint.address],
log: true,
deterministicDeployment: true,
})
}

deploy.dependencies = ['entrypoint']
deploy.tags = ['safe']

export default deploy
10 changes: 10 additions & 0 deletions modules/passkey/src/deploy/verifiers.ts
Original file line number Diff line number Diff line change
@@ -1,9 +1,19 @@
import type { DeployFunction } from 'hardhat-deploy/types'

import DaimoP256Verifier from '../vendor/daimo-eth/P256Verifier.json'

const deploy: DeployFunction = async ({ deployments, getNamedAccounts }) => {
const { deployer } = await getNamedAccounts()
const { deploy } = deployments

await deploy('DaimoP256Verifier', {
from: deployer,
contract: DaimoP256Verifier,
args: [],
deterministicDeployment: true,
log: true,
})

const FCLP256Verifier = await deploy('FCLP256Verifier', {
from: deployer,
args: [],
Expand Down
Loading

0 comments on commit 92e5bc0

Please sign in to comment.