2017-10-25 10:58:59 +08:00
|
|
|
"use strict";
|
|
|
|
|
|
|
|
const express = require('express');
|
|
|
|
const router = express.Router();
|
|
|
|
const auth = require('../../services/auth');
|
2017-10-27 09:16:21 +08:00
|
|
|
const sync = require('../../services/sync');
|
2017-11-10 09:52:47 +08:00
|
|
|
const syncUpdate = require('../../services/sync_update');
|
2017-11-01 07:34:58 +08:00
|
|
|
const sql = require('../../services/sql');
|
2017-11-03 08:48:02 +08:00
|
|
|
const options = require('../../services/options');
|
2017-11-22 11:11:27 +08:00
|
|
|
const content_hash = require('../../services/content_hash');
|
2017-12-20 11:04:51 +08:00
|
|
|
const utils = require('../../services/utils');
|
|
|
|
const log = require('../../services/log');
|
2017-11-22 11:11:27 +08:00
|
|
|
|
|
|
|
router.get('/check', auth.checkApiAuth, async (req, res, next) => {
|
|
|
|
res.send({
|
2017-12-16 10:14:10 +08:00
|
|
|
'hashes': await content_hash.getHashes(),
|
2017-11-22 11:11:27 +08:00
|
|
|
'max_sync_id': await sql.getSingleValue('SELECT MAX(id) FROM sync')
|
|
|
|
});
|
|
|
|
});
|
2017-10-25 10:58:59 +08:00
|
|
|
|
2017-10-29 23:22:41 +08:00
|
|
|
router.post('/now', auth.checkApiAuth, async (req, res, next) => {
|
2017-11-05 09:21:09 +08:00
|
|
|
res.send(await sync.sync());
|
2017-10-29 23:22:41 +08:00
|
|
|
});
|
|
|
|
|
2017-12-20 11:04:51 +08:00
|
|
|
async function fillSyncRows(entityName, entityKey) {
|
|
|
|
// cleanup sync rows for missing entities
|
|
|
|
await sql.execute(`
|
|
|
|
DELETE
|
|
|
|
FROM sync
|
|
|
|
WHERE sync.entity_name = '${entityName}'
|
|
|
|
AND sync.entity_id NOT IN (SELECT ${entityKey} FROM ${entityName})`);
|
|
|
|
|
|
|
|
const entityIds = await sql.getFlattenedResults(`SELECT ${entityKey} FROM ${entityName}`);
|
|
|
|
|
|
|
|
for (const entityId of entityIds) {
|
|
|
|
const existingRows = await sql.getSingleValue("SELECT COUNT(id) FROM sync WHERE entity_name = ? AND entity_id = ?", [entityName, entityId]);
|
|
|
|
|
|
|
|
// we don't want to replace existing entities (which would effectively cause full resync)
|
|
|
|
if (existingRows === 0) {
|
|
|
|
log.info(`Creating missing sync record for ${entityName} ${entityId}`);
|
|
|
|
|
|
|
|
await sql.insert("sync", {
|
|
|
|
entity_name: entityName,
|
|
|
|
entity_id: entityId,
|
|
|
|
source_id: "SYNC_FILL",
|
|
|
|
sync_date: utils.nowDate()
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
router.post('/fill-sync-rows', auth.checkApiAuth, async (req, res, next) => {
|
|
|
|
await sql.doInTransaction(async () => {
|
|
|
|
await fillSyncRows("notes", "note_id");
|
|
|
|
await fillSyncRows("notes_tree", "note_tree_id");
|
|
|
|
await fillSyncRows("notes_history", "note_history_id");
|
|
|
|
await fillSyncRows("recent_notes", "note_tree_id");
|
|
|
|
});
|
|
|
|
|
|
|
|
res.send({});
|
|
|
|
});
|
|
|
|
|
2017-12-14 12:03:48 +08:00
|
|
|
router.post('/force-full-sync', auth.checkApiAuth, async (req, res, next) => {
|
|
|
|
await sql.doInTransaction(async () => {
|
|
|
|
await options.setOption('last_synced_pull', 0);
|
|
|
|
await options.setOption('last_synced_push', 0);
|
|
|
|
});
|
|
|
|
|
|
|
|
// not awaiting for the job to finish (will probably take a long time)
|
|
|
|
sync.sync();
|
|
|
|
|
|
|
|
res.send({});
|
|
|
|
});
|
|
|
|
|
2017-11-01 07:34:58 +08:00
|
|
|
router.get('/changed', auth.checkApiAuth, async (req, res, next) => {
|
|
|
|
const lastSyncId = parseInt(req.query.lastSyncId);
|
2017-10-25 10:58:59 +08:00
|
|
|
|
2017-11-17 10:50:00 +08:00
|
|
|
res.send(await sql.getResults("SELECT * FROM sync WHERE id > ?", [lastSyncId]));
|
2017-10-27 09:16:21 +08:00
|
|
|
});
|
|
|
|
|
2017-11-01 07:34:58 +08:00
|
|
|
router.get('/notes/:noteId', auth.checkApiAuth, async (req, res, next) => {
|
|
|
|
const noteId = req.params.noteId;
|
2017-10-27 09:16:21 +08:00
|
|
|
|
2017-11-01 07:34:58 +08:00
|
|
|
res.send({
|
2017-11-30 09:47:01 +08:00
|
|
|
entity: await sql.getSingleResult("SELECT * FROM notes WHERE note_id = ?", [noteId])
|
2017-11-01 07:34:58 +08:00
|
|
|
});
|
2017-10-26 10:39:21 +08:00
|
|
|
});
|
|
|
|
|
2017-11-24 11:25:39 +08:00
|
|
|
router.get('/notes_tree/:noteTreeId', auth.checkApiAuth, async (req, res, next) => {
|
|
|
|
const noteTreeId = req.params.noteTreeId;
|
2017-10-26 10:39:21 +08:00
|
|
|
|
2017-11-24 11:25:39 +08:00
|
|
|
res.send(await sql.getSingleResult("SELECT * FROM notes_tree WHERE note_tree_id = ?", [noteTreeId]));
|
2017-11-01 07:34:58 +08:00
|
|
|
});
|
|
|
|
|
|
|
|
router.get('/notes_history/:noteHistoryId', auth.checkApiAuth, async (req, res, next) => {
|
|
|
|
const noteHistoryId = req.params.noteHistoryId;
|
|
|
|
|
|
|
|
res.send(await sql.getSingleResult("SELECT * FROM notes_history WHERE note_history_id = ?", [noteHistoryId]));
|
2017-10-27 09:16:21 +08:00
|
|
|
});
|
|
|
|
|
2017-11-03 08:48:02 +08:00
|
|
|
router.get('/options/:optName', auth.checkApiAuth, async (req, res, next) => {
|
|
|
|
const optName = req.params.optName;
|
|
|
|
|
|
|
|
if (!options.SYNCED_OPTIONS.includes(optName)) {
|
|
|
|
res.send("This option can't be synced.");
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
res.send(await sql.getSingleResult("SELECT * FROM options WHERE opt_name = ?", [optName]));
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
2017-11-19 06:17:46 +08:00
|
|
|
router.get('/notes_reordering/:noteTreeParentId', auth.checkApiAuth, async (req, res, next) => {
|
|
|
|
const noteTreeParentId = req.params.noteTreeParentId;
|
2017-11-03 10:55:22 +08:00
|
|
|
|
|
|
|
res.send({
|
2017-12-20 10:40:48 +08:00
|
|
|
parent_note_id: noteTreeParentId,
|
|
|
|
ordering: await sql.getMap("SELECT note_tree_id, note_position FROM notes_tree WHERE parent_note_id = ?", [noteTreeParentId])
|
2017-11-03 10:55:22 +08:00
|
|
|
});
|
|
|
|
});
|
|
|
|
|
2017-12-03 23:06:53 +08:00
|
|
|
router.get('/recent_notes/:noteTreeId', auth.checkApiAuth, async (req, res, next) => {
|
|
|
|
const noteTreeId = req.params.noteTreeId;
|
2017-11-05 12:16:02 +08:00
|
|
|
|
2017-12-03 23:06:53 +08:00
|
|
|
res.send(await sql.getSingleResult("SELECT * FROM recent_notes WHERE note_tree_id = ?", [noteTreeId]));
|
2017-11-05 12:16:02 +08:00
|
|
|
});
|
|
|
|
|
2017-10-30 10:22:30 +08:00
|
|
|
router.put('/notes', auth.checkApiAuth, async (req, res, next) => {
|
2017-11-30 09:47:01 +08:00
|
|
|
await syncUpdate.updateNote(req.body.entity, req.body.sourceId);
|
2017-10-30 10:22:30 +08:00
|
|
|
|
|
|
|
res.send({});
|
|
|
|
});
|
|
|
|
|
|
|
|
router.put('/notes_tree', auth.checkApiAuth, async (req, res, next) => {
|
2017-11-10 09:52:47 +08:00
|
|
|
await syncUpdate.updateNoteTree(req.body.entity, req.body.sourceId);
|
2017-10-30 10:22:30 +08:00
|
|
|
|
|
|
|
res.send({});
|
|
|
|
});
|
|
|
|
|
|
|
|
router.put('/notes_history', auth.checkApiAuth, async (req, res, next) => {
|
2017-11-10 09:52:47 +08:00
|
|
|
await syncUpdate.updateNoteHistory(req.body.entity, req.body.sourceId);
|
2017-10-26 10:39:21 +08:00
|
|
|
|
2017-10-27 09:16:21 +08:00
|
|
|
res.send({});
|
2017-10-25 10:58:59 +08:00
|
|
|
});
|
|
|
|
|
2017-11-03 10:55:22 +08:00
|
|
|
router.put('/notes_reordering', auth.checkApiAuth, async (req, res, next) => {
|
2017-11-10 09:52:47 +08:00
|
|
|
await syncUpdate.updateNoteReordering(req.body.entity, req.body.sourceId);
|
2017-11-03 10:55:22 +08:00
|
|
|
|
|
|
|
res.send({});
|
|
|
|
});
|
|
|
|
|
2017-11-03 08:48:02 +08:00
|
|
|
router.put('/options', auth.checkApiAuth, async (req, res, next) => {
|
2017-11-10 09:52:47 +08:00
|
|
|
await syncUpdate.updateOptions(req.body.entity, req.body.sourceId);
|
2017-11-03 08:48:02 +08:00
|
|
|
|
|
|
|
res.send({});
|
|
|
|
});
|
|
|
|
|
2017-11-05 12:16:02 +08:00
|
|
|
router.put('/recent_notes', auth.checkApiAuth, async (req, res, next) => {
|
2017-11-10 09:52:47 +08:00
|
|
|
await syncUpdate.updateRecentNotes(req.body.entity, req.body.sourceId);
|
2017-11-05 12:16:02 +08:00
|
|
|
|
|
|
|
res.send({});
|
|
|
|
});
|
|
|
|
|
2017-10-25 10:58:59 +08:00
|
|
|
module.exports = router;
|