From d38bb701e76ddc4ad18f24144b6170456d6c5faa Mon Sep 17 00:00:00 2001 From: Alva Swanson Date: Fri, 16 Feb 2024 14:48:32 +0100 Subject: [PATCH 1/2] Implement DirectoryHasNChildren Hamcrest Matcher --- .../persistence/DirectoryHasNChildren.java | 54 +++++++++++++++++++ 1 file changed, 54 insertions(+) create mode 100644 persistence/src/test/java/bisq/persistence/DirectoryHasNChildren.java diff --git a/persistence/src/test/java/bisq/persistence/DirectoryHasNChildren.java b/persistence/src/test/java/bisq/persistence/DirectoryHasNChildren.java new file mode 100644 index 00000000000..6dc47c1e9bc --- /dev/null +++ b/persistence/src/test/java/bisq/persistence/DirectoryHasNChildren.java @@ -0,0 +1,54 @@ +/* + * This file is part of Bisq. + * + * Bisq is free software: you can redistribute it and/or modify it + * under the terms of the GNU Affero General Public License as published by + * the Free Software Foundation, either version 3 of the License, or (at + * your option) any later version. + * + * Bisq is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU Affero General Public + * License for more details. + * + * You should have received a copy of the GNU Affero General Public License + * along with Bisq. If not, see . + */ + +package bisq.persistence; + +import java.nio.file.Path; + +import java.io.File; + +import java.util.Objects; + + + +import org.hamcrest.Description; +import org.hamcrest.Matcher; +import org.hamcrest.TypeSafeMatcher; + +public class DirectoryHasNChildren extends TypeSafeMatcher { + + private final int numberOfChildren; + + public DirectoryHasNChildren(int numberOfChildren) { + this.numberOfChildren = numberOfChildren; + } + + @Override + protected boolean matchesSafely(Path item) { + File[] files = item.toFile().listFiles(); + return Objects.requireNonNull(files).length == numberOfChildren; + } + + @Override + public void describeTo(Description description) { + description.appendText("has " + numberOfChildren + " children"); + } + + public static Matcher hasNChildren(int numberOfChildren) { + return new DirectoryHasNChildren(numberOfChildren); + } +} From 9d4d39650fc82cbf7bf66b2c984ea05a7f03ffb3 Mon Sep 17 00:00:00 2001 From: Alva Swanson Date: Fri, 16 Feb 2024 14:48:32 +0100 Subject: [PATCH 2/2] persistence: Implement RollingBackups --- .../RollingBackupCreationFailedException.java | 24 +++ .../java/bisq/persistence/RollingBackups.java | 69 +++++++ .../bisq/persistence/RollingBackupsTests.java | 179 ++++++++++++++++++ 3 files changed, 272 insertions(+) create mode 100644 persistence/src/main/java/bisq/persistence/RollingBackupCreationFailedException.java create mode 100644 persistence/src/main/java/bisq/persistence/RollingBackups.java create mode 100644 persistence/src/test/java/bisq/persistence/RollingBackupsTests.java diff --git a/persistence/src/main/java/bisq/persistence/RollingBackupCreationFailedException.java b/persistence/src/main/java/bisq/persistence/RollingBackupCreationFailedException.java new file mode 100644 index 00000000000..2bdef3d6ebc --- /dev/null +++ b/persistence/src/main/java/bisq/persistence/RollingBackupCreationFailedException.java @@ -0,0 +1,24 @@ +/* + * This file is part of Bisq. + * + * Bisq is free software: you can redistribute it and/or modify it + * under the terms of the GNU Affero General Public License as published by + * the Free Software Foundation, either version 3 of the License, or (at + * your option) any later version. + * + * Bisq is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU Affero General Public + * License for more details. + * + * You should have received a copy of the GNU Affero General Public License + * along with Bisq. If not, see . + */ + +package bisq.persistence; + +public class RollingBackupCreationFailedException extends RuntimeException { + public RollingBackupCreationFailedException(String message) { + super(message); + } +} diff --git a/persistence/src/main/java/bisq/persistence/RollingBackups.java b/persistence/src/main/java/bisq/persistence/RollingBackups.java new file mode 100644 index 00000000000..554a099ea35 --- /dev/null +++ b/persistence/src/main/java/bisq/persistence/RollingBackups.java @@ -0,0 +1,69 @@ +/* + * This file is part of Bisq. + * + * Bisq is free software: you can redistribute it and/or modify it + * under the terms of the GNU Affero General Public License as published by + * the Free Software Foundation, either version 3 of the License, or (at + * your option) any later version. + * + * Bisq is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU Affero General Public + * License for more details. + * + * You should have received a copy of the GNU Affero General Public License + * along with Bisq. If not, see . + */ + +package bisq.persistence; + +import java.io.File; + +public class RollingBackups { + private final File baseFile; + private final int numberOfBackups; + private final File parentDirFile; + private final String baseFileName; + + public RollingBackups(File baseFile, int numberOfBackups) { + if (numberOfBackups < 1) { + throw new IllegalArgumentException("Number of backup is " + numberOfBackups); + } + + this.baseFile = baseFile; + this.numberOfBackups = numberOfBackups; + parentDirFile = baseFile.getParentFile(); + baseFileName = baseFile.getName(); + } + + public void rollBackups() { + for (int i = numberOfBackups - 2; i >= 0; i--) { + File originalFile = new File(parentDirFile, baseFileName + "_" + i); + File backupFile = new File(parentDirFile, baseFileName + "_" + (i + 1)); + renameFile(originalFile, backupFile); + } + + File backupFile = new File(parentDirFile, baseFileName + "_0"); + renameFile(baseFile, backupFile); + } + + private void renameFile(File originalFile, File newFile) { + if (!originalFile.exists()) { + return; + } + + if (newFile.exists()) { + boolean isSuccess = newFile.delete(); + if (!isSuccess) { + throw new RollingBackupCreationFailedException("Couldn't delete " + newFile.getAbsolutePath() + + " before replacing it."); + } + } + + boolean isSuccess = originalFile.renameTo(newFile); + if (!isSuccess) { + throw new RollingBackupCreationFailedException("Couldn't rename " + originalFile.getAbsolutePath() + " to " + + newFile.getAbsolutePath()); + } + } +} diff --git a/persistence/src/test/java/bisq/persistence/RollingBackupsTests.java b/persistence/src/test/java/bisq/persistence/RollingBackupsTests.java new file mode 100644 index 00000000000..ce2902a2f4a --- /dev/null +++ b/persistence/src/test/java/bisq/persistence/RollingBackupsTests.java @@ -0,0 +1,179 @@ +/* + * This file is part of Bisq. + * + * Bisq is free software: you can redistribute it and/or modify it + * under the terms of the GNU Affero General Public License as published by + * the Free Software Foundation, either version 3 of the License, or (at + * your option) any later version. + * + * Bisq is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU Affero General Public + * License for more details. + * + * You should have received a copy of the GNU Affero General Public License + * along with Bisq. If not, see . + */ + +package bisq.persistence; + +import java.nio.file.Files; +import java.nio.file.Path; + +import java.io.File; +import java.io.IOException; + +import org.junit.jupiter.api.BeforeEach; +import org.junit.jupiter.api.Test; +import org.junit.jupiter.api.io.TempDir; + +import static bisq.persistence.DirectoryHasNChildren.hasNChildren; +import static org.hamcrest.CoreMatchers.is; +import static org.hamcrest.MatcherAssert.assertThat; +import static org.junit.jupiter.api.Assertions.assertThrows; + +public class RollingBackupsTests { + private Path baseFilePath; + + @BeforeEach + void setup(@TempDir Path tempDir) { + baseFilePath = tempDir.resolve("file"); + } + + @Test + void noBackup(@TempDir Path tempDir) { + File file = new File(tempDir.toFile(), "file"); + assertThrows(IllegalArgumentException.class, () -> new RollingBackups(file, 0)); + } + + @Test + void firstBackup(@TempDir Path tempDir) throws IOException { + Files.writeString(baseFilePath, "ABC"); + assertThat(tempDir, hasNChildren(1)); + + RollingBackups rollingBackups = new RollingBackups(baseFilePath.toFile(), 1); + rollingBackups.rollBackups(); + + assertThat(tempDir, hasNChildren(1)); + + File backupFile = tempDir.resolve("file_0").toFile(); + String backupFileContent = Files.readString(backupFile.toPath()); + assertThat(backupFileContent, is("ABC")); + } + + @Test + void oneBackupWithExistingFiles(@TempDir Path tempDir) throws IOException { + Files.writeString(baseFilePath, "NEW_CONTENT"); + + Path backupPath = tempDir.resolve("file_0"); + Files.writeString(backupPath, "OLD_CONTENT"); + + assertThat(tempDir, hasNChildren(2)); + + RollingBackups rollingBackups = new RollingBackups(baseFilePath.toFile(), 1); + rollingBackups.rollBackups(); + + assertThat(tempDir, hasNChildren(1)); + String backupFileContent = Files.readString(backupPath); + assertThat(backupFileContent, is("NEW_CONTENT")); + } + + @Test + void threeBackupsFirstBackup(@TempDir Path tempDir) throws IOException { + Files.writeString(baseFilePath, "NEW_CONTENT"); + assertThat(tempDir, hasNChildren(1)); + + RollingBackups rollingBackups = new RollingBackups(baseFilePath.toFile(), 3); + rollingBackups.rollBackups(); + + assertThat(tempDir, hasNChildren(1)); + + Path backupPath = tempDir.resolve("file_0"); + String backupFileContent = Files.readString(backupPath); + assertThat(backupFileContent, is("NEW_CONTENT")); + } + + @Test + void threeBackupsWithExistingFiles(@TempDir Path tempDir) throws IOException { + Files.writeString(baseFilePath, "A"); + + Path firstBackupPath = tempDir.resolve("file_0"); + Files.writeString(firstBackupPath, "B"); + + Path secondBackupPath = tempDir.resolve("file_1"); + Files.writeString(secondBackupPath, "C"); + + Path thirdBackupPath = tempDir.resolve("file_2"); + Files.writeString(thirdBackupPath, "D"); + + assertThat(tempDir, hasNChildren(4)); + + RollingBackups rollingBackups = new RollingBackups(baseFilePath.toFile(), 3); + rollingBackups.rollBackups(); + + assertThat(tempDir, hasNChildren(3)); + + String firstBackupFileContent = Files.readString(firstBackupPath); + assertThat(firstBackupFileContent, is("A")); + + String secondBackupFileContent = Files.readString(secondBackupPath); + assertThat(secondBackupFileContent, is("B")); + + String thirdBackupFileContent = Files.readString(thirdBackupPath); + assertThat(thirdBackupFileContent, is("C")); + } + + @Test + void threeBackupsFirstBackupMissing(@TempDir Path tempDir) throws IOException { + Files.writeString(baseFilePath, "A"); + + Path secondBackupPath = tempDir.resolve("file_1"); + Files.writeString(secondBackupPath, "C"); + + Path thirdBackupPath = tempDir.resolve("file_2"); + Files.writeString(thirdBackupPath, "D"); + + assertThat(tempDir, hasNChildren(3)); + + RollingBackups rollingBackups = new RollingBackups(baseFilePath.toFile(), 3); + rollingBackups.rollBackups(); + + assertThat(tempDir, hasNChildren(2)); + + Path firstBackupPath = tempDir.resolve("file_0"); + String firstBackupFileContent = Files.readString(firstBackupPath); + assertThat(firstBackupFileContent, is("A")); + + String thirdBackupFileContent = Files.readString(thirdBackupPath); + assertThat(thirdBackupFileContent, is("C")); + } + + @Test + void threeBackupsFileMissingInMiddles(@TempDir Path tempDir) throws IOException { + Files.writeString(baseFilePath, "A"); + + Path firstBackupPath = tempDir.resolve("file_0"); + Files.writeString(firstBackupPath, "B"); + + Path thirdBackupPath = tempDir.resolve("file_2"); + Files.writeString(thirdBackupPath, "D"); + + assertThat(tempDir, hasNChildren(3)); + + RollingBackups rollingBackups = new RollingBackups(baseFilePath.toFile(), 3); + rollingBackups.rollBackups(); + + assertThat(tempDir, hasNChildren(3)); + + String firstBackupFileContent = Files.readString(firstBackupPath); + assertThat(firstBackupFileContent, is("A")); + + Path secondBackupPath = tempDir.resolve("file_1"); + String secondBackupFileContent = Files.readString(secondBackupPath); + assertThat(secondBackupFileContent, is("B")); + + // Stays the same + String thirdBackupFileContent = Files.readString(thirdBackupPath); + assertThat(thirdBackupFileContent, is("D")); + } +}