主要改动: - 重构目录结构:合并子目录到根目录,简化项目结构 - 添加完整的查询 API:支持复杂条件查询、字段选择、游标模式 - 实现 LSM-Tree Compaction:7层结构、Score-based策略、后台异步合并 - 添加 Web UI:基于 Lit 的现代化管理界面,支持数据浏览和 Manifest 查看 - 完善文档:添加 README.md 和 examples/webui/README.md 新增功能: - Query Builder:链式查询 API,支持 Eq/Lt/Gt/In/Between/Contains 等操作符 - Web UI 组件:srdb-app、srdb-table-list、srdb-data-view、srdb-manifest-view 等 - 列选择持久化:自动保存到 localStorage - 刷新按钮:一键刷新当前视图 - 主题切换:深色/浅色主题支持 代码优化: - 使用 Go 1.24 新特性:range 7、min()、maps.Copy()、slices.Sort() - 统一组件命名:所有 Web Components 使用 srdb-* 前缀 - CSS 优化:提取共享样式,减少重复代码 - 清理遗留代码:删除未使用的方法和样式
221 lines
5.0 KiB
Go
221 lines
5.0 KiB
Go
package srdb
|
|
|
|
import (
|
|
"os"
|
|
"testing"
|
|
)
|
|
|
|
func TestVersionSetBasic(t *testing.T) {
|
|
dir := "./test_manifest"
|
|
os.RemoveAll(dir)
|
|
defer os.RemoveAll(dir)
|
|
|
|
// 创建 VersionSet
|
|
vs, err := NewVersionSet(dir)
|
|
if err != nil {
|
|
t.Fatalf("NewVersionSet failed: %v", err)
|
|
}
|
|
defer vs.Close()
|
|
|
|
// 检查初始状态
|
|
version := vs.GetCurrent()
|
|
if version.GetFileCount() != 0 {
|
|
t.Errorf("Expected 0 files, got %d", version.GetFileCount())
|
|
}
|
|
|
|
t.Log("VersionSet basic test passed!")
|
|
}
|
|
|
|
func TestVersionSetAddFile(t *testing.T) {
|
|
dir := "./test_manifest_add"
|
|
os.RemoveAll(dir)
|
|
defer os.RemoveAll(dir)
|
|
|
|
vs, err := NewVersionSet(dir)
|
|
if err != nil {
|
|
t.Fatalf("NewVersionSet failed: %v", err)
|
|
}
|
|
defer vs.Close()
|
|
|
|
// 添加文件
|
|
edit := NewVersionEdit()
|
|
edit.AddFile(&FileMetadata{
|
|
FileNumber: 1,
|
|
FileSize: 1024,
|
|
MinKey: 1,
|
|
MaxKey: 100,
|
|
RowCount: 100,
|
|
})
|
|
|
|
err = vs.LogAndApply(edit)
|
|
if err != nil {
|
|
t.Fatalf("LogAndApply failed: %v", err)
|
|
}
|
|
|
|
// 检查
|
|
version := vs.GetCurrent()
|
|
if version.GetFileCount() != 1 {
|
|
t.Errorf("Expected 1 file, got %d", version.GetFileCount())
|
|
}
|
|
|
|
files := version.GetSSTFiles()
|
|
if files[0].FileNumber != 1 {
|
|
t.Errorf("Expected file number 1, got %d", files[0].FileNumber)
|
|
}
|
|
|
|
t.Log("VersionSet add file test passed!")
|
|
}
|
|
|
|
func TestVersionSetDeleteFile(t *testing.T) {
|
|
dir := "./test_manifest_delete"
|
|
os.RemoveAll(dir)
|
|
defer os.RemoveAll(dir)
|
|
|
|
vs, err := NewVersionSet(dir)
|
|
if err != nil {
|
|
t.Fatalf("NewVersionSet failed: %v", err)
|
|
}
|
|
defer vs.Close()
|
|
|
|
// 添加两个文件
|
|
edit1 := NewVersionEdit()
|
|
edit1.AddFile(&FileMetadata{FileNumber: 1, FileSize: 1024, MinKey: 1, MaxKey: 100, RowCount: 100})
|
|
edit1.AddFile(&FileMetadata{FileNumber: 2, FileSize: 2048, MinKey: 101, MaxKey: 200, RowCount: 100})
|
|
vs.LogAndApply(edit1)
|
|
|
|
// 删除一个文件
|
|
edit2 := NewVersionEdit()
|
|
edit2.DeleteFile(1)
|
|
err = vs.LogAndApply(edit2)
|
|
if err != nil {
|
|
t.Fatalf("LogAndApply failed: %v", err)
|
|
}
|
|
|
|
// 检查
|
|
version := vs.GetCurrent()
|
|
if version.GetFileCount() != 1 {
|
|
t.Errorf("Expected 1 file, got %d", version.GetFileCount())
|
|
}
|
|
|
|
files := version.GetSSTFiles()
|
|
if files[0].FileNumber != 2 {
|
|
t.Errorf("Expected file number 2, got %d", files[0].FileNumber)
|
|
}
|
|
|
|
t.Log("VersionSet delete file test passed!")
|
|
}
|
|
|
|
func TestVersionSetRecover(t *testing.T) {
|
|
dir := "./test_manifest_recover"
|
|
os.RemoveAll(dir)
|
|
defer os.RemoveAll(dir)
|
|
|
|
// 第一次:创建并添加文件
|
|
vs1, err := NewVersionSet(dir)
|
|
if err != nil {
|
|
t.Fatalf("NewVersionSet failed: %v", err)
|
|
}
|
|
|
|
edit := NewVersionEdit()
|
|
edit.AddFile(&FileMetadata{FileNumber: 1, FileSize: 1024, MinKey: 1, MaxKey: 100, RowCount: 100})
|
|
edit.AddFile(&FileMetadata{FileNumber: 2, FileSize: 2048, MinKey: 101, MaxKey: 200, RowCount: 100})
|
|
vs1.LogAndApply(edit)
|
|
vs1.Close()
|
|
|
|
// 第二次:重新打开并恢复
|
|
vs2, err := NewVersionSet(dir)
|
|
if err != nil {
|
|
t.Fatalf("NewVersionSet recover failed: %v", err)
|
|
}
|
|
defer vs2.Close()
|
|
|
|
// 检查恢复的数据
|
|
version := vs2.GetCurrent()
|
|
if version.GetFileCount() != 2 {
|
|
t.Errorf("Expected 2 files after recover, got %d", version.GetFileCount())
|
|
}
|
|
|
|
files := version.GetSSTFiles()
|
|
if files[0].FileNumber != 1 || files[1].FileNumber != 2 {
|
|
t.Errorf("File numbers not correct after recover")
|
|
}
|
|
|
|
t.Log("VersionSet recover test passed!")
|
|
}
|
|
|
|
func TestVersionSetMultipleEdits(t *testing.T) {
|
|
dir := "./test_manifest_multiple"
|
|
os.RemoveAll(dir)
|
|
defer os.RemoveAll(dir)
|
|
|
|
vs, err := NewVersionSet(dir)
|
|
if err != nil {
|
|
t.Fatalf("NewVersionSet failed: %v", err)
|
|
}
|
|
defer vs.Close()
|
|
|
|
// 多次变更
|
|
for i := int64(1); i <= 10; i++ {
|
|
edit := NewVersionEdit()
|
|
edit.AddFile(&FileMetadata{
|
|
FileNumber: i,
|
|
FileSize: 1024 * i,
|
|
MinKey: (i-1)*100 + 1,
|
|
MaxKey: i * 100,
|
|
RowCount: 100,
|
|
})
|
|
err = vs.LogAndApply(edit)
|
|
if err != nil {
|
|
t.Fatalf("LogAndApply failed: %v", err)
|
|
}
|
|
}
|
|
|
|
// 检查
|
|
version := vs.GetCurrent()
|
|
if version.GetFileCount() != 10 {
|
|
t.Errorf("Expected 10 files, got %d", version.GetFileCount())
|
|
}
|
|
|
|
t.Log("VersionSet multiple edits test passed!")
|
|
}
|
|
|
|
func TestVersionEditEncodeDecode(t *testing.T) {
|
|
// 创建 VersionEdit
|
|
edit1 := NewVersionEdit()
|
|
edit1.AddFile(&FileMetadata{FileNumber: 1, FileSize: 1024, MinKey: 1, MaxKey: 100, RowCount: 100})
|
|
edit1.DeleteFile(2)
|
|
nextFile := int64(10)
|
|
edit1.SetNextFileNumber(nextFile)
|
|
lastSeq := int64(1000)
|
|
edit1.SetLastSequence(lastSeq)
|
|
|
|
// 编码
|
|
data, err := edit1.Encode()
|
|
if err != nil {
|
|
t.Fatalf("Encode failed: %v", err)
|
|
}
|
|
|
|
// 解码
|
|
edit2 := NewVersionEdit()
|
|
err = edit2.Decode(data)
|
|
if err != nil {
|
|
t.Fatalf("Decode failed: %v", err)
|
|
}
|
|
|
|
// 检查
|
|
if len(edit2.AddedFiles) != 1 {
|
|
t.Errorf("Expected 1 added file, got %d", len(edit2.AddedFiles))
|
|
}
|
|
if len(edit2.DeletedFiles) != 1 {
|
|
t.Errorf("Expected 1 deleted file, got %d", len(edit2.DeletedFiles))
|
|
}
|
|
if *edit2.NextFileNumber != 10 {
|
|
t.Errorf("Expected NextFileNumber 10, got %d", *edit2.NextFileNumber)
|
|
}
|
|
if *edit2.LastSequence != 1000 {
|
|
t.Errorf("Expected LastSequence 1000, got %d", *edit2.LastSequence)
|
|
}
|
|
|
|
t.Log("VersionEdit encode/decode test passed!")
|
|
}
|