2015-11-11 15:51:07 +01:00
|
|
|
/*
|
|
|
|
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
|
2015-11-14 15:03:38 +01:00
|
|
|
|
|
|
|
Needs to be browserified to work in the browser
|
2015-11-11 15:51:07 +01:00
|
|
|
*/
|
|
|
|
|
2015-11-14 18:26:56 +01:00
|
|
|
// EventEmitter used to communicate with clients
|
2015-11-14 15:03:38 +01:00
|
|
|
var EventEmitter = require('wolfy87-eventemitter')
|
|
|
|
|
2015-11-11 09:18:36 +01:00
|
|
|
function asObj(str,done){
|
|
|
|
var obj
|
|
|
|
try {
|
|
|
|
obj = JSON.parse(str)
|
|
|
|
} catch (e) {
|
2015-11-11 12:38:10 +01:00
|
|
|
return done(e,null)
|
2015-11-11 09:18:36 +01:00
|
|
|
}
|
2015-11-11 12:38:10 +01:00
|
|
|
done(null,obj)
|
2015-11-11 09:18:36 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
function replyAsObj(res,isJson,done){
|
|
|
|
if(res.readable){
|
|
|
|
// Is a stream
|
2015-11-14 12:06:37 +01:00
|
|
|
console.log('got stream')
|
2015-11-11 09:18:36 +01:00
|
|
|
res.setEncoding('utf8')
|
|
|
|
var data = ''
|
|
|
|
res.on('data',d => {
|
2015-11-14 12:06:37 +01:00
|
|
|
console.log('got stream data:',d)
|
2015-11-11 09:18:36 +01:00
|
|
|
data += d
|
|
|
|
})
|
|
|
|
res.on('end',() => {
|
|
|
|
if(isJson) {
|
|
|
|
asObj(data,done)
|
|
|
|
} else {
|
|
|
|
done(null,data)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
} else {
|
2015-11-14 16:26:03 +01:00
|
|
|
//console.log('got string:',res)
|
2015-11-11 09:18:36 +01:00
|
|
|
// Is a string
|
|
|
|
if(isJson){
|
|
|
|
asObj(res,done)
|
|
|
|
} else {
|
|
|
|
done(null,res)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
function BoardsAPI(ipfs){
|
|
|
|
this.ipfs = ipfs
|
2015-11-18 16:56:07 +01:00
|
|
|
this.version = 'ipfs:boards:version:dev'
|
2015-11-16 11:56:25 +01:00
|
|
|
this.baseurl = '/ipfs-boards-profile/'
|
2015-11-11 15:22:58 +01:00
|
|
|
this.users = {} // userID : profileHash
|
2015-11-14 18:38:46 +01:00
|
|
|
this.resolving_ipns = {} // to check if a resolve is already in progress
|
2015-11-14 18:26:56 +01:00
|
|
|
this.ee = new EventEmitter()
|
2015-11-14 23:55:55 +01:00
|
|
|
if(localStorage !== undefined){
|
|
|
|
// Use localStorage to store the IPNS cache
|
|
|
|
var stored = localStorage.getItem('ipfs-boards-user-cache')
|
|
|
|
try {
|
|
|
|
this.users = JSON.parse(stored)
|
|
|
|
if(this.users === null || this.users === undefined){
|
|
|
|
this.users = {}
|
|
|
|
}
|
|
|
|
} catch(e){
|
|
|
|
this.users = {}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
BoardsAPI.prototype.backupCache = function(){
|
|
|
|
if(localStorage !== undefined){
|
|
|
|
// Use localStorage to store the IPNS cache
|
|
|
|
localStorage.setItem('ipfs-boards-user-cache',JSON.stringify(this.users))
|
|
|
|
}
|
2015-11-11 15:22:58 +01:00
|
|
|
}
|
|
|
|
|
2015-11-14 18:26:56 +01:00
|
|
|
// Rewrote this to use event emitters. Should also add periodic rechecking
|
|
|
|
BoardsAPI.prototype.resolveIPNS = function(n,handler){
|
2015-11-16 13:20:15 +01:00
|
|
|
if(handler && handler.apply) this.ee.on(n,handler)
|
2015-11-11 12:38:10 +01:00
|
|
|
var cached = this.users[n]
|
|
|
|
if(cached){
|
2015-11-14 18:26:56 +01:00
|
|
|
this.ee.emit(n,cached)
|
2015-11-18 17:19:33 +01:00
|
|
|
console.log(n,'was cached',cached)
|
2015-11-18 16:56:07 +01:00
|
|
|
} else {
|
|
|
|
console.log(n,'not cached')
|
2015-11-11 12:38:10 +01:00
|
|
|
}
|
2015-11-14 18:38:46 +01:00
|
|
|
if(this.resolving_ipns[n] != true){
|
|
|
|
this.resolving_ipns[n] = true
|
|
|
|
this.ipfs.name.resolve(n,(err,r) => {
|
|
|
|
if(err){
|
|
|
|
// Communicate error
|
2015-11-18 15:40:29 +01:00
|
|
|
this.ee.emit('error',err)
|
|
|
|
} else {
|
2015-11-18 16:56:07 +01:00
|
|
|
var url = r.Path
|
|
|
|
if(url === undefined){
|
|
|
|
console.log('UNDEFINED URL',r)
|
|
|
|
}
|
|
|
|
if(this.users[n] != url) this.isUserProfile(url,(isit,err) => {
|
|
|
|
if(isit){
|
2015-11-18 15:40:29 +01:00
|
|
|
console.log(n,'is a user')
|
2015-11-18 17:19:33 +01:00
|
|
|
if(this.users[n] === undefined) this.ee.emit('user',n,url)
|
2015-11-18 16:56:07 +01:00
|
|
|
this.users[n] = url
|
|
|
|
this.ee.emit(n,url)
|
|
|
|
this.backupCache()
|
|
|
|
} else {
|
|
|
|
console.log(n,'not a user')
|
|
|
|
this.ee.emit(n,undefined,'not a valid profile: '+err)
|
2015-11-18 15:40:29 +01:00
|
|
|
}
|
|
|
|
this.resolving_ipns[n] = false
|
2015-11-18 16:56:07 +01:00
|
|
|
return true // Remove from listeners
|
2015-11-18 15:40:29 +01:00
|
|
|
})
|
2015-11-14 18:38:46 +01:00
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
2015-11-14 18:26:56 +01:00
|
|
|
return this.ee
|
2015-11-11 12:38:10 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
BoardsAPI.prototype.isUserProfile = function(addr,done){
|
2015-11-18 15:40:29 +01:00
|
|
|
if(addr === undefined) return console.log('Asked to check if undefined is a profile')
|
2015-11-16 11:56:25 +01:00
|
|
|
this.ipfs.cat(addr+this.baseurl+'ipfs-boards-version.txt',(err,r) => {
|
2015-11-18 16:56:07 +01:00
|
|
|
if(err) return done(false,err)
|
2015-11-11 12:38:10 +01:00
|
|
|
replyAsObj(r,false,(_,res) => {
|
|
|
|
var v = res.trim()
|
2015-11-18 16:56:07 +01:00
|
|
|
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+'"')
|
2015-11-11 12:38:10 +01:00
|
|
|
})
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
BoardsAPI.prototype.searchUsers = function(){
|
2015-11-11 09:18:36 +01:00
|
|
|
// Look at our peers
|
2015-11-11 12:38:10 +01:00
|
|
|
this.ipfs.swarm.peers((err,r) => {
|
|
|
|
if(err) return console.log(err)
|
|
|
|
replyAsObj(r,true,(e,reply) => {
|
2015-11-11 15:22:58 +01:00
|
|
|
console.log('Checking',reply.Strings.length,'peers')
|
2015-11-14 15:03:38 +01:00
|
|
|
reply.Strings.forEach(item => {
|
2015-11-18 16:56:07 +01:00
|
|
|
var ss = item.split('/')
|
|
|
|
var n = ss[ss.length-1]
|
|
|
|
this.resolveIPNS(n)
|
2015-11-14 15:03:38 +01:00
|
|
|
})
|
2015-11-11 09:18:36 +01:00
|
|
|
})
|
|
|
|
})
|
2015-11-18 16:56:07 +01:00
|
|
|
// Look for who has the correct version file, they probably have a profile
|
|
|
|
// Disabled at the moment
|
|
|
|
/*
|
|
|
|
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)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
*/
|
2015-11-14 18:26:56 +01:00
|
|
|
return this.ee
|
2015-11-11 09:18:36 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
BoardsAPI.prototype.getProfile = function(userID,done){
|
2015-11-14 12:06:37 +01:00
|
|
|
console.log('profile requested for',userID)
|
2015-11-14 18:26:56 +01:00
|
|
|
this.resolveIPNS(userID,(url,err) => {
|
2015-11-11 09:18:36 +01:00
|
|
|
if(err){
|
2015-11-16 11:56:25 +01:00
|
|
|
this.ee.emit('error',err)
|
2015-11-11 09:18:36 +01:00
|
|
|
done(err,null)
|
2015-11-14 15:03:38 +01:00
|
|
|
} else {
|
|
|
|
// Download actual profile
|
2015-11-16 11:56:25 +01:00
|
|
|
this.ipfs.cat(url+this.baseurl+'profile.json',(err2,res) => {
|
2015-11-14 15:03:38 +01:00
|
|
|
if(err2){
|
2015-11-16 11:56:25 +01:00
|
|
|
this.ee.emit('error',err2)
|
2015-11-14 15:03:38 +01:00
|
|
|
done(err2,null)
|
|
|
|
} else {
|
|
|
|
// It already returns a JSON?
|
2015-11-18 16:56:07 +01:00
|
|
|
this.ee.emit('profile for '+userID,res)
|
2015-11-14 15:03:38 +01:00
|
|
|
done(null,res)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
// Get other info
|
2015-11-16 11:56:25 +01:00
|
|
|
this.ipfs.ls(url+this.baseurl+'boards/',(err2,res) => {
|
2015-11-14 15:03:38 +01:00
|
|
|
if(!err2){
|
|
|
|
var l = res.Objects[0].Links.map(i => {
|
|
|
|
return { name: i.Name, hash: i.Hash }
|
|
|
|
})
|
2015-11-16 13:20:15 +01:00
|
|
|
this.ee.emit('boards for '+userID,l)
|
2015-11-16 11:56:25 +01:00
|
|
|
} else {
|
|
|
|
this.ee.emit('error',err2)
|
|
|
|
}
|
2015-11-14 15:03:38 +01:00
|
|
|
})
|
|
|
|
}
|
2015-11-14 18:26:56 +01:00
|
|
|
return true // remove myself from listeners
|
2015-11-11 09:18:36 +01:00
|
|
|
})
|
2015-11-14 18:26:56 +01:00
|
|
|
return this.ee
|
2015-11-11 09:18:36 +01:00
|
|
|
}
|
|
|
|
|
2015-11-16 14:35:27 +01:00
|
|
|
BoardsAPI.prototype.getBoardSettings = function(userID,board){
|
2015-11-14 18:26:56 +01:00
|
|
|
this.resolveIPNS(userID,(r,e) => {
|
2015-11-14 16:26:03 +01:00
|
|
|
if(e){
|
2015-11-16 11:56:25 +01:00
|
|
|
this.ee.emit('error',e)
|
2015-11-11 09:18:36 +01:00
|
|
|
} else {
|
2015-11-16 11:56:25 +01:00
|
|
|
var url = r+this.baseurl+'boards/'+board+'/settings.json'
|
2015-11-16 14:35:27 +01:00
|
|
|
this.ipfs.cat(url,(err,settings) => {
|
2015-11-14 16:26:03 +01:00
|
|
|
if(err){
|
2015-11-16 11:56:25 +01:00
|
|
|
this.ee.emit('error',err)
|
2015-11-14 16:26:03 +01:00
|
|
|
} else {
|
2015-11-16 14:35:27 +01:00
|
|
|
// SETTINGS file is here, need to parse it a little bit
|
|
|
|
this.ee.emit('settings for '+board+'@'+userID,settings,r)
|
|
|
|
if(settings.whitelist == true){
|
|
|
|
// Get the whitelist
|
|
|
|
var url = r+this.baseurl+'boards/'+board+'/whitelist'
|
|
|
|
this.ipfs.cat(url,(err,whitelist) => {
|
|
|
|
if(err){
|
|
|
|
this.ee.emit('error',err)
|
|
|
|
// Emit an empty whitelist.
|
|
|
|
this.emit('whitelist for '+board+'@'+userID,[])
|
|
|
|
} else {
|
|
|
|
// Send whitelist
|
|
|
|
var w = whitelist.split(' ')
|
|
|
|
this.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)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
2015-11-14 16:26:03 +01:00
|
|
|
}
|
|
|
|
})
|
2015-11-11 09:18:36 +01:00
|
|
|
}
|
2015-11-14 18:26:56 +01:00
|
|
|
return true // remove myself from listeners
|
2015-11-11 09:18:36 +01:00
|
|
|
})
|
2015-11-16 14:35:27 +01:00
|
|
|
return this.ee
|
2015-11-11 09:18:36 +01:00
|
|
|
}
|
|
|
|
|
2015-11-14 15:03:38 +01:00
|
|
|
BoardsAPI.prototype.getPostsInBoard = function(adminID,board){
|
2015-11-16 14:35:27 +01:00
|
|
|
var downloadPost = hash => {
|
|
|
|
this.ipfs.cat(hash,(err2,r) => {
|
|
|
|
if(err2){
|
|
|
|
this.ee.emit('error',err2)
|
|
|
|
console.log('Could not download post',hash,'of',board+'@'+adminID)
|
|
|
|
} else {
|
|
|
|
// It already returns a JSON?
|
|
|
|
this.ee.emit('post in '+board+'@'+adminID,r,hash)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
this.getBoardSettings(adminID,board)
|
|
|
|
this.ee.on('settings for'+board+'@'+adminID,function(settings,addr){
|
|
|
|
// Download posts based on settings
|
|
|
|
if(settings.approval_required == true){
|
|
|
|
// Get approved posts list
|
|
|
|
var a = addr+this.baseurl+'boards/'+board+'/approved/posts/'
|
|
|
|
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 posts for '+board+'@'+adminID,ret)
|
|
|
|
// Automatically download approved posts
|
|
|
|
ret.forEach(item => downloadPost(item.hash))
|
|
|
|
}
|
|
|
|
})
|
|
|
|
if(settings.whitelist == true){
|
|
|
|
// TODO: Download all posts from whitelisted users
|
|
|
|
}
|
|
|
|
} else if(settings.whitelist_only == true){
|
|
|
|
// TODO: download all posts from whitelisted users
|
|
|
|
} else if(settings.blacklist == true){
|
|
|
|
// TODO: get the blacklist, then start downloading posts from everyone not in the blacklist
|
|
|
|
}
|
2015-11-11 15:22:58 +01:00
|
|
|
})
|
2015-11-16 14:35:27 +01:00
|
|
|
// Get the admin's posts
|
2015-11-16 11:56:25 +01:00
|
|
|
this.getUserPostListInBoard(adminID,board,(err,res) => {
|
2015-11-14 15:03:38 +01:00
|
|
|
if(err){
|
|
|
|
console.log(err)
|
2015-11-16 14:35:27 +01:00
|
|
|
} else res.forEach(item => downloadPost(item.hash))
|
2015-11-14 15:03:38 +01:00
|
|
|
})
|
2015-11-14 18:26:56 +01:00
|
|
|
return this.ee
|
2015-11-11 09:18:36 +01:00
|
|
|
}
|
|
|
|
|
2015-11-14 15:03:38 +01:00
|
|
|
BoardsAPI.prototype.getUserPostListInBoard = function(user,board,done){
|
2015-11-14 18:26:56 +01:00
|
|
|
this.resolveIPNS(user,(url,err) => {
|
2015-11-11 15:22:58 +01:00
|
|
|
if(err){
|
2015-11-16 11:56:25 +01:00
|
|
|
this.ee.emit('error',err)
|
2015-11-11 15:22:58 +01:00
|
|
|
done(err)
|
2015-11-16 11:56:25 +01:00
|
|
|
} else this.ipfs.ls(url+this.baseurl+'posts/'+board,(e,r) => {
|
2015-11-11 15:22:58 +01:00
|
|
|
if(e){
|
2015-11-16 11:56:25 +01:00
|
|
|
this.ee.emit('error',e)
|
2015-11-11 15:22:58 +01:00
|
|
|
done(e)
|
2015-11-14 15:03:38 +01:00
|
|
|
} else if(r && !r.split){
|
|
|
|
console.log('Found',r.Objects[0].Links.length,'posts in',board,'at',user)
|
2015-11-16 11:56:25 +01:00
|
|
|
this.ee.emit('post count',board,user,r.Objects[0].Links.length)
|
2015-11-14 15:03:38 +01:00
|
|
|
var l = r.Objects[0].Links.map(i => {
|
2015-11-11 15:22:58 +01:00
|
|
|
return { date: i.Name, hash: i.Hash }
|
|
|
|
})
|
2015-11-14 15:03:38 +01:00
|
|
|
done(null,l)
|
2015-11-11 15:22:58 +01:00
|
|
|
}
|
|
|
|
})
|
2015-11-14 18:26:56 +01:00
|
|
|
return true // remove myself from listeners
|
2015-11-11 15:22:58 +01:00
|
|
|
})
|
2015-11-16 11:56:25 +01:00
|
|
|
return this.ee
|
2015-11-11 09:18:36 +01:00
|
|
|
}
|
|
|
|
|
2015-11-16 11:56:25 +01:00
|
|
|
BoardsAPI.prototype.getCommentsFor = function(parent,board,adminID){
|
2015-11-14 23:55:55 +01:00
|
|
|
// Create an EventEmitter, start looking and emit an event for every new comment
|
2015-11-16 11:56:25 +01:00
|
|
|
// Consider the rules of @adminID#board
|
2015-11-11 09:18:36 +01:00
|
|
|
}
|
|
|
|
|
2015-11-14 23:55:55 +01:00
|
|
|
// API for publishing content and managing to be done later...
|
2015-11-11 09:18:36 +01:00
|
|
|
|
|
|
|
// Initialize API
|
|
|
|
BoardsAPI.prototype.init = function(done){
|
|
|
|
this.ipfs.id( (err, res) => {
|
|
|
|
if(err){
|
2015-11-16 11:56:25 +01:00
|
|
|
console.log('Error while getting OWN ID:',err)
|
|
|
|
this.ee.emit('error',err)
|
2015-11-11 09:18:36 +01:00
|
|
|
done(err)
|
2015-11-14 23:55:55 +01:00
|
|
|
} else if(res.ID){
|
2015-11-11 09:18:36 +01:00
|
|
|
console.log('I am',res.ID)
|
|
|
|
this.id = res.ID
|
2015-11-18 15:40:29 +01:00
|
|
|
this.resolveIPNS(res.ID)
|
2015-11-11 09:18:36 +01:00
|
|
|
console.log('Version is',this.version)
|
2015-11-18 16:56:07 +01:00
|
|
|
this.ipfs.add(new Buffer('ipfs:boards:version:'+this.version),{n: true},(err2,r) => {
|
2015-11-16 11:56:25 +01:00
|
|
|
if(err2){
|
|
|
|
this.ee.emit('error',err2)
|
|
|
|
console.log('Error while calculating version hash:',err2)
|
|
|
|
done(err2)
|
2015-11-14 23:55:55 +01:00
|
|
|
} else {
|
2015-11-18 16:56:07 +01:00
|
|
|
if(r && r.Hash) this.version_hash = r.Hash
|
2015-11-15 21:10:58 +01:00
|
|
|
if(r && r[0] && r[0].Hash) this.version_hash = r[0].Hash
|
2015-11-14 23:55:55 +01:00
|
|
|
console.log('Version hash is',this.version_hash)
|
|
|
|
done(null)
|
2015-11-14 15:03:38 +01:00
|
|
|
}
|
2015-11-11 09:18:36 +01:00
|
|
|
})
|
|
|
|
}
|
|
|
|
})
|
2015-11-16 13:20:15 +01:00
|
|
|
this.ipfs.version((err,res) => {
|
|
|
|
if(err){
|
|
|
|
this.ee.emit('error',err)
|
|
|
|
} else {
|
|
|
|
this.ipfs_version = res.Version
|
|
|
|
console.log('IPFS Version is',res.Version)
|
|
|
|
}
|
|
|
|
})
|
2015-11-11 09:18:36 +01:00
|
|
|
}
|
|
|
|
|
2015-11-16 14:35:27 +01:00
|
|
|
BoardsAPI.prototype.getEventEmitter = function(){
|
|
|
|
return this.ee
|
|
|
|
}
|
|
|
|
|
2015-11-11 09:18:36 +01:00
|
|
|
module.exports = BoardsAPI
|