/* This file contains the IPFS Boards API. It's a simple abstraction over the js-ipfs-api that also provides an additional level of caching for this particular application. Let's hope it turns out decent Needs to be browserified to work in the browser */ // EventEmitter used to communicate with clients var EventEmitter = require('wolfy87-eventemitter') var asyncjs = require('async') var semver = require('semver') function asObj (str, done) { if (str.toString) str = str.toString() if (typeof str === 'string') { var obj try { obj = JSON.parse(str) } catch (e) { console.log('error parsing:', str, 'Error:', e) return done(e, undefined) } done(null, obj) } else { console.log('not string:', str) done('not string: ' + str, undefined) } } function replyAsObj (res, isJson, done) { if (res.readable) { // Is a stream console.log('got stream') res.setEncoding('utf8') var data = '' res.on('data', d => { console.log('got stream data:', d) data += d }) res.on('end', () => { if (isJson) { asObj(data, done) } else { done(null, data) } }) } else if (res.split || res.toString) { // console.log('got string or buffer:',res) if (res.toString) res = res.toString() // Is a string if (isJson) { asObj(res, done) } else { done(null, res) } } } function BoardsAPI (ipfs) { this.ipfs = ipfs this.version = 'ipfs:boards:version:dev' this.baseurl = '/ipfs-boards-profile/' this.users = [] // list of IPNS names this.resolvingIPNS = {} this.ee = new EventEmitter() if (window && window.localStorage !== undefined) { this.ee.on('init', e => { if (e) { console.log('init failed') this.init_error = e } }) // Use localStorage to store the IPNS cache var stored = window.localStorage.getItem('ipfs-boards-user-cache') try { this.users = JSON.parse(stored) if (this.users === null || this.users === undefined || !this.users.indexOf || !this.users.push) { this.users = [] } } catch (e) { this.users = [] } } } BoardsAPI.prototype.createProfile = function (profile, done) { console.log('Generating profile:', profile) try { var profile_str = JSON.stringify(profile) } catch (e) { console.log('Error, invalid profile:', e) return done(e) } asyncjs.waterfall([ // Create required directories cb => this.ipfs.files.mkdir('/ipfs-boards-profile/boards', { p: true }, cb), (e, cb) => this.ipfs.files.mkdir('/ipfs-boards-profile/comments', { p: true }, cb), (e, cb) => this.ipfs.files.mkdir('/ipfs-boards-profile/posts', { p: true }, cb), (e, cb) => { // Remove old profile files if present var path = '/ipfs-boards-profile/ipfs-boards-version.txt' this.ipfs.files.rm(path, { r: true }, res => { var path = '/ipfs-boards-profile/profile.json' this.ipfs.files.rm(path, { r: true }, res => cb()) }) }, cb => { // Add profile version file var path = '/ipfs-boards-profile/ipfs-boards-version.txt' var version_hash = '/ipfs/' + this.version_hash this.ipfs.files.cp([version_hash, path], cb) }, (e, cb) => { // Serialize profile and add to IPFS this.ipfs.add(new Buffer(profile_str), cb) }, (res, cb) => { // Move profile into mfs console.log('added profile to IPFS:', res.Hash) var profilepath = '/ipfs/' + res.Hash this.ipfs.files.cp([profilepath, '/ipfs-boards-profile/profile.json'], cb) }, (e, cb) => this.ipfs.files.stat('/', cb), (res, cb) => { var profile_hash = res.Hash console.log('Publishing profile...') this.ipfs.name.publish(profile_hash, cb) } ], done) } BoardsAPI.prototype.createBoard = function (board, done) { } BoardsAPI.prototype.backupCache = function () { if (window && window.localStorage !== undefined) { // Use localStorage to store the IPNS cache window.localStorage.setItem('ipfs-boards-user-cache', JSON.stringify(this.users)) } } // Rewrote this to use event emitters. Should also add periodic rechecking BoardsAPI.prototype.resolveIPNS = function (n, handler) { if (handler && handler.apply) this.ee.on(n, handler) if (!this.resolvingIPNS[n]) { this.resolvingIPNS[n] = true this.ipfs.name.resolve(n, (err, r) => { delete this.resolvingIPNS[n] if (err) { // Communicate error this.ee.emit('error', err) if (handler && handler.apply) handler(undefined, err) } else { var url = r.Path if (url === undefined) { console.log('Could not resolve', n) this.ee.emit('error', r.Message) } else if (this.users.indexOf(n) < 0) { // new find this.isUserProfile(url, (isit, err) => { if (isit) { console.log(n, 'is a user') this.ee.emit(n, url) if (this.users.indexOf(n) < 0) { this.ee.emit('user', n, url) this.users.push(n) this.backupCache() } } else { console.log(n, 'not a valid profile:', err) this.ee.emit(n, undefined, 'not a valid profile: ' + err) } return true // Remove from listeners }) } else { // Already known this.ee.emit(n, url) } } }) } return this.ee } BoardsAPI.prototype.isUserProfile = function (addr, done) { if (addr === undefined) return console.log('Asked to check if undefined is a profile') this.ipfs.cat(addr + this.baseurl + 'ipfs-boards-version.txt', (err, r) => { if (err) return done(false, err) replyAsObj(r, false, (_, res) => { if (!res || !res.trim) { console.log('Could not read version from', addr) } else { var v = res.trim() console.log('Version in profile snapshot', addr, 'is', v) if (v === this.version) { done(true) } else { done(false, 'version mismatch: is "' + v + '" but should be "' + this.version + '"') } } }) }) } BoardsAPI.prototype.searchUsers = function () { // Look at our peers this.ipfs.swarm.peers((err, r) => { if (err) return console.log(err) replyAsObj(r, true, (e, reply) => { if (e) { this.ee.emit('error', e) return console.log('There was an error while getting swarm peers:', e) } console.log('Checking', reply.Strings.length, 'peers') // reply.Strings.forEach(item => { var f = (item, done) => { var ss = item.split('/') var n = ss[ss.length - 1] this.ee.once(n, (res, err) => done(err)) this.resolveIPNS(n) } asyncjs.eachSeries(reply.Strings, f.bind(this)) }) }) // Look for who has the correct version file, they probably have a profile /* this.ipfs.dht.findprovs(this.version_hash, (err,res) => { if(err){ console.log('DHT FINDPROVS err',err) } else if(res.readable){ console.log('DHT FINDPROVS stream',res) } else { console.log('DHT FINDPROVS string',res) } })*/ return this.ee } BoardsAPI.prototype.getProfile = function (userID, done) { this.resolveIPNS(userID, (url, err) => { if (err) { this.ee.emit('error', err) done(err, null) } else { // Download actual profile this.ipfs.cat(url + this.baseurl + 'profile.json', (err2, res) => { if (err2) { this.ee.emit('error', err2) done(err2, null) } else { // TODO: JSON parse error handling var p = JSON.parse(res.toString()) this.ee.emit('profile for ' + userID, p) done(null, p) } }) // Get other info this.ipfs.ls(url + this.baseurl + 'boards/', (err2, res) => { if (!err2) { var l = res.Objects[0].Links.map(i => { return { name: i.Name, hash: i.Hash } }) this.ee.emit('boards for ' + userID, l) } else { this.ee.emit('error', err2) } }) } return true // remove myself from listeners }) return this.ee } BoardsAPI.prototype.getBoardSettings = function (userID, board, done) { if (!userID) { return console.log('Invalid USERID', userID) } if (!board) { return console.log('Invalid BOARD', board) } this.resolveIPNS(userID, (r, e) => { if (e) { this.ee.emit('error', e) } else { var url = r + this.baseurl + 'boards/' + board + '/settings.json' this.ipfs.cat(url, (err, resp) => { // TODO: error handling json conversion var settings = JSON.parse(resp.toString()) if (err) { this.ee.emit('error', err) if (done && done.apply) done(err) } else { // SETTINGS file is here, need to parse it a little bit this.ee.emit('settings for ' + board + '@' + userID, settings, r) if (done && done.apply) done(null, settings) if (settings.whitelist === true) { // Get the whitelist var url = r + this.baseurl + 'boards/' + board + '/whitelist' this.ipfs.cat(url, (err, res) => { if (err) { this.ee.emit('error', err) // Emit an empty whitelist. this.ee.emit('whitelist for ' + board + '@' + userID, []) } else { replyAsObj(res, false, (err, whitelist) => { if (err) { // Emit an empty whitelist. this.ee.emit('whitelist for ' + board + '@' + userID, []) } else { // Send whitelist var w = whitelist.split(' ').map(x => x.trim()) this.ee.emit('whitelist for ' + board + '@' + userID, w) } }) } }) } if (!settings.whitelist_only && !settings.approval_required && settings.blacklist === true) { // Get the blacklist var u = r + this.baseurl + 'boards/' + board + '/blacklist' this.ipfs.cat(u, (err, blacklist) => { if (err) { this.ee.emit('error', err) } else { // Send blacklist var w = blacklist.split(' ') this.emit('blacklist for ' + board + '@' + userID, w) } }) } } }) } return true // remove myself from listeners }) return this.ee } BoardsAPI.prototype.downloadPost = function (hash, adminID, board, op, done) { console.log('Downloading post', hash) this.ipfs.cat(hash, (err2, r) => { if (err2) { this.ee.emit('error', err2) console.log('Could not download post', hash, 'of', board + '@' + adminID) if (done && done.apply) done(err2) } else { replyAsObj(r, true, (err, post) => { if (err) return // TODO: add JSON parsing error handling post.hash = hash if (op) post.op = op // Inject op if (board) { if (adminID) this.ee.emit('post in ' + board + '@' + adminID, post, hash) else this.ee.emit('post in ' + board, post, hash) } this.ee.emit(hash, post, adminID, board) if (done && done.apply) done(null, post) }) } }) return this.ee } BoardsAPI.prototype.retrieveListOfApproved = function (what, addr, adminID, board) { var a = addr + this.baseurl + 'boards/' + board + '/approved/' + what + '/' this.ipfs.ls(a, (err, res) => { if (err) { this.ee.emit('error', err) } else { // Send approved posts list var ret = res.Objects[0].Links.map(item => { return { date: item.Name, hash: item.Hash } }) this.emit('approved ' + what + ' for ' + board + '@' + adminID, ret) } }) } BoardsAPI.prototype.getAllowedContentProducers = function (adminID, board, options) { if (!options) return this.ee.on('settings for ' + board + '@' + adminID, function (settings, addr) { // Get stuff based on settings if (settings.approval_required === true) { // Get approved posts list if (options.posts) this.retrieveListOfApproved('posts', addr, adminID, board) // Get approved comments list if (options.comments) this.retrieveListOfApproved('comments', addr, adminID, board) } else if (settings.whitelist_only === true) { // TODO: emit all whitelisted users } else if (settings.blacklist === true) { // TODO: emit all users not in the blacklist } }) this.getBoardSettings(adminID, board) return this.ee } BoardsAPI.prototype.getPostsInBoard = function (adminID, board) { if (adminID) { this.ee.on('approved posts for ' + board + '@' + adminID, ret => { // Automatically download approved posts ret.forEach(item => this.downloadPost(item.hash, adminID, board)) }) this.ee.on('whitelist for ' + board + '@' + adminID, whitelist => { // download posts for each user in whitelist whitelist.forEach(item => { this.getUserPostListInBoard(item, board, (err, postList) => { if (err) return postList.forEach(i => this.downloadPost(i.hash, adminID, board, item)) }) }) }) // Get allowed content and content producers this.getAllowedContentProducers(adminID, board, { posts: true }) // Get the admin's posts this.getUserPostListInBoard(adminID, board, (err, res) => { if (err) { console.log(err) } else res.forEach(item => this.downloadPost(item.hash, adminID, board, adminID)) }) } else { // TODO: Download all posts in board from everyone // Download my posts this.getUserPostListInBoard(this.id, board, (err, res) => { if (err) { console.log(err) } else res.forEach(item => this.downloadPost(item.hash, undefined, board, this.id)) }) } return this.ee } BoardsAPI.prototype.getUserPostListInBoard = function (user, board, done) { this.resolveIPNS(user, (url, err) => { if (err) { this.ee.emit('error', err) done(err) } else { this.ipfs.ls(url + this.baseurl + 'posts/' + board, (e, r) => { if (e) { this.ee.emit('error', e) done(e) } else if (r && !r.split) { console.log('Found', r.Objects[0].Links.length, 'posts in', board, 'at', user) this.ee.emit('post count', board, user, r.Objects[0].Links.length) var l = r.Objects[0].Links.map(i => { return { date: i.Name, hash: i.Hash } }) done(null, l) } }) } return true // remove myself from listeners }) return this.ee } BoardsAPI.prototype.downloadComment = function (hash, adminID, board, target, done) { if (!done && typeof target === 'function') { done = target target = undefined } console.log('target', target) this.ipfs.cat(hash, (err2, r) => { if (err2) { this.ee.emit('error', err2) console.log('Could not download comment', hash, 'of', board + '@' + adminID) if (done) done(err2) } else { // TODO: add JSON parsing error handling var cmnt = JSON.parse(r.toString()) cmnt.hash = hash if (target) { cmnt.original_parent = cmnt.parent cmnt.parent = target } this.ee.emit(hash, cmnt, adminID, board) this.ee.emit('comment for ' + (target || cmnt.parent), cmnt) if (done) done(null, cmnt) } }) return this.ee } BoardsAPI.prototype.getCommentsFor = function (parent, board, adminID, target) { if (!parent || !board || !adminID) { return console.log('malformed arguments:', parent, board, adminID) } // figure out if there's a previous version of the item this.ipfs.cat(parent, (err, res) => { if (err) { this.ee.emit('error', err) } else { replyAsObj(res, true, (err2, obj) => { if (err2) { this.ee.emit('error', err2) } else if (typeof obj.previous === 'string') { // Also get comments for the previous version of the parent! this.getCommentsFor(obj.previous, board, adminID, parent) } }) } }) // get the admin's comments this.getUserCommentList(parent, adminID, (err, res) => { if (!err) { res.forEach(item => this.downloadComment(item.hash, adminID, board, target)) } }) // Download comments from whitelisted this.ee.on('whitelist for ' + board + '@' + adminID, whitelist => { // download posts for each user in whitelist whitelist.forEach(item => { this.getUserCommentList(parent, item, (err, res) => { if (err) return res.forEach(i => this.downloadComment(i.hash, adminID, board, target)) }) }) }) // Handle approved comments this.ee.on('approved comments for ' + board + '@' + adminID, ret => { ret.forEach(item => this.downloadComment(item.hash, adminID, board, target)) }) this.getAllowedContentProducers(adminID, board, { comments: true }) } BoardsAPI.prototype.getUserCommentList = function (parent, user, done) { if (!parent || !user) { return console.log('Malformed arguments:', parent, user) } this.resolveIPNS(user, (url, err) => { if (err) { this.ee.emit('error', err) done(err) } else { this.ipfs.ls(url + this.baseurl + 'comments/' + parent, (e, r) => { if (e) { this.ee.emit('error', e) done(e) } else if (r && !r.split) { if (r.Objects && r.Objects[0]) { // If this is not true, then there are no comments console.log('Found', r.Objects[0].Links.length, 'comments for', parent, 'at', user) var l = r.Objects[0].Links.map(i => { return { date: i.Name, hash: i.Hash } }) done(null, l) } } }) } return true // remove myself from listeners }) return this.ee } // API for publishing content and managing to be done later... // Initialize API BoardsAPI.prototype.init = function (done) { if (this.isInit) return this.ipfs.id((err, res) => { if (err) { console.log('Error while getting OWN ID:', err) this.ee.emit('error', err) this.ee.emit('init', err) if (done && done.apply) done(err) } else if (res.ID) { console.log('I am', res.ID) this.id = res.ID this.resolveIPNS(res.ID) console.log('Version is', this.version) this.ipfs.add(new Buffer(this.version), (err2, r) => { if (err2) { this.ee.emit('error', err2) console.log('Error while calculating version hash:', err2) this.ee.emit('init', err2) if (done && done.apply) done(err2) } else { if (r && r.Hash) this.version_hash = r.Hash if (r && r[0] && r[0].Hash) this.version_hash = r[0].Hash console.log('Version hash is', this.version_hash) this.ipfs.version((err, res) => { if (err) { this.ee.emit('error', err) this.ee.emit('init', err) console.log('Error while getting ipfs version:', err) if (done && done.apply) done(err) } else { this.ipfs_version = res.Version.split('-')[0] console.log('IPFS Version is', res.Version) if (semver.satisfies(this.ipfs_version, '~0.4.0')) { console.log('IPFS version is supported') this.ee.emit('init', undefined) this.isInit = true delete this.init_error if (done && done.apply) done(null) } else { var e = { Message: 'IPFS Version not supported. This app supports go-ipfs 0.4.x' } if (done && done.apply) done(e) console.log('Error:', e.Message) this.ee.emit('error', e) this.ee.emit('init', e) } } }) } }) } }) } BoardsAPI.prototype.getEventEmitter = function () { return this.ee } BoardsAPI.prototype.getUsers = function () { return this.users } BoardsAPI.prototype.getMyID = function () { return this.id } module.exports = BoardsAPI