Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Refine BackupManager #9054

Merged
merged 29 commits into from
Aug 15, 2022
Merged
Show file tree
Hide file tree
Changes from 6 commits
Commits
Show all changes
29 commits
Select commit Hold shift + click to select a range
8051bfd
Refine BackupManager
koppor Aug 13, 2022
83938bf
Fix BackupManager
koppor Aug 13, 2022
b607fff
Exchange "bak" and "sav"
koppor Aug 13, 2022
266f997
Refine backup message to show backup path
koppor Aug 13, 2022
dc5710a
Merge remote-tracking branch 'origin/main' into ten-saves-for-backup-…
koppor Aug 14, 2022
78065e2
Introduce BackupFileUtil
koppor Aug 14, 2022
3592ad4
Use AtomicFileWriter
koppor Aug 14, 2022
7c6fd38
Update src/main/java/org/jabref/logic/util/io/BackupFileUtil.java
koppor Aug 14, 2022
e7660e3
More comments
koppor Aug 14, 2022
b2da978
Align JabRefGUI#openLastEditedDatabases() with OpenDatabaseAction#loa…
koppor Aug 14, 2022
d26e218
Remove unused parameter "raisePanel"
koppor Aug 14, 2022
d8b48dd
WIP: Fix opening of databases i) on startup, ii) using File > Open
koppor Aug 14, 2022
9c5cac5
Experiment with needsBackup
koppor Aug 14, 2022
457c4a5
Fix variable name
koppor Aug 15, 2022
3f1ffc6
Remove obsolete method
koppor Aug 15, 2022
e171c04
Remove duplicate code
koppor Aug 15, 2022
b580577
Switch from DelayTaskThrottler to Java's ScheduledThreadPoolExecutor
koppor Aug 15, 2022
7aa1b76
Group together code cloned methods
koppor Aug 15, 2022
7f04fa3
Enable "needsBackup" strategy
koppor Aug 15, 2022
bf18715
Ensure that backup is a recent one
koppor Aug 15, 2022
dacd300
Fix checkstyle
koppor Aug 15, 2022
2f8a75e
Removed JabRefGUI import
calixtus Aug 15, 2022
2db1fab
Introdce OS.APP_DIR_APP_*
koppor Aug 15, 2022
1c45014
Fix checkstyle
koppor Aug 15, 2022
e722e3d
Merge remote-tracking branch 'origin/main' into ten-saves-for-backup-…
koppor Aug 15, 2022
c317aea
Try to fix tests
koppor Aug 15, 2022
122a236
Remove check for concrete hash code
koppor Aug 15, 2022
1e2c2fa
Fix checkstyle
koppor Aug 15, 2022
80797de
Refine comment on expected behavior
koppor Aug 15, 2022
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
3 changes: 3 additions & 0 deletions CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -11,13 +11,16 @@ Note that this project **does not** adhere to [Semantic Versioning](http://semve

### Added

- In case a backup is found, the filename of the backup is shown.
- We integrated a new three-way merge UI for merging entries in the Entries Merger Dialog, the Duplicate Resolver Dialog, the Entry Importer Dialog, and the External Changes Resolver Dialog. [#8945](https://github.com/JabRef/jabref/pull/8945)
- We added the ability to merge groups, keywords, comments and files when merging entries. [#9022](https://github.com/JabRef/jabref/pull/9022)

### Changed

- We improved the Citavi Importer to also import so called Knowledge-items into the field `comment` of the corresponding entry [#9025](https://github.com/JabRef/jabref/issues/9025)
- We removed wrapping of string constants when writing to a `.bib` file.
- We call backup files `.bak` and temporary writing files now `.sav`.
- JabRef keeps 10 older versions of a `.bib` file in the [user data dir](https://github.com/harawata/appdirs#supported-directories) (instead of a single `.sav` (now: `.bak`) file in the directory of the `.bib` file)
- We changed the button label from "Return to JabRef" to "Return to library" to better indicate the purpose of the action.

### Fixed
Expand Down
4 changes: 2 additions & 2 deletions src/main/java/org/jabref/gui/JabRefGUI.java
Original file line number Diff line number Diff line change
Expand Up @@ -77,8 +77,8 @@ private void openWindow(Stage mainStage) {
if (guiPreferences.isWindowMaximised()) {
mainStage.setMaximized(true);
} else if ((Screen.getScreens().size() == 1) && isWindowPositionOutOfBounds()) {
// corrects the Window, if its outside of the mainscreen
LOGGER.debug("The Jabref Window is outside the Main Monitor\n");
// corrects the Window, if it is outside the mainscreen
LOGGER.debug("The Jabref window is outside the main screen\n");
mainStage.setX(0);
mainStage.setY(0);
mainStage.setWidth(1024);
Expand Down
7 changes: 6 additions & 1 deletion src/main/java/org/jabref/gui/dialogs/BackupUIManager.java
Original file line number Diff line number Diff line change
Expand Up @@ -6,6 +6,8 @@
import org.jabref.gui.util.DefaultTaskExecutor;
import org.jabref.logic.autosaveandbackup.BackupManager;
import org.jabref.logic.l10n.Localization;
import org.jabref.logic.util.BackupFileType;
import org.jabref.logic.util.io.BackupFileUtil;

/**
* Stores all user dialogs related to {@link BackupManager}.
Expand All @@ -17,7 +19,10 @@ private BackupUIManager() {

public static void showRestoreBackupDialog(DialogService dialogService, Path originalPath) {
String content = new StringBuilder()
.append(Localization.lang("A backup file for '%0' was found.", originalPath.getFileName().toString()))
.append(Localization.lang("A backup file for '%0' was found at '%1'.",
originalPath.getFileName().toString(),
// We need to determine the path "manually" as the path does not get passed through when a diff is detected.
BackupFileUtil.getPathOfLatestExisingBackupFile(originalPath, BackupFileType.BACKUP).map(Path::toString).orElse(Localization.lang("File not found"))))
.append("\n")
.append(Localization.lang("This could indicate that JabRef did not shut down cleanly last time the file was used."))
.append("\n\n")
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -21,7 +21,7 @@ public class FileExtensionViewModel {

FileExtensionViewModel(FileType fileType, FilePreferences filePreferences) {
this.description = Localization.lang("%0 file", fileType.getName());
this.extensions = fileType.getExtensionsWithDot();
this.extensions = fileType.getExtensionsWithAsteriskAndDot();
this.filePreferences = filePreferences;
}

Expand Down
6 changes: 3 additions & 3 deletions src/main/java/org/jabref/gui/util/FileFilterConverter.java
Original file line number Diff line number Diff line change
Expand Up @@ -28,11 +28,11 @@ private FileFilterConverter() {

public static FileChooser.ExtensionFilter toExtensionFilter(FileType fileType) {
String description = Localization.lang("%0 file", fileType.toString());
return new FileChooser.ExtensionFilter(description, fileType.getExtensionsWithDot());
return new FileChooser.ExtensionFilter(description, fileType.getExtensionsWithAsteriskAndDot());
}

public static FileChooser.ExtensionFilter toExtensionFilter(String description, FileType fileType) {
return new FileChooser.ExtensionFilter(description, fileType.getExtensionsWithDot());
return new FileChooser.ExtensionFilter(description, fileType.getExtensionsWithAsteriskAndDot());
}

public static Optional<Importer> getImporter(FileChooser.ExtensionFilter extensionFilter, Collection<Importer> importers) {
Expand All @@ -46,7 +46,7 @@ public static Optional<Exporter> getExporter(FileChooser.ExtensionFilter extensi
public static FileChooser.ExtensionFilter forAllImporters(SortedSet<Importer> importers) {
List<FileType> fileTypes = importers.stream().map(Importer::getFileType).collect(Collectors.toList());
List<String> flatExtensions = fileTypes.stream()
.flatMap(type -> type.getExtensionsWithDot().stream())
.flatMap(type -> type.getExtensionsWithAsteriskAndDot().stream())
.collect(Collectors.toList());

return new FileChooser.ExtensionFilter(Localization.lang("Available import formats"), flatExtensions);
Expand Down
106 changes: 79 additions & 27 deletions src/main/java/org/jabref/logic/autosaveandbackup/BackupManager.java
Original file line number Diff line number Diff line change
@@ -1,25 +1,29 @@
package org.jabref.logic.autosaveandbackup;

import java.io.IOException;
import java.io.Writer;
import java.nio.charset.Charset;
import java.nio.charset.StandardCharsets;
import java.nio.file.Files;
import java.nio.file.Path;
import java.nio.file.StandardCopyOption;
import java.util.HashSet;
import java.util.List;
import java.util.Optional;
import java.util.Queue;
import java.util.Set;
import java.util.concurrent.BlockingQueue;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.LinkedBlockingQueue;

import org.jabref.logic.bibtex.InvalidFieldValueException;
import org.jabref.logic.exporter.AtomicFileWriter;
import org.jabref.logic.exporter.BibWriter;
import org.jabref.logic.exporter.BibtexDatabaseWriter;
import org.jabref.logic.exporter.SavePreferences;
import org.jabref.logic.util.BackupFileType;
import org.jabref.logic.util.CoarseChangeFilter;
import org.jabref.logic.util.DelayTaskThrottler;
import org.jabref.logic.util.io.FileUtil;
import org.jabref.logic.util.io.BackupFileUtil;
import org.jabref.model.database.BibDatabaseContext;
import org.jabref.model.database.event.BibDatabaseContextChangedEvent;
import org.jabref.model.entry.BibEntryTypesManager;
Expand All @@ -40,8 +44,7 @@ public class BackupManager {

private static final Logger LOGGER = LoggerFactory.getLogger(BackupManager.class);

// This differs from org.jabref.logic.exporter.AtomicFileOutputStream.BACKUP_EXTENSION, which is used for copying the .bib away before overwriting on save.
private static final String AUTOSAVE_FILE_EXTENSION = ".sav";
private static final int MAXIMUM_BACKUP_FILE_COUNT = 10;

private static Set<BackupManager> runningInstances = new HashSet<>();

Expand All @@ -51,6 +54,11 @@ public class BackupManager {
private final CoarseChangeFilter changeFilter;
private final BibEntryTypesManager entryTypesManager;


// Contains a list of all backup paths
// During a write, the less recent backup file is deleted
private final Queue<Path> backupFilesQueue = new LinkedBlockingQueue<>();

private BackupManager(BibDatabaseContext bibDatabaseContext, BibEntryTypesManager entryTypesManager, PreferencesService preferences) {
this.bibDatabaseContext = bibDatabaseContext;
this.entryTypesManager = entryTypesManager;
Expand All @@ -61,8 +69,18 @@ private BackupManager(BibDatabaseContext bibDatabaseContext, BibEntryTypesManage
changeFilter.registerListener(this);
}

static Path getBackupPath(Path originalPath) {
return FileUtil.addExtension(originalPath, AUTOSAVE_FILE_EXTENSION);
/**
* Determines the most recent backup file name
*/
static Path getBackupPathForNewBackup(Path originalPath) {
return BackupFileUtil.getPathForNewBackupFileAndCreateDirectory(originalPath, BackupFileType.BACKUP);
}

/**
* Determines the most recent existing backup file name
*/
static Optional<Path> getLatestBackupPath(Path originalPath) {
return BackupFileUtil.getPathOfLatestExisingBackupFile(originalPath, BackupFileType.BACKUP);
}

/**
Expand Down Expand Up @@ -99,13 +117,13 @@ public static void shutdown(BibDatabaseContext bibDatabaseContext) {
* the user checks the output.
*/
public static boolean backupFileDiffers(Path originalPath) {
Path backupPath = getBackupPath(originalPath);
if (!Files.exists(backupPath) || Files.isDirectory(backupPath)) {
Optional<Path> backupPath = getLatestBackupPath(originalPath);
if (backupPath.isEmpty()) {
return false;
}

try {
return Files.mismatch(originalPath, backupPath) != -1L;
return Files.mismatch(originalPath, backupPath.get()) != -1L;
} catch (IOException e) {
LOGGER.debug("Could not compare original file and backup file.", e);
// User has to investigate in this case
Expand All @@ -119,28 +137,50 @@ public static boolean backupFileDiffers(Path originalPath) {
* @param originalPath Path to the file which should be equalized to the backup file.
*/
public static void restoreBackup(Path originalPath) {
Path backupPath = getBackupPath(originalPath);
Optional<Path> backupPath = getLatestBackupPath(originalPath);
if (backupPath.isEmpty()) {
LOGGER.error("There is no backup file");
return;
}
try {
Files.copy(backupPath, originalPath, StandardCopyOption.REPLACE_EXISTING);
Files.copy(backupPath.get(), originalPath, StandardCopyOption.REPLACE_EXISTING);
} catch (IOException e) {
LOGGER.error("Error while restoring the backup file.", e);
}
}

private Optional<Path> determineBackupPath() {
return bibDatabaseContext.getDatabasePath().map(BackupManager::getBackupPath);
private Optional<Path> determineBackupPathForNewBackup() {
return bibDatabaseContext.getDatabasePath().map(BackupManager::getBackupPathForNewBackup);
}

/**
* This method is called as soon as the scheduler says: "Do the backup"
*
* <em>SIDE EFFECT: Deletes oldest backup file</em>
*
* @param backupPath the path where the library should be backed up to
*/
private void performBackup(Path backupPath) {
// We opted for "while" to delete backups in case there are more than 10
while (backupFilesQueue.size() >= MAXIMUM_BACKUP_FILE_COUNT) {
Path lessRecentBackupFile = backupFilesQueue.poll();
try {
Files.delete(lessRecentBackupFile);
} catch (IOException e) {
LOGGER.error("Could not delete backup file {}", lessRecentBackupFile, e);
}
}

// code similar to org.jabref.gui.exporter.SaveDatabaseAction.saveDatabase
GeneralPreferences generalPreferences = preferences.getGeneralPreferences();
SavePreferences savePreferences = preferences.getSavePreferences()
.withMakeBackup(false);
Charset encoding = bibDatabaseContext.getMetaData().getEncoding().orElse(StandardCharsets.UTF_8);
try (AtomicFileWriter fileWriter = new AtomicFileWriter(backupPath, encoding)) {
BibWriter bibWriter = new BibWriter(fileWriter, bibDatabaseContext.getDatabase().getNewLineSeparator());
try (Writer writer = Files.newBufferedWriter(backupPath, encoding)) {
BibWriter bibWriter = new BibWriter(writer, bibDatabaseContext.getDatabase().getNewLineSeparator());
new BibtexDatabaseWriter(bibWriter, generalPreferences, savePreferences, entryTypesManager)
.saveDatabase(bibDatabaseContext);
backupFilesQueue.add(backupPath);
} catch (IOException e) {
logIfCritical(backupPath, e);
}
Expand All @@ -167,7 +207,30 @@ public synchronized void listen(@SuppressWarnings("unused") BibDatabaseContextCh
}

private void startBackupTask() {
throttler.schedule(() -> determineBackupPath().ifPresent(this::performBackup));
fillQueue();

// We need to determine the backup path on each action, because the user might have saved the file to a different location
throttler.schedule(() -> determineBackupPathForNewBackup().ifPresent(this::performBackup));
}

private void fillQueue() {
Path backupDir = BackupFileUtil.getAppDataBackupDir();
if (!Files.exists(backupDir)) {
return;
}
bibDatabaseContext.getDatabasePath().ifPresent(databasePath -> {
// code similar to {@link org.jabref.logic.util.io.BackupFileUtil.getPathOfLatestExisingBackupFile}
final String prefix = BackupFileUtil.getUniqueFilePrefix(databasePath) + "--" + databasePath.getFileName();
try {
List<Path> allSavFiles = Files.list(backupDir)
// just list the .sav belonging to the given targetFile
.filter(p -> p.getFileName().toString().startsWith(prefix))
.sorted().toList();
backupFilesQueue.addAll(allSavFiles);
} catch (IOException e) {
LOGGER.error("Could not determine most recent file", e);
}
});
}

/**
Expand All @@ -178,16 +241,5 @@ private void shutdown() {
changeFilter.unregisterListener(this);
changeFilter.shutdown();
throttler.shutdown();
determineBackupPath().ifPresent(this::deleteBackupFile);
}

private void deleteBackupFile(Path backupPath) {
try {
if (Files.exists(backupPath) && !Files.isDirectory(backupPath)) {
Files.delete(backupPath);
}
} catch (IOException e) {
LOGGER.error("Error while deleting the backup file.", e);
}
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -12,6 +12,7 @@
import java.util.EnumSet;
import java.util.Set;

import org.jabref.logic.util.BackupFileType;
import org.jabref.logic.util.io.FileUtil;

import org.slf4j.Logger;
Expand Down Expand Up @@ -47,7 +48,7 @@ public class AtomicFileOutputStream extends FilterOutputStream {
private static final Logger LOGGER = LoggerFactory.getLogger(AtomicFileOutputStream.class);

private static final String TEMPORARY_EXTENSION = ".tmp";
private static final String BACKUP_EXTENSION = ".bak";
private static final String SAVE_EXTENSION = BackupFileType.SAVE.getExtensions().get(0);

/**
* The file we want to create/replace.
Expand Down Expand Up @@ -76,7 +77,7 @@ public AtomicFileOutputStream(Path path, boolean keepBackup) throws IOException

this.targetFile = path;
this.temporaryFile = getPathOfTemporaryFile(path);
this.backupFile = getPathOfBackupFile(path);
this.backupFile = getPathOfSaveBackupFile(path);
this.keepBackup = keepBackup;

try {
Expand Down Expand Up @@ -104,8 +105,8 @@ private static Path getPathOfTemporaryFile(Path targetFile) {
return FileUtil.addExtension(targetFile, TEMPORARY_EXTENSION);
}

private static Path getPathOfBackupFile(Path targetFile) {
return FileUtil.addExtension(targetFile, BACKUP_EXTENSION);
private static Path getPathOfSaveBackupFile(Path targetFile) {
return FileUtil.addExtension(targetFile, SAVE_EXTENSION);
}

/**
Expand Down
32 changes: 32 additions & 0 deletions src/main/java/org/jabref/logic/util/BackupFileType.java
Original file line number Diff line number Diff line change
@@ -0,0 +1,32 @@
package org.jabref.logic.util;

import java.util.Collections;
import java.util.List;

public enum BackupFileType implements FileType {

// Used at BackupManager
BACKUP("Backup", "bak"),

// Used when writing the .bib file. See {@link org.jabref.logic.exporter.AtomicFileWriter}
// Used for copying the .bib away before overwriting on save.
SAVE("AutoSaveFile", "sav");

private final List<String> extensions;
private final String name;

BackupFileType(String name, String extension) {
this.extensions = Collections.singletonList(extension);
this.name = name;
}

@Override
public List<String> getExtensions() {
return extensions;
}

@Override
public String getName() {
return this.name;
}
}
2 changes: 1 addition & 1 deletion src/main/java/org/jabref/logic/util/FileType.java
Original file line number Diff line number Diff line change
Expand Up @@ -8,7 +8,7 @@
*/
public interface FileType {

default List<String> getExtensionsWithDot() {
default List<String> getExtensionsWithAsteriskAndDot() {
return getExtensions().stream()
.map(extension -> "*." + extension)
.collect(Collectors.toList());
Expand Down
Loading