mirror of
https://github.com/MarSeventh/CloudFlare-ImgBed.git
synced 2026-02-01 22:47:37 +00:00
- Add HuggingFace API wrapper class (huggingfaceAPI.js) - Support upload, download, delete operations via HuggingFace Hub API - Support public repos (unlimited storage) and private repos (100GB limit) - Private repos: server proxies requests with Authorization header - Auto-create repo if not exists (with write token) - Add HuggingFace to auto-retry channel list - Environment variables: HF_TOKEN, HF_REPO, HF_PRIVATE - Support load balancing for multiple HuggingFace channels
133 lines
4.9 KiB
JavaScript
133 lines
4.9 KiB
JavaScript
import { getUploadConfig } from '../api/manage/sysConfig/upload';
|
||
import { getSecurityConfig } from '../api/manage/sysConfig/security';
|
||
import { getPageConfig } from '../api/manage/sysConfig/page';
|
||
import { getOthersConfig } from '../api/manage/sysConfig/others';
|
||
import { getDatabase } from './databaseAdapter.js';
|
||
import { getIndexMeta } from './indexManager.js';
|
||
|
||
/**
|
||
* 根据容量限制过滤渠道
|
||
* @param {Object} context - 上下文对象(包含 env)
|
||
* @param {Array} channels - 渠道列表
|
||
* @returns {Array} 过滤后的渠道列表
|
||
*/
|
||
async function filterChannelsByQuota(context, channels) {
|
||
// 先检查是否有任何渠道启用了容量限制,如果都没启用则跳过 KV 读取
|
||
const hasQuotaEnabled = channels.some(ch => ch.quota?.enabled && ch.quota?.limitGB);
|
||
if (!hasQuotaEnabled) {
|
||
return channels; // 无需读取 KV,直接返回所有渠道
|
||
}
|
||
|
||
// 获取索引元数据(只需 1 次读取)
|
||
const indexMeta = await getIndexMeta(context);
|
||
const channelStats = indexMeta.channelStats || {};
|
||
|
||
const result = [];
|
||
for (const channel of channels) {
|
||
// 未启用容量限制,直接通过
|
||
if (!channel.quota?.enabled || !channel.quota?.limitGB) {
|
||
result.push(channel);
|
||
continue;
|
||
}
|
||
|
||
try {
|
||
// 从索引元数据中获取该渠道的容量统计
|
||
const stats = channelStats[channel.name] || { usedMB: 0, fileCount: 0 };
|
||
|
||
const usedGB = stats.usedMB / 1024;
|
||
const limitGB = channel.quota.limitGB;
|
||
const threshold = channel.quota.threshold || 95;
|
||
|
||
// 未超过阈值,渠道可用
|
||
if ((usedGB / limitGB) * 100 < threshold) {
|
||
result.push(channel);
|
||
} else {
|
||
console.log(`Channel ${channel.name} quota exceeded: ${usedGB.toFixed(2)}GB / ${limitGB}GB (${threshold}% threshold)`);
|
||
}
|
||
} catch (error) {
|
||
console.error(`Failed to check quota for channel ${channel.name}:`, error);
|
||
// 检查失败时保守处理,允许使用该渠道
|
||
result.push(channel);
|
||
}
|
||
}
|
||
return result;
|
||
}
|
||
|
||
export async function fetchUploadConfig(env, context = null) {
|
||
try {
|
||
const db = getDatabase(env);
|
||
const settings = await getUploadConfig(db, env);
|
||
// 去除 已禁用 的渠道
|
||
settings.telegram.channels = settings.telegram.channels.filter((channel) => channel.enabled);
|
||
settings.cfr2.channels = settings.cfr2.channels.filter((channel) => channel.enabled);
|
||
settings.s3.channels = settings.s3.channels.filter((channel) => channel.enabled);
|
||
settings.discord.channels = settings.discord.channels.filter((channel) => channel.enabled);
|
||
settings.huggingface.channels = settings.huggingface.channels.filter((channel) => channel.enabled);
|
||
|
||
// 根据容量限制过滤渠道(仅 R2 和 S3)
|
||
// 需要 context 来调用 getIndexMeta
|
||
if (context) {
|
||
settings.cfr2.channels = await filterChannelsByQuota(context, settings.cfr2.channels);
|
||
settings.s3.channels = await filterChannelsByQuota(context, settings.s3.channels);
|
||
}
|
||
|
||
return settings;
|
||
} catch (error) {
|
||
console.error('Failed to fetch upload config:', error);
|
||
// 返回默认配置
|
||
return {
|
||
telegram: { channels: [] },
|
||
cfr2: { channels: [] },
|
||
s3: { channels: [] },
|
||
discord: { channels: [] },
|
||
huggingface: { channels: [] }
|
||
};
|
||
}
|
||
}
|
||
|
||
export async function fetchSecurityConfig(env) {
|
||
try {
|
||
const db = getDatabase(env);
|
||
const settings = await getSecurityConfig(db, env);
|
||
return settings;
|
||
} catch (error) {
|
||
console.error('Failed to fetch security config:', error);
|
||
// 返回默认配置
|
||
return {
|
||
auth: {
|
||
user: { authCode: "" },
|
||
admin: { adminUsername: "", adminPassword: "" }
|
||
},
|
||
upload: {
|
||
moderate: { enabled: false, channel: "default", moderateContentApiKey: "", nsfwApiPath: "" }
|
||
},
|
||
access: { allowedDomains: "", whiteListMode: false }
|
||
};
|
||
}
|
||
}
|
||
|
||
export async function fetchPageConfig(env) {
|
||
try {
|
||
const db = getDatabase(env);
|
||
const settings = await getPageConfig(db, env);
|
||
return settings;
|
||
} catch (error) {
|
||
console.error('Failed to fetch page config:', error);
|
||
// 返回默认配置
|
||
return { config: [] };
|
||
}
|
||
}
|
||
|
||
export async function fetchOthersConfig(env) {
|
||
try {
|
||
const db = getDatabase(env);
|
||
const settings = await getOthersConfig(db, env);
|
||
return settings;
|
||
} catch (error) {
|
||
console.error('Failed to fetch others config:', error);
|
||
// 返回默认配置
|
||
return {
|
||
telemetry: { enabled: false }
|
||
};
|
||
}
|
||
} |