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

[WIP] Fix database sync #11879

Open
wants to merge 57 commits into
base: main
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from 10 commits
Commits
Show all changes
57 commits
Select commit Hold shift + click to select a range
443765f
Refactor method signature
koppor Oct 3, 2024
f3921c2
Rename maps
koppor Oct 3, 2024
92a2287
Cleanup EntriesAddedEvent
koppor Oct 3, 2024
f0465a6
Clean up EntriesRemovedEvent
koppor Oct 3, 2024
95fe7f6
Disable save actions for SQL
koppor Oct 3, 2024
279da5b
Fix logger string
koppor Oct 3, 2024
e19bc1a
Add TODO
koppor Oct 3, 2024
90e067d
Remove MySQL and Oracle dependencies
koppor Oct 3, 2024
35d6411
Remove PullChangesFromSharedAction
koppor Oct 3, 2024
f4f40c6
Improve doc and code comments
koppor Oct 3, 2024
7e459be
Mark variable to be removed
koppor Oct 4, 2024
f901151
Fix class name
koppor Oct 4, 2024
f6a2fe5
Introduce BibEntry.ENTRY_LINK_SEPARATOR
koppor Oct 4, 2024
8a11cb9
Cleanup code in CitationKeyListener
koppor Oct 4, 2024
fd958b6
More Java 8
koppor Oct 4, 2024
fd7697b
Minor code updates
koppor Oct 4, 2024
770cef0
Begin: separate metadata sync with all-entry-sync
koppor Oct 4, 2024
2c8faa7
Use "ON CONFLICT" of PostgreSQL
koppor Oct 4, 2024
dcf605f
Remove non-used get
koppor Oct 4, 2024
55bb387
Fix language
koppor Oct 4, 2024
fd3c184
Fix module-info.java
koppor Oct 4, 2024
5c863dc
Switch from "external" Postgres to embedded Postgres
koppor Oct 4, 2024
7e0ba3e
Remove escape and escape_Table (and add indexes)
koppor Oct 4, 2024
407b307
Fix method name "isFiltered" and also link reasoning
koppor Oct 4, 2024
f19101b
Merge branch 'main' into fix-sync
Siedlerchr Oct 4, 2024
5a36849
Integrate PostgreSQLProcessor in DBMSProcessor
koppor Oct 4, 2024
c1dda85
Let pgConnection do the waiting work (not the Java thread)
koppor Oct 4, 2024
323118f
Compilefix
koppor Oct 4, 2024
9f099ed
Change signature
koppor Oct 4, 2024
98242cf
Improve variable name
koppor Oct 4, 2024
dd6fb1b
Fix package name (and class name) for notifications
koppor Oct 4, 2024
451c971
Reorder methods
koppor Oct 4, 2024
2577ac6
Switch from UUID to CUID2
koppor Oct 4, 2024
b17aa35
Reorder fields
koppor Oct 4, 2024
76da442
Remove unused method
koppor Oct 4, 2024
72ed0bb
Write "Id" (instead of ID)
koppor Oct 4, 2024
c1fbcc4
Rename "Revision" to "Version"
koppor Oct 4, 2024
5b83b84
Streamline insertEntries
koppor Oct 4, 2024
6c1c2b0
Begin to revine remote-storage-sql.md
koppor Oct 5, 2024
2e66e10
Merge branch 'fix-sync' of github.com:JabRef/jabref into fix-sync
koppor Oct 7, 2024
383c665
Merge branch 'main' into fix-sync
koppor Oct 7, 2024
5bc7abd
Some minor code comments and SQL optimization
koppor Oct 7, 2024
f919bbd
Merge branch 'main' into fix-sync
calixtus Oct 7, 2024
c901b84
fix compile errors
Siedlerchr Oct 8, 2024
d2afcb4
Merge remote-tracking branch 'upstream/main' into fix-sync
Siedlerchr Oct 8, 2024
cedafb1
Merge branch 'fix-sync' of github.com:JabRef/jabref into fix-sync
koppor Oct 8, 2024
d1fd3f0
WIP: notificatoin
koppor Oct 9, 2024
e4c7130
Merge branch 'main' into fix-sync
koppor Oct 9, 2024
598d83e
Fix compile error (and typo)
koppor Oct 9, 2024
a78afc9
Add databaseName as parameter
koppor Oct 10, 2024
7eb4b07
Compilefix
koppor Oct 10, 2024
a9a2708
Merge branch 'fix-sync' of github.com:JabRef/jabref into fix-sync
koppor Oct 10, 2024
bc68c19
Remove unnecessary JavaDoc
koppor Oct 10, 2024
dd64830
Fix tests
koppor Oct 10, 2024
801acdf
Merge branch 'main' into fix-sync
koppor Nov 7, 2024
1b896d0
Compile fix
koppor Nov 7, 2024
2dd2beb
Merge branch 'main' into fix-sync
koppor Nov 12, 2024
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
19 changes: 2 additions & 17 deletions .github/workflows/tests.yml
Original file line number Diff line number Diff line change
Expand Up @@ -210,6 +210,8 @@ jobs:
# needed because the postgres container does not provide a healthcheck
options: --health-cmd pg_isready --health-interval 10s --health-timeout 5s --health-retries 5
steps:
- name: Shutdown Ubuntu MySQL
run: sudo service mysql stop # Shutdown the Default MySQL to save memory, "sudo" is necessary, please do not remove it
- name: Checkout source
uses: actions/checkout@v4
with:
Expand All @@ -229,23 +231,6 @@ jobs:
env:
CI: "true"
DBMS: "postgresql"
- name: Shutdown Ubuntu MySQL
run: sudo service mysql stop # Shutdown the Default MySQL, "sudo" is necessary, please not remove it
- name: Start custom MySQL
uses: mirromutth/[email protected]
with:
host port: 3800
container port: 3307
character set server: 'utf8'
collation server: 'utf8_general_ci'
mysql version: '8.0'
mysql database: 'jabref'
mysql root password: 'root'
- name: Run tests on MySQL
run: ./gradlew databaseTest --rerun-tasks
env:
CI: "true"
DBMS: "mysql"

guitests:
name: GUI tests
Expand Down
2 changes: 1 addition & 1 deletion CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -89,7 +89,7 @@ Note that this project **does not** adhere to [Semantic Versioning](https://semv
- We removed support for case-sensitive and exact search. [#11542](https://github.com/JabRef/jabref/pull/11542)
- We removed the description of search strings. [#11542](https://github.com/JabRef/jabref/pull/11542)
- We removed support for importing using the SilverPlatterImporter (`Record INSPEC`). [#11576](https://github.com/JabRef/jabref/pull/11576)

- We removed support for MySQL/MariaDB and Oracle.



Expand Down
24 changes: 0 additions & 24 deletions build.gradle
Original file line number Diff line number Diff line change
Expand Up @@ -205,18 +205,10 @@ dependencies {

implementation 'com.fasterxml:aalto-xml:1.3.3'

implementation group: 'org.mariadb.jdbc', name: 'mariadb-java-client', version: '2.7.9'

implementation 'org.postgresql:postgresql:42.7.4'

// Support unix socket connection types
implementation 'com.kohlschutter.junixsocket:junixsocket-core:2.10.1'
implementation 'com.kohlschutter.junixsocket:junixsocket-mysql:2.10.0'

implementation ('com.oracle.ojdbc:ojdbc10:19.3.0.0') {
// causing module issues
exclude module: 'oraclepki'
}

implementation ('com.google.guava:guava:33.1.0-jre') {
// TODO: Remove this as soon as https://github.com/google/guava/issues/2960 is fixed
Expand Down Expand Up @@ -803,24 +795,8 @@ jlink {
uses 'kong.unirest.core.json.JsonEngine'
uses 'org.eclipse.jgit.lib.Signer'
uses 'org.eclipse.jgit.transport.SshSessionFactory'
uses 'org.mariadb.jdbc.LocalInfileInterceptor'
uses 'org.mariadb.jdbc.authentication.AuthenticationPlugin'
uses 'org.mariadb.jdbc.credential.CredentialPlugin'
uses 'org.mariadb.jdbc.tls.TlsSocketPlugin'

provides 'org.mariadb.jdbc.tls.TlsSocketPlugin' with 'org.mariadb.jdbc.internal.protocol.tls.DefaultTlsSocketPlugin'
provides 'java.sql.Driver' with 'org.postgresql.Driver'
provides 'org.mariadb.jdbc.authentication.AuthenticationPlugin' with 'org.mariadb.jdbc.internal.com.send.authentication.CachingSha2PasswordPlugin',
'org.mariadb.jdbc.internal.com.send.authentication.ClearPasswordPlugin',
'org.mariadb.jdbc.internal.com.send.authentication.Ed25519PasswordPlugin',
'org.mariadb.jdbc.internal.com.send.authentication.NativePasswordPlugin',
'org.mariadb.jdbc.internal.com.send.authentication.OldPasswordPlugin',
'org.mariadb.jdbc.internal.com.send.authentication.SendGssApiAuthPacket',
'org.mariadb.jdbc.internal.com.send.authentication.SendPamAuthPacket',
'org.mariadb.jdbc.internal.com.send.authentication.Sha256PasswordPlugin'
provides 'org.mariadb.jdbc.credential.CredentialPlugin' with 'org.mariadb.jdbc.credential.aws.AwsIamCredentialPlugin',
'org.mariadb.jdbc.credential.env.EnvCredentialPlugin',
'org.mariadb.jdbc.credential.system.PropertiesCredentialPlugin'
provides 'java.security.Provider' with 'org.bouncycastle.jce.provider.BouncyCastleProvider',
'org.bouncycastle.pqc.jcajce.provider.BouncyCastlePQCProvider'
provides 'kong.unirest.core.json.JsonEngine' with 'kong.unirest.modules.gson.GsonEngine';
Expand Down
15 changes: 13 additions & 2 deletions docs/code-howtos/remote-storage-sql.md
Original file line number Diff line number Diff line change
Expand Up @@ -9,14 +9,25 @@ For user documentation, see <https://docs.jabref.org/collaborative-work/sqldatab

## Handling large shared databases

Synchronization times may get long when working with a large database containing several thousand entries. Therefore, synchronization only happens if several conditions are fulfilled:
Synchronization times may get long when working with a large database containing several thousand entries.
Therefore, we use PostgreSQL's `LISTEN` and `NOTIFY` commands to inform the client about changes in the database on an entry level.

Background reading: <https://www.baeldung.com/spring-postgresql-message-broker>.

## Handling synchronization of "micro-edits"

It causes too much load both on the server and at all subscribed clients to synchronize every single letter change.
Therefore, synchronization only happens if several conditions are fulfilled:

* Edit to another field.
* Major changes have been made (pasting or deleting more than one character).

Class `org.jabref.logic.util.CoarseChangeFilter.java` checks both conditions.

Remaining changes that have not been synchronized yet are saved at closing the database rendering additional closing time. Saving is realized in `org.jabref.logic.shared.DBMSSynchronizer.java`. Following methods account for synchronization modes:
Remaining changes that have not been synchronized yet are saved at closing the database rendering additional closing time.
Saving is realized in `org.jabref.logic.shared.DBMSSynchronizer.java`.

Following methods account for synchronization modes:

* `pullChanges` synchronizes the database unconditionally.
* `pullLastEntryChanges` synchronizes only if there are remaining entry changes. It is invoked when closing the shared database (`closeSharedDatabase`).
Expand Down
1 change: 0 additions & 1 deletion src/main/java/org/jabref/gui/actions/StandardActions.java
Original file line number Diff line number Diff line change
Expand Up @@ -75,7 +75,6 @@ public enum StandardActions implements Action {
REMOTE_DB(Localization.lang("Shared database"), IconTheme.JabRefIcons.REMOTE_DATABASE),
EXPORT_SELECTED(Localization.lang("Export selected entries"), KeyBinding.EXPORT_SELECTED),
CONNECT_TO_SHARED_DB(Localization.lang("Connect to shared database"), IconTheme.JabRefIcons.CONNECT_DB),
PULL_CHANGES_FROM_SHARED_DB(Localization.lang("Pull changes from shared database"), KeyBinding.PULL_CHANGES_FROM_SHARED_DATABASE),
CLOSE_LIBRARY(Localization.lang("Close library"), Localization.lang("Close the current library"), IconTheme.JabRefIcons.CLOSE, KeyBinding.CLOSE_DATABASE),
CLOSE_OTHER_LIBRARIES(Localization.lang("Close others"), Localization.lang("Close other libraries"), IconTheme.JabRefIcons.CLOSE),
CLOSE_ALL_LIBRARIES(Localization.lang("Close all"), Localization.lang("Close all libraries"), IconTheme.JabRefIcons.CLOSE),
Expand Down
4 changes: 1 addition & 3 deletions src/main/java/org/jabref/gui/frame/MainMenu.java
Original file line number Diff line number Diff line change
Expand Up @@ -62,7 +62,6 @@
import org.jabref.gui.push.PushToApplicationCommand;
import org.jabref.gui.search.RebuildFulltextSearchIndexAction;
import org.jabref.gui.shared.ConnectToSharedDatabaseCommand;
import org.jabref.gui.shared.PullChangesFromSharedAction;
import org.jabref.gui.sidepane.SidePane;
import org.jabref.gui.sidepane.SidePaneType;
import org.jabref.gui.slr.EditExistingStudyAction;
Expand Down Expand Up @@ -171,8 +170,7 @@ private void createMenu() {
new SeparatorMenuItem(),

factory.createSubMenu(StandardActions.REMOTE_DB,
factory.createMenuItem(StandardActions.CONNECT_TO_SHARED_DB, new ConnectToSharedDatabaseCommand(frame, dialogService)),
factory.createMenuItem(StandardActions.PULL_CHANGES_FROM_SHARED_DB, new PullChangesFromSharedAction(stateManager))),
factory.createMenuItem(StandardActions.CONNECT_TO_SHARED_DB, new ConnectToSharedDatabaseCommand(frame, dialogService))),

new SeparatorMenuItem(),

Expand Down
1 change: 0 additions & 1 deletion src/main/java/org/jabref/gui/keyboard/KeyBinding.java
Original file line number Diff line number Diff line change
Expand Up @@ -90,7 +90,6 @@ public enum KeyBinding {
SHOW_PREFS("Preferences", Localization.lang("Preferences"), "ctrl+,", KeyBindingCategory.FILE),
OPEN_URL_OR_DOI("Open URL or DOI", Localization.lang("Open URL or DOI"), "F3", KeyBindingCategory.TOOLS),
PASTE("Paste", Localization.lang("Paste"), "ctrl+V", KeyBindingCategory.EDIT),
PULL_CHANGES_FROM_SHARED_DATABASE("Pull changes from shared database", Localization.lang("Pull changes from shared database"), "ctrl+shift+R", KeyBindingCategory.FILE),
PREVIOUS_PREVIEW_LAYOUT("Previous preview layout", Localization.lang("Previous preview layout"), "shift+F9", KeyBindingCategory.VIEW),
PREVIOUS_LIBRARY("Previous library", Localization.lang("Previous library"), "ctrl+PAGE_UP", KeyBindingCategory.VIEW),
SCROLL_TO_NEXT_MATCH_CATEGORY("Scroll to next match category", Localization.lang("Scroll to next match category"), "right", KeyBindingCategory.VIEW),
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -18,7 +18,6 @@ public Map<KeyBinding, String> getKeyBindings() {
final Map<KeyBinding, String> keyBindings = new HashMap<>();

// Clear conflicting default presets
keyBindings.put(KeyBinding.PULL_CHANGES_FROM_SHARED_DATABASE, "");
keyBindings.put(KeyBinding.COPY_PREVIEW, "");

// Add new entry presets
Expand Down

This file was deleted.

29 changes: 9 additions & 20 deletions src/main/java/org/jabref/logic/shared/DBMSProcessor.java
Original file line number Diff line number Diff line change
Expand Up @@ -59,16 +59,12 @@ public boolean checkBaseIntegrity() throws SQLException {
boolean databasePassesIntegrityCheck = false;
DBMSType type = this.connectionProperties.getType();
Map<String, String> metadata = getSharedMetaData();
if (type == DBMSType.POSTGRESQL || type == DBMSType.MYSQL) {
String metadataVersion = metadata.get(MetaData.VERSION_DB_STRUCT);
if (metadataVersion != null) {
int VERSION_DB_STRUCT = Integer.parseInt(metadata.getOrDefault(MetaData.VERSION_DB_STRUCT, "").replace(";", ""));
if (VERSION_DB_STRUCT == getCURRENT_VERSION_DB_STRUCT()) {
databasePassesIntegrityCheck = true;
}
String metadataVersion = metadata.get(MetaData.VERSION_DB_STRUCT);
if (metadataVersion != null) {
int VERSION_DB_STRUCT = Integer.parseInt(metadata.getOrDefault(MetaData.VERSION_DB_STRUCT, "").replace(";", ""));
if (VERSION_DB_STRUCT == getCURRENT_VERSION_DB_STRUCT()) {
databasePassesIntegrityCheck = true;
}
} else {
databasePassesIntegrityCheck = checkTableAvailability("ENTRY", "FIELD", "METADATA");
}
return databasePassesIntegrityCheck;
}
Expand Down Expand Up @@ -291,7 +287,7 @@ protected void insertIntoFieldTable(List<BibEntry> bibEntries) {
preparedFieldStatement.executeUpdate();
}
} catch (SQLException e) {
LOGGER.error("SQL Error: ", e);
LOGGER.error("SQL Error", e);
}
}

Expand All @@ -302,6 +298,7 @@ protected void insertIntoFieldTable(List<BibEntry> bibEntries) {
* @throws SQLException in case of error
*/
public void updateEntry(BibEntry localBibEntry) throws OfflineLockException, SQLException {
// FIXME: either two connections (one with auto commit and one without) or better auto commit state - this line here can lead to issues if autocommit is required in a parallel thread
connection.setAutoCommit(false); // disable auto commit due to transaction

try {
Expand Down Expand Up @@ -345,7 +342,7 @@ public void updateEntry(BibEntry localBibEntry) throws OfflineLockException, SQL
throw new OfflineLockException(localBibEntry, sharedBibEntry);
}
} catch (SQLException e) {
LOGGER.error("SQL Error: ", e);
LOGGER.error("SQL Error", e);
connection.rollback(); // undo changes made in current transaction
} finally {
connection.setAutoCommit(true); // enable auto commit mode again
Expand Down Expand Up @@ -666,15 +663,7 @@ public void setSharedMetaData(Map<String, String> data) throws SQLException {
* Returns a new instance of the abstract type {@link DBMSProcessor}
*/
public static DBMSProcessor getProcessorInstance(DatabaseConnection connection) {
DBMSType type = connection.getProperties().getType();
if (type == DBMSType.MYSQL) {
return new MySQLProcessor(connection);
} else if (type == DBMSType.POSTGRESQL) {
return new PostgreSQLProcessor(connection);
} else if (type == DBMSType.ORACLE) {
return new OracleProcessor(connection);
}
return null; // can never happen except new types were added without updating this method.
return new PostgreSQLProcessor(connection);
}

public DatabaseConnectionProperties getDBMSConnectionProperties() {
Expand Down
6 changes: 5 additions & 1 deletion src/main/java/org/jabref/logic/shared/DBMSSynchronizer.java
Original file line number Diff line number Diff line change
Expand Up @@ -80,7 +80,9 @@ public void listen(EntriesAddedEvent event) {
// While synchronizing the local database (see synchronizeLocalDatabase() below), some EntriesEvents may be posted.
// In this case DBSynchronizer should not try to insert the bibEntry entry again (but it would not harm).
if (isEventSourceAccepted(event) && checkCurrentConnection()) {
// TODO: Make use of org.jabref.model.metadata.event.MetaDataChangedEvent (and also do Added/Removed there)
synchronizeLocalMetaData();

pullWithLastEntry();
synchronizeLocalDatabase();
dbmsProcessor.insertEntries(event.getBibEntries());
Expand Down Expand Up @@ -247,7 +249,9 @@ public void synchronizeSharedEntry(BibEntry bibEntry) {
return;
}
try {
BibDatabaseWriter.applySaveActions(bibEntry, metaData, fieldPreferences); // perform possibly existing save actions
// TODO: Either reenable (compare with fix https://github.com/JabRef/jabref/pull/11282) or write user documentation that "cleanup actions" should be used or introduce SQL databse cleanup (stored procedure, ...)
// BibDatabaseWriter.applySaveActions(bibEntry, metaData, fieldPreferences); // perform possibly existing save actions

dbmsProcessor.updateEntry(bibEntry);
} catch (OfflineLockException exception) {
eventBus.post(new UpdateRefusedEvent(bibDatabaseContext, exception.getLocalBibEntry(), exception.getSharedBibEntry()));
Expand Down
4 changes: 1 addition & 3 deletions src/main/java/org/jabref/logic/shared/DBMSType.java
Original file line number Diff line number Diff line change
Expand Up @@ -7,9 +7,7 @@
* Enumerates all supported database systems (DBMS) by JabRef.
*/
public enum DBMSType {
POSTGRESQL("PostgreSQL", "org.postgresql.Driver", "jdbc:postgresql://%s:%d/%s", 5432),
MYSQL("MySQL", "org.mariadb.jdbc.Driver", "jdbc:mariadb://%s:%d/%s", 3306),
ORACLE("Oracle", "oracle.jdbc.driver.OracleDriver", "jdbc:oracle:thin:@%s:%d/%s", 1521);
POSTGRESQL("PostgreSQL", "org.postgresql.Driver", "jdbc:postgresql://%s:%d/%s", 5432);

private final String type;
private final String driverPath;
Expand Down
86 changes: 0 additions & 86 deletions src/main/java/org/jabref/logic/shared/MySQLProcessor.java

This file was deleted.

Loading
Loading