yandere_fe/src/modules/serverSideStorage.js

372 lines
12 KiB
JavaScript
Raw Normal View History

2022-08-03 15:56:52 -07:00
import { toRaw } from 'vue'
import { isEqual, uniqWith, cloneDeep, set, get, clamp } from 'lodash'
2022-08-04 12:09:42 -07:00
import { CURRENT_UPDATE_COUNTER } from 'src/components/update_notification/update_notification.js'
2022-08-03 15:56:52 -07:00
2022-08-04 07:20:11 -07:00
export const VERSION = 1
2022-08-04 12:09:42 -07:00
export const NEW_USER_DATE = new Date('2022-08-04') // date of writing this, basically
2022-08-03 15:56:52 -07:00
2022-08-04 07:20:11 -07:00
export const COMMAND_TRIM_FLAGS = 1000
export const COMMAND_TRIM_FLAGS_AND_RESET = 1001
2022-08-03 15:56:52 -07:00
2022-08-04 12:09:42 -07:00
export const defaultState = {
2022-08-04 07:20:11 -07:00
// do we need to update data on server?
2022-08-03 15:56:52 -07:00
dirty: false,
// storage of flags - stuff that can only be set and incremented
flagStorage: {
updateCounter: 0, // Counter for most recent update notification seen
reset: 0 // special flag that can be used to force-reset all flags, debug purposes only
// special reset codes:
// 1000: trim keys to those known by currently running FE
// 1001: same as above + reset everything to 0
},
2022-08-09 16:19:07 -07:00
prefsStorage: {
_journal: [],
simple: {
dontShowUpdateNotifs: false
}
2022-08-09 16:19:07 -07:00
},
2022-08-03 15:56:52 -07:00
// raw data
raw: null,
// local cache
cache: null
}
2022-08-04 12:09:42 -07:00
export const newUserFlags = {
2022-08-03 15:56:52 -07:00
...defaultState.flagStorage,
2022-08-04 12:09:42 -07:00
updateCounter: CURRENT_UPDATE_COUNTER // new users don't need to see update notification
2022-08-03 15:56:52 -07:00
}
export const _moveItemInArray = (array, value, movement) => {
const oldIndex = array.indexOf(value)
const newIndex = oldIndex + movement
const newArray = [...array]
// remove old
newArray.splice(oldIndex, 1)
// add new
newArray.splice(clamp(newIndex, 0, newArray.length + 1), 0, value)
return newArray
}
2022-08-04 07:20:11 -07:00
const _wrapData = (data) => ({
...data,
_timestamp: Date.now(),
_version: VERSION
})
2022-08-04 12:09:42 -07:00
const _checkValidity = (data) => data._timestamp > 0 && data._version > 0
2022-08-04 07:20:11 -07:00
export const _getRecentData = (cache, live) => {
const result = { recent: null, stale: null, needUpload: false }
const cacheValid = _checkValidity(cache || {})
const liveValid = _checkValidity(live || {})
2022-08-04 12:09:42 -07:00
if (!liveValid && cacheValid) {
2022-08-04 07:20:11 -07:00
result.needUpload = true
console.debug('Nothing valid stored on server, assuming cache to be source of truth')
result.recent = cache
result.stale = live
2022-08-04 12:09:42 -07:00
} else if (!cacheValid && liveValid) {
2022-08-04 07:20:11 -07:00
console.debug('Valid storage on server found, no local cache found, using live as source of truth')
result.recent = live
result.stale = cache
2022-08-04 12:09:42 -07:00
} else if (cacheValid && liveValid) {
2022-08-04 07:20:11 -07:00
console.debug('Both sources have valid data, figuring things out...')
if (live._timestamp === cache._timestamp && live._version === cache._version) {
console.debug('Same version/timestamp on both source, source of truth irrelevant')
result.recent = cache
result.stale = live
} else {
console.debug('Different timestamp, figuring out which one is more recent')
if (live._timestamp < cache._timestamp) {
result.recent = cache
result.stale = live
} else {
result.recent = live
result.stale = cache
}
}
2022-08-04 12:09:42 -07:00
} else {
console.debug('Both sources are invalid, start from scratch')
result.needUpload = true
2022-08-04 07:20:11 -07:00
}
return result
}
2022-08-04 12:09:42 -07:00
export const _getAllFlags = (recent, stale) => {
2022-08-04 07:20:11 -07:00
return Array.from(new Set([
2022-08-04 12:09:42 -07:00
...Object.keys(toRaw((recent || {}).flagStorage || {})),
...Object.keys(toRaw((stale || {}).flagStorage || {}))
2022-08-04 07:20:11 -07:00
]))
}
export const _mergeFlags = (recent, stale, allFlagKeys) => {
2022-08-09 16:59:08 -07:00
if (!stale.flagStorage) return recent.flagStorage
if (!recent.flagStorage) return stale.flagStorage
2022-08-04 07:20:11 -07:00
return Object.fromEntries(allFlagKeys.map(flag => {
const recentFlag = recent.flagStorage[flag]
const staleFlag = stale.flagStorage[flag]
// use flag that is of higher value
2022-08-04 12:09:42 -07:00
return [flag, Number((recentFlag > staleFlag ? recentFlag : staleFlag) || 0)]
2022-08-04 07:20:11 -07:00
}))
}
const _mergeJournal = (a, b) => uniqWith(
2022-08-10 16:23:58 -07:00
// TODO use groupBy to group by path, then trim them depending on operations,
// i.e. if field got removed in the end - no need to sort it beforehand, if field
// got re-added no need to remove it and add it etc.
[
...(Array.isArray(a) ? a : []),
...(Array.isArray(b) ? b : [])
].sort((a, b) => a.timestamp > b.timestamp ? -1 : 1),
(a, b) => {
if (a.operation !== b.operation) return false
switch (a.operation) {
case 'set':
case 'arrangeSet':
return a.path === b.path
default:
return a.path === b.path && a.timestamp === b.timestamp
}
}
).reverse()
2022-08-09 16:19:07 -07:00
export const _mergePrefs = (recent, stale, allFlagKeys) => {
if (!stale) return recent
if (!recent) return stale
const { _journal: recentJournal, ...recentData } = recent
const { _journal: staleJournal } = stale
/** Journal entry format:
* path: path to entry in prefsStorage
* timestamp: timestamp of the change
* operation: operation type
* arguments: array of arguments, depends on operation type
*
* currently only supported operation type is "set" which just sets the value
* to requested one. Intended only to be used with simple preferences (boolean, number)
* shouldn't be used with collections!
*/
const resultOutput = { ...recentData }
const totalJournal = _mergeJournal(staleJournal, recentJournal)
2022-08-09 16:19:07 -07:00
totalJournal.forEach(({ path, timestamp, operation, args }) => {
if (path.startsWith('_')) {
console.error(`journal contains entry to edit internal (starts with _) field '${path}', something is incorrect here, ignoring.`)
return
}
switch (operation) {
case 'set':
set(resultOutput, path, args[0])
break
case 'addToCollection':
set(resultOutput, path, Array.from(new Set(get(resultOutput, path)).add(args[0])))
break
case 'removeFromCollection':
set(resultOutput, path, Array.from(new Set(get(resultOutput, path)).remove(args[0])))
break
case 'reorderCollection': {
const [value, movement] = args
set(resultOutput, path, _moveItemInArray(get(resultOutput, path), value, movement))
break
}
2022-08-09 16:19:07 -07:00
default:
console.error(`Unknown journal operation: '${operation}', did we forget to run reverse migrations beforehand?`)
}
})
return { ...resultOutput, _journal: totalJournal }
}
2022-08-04 12:09:42 -07:00
export const _resetFlags = (totalFlags, knownKeys = defaultState.flagStorage) => {
let result = { ...totalFlags }
const allFlagKeys = Object.keys(totalFlags)
2022-08-04 07:20:11 -07:00
// flag reset functionality
if (totalFlags.reset >= COMMAND_TRIM_FLAGS && totalFlags.reset <= COMMAND_TRIM_FLAGS_AND_RESET) {
console.debug('Received command to trim the flags')
2022-08-04 12:09:42 -07:00
const knownKeysSet = new Set(Object.keys(knownKeys))
// Trim
result = {}
2022-08-04 07:20:11 -07:00
allFlagKeys.forEach(flag => {
2022-08-04 12:09:42 -07:00
if (knownKeysSet.has(flag)) {
result[flag] = totalFlags[flag]
2022-08-04 07:20:11 -07:00
}
})
2022-08-04 12:09:42 -07:00
// Reset
2022-08-04 07:20:11 -07:00
if (totalFlags.reset === COMMAND_TRIM_FLAGS_AND_RESET) {
// 1001 - and reset everything to 0
console.debug('Received command to reset the flags')
2022-08-04 12:09:42 -07:00
Object.keys(knownKeys).forEach(flag => { result[flag] = 0 })
2022-08-04 07:20:11 -07:00
}
} else if (totalFlags.reset > 0 && totalFlags.reset < 9000) {
console.debug('Received command to reset the flags')
2022-08-04 12:09:42 -07:00
allFlagKeys.forEach(flag => { result[flag] = 0 })
2022-08-04 07:20:11 -07:00
}
2022-08-04 12:09:42 -07:00
result.reset = 0
return result
2022-08-04 07:20:11 -07:00
}
export const _doMigrations = (cache) => {
2022-08-04 12:09:42 -07:00
if (!cache) return cache
2022-08-04 07:20:11 -07:00
if (cache._version < VERSION) {
console.debug('Local cached data has older version, seeing if there any migrations that can be applied')
// no migrations right now since we only have one version
console.debug('No migrations found')
}
if (cache._version > VERSION) {
console.debug('Local cached data has newer version, seeing if there any reverse migrations that can be applied')
// no reverse migrations right now but we leave a possibility of loading a hotpatch if need be
if (window._PLEROMA_HOTPATCH) {
if (window._PLEROMA_HOTPATCH.reverseMigrations) {
console.debug('Found hotpatch migration, applying')
2022-08-07 16:18:29 -07:00
return window._PLEROMA_HOTPATCH.reverseMigrations.call({}, 'serverSideStorage', { from: cache._version, to: VERSION }, cache)
2022-08-04 07:20:11 -07:00
}
}
}
return cache
}
2022-08-04 12:09:42 -07:00
export const mutations = {
setServerSideStorage (state, userData) {
const live = userData.storage
state.raw = live
let cache = state.cache
2022-08-04 07:20:11 -07:00
2022-08-04 12:09:42 -07:00
cache = _doMigrations(cache)
2022-08-04 07:20:11 -07:00
2022-08-04 12:09:42 -07:00
let { recent, stale, needsUpload } = _getRecentData(cache, live)
2022-08-04 07:20:11 -07:00
2022-08-04 12:09:42 -07:00
const userNew = userData.created_at > NEW_USER_DATE
const flagsTemplate = userNew ? newUserFlags : defaultState.flagStorage
let dirty = false
2022-08-04 07:20:11 -07:00
2022-08-04 12:09:42 -07:00
if (recent === null) {
console.debug(`Data is empty, initializing for ${userNew ? 'new' : 'existing'} user`)
recent = _wrapData({
2022-08-09 16:19:07 -07:00
flagStorage: { ...flagsTemplate },
prefsStorage: { ...defaultState.prefsStorage }
2022-08-04 12:09:42 -07:00
})
}
2022-08-04 07:20:11 -07:00
2022-08-04 12:09:42 -07:00
if (!needsUpload && recent && stale) {
console.debug('Checking if data needs merging...')
// discarding timestamps and versions
const { _timestamp: _0, _version: _1, ...recentData } = recent
const { _timestamp: _2, _version: _3, ...staleData } = stale
dirty = !isEqual(recentData, staleData)
console.debug(`Data ${dirty ? 'needs' : 'doesn\'t need'} merging`)
}
2022-08-04 07:20:11 -07:00
2022-08-04 12:09:42 -07:00
const allFlagKeys = _getAllFlags(recent, stale)
let totalFlags
2022-08-09 16:19:07 -07:00
let totalPrefs
2022-08-04 12:09:42 -07:00
if (dirty) {
// Merge the flags
2022-08-09 16:19:07 -07:00
console.debug('Merging the data...')
2022-08-04 12:09:42 -07:00
totalFlags = _mergeFlags(recent, stale, allFlagKeys)
2022-08-09 16:19:07 -07:00
totalPrefs = _mergePrefs(recent.prefsStorage, stale.prefsStorage)
2022-08-04 12:09:42 -07:00
} else {
totalFlags = recent.flagStorage
2022-08-09 16:19:07 -07:00
totalPrefs = recent.prefsStorage
2022-08-04 12:09:42 -07:00
}
2022-08-04 07:20:11 -07:00
2022-08-04 12:09:42 -07:00
totalFlags = _resetFlags(totalFlags)
2022-08-04 07:20:11 -07:00
2022-08-04 12:09:42 -07:00
recent.flagStorage = totalFlags
2022-08-09 16:19:07 -07:00
recent.prefsStorage = totalPrefs
2022-08-04 07:20:11 -07:00
2022-08-04 12:09:42 -07:00
state.dirty = dirty || needsUpload
state.cache = recent
// set local timestamp to smaller one if we don't have any changes
if (stale && recent && !state.dirty) {
state.cache._timestamp = Math.min(stale._timestamp, recent._timestamp)
2022-08-03 15:56:52 -07:00
}
2022-08-04 12:09:42 -07:00
state.flagStorage = state.cache.flagStorage
2022-08-09 16:59:08 -07:00
state.prefsStorage = state.cache.prefsStorage
2022-08-04 12:09:42 -07:00
},
setFlag (state, { flag, value }) {
state.flagStorage[flag] = value
state.dirty = true
},
setPreference (state, { path, value }) {
if (path.startsWith('_')) {
console.error(`tried to edit internal (starts with _) field '${path}', ignoring.`)
return
}
set(state.prefsStorage, path, value)
state.prefsStorage._journal = [
...state.prefsStorage._journal,
{ command: 'set', path, args: [value], timestamp: Date.now() }
]
},
addCollectionPreference (state, { path, value }) {
if (path.startsWith('_')) {
console.error(`tried to edit internal (starts with _) field '${path}', ignoring.`)
return
}
const collection = new Set(get(state.prefsStorage, path))
collection.add(value)
set(state.prefsStorage, path, collection)
state.prefsStorage._journal = [
...state.prefsStorage._journal,
{ command: 'addToCollection', path, args: [value], timestamp: Date.now() }
]
},
removeCollectionPreference (state, { path, value }) {
if (path.startsWith('_')) {
console.error(`tried to edit internal (starts with _) field '${path}', ignoring.`)
return
}
const collection = new Set(get(state.prefsStorage, path))
collection.remove(value)
set(state.prefsStorage, path, collection)
state.prefsStorage._journal = [
...state.prefsStorage._journal,
{ command: 'removeFromCollection', path, args: [value], timestamp: Date.now() }
]
},
reorderCollectionPreference (state, { path, value, movement }) {
if (path.startsWith('_')) {
console.error(`tried to edit internal (starts with _) field '${path}', ignoring.`)
return
}
const collection = get(state.prefsStorage, path)
const newCollection = _moveItemInArray(collection, value, movement)
set(state.prefsStorage, path, newCollection)
state.prefsStorage._journal = [
...state.prefsStorage._journal,
{ command: 'arrangeCollection', path, args: [value], timestamp: Date.now() }
]
},
updateCache (state) {
state.prefsStorage._journal = _mergeJournal(state.prefsStorage._journal)
state.cache = _wrapData({
flagStorage: toRaw(state.flagStorage),
prefsStorage: toRaw(state.prefsStorage)
})
2022-08-04 12:09:42 -07:00
}
}
const serverSideStorage = {
state: {
...cloneDeep(defaultState)
2022-08-03 15:56:52 -07:00
},
2022-08-04 12:09:42 -07:00
mutations,
2022-08-03 15:56:52 -07:00
actions: {
pushServerSideStorage ({ state, rootState, commit }, { force = false } = {}) {
const needPush = state.dirty || force
if (!needPush) return
commit('updateCache')
2022-08-03 15:56:52 -07:00
const params = { pleroma_settings_store: { 'pleroma-fe': state.cache } }
rootState.api.backendInteractor
.updateProfile({ params })
.then((user) => commit('setServerSideStorage', user))
state.dirty = false
}
}
}
export default serverSideStorage