mirror of
https://github.com/streetwriters/notesnook.git
synced 2025-12-25 16:09:42 +01:00
132 lines
3.7 KiB
TypeScript
132 lines
3.7 KiB
TypeScript
/*
|
|
This file is part of the Notesnook project (https://notesnook.com/)
|
|
|
|
Copyright (C) 2023 Streetwriters (Private) Limited
|
|
|
|
This program is free software: you can redistribute it and/or modify
|
|
it under the terms of the GNU General Public License as published by
|
|
the Free Software Foundation, either version 3 of the License, or
|
|
(at your option) any later version.
|
|
|
|
This program 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 General Public License for more details.
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
import NetInfo from "@react-native-community/netinfo";
|
|
import { db } from "../common/database";
|
|
import { DatabaseLogger } from "../common/database/index";
|
|
import { initAfterSync } from "../stores/index";
|
|
import { SyncStatus, useUserStore } from "../stores/use-user-store";
|
|
import BackgroundSync from "./background-sync";
|
|
import { ToastManager } from "./event-manager";
|
|
import SettingsService from "./settings";
|
|
|
|
export const ignoredMessages = [
|
|
"Sync already running",
|
|
"Not allowed to start service intent",
|
|
"WebSocket failed to connect",
|
|
"Failed to start the HttpConnection before",
|
|
"Could not connect to the Sync server",
|
|
"Network request failed"
|
|
];
|
|
let pendingSync: any = undefined;
|
|
let syncTimer: NodeJS.Timeout;
|
|
let lastSyncType = "full";
|
|
|
|
const run = async (
|
|
context = "global",
|
|
forced = false,
|
|
type: "full" | "send" | "fetch" = "full",
|
|
onCompleted?: (status?: number) => void,
|
|
lastSyncTime?: number
|
|
) => {
|
|
if (useUserStore.getState().syncing) {
|
|
DatabaseLogger.info("Sync in progress");
|
|
pendingSync = {
|
|
forced,
|
|
type: type,
|
|
context: context,
|
|
onCompleted,
|
|
lastSyncTime
|
|
};
|
|
return;
|
|
}
|
|
|
|
if (pendingSync) {
|
|
pendingSync = undefined;
|
|
DatabaseLogger.info("Running pending sync...");
|
|
}
|
|
|
|
clearTimeout(syncTimer);
|
|
syncTimer = setTimeout(async () => {
|
|
const userstore = useUserStore.getState();
|
|
userstore.setSyncing(true);
|
|
const user = await db.user.getUser();
|
|
|
|
if (!user || SettingsService.get().disableSync) {
|
|
userstore.setSyncing(false);
|
|
initAfterSync(type != "send" ? "full" : "send");
|
|
pendingSync = undefined;
|
|
return onCompleted?.(SyncStatus.Failed);
|
|
}
|
|
|
|
lastSyncType = type != "send" ? "full" : "send";
|
|
|
|
let error: Error | undefined = undefined;
|
|
|
|
try {
|
|
await BackgroundSync.doInBackground(async () => {
|
|
try {
|
|
await db.sync({
|
|
type: type,
|
|
force: forced,
|
|
offlineMode: SettingsService.get().offlineMode
|
|
});
|
|
} catch (e) {
|
|
error = e as Error;
|
|
}
|
|
});
|
|
|
|
if (error) throw error;
|
|
} catch (e) {
|
|
error = e as Error;
|
|
DatabaseLogger.error(error, "[Client] Failed to sync");
|
|
if (
|
|
!ignoredMessages.find((message) => error?.message?.includes(message)) &&
|
|
userstore.user
|
|
) {
|
|
const status = await NetInfo.fetch();
|
|
if (status.isConnected && status.isInternetReachable) {
|
|
ToastManager.error(e as Error, "Sync failed", context);
|
|
}
|
|
}
|
|
} finally {
|
|
userstore.setSyncing(
|
|
false,
|
|
error ? SyncStatus.Failed : SyncStatus.Passed
|
|
);
|
|
onCompleted?.(error ? SyncStatus.Failed : SyncStatus.Passed);
|
|
if (pendingSync)
|
|
Sync.run(
|
|
pendingSync.context,
|
|
pendingSync.forced,
|
|
pendingSync.type,
|
|
pendingSync.onCompleted,
|
|
pendingSync.lastSyncTime
|
|
);
|
|
}
|
|
}, 300);
|
|
};
|
|
|
|
const Sync = {
|
|
run,
|
|
getLastSyncType: () => lastSyncType
|
|
};
|
|
|
|
export default Sync;
|