sync: align local with upstream main
This commit is contained in:
@@ -5,6 +5,7 @@
|
||||
|
||||
import { zipSync, unzipSync, strToU8, strFromU8 } from '../../../../libs/fflate.mjs';
|
||||
import { getContext } from '../../../../../../../extensions.js';
|
||||
import { getRequestHeaders } from '../../../../../../../../script.js';
|
||||
import { xbLog } from '../../../../core/debug-core.js';
|
||||
import {
|
||||
getMeta,
|
||||
@@ -72,6 +73,37 @@ function downloadBlob(blob, filename) {
|
||||
document.body.removeChild(a);
|
||||
URL.revokeObjectURL(url);
|
||||
}
|
||||
// 二进制 Uint8Array → base64(分块处理,避免 btoa 栈溢出)
|
||||
function uint8ToBase64(uint8) {
|
||||
const CHUNK = 0x8000;
|
||||
let result = '';
|
||||
for (let i = 0; i < uint8.length; i += CHUNK) {
|
||||
result += String.fromCharCode.apply(null, uint8.subarray(i, i + CHUNK));
|
||||
}
|
||||
return btoa(result);
|
||||
}
|
||||
|
||||
// base64 → Uint8Array
|
||||
function base64ToUint8(base64) {
|
||||
const binary = atob(base64);
|
||||
const bytes = new Uint8Array(binary.length);
|
||||
for (let i = 0; i < binary.length; i++) {
|
||||
bytes[i] = binary.charCodeAt(i);
|
||||
}
|
||||
return bytes;
|
||||
}
|
||||
|
||||
// 服务器备份文件名
|
||||
function getBackupFilename(chatId) {
|
||||
// chatId 可能含中文/特殊字符,ST 只接受 [a-zA-Z0-9_-]
|
||||
// 用简单 hash 生成安全文件名
|
||||
let hash = 0;
|
||||
for (let i = 0; i < chatId.length; i++) {
|
||||
hash = ((hash << 5) - hash + chatId.charCodeAt(i)) | 0;
|
||||
}
|
||||
const safe = (hash >>> 0).toString(36);
|
||||
return `LWB_VectorBackup_${safe}.zip`;
|
||||
}
|
||||
|
||||
// ═══════════════════════════════════════════════════════════════════════════
|
||||
// 导出
|
||||
@@ -383,3 +415,465 @@ export async function importVectors(file, onProgress) {
|
||||
fingerprintMismatch,
|
||||
};
|
||||
}
|
||||
// ═══════════════════════════════════════════════════════════════════════════
|
||||
// 备份到服务器
|
||||
// ═══════════════════════════════════════════════════════════════════════════
|
||||
|
||||
export async function backupToServer(onProgress) {
|
||||
const { chatId } = getContext();
|
||||
if (!chatId) {
|
||||
throw new Error('未打开聊天');
|
||||
}
|
||||
|
||||
onProgress?.('读取数据...');
|
||||
|
||||
const meta = await getMeta(chatId);
|
||||
const chunks = await getAllChunks(chatId);
|
||||
const chunkVectors = await getAllChunkVectors(chatId);
|
||||
const eventVectors = await getAllEventVectors(chatId);
|
||||
const stateAtoms = getStateAtoms();
|
||||
const stateVectors = await getAllStateVectors(chatId);
|
||||
|
||||
if (chunkVectors.length === 0 && eventVectors.length === 0 && stateVectors.length === 0) {
|
||||
throw new Error('没有可备份的向量数据');
|
||||
}
|
||||
|
||||
const dims = chunkVectors[0]?.vector?.length
|
||||
|| eventVectors[0]?.vector?.length
|
||||
|| stateVectors[0]?.vector?.length
|
||||
|| 0;
|
||||
if (dims === 0) {
|
||||
throw new Error('无法确定向量维度');
|
||||
}
|
||||
|
||||
onProgress?.('构建索引...');
|
||||
|
||||
const sortedChunks = [...chunks].sort((a, b) => a.chunkId.localeCompare(b.chunkId));
|
||||
const chunkVectorMap = new Map(chunkVectors.map(cv => [cv.chunkId, cv.vector]));
|
||||
|
||||
const chunksJsonl = sortedChunks.map(c => JSON.stringify({
|
||||
chunkId: c.chunkId,
|
||||
floor: c.floor,
|
||||
chunkIdx: c.chunkIdx,
|
||||
speaker: c.speaker,
|
||||
isUser: c.isUser,
|
||||
text: c.text,
|
||||
textHash: c.textHash,
|
||||
})).join('\n');
|
||||
|
||||
const chunkVectorsOrdered = sortedChunks.map(c => chunkVectorMap.get(c.chunkId) || new Array(dims).fill(0));
|
||||
|
||||
onProgress?.('压缩向量...');
|
||||
|
||||
const sortedEventVectors = [...eventVectors].sort((a, b) => a.eventId.localeCompare(b.eventId));
|
||||
const eventsJsonl = sortedEventVectors.map(ev => JSON.stringify({
|
||||
eventId: ev.eventId,
|
||||
})).join('\n');
|
||||
const eventVectorsOrdered = sortedEventVectors.map(ev => ev.vector);
|
||||
|
||||
const sortedStateVectors = [...stateVectors].sort((a, b) => String(a.atomId).localeCompare(String(b.atomId)));
|
||||
const stateVectorsOrdered = sortedStateVectors.map(v => v.vector);
|
||||
const rDims = sortedStateVectors.find(v => v.rVector?.length)?.rVector?.length || dims;
|
||||
const stateRVectorsOrdered = sortedStateVectors.map(v =>
|
||||
v.rVector?.length ? v.rVector : new Array(rDims).fill(0)
|
||||
);
|
||||
const stateVectorsJsonl = sortedStateVectors.map(v => JSON.stringify({
|
||||
atomId: v.atomId,
|
||||
floor: v.floor,
|
||||
hasRVector: !!(v.rVector?.length),
|
||||
rDims: v.rVector?.length || 0,
|
||||
})).join('\n');
|
||||
|
||||
const manifest = {
|
||||
version: EXPORT_VERSION,
|
||||
exportedAt: Date.now(),
|
||||
chatId,
|
||||
fingerprint: meta.fingerprint || '',
|
||||
dims,
|
||||
chunkCount: sortedChunks.length,
|
||||
chunkVectorCount: chunkVectors.length,
|
||||
eventCount: sortedEventVectors.length,
|
||||
stateAtomCount: stateAtoms.length,
|
||||
stateVectorCount: stateVectors.length,
|
||||
stateRVectorCount: sortedStateVectors.filter(v => v.rVector?.length).length,
|
||||
rDims,
|
||||
lastChunkFloor: meta.lastChunkFloor ?? -1,
|
||||
};
|
||||
|
||||
onProgress?.('打包文件...');
|
||||
|
||||
const zipData = zipSync({
|
||||
'manifest.json': strToU8(JSON.stringify(manifest, null, 2)),
|
||||
'chunks.jsonl': strToU8(chunksJsonl),
|
||||
'chunk_vectors.bin': float32ToBytes(chunkVectorsOrdered, dims),
|
||||
'events.jsonl': strToU8(eventsJsonl),
|
||||
'event_vectors.bin': float32ToBytes(eventVectorsOrdered, dims),
|
||||
'state_atoms.json': strToU8(JSON.stringify(stateAtoms)),
|
||||
'state_vectors.jsonl': strToU8(stateVectorsJsonl),
|
||||
'state_vectors.bin': stateVectorsOrdered.length
|
||||
? float32ToBytes(stateVectorsOrdered, dims)
|
||||
: new Uint8Array(0),
|
||||
'state_r_vectors.bin': stateRVectorsOrdered.length
|
||||
? float32ToBytes(stateRVectorsOrdered, rDims)
|
||||
: new Uint8Array(0),
|
||||
}, { level: 1 });
|
||||
|
||||
onProgress?.('上传到服务器...');
|
||||
|
||||
const base64 = uint8ToBase64(zipData);
|
||||
const filename = getBackupFilename(chatId);
|
||||
|
||||
const res = await fetch('/api/files/upload', {
|
||||
method: 'POST',
|
||||
headers: getRequestHeaders(),
|
||||
body: JSON.stringify({ name: filename, data: base64 }),
|
||||
});
|
||||
if (!res.ok) {
|
||||
throw new Error(`服务器返回 ${res.status}`);
|
||||
}
|
||||
|
||||
// 新增:安全读取 path 字段
|
||||
let uploadedPath = null;
|
||||
try {
|
||||
const resJson = await res.json();
|
||||
if (typeof resJson?.path === 'string') uploadedPath = resJson.path;
|
||||
} catch (_) { /* JSON 解析失败时 uploadedPath 保持 null */ }
|
||||
|
||||
// 新增:写清单(独立 try/catch,失败不影响原有备份返回)
|
||||
try {
|
||||
await upsertManifestEntry({
|
||||
filename,
|
||||
serverPath: uploadedPath,
|
||||
size: zipData.byteLength,
|
||||
chatId,
|
||||
backupTime: new Date().toISOString(),
|
||||
});
|
||||
} catch (e) {
|
||||
xbLog.warn(MODULE_ID, `清单写入失败(不影响备份结果): ${e.message}`);
|
||||
}
|
||||
|
||||
const sizeMB = (zipData.byteLength / 1024 / 1024).toFixed(2);
|
||||
xbLog.info(MODULE_ID, `备份完成: ${filename} (${sizeMB}MB)`);
|
||||
|
||||
return {
|
||||
filename,
|
||||
size: zipData.byteLength,
|
||||
chunkCount: sortedChunks.length,
|
||||
eventCount: sortedEventVectors.length,
|
||||
};
|
||||
}
|
||||
|
||||
// ═══════════════════════════════════════════════════════════════════════════
|
||||
// 从服务器恢复
|
||||
// ═══════════════════════════════════════════════════════════════════════════
|
||||
|
||||
export async function restoreFromServer(onProgress) {
|
||||
const { chatId } = getContext();
|
||||
if (!chatId) {
|
||||
throw new Error('未打开聊天');
|
||||
}
|
||||
|
||||
onProgress?.('从服务器下载...');
|
||||
|
||||
const filename = getBackupFilename(chatId);
|
||||
const res = await fetch(`/user/files/${filename}`, {
|
||||
headers: getRequestHeaders(),
|
||||
cache: 'no-cache',
|
||||
});
|
||||
|
||||
if (!res.ok) {
|
||||
if (res.status === 404) {
|
||||
throw new Error('服务器上没有找到此聊天的备份');
|
||||
}
|
||||
throw new Error(`服务器返回 ${res.status}`);
|
||||
}
|
||||
|
||||
const arrayBuffer = await res.arrayBuffer();
|
||||
if (!arrayBuffer || arrayBuffer.byteLength === 0) {
|
||||
throw new Error('服务器上没有找到此聊天的备份');
|
||||
}
|
||||
|
||||
onProgress?.('解压文件...');
|
||||
|
||||
const zipData = new Uint8Array(arrayBuffer);
|
||||
|
||||
let unzipped;
|
||||
try {
|
||||
unzipped = unzipSync(zipData);
|
||||
} catch (e) {
|
||||
throw new Error('备份文件格式错误,无法解压');
|
||||
}
|
||||
|
||||
if (!unzipped['manifest.json']) {
|
||||
throw new Error('缺少 manifest.json');
|
||||
}
|
||||
|
||||
const manifest = JSON.parse(strFromU8(unzipped['manifest.json']));
|
||||
|
||||
if (![1, 2].includes(manifest.version)) {
|
||||
throw new Error(`不支持的版本: ${manifest.version}`);
|
||||
}
|
||||
|
||||
onProgress?.('校验数据...');
|
||||
|
||||
const vectorCfg = getVectorConfig();
|
||||
const currentFingerprint = vectorCfg ? getEngineFingerprint(vectorCfg) : '';
|
||||
const fingerprintMismatch = manifest.fingerprint && currentFingerprint && manifest.fingerprint !== currentFingerprint;
|
||||
const chatIdMismatch = manifest.chatId !== chatId;
|
||||
|
||||
const warnings = [];
|
||||
if (fingerprintMismatch) {
|
||||
warnings.push(`向量引擎不匹配(文件: ${manifest.fingerprint}, 当前: ${currentFingerprint}),导入后需重新生成`);
|
||||
}
|
||||
if (chatIdMismatch) {
|
||||
warnings.push(`聊天ID不匹配(文件: ${manifest.chatId}, 当前: ${chatId})`);
|
||||
}
|
||||
|
||||
onProgress?.('解析数据...');
|
||||
|
||||
const chunksJsonl = unzipped['chunks.jsonl'] ? strFromU8(unzipped['chunks.jsonl']) : '';
|
||||
const chunkMetas = chunksJsonl.split('\n').filter(Boolean).map(line => JSON.parse(line));
|
||||
|
||||
const chunkVectorsBytes = unzipped['chunk_vectors.bin'];
|
||||
const chunkVectors = chunkVectorsBytes ? bytesToFloat32(chunkVectorsBytes, manifest.dims) : [];
|
||||
|
||||
const eventsJsonl = unzipped['events.jsonl'] ? strFromU8(unzipped['events.jsonl']) : '';
|
||||
const eventMetas = eventsJsonl.split('\n').filter(Boolean).map(line => JSON.parse(line));
|
||||
|
||||
const eventVectorsBytes = unzipped['event_vectors.bin'];
|
||||
const eventVectors = eventVectorsBytes ? bytesToFloat32(eventVectorsBytes, manifest.dims) : [];
|
||||
|
||||
const stateAtoms = unzipped['state_atoms.json']
|
||||
? JSON.parse(strFromU8(unzipped['state_atoms.json']))
|
||||
: [];
|
||||
|
||||
const stateVectorsJsonl = unzipped['state_vectors.jsonl'] ? strFromU8(unzipped['state_vectors.jsonl']) : '';
|
||||
const stateVectorMetas = stateVectorsJsonl.split('\n').filter(Boolean).map(line => JSON.parse(line));
|
||||
|
||||
const stateVectorsBytes = unzipped['state_vectors.bin'];
|
||||
const stateVectors = (stateVectorsBytes && stateVectorMetas.length)
|
||||
? bytesToFloat32(stateVectorsBytes, manifest.dims)
|
||||
: [];
|
||||
const stateRVectorsBytes = unzipped['state_r_vectors.bin'];
|
||||
const stateRVectors = (stateRVectorsBytes && stateVectorMetas.length)
|
||||
? bytesToFloat32(stateRVectorsBytes, manifest.rDims || manifest.dims)
|
||||
: [];
|
||||
const hasRVectorMeta = stateVectorMetas.some(m => typeof m.hasRVector === 'boolean');
|
||||
|
||||
if (chunkMetas.length !== chunkVectors.length) {
|
||||
throw new Error(`chunk 数量不匹配: 元数据 ${chunkMetas.length}, 向量 ${chunkVectors.length}`);
|
||||
}
|
||||
if (eventMetas.length !== eventVectors.length) {
|
||||
throw new Error(`event 数量不匹配: 元数据 ${eventMetas.length}, 向量 ${eventVectors.length}`);
|
||||
}
|
||||
if (stateVectorMetas.length !== stateVectors.length) {
|
||||
throw new Error(`state 向量数量不匹配: 元数据 ${stateVectorMetas.length}, 向量 ${stateVectors.length}`);
|
||||
}
|
||||
if (stateRVectors.length > 0 && stateVectorMetas.length !== stateRVectors.length) {
|
||||
throw new Error(`state r-vector count mismatch: meta=${stateVectorMetas.length}, vectors=${stateRVectors.length}`);
|
||||
}
|
||||
|
||||
onProgress?.('清空旧数据...');
|
||||
|
||||
await clearAllChunks(chatId);
|
||||
await clearEventVectors(chatId);
|
||||
await clearStateVectors(chatId);
|
||||
clearStateAtoms();
|
||||
|
||||
onProgress?.('写入数据...');
|
||||
|
||||
if (chunkMetas.length > 0) {
|
||||
const chunksToSave = chunkMetas.map(meta => ({
|
||||
chunkId: meta.chunkId,
|
||||
floor: meta.floor,
|
||||
chunkIdx: meta.chunkIdx,
|
||||
speaker: meta.speaker,
|
||||
isUser: meta.isUser,
|
||||
text: meta.text,
|
||||
textHash: meta.textHash,
|
||||
}));
|
||||
await saveChunks(chatId, chunksToSave);
|
||||
|
||||
const chunkVectorItems = chunkMetas.map((meta, idx) => ({
|
||||
chunkId: meta.chunkId,
|
||||
vector: chunkVectors[idx],
|
||||
}));
|
||||
await saveChunkVectors(chatId, chunkVectorItems, manifest.fingerprint);
|
||||
}
|
||||
|
||||
if (eventMetas.length > 0) {
|
||||
const eventVectorItems = eventMetas.map((meta, idx) => ({
|
||||
eventId: meta.eventId,
|
||||
vector: eventVectors[idx],
|
||||
}));
|
||||
await saveEventVectors(chatId, eventVectorItems, manifest.fingerprint);
|
||||
}
|
||||
|
||||
if (stateAtoms.length > 0) {
|
||||
saveStateAtoms(stateAtoms);
|
||||
}
|
||||
|
||||
if (stateVectorMetas.length > 0) {
|
||||
const stateVectorItems = stateVectorMetas.map((meta, idx) => ({
|
||||
atomId: meta.atomId,
|
||||
floor: meta.floor,
|
||||
vector: stateVectors[idx],
|
||||
rVector: (stateRVectors[idx] && (!hasRVectorMeta || meta.hasRVector)) ? stateRVectors[idx] : null,
|
||||
}));
|
||||
await saveStateVectors(chatId, stateVectorItems, manifest.fingerprint);
|
||||
}
|
||||
|
||||
await updateMeta(chatId, {
|
||||
fingerprint: manifest.fingerprint,
|
||||
lastChunkFloor: manifest.lastChunkFloor,
|
||||
});
|
||||
|
||||
xbLog.info(MODULE_ID, `从服务器恢复完成: ${chunkMetas.length} chunks, ${eventMetas.length} events, ${stateAtoms.length} state atoms`);
|
||||
|
||||
return {
|
||||
chunkCount: chunkMetas.length,
|
||||
eventCount: eventMetas.length,
|
||||
warnings,
|
||||
fingerprintMismatch,
|
||||
};
|
||||
}
|
||||
|
||||
// ═══════════════════════════════════════════════════════════════════════════
|
||||
// 备份清单管理
|
||||
// ═══════════════════════════════════════════════════════════════════════════
|
||||
|
||||
const BACKUP_MANIFEST = 'LWB_BackupManifest.json';
|
||||
|
||||
// 宽容解析:非数组/JSON 失败/字段异常时清洗,不抛错
|
||||
async function fetchManifest() {
|
||||
try {
|
||||
const res = await fetch(`/user/files/${BACKUP_MANIFEST}`, {
|
||||
headers: getRequestHeaders(),
|
||||
cache: 'no-cache',
|
||||
});
|
||||
if (!res.ok) return [];
|
||||
const raw = await res.json();
|
||||
if (!Array.isArray(raw)) return [];
|
||||
return raw.map(normalizeManifestEntry).filter(Boolean);
|
||||
} catch (_) {
|
||||
return [];
|
||||
}
|
||||
}
|
||||
|
||||
// 标准化单条条目字段,非法 filename 直接丢弃,其余字段降级
|
||||
function normalizeManifestEntry(raw) {
|
||||
if (!raw || typeof raw !== 'object') return null;
|
||||
const filename = typeof raw.filename === 'string' ? raw.filename : null;
|
||||
if (!filename || !/^LWB_VectorBackup_[a-z0-9]+\.zip$/.test(filename)) return null;
|
||||
const rawPath = typeof raw.serverPath === 'string' ? raw.serverPath.replace(/^\/+/, '') : null;
|
||||
return {
|
||||
filename,
|
||||
serverPath: rawPath,
|
||||
size: typeof raw.size === 'number' ? raw.size : null,
|
||||
chatId: typeof raw.chatId === 'string' ? raw.chatId : null,
|
||||
backupTime: typeof raw.backupTime === 'string' ? raw.backupTime : null,
|
||||
};
|
||||
}
|
||||
|
||||
// 安全推导/校验 serverPath:缺失时推导,与 filename 不一致时拒绝
|
||||
function buildSafeServerPath(filename, serverPath) {
|
||||
const expected = `user/files/${filename}`;
|
||||
if (!serverPath) return expected;
|
||||
const normalized = serverPath.replace(/^\/+/, '');
|
||||
if (normalized !== expected) {
|
||||
throw new Error(`serverPath 不安全: ${serverPath}`);
|
||||
}
|
||||
return normalized;
|
||||
}
|
||||
|
||||
// 读-改(upsert by filename)-写回-验证,失败最多重试 2 次
|
||||
async function upsertManifestEntry({ filename, serverPath, size, chatId, backupTime }) {
|
||||
if (typeof serverPath === 'string') serverPath = serverPath.replace(/^\/+/, '');
|
||||
const MAX_RETRIES = 3;
|
||||
for (let attempt = 0; attempt < MAX_RETRIES; attempt++) {
|
||||
// 读取现有清单
|
||||
const existing = await fetchManifest();
|
||||
|
||||
// upsert by filename
|
||||
const idx = existing.findIndex(e => e.filename === filename);
|
||||
const entry = { filename, serverPath, size, chatId, backupTime };
|
||||
if (idx >= 0) {
|
||||
existing[idx] = entry;
|
||||
} else {
|
||||
existing.push(entry);
|
||||
}
|
||||
|
||||
// 上传清单
|
||||
const json = JSON.stringify(existing, null, 2);
|
||||
const base64 = uint8ToBase64(new TextEncoder().encode(json));
|
||||
const res = await fetch('/api/files/upload', {
|
||||
method: 'POST',
|
||||
headers: getRequestHeaders(),
|
||||
body: JSON.stringify({ name: BACKUP_MANIFEST, data: base64 }),
|
||||
});
|
||||
if (!res.ok) throw new Error(`清单上传失败: ${res.status}`);
|
||||
|
||||
// 写后立即重读验证
|
||||
const verified = await fetchManifest();
|
||||
if (verified.some(e => e.filename === filename)) return;
|
||||
|
||||
// 最后一次仍失败才抛出
|
||||
if (attempt === MAX_RETRIES - 1) {
|
||||
throw new Error('清单写入后验证失败,重试已耗尽');
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// 删除前校验 + POST /api/files/delete + 更新清单
|
||||
async function deleteServerBackup(filename, serverPath) {
|
||||
// 安全校验
|
||||
if (!/^LWB_VectorBackup_[a-z0-9]+\.zip$/.test(filename)) {
|
||||
throw new Error(`非法文件名: ${filename}`);
|
||||
}
|
||||
const safePath = buildSafeServerPath(filename, serverPath || null);
|
||||
|
||||
// 物理删除
|
||||
const res = await fetch('/api/files/delete', {
|
||||
method: 'POST',
|
||||
headers: getRequestHeaders(),
|
||||
body: JSON.stringify({ path: safePath }),
|
||||
});
|
||||
if (!res.ok) {
|
||||
const err = new Error(`删除失败: ${res.status}`);
|
||||
err.status = res.status;
|
||||
err.method = 'DELETE';
|
||||
throw err;
|
||||
}
|
||||
|
||||
// 更新清单(删除条目)
|
||||
try {
|
||||
const existing = await fetchManifest();
|
||||
const filtered = existing.filter(e => e.filename !== filename);
|
||||
const json = JSON.stringify(filtered, null, 2);
|
||||
const base64 = uint8ToBase64(new TextEncoder().encode(json));
|
||||
const upRes = await fetch('/api/files/upload', {
|
||||
method: 'POST',
|
||||
headers: getRequestHeaders(),
|
||||
body: JSON.stringify({ name: BACKUP_MANIFEST, data: base64 }),
|
||||
});
|
||||
if (!upRes.ok) {
|
||||
throw new Error('zip 已删除,但清单更新失败,请手动刷新');
|
||||
}
|
||||
} catch (e) {
|
||||
// zip 删成功但清单更新失败 → 抛"部分成功"错误
|
||||
const partialErr = new Error(e.message || 'zip 已删除,清单同步失败');
|
||||
partialErr.partial = true;
|
||||
throw partialErr;
|
||||
}
|
||||
}
|
||||
|
||||
// 集中判断 404/405/method not allowed/unsupported
|
||||
function isDeleteUnsupportedError(err) {
|
||||
if (!err) return false;
|
||||
const status = err.status;
|
||||
if (status === 404 || status === 405) return true;
|
||||
const msg = String(err.message || '').toLowerCase();
|
||||
return msg.includes('method not allowed') || msg.includes('unsupported') || msg.includes('not found');
|
||||
}
|
||||
|
||||
export { fetchManifest, deleteServerBackup, isDeleteUnsupportedError, getBackupFilename };
|
||||
|
||||
Reference in New Issue
Block a user