Repaired race condition with ParallelZip. Hashing now works!
							parent
							
								
									5367a00cdc
								
							
						
					
					
						commit
						2efe112157
					
				|  | @ -19,16 +19,14 @@ | |||
| package net.szum123321.textile_backup; | ||||
| 
 | ||||
| import net.minecraft.server.MinecraftServer; | ||||
| import net.szum123321.textile_backup.core.digest.BalticHash; | ||||
| import net.szum123321.textile_backup.core.digest.Hash; | ||||
| import net.szum123321.textile_backup.core.Utilities; | ||||
| 
 | ||||
| import net.szum123321.textile_backup.core.create.MakeBackupRunnable; | ||||
| import net.szum123321.textile_backup.core.restore.AwaitThread; | ||||
| import org.apache.commons.io.FileUtils; | ||||
| import org.tukaani.xz.check.CRC64; | ||||
| 
 | ||||
| import java.nio.ByteBuffer; | ||||
| import java.nio.ByteOrder; | ||||
| import java.nio.file.Files; | ||||
| import java.nio.file.Path; | ||||
| import java.time.format.DateTimeFormatter; | ||||
|  | @ -44,33 +42,9 @@ public class Globals { | |||
|     public static final Globals INSTANCE = new Globals(); | ||||
|     private static final TextileLogger log = new TextileLogger(TextileBackup.MOD_NAME); | ||||
|     public static final DateTimeFormatter defaultDateTimeFormatter = DateTimeFormatter.ofPattern("yyyy-MM-dd_HH.mm.ss"); | ||||
|     public static final Supplier<Hash> CHECKSUM_SUPPLIER = () -> new Hash() { | ||||
|         private final CRC64 crc = new CRC64(); | ||||
|         @Override | ||||
|         public void update(byte b) { | ||||
|             crc.update(new byte[]{b}); | ||||
|         } | ||||
|     public static final Supplier<Hash> CHECKSUM_SUPPLIER = BalticHash::new; | ||||
| 
 | ||||
|         @Override | ||||
|         public void update(long b) { | ||||
|             ByteBuffer v = ByteBuffer.allocate(8).order(ByteOrder.LITTLE_ENDIAN); | ||||
|             v.putLong(b); | ||||
|             crc.update(v.array()); | ||||
|         } | ||||
| 
 | ||||
|         @Override | ||||
|         public void update(byte[] b, int off, int len) { | ||||
|             crc.update(b, off, len); | ||||
|         } | ||||
| 
 | ||||
|         @Override | ||||
|         public long getValue() { | ||||
|             ByteBuffer b = ByteBuffer.wrap(crc.finish()); | ||||
|             return b.getLong(); | ||||
|         } | ||||
|     }; | ||||
| 
 | ||||
|     private ExecutorService executorService = null;// = Executors.newSingleThreadExecutor();
 | ||||
|     private ExecutorService executorService = null;//TODO: AAAAAAAAAAAAAAA MEMORY LEAK!!!!!!!!!
 | ||||
|     public final AtomicBoolean globalShutdownBackupFlag = new AtomicBoolean(true); | ||||
|     public boolean disableWatchdog = false; | ||||
|     private boolean disableTMPFiles = false; | ||||
|  |  | |||
|  | @ -30,6 +30,7 @@ public record BackupContext(@NotNull MinecraftServer server, | |||
|                             ServerCommandSource commandSource, | ||||
|                             ActionInitiator initiator, | ||||
|                             boolean save, | ||||
|                             boolean cleanup, | ||||
|                             String comment, | ||||
|                             LocalDateTime startDate) { | ||||
| 
 | ||||
|  | @ -46,6 +47,7 @@ public record BackupContext(@NotNull MinecraftServer server, | |||
|         private ServerCommandSource commandSource; | ||||
|         private ActionInitiator initiator; | ||||
|         private boolean save; | ||||
|         private boolean cleanup; | ||||
|         private String comment; | ||||
| 
 | ||||
|         private boolean guessInitiator; | ||||
|  | @ -55,6 +57,7 @@ public record BackupContext(@NotNull MinecraftServer server, | |||
|             this.commandSource = null; | ||||
|             this.initiator = null; | ||||
|             this.save = false; | ||||
|             cleanup = true; //defaults
 | ||||
|             this.comment = null; | ||||
| 
 | ||||
|             guessInitiator = false; | ||||
|  | @ -94,6 +97,11 @@ public record BackupContext(@NotNull MinecraftServer server, | |||
|             return this; | ||||
|         } | ||||
| 
 | ||||
|         public Builder dontCleanup() { | ||||
|             this.cleanup = false; | ||||
|             return this; | ||||
|         } | ||||
| 
 | ||||
|         public BackupContext build() { | ||||
|             if (guessInitiator) { | ||||
|                 initiator = commandSource.getEntity() instanceof PlayerEntity ? ActionInitiator.Player : ActionInitiator.ServerConsole; | ||||
|  | @ -106,7 +114,7 @@ public record BackupContext(@NotNull MinecraftServer server, | |||
|                 else throw new RuntimeException("Neither MinecraftServer or ServerCommandSource were provided!"); | ||||
|             } | ||||
| 
 | ||||
|             return new BackupContext(server, commandSource, initiator, save, comment, LocalDateTime.now()); | ||||
|             return new BackupContext(server, commandSource, initiator, save, cleanup, comment, LocalDateTime.now()); | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  |  | |||
|  | @ -29,15 +29,19 @@ import java.io.InputStream; | |||
| import java.nio.file.Files; | ||||
| import java.nio.file.Path; | ||||
| import java.util.Optional; | ||||
| import java.util.concurrent.CountDownLatch; | ||||
| 
 | ||||
| public record FileInputStreamSupplier(Path path, String name, FileTreeHashBuilder hashTreeBuilder, BrokenFileHandler brokenFileHandler) implements InputSupplier { | ||||
| public record FileInputStreamSupplier(Path path, String name, FileTreeHashBuilder hashTreeBuilder, BrokenFileHandler brokenFileHandler, CountDownLatch latch) implements InputSupplier { | ||||
|     private final static TextileLogger log = new TextileLogger(TextileBackup.MOD_NAME); | ||||
| 
 | ||||
|     @Override | ||||
|     public InputStream getInputStream() throws IOException { | ||||
|         try { | ||||
|             return new HashingInputStream(Files.newInputStream(path), path, hashTreeBuilder, brokenFileHandler); | ||||
|             return new HashingInputStream(Files.newInputStream(path), path, hashTreeBuilder, brokenFileHandler, latch); | ||||
|         } catch (IOException e) { | ||||
|             //Probably good idea to just put it here. In the case an exception is thrown here, it would be probable
 | ||||
|             //The latch would never be lifted
 | ||||
|             latch.countDown(); | ||||
|             brokenFileHandler.handle(path, e); | ||||
|             throw e; | ||||
|         } | ||||
|  | @ -47,9 +51,7 @@ public record FileInputStreamSupplier(Path path, String name, FileTreeHashBuilde | |||
|     public Optional<Path> getPath() { return Optional.of(path); } | ||||
| 
 | ||||
|     @Override | ||||
|     public long size() throws IOException { | ||||
|         return Files.size(path); | ||||
|     } | ||||
|     public long size() throws IOException { return Files.size(path); } | ||||
| 
 | ||||
|     @Override | ||||
|     public String getName() { | ||||
|  |  | |||
|  | @ -102,8 +102,8 @@ public class MakeBackupRunnable implements Callable<Void> { | |||
|                 case TAR -> new AbstractTarArchiver().createArchive(world, outFile, context, coreCount); | ||||
|             } | ||||
| 
 | ||||
|             if(!Globals.INSTANCE.getQueueExecutor().isShutdown()) | ||||
|                 Globals.INSTANCE.getQueueExecutor().submit(new Cleanup(context.commandSource(), Utilities.getLevelName(context.server()))); | ||||
|             if(context.cleanup()) | ||||
|                 new Cleanup(context.commandSource(), Utilities.getLevelName(context.server())).call(); | ||||
| 
 | ||||
|             if (config.get().broadcastBackupDone) Utilities.notifyPlayers(context.server(), "Done!"); | ||||
|             else log.sendInfoAL(context, "Done!"); | ||||
|  |  | |||
|  | @ -34,7 +34,9 @@ import java.nio.file.Path; | |||
| import java.time.Duration; | ||||
| import java.time.Instant; | ||||
| import java.util.Optional; | ||||
| import java.util.concurrent.CountDownLatch; | ||||
| import java.util.concurrent.ExecutionException; | ||||
| import java.util.concurrent.atomic.AtomicInteger; | ||||
| import java.util.stream.Stream; | ||||
| 
 | ||||
| /** | ||||
|  | @ -53,9 +55,18 @@ public abstract class AbstractCompressor { | |||
|              BufferedOutputStream bufferedOutputStream = new BufferedOutputStream(outStream); | ||||
|              OutputStream arc = createArchiveOutputStream(bufferedOutputStream, ctx, coreLimit); | ||||
|              Stream<Path> fileStream = Files.walk(inputFile)) { | ||||
| 
 | ||||
|             AtomicInteger fileCounter = new AtomicInteger(0); | ||||
| 
 | ||||
|             var it = fileStream | ||||
|                     .filter(path -> !Utilities.isBlacklisted(inputFile.relativize(path))) | ||||
|                     .filter(Files::isRegularFile).iterator(); | ||||
|                     .filter(Files::isRegularFile) | ||||
|                     .peek(x -> fileCounter.incrementAndGet()).toList() | ||||
|                     .iterator(); | ||||
| 
 | ||||
|             log.info("File count: {}", fileCounter.get()); | ||||
| 
 | ||||
|             CountDownLatch latch = new CountDownLatch(fileCounter.get()); | ||||
| 
 | ||||
|             while(it.hasNext()) { | ||||
|                 Path file = it.next(); | ||||
|  | @ -66,7 +77,8 @@ public abstract class AbstractCompressor { | |||
|                                     file, | ||||
|                                     inputFile.relativize(file).toString(), | ||||
|                                     fileHashBuilder, | ||||
|                                     brokenFileHandler), | ||||
|                                     brokenFileHandler, | ||||
|                                     latch), | ||||
|                             arc | ||||
|                     ); | ||||
|                 } catch (IOException e) { | ||||
|  | @ -78,37 +90,21 @@ public abstract class AbstractCompressor { | |||
|                 } | ||||
|             } | ||||
| 
 | ||||
|             latch.await(); | ||||
| 
 | ||||
|             Instant now = Instant.now(); | ||||
| 
 | ||||
|             CompressionStatus status = new CompressionStatus ( | ||||
|                     fileHashBuilder.getValue(), | ||||
|                     brokenFileHandler.get(), | ||||
|                     ctx.startDate(), start.toEpochMilli(), now.toEpochMilli(), | ||||
|                     brokenFileHandler.get() | ||||
|                     TextileBackup.VERSION | ||||
|             ); | ||||
| 
 | ||||
|             addEntry(new StatusFileInputSupplier(status.serialize()), arc); | ||||
| 
 | ||||
|             finish(arc); | ||||
|         } /*catch(NoSpaceLeftOnDeviceException e) { | ||||
|             log.error(""" | ||||
|             CRITICAL ERROR OCCURRED! | ||||
|             The backup is corrupt! | ||||
|             Don't panic! This is a known issue! | ||||
|             For help see: https://github.com/Szum123321/textile_backup/wiki/ZIP-Problems
 | ||||
|             In case this isn't it here's also the exception itself""", e); | ||||
| 
 | ||||
|             if(ctx.initiator() == ActionInitiator.Player) { | ||||
|                 log.sendError(ctx, "Backup failed. The file is corrupt."); | ||||
|                 log.error("For help see: https://github.com/Szum123321/textile_backup/wiki/ZIP-Problems"); | ||||
|             } | ||||
|             if(ConfigHelper.INSTANCE.get().errorErrorHandlingMode.isStrict()) keep = false; | ||||
|         } catch (IOException | InterruptedException | ExecutionException e) { | ||||
|             log.error("An exception occurred!", e); | ||||
|             if(ctx.initiator() == ActionInitiator.Player) | ||||
|                 log.sendError(ctx, "Something went wrong while compressing files!"); | ||||
|             if(ConfigHelper.INSTANCE.get().errorErrorHandlingMode.isStrict()) keep = false; | ||||
| 
 | ||||
|         } */finally { | ||||
|         } finally { | ||||
|             close(); | ||||
|         } | ||||
| 
 | ||||
|  |  | |||
|  | @ -27,19 +27,23 @@ import java.io.IOException; | |||
| import java.nio.charset.StandardCharsets; | ||||
| import java.nio.file.Files; | ||||
| import java.nio.file.Path; | ||||
| import java.util.concurrent.atomic.AtomicBoolean; | ||||
| 
 | ||||
| public class FileTreeHashBuilder { | ||||
|     private final static TextileLogger log = new TextileLogger(TextileBackup.MOD_NAME); | ||||
|     private final Object lock = new Object(); | ||||
|     private long hash = 0, filesProcessed = 0, filesTotalSize = 0; | ||||
|     private final AtomicBoolean closed = new AtomicBoolean(false); | ||||
| 
 | ||||
|     public void update(Path path, long newHash) throws IOException { | ||||
|         if(path.getFileName().toString().equals(CompressionStatus.DATA_FILENAME)) return; | ||||
|         //log.info("Putting: {}, {}", path, newHash);
 | ||||
|         if(closed.get()) throw new RuntimeException("Hash Builder already closed!"); | ||||
| 
 | ||||
|         var hasher = Globals.CHECKSUM_SUPPLIER.get(); | ||||
|         if(path.getFileName().toString().equals(CompressionStatus.DATA_FILENAME)) return; | ||||
| 
 | ||||
|         long size = Files.size(path); | ||||
| 
 | ||||
|         var hasher = Globals.CHECKSUM_SUPPLIER.get(); | ||||
| 
 | ||||
|         hasher.update(path.getFileName().toString().getBytes(StandardCharsets.UTF_8)); | ||||
|         hasher.update(newHash); | ||||
| 
 | ||||
|  | @ -53,7 +57,10 @@ public class FileTreeHashBuilder { | |||
| 
 | ||||
|     public long getValue() { | ||||
|         var hasher = Globals.CHECKSUM_SUPPLIER.get(); | ||||
|         closed.set(true); | ||||
| 
 | ||||
|         synchronized (lock) { | ||||
|             log.debug("Closing: files: {}, bytes {}, raw hash {}", filesProcessed, filesTotalSize, hash); | ||||
|             hasher.update(hash); | ||||
|             hasher.update(filesProcessed); | ||||
|             hasher.update(filesTotalSize); | ||||
|  | @ -61,3 +68,4 @@ public class FileTreeHashBuilder { | |||
|             return hasher.getValue(); | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  |  | |||
|  | @ -19,39 +19,28 @@ | |||
| package net.szum123321.textile_backup.core.digest; | ||||
| 
 | ||||
| import net.szum123321.textile_backup.Globals; | ||||
| import net.szum123321.textile_backup.TextileBackup; | ||||
| import net.szum123321.textile_backup.TextileLogger; | ||||
| import net.szum123321.textile_backup.core.DataLeftException; | ||||
| import net.szum123321.textile_backup.core.create.BrokenFileHandler; | ||||
| import org.jetbrains.annotations.NotNull; | ||||
| 
 | ||||
| import java.io.*; | ||||
| import java.nio.file.Path; | ||||
| import java.util.concurrent.CountDownLatch; | ||||
| 
 | ||||
| //This class calculates a hash of the file on the input stream, submits it to FileTreeHashBuilder.
 | ||||
| //In case the underlying stream hasn't been read completely in, puts it into BrokeFileHandler
 | ||||
| public class HashingInputStream extends FilterInputStream { | ||||
|     private final static TextileLogger log = new TextileLogger(TextileBackup.MOD_NAME); | ||||
|     private final Path path; | ||||
|     private final Hash hasher = Globals.CHECKSUM_SUPPLIER.get(); | ||||
|     private final FileTreeHashBuilder hashBuilder; | ||||
|     private final BrokenFileHandler brokenFileHandler; | ||||
|     private final CountDownLatch latch; | ||||
| 
 | ||||
|     private int cnt = 0; | ||||
| 
 | ||||
|     @Override | ||||
|     public synchronized void reset() throws IOException { | ||||
|         log.info("Called reset! {}", path); | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
|     public boolean markSupported() { | ||||
|         return false; | ||||
|     } | ||||
| 
 | ||||
|     public HashingInputStream(InputStream in, Path path, FileTreeHashBuilder hashBuilder, BrokenFileHandler brokenFileHandler) { | ||||
|     public HashingInputStream(InputStream in, Path path, FileTreeHashBuilder hashBuilder, BrokenFileHandler brokenFileHandler, CountDownLatch latch) { | ||||
|         super(in); | ||||
|         this.path = path; | ||||
|         this.latch = latch; | ||||
|         this.hashBuilder = hashBuilder; | ||||
|         this.brokenFileHandler = brokenFileHandler; | ||||
|     } | ||||
|  | @ -59,34 +48,29 @@ public class HashingInputStream extends FilterInputStream { | |||
|     @Override | ||||
|     public int read(byte @NotNull [] b, int off, int len) throws IOException { | ||||
|         int i = in.read(b, off, len); | ||||
|         if(i > -1) { | ||||
|             hasher.update(b, off, i); | ||||
|             cnt += i; | ||||
|         } | ||||
|         if(i != -1) hasher.update(b, off, i); | ||||
|         return i; | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
|     public int read() throws IOException { | ||||
|         int i = in.read(); | ||||
|         if(i > -1) { | ||||
|             hasher.update(i); | ||||
|             cnt++; | ||||
|         } | ||||
|         if(i != -1) hasher.update(i); | ||||
|         return i; | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
|     public boolean markSupported() { | ||||
|         return false; | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
|     public void close() throws IOException { | ||||
|         if(in.available() == 0) { | ||||
|             long val = hasher.getValue(); | ||||
|             hashBuilder.update(path, val); | ||||
|             log.info("Read in {}, of {}, with hash {}", path, cnt, val); | ||||
|         } | ||||
|         else { | ||||
|             brokenFileHandler.handle(path, new DataLeftException(in.available())); | ||||
|             //log.info("bad file {} {}",path, cnt);
 | ||||
|         } | ||||
|         if(in.available() == 0) hashBuilder.update(path, hasher.getValue()); | ||||
|         else brokenFileHandler.handle(path, new DataLeftException(in.available())); | ||||
| 
 | ||||
|         latch.countDown(); | ||||
| 
 | ||||
|         super.close(); | ||||
|     } | ||||
| } | ||||
|  |  | |||
|  | @ -67,6 +67,8 @@ public class RestoreBackupRunnable implements Runnable { | |||
|             return; | ||||
|         } | ||||
| 
 | ||||
|         //By making a separate thread we can start unpacking an old backup instantly
 | ||||
|         //Let the server shut down gracefully, and wait for the old world backup to complete
 | ||||
|         FutureTask<Void> waitForShutdown = new FutureTask<>(() -> { | ||||
|             ctx.server().getThread().join(); //wait for server to die and save all its state
 | ||||
|             if(config.get().backupOldWorlds) { | ||||
|  | @ -75,6 +77,7 @@ public class RestoreBackupRunnable implements Runnable { | |||
|                                 .newBackupContextBuilder() | ||||
|                                 .setServer(ctx.server()) | ||||
|                                 .setInitiator(ActionInitiator.Restore) | ||||
|                                 .dontCleanup() | ||||
|                                 .setComment("Old_World" + (ctx.comment() != null ? "_" + ctx.comment() : "")) | ||||
|                                 .build() | ||||
|                 ).call(); | ||||
|  | @ -82,7 +85,7 @@ public class RestoreBackupRunnable implements Runnable { | |||
|             return null; | ||||
|         }); | ||||
| 
 | ||||
|         new Thread(waitForShutdown).start(); | ||||
|         new Thread(waitForShutdown, "Server shutdown wait thread").start(); | ||||
| 
 | ||||
|         try { | ||||
|             log.info("Starting decompression..."); | ||||
|  | @ -97,11 +100,14 @@ public class RestoreBackupRunnable implements Runnable { | |||
|             CompressionStatus status = CompressionStatus.readFromFile(tmp); | ||||
|             Files.delete(tmp.resolve(CompressionStatus.DATA_FILENAME)); | ||||
| 
 | ||||
|             log.info("Waiting for server to fully terminate..."); | ||||
| 
 | ||||
|             //locks until the backup is finished
 | ||||
|             waitForShutdown.get(); | ||||
| 
 | ||||
|             log.info("Status: {}", status); | ||||
| 
 | ||||
|             //TODO: check broken file array
 | ||||
|             boolean valid = status.isValid(hash); | ||||
|             if(valid || !config.get().errorErrorHandlingMode.verify()) { | ||||
|                 if(valid) log.info("Backup valid. Restoring"); | ||||
|  | @ -121,12 +127,11 @@ public class RestoreBackupRunnable implements Runnable { | |||
|             log.error("An exception occurred while trying to restore a backup!", e); | ||||
|         } finally { | ||||
|             //Regardless of what happened, we should still clean up
 | ||||
|            /* if(Files.exists(tmp)) { | ||||
|             if(Files.exists(tmp)) { | ||||
|                 try { | ||||
|                     Utilities.deleteDirectory(tmp); | ||||
|                 } catch (IOException ignored) {} | ||||
|             }*/ | ||||
|             //TODO: uncomment
 | ||||
|             } | ||||
|         } | ||||
| 
 | ||||
|         //in case we're playing on client
 | ||||
|  |  | |||
		Loading…
	
		Reference in New Issue