// SPDX-License-Identifier: MIT
pragma solidity ^0.8.0;
import "@openzeppelin/contracts/security/ReentrancyGuard.sol";
contract LendingPool is ReentrancyGuard {
mapping(address => uint256) public collateral; // 用户抵押的 JU 数量
mapping(address => uint256) public borrowed; // 用户借出的 JU 数量
uint256 public constant LTV = 50; // 贷款价值比(Loan-to-Value),50%
uint256 public constant INTEREST_RATE = 5; // 年化利率 5%(简化为每次还款收取)
event Deposited(address indexed user, uint256 amount);
event Borrowed(address indexed user, uint256 amount);
event Repaid(address indexed user, uint256 amount);
event Withdrawn(address indexed user, uint256 amount);
constructor() {
// 无需参数,JU 是原生代币
}
// 存入 JU 作为抵押
function depositCollateral() external payable nonReentrant {
require(msg.value > 0, "Amount must be greater than 0");
collateral[msg.sender] += msg.value;
emit Deposited(msg.sender, msg.value);
}
// 借出 JU,基于抵押的 50% LTV
function borrow(uint256 amount) external nonReentrant {
require(amount > 0, "Amount must be greater than 0");
uint256 maxBorrow = (collateral[msg.sender] * LTV) / 100;
require(borrowed[msg.sender] + amount <= maxBorrow, "Exceeds max borrowable amount");
borrowed[msg.sender] += amount;
(bool success, ) = msg.sender.call{value: amount}("");
require(success, "Transfer failed");
emit Borrowed(msg.sender, amount);
}
// 偿还借款(含利息)
function repay(uint256 amount) external payable nonReentrant {
require(amount > 0, "Amount must be greater than 0");
require(borrowed[msg.sender] >= amount, "Repay exceeds borrowed amount");
uint256 interest = (amount * INTEREST_RATE) / 100;
uint256 totalRepay = amount + interest;
require(msg.value >= totalRepay, "Insufficient payment");
borrowed[msg.sender] -= amount;
emit Repaid(msg.sender, amount);
// 退回多余的 JU
if (msg.value > totalRepay) {
(bool success, ) = msg.sender.call{value: msg.value - totalRepay}("");
require(success, "Refund failed");
}
}
// 提取抵押(需先还清借款)
function withdrawCollateral(uint256 amount) external nonReentrant {
require(amount > 0, "Amount must be greater than 0");
require(collateral[msg.sender] >= amount, "Insufficient collateral");
require(borrowed[msg.sender] == 0, "Must repay all loans first");
collateral[msg.sender] -= amount;
(bool success, ) = msg.sender.call{value: amount}("");
require(success, "Transfer failed");
emit Withdrawn(msg.sender, amount);
}
// 查看用户状态
function getUserStatus(address user) external view returns (uint256, uint256, uint256) {
uint256 maxBorrow = (collateral[user] * LTV) / 100;
return (collateral[user], borrowed[user], maxBorrow);
}
}
const HDWalletProvider = require("@truffle/hdwallet-provider");
const mnemonic = "你的 MetaMask 助记词"; // 替换为你的助记词
module.exports = {
networks: {
juchain_testnet: {
provider: () => new HDWalletProvider(mnemonic, "https://testnet-rpc.juchain.org"),
network_id: 202599,
gas: 5500000,
gasPrice: 1000000000, // 1 gwei
},
},
compilers: {
solc: {
version: "0.8.0",
},
},
};
<!DOCTYPE html>
<html>
<head>
<title>JuChain Lending dApp</title>
<style>
body { font-family: Arial, sans-serif; text-align: center; margin-top: 50px; }
button { padding: 10px 20px; margin: 5px; }
input { padding: 5px; margin: 5px; }
</style>
</head>
<body>
<h1>JuChain Lending Pool</h1>
<p>Collateral: <span id="collateral">0</span> JU</p>
<p>Borrowed: <span id="borrowed">0</span> JU</p>
<p>Max Borrowable: <span id="maxBorrow">0</span> JU</p>
<input id="amount" type="number" placeholder="Amount in JU" />
<br />
<button onclick="deposit()">Deposit Collateral</button>
<button onclick="borrow()">Borrow JU</button>
<button onclick="repay()">Repay Loan</button>
<button onclick="withdraw()">Withdraw Collateral</button>
<script src="https://cdn.jsdelivr.net/npm/web3@1.5.3/dist/web3.min.js"></script>
<script>
const contractAddress = "你的 LendingPool 合约地址"; // 替换为实际地址
const abi = [ /* 从 build/contracts/LendingPool.json 获取 ABI */ ]; // 替换为完整 ABI
let web3, lending;
async function init() {
if (window.ethereum) {
web3 = new Web3(window.ethereum);
await window.ethereum.enable();
lending = new web3.eth.Contract(abi, contractAddress);
updateStatus();
} else {
alert("Please install MetaMask!");
}
}
async function updateStatus() {
const accounts = await web3.eth.getAccounts();
const status = await lending.methods.getUserStatus(accounts[0]).call();
document.getElementById("collateral").innerText = web3.utils.fromWei(status[0]);
document.getElementById("borrowed").innerText = web3.utils.fromWei(status[1]);
document.getElementById("maxBorrow").innerText = web3.utils.fromWei(status[2]);
}
async function deposit() {
const accounts = await web3.eth.getAccounts();
const amount = web3.utils.toWei(document.getElementById("amount").value);
await lending.methods.depositCollateral().send({ from: accounts[0], value: amount });
updateStatus();
}
async function borrow() {
const accounts = await web3.eth.getAccounts();
const amount = web3.utils.toWei(document.getElementById("amount").value);
await lending.methods.borrow(amount).send({ from: accounts[0] });
updateStatus();
}
async function repay() {
const accounts = await web3.eth.getAccounts();
const amount = web3.utils.toWei(document.getElementById("amount").value);
const interest = (BigInt(amount) * 5n) / 100n;
const total = (BigInt(amount) + interest).toString();
await lending.methods.repay(amount).send({ from: accounts[0], value: total });
updateStatus();
}
async function withdraw() {
const accounts = await web3.eth.getAccounts();
const amount = web3.utils.toWei(document.getElementById("amount").value);
await lending.methods.withdrawCollateral(amount).send({ from: accounts[0] });
updateStatus();
}
init();
</script>
</body>
</html>