1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
| package com.yinlongfei.lightweight.database.metadata;
import com.yinlongfei.lightweight.database.storage.Row;
import javax.json.Json;
import javax.json.JsonObject;
import java.nio.file.Files;
import java.nio.file.Path;
import java.util.*;
import java.util.stream.Collectors;
public class MetadataManager {
private final Map<String, TableMetadata> tables = new HashMap<>();
private final Path dataDir;
private final boolean isMemoryMode;
public MetadataManager(String url) {
this.dataDir = Path.of(url.replace("jdbc:lightweight:", ""));
this.isMemoryMode = url.contains("memory");
if (!isMemoryMode && !Files.exists(dataDir)) {
try {
Files.createDirectories(dataDir);
} catch (Exception e) {
throw new RuntimeException("Failed to create metadata directory", e);
}
}
loadMetadata(); // 初始化时加载元数据
}
public void addTable(String tableName, List<String> columnNames, List<String> columnTypes, List<String> indexedColumns) {
List<ColumnDefinition> columns = new ArrayList<>();
for (int i = 0; i < columnNames.size(); i++) {
boolean isPrimaryKey = indexedColumns.contains(columnNames.get(i));
columns.add(new ColumnDefinition(columnNames.get(i), columnTypes.get(i), isPrimaryKey));
}
tables.put(tableName, new TableMetadata(tableName, columns, indexedColumns));
if (!isMemoryMode) persistMetadata();
}
public void dropTable(String tableName) {
tables.remove(tableName);
if (!isMemoryMode) persistMetadata();
}
public double getRowCount(String tableName) {
TableMetadata table = tables.get(tableName);
return table != null ? table.getRowCount() : 0;
}
public void updateRowCount(String tableName, double rowCount) {
TableMetadata table = tables.get(tableName);
if (table != null) {
table.setRowCount(rowCount);
if (!isMemoryMode) persistMetadata();
}
}
public List<String> getIndexedColumns(String tableName) {
TableMetadata table = tables.get(tableName);
return table != null ? table.getIndexedColumns() : Collections.emptyList();
}
public String getTableNameForRow(Row row) {
// 根据Row内容查找所属表(简化为遍历)
return tables.entrySet().stream()
.filter(entry -> entry.getValue().getColumns().stream()
.allMatch(col -> row.getColumns().containsKey(col.getName())))
.map(Map.Entry::getKey)
.findFirst()
.orElse(null);
}
private void persistMetadata() {
try {
JsonObject json = Json.createObjectBuilder()
.add("tables", Json.createObjectBuilder()
.addAll(tables.entrySet().stream()
.collect(Collectors.toMap(
Map.Entry::getKey,
entry -> Json.createObjectBuilder()
.add("columns", Json.createArrayBuilder(
entry.getValue().getColumns().stream()
.map(col -> Json.createObjectBuilder()
.add("name", col.getName())
.add("type", col.getType())
.add("isPrimaryKey", col.isPrimaryKey())
.build())
.collect(Collectors.toList())))
.add("indexedColumns", Json.createArrayBuilder(entry.getValue().getIndexedColumns()))
.add("rowCount", entry.getValue().getRowCount())
.build()))))
.build();
Files.writeString(dataDir.resolve("metadata.json"), json.toString());
} catch (Exception e) {
throw new RuntimeException("Failed to persist metadata", e);
}
}
private void loadMetadata() {
if (isMemoryMode || !Files.exists(dataDir.resolve("metadata.json"))) return;
try {
String content = Files.readString(dataDir.resolve("metadata.json"));
JsonObject json = Json.createReader(new StringReader(content)).readObject();
JsonObject tablesJson = json.getJsonObject("tables");
for (String tableName : tablesJson.keySet()) {
JsonObject tableJson = tablesJson.getJsonObject(tableName);
List<ColumnDefinition> columns = tableJson.getJsonArray("columns").stream()
.map(obj -> {
JsonObject col = (JsonObject) obj;
return new ColumnDefinition(col.getString("name"), col.getString("type"), col.getBoolean("isPrimaryKey"));
})
.collect(Collectors.toList());
List<String> indexedColumns = tableJson.getJsonArray("indexedColumns").stream()
.map(Object::toString)
.collect(Collectors.toList());
TableMetadata table = new TableMetadata(tableName, columns, indexedColumns);
table.setRowCount(tableJson.getJsonNumber("rowCount").doubleValue());
tables.put(tableName, table);
}
} catch (Exception e) {
throw new RuntimeException("Failed to load metadata", e);
}
}
}
|