Skip to content
New issue

Have a question about this project? # for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “#”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? # to your account

feat: add section for claiming staking rewards in v2, tested in arb sepolia #1913

Draft
wants to merge 1 commit into
base: dev
Choose a base branch
from

Conversation

kemuru
Copy link
Contributor

@kemuru kemuru commented Mar 10, 2025

first iteration to claim. it claimed correctly. i tested deploying a contract to arb sepolia, sending airdrop to staked jurors in devnet, and claiming with my own wallet.

TODO: implement the design from Plinio

and this PR cannot be merged until we actually release the Staking Rewards for V2

PR-Codex overview

This PR introduces the StakingRewardsClaimModal component to the Profile page, allowing users to claim staking rewards. It fetches claims from IPFS and enables users to execute a claim transaction on the blockchain.

Detailed summary

  • Added import for StakingRewardsClaimModal in index.tsx.
  • Introduced StakingRewardsClaimModal component in StakingRewardsClaimModal.tsx.
  • Implemented fetching of user claims from IPFS.
  • Created a function to handle claiming tokens.
  • Displayed total claimable tokens and a button to claim them.
  • Managed loading and claimed states for user feedback.

✨ Ask PR-Codex anything about this PR by commenting with /codex {your question}

Summary by CodeRabbit

  • New Features
    • Introduced a new interactive rewards claim modal that allows users to view their total claimable tokens.
    • Integrated the modal into the profile page, providing real-time status updates, a loading state during processing, and confirmation once rewards have been successfully claimed.

Copy link
Contributor

coderabbitai bot commented Mar 10, 2025

Walkthrough

This pull request introduces a new React component, ClaimModal, in the staking rewards claim modal file. The component uses wagmi hooks to manage user account information and contract interactions. It fetches claim data from IPFS based on chain-specific snapshots and processes information such as month, balance, and Merkle proof. Upon user interaction, it initiates a claim transaction using writeContractAsync, with error handling and state updates for UI feedback. Additionally, the new component is integrated within the Profile page to display when a user is connected or when a search parameter is provided.

Changes

File(s) Change Summary
web/src/pages/Profile/Staking...ClaimModal.tsx Added new ClaimModal component which incorporates wagmi hooks for account management, fetches claims from IPFS based on chain snapshots, processes claim data, and executes claim transactions via writeContractAsync with proper error handling and state updates.
web/src/pages/Profile/index.tsx Integrated the StakingRewardsClaimModal component into the Profile page, rendering it unconditionally when the user is connected or a search address is provided.

Sequence Diagram(s)

sequenceDiagram
    participant U as User
    participant M as ClaimModal
    participant IP as IPFS
    participant C as Contract

    U->>+M: Open Claim Modal
    M->>+IP: Fetch claim data based on snapshots
    IP-->>-M: Return claim details (month, balance, Merkle proof)
    M->>+C: Invoke writeContractAsync to process claim
    C-->>-M: Confirm transaction / return error
    M-->>U: Update UI with claim success or error message
Loading

Poem

In the realm of code so spry,
I hop along with a twinkling eye.
A modal blossoms, fresh and new,
Claims and contracts in a joyful brew.
Tokens claimed with unicorn might,
This rabbit cheers the change out bright! 🐰✨

✨ Finishing Touches
  • 📝 Generate Docstrings

🪧 Tips

Chat

There are 3 ways to chat with CodeRabbit:

  • Review comments: Directly reply to a review comment made by CodeRabbit. Example:
    • I pushed a fix in commit <commit_id>, please review it.
    • Generate unit testing code for this file.
    • Open a follow-up GitHub issue for this discussion.
  • Files and specific lines of code (under the "Files changed" tab): Tag @coderabbitai in a new review comment at the desired location with your query. Examples:
    • @coderabbitai generate unit testing code for this file.
    • @coderabbitai modularize this function.
  • PR comments: Tag @coderabbitai in a new PR comment to ask questions about the PR branch. For the best results, please provide a very specific query, as very limited context is provided in this mode. Examples:
    • @coderabbitai gather interesting stats about this repository and render them as a table. Additionally, render a pie chart showing the language distribution in the codebase.
    • @coderabbitai read src/utils.ts and generate unit testing code.
    • @coderabbitai read the files in the src/scheduler package and generate a class diagram using mermaid and a README in the markdown format.
    • @coderabbitai help me debug CodeRabbit configuration file.

Note: Be mindful of the bot's finite context window. It's strongly recommended to break down tasks such as reading entire modules into smaller chunks. For a focused discussion, use review comments to chat about specific files and their changes, instead of using the PR comments.

CodeRabbit Commands (Invoked using PR comments)

  • @coderabbitai pause to pause the reviews on a PR.
  • @coderabbitai resume to resume the paused reviews.
  • @coderabbitai review to trigger an incremental review. This is useful when automatic reviews are disabled for the repository.
  • @coderabbitai full review to do a full review from scratch and review all the files again.
  • @coderabbitai summary to regenerate the summary of the PR.
  • @coderabbitai generate docstrings to generate docstrings for this PR.
  • @coderabbitai resolve resolve all the CodeRabbit review comments.
  • @coderabbitai configuration to show the current CodeRabbit configuration for the repository.
  • @coderabbitai help to get help.

Other keywords and placeholders

  • Add @coderabbitai ignore anywhere in the PR description to prevent this PR from being reviewed.
  • Add @coderabbitai summary to generate the high-level summary at a specific location in the PR description.
  • Add @coderabbitai or @coderabbitai title anywhere in the PR title to generate the title automatically.

CodeRabbit Configuration File (.coderabbit.yaml)

  • You can programmatically configure CodeRabbit by adding a .coderabbit.yaml file to the root of your repository.
  • Please see the configuration documentation for more information.
  • If your editor has YAML language server enabled, you can add the path at the top of this file to enable auto-completion and validation: # yaml-language-server: $schema=https://coderabbit.ai/integrations/schema.v2.json

Documentation and Community

  • Visit our Documentation for detailed information on how to use CodeRabbit.
  • Join our Discord Community to get help, request features, and share feedback.
  • Follow us on X/Twitter for updates and announcements.

Copy link

netlify bot commented Mar 10, 2025

Deploy Preview for kleros-v2-university ready!

Name Link
🔨 Latest commit 0500272
🔍 Latest deploy log https://app.netlify.com/sites/kleros-v2-university/deploys/67ce65f67da28800080528db
😎 Deploy Preview https://deploy-preview-1913--kleros-v2-university.netlify.app
📱 Preview on mobile
Toggle QR Code...

QR Code

Use your smartphone camera to open QR code link.

To edit notification comments on pull requests, go to your Netlify site configuration.

Copy link

netlify bot commented Mar 10, 2025

Deploy Preview for kleros-v2-testnet ready!

Name Link
🔨 Latest commit 0500272
🔍 Latest deploy log https://app.netlify.com/sites/kleros-v2-testnet/deploys/67ce65f665c95f0008b8b1b5
😎 Deploy Preview https://deploy-preview-1913--kleros-v2-testnet.netlify.app
📱 Preview on mobile
Toggle QR Code...

QR Code

Use your smartphone camera to open QR code link.

To edit notification comments on pull requests, go to your Netlify site configuration.

Copy link

netlify bot commented Mar 10, 2025

Deploy Preview for kleros-v2-testnet-devtools ready!

Name Link
🔨 Latest commit 0500272
🔍 Latest deploy log https://app.netlify.com/sites/kleros-v2-testnet-devtools/deploys/67ce65f625176a0008992e3f
😎 Deploy Preview https://deploy-preview-1913--kleros-v2-testnet-devtools.netlify.app
📱 Preview on mobile
Toggle QR Code...

QR Code

Use your smartphone camera to open QR code link.

To edit notification comments on pull requests, go to your Netlify site configuration.

Copy link

codeclimate bot commented Mar 10, 2025

Code Climate has analyzed commit 0500272 and detected 2 issues on this pull request.

Here's the issue category breakdown:

Category Count
Complexity 2

View more on Code Climate.

Copy link
Contributor

@coderabbitai coderabbitai bot left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Actionable comments posted: 4

🧹 Nitpick comments (3)
web/src/pages/Profile/StakingRewardsClaimModal.tsx (3)

70-70: Consider dynamic gas limit for different chains

The hardcoded gas limit of 500000 may not be appropriate for all chains or transaction conditions.

- gasLimit: 500000,
+ gasLimit: chainId === 421614 ? 500000 : 300000, // Adjust values based on specific chain requirements

Even better, you could define these values in your chain configuration object:

const chainIdToParams = {
    contractAddress: "0x9DdAeD4e2Ba34d59025c1A549311F621a8ff9b7b",
    snapshots: ["QmQBupnUD9zt2dzZcB6tNAENiWtmwfWeKDuZbWEWoKs7s2/arbSepolia-snapshot-2025-02.json"],
    startMonth: 2,
+   gasLimit: 500000,
  },
    // ...
+   gasLimit: 300000,
  },
};

// Then in your code:
gasLimit: chainParams.gasLimit,

84-98: Enhance UI with proper styling and feedback

The current UI is very minimalistic without proper styling or integration with the app's design system. Consider enhancing it to match the application's styling pattern.

The current implementation lacks:

  1. Proper styling consistent with the application
  2. Loading indicators that match the app's design
  3. Error messages for failed transactions
  4. Empty state when no claims are available

Consider wrapping your component in a styled container and adding more comprehensive UI elements to match the rest of the application.


102-102: Component name in export doesn't match filename

The file is named StakingRewardsClaimModal.tsx but the exported component is ClaimModal. This could lead to confusion and potential issues with imports.

- export default ClaimModal;
+ export default StakingRewardsClaimModal;

And update the component name:

- const ClaimModal = () => {
+ const StakingRewardsClaimModal = () => {
📜 Review details

Configuration used: CodeRabbit UI
Review profile: CHILL
Plan: Pro

📥 Commits

Reviewing files that changed from the base of the PR and between 8978cd9 and 0500272.

📒 Files selected for processing (2)
  • web/src/pages/Profile/StakingRewardsClaimModal.tsx (1 hunks)
  • web/src/pages/Profile/index.tsx (2 hunks)
⏰ Context from checks skipped due to timeout of 90000ms (16)
  • GitHub Check: Redirect rules - kleros-v2-testnet-devtools
  • GitHub Check: Header rules - kleros-v2-testnet-devtools
  • GitHub Check: Pages changed - kleros-v2-testnet-devtools
  • GitHub Check: Redirect rules - kleros-v2-testnet
  • GitHub Check: Redirect rules - kleros-v2-testnet
  • GitHub Check: Header rules - kleros-v2-testnet
  • GitHub Check: Header rules - kleros-v2-testnet
  • GitHub Check: Pages changed - kleros-v2-testnet
  • GitHub Check: Pages changed - kleros-v2-testnet
  • GitHub Check: Redirect rules - kleros-v2-university
  • GitHub Check: Header rules - kleros-v2-university
  • GitHub Check: Pages changed - kleros-v2-university
  • GitHub Check: SonarCloud
  • GitHub Check: contracts-testing
  • GitHub Check: Analyze (javascript)
  • GitHub Check: Mend Security Check
🔇 Additional comments (2)
web/src/pages/Profile/index.tsx (2)

22-22: Import naming matches export in file

The import name StakingRewardsClaimModal matches the export in the related file, which is good for code organization.


98-98: Properly implemented StakingRewardsClaimModal

The modal is correctly conditionally rendered within the connected user view and positioned appropriately in the component hierarchy. No props are being passed, which is consistent with the component's self-contained implementation.

Comment on lines +60 to +80
const handleClaim = async () => {
if (!claims.length || !account) return;
setLoading(true);

try {
await writeContractAsync({
abi: claimMonthsAbi,
address: chainParams.contractAddress,
functionName: "claimMonths",
args: [account, claims],
gasLimit: 500000,
});

setClaimed(true);
} catch (error) {
console.error("Transaction failed:", error);
setClaimed(false);
}

setLoading(false);
};
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

🛠️ Refactor suggestion

Improve error handling and user feedback for transactions

The current implementation logs errors to the console but doesn't provide meaningful feedback to the user when a transaction fails.

const handleClaim = async () => {
  if (!claims.length || !account) return;
  setLoading(true);
+ setError(null); // Add error state

  try {
    await writeContractAsync({
      abi: claimMonthsAbi,
      address: chainParams.contractAddress,
      functionName: "claimMonths",
      args: [account, claims],
      gasLimit: 500000,
    });

    setClaimed(true);
  } catch (error) {
    console.error("Transaction failed:", error);
+   // Provide a user-friendly error message
+   setError(error.message || "Transaction failed. Please try again.");
    setClaimed(false);
  }

  setLoading(false);
};

Also, don't forget to add the error state to your component:

const [error, setError] = useState(null);

And display it in your UI:

{error && <div className="error-message">{error}</div>}

Comment on lines +8 to +19
const chainIdToParams = {
421614: {
contractAddress: "0x9DdAeD4e2Ba34d59025c1A549311F621a8ff9b7b",
snapshots: ["QmQBupnUD9zt2dzZcB6tNAENiWtmwfWeKDuZbWEWoKs7s2/arbSepolia-snapshot-2025-02.json"],
startMonth: 2,
},
42161: {
contractAddress: "",
snapshots: [],
startMonth: 2,
},
};
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

⚠️ Potential issue

Missing contract configuration for production chain

The Arbitrum One (42161) configuration has empty values for contractAddress and snapshots. This will prevent the feature from working in production.

-    contractAddress: "",
-    snapshots: [],
+    contractAddress: "0xActualProductionAddress",
+    snapshots: ["actual/production/snapshot/path.json"],
    startMonth: 2,
  },
📝 Committable suggestion

‼️ IMPORTANT
Carefully review the code before committing. Ensure that it accurately replaces the highlighted code, contains no missing lines, and has no issues with indentation. Thoroughly test & benchmark the code to ensure it meets the requirements.

Suggested change
const chainIdToParams = {
421614: {
contractAddress: "0x9DdAeD4e2Ba34d59025c1A549311F621a8ff9b7b",
snapshots: ["QmQBupnUD9zt2dzZcB6tNAENiWtmwfWeKDuZbWEWoKs7s2/arbSepolia-snapshot-2025-02.json"],
startMonth: 2,
},
42161: {
contractAddress: "",
snapshots: [],
startMonth: 2,
},
};
const chainIdToParams = {
421614: {
contractAddress: "0x9DdAeD4e2Ba34d59025c1A549311F621a8ff9b7b",
snapshots: ["QmQBupnUD9zt2dzZcB6tNAENiWtmwfWeKDuZbWEWoKs7s2/arbSepolia-snapshot-2025-02.json"],
startMonth: 2,
},
42161: {
contractAddress: "0xActualProductionAddress",
snapshots: ["actual/production/snapshot/path.json"],
startMonth: 2,
},
};

Comment on lines +34 to +56
useEffect(() => {
const fetchClaims = async () => {
if (!account || !chainParams) return;

const userClaims = [];
for (let index = 0; index < chainParams.snapshots.length; index++) {
const response = await fetch(`${ipfsEndpoint}/ipfs/${chainParams.snapshots[index]}`);
const snapshot = await response.json();
const claim = snapshot.merkleTree.claims[account];

if (claim) {
userClaims.push({
month: chainParams.startMonth + index,
balance: BigInt(claim.value.hex),
merkleProof: claim.proof,
});
}
}
setClaims(userClaims);
};

fetchClaims();
}, [account, chainParams]);
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

🛠️ Refactor suggestion

Add error handling for IPFS fetch operations

The function fetches data from IPFS without proper error handling, which could lead to unhandled promise rejections if the network request fails.

const fetchClaims = async () => {
  if (!account || !chainParams) return;

  const userClaims = [];
  for (let index = 0; index < chainParams.snapshots.length; index++) {
    try {
      const response = await fetch(`${ipfsEndpoint}/ipfs/${chainParams.snapshots[index]}`);
+     if (!response.ok) {
+       console.error(`Failed to fetch snapshot: ${response.status} ${response.statusText}`);
+       continue;
+     }
      const snapshot = await response.json();
      const claim = snapshot.merkleTree.claims[account];

      if (claim) {
        userClaims.push({
          month: chainParams.startMonth + index,
          balance: BigInt(claim.value.hex),
          merkleProof: claim.proof,
        });
      }
+   } catch (error) {
+     console.error(`Error fetching snapshot ${index}:`, error);
+   }
  }
  setClaims(userClaims);
};
📝 Committable suggestion

‼️ IMPORTANT
Carefully review the code before committing. Ensure that it accurately replaces the highlighted code, contains no missing lines, and has no issues with indentation. Thoroughly test & benchmark the code to ensure it meets the requirements.

Suggested change
useEffect(() => {
const fetchClaims = async () => {
if (!account || !chainParams) return;
const userClaims = [];
for (let index = 0; index < chainParams.snapshots.length; index++) {
const response = await fetch(`${ipfsEndpoint}/ipfs/${chainParams.snapshots[index]}`);
const snapshot = await response.json();
const claim = snapshot.merkleTree.claims[account];
if (claim) {
userClaims.push({
month: chainParams.startMonth + index,
balance: BigInt(claim.value.hex),
merkleProof: claim.proof,
});
}
}
setClaims(userClaims);
};
fetchClaims();
}, [account, chainParams]);
useEffect(() => {
const fetchClaims = async () => {
if (!account || !chainParams) return;
const userClaims = [];
for (let index = 0; index < chainParams.snapshots.length; index++) {
try {
const response = await fetch(`${ipfsEndpoint}/ipfs/${chainParams.snapshots[index]}`);
if (!response.ok) {
console.error(`Failed to fetch snapshot: ${response.status} ${response.statusText}`);
continue;
}
const snapshot = await response.json();
const claim = snapshot.merkleTree.claims[account];
if (claim) {
userClaims.push({
month: chainParams.startMonth + index,
balance: BigInt(claim.value.hex),
merkleProof: claim.proof,
});
}
} catch (error) {
console.error(`Error fetching snapshot ${index}:`, error);
}
}
setClaims(userClaims);
};
fetchClaims();
}, [account, chainParams]);

Comment on lines +27 to +28
const chainId = DEFAULT_CHAIN;
const chainParams = chainIdToParams[chainId] ?? chainIdToParams[DEFAULT_CHAIN];
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

🛠️ Refactor suggestion

Hardcoded chain ID needs to be dynamic

The component uses a hardcoded DEFAULT_CHAIN rather than the user's connected chain, which could lead to incorrect contract interactions if the user is connected to a different network.

- const chainId = DEFAULT_CHAIN;
+ const { chain } = useNetwork();
+ const chainId = chain?.id || DEFAULT_CHAIN;

Make sure to add useNetwork to your wagmi imports.

📝 Committable suggestion

‼️ IMPORTANT
Carefully review the code before committing. Ensure that it accurately replaces the highlighted code, contains no missing lines, and has no issues with indentation. Thoroughly test & benchmark the code to ensure it meets the requirements.

Suggested change
const chainId = DEFAULT_CHAIN;
const chainParams = chainIdToParams[chainId] ?? chainIdToParams[DEFAULT_CHAIN];
const { chain } = useNetwork();
const chainId = chain?.id || DEFAULT_CHAIN;
const chainParams = chainIdToParams[chainId] ?? chainIdToParams[DEFAULT_CHAIN];

Copy link

netlify bot commented Mar 10, 2025

Deploy Preview for kleros-v2-neo ready!

Name Link
🔨 Latest commit 0500272
🔍 Latest deploy log https://app.netlify.com/sites/kleros-v2-neo/deploys/67ce65f67da28800080528dd
😎 Deploy Preview https://deploy-preview-1913--kleros-v2-neo.netlify.app
📱 Preview on mobile
Toggle QR Code...

QR Code

Use your smartphone camera to open QR code link.

To edit notification comments on pull requests, go to your Netlify site configuration.

# for free to join this conversation on GitHub. Already have an account? # to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

Staking Rewards Claim PNK
1 participant