This is a proof of concept for a 'decentralized social network'.
The demo revolves around two main concepts:
Content Addressing: Data is stored and distributed in a content addressed manner. A user's profile & post history can be represented by a single hash.
User-controlled Keys & Auth: Users register a public key to their account and sign updates & authorization tokens in the form of ucans
Requires Node>=15, and yarn
There are four components to the demo:
- a "bluesky" server (federated data & identity management)
- the frontend (simple micro-blogging application)
- the command-line (simple micro-blogging CLI)
- a third-party server that a user can delegate permission to to post on it's behalf (think Buffer)
This project is setup in a yarn workspace. To install dependencies for all sub-projects, just run yarn
from the project root. To build all sub-projects, run yarn build
from the project root.
Run the main bluesky server
cd server
yarn dev
Server will be running at http://localhost:2583
In another console tab, run the frontend
cd frontend
yarn dev
Go to http://localhost:3005
to try the demo.
In another console tab, run the CLI
cd cli
# build, if needed, with `yarn build`
yarn cli
To enable third-party posting, run the third-party server as well in another console
cd third-party
yarn dev
Server will be running on http://localhost:2584
- Ucans: Ucans are distributed user-controlled & signed authorization tokens. They are signed JWTs that indicate what a given user (or keypair) is capable of. A user registers a "root DID" that has full account access, and using that keypair, can delegate some subset of their capabilities (such as posting) to another device, user, or third-party platform.
- CAR files: CAR files are "Content Addressable aRchives". They encode and serialize some set of content addressable objects and allow you to indicate a "root" of the content addressed structure. CAR files allow us to take advantage of content addressing while avoiding the performance hit of content discovery in an in-browser DHT.
- Hash Array Mapped Trie(HAMT): A data structure that functions like a hashmap, but under the hood stores values in a trie.
The user creates an keypair and signs an empty (no attenuation) UCAN. This servers as proof of ownership of the key.
They then send the signed ucan and their requested username to the server.
The server parses and validates the UCAN, checks to make sure the username is available, and registers the DID that issued the UCAN to the requested username.
Note: currently keys are stored in localStorage, which is not a safe location in production
The user maintains a merkelized "User Store" that wraps around a HAMT (described in more detail later). When they create a new post, they add it to the HAMT, indexed by the current post count.
The user signs a UCAN with a valid POST
permission for their username.
The user serializes their user store to a CAR file, and sends it to the server, adding the encoded UCAN to the request in the form of a Bearer token.
The server validates the UCAN, ensuring that it has valid POST
permission, decodes the CAR file, stores it in it's datastore (currently: memory), and updates the user's data root to the root of the CAR file.
Posts are all public for the time being, so no authentication is needed.
The user requests the data for a given DID.
The server serializes the user store to a CAR file and sends it to the user.
The user decodes the CAR file and displays the posts to the user.
The user store is a content-addressed merkelized data structure that encodes some basic information about a user and their post history.
Posts are stored in a HAMT, indexed by the current post count.
An dag-cbor encoded IPLD block contains some basic information about a user and a pointer to the current root of the HAMT.
// dag-cbor encoded user root
type User = {
name: string
did: string
nextPost: number
postsRoot: CID
follows: Follow[]
}
// stored in HAMT
type Post = {
user: string
text: string
}
type Follow = {
username: string
did: string
}
The CID of the IPLD-encoded "User" block is then signed by the user and added to to an "Commit" IPLD object:
type Commit = {
user: CID
sig: Uint8Array
}
The CID of the Commit serves as the root of the datastructure.
User data is persisted in Level which is a simple key-value store. This Blockstore is a mapping of CID -> Bytes
and represents the consituent blocks that make up the UserStore.
The server uses Ucans for authorization.
Body:
{
name: string
}
Auth: A Ucan no attenuation. This token is just used to prove key ownership.
Body: Binary CAR file representing a valid user store
Auth: A valid Ucan with attenuation for the following resource:
{
'bluesky': '${USERNAME}'
'cap': 'POST'
}
Params:
id
: User's DID
Returns: An array of all user DIDs
[
did:key:abcdef.....,
did:key:123456....,
...
]
Params:
id
: User's DID
Returns: Binary CAR file representing the user's current user store
Returns: The server's DID
{
id: "did:key:z6Mkmi4eUvWtRAP6PNB7MnGfUFdLkGe255ftW9sGo28uv44g"
}
Params:
resource
: The user's name
Returns: A (very sparse) webfinger document. It currently only contains the user's DID.
{
id: "did:key:zAbc...."
}