npm package discovery and stats viewer.

Discover Tips

  • General search

    [free text search, go nuts!]

  • Package details

    pkg:[package-name]

  • User packages

    @[username]

Sponsor

Optimize Toolset

I’ve always been into building performant and accessible sites, but lately I’ve been taking it extremely seriously. So much so that I’ve been building a tool to help me optimize and monitor the sites that I build to make sure that I’m making an attempt to offer the best experience to those who visit them. If you’re into performant, accessible and SEO friendly sites, you might like it too! You can check it out at Optimize Toolset.

About

Hi, 👋, I’m Ryan Hefner  and I built this site for me, and you! The goal of this site was to provide an easy way for me to check the stats on my npm packages, both for prioritizing issues and updates, and to give me a little kick in the pants to keep up on stuff.

As I was building it, I realized that I was actually using the tool to build the tool, and figured I might as well put this out there and hopefully others will find it to be a fast and useful way to search and browse npm packages as I have.

If you’re interested in other things I’m working on, follow me on Twitter or check out the open source projects I’ve been publishing on GitHub.

I am also working on a Twitter bot for this site to tweet the most popular, newest, random packages from npm. Please follow that account now and it will start sending out packages soon–ish.

Open Software & Tools

This site wouldn’t be possible without the immense generosity and tireless efforts from the people who make contributions to the world and share their work via open source initiatives. Thank you 🙏

© 2024 – Pkg Stats / Ryan Hefner

@beaker/libfritter

v2.2.6

Published

Data definitions and methods for Fritter

Downloads

30

Readme

Table of Contents generated with DocToc

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'
} */

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', ...}]

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.

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 include threadRoot.
    • threadRoot String. The URL of the root post in the thread. Only needed in a reply; must also include threadParent.
    • 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. Multi-author filter.
    • rootPostsOnly Boolean. Remove posts in the feed that are replies
    • after 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 of countVotesFor.

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. Multi-author filter.
    • rootPostsOnly Boolean. Remove posts in the feed that are replies
    • after 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. 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.