1
0
mirror of https://github.com/ZeroCatDev/Classworks.git synced 2025-09-05 09:29:23 +00:00

Implement data migration tool and enhance KV storage support. Add automatic redirection for legacy data providers, improve user interface for migration settings, and update data handling for Classworks cloud storage. Refactor components for better integration with new storage options.

This commit is contained in:
SunWuyuan 2025-05-10 20:51:56 +08:00
parent c0e33dcbf9
commit a542e9b91f
No known key found for this signature in database
GPG Key ID: A6A54CF66F56BB64
11 changed files with 355 additions and 204 deletions

View File

@ -10,17 +10,51 @@
</template> </template>
<script setup> <script setup>
import { onMounted } from 'vue'; import { onMounted, watch } from 'vue';
import { useTheme } from 'vuetify'; import { useTheme } from 'vuetify';
import { getSetting } from '@/utils/settings'; import { getSetting } from '@/utils/settings';
import { useRouter, useRoute } from 'vue-router';
const theme = useTheme(); const theme = useTheme();
const router = useRouter();
const route = useRoute();
onMounted(() => { onMounted(() => {
// //
const savedTheme = getSetting('theme.mode'); const savedTheme = getSetting('theme.mode');
theme.global.name.value = savedTheme; theme.global.name.value = savedTheme;
//
checkProviderType();
}); });
//
function checkProviderType() {
const currentProvider = getSetting('server.provider');
//
if ((currentProvider === 'server' || currentProvider === 'indexedDB') &&
route.path !== '/datamigration') {
console.log('检测到旧的数据提供者类型,正在重定向到数据迁移页面...');
router.push({
path: '/datamigration',
query: {
reason: 'legacy_provider',
provider: currentProvider
}
});
}
}
//
watch(
() => route.path,
(newPath) => {
if (newPath !== '/datamigration') {
checkProviderType();
}
}
);
</script> </script>
<style> <style>

View File

@ -82,14 +82,14 @@
<!-- 统一的数据显示卡片 --> <!-- 统一的数据显示卡片 -->
<v-card class="mb-6"> <v-card class="mb-6">
<v-card-title class="d-flex align-center"> <v-card-title class="d-flex align-center">
<span>{{ migrationType === 'local' ? '本地数据库内容' : '服务器数据预览' }}</span> <span>{{ migrationType === 'local' ? '本地数据库内容' : '服务器数据内容' }}</span>
<v-spacer></v-spacer> <v-spacer></v-spacer>
<v-btn <v-btn
color="primary" color="primary"
@click="migrationType === 'local' ? scanLocalDatabase() : previewServerData()" @click="migrationType === 'local' ? scanLocalDatabase() : previewServerData()"
:loading="loading || scanning" :loading="loading || scanning"
> >
{{ migrationType === 'local' ? '扫描数据' : '预览数据' }} {{ migrationType === 'local' ? '扫描数据' : '加载数据' }}
</v-btn> </v-btn>
</v-card-title> </v-card-title>
<v-card-text> <v-card-text>
@ -99,7 +99,7 @@
> >
{{ migrationType === 'local' {{ migrationType === 'local'
? '尚未扫描本地数据或未找到可迁移的数据。点击"扫描数据"按钮开始扫描。' ? '尚未扫描本地数据或未找到可迁移的数据。点击"扫描数据"按钮开始扫描。'
: '尚未预览服务器数据或未找到可迁移的数据。点击"预览数据"按钮开始查询。' : '尚未预览服务器数据或未找到可迁移的数据。点击"加载数据"按钮开始查询。'
}} }}
</v-alert> </v-alert>
@ -478,7 +478,8 @@ export default {
const res = await axios.get(homeworkUrl, { const res = await axios.get(homeworkUrl, {
headers: this.getRequestHeaders() headers: this.getRequestHeaders()
}); });
if (res.data) { if (res.data&&res.data.status!=false) {
console.log(res.data);
this.serverItems.push({ this.serverItems.push({
type: 'homework', type: 'homework',
key: `homework_${this.classNumber}_${dateStr}`, key: `homework_${this.classNumber}_${dateStr}`,
@ -624,7 +625,7 @@ export default {
if (itemType === 'config') { if (itemType === 'config') {
// : classNumber/classworks-config // : classNumber/classworks-config
await db.put("kv", JSON.stringify(value), `${this.classNumber}/classworks-config`); await db.put("kv", JSON.stringify(value), `classworks-config`);
return { success: true, message: '配置已迁移' }; return { success: true, message: '配置已迁移' };
} else { } else {
// : classNumber/classworks-data-YYYYMMDD // : classNumber/classworks-data-YYYYMMDD
@ -644,7 +645,7 @@ export default {
} }
} }
await db.put("kv", JSON.stringify(value), `${this.classNumber}/classworks-data-${dateStr}`); await db.put("kv", JSON.stringify(value), `classworks-data-${dateStr}`);
return { success: true, message: `${dateStr} 数据已迁移` }; return { success: true, message: `${dateStr} 数据已迁移` };
} }
} catch (error) { } catch (error) {

View File

@ -169,10 +169,9 @@ export default {
'dark': '深色' 'dark': '深色'
}, },
'server.provider': { 'server.provider': {
'server': '远程服务器',
'indexedDB': '本地存储',
'kv-local': 'KV本地存储', 'kv-local': 'KV本地存储',
'kv-server': 'KV远程服务器' 'kv-server': 'KV远程服务器',
'classworkscloud': 'Classworks云端存储'
} }
}, },
// //

View File

@ -2,7 +2,7 @@
<settings-card title="数据源设置" icon="mdi-database-cog"> <settings-card title="数据源设置" icon="mdi-database-cog">
<v-list> <v-list>
<!-- 服务器模式设置 --> <!-- 服务器模式设置 -->
<template v-if="currentProvider === 'server' || currentProvider === 'kv-server'"> <template v-if="currentProvider === 'kv-server' || currentProvider === 'classworkscloud'">
<v-list-item> <v-list-item>
<template #prepend> <template #prepend>
<v-icon icon="mdi-lan-connect" class="mr-3" /> <v-icon icon="mdi-lan-connect" class="mr-3" />
@ -13,17 +13,29 @@
测试连接 测试连接
</v-btn> </v-btn>
</template> </template>
</v-list-item><!-- 数据迁移仅对KV本地存储有效 -->
<v-list-item>
<template #prepend>
<v-icon icon="mdi-database-import" class="mr-3" />
</template>
<v-list-item-title>迁移旧数据</v-list-item-title>
<v-list-item-subtitle>将旧的存储格式数据转移到新的KV存储</v-list-item-subtitle>
<template #append>
<v-btn :loading="migrateLoading" variant="tonal" @click="migrateData">
迁移
</v-btn>
</template>
</v-list-item> </v-list-item>
</template> </template>
<!-- IndexedDB设置 --> <!-- 本地存储设置 -->
<template v-if="currentProvider === 'indexedDB' || currentProvider === 'kv-local'"> <template v-if="currentProvider === 'kv-local'">
<v-list-item> <v-list-item>
<template #prepend> <template #prepend>
<v-icon icon="mdi-database" class="mr-3" /> <v-icon icon="mdi-database" class="mr-3" />
</template> </template>
<v-list-item-title>清除数据库缓存</v-list-item-title> <v-list-item-title>清除数据库缓存</v-list-item-title>
<v-list-item-subtitle>这将清除所有IndexedDB中的数据</v-list-item-subtitle> <v-list-item-subtitle>这将清除所有本地数据库中的数据</v-list-item-subtitle>
<template #append> <template #append>
<v-btn color="error" variant="tonal" @click="confirmClearIndexedDB"> <v-btn color="error" variant="tonal" @click="confirmClearIndexedDB">
清除 清除
@ -40,8 +52,8 @@
</template> </template>
</v-list-item> </v-list-item>
<!-- 显示机器ID仅对KV本地存储有效 --> <!-- 显示机器ID -->
<v-list-item v-if="currentProvider === 'kv-local'"> <v-list-item>
<template #prepend> <template #prepend>
<v-icon icon="mdi-identifier" class="mr-3" /> <v-icon icon="mdi-identifier" class="mr-3" />
</template> </template>
@ -76,31 +88,6 @@
</v-btn> </v-btn>
</template> </template>
</v-list-item> </v-list-item>
<v-list-item>
<template #prepend>
<v-icon icon="mdi-database-sync" class="mr-3" />
</template>
<v-list-item-title>高级数据迁移工具</v-list-item-title>
<v-list-item-subtitle>更强大的数据迁移工具支持从本地或服务器迁移到KV存储</v-list-item-subtitle>
<template #append>
<v-btn variant="tonal" color="primary" to="/datamigration">
打开
</v-btn>
</template>
</v-list-item>
<v-list-item>
<template #prepend>
<v-icon icon="mdi-lan-connect" class="mr-3" />
</template>
<v-list-item-title>CSES转WakeUP工具</v-list-item-title>
<template #append>
<v-btn variant="tonal" to="/cses2wakeup">
查看
</v-btn>
</template>
</v-list-item>
</v-list> </v-list>
<!-- 确认对话框 --> <!-- 确认对话框 -->
@ -122,7 +109,6 @@
import SettingsCard from "@/components/SettingsCard.vue"; import SettingsCard from "@/components/SettingsCard.vue";
import { getSetting } from "@/utils/settings"; import { getSetting } from "@/utils/settings";
import axios from "axios"; import axios from "axios";
import { getMachineId } from "@/utils/providers/kvProvider";
export default { export default {
name: "DataProviderSettingsCard", name: "DataProviderSettingsCard",
@ -147,7 +133,7 @@ export default {
}, },
isKvProvider() { isKvProvider() {
return this.currentProvider === 'kv-local' || this.currentProvider === 'kv-server'; return this.currentProvider === 'kv-local' || this.currentProvider === 'kv-server' || this.currentProvider === 'classworkscloud';
} }
}, },
@ -155,7 +141,7 @@ export default {
// KVID // KVID
if (this.currentProvider === 'kv-local') { if (this.currentProvider === 'kv-local') {
try { try {
this.machineId = await getMachineId(); this.machineId = getSetting('device.uuid');
} catch (error) { } catch (error) {
console.error("获取机器ID失败:", error); console.error("获取机器ID失败:", error);
} }

View File

@ -12,12 +12,18 @@
</p> </p>
</v-alert> </v-alert>
<v-alert v-if="isClassworksCloud" type="info" color="success" variant="tonal" class="my-2">
<v-alert-title>Classworks云端存储</v-alert-title>
<p>Classworks云端存储是官方提供的存储解决方案自动配置了最优的访问设置</p>
<p>使用此选项时服务器域名和网站令牌将自动配置无需手动设置</p>
</v-alert>
<v-divider class="my-2" /> <v-divider class="my-2" />
<setting-item setting-key="server.domain" title="服务器域名" /> <setting-item setting-key="server.domain" title="服务器域名" :disabled="isClassworksCloud" />
<v-divider class="my-2" /> <v-divider class="my-2" />
<setting-item setting-key="server.classNumber" title="班号" /> <setting-item setting-key="server.classNumber" title="班号" />
<v-divider class="my-2" /> <v-divider class="my-2" />
<setting-item setting-key="server.siteKey" title="网站令牌"> <setting-item setting-key="server.siteKey" title="网站令牌" :disabled="isClassworksCloud">
<template #description> <template #description>
用于后端验证请求的安全令牌如需要请从系统管理员获取 用于后端验证请求的安全令牌如需要请从系统管理员获取
</template> </template>
@ -53,8 +59,11 @@ export default {
isKvProvider() { isKvProvider() {
return this.currentProvider === 'kv-local' || this.currentProvider === 'kv-server'; return this.currentProvider === 'kv-local' || this.currentProvider === 'kv-server';
}, },
isClassworksCloud() {
return this.currentProvider === 'classworkscloud';
},
useServer() { useServer() {
return this.currentProvider === 'server' || this.currentProvider === 'kv-server'; return this.currentProvider === 'server' || this.currentProvider === 'kv-server' || this.currentProvider === 'classworkscloud';
} }
} }
}; };

View File

@ -3,36 +3,205 @@
<v-row> <v-row>
<v-col cols="12"> <v-col cols="12">
<div class="d-flex align-center mb-6"> <div class="d-flex align-center mb-6">
<v-icon size="x-large" color="primary" class="mr-3">mdi-database-sync</v-icon> <v-icon size="x-large" color="primary" class="mr-3"
>mdi-database-sync</v-icon
>
<div> <div>
<h1 class="text-h4">数据迁移工具</h1> <h1 class="text-h4">数据迁移工具</h1>
<div class="text-subtitle-1 text-grey">将现有数据迁移至 KV 存储系统</div> <div class="text-subtitle-1 text-grey">
将现有数据迁移至 KV 存储系统
</div>
</div> </div>
</div> </div>
<v-card class="mb-6" variant="tonal" color="info" density="compact"> <v-card class="mb-6" variant="tonal" color="info" density="compact">
<v-card-text class="d-flex align-center"> <v-card-text class="d-flex align-center">
<v-icon color="info" class="mr-2">mdi-information-outline</v-icon> <v-icon color="info" class="mr-2">mdi-information-outline</v-icon>
<span>使用此工具可以将数据从旧存储系统迁移到新的 KV 存储系统选择本地或云端迁移以确保数据不会丢失</span> <span
>使用此工具可以将数据从旧存储系统迁移到新的 KV
存储系统选择本地或云端迁移以确保数据不会丢失</span
>
</v-card-text> </v-card-text>
</v-card> </v-card>
<MigrationTool /> <MigrationTool ref="migrationTool" />
</v-col> </v-col>
</v-row> </v-row>
<!-- 一键迁移对话框 -->
<v-dialog v-model="showMigrationDialog" max-width="500" persistent>
<v-card>
<v-card-title class="text-h5 d-flex align-center">
<v-icon color="primary" size="large" class="mr-3"
>mdi-database-sync</v-icon
>
一键数据迁移
</v-card-title>
<v-card-text class="mt-4">
<p>
系统将自动读取您的配置并将过去半年的数据迁移至Classworks
KV数据库中
</p>
<v-alert
color="info"
variant="outlined"
density="compact"
class="mt-4"
icon="mdi-information-outline"
>
<ul class="ml-3 mt-1">
<li>数据源: {{ dataSourceText }}</li>
<li>班级: {{ classNumber }}</li>
<li>服务器: {{ serverDomain || "本地存储" }}</li>
<li>
迁移范围: {{ formatDate(sixMonthsAgo) }}
{{ formatDate(today) }}
</li>
</ul>
</v-alert>
</v-card-text>
<v-card-actions>
<v-spacer></v-spacer>
<v-btn
color="grey-darken-1"
variant="text"
@click="showMigrationDialog = false"
>
稍后再说
</v-btn>
<v-btn
color="primary"
size="large"
variant="elevated"
@click="startAutoMigration"
:loading="isAutoMigrating"
:disabled="isAutoMigrating"
>
<v-icon left class="mr-2">mdi-database-export</v-icon>
开始一键迁移
</v-btn>
</v-card-actions>
</v-card>
</v-dialog>
</v-container> </v-container>
</template> </template>
<script> <script>
import MigrationTool from '@/components/MigrationTool.vue'; import MigrationTool from "@/components/MigrationTool.vue";
import { getSetting, setSetting } from "@/utils/settings";
export default { export default {
name: 'DataMigrationPage', name: "DataMigrationPage",
components: { components: {
MigrationTool MigrationTool,
},
data() {
const today = new Date();
const sixMonthsAgo = new Date();
sixMonthsAgo.setMonth(today.getMonth() - 3);
return {
showMigrationDialog: false,
isAutoMigrating: false,
today,
sixMonthsAgo,
classNumber: "",
serverDomain: "",
dataProvider: "",
};
},
computed: {
dataSourceText() {
switch (this.dataProvider) {
case "server":
return "服务器";
case "indexeddb":
return "本地数据库";
case "kv-local":
return "本地 KV 存储";
case "kv-server":
return "远程 KV 存储";
case "classworkscloud":
return "Classworks 云";
default:
return "未知来源";
}
},
},
async mounted() {
this.loadSettings();
if (this.serverDomain == "https://class.wuyuan.dev") {
await this.startAutoMigration();
this.$router.push("/");
}
},
methods: {
loadSettings() {
this.classNumber = getSetting("server.classNumber");
this.serverDomain = getSetting("server.domain");
this.dataProvider = getSetting("server.provider");
this.showMigrationDialog =
this.dataProvider === "server" || this.dataProvider === "indexeddb";
},
formatDate(date) {
return date.toLocaleDateString();
},
async startAutoMigration() {
if (!this.$refs.migrationTool) {
console.error("MigrationTool组件引用不可用");
return;
}
this.isAutoMigrating = true;
try {
//
const migrationTool = this.$refs.migrationTool;
migrationTool.classNumber = this.classNumber;
migrationTool.migrationType =
this.dataProvider === "server" ? "server" : "local";
migrationTool.serverUrl = this.serverDomain;
migrationTool.targetStorage = "kv-server";
//migrationTool.targetServerUrl = this.serverDomain;
//
migrationTool.startDate = this.formatDateString(this.sixMonthsAgo);
migrationTool.endDate = this.formatDateString(this.today);
//
if (this.dataProvider === "server") {
//
await migrationTool.previewServerData();
} else {
//
await migrationTool.scanLocalDatabase();
}
//
if (migrationTool.displayItems.length > 0) {
await migrationTool.startMigration();
} else {
console.warn("没有找到可迁移的数据");
}
setSetting("server.provider", "classworkscloud");
} catch (error) {
console.error("自动迁移失败:", error);
} finally {
this.isAutoMigrating = false;
this.showMigrationDialog = false;
}
},
formatDateString(date) {
const year = date.getFullYear();
const month = String(date.getMonth() + 1).padStart(2, "0");
const day = String(date.getDate()).padStart(2, "0");
return `${year}-${month}-${day}`;
},
}, },
metaInfo: { metaInfo: {
title: '数据迁移工具' title: "数据迁移工具",
} },
} };
</script> </script>

View File

@ -625,6 +625,7 @@
import MessageLog from "@/components/MessageLog.vue"; import MessageLog from "@/components/MessageLog.vue";
import RandomPicker from "@/components/RandomPicker.vue"; // import RandomPicker from "@/components/RandomPicker.vue"; //
import dataProvider from "@/utils/dataProvider"; import dataProvider from "@/utils/dataProvider";
import { kvProvider } from "@/utils/providers/kvProvider";
import { import {
getSetting, getSetting,
watchSettings, watchSettings,
@ -1086,7 +1087,6 @@ export default {
try { try {
this.loading.download = true; this.loading.download = true;
const response = await dataProvider.loadData( const response = await dataProvider.loadData(
this.provider,
this.dataKey, this.dataKey,
this.state.dateString this.state.dateString
); );
@ -1192,10 +1192,9 @@ export default {
try { try {
this.loading.upload = true; this.loading.upload = true;
const response = await dataProvider.saveData( const response = await dataProvider.saveData(
this.provider,
this.dataKey, this.dataKey,
this.state.boardData, this.state.boardData,
this.state.dateString // dateString this.state.dateString
); );
if (!response.success) { if (!response.success) {
@ -1211,10 +1210,16 @@ export default {
async loadConfig() { async loadConfig() {
try { try {
const response = await dataProvider.loadConfig( // 使kvProvider
this.provider, const provider = getSetting("server.provider");
this.dataKey const useServer = provider === "kv-server" || provider === "classworkscloud";
); let response;
if (useServer) {
response = await kvProvider.server.loadConfig();
} else {
response = await kvProvider.local.loadConfig();
}
if (!response.success) { if (!response.success) {
throw new Error(response.error.message); throw new Error(response.error.message);
@ -1325,7 +1330,7 @@ export default {
const classNum = getSetting("server.classNumber"); const classNum = getSetting("server.classNumber");
this.provider = provider; this.provider = provider;
this.dataKey = provider === "server" ? `${domain}/${classNum}` : classNum; this.dataKey = provider === "server" || provider === "classworkscloud" ? `${domain}/${classNum}` : classNum;
this.state.classNumber = classNum; this.state.classNumber = classNum;
}, },

View File

@ -127,7 +127,7 @@
:is-mobile="isMobile" :is-mobile="isMobile"
:unsaved-changes="hasUnsavedChanges" :unsaved-changes="hasUnsavedChanges"
@save="saveStudents" @save="saveStudents"
@reload="loadStudentList" @reload="loadStudents"
@update:modelValue="handleStudentDataChange" @update:modelValue="handleStudentDataChange"
/> />
</v-col> </v-col>
@ -187,7 +187,7 @@ import SettingsCard from '@/components/SettingsCard.vue';
import StudentListCard from '@/components/settings/StudentListCard.vue'; import StudentListCard from '@/components/settings/StudentListCard.vue';
import AboutCard from '@/components/settings/AboutCard.vue'; import AboutCard from '@/components/settings/AboutCard.vue';
import '../styles/settings.scss'; import '../styles/settings.scss';
import dataProvider from '@/utils/dataProvider'; import { kvProvider } from '@/utils/providers/kvProvider';
import SettingsExplorer from '@/components/settings/SettingsExplorer.vue'; import SettingsExplorer from '@/components/settings/SettingsExplorer.vue';
import SettingsLinkGenerator from '@/components/SettingsLinkGenerator.vue'; import SettingsLinkGenerator from '@/components/SettingsLinkGenerator.vue';
export default { export default {
@ -306,7 +306,7 @@ export default {
this.unwatchSettings = watchSettings(() => { this.unwatchSettings = watchSettings(() => {
this.loadAllSettings(); this.loadAllSettings();
}); });
this.loadStudentList(); this.loadStudents();
this.refreshDebugConfig(); this.refreshDebugConfig();
// , // ,
@ -366,30 +366,34 @@ export default {
this.$message.error(title, content); this.$message.error(title, content);
}, },
async loadStudentList() { async loadStudents() {
this.studentsError = null;
try { try {
this.loading.students = true; this.loading.students = true;
this.studentsError = null;
const domain = getSetting('server.domain');
const classNum = getSetting('server.classNumber'); const classNum = getSetting('server.classNumber');
const provider = getSetting('server.provider');
if (!classNum) { if (!classNum) {
throw new Error('请先设置班号'); throw new Error('请先设置班号');
} }
const key = provider === 'server' ? `${domain}/${classNum}` : classNum; const provider = getSetting('server.provider');
const res = await dataProvider.loadConfig(provider, key); const useServer = provider === 'kv-server' || provider === 'classworkscloud';
let response;
if (!res.success) { if (useServer) {
throw new Error(res.error.message); response = await kvProvider.server.loadConfig();
} else {
response = await kvProvider.local.loadConfig();
} }
if (res.data && Array.isArray(res.data.studentList)) { if (!response.success) {
this.studentData.list = res.data.studentList; throw new Error(response.error.message);
this.studentData.text = res.data.studentList.join('\n'); }
this.lastSavedData = [...res.data.studentList];
if (response.data && Array.isArray(response.data.studentList)) {
this.studentData.list = response.data.studentList;
this.studentData.text = response.data.studentList.join('\n');
this.lastSavedData = [...response.data.studentList];
this.hasUnsavedChanges = false; this.hasUnsavedChanges = false;
} }
} catch (error) { } catch (error) {
@ -403,21 +407,28 @@ export default {
async saveStudents() { async saveStudents() {
try { try {
const domain = getSetting('server.domain');
const classNum = getSetting('server.classNumber'); const classNum = getSetting('server.classNumber');
const provider = getSetting('server.provider');
if (!classNum) { if (!classNum) {
throw new Error('请先设置班号'); throw new Error('请先设置班号');
} }
const key = provider === 'server' ? `${domain}/${classNum}` : classNum; const provider = getSetting('server.provider');
const res = await dataProvider.saveConfig(provider, key, { const useServer = provider === 'kv-server' || provider === 'classworkscloud';
studentList: this.studentData.list, let response;
});
if (!res.success) { if (useServer) {
throw new Error(res.error.message); response = await kvProvider.server.saveConfig({
studentList: this.studentData.list,
});
} else {
response = await kvProvider.local.saveConfig({
studentList: this.studentData.list,
});
}
if (!response.success) {
throw new Error(response.error.message);
} }
// //

View File

@ -1,5 +1,3 @@
import { serverProvider } from "./providers/server";
import { indexedDBProvider } from "./providers/indexedDB";
import { kvProvider } from "./providers/kvProvider"; import { kvProvider } from "./providers/kvProvider";
import { getSetting } from "./settings"; import { getSetting } from "./settings";
@ -14,84 +12,28 @@ export const formatError = (message, code = "UNKNOWN_ERROR") => ({
error: { code, message }, error: { code, message },
}); });
// Legacy providers // Main data provider with simplified API
const legacyProviders = {
server: serverProvider,
indexedDB: indexedDBProvider,
};
// New KV provider
const newProviders = {
kv: kvProvider,
};
// Main data provider with support for both legacy and new API
export default { export default {
// Provider API methods // Provider API methods
loadData: (provider, key, date) => { loadData: async (key, date) => {
if (legacyProviders[provider]) { const provider = getSetting("server.provider");
return legacyProviders[provider]?.loadData(key, date); const useServer = provider === "kv-server" || provider === "classworkscloud";
}
// If using new KV provider if (useServer) {
if (provider === "kv-local") { return kvProvider.server.loadData(key, date);
const classNumber = key.split("/").pop(); } else {
return newProviders.kv.local.loadData(classNumber, date); return kvProvider.local.loadData(date);
} else if (provider === "kv-server") {
const classNumber = key.split("/").pop();
const serverUrl = getSetting("server.domain");
return newProviders.kv.server.loadData(serverUrl, classNumber, date);
} }
}, },
saveData: (provider, key, data, date) => { saveData: async (key, data, date) => {
if (legacyProviders[provider]) { const provider = getSetting("server.provider");
return legacyProviders[provider]?.saveData(key, data, date); const useServer = provider === "kv-server" || provider === "classworkscloud";
}
// If using new KV provider if (useServer) {
if (provider === "kv-local") { return kvProvider.server.saveData(key, data, date);
const classNumber = key.split("/").pop(); } else {
return newProviders.kv.local.saveData(classNumber, data, date); return kvProvider.local.saveData(data, date);
} else if (provider === "kv-server") {
const classNumber = key.split("/").pop();
const serverUrl = getSetting("server.domain");
return newProviders.kv.server.saveData(
serverUrl,
classNumber,
data,
date
);
}
},
loadConfig: (provider, key) => {
if (legacyProviders[provider]) {
return legacyProviders[provider]?.loadConfig(key);
}
// If using new KV provider
if (provider === "kv-local") {
const classNumber = key.split("/").pop();
return newProviders.kv.local.loadConfig(classNumber);
} else if (provider === "kv-server") {
const serverUrl = getSetting("server.domain");
return newProviders.kv.server.loadConfig(serverUrl);
}
},
saveConfig: (provider, key, config) => {
if (legacyProviders[provider]) {
return legacyProviders[provider]?.saveConfig(key, config);
}
// If using new KV provider
if (provider === "kv-local") {
const classNumber = key.split("/").pop();
return newProviders.kv.local.saveConfig(classNumber, config);
} else if (provider === "kv-server") {
const serverUrl = getSetting("server.domain");
return newProviders.kv.server.saveConfig(serverUrl, config);
} }
}, },
}; };

View File

@ -23,11 +23,6 @@ const getHeaders = () => {
return headers; return headers;
}; };
// Get machine UUID from settings
const getMachineId = () => {
return getSetting("device.uuid");
};
// Removed migrateToKvStorage function - now handled by the dedicated migration tool // Removed migrateToKvStorage function - now handled by the dedicated migration tool
const initDB = async () => { const initDB = async () => {
@ -58,17 +53,13 @@ const formatDateForKey = (date) => {
export const kvProvider = { export const kvProvider = {
// Local storage provider // Local storage provider
local: { local: {
async loadData(classNumber, date) { async loadData(date) {
try { try {
if (!classNumber) {
return formatError("请先设置班号", "CONFIG_ERROR");
}
const formattedDate = formatDateForKey(date); const formattedDate = formatDateForKey(date);
const key = `${DATA_KEY_PREFIX}${formattedDate}`; const key = `${DATA_KEY_PREFIX}${formattedDate}`;
const db = await initDB(); const db = await initDB();
const data = await db.get("kv", `${classNumber}/${key}`); const data = await db.get("kv", key);
if (!data) { if (!data) {
const today = new Date().toISOString().split("T")[0]; const today = new Date().toISOString().split("T")[0];
@ -91,31 +82,23 @@ export const kvProvider = {
} }
}, },
async saveData(classNumber, data, date) { async saveData(data, date) {
try { try {
if (!classNumber) {
return formatError("请先设置班号", "CONFIG_ERROR");
}
const formattedDate = formatDateForKey(date); const formattedDate = formatDateForKey(date);
const key = `${DATA_KEY_PREFIX}${formattedDate}`; const key = `${DATA_KEY_PREFIX}${formattedDate}`;
const db = await initDB(); const db = await initDB();
await db.put("kv", JSON.stringify(data), `${classNumber}/${key}`); await db.put("kv", JSON.stringify(data), key);
return formatResponse(null, "保存成功"); return formatResponse(null, "保存成功");
} catch (error) { } catch (error) {
return formatError("保存本地数据失败:" + error); return formatError("保存本地数据失败:" + error);
} }
}, },
async loadConfig(classNumber) { async loadConfig() {
try { try {
if (!classNumber) {
return formatError("请先设置班号", "CONFIG_ERROR");
}
const db = await initDB(); const db = await initDB();
const config = await db.get("kv", `${classNumber}/${CONFIG_KEY}`); const config = await db.get("kv", CONFIG_KEY);
if (!config) { if (!config) {
return formatResponse({ return formatResponse({
@ -134,14 +117,10 @@ export const kvProvider = {
} }
}, },
async saveConfig(classNumber, config) { async saveConfig(config) {
try { try {
if (!classNumber) {
return formatError("请先设置班号", "CONFIG_ERROR");
}
const db = await initDB(); const db = await initDB();
await db.put("kv", JSON.stringify(config), `${classNumber}/${CONFIG_KEY}`); await db.put("kv", JSON.stringify(config), CONFIG_KEY);
return formatResponse(null, "保存成功"); return formatResponse(null, "保存成功");
} catch (error) { } catch (error) {
return formatError("保存本地配置失败:" + error); return formatError("保存本地配置失败:" + error);
@ -151,9 +130,10 @@ export const kvProvider = {
// Server storage provider // Server storage provider
server: { server: {
async loadData(serverUrl, classNumber, date) { async loadData(classNumber, date) {
try { try {
const machineId = getMachineId(); const serverUrl = getSetting("server.domain");
const machineId = getSetting("device.uuid");
const formattedDate = formatDateForKey(date); const formattedDate = formatDateForKey(date);
const key = `${DATA_KEY_PREFIX}${formattedDate}`; const key = `${DATA_KEY_PREFIX}${formattedDate}`;
@ -185,9 +165,10 @@ export const kvProvider = {
} }
}, },
async saveData(serverUrl, classNumber, data, date) { async saveData(classNumber, data, date) {
try { try {
const machineId = getMachineId(); const serverUrl = getSetting("server.domain");
const machineId = getSetting("device.uuid");
const formattedDate = formatDateForKey(date); const formattedDate = formatDateForKey(date);
const key = `${DATA_KEY_PREFIX}${formattedDate}`; const key = `${DATA_KEY_PREFIX}${formattedDate}`;
@ -203,9 +184,10 @@ export const kvProvider = {
} }
}, },
async loadConfig(serverUrl) { async loadConfig() {
try { try {
const machineId = getMachineId(); const serverUrl = getSetting("server.domain");
const machineId = getSetting("device.uuid");
const res = await axios.get(`${serverUrl}/${machineId}/${CONFIG_KEY}`, { const res = await axios.get(`${serverUrl}/${machineId}/${CONFIG_KEY}`, {
headers: getHeaders() headers: getHeaders()
}); });
@ -230,9 +212,10 @@ export const kvProvider = {
} }
}, },
async saveConfig(serverUrl, config) { async saveConfig(config) {
try { try {
const machineId = getMachineId(); const serverUrl = getSetting("server.domain");
const machineId = getSetting("device.uuid");
await axios.post(`${serverUrl}/${machineId}/${CONFIG_KEY}`, config, { await axios.post(`${serverUrl}/${machineId}/${CONFIG_KEY}`, config, {
headers: getHeaders() headers: getHeaders()
}); });
@ -247,4 +230,3 @@ export const kvProvider = {
} }
}; };
export { getMachineId };

View File

@ -74,6 +74,12 @@ function generateUUID() {
}); });
} }
// 新增: Classworks云端存储的默认设置
const classworksCloudDefaults = {
"server.domain": "http://localhost:3030",
"server.siteKey": "123456",
};
/** /**
* 所有配置项的定义 * 所有配置项的定义
* @type {Object.<string, SettingDefinition>} * @type {Object.<string, SettingDefinition>}
@ -182,11 +188,11 @@ const settingsDefinitions = {
}, },
"server.provider": { "server.provider": {
type: "string", type: "string",
default: "indexedDB", default: "kv-local",
validate: (value) => ["server", "indexedDB", "kv-local", "kv-server"].includes(value), validate: (value) => ["kv-local", "kv-server", "classworkscloud"].includes(value),
description: "数据提供者", description: "数据提供者",
icon: "mdi-database", icon: "mdi-database",
// 选择数据存储方式:使用本地IndexedDB或远程服务器 // 选择数据存储方式:使用本地存储或远程服务器
}, },
// 刷新设置 // 刷新设置
@ -480,6 +486,13 @@ class SettingsManagerClass {
} }
} }
// 检查是否使用Classworks云端存储并覆盖特定设置
if (this.settingsCache["server.provider"] === "classworkscloud") {
if (classworksCloudDefaults[key] !== undefined) {
return classworksCloudDefaults[key];
}
}
const value = this.settingsCache[key]; const value = this.settingsCache[key];
return value !== undefined ? value : definition.default; return value !== undefined ? value : definition.default;
} }