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
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
|
package com.keuin.kbackupfabric;
import com.keuin.kbackupfabric.metadata.BackupMetadata;
import com.keuin.kbackupfabric.metadata.MetadataHolder;
import com.keuin.kbackupfabric.operation.AbstractConfirmableOperation;
import com.keuin.kbackupfabric.util.BackupFilesystemUtil;
import com.keuin.kbackupfabric.util.BackupNameTimeFormatter;
import com.keuin.kbackupfabric.util.PrintUtil;
import com.keuin.kbackupfabric.worker.BackupWorker;
import com.mojang.brigadier.arguments.StringArgumentType;
import com.mojang.brigadier.context.CommandContext;
import net.minecraft.server.MinecraftServer;
import net.minecraft.server.command.ServerCommandSource;
import java.io.File;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.List;
import java.util.Objects;
import static com.keuin.kbackupfabric.util.BackupFilesystemUtil.*;
import static com.keuin.kbackupfabric.util.PrintUtil.*;
public final class KBCommands {
private static final int SUCCESS = 1;
private static final int FAILED = -1;
//private static final Logger LOGGER = LogManager.getLogger();
private static final List<String> backupNameList = new ArrayList<>(); // index -> backupName
private static AbstractConfirmableOperation pendingOperation = null;
/**
* Print the help menu.
*
* @param context the context.
* @return stat code.
*/
public static int help(CommandContext<ServerCommandSource> context) {
msgInfo(context, "==== KBackup Manual ====");
msgInfo(context, "/kb /kb help Print help menu.");
msgInfo(context, "/kb list Show all backups.");
msgInfo(context, "/kb backup [backup_name] Backup the whole level to backup_name. The default name is current system time.");
msgInfo(context, "/kb restore <backup_name> Delete the whole current level and restore from given backup. /kb restore is identical with /kb list.");
msgInfo(context, "/kb confirm Confirm and start restoring.");
msgInfo(context, "/kb cancel Cancel the restoration to be confirmed. If cancelled, /kb confirm will not run.");
return SUCCESS;
}
/**
* Print the help menu. (May show extra info during the first run after restoring)
*
* @param context the context.
* @return stat code.
*/
public static int kb(CommandContext<ServerCommandSource> context) {
int statCode = list(context);
if (MetadataHolder.hasMetadata()) {
// Output metadata info
msgStress(context, "Restored from backup " + MetadataHolder.getMetadata().getBackupName());
}
return statCode;
}
/**
* List all existing backups.
*
* @param context the context.
* @return stat code.
*/
public static int list(CommandContext<ServerCommandSource> context) {
msgInfo(context, "Available backups: (file is not checked, manipulation may affect this plugin)");
MinecraftServer server = context.getSource().getMinecraftServer();
File[] files = getBackupSaveDirectory(server).listFiles(
(dir, name) -> dir.isDirectory() && name.toLowerCase().endsWith(".zip") && name.toLowerCase().startsWith(getBackupFileNamePrefix())
);
backupNameList.clear();
if (files != null) {
int i = 0;
for (File file : files) {
++i;
String backupName = getBackupName(file.getName());
backupNameList.add(backupName);
msgInfo(context, String.format("[%d] %s, size: %.1fMB", i, backupName, file.length() * 1.0 / 1024 / 1024));
}
}
return SUCCESS;
}
/**
* Backup with context parameter backupName.
*
* @param context the context.
* @return stat code.
*/
public static int backup(CommandContext<ServerCommandSource> context) {
//KBMain.backup("name")
String backupName = StringArgumentType.getString(context, "backupName");
if (backupName.matches("[0-9]*")) {
// Numeric param is not allowed
backupName = String.format("a%s", backupName);
msgWarn(context, String.format("Pure numeric name is not allowed. Renaming to %s", backupName));
}
return doBackup(context, backupName);
}
/**
* Delete an existing backup with context parameter backupName.
* Simply set the pending backupName to given backupName, for the second confirmation.
*
* @param context the context.
* @return stat code.
*/
public static int delete(CommandContext<ServerCommandSource> context) {
String backupName = parseBackupName(context, StringArgumentType.getString(context, "backupName"));
MinecraftServer server = context.getSource().getMinecraftServer();
if (backupName == null)
return list(context); // Show the list and return
// Validate backupName
if (!isBackupNameValid(backupName, server)) {
// Invalid backupName
msgErr(context, "Invalid backup name! Please check your input. The list index number is also valid.");
return FAILED;
}
// Update pending task
pendingOperation = AbstractConfirmableOperation.createDeleteOperation(context, backupName);
msgWarn(context, String.format("DELETION WARNING: The deletion is irreversible! You will lose the backup %s permanently. Use /kb confirm to start or /kb cancel to abort.", backupName), true);
return SUCCESS;
}
/**
* Restore with context parameter backupName.
* Simply set the pending backupName to given backupName, for the second confirmation.
*
* @param context the context.
* @return stat code.
*/
public static int restore(CommandContext<ServerCommandSource> context) {
//KBMain.restore("name")
MinecraftServer server = context.getSource().getMinecraftServer();
String backupName = parseBackupName(context, StringArgumentType.getString(context, "backupName"));
backupName = parseBackupName(context, backupName);
if (backupName == null)
return list(context); // Show the list and return
// Validate backupName
if (!isBackupNameValid(backupName, server)) {
// Invalid backupName
msgErr(context, "Invalid backup name! Please check your input. The list index number is also valid.", false);
return FAILED;
}
// Update pending task
pendingOperation = AbstractConfirmableOperation.createRestoreOperation(context, backupName);
msgWarn(context, String.format("RESET WARNING: You will LOSE YOUR CURRENT WORLD PERMANENTLY! The worlds will be replaced with backup %s . Use /kb confirm to start or /kb cancel to abort.", backupName), true);
return SUCCESS;
}
/**
* Backup with default name.
*
* @param context the context.
* @return stat code.
*/
public static int backupWithDefaultName(CommandContext<ServerCommandSource> context) {
return doBackup(context, "noname");
}
private static int doBackup(CommandContext<ServerCommandSource> context, String customName) {
// Real backup name (compatible with legacy backup): date_name, such as 2020-04-23_21-03-00_test
//KBMain.backup("name")
String backupName = BackupNameTimeFormatter.getTimeString() + "_" + customName;
// Validate file name
final char[] ILLEGAL_CHARACTERS = {'/', '\n', '\r', '\t', '\0', '\f', '`', '?', '*', '\\', '<', '>', '|', '\"', ':'};
for (char c : ILLEGAL_CHARACTERS) {
if (backupName.contains(String.valueOf(c))) {
msgErr(context, String.format("Name cannot contain special character \"%c\".", c));
return FAILED;
}
}
// Do backup
BackupMetadata metadata = new BackupMetadata(System.currentTimeMillis(), backupName);
PrintUtil.info("Invoking backup worker ...");
BackupWorker.invoke(context, backupName, metadata);
return SUCCESS;
}
/**
* Restore with context parameter backupName.
*
* @param context the context.
* @return stat code.
*/
public static int confirm(CommandContext<ServerCommandSource> context) {
if (pendingOperation == null) {
msgWarn(context, "Nothing to be confirmed. Please execute /kb restore <backup_name> first.");
return FAILED;
}
AbstractConfirmableOperation operation = pendingOperation;
pendingOperation = null;
return operation.confirm() ? SUCCESS : FAILED; // block compiler's complain.
}
/**
* Cancel the execution to be confirmed.
*
* @param context the context.
* @return stat code.
*/
public static int cancel(CommandContext<ServerCommandSource> context) {
if (pendingOperation != null) {
PrintUtil.msgInfo(context, String.format("The %s has been cancelled.", pendingOperation.toString()), true);
pendingOperation = null;
return SUCCESS;
} else {
msgErr(context, "Nothing to cancel.");
return FAILED;
}
}
/**
* Show the most recent backup.
* If there is no available backup, print specific info.
*
* @param context the context.
* @return stat code.
*/
public static int prev(CommandContext<ServerCommandSource> context) {
try {
// List all backups
MinecraftServer server = context.getSource().getMinecraftServer();
List<File> files = Arrays.asList(Objects.requireNonNull(getBackupSaveDirectory(server).listFiles()));
files.removeIf(f -> !f.getName().startsWith(BackupFilesystemUtil.getBackupFileNamePrefix()));
files.sort((x, y) -> (int) (BackupFilesystemUtil.getBackupTimeFromBackupFileName(y.getName()) - BackupFilesystemUtil.getBackupTimeFromBackupFileName(x.getName())));
File prevBackupFile = files.get(0);
String backupName = getBackupName(prevBackupFile.getName());
int i = backupNameList.indexOf(backupName);
if (i == -1) {
backupNameList.add(backupName);
i = backupNameList.size();
} else {
++i;
}
msgInfo(context, String.format("The most recent backup: [%d] %s , size: %s", i, backupName, humanFileSize(prevBackupFile.length())));
} catch (NullPointerException e) {
msgInfo(context, "There are no backups available.");
} catch (SecurityException ignored) {
msgErr(context, "Failed to read file.");
return FAILED;
}
return SUCCESS;
}
private static String parseBackupName(CommandContext<ServerCommandSource> context, String userInput) {
try {
String backupName = StringArgumentType.getString(context, "backupName");
if (backupName.matches("[0-9]*")) {
// If numeric input
int index = Integer.parseInt(backupName) - 1;
return backupNameList.get(index); // Replace input number with real backup name.
}
} catch (NumberFormatException | IndexOutOfBoundsException ignored) {
}
return userInput;
}
}
|