forked from terminalnetwork/syncthing-hooks
-
Notifications
You must be signed in to change notification settings - Fork 0
/
index.js
executable file
·87 lines (78 loc) · 2.59 KB
/
index.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
#!/usr/bin/env node
const { fetchNewEvents } = require('./api');
const { collectHooks, runHook } = require('./hooks');
require('dotenv').config();
const state = {
seenIds: null,
mostRecentEventForFolder: null,
promisesForHooks: new Map(),
};
const getMostRecentEvents = (events, monitoredFolders) => {
const mostRecentEventForFolder = events
.filter(x => x.data && x.data.folder)
.filter(x => monitoredFolders.has(x.data.folder))
.reduce(
(acc, x) => {
const date = new Date(x.time);
const existingDate = acc[x.data.folder];
acc[x.data.folder] =
existingDate && existingDate > date ? existingDate : date;
return acc;
},
state.mostRecentEventForFolder
? { ...state.mostRecentEventForFolder }
: {}
);
state.mostRecentEventForFolder = mostRecentEventForFolder;
return mostRecentEventForFolder;
};
const convertRecentEventDatesToDelta = () => {
const now = new Date().getTime();
return Object.entries(state.mostRecentEventForFolder)
.map(([folder, date]) => [folder, now - date.getTime()])
.reduce((acc, [path, delta]) => {
acc[path] = delta;
return acc;
}, {});
};
const poll = async () => {
const { events, seenIds } = await fetchNewEvents(state.seenIds);
const hooks = await collectHooks();
const monitoredFolders = new Set(hooks.map(x => x.folder));
const deltaForFolders =
getMostRecentEvents(events, monitoredFolders) &&
convertRecentEventDatesToDelta();
hooks
.filter(x => deltaForFolders[x.folder])
.forEach(hook => {
const timeToWait = hook.time - deltaForFolders[hook.folder];
if (timeToWait < 0) {
const existingPromise = state.promisesForHooks[hook.path];
if (existingPromise) {
console.log(
`hook "${hook.path}" was skipped because it is already running`
);
} else {
delete state.mostRecentEventForFolder[hook.folder];
console.log(`running hook "${hook.path}"`);
const promise = runHook(hook);
state.promisesForHooks[hook.path] = promise;
promise
.then(() => {
console.log(`successfully ran hook "${hook.path}"`);
})
.catch(error => {
console.error(`failed to run hook "${hook.path}": ${error}`);
})
.finally(() => {
delete state.promisesForHooks[hook.path];
});
}
} else {
console.log(`scheduled hook "${hook.path}" to run in ${timeToWait}ms`);
}
});
state.seenIds = seenIds;
};
setInterval(poll, 1000);
poll();