(async () => {
const appKey = '';
const jsPath = '/public/goeasy-lite-2.13.2.min.js';
const channel = 'siyuan_sync';
const autoSync = true;
const autoSyncInterval = 30;
const notifyOnSyncFailed = true;
const syncActions = [
'/api/transactions',
'/api/filetree/createDoc',
'/api/filetree/removeDoc',
'/api/filetree/renameDoc',
'/api/filetree/moveDocs',
'/api/notebook/createNotebook',
'/api/notebook/removeNotebook',
'/api/notebook/renameNotebook',
'/api/notebook/changeSortNotebook',
'/api/sync/performSync',
'/api/snippet/setSnippet',
'/api/setting/setEditor',
'/api/setting/setFiletree',
'/api/setting/setFlashcard',
'/api/setting/setAI',
'/api/setting/setExport',
'/api/setting/setAppearance',
'/api/setting/setBazaar',
'/api/setting/setKeymap',
'/api/sync/setSyncProvider',
'/api/file/putFile',
];
const debug = false;
let goEasy,
sendMessageTimer,
isRemoteSyncing = false,
isLocalSyncing = false;
loadJS(jsPath, function () {
if (!checkGoEasy()) return;
goEasy = GoEasy.getInstance({
host: 'hangzhou.goeasy.io',
appkey: appKey,
modules: ['pubsub']
});
goEasy.connect({
id: siyuan.ws.app.appId,
data: {"workspaceDir": siyuan.config.system.workspaceDir, "name": siyuan.config.system.name},
onSuccess: function () {
console.log("GoEasy connect successfully.")
},
onFailed: function (error) {
console.log("Failed to connect GoEasy, code:" + error.code + ",error:" + error.content);
}
});
goEasy.pubsub.subscribe({
channel: channel,
presence: { enable: true },
onMessage: async function (message) {
try {
const data = JSON.parse(message.content);
if(!data.appId || !data.repoKey || !data.time || !data.action) {
console.log('message content error: ', message.content);
return;
}
data.repoKey = decrypt(genKeyByAppIdAndTime(data.appId, data.time), data.repoKey);
const appId = siyuan.ws.app.appId;
if (data.appId !== appId && data.repoKey === siyuan.config.repo.key && data.action === 'sync' && !isRemoteSyncing) {
isRemoteSyncing = true;
const result = await sync();
isRemoteSyncing = false;
if (result && result.code === 0) {
debugInfo('收到消息,已同步成功 data.appId:', data.appId);
} else {
if(notifyOnSyncFailed) showErrorMessage("从远程同步失败,请手动同步");
console.log('remote sync failed data.appId:', data.appId, result);
}
} else {
debugInfo('收到消息,但不符合同步条件,已忽略本次同步');
}
} catch (e) {
isRemoteSyncing = false;
console.log('message data: ', message.content);
console.error(e);
}
},
onSuccess: function () {
console.log("Channel " + channel + " 订阅成功");
},
onFailed: function (error) {
console.log("Channel " + channel + "订阅失败, 错误编码:" + error.code + " 错误信息:" + error.content)
}
});
listenChange();
listenSyncBtnClick();
});
function getUsers() {
return new Promise((resolve, reject) => {
goEasy.pubsub.hereNow({
channel: channel,
limit: 20,
onSuccess: resolve,
onFailed: reject
});
});
}
async function listenChange() {
let isFetchOverridden = false;
if (!isFetchOverridden) {
const originalFetch = window.fetch;
window.fetch = async function (url, ...args) {
try {
const response = await originalFetch(url, ...args);
if (syncActions.some(item => url.endsWith(item))) {
if (siyuan.config.sync.enabled && siyuan.config.sync.provider !== 0) {
let users;
try {
users = await getUsers();
} catch (e) {
console.warn(e);
}
if(users) {
if(users.code === 200 && users.content?.amount > 1){
if (url.endsWith('/api/sync/performSync')) {
delaySendMessage();
} else {
if (autoSync) delaySync();
}
}
} else {
if (url.endsWith('/api/sync/performSync')) {
delaySendMessage();
} else {
if (autoSync) delaySync();
}
}
} else {
debugInfo("监听到文件变动,但未开启同步,已忽略本次同步");
}
}
return response;
} catch (error) {
throw error;
}
};
isFetchOverridden = true;
}
}
function listenSyncBtnClick() {
const syncBtn = document.querySelector(isMobile() ? "#toolbarSync" : "#barSync svg");
syncBtn.addEventListener("click", async function () {
willSync(false);
});
}
function delaySendMessage() {
if (sendMessageTimer) clearTimeout(sendMessageTimer);
debugInfo("监听到文件变动,即将在" + autoSyncInterval + "秒后同步数据");
sendMessageTimer = setTimeout(async () => {
const appId = siyuan.ws.app.appId;
const time = new Date().getTime();
try {
sendMessage({
appId: appId,
repoKey: encrypt(genKeyByAppIdAndTime(appId, time), siyuan.config.repo.key),
action: 'sync',
time: time,
});
} catch (e) {
console.log('消息发送失败 appId:', appId);
console.error(e);
}
}, autoSyncInterval * 1000);
}
function delaySync() {
if (isLocalSyncing) {
console.log('本地正在同步中,已忽略本次同步');
return;
}
if (sendMessageTimer) clearTimeout(sendMessageTimer);
debugInfo("监听到文件变动,即将在" + autoSyncInterval + "秒后同步数据");
willSync(true);
sendMessageTimer = setTimeout(async () => {
willSync(false);
const appId = siyuan.ws.app.appId;
const time = new Date().getTime();
try {
isLocalSyncing = true;
const result = await sync();
isLocalSyncing = false;
if (result && result.code === 0) {
sendMessage({
appId: appId,
repoKey: encrypt(genKeyByAppIdAndTime(appId, time), siyuan.config.repo.key),
action: 'sync',
time: time,
});
} else {
if(notifyOnSyncFailed) showErrorMessage("本地同步失败,请手动同步");
console.log('local sync failed appId:', appId, result);
}
} catch (e) {
isLocalSyncing = false;
console.log('local sync error: ', e);
}
}, autoSyncInterval * 1000);
}
async function sync(payload = {}) {
return await fetchSyncPost('/api/sync/performSync?by=sync-js', payload || {});
}
function sendMessage(msg) {
if (!checkGoEasy()) return;
let appId = '';
if (typeof msg !== 'string') {
appId = msg.appId;
msg = JSON.stringify(msg);
}
goEasy.pubsub.publish({
channel: channel,
message: msg,
qos: 1,
onSuccess: function () {
debugInfo('消息发送成功 appId:', appId);
},
onFailed: function (error) {
if(notifyOnSyncFailed) showErrorMessage("消息发送失败,请检查网络连接状态");
console.log("消息发送失败,错误编码:" + error.code + " 错误信息:" + error.content);
}
});
}
function checkGoEasy() {
if (typeof goEasy === 'undefined' && typeof GoEasy === 'undefined') {
showErrorMessage('GoEasy SDK加载失败,请检查网络连接');
console.error('GoEasy SDK加载失败,请检查网络连接');
return false;
}
return true;
}
function willSync(yes = true) {
const syncBtn = document.querySelector(isMobile() ? "#toolbarSync" : "#barSync svg");
if(yes) {
if(syncBtn) syncBtn.style.color='red';
} else {
if(syncBtn) syncBtn.style.color='';
}
}
function loadJS(src, callback) {
var script = document.createElement('script');
script.type = 'text/javascript';
script.src = src;
script.onload = function () {
callback();
};
script.onerror = function () {
showErrorMessage('Failed to load script: ' + src);
console.error('Failed to load script: ' + src);
};
document.head.appendChild(script);
}
function showErrorMessage(message, delay) {
fetchSyncPost("/api/notification/pushErrMsg", {
"msg": message,
"timeout": delay || 7000
});
}
async function fetchSyncPost(url, data, returnType = 'json') {
const init = {
method: "POST",
};
if (data) {
if (data instanceof FormData) {
init.body = data;
} else {
init.body = JSON.stringify(data);
}
}
try {
const res = await fetch(url, init);
const res2 = returnType === 'json' ? await res.json() : await res.text();
return res2;
} catch (e) {
console.log(e);
return returnType === 'json' ? { code: e.code || 1, msg: e.message || "", data: null } : "";
}
}
function genKeyByAppIdAndTime(appId, time) {
const key = '--sy-sync-js' + appId + appKey + channel + time;
return encrypt(key.split('').reverse().join(''), key);
}
function encrypt(key, data) {
let encrypted = '';
for (let i = 0; i < data.length; i++) {
const charCode = data.charCodeAt(i) ^ key.charCodeAt(i % key.length);
encrypted += String.fromCharCode(charCode);
}
return btoa(encrypted);
}
function decrypt(key, encryptedData) {
const decryptedData = atob(encryptedData);
let decrypted = '';
for (let i = 0; i < decryptedData.length; i++) {
const charCode = decryptedData.charCodeAt(i) ^ key.charCodeAt(i % key.length);
decrypted += String.fromCharCode(charCode);
}
return decrypted;
}
function isMobile() {
return document.getElementById("sidebar") ? true : false;
}
function debugInfo(msg1, msg2, msg3, msg4, msg5){
if(debug) console.log(msg1, msg2, msg3);
}
})();