var {Value, computed} = require('mutant') var pull = require('pull-stream') var nest = require('depnest') var ref = require('ssb-ref') var colorHash = new (require('color-hash'))() var fallbackImageUrl = '' exports.needs = nest({ 'sbot.pull.stream': 'first', 'blob.sync.url': 'first', 'about.sync.shortFeedId': 'first', 'keys.sync.id': 'first' }) exports.gives = nest({ 'about.obs': [ 'name', 'description', 'image', 'imageUrl', 'names', 'images', 'color', 'latestValue', 'valueFrom', 'socialValue', 'groupedValues' ] }) exports.create = function (api) { var syncValue = Value(false) var sync = computed(syncValue, x => x) var cache = null return nest({ 'about.obs': { // quick helpers, probably should deprecate! name: (id) => socialValue(id, 'name', api.about.sync.shortFeedId(id)), description: (id) => socialValue(id, 'description'), image: (id) => socialValue(id, 'image'), names: (id) => groupedValues(id, 'name'), images: (id) => groupedValues(id, 'image'), color: (id) => computed(id, (id) => colorHash.hex(id)), imageUrl: (id) => computed(socialValue(id, 'image'), (blobId) => { return blobId ? api.blob.sync.url(blobId) : fallbackImageUrl }), // custom abouts (the future!) valueFrom, latestValue, socialValue, groupedValues } }) function valueFrom (id, key, author) { if (!ref.isLink(id)) throw new Error('About requires an ssb ref!') return withSync(computed([get(id), key, author], getValueFrom)) } function latestValue (id, key) { if (!ref.isLink(id)) throw new Error('About requires an ssb ref!') return withSync(computed([get(id), key], getLatestValue)) } function socialValue (id, key, defaultValue) { if (!ref.isLink(id)) throw new Error('About requires an ssb ref!') var yourId = api.keys.sync.id() return withSync(computed([get(id), key, id, yourId, defaultValue], getSocialValue)) } function groupedValues (id, key) { if (!ref.isLink(id)) throw new Error('About requires an ssb ref!') return withSync(computed([get(id), key], getGroupedValues)) } function withSync (obs) { obs.sync = sync return obs } function get (id) { if (!ref.isLink(id)) throw new Error('About requires an ssb ref!') load() if (!cache[id]) { cache[id] = Value({}) } return cache[id] } function load () { if (!cache) { cache = {} pull( api.sbot.pull.stream(sbot => sbot.about.stream({live: true})), pull.drain(item => { for (var target in item) { var state = get(target) var lastState = state() var values = item[target] var changed = false for (var key in values) { var valuesForKey = lastState[key] = lastState[key] || {} for (var author in values[key]) { var value = values[key][author] if (!valuesForKey[author] || value[1] > valuesForKey[author][1]) { valuesForKey[author] = value changed = true } } } if (changed) { state.set(lastState) } } if (!syncValue()) { syncValue.set(true) } }) ) } } } function getSocialValue (lookup, key, id, yourId, fallback) { var result = lookup[key] ? getValue(lookup[key][yourId]) || getValue(lookup[key][id]) || highestRank(lookup[key]) : null if (result != null) { return result } else { return fallback || null } } function getGroupedValues (lookup, key) { var values = {} for (var author in lookup[key]) { var value = getValue(lookup[key][author]) if (value != null) { values[value] = values[value] || [] values[value].push(author) } } return values } function highestRank (lookup) { var counts = {} var highestCount = 0 var currentHighest = null for (var key in lookup) { var value = getValue(lookup[key]) if (value != null) { counts[value] = (counts[value] || 0) + 1 if (counts[value] > highestCount) { currentHighest = value highestCount = counts[value] } } } return currentHighest } function getValue (item) { if (item && item[0]) { if (typeof item[0] === 'string') { return item[0] } else if (item[0] && item[0].link && ref.isLink(item[0].link) && !item[0].remove) { return item[0].link } } } function getLatestValue (lookup, key) { var latestTime = 0 var latestValue = null if (lookup[key]) { for (var author in lookup[key]) { if (Array.isArray(lookup[key][author])) { var value = lookup[key][author][0] var timestamp = lookup[key][author][1] if (timestamp > latestTime) { latestTime = timestamp latestValue = value } } } } return latestValue } function getValueFrom (lookup, key, author) { if (lookup[key] && Array.isArray(lookup[key][author])) { return lookup[key][author][0] } }