Skip to content

Latest commit

 

History

History
238 lines (159 loc) · 11.5 KB

README.md

File metadata and controls

238 lines (159 loc) · 11.5 KB

Crowded Fund - EVM implementation of Campaigns Escrow Contract

Provide a way to raise funds, for yourself as an organization, or on behalf of an organization, through donations, raising on behalf of organizations has an approval process to indicate whether the campaign is "official". Contract can also function as an escrow with minimum target amounts, refunding donors if the target is not met. Campaigns can be time-based.

An EVM adaptation of the NEAR Campaign contracts as part of Potlock Protocol https://github.com/PotLock/core/tree/feat/campaigns/contracts/campaigns For more on Potlock https://docs.potlock.io

🏗 Scaffold-ETH 2

⚙️ Built using NextJS, RainbowKit, Hardhat, Wagmi, Viem, and Typescript.
  • Contract Hot Reload: Your frontend auto-adapts to your smart contract as you edit it.
  • 🪝 Custom hooks: Collection of React hooks wrapper around wagmi to simplify interactions with smart contracts with typescript autocompletion.
  • 🧱 Components: Collection of common web3 components to quickly build your frontend.
  • 🔥 Burner Wallet & Local Faucet: Quickly test your application with a burner wallet and local faucet.
  • 🔐 Integration with Wallet Providers: Connect to different wallet providers and interact with the Ethereum network.

Debug Contracts tab

Requirements

Before you begin, you need to install the following tools:

Quickstart

To get started with Scaffold-ETH 2, follow the steps below:

  1. Clone this repo & install dependencies
git clone https://github.com/scaffold-eth/scaffold-eth-2.git
cd scaffold-eth-2
yarn install
  1. Run a local network in the first terminal:
yarn chain

This command starts a local Ethereum network using Hardhat. The network runs on your local machine and can be used for testing and development. You can customize the network configuration in hardhat.config.ts.

  1. On a second terminal, deploy the test contract:
yarn deploy

This command deploys a test smart contract to the local network. The contract is located in packages/hardhat/contracts and can be modified to suit your needs. The yarn deploy command uses the deploy script located in packages/hardhat/deploy to deploy the contract to the network. You can also customize the deploy script.

  1. On a third terminal, start your NextJS app:
yarn start

Visit your app on: http://localhost:3000. You can interact with your smart contract using the Debug Contracts page. You can tweak the app config in packages/nextjs/scaffold.config.ts.

What's next:

  • Edit your smart contract Campaigns.sol in packages/hardhat/contracts
  • Edit your frontend homepage at packages/nextjs/app/page.tsx. For guidance on routing and configuring pages/layouts checkout the Next.js documentation.
  • Edit your deployment scripts in packages/hardhat/deploy
  • Edit your smart contract test in: packages/hardhat/test. To run test use yarn hardhat:test
  • You can add your Alchemy API Key in scaffold.config.ts if you want more reliability in your RPC requests.

Documentation

Visit our docs to learn how to start building with Scaffold-ETH 2.

To know more about its features, check out our website.

Contributing to Scaffold-ETH 2

We welcome contributions to Scaffold-ETH 2!

Please see CONTRIBUTING.MD for more information and guidelines for contributing to Scaffold-ETH 2.

Solidity Campaigns Contract Overview

Overview

This smart contract implements a decentralized fundraising platform on the Ethereum blockchain. It allows users to create and manage fundraising campaigns, accept donations in ETH or ERC20 tokens, and handle the distribution of funds to recipients and fee collectors. The contract is inspired by the PotLock campaigns contract.

Features

  • Create and manage fundraising campaigns
  • Accept donations in ETH or ERC20 tokens
  • Implement referral and creator fee systems
  • Escrow functionality for campaigns with minimum funding goals
  • Refund mechanism for failed campaigns
  • Flexible fee structure with configurable protocol, referral, and creator fees

Contract Structure

The main contract Campaigns.sol contains the following key components:

  1. Campaign and Donation structs
  2. State variables for managing campaign and donation IDs, fees, and configuration
  3. Mappings for storing campaign and donation data
  4. Functions for creating and updating campaigns
  5. Donation processing and refund functions
  6. Getter functions for retrieving campaign and donation information
  7. Admin functions for updating fee structures and withdrawing stuck funds

Key Functions

Campaign Management

  • createCampaign: Create a new fundraising campaign
  • updateCampaign: Update an existing campaign (only by owner, before start)

Donation Handling

  • donate: Make a donation to a specific campaign
  • processEscrowedDonations: Process escrowed donations after reaching the minimum goal
  • refundDonations: Refund donations for failed campaigns

Information Retrieval

  • getCampaign: Get campaign details
  • getDonation: Get donation details
  • getCampaignsByOwner: Get campaigns created by a specific owner
  • getCampaignsByRecipient: Get campaigns for a specific recipient
  • getDonationsByCampaign: Get donations for a specific campaign
  • getDonationsByDonor: Get donations made by a specific donor

Admin Functions

  • Update fee structures (protocol, referral, creator)
  • Withdraw stuck ETH or ERC20 tokens (emergency functions)

Security Considerations

  • Uses OpenZeppelin's Ownable and ReentrancyGuard to enhance security
  • Implements input validation to ensure data integrity
  • Includes emergency functions to withdraw stuck funds, accessible only by the owner

Gas Optimization

  • Uses uint256 for most numeric values
  • Employs structs to group related data
  • Emits events for important state changes

Differences from PotLock Implementation

  • This Ethereum implementation uses Solidity instead of Rust
  • Adapts to Ethereum's transaction model and global state storage
  • Utilizes OpenZeppelin libraries for standard functionalities

Future Improvements

  • Implement a multi-signature wallet for the protocol fee recipient and contract ownership
  • Add support for batch operations to optimize gas usage
  • Implement a time-lock mechanism for critical parameter changes
  • Develop a factory contract for easier deployment and management of multiple campaign contracts

Recent Updates

  1. Claim Functionality: Implemented a pull-based refund system with claimRefund and claimAllRefunds functions, allowing donors to claim refunds for failed campaigns.

  2. Official Campaign Status: Added an isOfficial boolean to the Campaign struct, which is set to true if the campaign creator is also the recipient. This can be updated using the setOfficialStatus function.

  3. Commit-Reveal Scheme: Implemented a two-step process for creating campaigns using a commit-reveal scheme to prevent front-running.

  4. Gas Optimization: Improved gas efficiency by using uint256 for most numeric values and employing structs to group related data.

Key Functions

  • createCampaignCommitment: Create a new campaign commitment
  • revealCampaign: Reveal the campaign details after commitment
  • updateCampaign: Update an existing campaign (only by owner, before start)
  • donate: Make a donation to a specific campaign
  • processEscrowedDonations: Process escrowed donations after reaching the minimum goal
  • claimRefund: Claim a refund for a specific donation
  • claimAllRefunds: Claim all refunds for a campaign
  • setOfficialStatus: Set the official status of a campaign (only by recipient)

Testing Framework

The contract is tested using the Hardhat testing framework with Chai assertions. The test file Campaigns.ts contains various test suites that cover different aspects of the contract functionality.

Key Test Suites

  1. Deployment: Tests the initial state of the contract after deployment.
  2. Campaign Creation and Management: Tests creating, revealing, and updating campaigns.
  3. Donations: Tests making donations, processing escrowed donations, and handling campaign end conditions.
  4. Refunds: Tests the refund claiming process for failed campaigns.
  5. Admin Functions: Tests owner-only functions for updating fee structures and withdrawing stuck funds.
  6. Edge Cases and Attack Vectors: Tests various edge cases and potential attack scenarios.
  7. Gas Limit and Large Scale Operations: Tests handling of large numbers of donations and refunds.
  8. Official Campaign Status: Tests setting and checking the official status of campaigns.

How the Testing Framework Works

  1. The tests use the ethers library to interact with the smart contract.
  2. Before each test suite, the contract is deployed to a local Hardhat network.
  3. Test cases use async/await syntax to handle asynchronous contract interactions.
  4. Chai assertions (expect) are used to verify the expected outcomes of contract function calls.
  5. The ethers.provider is used to manipulate the blockchain state (e.g., increasing time) for testing time-dependent functionality.
  6. Events emitted by the contract are captured and verified in relevant test cases.

To run the tests, use the command yarn hardhat test in the project directory.

For more detailed information on the contract implementation and tests, please refer to the Campaigns.sol and Campaigns.ts files in the repository.

Campigns and Referrals

Referrals are utilized for campaign creators, Potlock protocol, and referrers. These amounts are also escrowed until the campaign is successful.

Referral fees are calculated at the time of donation . When a user makes a donation, the contract calculates the referral fee based on the referralFeeBasisPoints set for the campaign. For campaigns with a minimum amount (escrow-based): If the campaign's minimum amount has not been reached, all funds, including the referral fee, are held in escrow1 . The escrowBalance of the campaign is increased by the full donation amount. The referral fee is not immediately paid out. Instead, it's stored as part of the Donation struct1 . If the campaign reaches its minimum amount, the processEscrowedDonations function can be called1 . This function will transfer the escrowed funds, including paying out the referral fees for all donations. If the campaign ends without reaching its minimum amount, donors can claim refunds using the claimRefund or claimAllRefunds functions1 . In this case, the full donation amount is refunded to the donor, including the portion that would have been the referral fee. For campaigns without a minimum amount or when the minimum has already been reached: The referral fee is immediately transferred to the referrer at the time of donation