2025-12-18 13:07:35 +08:00
|
|
|
|
import { ethers, upgrades } from "hardhat";
|
|
|
|
|
|
import * as fs from "fs";
|
|
|
|
|
|
import * as path from "path";
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
|
* 配置 Lending 借贷池参数并部署代理
|
2025-12-26 13:23:50 +08:00
|
|
|
|
* 包含:配置市场参数、添加 YT 抵押资产、部署 Lending 代理
|
2025-12-18 13:07:35 +08:00
|
|
|
|
*/
|
|
|
|
|
|
async function main() {
|
|
|
|
|
|
const [deployer] = await ethers.getSigners();
|
|
|
|
|
|
console.log("\n==========================================");
|
|
|
|
|
|
console.log("⚙️ 配置 Lending 借贷池");
|
|
|
|
|
|
console.log("==========================================");
|
|
|
|
|
|
console.log("配置账户:", deployer.address, "\n");
|
|
|
|
|
|
|
|
|
|
|
|
// ========== 读取部署信息 ==========
|
|
|
|
|
|
const deploymentsPath = path.join(__dirname, "../../deployments-lending.json");
|
|
|
|
|
|
if (!fs.existsSync(deploymentsPath)) {
|
|
|
|
|
|
throw new Error("未找到部署信息文件,请先运行 07-deployLending.ts");
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const network = await ethers.provider.getNetwork();
|
|
|
|
|
|
const chainId = network.chainId.toString();
|
|
|
|
|
|
const deployments = JSON.parse(fs.readFileSync(deploymentsPath, "utf-8"))[chainId];
|
|
|
|
|
|
|
|
|
|
|
|
if (!deployments) {
|
|
|
|
|
|
throw new Error(`未找到网络 ${chainId} 的部署信息`);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
console.log("📋 使用已部署的合约:");
|
|
|
|
|
|
console.log(" LendingFactory:", deployments.lendingFactory);
|
2025-12-26 13:23:50 +08:00
|
|
|
|
console.log(" LendingPriceFeed (Proxy):", deployments.lendingPriceFeed);
|
|
|
|
|
|
if (deployments.lendingPriceFeedImpl) {
|
|
|
|
|
|
console.log(" LendingPriceFeed (Impl):", deployments.lendingPriceFeedImpl);
|
|
|
|
|
|
}
|
|
|
|
|
|
console.log(" Configurator (Proxy):", deployments.configuratorProxy);
|
|
|
|
|
|
console.log(" Lending (Impl):", deployments.lendingImpl);
|
|
|
|
|
|
console.log(" USDC Address:", deployments.usdcAddress);
|
|
|
|
|
|
console.log(" USDC Price Feed:", deployments.usdcPriceFeed, "\n");
|
2025-12-18 13:07:35 +08:00
|
|
|
|
|
2025-12-26 13:23:50 +08:00
|
|
|
|
// ========== 读取 YT Vault 部署信息 ==========
|
|
|
|
|
|
const vaultDeploymentsPath = path.join(__dirname, "../../deployments-vault-system.json");
|
|
|
|
|
|
if (!fs.existsSync(vaultDeploymentsPath)) {
|
|
|
|
|
|
throw new Error("未找到 YT Vault 部署信息文件,请先部署 YT Vault 系统");
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const vaultDeployments = JSON.parse(fs.readFileSync(vaultDeploymentsPath, "utf-8"));
|
|
|
|
|
|
if (!vaultDeployments.vaults || vaultDeployments.vaults.length === 0) {
|
|
|
|
|
|
throw new Error("未找到已部署的 YT Vault,请先创建至少一个 YT Vault");
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
console.log("📋 找到 YT Vaults:", vaultDeployments.vaults.length);
|
|
|
|
|
|
vaultDeployments.vaults.forEach((vault: any, index: number) => {
|
|
|
|
|
|
console.log(` ${index + 1}. ${vault.name} (${vault.symbol}): ${vault.address}`);
|
|
|
|
|
|
});
|
|
|
|
|
|
console.log();
|
2025-12-18 13:07:35 +08:00
|
|
|
|
|
2025-12-26 13:23:50 +08:00
|
|
|
|
// ========== 第一阶段:配置参数 ==========
|
|
|
|
|
|
console.log("⚙️ Phase 1: 准备配置参数");
|
2025-12-18 13:07:35 +08:00
|
|
|
|
|
2025-12-26 13:23:50 +08:00
|
|
|
|
const USDC = {
|
|
|
|
|
|
address: deployments.usdcAddress,
|
2026-01-08 11:30:31 +08:00
|
|
|
|
decimals: 18
|
2025-12-18 13:07:35 +08:00
|
|
|
|
};
|
2025-12-26 13:23:50 +08:00
|
|
|
|
|
|
|
|
|
|
// 选择要作为抵押品的 YT Vaults(可以选择多个)
|
|
|
|
|
|
// todo: 根据需要修改这里,选择哪些 YT Vault 作为抵押品
|
|
|
|
|
|
const selectedVaults = vaultDeployments.vaults.slice(0, 3); // 默认选择前3个
|
2025-12-18 13:07:35 +08:00
|
|
|
|
|
2025-12-26 13:23:50 +08:00
|
|
|
|
console.log(" 选择的抵押品 YT Vaults:");
|
|
|
|
|
|
selectedVaults.forEach((vault: any, index: number) => {
|
|
|
|
|
|
console.log(` ${index + 1}. ${vault.name}: ${vault.address}`);
|
|
|
|
|
|
});
|
|
|
|
|
|
console.log();
|
|
|
|
|
|
|
|
|
|
|
|
// 准备抵押资产配置
|
|
|
|
|
|
const assetConfigs = selectedVaults.map((vault: any) => ({
|
|
|
|
|
|
asset: vault.address,
|
|
|
|
|
|
decimals: 18, // YT Token 都是 18 decimals
|
|
|
|
|
|
borrowCollateralFactor: ethers.parseUnits("0.80", 18), // 80% LTV
|
|
|
|
|
|
liquidateCollateralFactor: ethers.parseUnits("0.85", 18), // 85% 清算线
|
|
|
|
|
|
liquidationFactor: ethers.parseUnits("0.95", 18), // 95% (配合 storeFrontPriceFactor 产生折扣)
|
|
|
|
|
|
supplyCap: ethers.parseUnits("100000", 18) // 最多 10 万 YT
|
|
|
|
|
|
}));
|
2025-12-18 13:07:35 +08:00
|
|
|
|
|
|
|
|
|
|
// ========== 第二阶段:准备配置参数 ==========
|
|
|
|
|
|
console.log("⚙️ Phase 2: 准备市场配置参数");
|
|
|
|
|
|
|
|
|
|
|
|
const configuration = {
|
|
|
|
|
|
baseToken: USDC.address,
|
2025-12-26 13:23:50 +08:00
|
|
|
|
lendingPriceSource: deployments.lendingPriceFeed,
|
2025-12-18 13:07:35 +08:00
|
|
|
|
|
|
|
|
|
|
// 利率模型参数(年化利率,18位精度)
|
|
|
|
|
|
// 注意:这些年化利率会在 initialize 时自动转换为每秒利率
|
|
|
|
|
|
// 转换公式:perSecondRate = perYearRate / 31,536,000
|
|
|
|
|
|
supplyKink: ethers.parseUnits("0.8", 18), // 80% 利用率拐点
|
|
|
|
|
|
supplyPerYearInterestRateSlopeLow: ethers.parseUnits("0.03", 18), // 3% APY
|
|
|
|
|
|
supplyPerYearInterestRateSlopeHigh: ethers.parseUnits("0.4", 18), // 40% APY
|
|
|
|
|
|
supplyPerYearInterestRateBase: ethers.parseUnits("0", 18), // 0% 基础
|
|
|
|
|
|
|
|
|
|
|
|
borrowKink: ethers.parseUnits("0.8", 18), // 80% 利用率拐点
|
|
|
|
|
|
borrowPerYearInterestRateSlopeLow: ethers.parseUnits("0.05", 18), // 5% APY
|
|
|
|
|
|
borrowPerYearInterestRateSlopeHigh: ethers.parseUnits("1.5", 18), // 150% APY
|
|
|
|
|
|
borrowPerYearInterestRateBase: ethers.parseUnits("0.015", 18), // 1.5% 基础
|
|
|
|
|
|
|
|
|
|
|
|
storeFrontPriceFactor: ethers.parseUnits("0.5", 18), // 50% 清算折扣
|
|
|
|
|
|
baseBorrowMin: ethers.parseUnits("100", USDC.decimals), // 最小借 100 USDC
|
|
|
|
|
|
targetReserves: ethers.parseUnits("5000000", USDC.decimals), // 目标储备 500 万
|
|
|
|
|
|
|
2025-12-26 13:23:50 +08:00
|
|
|
|
assetConfigs: assetConfigs
|
2025-12-18 13:07:35 +08:00
|
|
|
|
};
|
|
|
|
|
|
|
2025-12-26 13:23:50 +08:00
|
|
|
|
console.log("✅ 配置参数已准备");
|
|
|
|
|
|
console.log(" 基础资产: USDC (6 decimals)");
|
|
|
|
|
|
console.log(" 价格源: LendingPriceFeed");
|
|
|
|
|
|
console.log(" 抵押资产数量:", assetConfigs.length);
|
|
|
|
|
|
console.log(" Supply Kink: 80%");
|
|
|
|
|
|
console.log(" Borrow Kink: 80%");
|
|
|
|
|
|
console.log(" 最小借款: 100 USDC");
|
|
|
|
|
|
console.log(" 目标储备: 5,000,000 USDC\n");
|
2025-12-18 13:07:35 +08:00
|
|
|
|
|
2025-12-26 13:23:50 +08:00
|
|
|
|
// ========== 第三阶段:部署 Lending 代理 ==========
|
|
|
|
|
|
console.log("⚙️ Phase 3: 部署 Lending 代理");
|
2025-12-18 13:07:35 +08:00
|
|
|
|
|
|
|
|
|
|
const Lending = await ethers.getContractFactory("Lending");
|
|
|
|
|
|
|
|
|
|
|
|
// 使用 upgrades 插件部署 UUPS 代理
|
|
|
|
|
|
const lending = await upgrades.deployProxy(
|
|
|
|
|
|
Lending,
|
2025-12-26 13:23:50 +08:00
|
|
|
|
[configuration],
|
2025-12-18 13:07:35 +08:00
|
|
|
|
{
|
|
|
|
|
|
kind: "uups",
|
|
|
|
|
|
initializer: "initialize"
|
|
|
|
|
|
}
|
|
|
|
|
|
);
|
|
|
|
|
|
await lending.waitForDeployment();
|
|
|
|
|
|
const lendingProxyAddress = await lending.getAddress();
|
2025-12-26 13:23:50 +08:00
|
|
|
|
console.log("✅ Lending Proxy 已部署:", lendingProxyAddress);
|
2025-12-18 13:07:35 +08:00
|
|
|
|
|
|
|
|
|
|
// 获取实现合约地址(验证)
|
|
|
|
|
|
const lendingImplAddress = await upgrades.erc1967.getImplementationAddress(lendingProxyAddress);
|
2025-12-26 13:23:50 +08:00
|
|
|
|
console.log("✅ Lending Implementation (验证):", lendingImplAddress);
|
|
|
|
|
|
|
|
|
|
|
|
// 验证基本信息
|
|
|
|
|
|
console.log("\n📊 验证部署信息:");
|
|
|
|
|
|
const baseToken = await lending.baseToken();
|
|
|
|
|
|
const priceSource = await lending.lendingPriceSource();
|
|
|
|
|
|
const totalSupply = await lending.getTotalSupply();
|
|
|
|
|
|
const totalBorrow = await lending.getTotalBorrow();
|
|
|
|
|
|
|
|
|
|
|
|
console.log(" Base Token:", baseToken);
|
|
|
|
|
|
console.log(" Price Source:", priceSource);
|
|
|
|
|
|
console.log(" Total Supply:", totalSupply.toString());
|
|
|
|
|
|
console.log(" Total Borrow:", totalBorrow.toString());
|
|
|
|
|
|
console.log();
|
2025-12-18 13:07:35 +08:00
|
|
|
|
|
|
|
|
|
|
// ========== 保存部署信息 ==========
|
|
|
|
|
|
deployments.lendingProxy = lendingProxyAddress;
|
2025-12-26 13:23:50 +08:00
|
|
|
|
deployments.collateralAssets = selectedVaults.map((v: any) => ({
|
|
|
|
|
|
name: v.name,
|
|
|
|
|
|
symbol: v.symbol,
|
|
|
|
|
|
address: v.address
|
|
|
|
|
|
}));
|
2025-12-18 13:07:35 +08:00
|
|
|
|
deployments.configTimestamp = new Date().toISOString();
|
|
|
|
|
|
|
|
|
|
|
|
const allDeployments = JSON.parse(fs.readFileSync(deploymentsPath, "utf-8"));
|
|
|
|
|
|
allDeployments[chainId] = deployments;
|
|
|
|
|
|
fs.writeFileSync(deploymentsPath, JSON.stringify(allDeployments, null, 2));
|
|
|
|
|
|
|
|
|
|
|
|
console.log("💾 配置信息已保存到:", deploymentsPath);
|
|
|
|
|
|
|
|
|
|
|
|
// ========== 配置总结 ==========
|
2025-12-26 13:23:50 +08:00
|
|
|
|
console.log("\n🎉 部署和配置完成!");
|
2025-12-18 13:07:35 +08:00
|
|
|
|
console.log("=====================================");
|
|
|
|
|
|
console.log("Lending Proxy: ", lendingProxyAddress);
|
2025-12-26 13:23:50 +08:00
|
|
|
|
console.log("Base Token (USDC): ", USDC.address);
|
|
|
|
|
|
console.log("Price Feed: ", deployments.lendingPriceFeed);
|
2025-12-18 13:07:35 +08:00
|
|
|
|
console.log("Collateral Assets: ", configuration.assetConfigs.length);
|
|
|
|
|
|
console.log("Supply Kink: ", "80%");
|
|
|
|
|
|
console.log("Borrow Kink: ", "80%");
|
|
|
|
|
|
console.log("Min Borrow: ", "100 USDC");
|
2025-12-26 13:23:50 +08:00
|
|
|
|
console.log("Target Reserves: ", "5,000,000 USDC");
|
|
|
|
|
|
console.log("=====================================");
|
|
|
|
|
|
console.log("\n📋 抵押资产列表:");
|
|
|
|
|
|
selectedVaults.forEach((vault: any, index: number) => {
|
|
|
|
|
|
console.log(` ${index + 1}. ${vault.name} (${vault.symbol})`);
|
|
|
|
|
|
console.log(` 地址: ${vault.address}`);
|
|
|
|
|
|
console.log(` LTV: 80%, 清算线: 85%`);
|
|
|
|
|
|
});
|
2025-12-18 13:07:35 +08:00
|
|
|
|
console.log("=====================================\n");
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
main()
|
|
|
|
|
|
.then(() => process.exit(0))
|
|
|
|
|
|
.catch((error) => {
|
|
|
|
|
|
console.error(error);
|
|
|
|
|
|
process.exit(1);
|
|
|
|
|
|
});
|
|
|
|
|
|
|