Awesome
<!-- START doctoc generated TOC please keep comment here to allow auto update --> <!-- DON'T EDIT THIS SECTION, INSTEAD RE-RUN doctoc TO UPDATE -->Table of Contents generated with DocToc
- libfritter
- Usage
- API
- new LibFritter([opts])
- fritter.db
- fritter.setUser(archive)
- fritter.prepareArchive(archive)
- fritter.social.getProfile(archive)
- fritter.social.setProfile(archive, profile)
- fritter.social.setAvatar(archive, imgDataBuffer, extension)
- fritter.social.follow(archive, targetUser[, targetUserName])
- fritter.social.unfollow(archive, targetUser)
- fritter.social.listFollowers(archive)
- fritter.social.countFollowers(archive)
- fritter.social.listFriends(archive)
- fritter.social.countFriends(archive)
- fritter.social.isFollowing(archiveA, archiveB)
- fritter.social.isFriendsWith(archiveA, archiveB)
- fritter.feed.post(archive, post)
- fritter.feed.listPosts([opts])
- fritter.feed.countPosts([opts])
- fritter.feed.getThread(url[, opts])
- fritter.feed.vote(archive, data)
- fritter.feed.listVotesFor(subject)
- fritter.notifications.listNotifications([opts])
- fritter.notifications.countNotifications([opts])
libfritter
Data definitions and methods for Fritter, a Twitter clone built on top of Dat. Uses WebDB to read and write records on the Dat network.
See the Fritter app to see this library in use.
const LibFritter = require('@beaker/libfritter')
const fritter = new LibFritter()
await fritter.db.open()
await fritter.db.indexArchive('dat://bob.com')
await fritter.social.getProfile('dat://bob.com') // => ...
Schemas:
- Profile. The schema for user profiles. A very simple "social media" profile: name, bio, profile pic, and a list of followed users.
- Post. The schema for feed posts. Like in Twitter, posts are microblog posts, and can be in reply to other Fritter posts.
- Vote. The schema for votes. In Fritter, only upvotes are used.
Usage
Getting started
LibFritter provides a set of methods to be used on top of a WebDB instance.
Setup will always include the following steps:
// create the libfritter instance
const fritter = new LibFritter()
// open the webdb
await fritter.db.open()
WebDB maintains an index which will determine who shows up in the feed, and whether any read method works for a given archive.
(For instance, you can't call getProfile()
on an archive that hasn't been indexed.)
You can manage the index's membership using WebDB's methods:
// add a user
await fritter.db.indexArchive('dat://bob.com')
// remove a user
await fritter.db.unindexArchive('dat://bob.com')
You can also add individual files to the index, which is helpful when the user navigates to a thread:
// add an individual file
await fritter.db.indexFile('dat://bob.com/posts/1.json')
When you create a dat archive for the local user, you'll want to call prepareArchive()
to setup the folder structure:
var alice = DatArchive.create({title: 'Alice', description: 'My Fritter profile'})
await fritter.prepareArchive(alice)
Profiles
User profiles include a name
, bio
, and an avatar image.
await fritter.social.setProfile(alice, {
name: 'Alice',
bio: 'A cool hacker'
})
await fritter.social.setAvatar(alice, 'iVBORw...rkJggg==', 'png')
await fritter.social.getProfile(alice) /* => {
name: 'Alice',
bio: 'A cool hacker',
avatar: '/avatar.png'
} */
- fritter.social.getProfile(archive)
- fritter.social.setProfile(archive, profile)
- fritter.social.setAvatar(archive, imgDataBuffer, extension)
Social
Every user maintains a public list of other users they follow. You can modify and examine the social graph using these methods.
await fritter.social.follow(alice, bob)
await fritter.social.follow(alice, 'dat://bob.com') // (urls work too)
await fritter.social.listFollowers(bob) // => [{name: 'Alice', bio: 'A cool hacker', ...}]
- fritter.social.follow(archive, targetUser[, targetUserName])
- fritter.social.unfollow(archive, targetUser)
- fritter.social.listFollowers(archive)
- fritter.social.countFollowers(archive)
- fritter.social.listFriends(archive)
- fritter.social.countFriends(archive)
- fritter.social.isFollowing(archiveA, archiveB)
- fritter.social.isFriendsWith(archiveA, archiveB)
Feed
The feed contains simple text-based posts.
// posting a new thread
await fritter.feed.post(alice, {
text: 'Hello, world!',
})
// posting a reply
await fritter.feed.post(alice, {
text: 'Hello, world!',
threadParent: parent.getRecordURL(), // url of message replying to
threadRoot: root.getRecordURL(), // url of topmost ancestor message
createdAt: Date.parse('04 Dec 2017 00:12:00 GMT') // optional
})
The list method will show any indexed posts.
await fritter.feed.listPosts({
fetchAuthor: true,
countVotes: true,
reverse: true,
rootPostsOnly: true,
countReplies: true
})
You can view the posts of an individual user by adding the author
filter, and also narrow down the feed to only include the followed users using the authors
filter.
- fritter.feed.post(archive, post)
- fritter.feed.listPosts([opts])
- fritter.feed.countPosts([opts])
- fritter.feed.getThread(url[, opts])
Like / Unlike
Users can like posts using the votes.
await fritter.feed.vote(alice, {vote: 1, subject: 'dat://bob.com/posts/1.json'})
await fritter.feed.listVotesFor('dat://bob.com/posts/1.json') /* => {
up: 1,
down: 0,
value: 1,
upVoters: ['dat://alice.com']
}
Notifications
You can view recent notifications (mentions, likes and replies on your posts) using the notifications api.
await fritter.notifications.listNotifications() /* => [
{ type: 'mention',
url: 'dat://bob.com/posts/0jc7w0d5cd.json',
createdAt: 15155171572345 },
{ type: 'reply',
url: 'dat://alice.com/posts/0jc7w07be.json',
createdAt: 1515517526289 },
{ type: 'vote',
vote: 1,
subject: 'dat://alice.com/posts/0jc7w079o.json',
origin: 'dat://bob.com',
createdAt: 1515517526308 }
]*/
API
new LibFritter([opts])
const fritter = new LibFritter()
opts
Object.mainIndex
String. The name (in the browser) or path (in node) of the main indexes. Defaults to'fritter'
.DatArchive
Constructor. The class constructor for dat archive instances. If in node, you should specify node-dat-archive.
Create a new LibFritter
instance.
The mainIndex
will control where the indexes are stored.
You can specify different names to run multiple LibFritter instances at once.
fritter.db
The WebDB instance.
fritter.setUser(archive)
fritter.setUser(alice)
archive
DatArchive. The archive which represents the local user.
Sets the local user. Used in notifications to know which posts should be indexed.
fritter.prepareArchive(archive)
await fritter.prepareArchive(alice)
archive
DatArchive. The archive to prepare for use in fritter.
Create needed folders for writing to an archive. This should be called on any archive that represents the local user.
fritter.social.getProfile(archive)
await fritter.social.getProfile(alice) // => {name: 'Alice', bio: 'A cool hacker', avatar: '/avatar.png'}
archive
DatArchive or String. The archive to read.
Get the profile data of the given archive.
fritter.social.setProfile(archive, profile)
await fritter.social.setProfile(alice, {name: 'Alice', bio: 'A cool hacker'})
archive
DatArchive or String. The archive to modify.profile
Object.name
String.bio
String.
Set the profile data of the given archive.
fritter.social.setAvatar(archive, imgDataBuffer, extension)
await fritter.social.setAvatar(alice, myPngData, 'png')
archive
DatArchive or String. The archive to modify.imgDataBuffer
String, ArrayBuffer, or Buffer. The image data to store. If a string, must be base64-encoded.extensions
String. The file-extension of the avatar.
Set the avatar image of the given archive.
fritter.social.follow(archive, targetUser[, targetUserName])
await fritter.social.follow(alice, bob, 'Bob')
archive
DatArchive or String. The archive to modify.targetUser
DatArchive or String. The archive to follow.targetUserName
String. The name of the archive being followed.
Add to the follow-list of the given archive.
fritter.social.unfollow(archive, targetUser)
await fritter.social.unfollow(alice, bob)
archive
DatArchive or String. The archive to modify.targetUser
DatArchive or String. The archive to unfollow.
Remove from the follow-list of the given archive.
fritter.social.listFollowers(archive)
await fritter.social.listFollowers(alice)
archive
DatArchive or String. The archive to find followers of.
List users in db that follow the given archive.
fritter.social.countFollowers(archive)
await fritter.social.countFollowers(alice)
archive
DatArchive or String. The archive to find followers of.
Count users in db that follow the given archive.
fritter.social.listFriends(archive)
await fritter.social.listFriends(alice)
archive
DatArchive or String. The archive to find friends of.
List users in db that mutually follow the given archive.
fritter.social.countFriends(archive)
await fritter.social.countFriends(alice)
archive
DatArchive or String. The archive to find friends of.
Count users in db that mutually follow the given archive.
fritter.social.isFollowing(archiveA, archiveB)
await fritter.social.isFollowing(alice, bob) // => true
archiveA
DatArchive or String. The archive to test.archiveB
DatArchive or String. The follow target.
Test if archiveA
is following archiveB
.
fritter.social.isFriendsWith(archiveA, archiveB)
await fritter.social.isFriendsWith(alice, bob) // => true
archiveA
DatArchive or String.archiveB
DatArchive or String.
Test if archiveA
and archiveB
are mutually following each other.
fritter.feed.post(archive, post)
// posting a new thread
await fritter.feed.post(alice, {
text: 'Hello, world!',
})
// posting a reply
await fritter.feed.post(alice, {
text: 'Hello, world!',
threadParent: parent.getRecordURL(), // url of message replying to
threadRoot: root.getRecordURL() // url of topmost ancestor message
})
archive
DatArchive or String. The archive to modify.post
Object.text
String. The content of the post.threadParent
String. The URL of the parent post in the thread. Only needed in a reply; must also includethreadRoot
.threadRoot
String. The URL of the root post in the thread. Only needed in a reply; must also includethreadParent
.mentions
Array<{url: String, name: String}. An array of users mentioned in the posts, who should be pinged.
Post a new message to the feed.
fritter.feed.listPosts([opts])
await fritter.feed.listPosts({limit: 30})
opts
Object.author
String | DatArchive. Single-author filter.authors
Array<String>. Multi-author filter.rootPostsOnly
Boolean. Remove posts in the feed that are repliesafter
Number. Filter out posts created before the given timestamp.before
Number. Filter out posts created after the given timestamp.limit
Number. Add a limit to the number of results given.offset
Number. Add an offset to the results given. Useful in pagination.reverse
Boolean. Reverse the order of the output.fetchAuthor
Boolean. Populate the.author
attribute of the result objects with the author's profile record.countReplies
Boolean. Populate the.replies
attribute of the result objects with number of replies to the post.countVotes
Boolean. Populate the.votes
attribute of the result objects with the results ofcountVotesFor
.
Fetch a list of posts in the feed index.
fritter.feed.countPosts([opts])
await fritter.feed.countPosts({author: alice})
opts
Object.author
String | DatArchive. Single-author filter.authors
Array<String>. Multi-author filter.rootPostsOnly
Boolean. Remove posts in the feed that are repliesafter
Number. Filter out posts created before the given timestamp.before
Number. Filter out posts created after the given timestamp.limit
Number. Add a limit to the number of results given.offset
Number. Add an offset to the results given. Useful in pagination.
Count posts in the feed index.
fritter.feed.getThread(url[, opts])
await fritter.feed.getThread('dat://alice.com/posts/1.json')
url
String. The URL of the thread.opts
Object.authors
Array<String>. Filter the posts in the thread down to those published by the given list of archive urls.
Fetch a discussion thread, including all replies.
fritter.feed.vote(archive, data)
await fritter.feed.vote(alice, {
vote: 1,
subject: 'dat://bob.com/posts/1.json'
})
archive
DatArchive or String. The archive to modify.data
Object.vote
Number. The vote value. Must be -1 (dislike), 0 (no opinion), or 1 (like).subject
String. The url of the item being voted on.
Publish a vote on the given subject.
fritter.feed.listVotesFor(subject)
await fritter.feed.listVotesFor('dat://bob.com/posts/1.json')
subject
String. The url of the item.
Returns a vote tabulation of the given subject.
fritter.notifications.listNotifications([opts])
await fritter.notifications.listNotifications({limit: 30})
opts
Object.after
Number. Filter out notifications created before the given timestamp.before
Number. Filter out notifications created after the given timestamp.limit
Number. Add a limit to the number of results given.offset
Number. Add an offset to the results given. Useful in pagination.reverse
Boolean. Reverse the order of the output.fetchAuthor
Boolean. Populate the.author
attribute of the result objects with the author's profile record.fetchPost
Boolean. Populate the.post
attribute of the result objects with the post that's the subject of the notification.
Fetch a list of events in the notifications index.
fritter.notifications.countNotifications([opts])
await fritter.notifications.countNotifications()
opts
Object.after
Number. Filter out notifications created before the given timestamp.before
Number. Filter out notifications created after the given timestamp.limit
Number. Add a limit to the number of results given.offset
Number. Add an offset to the results given. Useful in pagination.
Fetch a count of events in the notifications index.