2017-08-15 03:46:04 +02:00
|
|
|
const jsonDiffer = require('fast-json-patch')
|
|
|
|
const clone = require('clone')
|
2018-04-19 20:29:26 +02:00
|
|
|
/** @module*/
|
2017-08-15 03:46:04 +02:00
|
|
|
module.exports = {
|
|
|
|
generateHistoryEntry,
|
|
|
|
replayHistory,
|
|
|
|
snapshotFromTxMeta,
|
|
|
|
migrateFromSnapshotsToDiffs,
|
|
|
|
}
|
|
|
|
|
2018-04-19 20:29:26 +02:00
|
|
|
/**
|
|
|
|
converts non-initial history entries into diffs
|
|
|
|
@param longHistory {array}
|
|
|
|
@returns {array}
|
|
|
|
*/
|
2017-10-21 21:06:39 +02:00
|
|
|
function migrateFromSnapshotsToDiffs (longHistory) {
|
2017-08-15 03:46:04 +02:00
|
|
|
return (
|
|
|
|
longHistory
|
|
|
|
// convert non-initial history entries into diffs
|
2019-07-31 22:17:11 +02:00
|
|
|
.map((entry, index) => {
|
2019-11-20 01:03:20 +01:00
|
|
|
if (index === 0) {
|
|
|
|
return entry
|
|
|
|
}
|
2019-07-31 22:17:11 +02:00
|
|
|
return generateHistoryEntry(longHistory[index - 1], entry)
|
|
|
|
})
|
2017-08-15 03:46:04 +02:00
|
|
|
)
|
|
|
|
}
|
|
|
|
|
2018-04-19 20:29:26 +02:00
|
|
|
/**
|
2018-05-10 13:26:02 +02:00
|
|
|
Generates an array of history objects sense the previous state.
|
2018-05-10 13:43:31 +02:00
|
|
|
The object has the keys
|
2018-05-10 13:26:02 +02:00
|
|
|
op (the operation performed),
|
|
|
|
path (the key and if a nested object then each key will be seperated with a `/`)
|
|
|
|
value
|
|
|
|
with the first entry having the note and a timestamp when the change took place
|
2018-04-19 20:29:26 +02:00
|
|
|
@param previousState {object} - the previous state of the object
|
|
|
|
@param newState {object} - the update object
|
2019-12-17 00:24:52 +01:00
|
|
|
@param {string} [note] - a optional note for the state change
|
2018-05-10 13:26:02 +02:00
|
|
|
@returns {array}
|
2018-04-19 20:29:26 +02:00
|
|
|
*/
|
2017-10-21 21:06:39 +02:00
|
|
|
function generateHistoryEntry (previousState, newState, note) {
|
2017-10-02 22:44:11 +02:00
|
|
|
const entry = jsonDiffer.compare(previousState, newState)
|
|
|
|
// Add a note to the first op, since it breaks if we append it to the entry
|
2018-05-10 13:26:02 +02:00
|
|
|
if (entry[0]) {
|
2019-11-20 01:03:20 +01:00
|
|
|
if (note) {
|
|
|
|
entry[0].note = note
|
|
|
|
}
|
2018-05-10 13:26:02 +02:00
|
|
|
|
2018-05-16 22:05:07 +02:00
|
|
|
entry[0].timestamp = Date.now()
|
2018-05-10 13:26:02 +02:00
|
|
|
}
|
2017-10-02 22:44:11 +02:00
|
|
|
return entry
|
2017-08-15 03:46:04 +02:00
|
|
|
}
|
|
|
|
|
2018-04-19 20:29:26 +02:00
|
|
|
/**
|
|
|
|
Recovers previous txMeta state obj
|
2018-05-10 13:26:02 +02:00
|
|
|
@returns {object}
|
2018-04-19 20:29:26 +02:00
|
|
|
*/
|
2017-10-21 21:06:39 +02:00
|
|
|
function replayHistory (_shortHistory) {
|
2017-10-02 22:14:42 +02:00
|
|
|
const shortHistory = clone(_shortHistory)
|
2017-08-15 03:46:04 +02:00
|
|
|
return shortHistory.reduce((val, entry) => jsonDiffer.applyPatch(val, entry).newDocument)
|
|
|
|
}
|
|
|
|
|
2018-04-19 20:29:26 +02:00
|
|
|
/**
|
2018-04-25 20:13:51 +02:00
|
|
|
@param txMeta {Object}
|
2018-04-19 20:29:26 +02:00
|
|
|
@returns {object} a clone object of the txMeta with out history
|
|
|
|
*/
|
2017-10-21 21:06:39 +02:00
|
|
|
function snapshotFromTxMeta (txMeta) {
|
2017-08-15 03:46:04 +02:00
|
|
|
// create txMeta snapshot for history
|
|
|
|
const snapshot = clone(txMeta)
|
|
|
|
// dont include previous history in this snapshot
|
|
|
|
delete snapshot.history
|
|
|
|
return snapshot
|
2017-10-21 21:06:39 +02:00
|
|
|
}
|