Skip to content
This repository was archived by the owner on Dec 18, 2024. It is now read-only.

Latest commit

 

History

History
321 lines (223 loc) · 16.6 KB

01_Basics.md

File metadata and controls

321 lines (223 loc) · 16.6 KB

Chapter 1 - Laying the Foundation

The basics of OrbitDB include installing OrbitDB (and IPFS), setting up an isomorphic project that runs in both Node.js and the browser, creating databases, and understanding how to choose datastores.

Installing the requirements: IPFS and OrbitDB

Before you start working on the core functionality behind the peer-to-peer sharing of sheet music, you’ll need to get the foundations of our distributed database set up.

You will need to get the code for OrbitDB and its dependency, IPFS, and make it available to your project. The process is different between the browser and Node.js, so we cover both here.

Note: Both OrbitDB and js-ipfs are open source, which give you the ability to build and even contribute to the code. This will be covered in detail in Part 3.

In Node.js

Choose a project directory and cd to there from your command line. Then run the following command:

$ npm init --yes # use npm defaults, you can edit this later
$ npm install --save orbit-db ipfs

This will create a package.json, package-lock.json, and node_modules folder.

Note: If you're running on a Windows prompt, or if you don't have certain build tools like g++ and python installed, you may see a noisy console output with lots of warnings and errors. Keep going, your code should still run.

If you want to use Git to track your progress, we also suggest the following:

$ git init
$ echo node_modules > .gitignore

Of course, be careful before copying and pasting any commands anyone ever tells you into your terminal. If you don't understand a command, figure out what it is supposed to do, before copying it over. Copy and paste at your own risk.

Note: This code was tested on Node v11.14.0. Your mileage for other versions may vary.

In the Browser

If you're using the browser for this tutorial, we recommend using unpkg for obtaining pre-built, minified versions of both IPFS and OrbitDB. Simply include these in your HTML:

<script src="https://unpkg.com/ipfs@0.55.1/dist/index.min.js"></script>
<script src="https://unpkg.com/orbit-db@0.26.1/dist/orbitdb.min.js"></script>

You will now have global Ipfs and OrbitDB objects available to you. You will see how we'll use these later.

Creating the isomorphic bookends

Since OrbitDB works in the browser and Node.js, you're going to want to make the library as isomorphic as possible. This means we want the same code to run in the browser as runs in REPL or local environment. Luckily, you will have the luxury of using the same language, JavaScript, for both Node.js and browser environments.

Create a new file called newpieceplease.js and put this code in there:

class NewPiecePlease {
  constructor (Ipfs, OrbitDB) {
    this.Ipfs = Ipfs
    this.OrbitDB = OrbitDB
  }
}

try {
    const Ipfs = require('ipfs')
    const OrbitDB = require('orbit-db')

    module.exports = exports = new NewPiecePlease(Ipfs, OrbitDB)
} catch (e) {
    window.NPP = new NewPiecePlease(window.Ipfs, window.OrbitDB)
}

Source: GitHub, or on IPFS at QmRZycUKy3MnRKRxkLu8jTzBEVHZovsYcbhdiwLQ221eBP.

In the browser, you can include this file in a script tag and have an NPP object at your disposal. In Node.js, you can simply call something like:

$ node

> const NPP = require('./newpieceplease')

Not much should happen either way, since there's not much code there yet. For now just make sure you can create the NPP constant.

What just happened?

Using some key JavaScript features, you have created the shell for our application that runs in both Node.js and the browser. It defines a new class called NewPiecePlease, with a constructor that takes two arguments

  1. IPFS for the js-ipfs constructor
  2. OrbitDB for the orbit-db constructor

From here on out, we will ignore these isometric bookends and concentrate wholly on the NewPiecePlease class.

Instantiating IPFS and OrbitDB

OrbitDB requires a running IPFS node to operate, so you will create one here and notify OrbitDB about it.

Note: We have designed Chapters 1 and 2 of the tutorial to work offline, not requiring any internet connectivity or connections to peers.

class NewPiecePlease {
  constructor (Ipfs, OrbitDB) {
    this.OrbitDB = OrbitDB
    this.Ipfs = Ipfs
  }

  async create() {
    this.node = await this.Ipfs.create({
      preload: { enabled: false },
      repo: './ipfs',
      EXPERIMENTAL: { pubsub: true },
      config: {
        Bootstrap: [],
        Addresses: { Swarm: [] }
      }
    })

    this._init()
  }

  async _init () {
    this.orbitdb = await this.OrbitDB.createInstance(this.node)
    this.defaultOptions = { accessController: {
      write: [this.orbitdb.identity.id]
      }
    }
    if(this.onready) this.onready();
  }
}

Source: GitHub.

This allows you to run something like the following in your application code:

NPP.onready = () => {
   console.log(NPP.orbitdb.identity.id)
}

NPP.create()

In the output you will see something called a "multihash", like QmPSicLtjhsVifwJftnxncFs4EwYTBEjKUzWweh1nAA87B. This is the identifier of your IPFS node. (You may have noticed we referenced multihashes above for the code examples: these are the multihashes you can use to download the example code files, if GitHub is down.)

What just happened?

Once calling create, the Ipfs.create line creates a new IPFS node. Note the default settings:

  • preload: { enabled: false } disables the use of so-called "pre-load" IPFS nodes. These nodes exist to help load balance the global network and prevent DDoS. However, these nodes can go down and cause errors. Since we are only working offline for now, we include this line to disable them.
  • repo: './ipfs' designates the path of the repo in Node.js only. In the browser, you can actually remove this line. The default setting is a folder called .jsipfs in your home directory. You will see why we choose this specific location for the folder later.
  • EXPERIMENTAL: { pubsub: true } enables IPFS pubsub, which is a method of communicating between nodes and is required for OrbitDB usage, despite whether or not we are connected to other peers.
  • config: { Bootstrap: [], Addresses: { Swarm: [] }} sets both our bootstrap peers list (peers that are loaded on instantiation) and swarm peers list (peers that can connect and disconnect at any time to empty. We will populate these later.

By running the code above, you have created a new IPFS node that works locally and is not connected to any peers. You have also loaded a new orbitdb object into memory, ready to create databases and manage data.

You are now ready to use OrbitDB!

What else happened in Node.js?

When you ran the code in Node.js, you created two folders in your project structure: 'orbitdb/ and ipfs/.

$ # slashes added to ls output for effect
$ ls orbitdb/
QmNrPunxswb2Chmv295GeCvK9FDusWaTr1ZrYhvWV9AtGM/

$ ls ipfs/
blocks/  config  datastore/  datastore_spec  keys/  version

Looking inside the orbitdb/ folder you will see that the subfolder has the same ID as orbitdb, as well as the IPFS node. This is purposeful, as this initial folder contains metadata that OrbitDB needs to operate. See Part 3 for detailed information about this.

The ipfs/ folder contains all of your IPFS data. Explaining this in depth is outside of the scope of this tutorial, but the curious can find out more here.

What else happened in the browser?

In the browser IPFS content is handled inside of IndexedDB, a persistent storage mechanism for browsers

An image showing the IPFS IndexedDB databases in Firefox

Note since you have not explicitly defined a database in the browser, no IndexedDB databases have been created for OrbitDB yet.

Caution! iOS and Android have been known to purge IndexedDB if storage space needs to be created inside of your phone. We recommend creating robust backup mechanisms at the application layer

Creating a database

Your users will want to create a catalog of musical pieces to practice. You will now create this database, and ensure that only that user can write to it.

Expand your _init function to the following:

  async _init () {
    this.orbitdb = await this.OrbitDB.createInstance(this.node)
    this.defaultOptions = { accessController: {
      write: [this.orbitdb.identity.id]
      }
    }
+   const docStoreOptions = {
+     ...this.defaultOptions,
+     indexBy: 'hash',
+   }
+   this.pieces = await this.orbitdb.docstore('pieces', docStoreOptions)
    if (this.onready) this.onready();
  }

Then, in your application code, run this:

NPP.onready = () => {
   console.log(NPP.pieces.id)
}

You will see something like the following as an output: /orbitdb/zdpuB3VvBJHqYCocN4utQrpBseHou88mq2DLh7bUkWviBQSE3/pieces. This is the id, or address (technically a multiaddress) of this database. It is important for you to not only know this, but also to understand what it is. This string is composed of 3 parts, separated by /s:

  1. The first bit, /orbitdb., is the protocol. It tells you that this address is an OrbitDB address.
  2. The second, or middle, part zdpuB3VvBJHqYCocN4utQrpBseHou88mq2DLh7bUkWviBQSE3 that is the most interesting. This is the Content ID (CID) of the database manifest, which contains:
    • The access control list of the database
    • The type of the database
    • The name of the database
  3. The final part is the name you provided, in this case pieces, which becomes the final part of the multiaddress

Note: Addresses that start with Qm… are typically CIDv0 content addresses, while addresses that start with zdpu…. are CIDv1. Misunderstanding OrbitDB addresses can lead to some very unexpected - sometimes hilarious, sometimes disastrous outcomes.

What just happened?

Your code created a local OrbitDB database, of type "docstore", writable only by the user who created it.

  • defaultOptions and docStoreOptions define the parameters for the database we are about to create.
    • accessController: { write: [orbitdb.identity.id] } defines the ACL, or "Access Control List". In this instance we are restricting write access to ONLY the OrbitDB instances identified by our particular id
    • indexBy: "hash" is a docstore-specific option, which specifies which field to index our database by
  • pieces = await orbitdb.docstore('pieces', options) is the magic line that creates the database. Once this line is completed, the database is open and can be acted upon.

Caution! A note about identity: Your public key is not your identity. We repeat, your public key is not your identity. That being said, it is often used as such for convenience's sake, and the lack of better alternatives. So, in the early parts of this tutorial we say "writable only to you" when we really mean "writable only by an OrbitDB instance on top of an IPFS node that has the correct id, which we are assuming is controlled by you."

What else happened in Node.js?

You will see some activity inside your project's orbitdb/ folder. This is good.

$ ls orbitdb/
12D3KooWNZa53tfgFUmUCw6fALEmkfCoubSyHwZpzkuQtwrMVg6L/

$ ls orbitdb/12D3KooWNZa53tfgFUmUCw6fALEmkfCoubSyHwZpzkuQtwrMVg6L/
cache/  keystore/

$ ls orbitdb/12D3KooWNZa53tfgFUmUCw6fALEmkfCoubSyHwZpzkuQtwrMVg6L/keystore/
000003.log  CURRENT  LOCK  LOG  MANIFEST-000002

You do not need to understand this fully for now, just know that it happened. Two subfolders, one being the original folder you saw when you instantiated OrbitDB, and now another that has the same address as your database.

What else happened in the browser?

Similarly, a new IndexedDB database was created to hold your OrbitDB-specific info, apart from the data itself which are still stored in IPFS.

An image showing the IPFS and OrbitDB IndexedDB databases in Firefox

This shows you one of OrbitDB's core strengths - the ability to manage a lot of complexity between its own internals and those of IPFS, providing a clear and clean API to manage the data that matters to you.

Choosing a datastore

OrbitDB organizes its functionality by separating different data management concerns, schemas and APIs into stores. We chose a docstore for you in the last chapter, but after this tutorial it will be your job to determine the right store for the job.

Each OrbitDB store has its own specific API methods to create, delete, retrieve and update data. In general, you can expect to always have something like a get and something like a put.

You have the following choices:

Name Description
log An immutable (append-only) log with traversable history. Useful for "latest N" use cases or as a message queue.
feed A mutable log with traversable history. Entries can be added and removed. Useful for "shopping cart" type of use cases, or for example as a feed of blog posts or "tweets".
keyvalue A simple key-value database that supports any JSON-serializable data, even nested objects.
docs A document database that stores JSON documents which can be indexed by a specified key. Useful for building search indices or version controlling documents and data.
counter An increment-only integer counter useful for counting events separate from log/feed data.

Also, OrbitDB developers can write their own stores if it suits them. This is an advanced topic and is covered in Part 3 of this book.

Key takeaways

  • OrbitDB is a distributed database layer which stores its raw data in IPFS
  • Both IPFS and OrbitDB work offline and online
  • OrbitDB instances have an ID which is the same as the underlying IPFS node's ID.
  • OrbitDB instances create databases which have unique addresses
  • Basic access rights to OrbitDB databases are managed using access control lists (or ACLs), based on the ID of the IPFS node performing the requests on the database
  • OrbitDB database addresses are hashes of the database's ACL, its type, and its name.
  • Since OrbitDB and IPFS are written in JavaScript, it is possible to build isomorphic applications that run in the browser and in Node.js
  • OrbitDB manages needed flexibility of schema and API de# functionality called stores.
  • OrbitDB comes with a handful of stores, and you can write your own.
  • Each store will have its own API, but you will generally have at least a get and a put

Now that you have laid the groundwork, you will learn how to work with data! Onward then, to Chapter 2: Managing Data.