Skip to content

ECDSA Ownership Module

What is it? 🤔

The ECDSA Ownership Registry Module or ECDSA Validation Module is integral to Biconomy's Modular Smart Account, enhancing transaction security and user authentication. This document outlines its functionality, benefits, and use cases.

What is the ECDSA Ownership Module? 🤔

This module allows Externally Owned Accounts (EOAs) to authorize and sign user operations (UserOps) for Smart Account. It operates similarly to non-modular Smart Account ownership but is reconstructed as a Validation Module within the Account Abstraction + Modular Smart Accounts ecosystem.

Key Functions 🔐

  • Single Signer Simplicity: Offers 1/1 multisig, single-signature control, ideal for both Web2 users and crypto-native users who already possess an EOA wallet.
  • ECDSA Signature Scheme: Utilizes the ECDSA secp256k1 curve for secure signing.
  • Flexible Owner Authentication: Supports various signer solutions like Privy, Fireblocks, Arcana Auth, Web3Auth, Magic, Capsule, Turnkey or Particle.
  • EIP-1271 Compliance: Allowing Smart Accounts to sign Ethereum messages for logging into dApps.

Use Cases 🌟

  1. Secure Transaction Signing: EOAs can securely authorize transactions for Smart Accounts.
  2. dApp Interaction: Simplifies the process of logging into dApps using Ethereum messages thanks to EIP-1271 support.
  3. Ownership Management: Mirrors traditional ownership systems, providing a familiar framework within a more advanced blockchain setting.

SDK Guide 📘

Initializing ECDSA Module 🌟

There is no need to do anything in order to use ECDSAOwnershipValidationModule, this module is set by default on every smart account.

Usage
import {
  createECDSAOwnershipValidationModule,
  createSmartAccountClient,
} from "@biconomy/accounts";
 
const ecdsaModuleConfig = { signer };
 
const defaultValidationModule = await createECDSAOwnershipValidationModule(
  ecdsaModuleConfig
);
 
// Signer for smart account taken from the module
const smartAccount = await createSmartAccountClient({
  bundlerUrl,
  defaultValidationModule, // Alternatively this can be ommitted, as it is the default validation module used when accounts are created
});
 
const saModuleAddress = await smartAccount.activeValidationModule.getAddress();
console.log(saModuleAddress === defaultValidationModule.moduleAddress); // true
Parameters

required params are explicitly mentioned

  • signer(Signer, required): Pass the signer instance into the validation module which gets used to detect owner of the smart account.
  • entryPointAddress(Hex): Defaults to "0x5ff137d4b0fdcd49dca30c7cf57e578a026d2789"
  • moduleAddress(Hex): Defaults to "0x0000001c5b32F37F5beA87BDD5374eB2aC54eA8e"
  • version(ModuleVersion): Defaults to "V1_0_0"
Returns
  • ecdsaOwnershipValidationModule (ECDSAOwnershipValidationModule): An instance of the Biconomy ECDSA ownership validation module.

Smart Contract Deep Dive 🛠️

This section dives into the EcdsaOwnershipRegistryModule for Biconomy Smart Accounts, focusing on key functionalities and security aspects.

Core Functionalities

User Operation Validation (validateUserOp) 🛡️

// Validates user operations signed by an EOA
    function validateUserOp(
        UserOperation calldata userOp,
        bytes32 userOpHash
    ) external view virtual override returns (uint256) {
        if (
            _verifySignature(
                userOpHash,
                userOp.signature[96:161],
                userOp.sender
            )
        ) {
            return VALIDATION_SUCCESS;
        }
        return SIG_VALIDATION_FAILED;
    }

Signature Verification (_verifySignature) 🔐

// Internal function to verify the signature of a smart account
function _verifySignature(bytes32 dataHash, bytes memory signature, address smartAccount)
    internal view returns (bool) {
    address expectedSigner = _smartAccountOwners[smartAccount];
    // Reverts if no owner is registered
    if (expectedSigner == address(0)) {
        revert NoOwnerRegisteredForSmartAccount(smartAccount);
    }
    // Checks for signature length and recovers the signer
    if (signature.length < 65) revert WrongSignatureLength();
    address recovered = (dataHash.toEthSignedMessageHash()).recover(signature);
    if (expectedSigner == recovered) {
        return true;
    }
    recovered = dataHash.recover(signature);
    return expectedSigner == recovered;
}
 
Validates a signature against a data hash and registered owner, supporting EIP-1271 standard.
function isValidSignature(
    bytes32 dataHash,
    bytes memory signature
) public view returns (bytes4);

Proper handling and verification of signatures are crucial for maintaining the integrity of the Smart Account. The isValidSignature function is essential for EIP-1271 support, validating that the signature matches the expected signer as per the EIP-1271 standard.

Security Considerations

  • Strict Ownership Rules: Only Externally Owned Accounts (EOAs) can authorize transactions, ensuring secure control over Smart Account operations.
  • Signature Verification: Implements robust methods for signature validation, crucial for preventing unauthorized access.

Interaction with Smart Accounts 🤝

The EcdsaOwnershipRegistryModule interacts with Smart Accounts primarily through its core functionalities:

  • Initialization and Ownership: During deployment, Smart Accounts use init method to invoke initForSmartAccount on modules, setting initial ownership and configurations.
  • User Operation Validation: When a Smart Account attempts to perform an operation, validateUserOp is invoked to ensure the action is authorized by the registered owner.
  • Signature Verification: The module uses _verifySignature to verify any signatures associated with transactions initiated by the Smart Account.