spec/index.md

885 lines
39 KiB
Markdown
Raw Normal View History

2018-10-15 23:37:28 +02:00
%%%
Title = "LBRY: A Decentralized Digital Content Marketplace"
area = "Internet"
[seriesInfo]
name = "Internet-Draft"
value = "draft-grintsvayg-00"
stream = "IETF"
status = "informational"
date = 2018-08-21T00:00:00Z
[[author]]
initials="A."
surname="Grintsvayg"
fullname="Alex Grintsvayg"
%%%
# LBRY: A Decentralized Digital Content Marketplace
A> Please excuse the unfinished state of this paper. It is being actively worked on. The content here is made available early because it contains useful information for developers.
A> For more technical information about LBRY, visit [lbry.tech](https://lbry.tech).
## Introduction
2018-10-19 03:15:00 +02:00
LBRY is a protocol for accessing and publishing digital content in a global, decentralized marketplace. Clients can use LBRY to publish, host, find, download, and pay for content — books, movies, music, or anything else. Anyone can participate and no permission is required, nor can anyone be blocked from participating. The system is distributed, so no single entity has unilateral control, nor will the removal of any single entity prevent the system from functioning.
2018-10-19 01:38:31 +02:00
TODO:
- why is it significant
- whom does it help
- why is it different/better than what existed before
2018-10-15 23:37:28 +02:00
## Table of Contents
2018-10-17 21:06:31 +02:00
<div id="toc">
2018-10-15 23:37:28 +02:00
<!--ts-->
* [Overview](#overview)
* [Conventions and Terminology](#conventions-and-terminology)
* [Blockchain](#blockchain)
* [Claims](#claims)
2018-10-24 22:41:52 +02:00
* [Claim Properties](#claim-properties)
* [Claim Example](#claim-example)
2018-10-15 23:37:28 +02:00
* [Claim Operations](#claim-operations)
* [Claimtrie](#claimtrie)
2018-10-24 22:41:52 +02:00
* [Claim Statuses](#claim-statuses)
2018-10-15 23:37:28 +02:00
* [Accepted](#accepted)
* [Abandoned](#abandoned)
* [Active](#active)
* [Controlling](#controlling)
* [Normalization](#normalization)
* [URLs](#urls)
2018-10-17 21:06:31 +02:00
* [Components](#components)
2018-10-24 22:41:52 +02:00
* [Grammar](#grammar)
2018-10-15 23:37:28 +02:00
* [Design Notes](#design-notes)
* [Transactions](#transactions)
* [Operations and Opcodes](#operations-and-opcodes)
* [Addresses](#addresses)
* [Proof of Payment](#proof-of-payment)
* [Consensus](#consensus)
* [Block Timing](#block-timing)
* [Difficulty Adjustment](#difficulty-adjustment)
2018-10-18 00:01:26 +02:00
* [Block Hash Algorithm](#block-hash-algorithm)
* [Block Rewards](#block-rewards)
2018-10-15 23:37:28 +02:00
* [Metadata](#metadata)
* [Metadata Specification](#metadata-specification)
* [Key Metadata Fields](#key-metadata-fields)
* [Streams and Stream Hashes](#streams-and-stream-hashes)
* [Fees and Fee Structure](#fees-and-fee-structure)
* [Identities](#identities)
* [Metadata Validation](#metadata-validation)
* [Data](#data)
* [Encoding and Decoding](#encoding-and-decoding)
* [Blobs](#blobs)
* [Streams](#streams)
2018-10-25 20:44:04 +02:00
* [Manifest Contents](#manifest-contents)
* [Stream Encoding](#stream-encoding)
2018-10-24 22:41:52 +02:00
* [Setup](#setup)
* [Content Blobs](#content-blobs)
* [Manifest Blob](#manifest-blob)
2018-10-25 20:44:04 +02:00
* [Stream Decoding](#stream-decoding)
* [Announce](#announce)
2018-10-15 23:37:28 +02:00
* [Distributed Hash Table](#distributed-hash-table)
2018-10-25 20:44:04 +02:00
* [Announcing to the DHT](#announcing-to-the-dht)
* [Download](#download)
* [Querying the DHT](#querying-the-dht)
2018-10-19 01:38:31 +02:00
* [Blob Exchange Protocol](#blob-exchange-protocol)
2018-10-25 20:44:04 +02:00
* [PriceCheck](#pricecheck)
* [DownloadCheck](#downloadcheck)
* [Download](#download-1)
* [UploadCheck](#uploadcheck)
* [Upload](#upload)
* [Reflector / BlobEx Upload](#reflector--blobex-upload)
2018-10-15 23:37:28 +02:00
* [Data Markets](#data-markets)
* [Conclusion](#conclusion)
<!--te-->
2018-10-17 21:06:31 +02:00
</div>
2018-10-15 23:37:28 +02:00
## Overview
2018-10-19 03:11:24 +02:00
This document defines the LBRY protocol, its components, and how they fit together. At its core, LBRY consists of several discrete components that are used together in order to provide the end-to-end capabilities of the protocol. There are two distributed data stores (blockchain and DHT), a peer-to-peer protocol for exchanging data, and several specifications for data structure, transformation, and retrieval.
2018-10-15 23:37:28 +02:00
2018-10-19 03:11:24 +02:00
This document assumes that the reader is familiar with Bitcoin and blockchain technology. It does not attempt to document the Bitcoin protocol or explain how it works. The [Bitcoin developer reference](https://bitcoin.org/en/developer-reference) is recommended for anyone wishing to understand the technical details.
2018-10-15 23:37:28 +02:00
## Conventions and Terminology
(Rather than this section, maybe we can use a syntax like brackets around keywords to inline key definitions?)
2018-10-19 01:38:31 +02:00
<dl>
<dt>file</dt>
2018-10-19 03:11:24 +02:00
<dd>A single piece of content published using LBRY.</dd>
2018-10-19 01:38:31 +02:00
<dt>blob</dt>
2018-10-19 03:11:24 +02:00
<dd>The unit of data transmission on the data network. A published file is split into many blobs.</dd>
2018-10-19 01:38:31 +02:00
<dt>stream</dt>
2018-10-19 03:11:24 +02:00
<dd>A set of blobs that can be reassembled into a file. Every stream has a descriptor blob and one or more content blobs.</dd>
2018-10-19 01:38:31 +02:00
<dt>blob hash</dt>
2018-10-19 03:11:24 +02:00
<dd>The output of a cryptographic hash function is applied to a blob. Hashes are used to uniquely identify blobs and to verify that the contents of the blob are correct. Unless otherwise specified, LBRY uses SHA384 as the hash function.</dd>
2018-10-19 01:38:31 +02:00
<dt>metadata</dt>
2018-10-19 03:11:24 +02:00
<dd>Information about the contents of a stream (e.g. creator, description, stream descriptor hash, etc). Metadata is stored in the blockchain.</dd>
2018-10-19 01:38:31 +02:00
<dt>claim</dt>
2018-10-19 03:11:24 +02:00
<dd>A single metadata entry in the blockchain.</dd>
<dt>name</dt>
<dd>A human-readable UTF8 string that is associated with a published claim.</dd>
2018-10-19 01:38:31 +02:00
<dt>channel</dt>
<dd>The unit of pseudonymous publisher identity. Claims may be part of a channel.</dd>
<dt>URL</dt>
2018-10-19 03:11:24 +02:00
<dd>A reference to a claim that specifies how to retrieve it.</dd>
2018-10-19 01:38:31 +02:00
</dl>
2018-10-15 23:37:28 +02:00
## Blockchain
2018-10-24 18:30:52 +02:00
<!-- done -->
2018-10-15 23:37:28 +02:00
The LBRY blockchain is a public, proof-of-work blockchain. It serves three key purposes:
1. An index of the content available on the network
2018-10-19 03:11:24 +02:00
2. A payment system and record of purchases for priced content
2018-10-24 18:30:52 +02:00
3. Trustful publisher identities
2018-10-15 23:37:28 +02:00
2018-10-17 21:06:31 +02:00
The LBRY blockchain is a fork of the [Bitcoin](https://bitcoin.org/bitcoin.pdf) blockchain, with substantial modifications. This document will not cover or specify any aspects of LBRY that are identical to Bitcoin, and will instead focus on the differences.
2018-10-15 23:37:28 +02:00
2018-10-24 22:41:52 +02:00
### Claims
<!-- done -->
2018-10-17 21:06:31 +02:00
A _claim_ is a single metadata entry in the blockchain. There are three types of claims:
2018-10-24 22:41:52 +02:00
<dl>
<dt>stream</dt>
<dd>Declare the availability, access method, and publisher of a stream of bytes (typically a file).</dd>
<dt>identity</dt>
<dd>Create a trustful pseudonym that can be used to identify the origin of stream claims.</dd>
<dt>support</dt>
<dd>Add their amount to a stream or identity claim.</dd>
</dl>
2018-10-15 23:37:28 +02:00
#### Claim Properties
Claims have 4 properties:
2018-10-24 18:30:52 +02:00
<dl>
<dt>claimId</dt>
<dd>A 20-byte hash unique among all claims. See [Claim Identifier Generation](#claim-identifier-generation).</dd>
<dt>name</dt>
<dd>A normalized UTF-8 string of up to 255 bytes used to address the claim. See [URLs](#urls) and [Normalization](#normalization).</dd>
<dt>amount</dt>
<dd>A quantity of tokens used to stake the claim. See [Controlling](#controlling).</dd>
<dt>value</dt>
<dd>Metadata about a piece of content or an identity. Empty for support claims. See [Metadata](#metadata).</dd>
2018-10-24 18:30:52 +02:00
</dl>
2018-10-24 22:41:52 +02:00
#### Claim Example
<!-- done -->
2018-10-15 23:37:28 +02:00
Here is an example stream claim:
2018-10-15 23:37:28 +02:00
```
{
"claimId": "fa3d002b67c4ff439463fcc0d4c80758e38a0aed",
"name": "lbry",
"amount": 100000000,
"value": "{\"ver\": \"0.0.3\", \"description\": \"What is LBRY? An introduction with Alex Tabarrok\",
\"license\": \"LBRY inc\", \"title\": \"What is LBRY?\", \"author\": \"Samuel Bryan\",
\"language\": \"en\", \"sources\": {\"lbry_sd_hash\":
\"e1e324bce7437540fac6707fa142cca44d76fc4e8e65060139a88ff7cdb218b4540cb9cff8bb3d5e06157ae6b08e5cb5\"},
\"content_type\": \"video/mp4\", \"nsfw\": false, \"thumbnail\":
\"https://s3.amazonaws.com/files.lbry.io/logo.png\"}",
"txid": "53ed05d9dfd728a94bedf952d67783bbe9da5d2ab436a84338bb53f0b85301b5",
"n": 0,
"height": 146117
}
2018-10-25 20:44:04 +02:00
```
2018-10-15 23:37:28 +02:00
2018-10-24 22:41:52 +02:00
#### Claim Operations
<!-- done -->
2018-10-15 23:37:28 +02:00
There are three claim operations: _create_, _update_, and _abandon_.
2018-10-15 23:37:28 +02:00
2018-10-24 18:30:52 +02:00
<dl>
<dt>create</dt>
<dd>Makes a new claim.</dd>
<dt>update</dt>
<dd>Changes the value or amount of an existing claim. Updates do not change the claim ID, so an updated claim retains any supports attached to it. </dd>
2018-10-24 18:30:52 +02:00
<dt>abandon</dt>
<dd>Withdraws a claim, freeing the associated credits to be used for other purposes.</dd>
2018-10-24 18:30:52 +02:00
</dl>
2018-10-15 23:37:28 +02:00
2018-10-24 22:41:52 +02:00
#### Claimtrie
<!-- done -->
2018-10-15 23:37:28 +02:00
2018-10-24 18:30:52 +02:00
The _claimtrie_ is the data structure used to store the set of all claims and prove the correctness of claim resolution.
2018-10-15 23:37:28 +02:00
2018-10-24 18:30:52 +02:00
The claimtrie is implemented as a [Merkle tree](https://en.wikipedia.org/wiki/Merkle_tree) that maps names to claims. Claims are stored as leaf nodes in the tree. Names are stored as the path from the root node to the leaf node.
2018-10-15 23:37:28 +02:00
The _root hash_ is the hash of the root node. It is stored in the header of each block in the blockchain. Nodes in the LBRY network use the root hash to efficiently and securely validate the state of the claimtrie.
2018-10-15 23:37:28 +02:00
2018-10-17 21:06:31 +02:00
Multiple claims can exist for the same name. They are all stored in the leaf node for that name, sorted in decreasing order by the total amount of credits backing each claim.
2018-10-19 01:38:31 +02:00
For more details on the specific claimtrie implementation, see [the source code](https://github.com/lbryio/lbrycrd/blob/master/src/claimtrie.cpp).
2018-10-15 23:37:28 +02:00
2018-10-24 22:41:52 +02:00
#### Claim Statuses
<!-- done -->
2018-10-15 23:37:28 +02:00
A claim can have one or more the following statuses at a given block.
2018-10-15 23:37:28 +02:00
2018-10-24 22:41:52 +02:00
##### Accepted
<!-- done -->
2018-10-15 23:37:28 +02:00
An _accepted_ claim is one that has been entered into the blockchain. This happens when the transaction containing the claim is included in a block.
Accepted claims do not appear in or affect the claimtrie state until they are [Active](#active).
2018-10-15 23:37:28 +02:00
2018-10-24 22:41:52 +02:00
##### Abandoned
<!-- done -->
2018-10-15 23:37:28 +02:00
An _abandoned_ claim is one that was withdrawn by its creator. Spending a transaction that contains a claim will cause that claim to become abandoned.
2018-10-15 23:37:28 +02:00
Abandoned stream and identity claims are no longer stored in the claimtrie. Abandoned support claims no longer contribute their amount to the sort order of claims listed in a leaf.
While data related to abandoned claims technically still resides in the blockchain, it is improper to use this data to fetch the associated content.
2018-10-15 23:37:28 +02:00
##### Active
An _active_ claim is an accepted and non-abandoned claim that has been in the blockchain long enough to be activated. The length of time required is called the _activation delay_.
The activation delay depends on the claim operation, the height of the current block, and the height at which the claimtrie state for that name last changed.
If the claim is an update or support to an already active claim, or if it is the first claim for a name, the claim becomes active as soon as it is accepted. Otherwise it becomes active at the block heigh determined by the following formula:
2018-10-15 23:37:28 +02:00
2018-10-25 20:53:09 +02:00
```
C + min(4032, floor((H-T) / 32))
```
2018-10-15 23:37:28 +02:00
Where:
2018-10-25 20:53:09 +02:00
2018-10-15 23:37:28 +02:00
- C = claim height (height when the claim was accepted)
- H = current height
- T = takeover height (the most recent height at which the claimtrie state for the name changed)
2018-10-15 23:37:28 +02:00
In plain English, the delay before a claim becomes active is equal to the claims height minus height of the last takeover, divided by 32. The delay is capped at 4032 blocks, which is 7 days of blocks at 2.5 minutes per block (our target block time). The max delay is reached 224 (7x32) days after the last takeover. The goal of this delay function is to give long-standing claimants time to respond to takeover attempts, while still keeping takeover times reasonable and allowing recent or contentious claims to be taken over quickly.
##### Controlling
2018-10-17 21:06:31 +02:00
The controlling claim is the claim that has the highest total effective amount, which is the sum of its own amount and the amounts of all of its supports. It must be active and cannot itself be a support.
2018-10-15 23:37:28 +02:00
2018-10-17 21:06:31 +02:00
Only one claim can be controlling for a given name at a given block. To determine which claim is controlling for a given name at a given block, the following algorithm is used:
2018-10-15 23:37:28 +02:00
1. For each active claim for the name, add up the amount of the claim and the amount of all the active supports for that claim.
1. Determine if a takeover is happening
2018-10-17 21:06:31 +02:00
1. If the claim with the greatest total is the controlling claim from the previous block, then nothing changes. That claim is still controlling at this block.
1. Otherwise, a takeover is occurring. Set the takeover height for this name to the current height, recalculate which claims and supports are now active, and then perform step 1 again.
2018-10-15 23:37:28 +02:00
1. At this point, the claim with the greatest total is the controlling claim at this block.
2018-10-19 03:11:24 +02:00
The purpose of 2b is to handle the case when multiple competing claims are made on the same name in different blocks, and one of those claims becomes active but another still-inactive claim has the greatest amount. Step 2b will cause the greater claim to also activate and become the controlling claim.
2018-10-15 23:37:28 +02:00
Here is a step-by-step example to illustrate the different scenarios. All claims are for the same name.
**Block 13:** Claim A for 10LBC is accepted. It is the first claim, so it immediately becomes active and controlling.
2018-10-17 21:06:31 +02:00
<br>State: A(10) is controlling
2018-10-15 23:37:28 +02:00
2018-10-17 21:06:31 +02:00
**Block 1001:** Claim B for 20LBC is accepted. Its activation height is `1001 + min(4032, floor((1001-13) / 32)) = 1001 + 30 = 1031`.
<br>State: A(10) is controlling, B(20) is accepted.
2018-10-15 23:37:28 +02:00
**Block 1010:** Support X for 14LBC for claim A is accepted. Since it is a support for the controlling claim, it activates immediately.
2018-10-17 21:06:31 +02:00
<br>State: A(10+14) is controlling, B(20) is accepted.
2018-10-15 23:37:28 +02:00
2018-10-17 21:06:31 +02:00
**Block 1020:** Claim C for 50LBC is accepted. The activation height is `1020 + min(4032, floor((1020-13) / 32)) = 1020 + 31 = 1051`.
<br>State: A(10+14) is controlling, B(20) is accepted, C(50) is accepted.
2018-10-15 23:37:28 +02:00
**Block 1031:** Claim B activates. It has 20LBC, while claim A has 24LBC (10 original + 14 from support X). There is no takeover, and claim A remains controlling.
2018-10-17 21:06:31 +02:00
<br>State: A(10+14) is controlling, B(20) is active, C(50) is accepted.
2018-10-15 23:37:28 +02:00
2018-10-17 21:06:31 +02:00
**Block 1040:** Claim D for 300LBC is accepted. The activation height is `1040 + min(4032, floor((1040-13) / 32)) = 1040 + 32 = 1072`.
<br>State: A(10+14) is controlling, B(20) is active, C(50) is accepted, D(300) is accepted.
2018-10-15 23:37:28 +02:00
**Block 1051:** Claim C activates. It has 50LBC, while claim A has 24LBC, so a takeover is initiated. The takeover height for this name is set to 1051, and therefore the activation delay for all the claims becomes `min(4032, floor((1051-1051) / 32)) = 0`. All the claims become active. The totals for each claim are recalculated, and claim D becomes controlling because it has the highest total.
2018-10-17 21:06:31 +02:00
<br>State: A(10+14) is active, B(20) is active, C(50) is active, D(300) is controlling.
2018-10-15 23:37:28 +02:00
#### Normalization
2018-10-24 16:42:20 +02:00
Names in the claimtrie are normalized to avoid confusion due to Unicode equivalence or casing. All names are normalized using the NFD normalization form, then lowercased using the en_US locale.
2018-10-15 23:37:28 +02:00
### URLs
2018-10-19 03:11:24 +02:00
URLs are human-readable references to claims. All URLs contain a name, and can be resolved to a specific claim for that name. The ultimate purpose of much of the claim design, including controlling claims and the claimtrie structure, is to provide human-readable URLs that can be trustfully resolved by clients that have don't have a full copy of the blockchain.
2018-10-15 23:37:28 +02:00
2018-10-17 21:06:31 +02:00
#### Components
2018-10-15 23:37:28 +02:00
2018-10-19 03:11:24 +02:00
A URL is a name with one or more modifiers. A bare name on its own will resolve to the controlling claim at the latest block height, for reasons covered in [Design Notes](#design-notes). Common URL structures are:
2018-10-15 23:37:28 +02:00
**Name:** a basic claim for a name
2018-10-19 03:15:00 +02:00
```
2018-10-19 03:11:24 +02:00
lbry:meet-LBRY
2018-10-19 03:15:00 +02:00
```
2018-10-15 23:37:28 +02:00
2018-10-19 03:11:24 +02:00
**Claim ID:** a claim for this name with this claim ID (does not have to be the controlling claim). Partial prefix matches are allowed.
2018-10-15 23:37:28 +02:00
2018-10-19 03:15:00 +02:00
```
lbry:meet-LBRY#7a0aa95c5023c21c098
2018-10-19 03:11:24 +02:00
lbry:meet-LBRY#7a
2018-10-19 03:15:00 +02:00
```
2018-10-15 23:37:28 +02:00
2018-10-19 03:11:24 +02:00
**Claim Sequence:** the Nth claim for this name, in the order the claims entered the blockchain. N must be a positive number. This can be used to determine which claim came first, rather than which claim has the most support.
2018-10-15 23:37:28 +02:00
2018-10-19 03:15:00 +02:00
```
2018-10-19 03:11:24 +02:00
lbry:meet-LBRY:1
2018-10-19 03:15:00 +02:00
```
2018-10-15 23:37:28 +02:00
2018-10-19 03:11:24 +02:00
**Bid Position:** the Nth claim for this name, in order of most support to least support. N must be a positive number. This is useful for resolving non-winning bids in bid order, e.g. if you want to list the top three winning claims in a voting contest or want to ignore the activation delay.
2018-10-15 23:37:28 +02:00
2018-10-19 03:15:00 +02:00
```
lbry:meet-LBRY$2
2018-10-19 03:11:24 +02:00
lbry:meet-LBRY$3
2018-10-19 03:15:00 +02:00
```
2018-10-15 23:37:28 +02:00
2018-10-19 03:11:24 +02:00
**Query Params:** extra parameters (reserved for future use)
2018-10-19 03:15:00 +02:00
```
2018-10-19 03:11:24 +02:00
lbry:meet-LBRY?arg=value+arg2=value2
2018-10-19 03:15:00 +02:00
```
2018-10-15 23:37:28 +02:00
2018-10-19 03:11:24 +02:00
**Channel:** a claim for a channel
2018-10-15 23:37:28 +02:00
2018-10-19 03:15:00 +02:00
```
2018-10-19 03:11:24 +02:00
lbry:@lbry
2018-10-19 03:15:00 +02:00
```
2018-10-15 23:37:28 +02:00
2018-10-19 03:11:24 +02:00
**Claim in Channel:** URLS with a channel and a claim name are resolved in two steps. First the channel is resolved to get the claim for that channel. Then the name is resolved to get the appropriate claim from among the claims in the channel.
2018-10-15 23:37:28 +02:00
2018-10-19 03:15:00 +02:00
```
2018-10-19 03:11:24 +02:00
lbry:@lbry/meet-LBRY
2018-10-19 03:15:00 +02:00
```
2018-10-15 23:37:28 +02:00
2018-10-24 16:42:20 +02:00
#### Grammar
2018-10-15 23:37:28 +02:00
2018-10-24 16:42:20 +02:00
The full URL grammar is defined using [Xquery EBNF notation](https://www.w3.org/TR/2017/REC-xquery-31-20170321/#EBNFNotation):
2018-10-17 21:06:31 +02:00
<!-- see http://bottlecaps.de/rr/ui for visuals-->
2018-10-15 23:37:28 +02:00
```
2018-10-19 01:38:31 +02:00
URL ::= Scheme Path Query?
2018-10-17 21:06:31 +02:00
2018-10-24 16:42:20 +02:00
Scheme ::= 'lbry://'
2018-10-19 01:38:31 +02:00
Path ::= ClaimNameAndModifier | ChannelAndModifier ( '/' ClaimNameAndModifier )?
2018-10-17 21:06:31 +02:00
ClaimNameAndModifier ::= ClaimName Modifier?
ChannelAndModifier ::= Channel Modifier?
2018-10-24 16:42:20 +02:00
ClaimName ::= NameChar+
2018-10-17 21:06:31 +02:00
Channel ::= '@' ClaimName
Modifier ::= ClaimID | ClaimSequence | BidPosition
ClaimID ::= '#' Hex+
2018-10-19 03:11:24 +02:00
ClaimSequence ::= ':' PositiveNumber
BidPosition ::= '$' PositiveNumber
2018-10-17 21:06:31 +02:00
Query ::= '?' QueryParameterList
QueryParameterList ::= QueryParameter ( '&' QueryParameterList )*
QueryParameter ::= QueryParameterName ( '=' QueryParameterValue )?
2018-10-24 16:42:20 +02:00
QueryParameterName ::= NameChar+
QueryParameterValue ::= NameChar+
2018-10-17 21:06:31 +02:00
2018-10-19 03:11:24 +02:00
PositiveDigit ::= [123456789]
Digit ::= '0' | PositiveDigit
PositiveNumber ::= PositiveDigit Digit*
2018-10-17 21:06:31 +02:00
HexAlpha ::= [abcdef]
Hex ::= (Digit | HexAlpha)+
2018-10-24 16:42:20 +02:00
NameChar ::= Char - [=&#:$@?/] /* any character that is not reserved */
Char ::= #x9 | #xA | #xD | [#x20-#xD7FF] | [#xE000-#xFFFD] | [#x10000-#x10FFFF] /* any Unicode character, excluding the surrogate blocks, FFFE, and FFFF. */
2018-10-15 23:37:28 +02:00
```
#### Design Notes
2018-10-17 21:06:31 +02:00
Most existing public name schemes are first-come, first-serve. This leads to several bad outcomes. When the system is young, users are incentivized to register common names even if they don't intend to use them, in hopes of selling them to the proper owner in the future for an exorbitant price. In a centralized system, the authority may allow for appeals to reassign names based on trademark or other common use reasons. There may also be a process to "verify" that a name belongs to the entity you think it does (e.g. Twitter's verified accounts). Such processes are often arbitrary, change over time, involve significant transaction costs, and may still lead to names being used in ways that are contrary to user expectation (e.g. [nissan.com](http://nissan.com) is not what youd expect).
2018-10-15 23:37:28 +02:00
2018-10-17 21:06:31 +02:00
In a decentralized system, such approaches are not possible, so name squatting is especially dangerous (see Namecoin). Instead, LBRY creates an efficient allocation of names via a market. Following [Coase](https://en.wikipedia.org/wiki/Coase_theorem), we believe that if the rules for name ownership and exchange are clearly defined, transaction costs are low, and there is no information asymmetry, then control of URLs will flow to their highest-valued use.
2018-10-15 23:37:28 +02:00
2018-10-17 21:06:31 +02:00
Note that only vanity URLs (i.e. URLs without a ClaimID or or ClaimSequence modifier) have this property. Permanent URLs like `lbry://myclaimname#abc` exist and are available for the small cost of issuing a `create` claim transactions.
2018-10-15 23:37:28 +02:00
### Transactions
2018-10-19 03:11:24 +02:00
To support claims, the LBRY blockchain makes the following changes on top of Bitcoin.
2018-10-15 23:37:28 +02:00
#### Operations and Opcodes
2018-10-19 03:11:24 +02:00
To enable [claim operations](#claim-operations), three new opcodes were added to the blockchain scripting language: `OP_CLAIM_NAME`, `OP_SUPPORT_CLAIM`, and `OP_UPDATE_CLAIM` (in Bitcoin they are respectively `OP_NOP6`, `OP_NOP7`, and `OP_NOP8`). Each op code will push a zero on to the execution stack, and will trigger the claimtrie to perform calculations necessary for each operation. Below are the three supported transactions scripts using these opcodes.
2018-10-15 23:37:28 +02:00
```
OP_CLAIM_NAME <name> <value> OP_2DROP OP_DROP <pubKey>
2018-10-19 01:38:31 +02:00
2018-10-15 23:37:28 +02:00
OP_UPDATE_CLAIM <name> <claimId> <value> OP_2DROP OP_2DROP <pubKey>
2018-10-19 01:38:31 +02:00
2018-10-15 23:37:28 +02:00
OP_SUPPORT_CLAIM <name> <claimId> OP_2DROP OP_DROP <pubKey>
```
`<pubKey>` can be any valid Bitcoin payout script, so a claimtrie script is also a pay-to-pubkey script to a user-controlled address. Note that the zeros pushed onto the stack by the claimtrie opcodes and vectors are all dropped by `OP_2DROP` and `OP_DROP`. This means that claimtrie transactions exist as prefixes to Bitcoin payout scripts and can be spent just like standard transactions.
For example, a claim transaction setting the name “Fruit” to “Apple” and using a pay-to-pubkey script will have the following payout script:
```
2018-10-19 01:38:31 +02:00
OP_CLAIM_NAME Fruit Apple OP_2DROP OP_DROP OP_DUP OP_HASH160 <addressOne> OP_EQUALVERIFY OP_CHECKSIG
2018-10-15 23:37:28 +02:00
```
Like any standard Bitcoin transaction output script, it will be associated with a transaction hash and output index. The transaction hash and output index are concatenated and hashed to create the claimID for this claim. For the example above, let's say the above transaction hash is `7560111513bea7ec38e2ce58a58c1880726b1515497515fd3f470d827669ed43` and the output index is `1`. Then the claimID would be `529357c3422c6046d3fec76be2358004ba22e323`.
A support for this bid will have the following payout script:
```
2018-10-19 01:38:31 +02:00
OP_SUPPORT_CLAIM Fruit 529357c3422c6046d3fec76be2358004ba22e323 OP_2DROP OP_DROP OP_DUP OP_HASH160 <addressTwo> OP_EQUALVERIFY OP_CHECKSIG
2018-10-15 23:37:28 +02:00
```
And now let's say we want to update the original claim to change the value to “Banana”. An update transaction has a special requirement that it must spend the existing claim that it wishes to update in its redeem script. Otherwise, it will be considered invalid and will not make it into the claimtrie. Thus it will have the following redeem script:
```
<signature> <pubKeyForAddressOne>
```
This is identical to the standard way of redeeming a pay-to-pubkey script in Bitcoin.
The payout script for the update transaction is:
```
2018-10-19 01:38:31 +02:00
OP_UPDATE_CLAIM Fruit 529357c3422c6046d3fec76be2358004ba22e323 Banana OP_2DROP OP_2DROP OP_DUP OP_HASH160 <addressThree> OP_EQUALVERIFY OP_CHECKSIG
2018-10-15 23:37:28 +02:00
```
#### Addresses
2018-10-19 01:38:31 +02:00
The address version byte is set to `0x55` for standard (pay-to-public-key-hash) addresses and `0x7a` for multisig (pay-to-script-hash) addresses. P2PKH addresses start with the letter `b`, and P2SH addresses start with `r`.
2018-10-15 23:37:28 +02:00
2018-10-19 03:11:24 +02:00
All the chain parameters are defined [here](https://github.com/lbryio/lbrycrd/blob/master/src/chainparams.cpp).
2018-10-15 23:37:28 +02:00
#### Proof of Payment
2018-10-19 03:11:24 +02:00
TODO: Explain how transactions serve as proof that a client has made a valid payment for a piece of content.
2018-10-15 23:37:28 +02:00
### Consensus
2018-10-19 03:11:24 +02:00
LBRY makes a few small changes to consensus rules.
2018-10-15 23:37:28 +02:00
2018-10-18 00:01:26 +02:00
2018-10-15 23:37:28 +02:00
#### Block Timing
2018-10-19 03:11:24 +02:00
The target block time was lowered from 10 minutes to 2.5 minutes to facilitate faster transaction confirmation.
2018-10-15 23:37:28 +02:00
#### Difficulty Adjustment
2018-10-19 03:11:24 +02:00
The proof-of-work target is adjusted every block to better adapt to sudden changes in hash rate. The exact adjustment algorithm can be seen [here](https://github.com/lbryio/lbrycrd/blob/master/src/lbry.cpp).
2018-10-18 00:01:26 +02:00
#### Block Hash Algorithm
2018-10-19 03:11:24 +02:00
LBRY uses a combination of SHA256, SHA512, and RIPEMD160. The exact hashing algorithm can be seen [here](https://github.com/lbryio/lbrycrd/blob/master/src/hash.cpp#L18).
2018-10-18 00:01:26 +02:00
#### Block Rewards
2018-10-19 03:15:00 +02:00
The block reward schedule was adjusted to provide an initial testing period, a quick ramp-up to max block rewards, then a logarithmic decay to 0. The source for the algorithm is [here](https://github.com/lbryio/lbrycrd/blob/master/src/main.cpp#L1594).
2018-10-15 23:37:28 +02:00
## Metadata
2018-10-19 03:11:24 +02:00
Claim metadata is stored in a serialized format using [Protocol Buffers](https://developers.google.com/protocol-buffers/). This was chosen for several reasons:
- **Extensibility**. The metadata structure could grow to encompass thousands of fields for dozens of types of content. It should be easy to modify the structure while maintaining backward compatibility. Blockchain data is permanent and cannot be migrated.
- **Compactness**. Blockchain space is expensive. Data should be stored as compactly as possible.
- **Interoperability**. These definitions will be used by many projects written in different languages. Protocol buffers are language-independent and have great support for most popular languages.
2018-10-15 23:37:28 +02:00
2018-10-19 03:11:24 +02:00
No enforcement or validation on metadata happens at the blockchain level. Instead, metadata encoding, decoding, and validation is done by clients. This allows evolution of the metadata without changes to consensus rules.
2018-10-15 23:37:28 +02:00
### Metadata Specification
2018-10-19 03:11:24 +02:00
A useful index of available content must be succinct yet meaningful. It should be machine-readable, comprehensive, and should ideally point you toward the content youre looking for. LBRY achieves this by defining a set of common properties for streams.
2018-10-15 23:37:28 +02:00
The metadata contains structured information describing the content, such as the title, author, language, and so on.
Heres an example:
```
"metadata": {
"author": "",
"description": "All proceeds go to holly for buying toys, i will post the video with those toys on Xmas day",
"language": "en",
"license": "All rights reserved.",
"licenseUrl": "",
"nsfw": false,
"preview": "",
"thumbnail": "http://www.thetoydiscounter.com/happy.jpg",
"title": "Holly singing The Happy Working Song",
"source": {
2018-10-19 03:15:00 +02:00
"contentType": "video/mp4",
"source": "92b8aae7a901c56901fd5602c1f1acc0e63fb5492ef2a3cd5b9c631d92cab2e060e2a908baa922c24dee6c5229a98136",
"sourceType": "lbry_sd_hash",
"version": "_0_0_1"
},
2018-10-15 23:37:28 +02:00
"version": "_0_1_0"
}
```
2018-10-19 03:11:24 +02:00
Because the metadata structure can and does change frequently, a complete specification is omitted from this document. Instead, [github.com/lbryio/types](https://github.com/lbryio/types) should be consulted for the precise definition of current metadata structure.
2018-10-15 23:37:28 +02:00
### Key Metadata Fields
2018-10-19 03:11:24 +02:00
Despite not covering the full metadata structure, a few important metadata fields are highlighted below.
2018-10-15 23:37:28 +02:00
#### Streams and Stream Hashes
2018-10-24 16:42:20 +02:00
(The metadata property `lbry_sd_hash` contains a unique identifier to locate and find the content in the data network. Reference [[Data]].)
2018-10-15 23:37:28 +02:00
#### Fees and Fee Structure
- LBC
- Currencies?
2018-10-24 16:42:20 +02:00
- channel signatures and private keys
2018-10-15 23:37:28 +02:00
### Identities
2018-10-19 03:11:24 +02:00
Channels are the unit of identity in the LBRY system. A channel is a claim that start with `@` and contains a metadata structure for identities rather than content. The most important part of channel's metadata is the public key. Claims belonging to a channel are signed with the corresponding private key. A valid signature proves channel membership.
2018-10-15 23:37:28 +02:00
2018-10-19 03:11:24 +02:00
The purpose of channels is to allow content to be clustered under a single pseudonym or identity. This allows publishers to easily list all their content, maintain attribution, and build their brand.
2018-10-15 23:37:28 +02:00
Heres the value of an example channel claim:
```
"certificate": {
"keyType": "SECP256k1",
"publicKey": "3056301006072a8648ce3d020106052b8104000a0342
0004180488ffcb3d1825af538b0b952f0eba6933faa6
d8229609ac0aeadfdbcf49C59363aa5d77ff2b7ff06c
ddc07116b335a4a0849b1b524a4a69d908d69f1bcebb",
"version": "_0_0_1"
}
```
When a claim published into a channel, the claim data is signed and the following is added to the claim:
```
"publisherSignature": {
"channelClaimID": "2996b9a087c18456402b57cba6085b2a8fcc136d",
"signature": "bf82d53143155bb0cac1fd3d917c000322244b5aD17
e7865124db2ed33812ea66c9b0c3f390a65a9E2d452
e315e91ae695642847d88e90348ef3c1fa283a36a8",
"signatureType": "SECP256k1",
"version": "_0_0_1"
}
```
### Metadata Validation
2018-10-24 16:42:20 +02:00
Clients are responsible for validating metadata, including data structure and signatures.
2018-10-15 23:37:28 +02:00
(expand)
- Validation 101
- Channel / identity validation
## Data
(This portion covers how content is actually encoded and decoded, fetched, and announced. Expand/fix.)
2018-10-25 20:44:04 +02:00
2018-10-15 23:37:28 +02:00
### Encoding and Decoding
2018-10-25 20:44:04 +02:00
<!-- done -->
2018-10-15 23:37:28 +02:00
#### Blobs
2018-10-25 20:44:04 +02:00
<!-- done -->
The unit of data in our network is called a _blob_. A blob is an encrypted chunk of data up to 2MiB in size. Each blob is indexed by its _blob hash_, which is a SHA384 hash of the blob contents. Addressing blobs by their hashes simultaneously protects against naming collisions and ensures that the content you get is what you expect.
2018-10-24 22:41:52 +02:00
2018-10-25 20:44:04 +02:00
Blobs are encrypted using AES-256 in CBC mode and PKCS7 padding. In order to keep each encrypted blob at 2MiB max, a blob can hold at most 2097151 bytes (2MiB minus 1 byte) of plaintext data. The source code for the exact algorithm is available [here](https://github.com/lbryio/lbry.go/blob/master/stream/blob.go). The encryption key and IV for each blob is stored as described below.
2018-10-15 23:37:28 +02:00
#### Streams
2018-10-25 20:44:04 +02:00
<!-- done -->
2018-10-24 22:41:52 +02:00
Multiple blobs are combined into a _stream_. A stream may be a book, a movie, a CAD file, etc. All content on the network is shared as streams. Every stream begins with the _manifest blob_, followed by one or more _content blobs_. The content blobs hold the actual content of the stream. The manifest blob contains information necessary to find the content blobs and convert them into a file. This includes the hashes of the content blobs, their order in the stream, and cryptographic material for decrypting them.
The blob hash of the manifest blob is called the _stream hash_. It uniquely identifies each stream.
2018-10-25 20:44:04 +02:00
#### Manifest Contents
<!-- done -->
2018-10-24 22:41:52 +02:00
A manifest blob's contents are encoded using a canonical JSON encoding. The JSON encoding must be canonical to support consistent hashing and validation. The encoding is the same as standard JSON, but adds the following rules:
- Object keys must be quoted and lexicographically sorted.
- All strings are hex-encoded. Hex letters must be lowercase.
- Whitespace before, after, or between tokens is not permitted.
- Floating point numbers, leading zeros, and "minus 0" for integers are not permitted.
- Trailing commas after the last item in an array or object are not permitted.
2018-10-24 16:42:20 +02:00
2018-10-24 22:41:52 +02:00
Here's an example manifest, with whitespace added for readability:
<!-- originally from 053b2f0f0e82e7f022837382733d5f5817dcd67027103fe43f00fa7a6f9fa8742c1022a851616c1ac15d1c60e89db3f4 -->
2018-10-24 16:42:20 +02:00
```
{
2018-10-24 22:41:52 +02:00
"blobs":[
{
"blob_hash":"a6daea71be2bb89fab29a2a10face08143411a5245edcaa5efff48c2e459e7ec01ad20edfde6da43a932aca45b2cec61",
"iv":"ef6caef207a207ca5b14c0282d25ce21",
"length":2097152
},
{
"blob_hash":"bf2717e2c445052366d35bcd58edb108cbe947af122d8f76b4856db577aeeaa2def5b57dbb80f7b1531296bd3e0256fc",
"iv":"a37b291a37337fc1ff90ae655c244c1d",
"length":2097152
},
...,
{
"blob_hash":"322973617221ddfec6e53bff4b74b9c21c968cd32ba5a5094d84210e660c4b2ed0882b114a2392a08b06183f19330aaf",
"iv": "a00f5f458695bdc9d50d3dbbc7905abc",
"length": 600160
}
],
"filename":"6b706a7977755477704d632e6d7034",
2018-10-25 20:44:04 +02:00
"key":"94d89c0493c576057ac5f32eb0871180",
"version":1
2018-10-24 16:42:20 +02:00
}
```
2018-10-25 20:44:04 +02:00
The `key` field contains the key to decrypt the stream, and is optional. The key may be stored by a third party and made available to a client when presented with proof that the content was purchased. The `version` field is always 1. It is intended to signal structure changes in the future. The `length` field for each blob is the length of the encrypted blob, not the original file chunk.
2018-10-24 22:41:52 +02:00
Every stream must have at least two blobs - the manifest blob and a content blob. Consequently, zero-length streams are not allowed.
2018-10-25 20:44:04 +02:00
#### Stream Encoding
<!-- done -->
A file must be encoded into a stream before it can be published. Encoding involves breaking the file into chunks, encrypting the chunks into content blobs, and creating the manifest blob. Here are the steps:
2018-10-24 22:41:52 +02:00
##### Setup
2018-10-25 20:44:04 +02:00
<!-- done -->
2018-10-24 22:41:52 +02:00
1. Generate a random 32-byte key for the stream.
##### Content Blobs
2018-10-24 16:42:20 +02:00
2018-10-25 20:44:04 +02:00
<!-- done -->
2018-10-24 22:41:52 +02:00
1. Break the file into chunks of at most 2097151 bytes.
1. Generate a random IV for each chuck.
1. Pad each chunk using PKCS7 padding
1. Encrypt each chunk with AES-CBC using the stream key and the IV for that chunk.
1. An encrypted chunk is a blob.
2018-10-24 16:42:20 +02:00
2018-10-24 22:41:52 +02:00
##### Manifest Blob
2018-10-15 23:37:28 +02:00
2018-10-25 20:44:04 +02:00
<!-- done -->
2018-10-24 22:41:52 +02:00
1. Fill in the manifest data.
1. Encode the data using the canonical JSON encoding described above.
2018-10-25 20:44:04 +02:00
1. Compute the stream hash.
2018-10-19 01:38:31 +02:00
2018-10-24 22:41:52 +02:00
An implementation of this process is available [here](https://github.com/lbryio/lbry.go/tree/master/stream).
2018-10-25 20:44:04 +02:00
fixme: this link is for v0, not v1. need to implement v1 or drop the link.
2018-10-24 16:42:20 +02:00
2018-10-15 23:37:28 +02:00
2018-10-25 20:44:04 +02:00
#### Stream Decoding
<!-- done -->
Decoding a stream is like encoding in reverse, and with the added step of verifying that the expected blob hashes match the actual data.
1. Verify that the manifest blob hash matches the stream hash you expect.
1. Parse the manifest blob contents.
1. Verify the hashes of the content blobs.
1. Decrypt and remove the padding from each content blob using the key and IVs in the manifest.
1. Concatenate the decrypted chunks in order.
### Announce
After a [[stream]] is encoded, it must be _announced_ to the network. Announcing is the process of letting other nodes on the network know that you have content available for download. The LBRY networks tracks announced content using a distributed hash table.
2018-10-15 23:37:28 +02:00
#### Distributed Hash Table
2018-10-25 20:44:04 +02:00
_Distributed hash tables_ (or DHTs) have proven to be an effective way to build a decentralized content network. Our DHT implementation follows the [Kademlia](https://pdos.csail.mit.edu/~petar/papers/maymounkov-kademlia-lncs.pdf)
2018-10-15 23:37:28 +02:00
spec fairly closely, with some modifications.
2018-10-19 03:11:24 +02:00
A distributed hash table is a key-value store that is spread over multiple host nodes in a network. Nodes may join or leave the network anytime, with no central coordination necessary. Nodes communicate with each other using a peer-to-peer protocol to advertise what data they have and what they are best positioned to store.
2018-10-15 23:37:28 +02:00
2018-10-25 20:44:04 +02:00
When a host connects to the DHT, it announces the hash for every [[blob]] it wishes to share. Downloading a blob from the network requires querying the DHT for a list of hosts that announced that blobs hash (called _peers_), then requesting the blob from the peers directly.
#### Announcing to the DHT
A host announces a hash to the DHT in two steps. First, the host looks for nodes that are closest to the target hash that will be announced. Then the host announces the target hash to those nodes.
Finding the closest nodes is done via iterative `FindNode` DHT requests. The host starts with the closest nodes it knows about and sends a `FindNode(target_hash)` request to each of them. If any of the requests return nodes that are closer to the target hash, the host sends `FindNode` requests to those nodes to try to get even closer. When the `FindNode` requests no longer return nodes that are closer, the search ends.
Once the search is over, the host takes the 8 closest nodes it found and sends a `Store(target_hash)` request to them. The nodes receiving this request store the fact that the host is a peer for the target hash.
### Download
A client wishing to download a [[stream]] must first query the [[DHT]] to find peers hosting the [[blobs]] in that stream, then contact those peers directly to download the blobs directly.
#### Querying the DHT
Querying works almost the same way as [[announcing]]. A client looking for a target hash will start by sending iterative `FindValue(target_hash)` requests to the nodes it knows that are closest to the target hash. If a node receives a `FindValue` request and knows of any peers for the target hash, it will respond with a list of those peers. Otherwise, it will respond with the closest nodes to the target hash that it knows about. The client then queries those closer nodes using the same `FindValue` call. This way, each call either finds the client some peers, or brings it closer to finding those peers. If no peers are found and no closer nodes are being returned, the client will determine that the target hash is not available and will give up.
2018-10-15 23:37:28 +02:00
2018-10-19 01:38:31 +02:00
#### Blob Exchange Protocol
2018-10-25 20:44:04 +02:00
Downloading a blob from a peer is governed by the _Blob Exchange Protocol_. It is used by hosts and clients to check blob availability, exchange blobs, and negotiate data prices. The protocol is an RPC protocol using Protocol Buffers and the gRPC framework. It has five types of requests.
fixme: protocol does not **negotiate** anything right now. It just checks the price. Should we include negotiation in v1?
##### PriceCheck
PriceCheck gets the price that the server is charging for data transfer. It returns the prices in [[deweys]] per KB.
##### DownloadCheck
DownloadCheck checks whether the server has certain blobs available for download. For each hash in the request, the server returns a true or false to indicate whether the blob is available.
##### Download
2018-10-19 01:38:31 +02:00
2018-10-25 20:44:04 +02:00
Download requests the blob for a given hash. The response contains the blob, its hash, and the address where to send payment for the data transfer. If the blob is not available on the server, the response will instead contain an error.
2018-10-15 23:37:28 +02:00
2018-10-25 20:44:04 +02:00
##### UploadCheck
2018-10-15 23:37:28 +02:00
2018-10-25 20:44:04 +02:00
UploadCheck asks the server whether blobs can be uploaded to it. For each hash in the request, the server returns a true or false to indicate whether it would accept a given blob for upload. In addition, if any of the hashes in the request is a stream hash and the server has the manifest blob for that stream but is missing some content blobs, it may include the hashes of those content blobs in the response.
2018-10-15 23:37:28 +02:00
2018-10-25 20:44:04 +02:00
##### Upload
2018-10-15 23:37:28 +02:00
2018-10-25 20:44:04 +02:00
Upload sends a blob to the server. If uploading many blobs, the client should use the UploadCheck request to check which blobs the server actually needs. This avoids needlessly uploading blobs that the server already has. If a client tries to upload too many blobs that the server does not want, this may be consider a denial of service attack.
2018-10-15 23:37:28 +02:00
2018-10-25 20:44:04 +02:00
The protocol calls and message types are defined in detail [here](https://github.com/lbryio/lbry.go/blob/master/blobex/blobex.proto).
### Reflector / BlobEx Upload
2018-10-15 23:37:28 +02:00
### Data Markets
2018-10-25 20:44:04 +02:00
To incentivize hosts and reflectors, the blob exchange protocol supports payment for data.
2018-10-15 23:37:28 +02:00
(Price negotiation.)
<!--
### Data Market
Hosts in the DHT can treat blobs as opaque chunks of data. There is price negotiation mechanism for data. So some hosts can be
purely interested in storing data and selling it. They may create algorithms for what data is more in demand (e.g. the first content
blob in a stream is probably requested more often than the last blob).
Talk about reputation system for hosts.
Talk about how lightning can be used for streaming payments.
-->
## Conclusion
*TODO*
2018-10-25 20:53:09 +02:00
_Edit this on Github: https://github.com/lbryio/spec_
2018-10-15 23:37:28 +02:00
<!---
### Supports, Tips
Supports add weight to name claims. They are kind of like voting. You retain control of the credits.
Tips are supports where person you tip gains control of the credits.
### Discovery
### Search
Search will be handled primarily by external indexing services. There are many existing search solutions
that would ingest the blockchain data and build an index of the content. The novel aspect of our system is
that the credits committed to a claim are a strong signal of relevance.
### Tagging
Tags provide extra information for content discovery. A tag has a claim ID and a name. Tags can be created,
supported, updated, and abandoned, just like claims. One key difference is that tag supports may be
labeled “negative” supports. A negative support reduces the effective amount of credits attached to a
tag. This is a signal that the tag is not a good fit for the content of the claim.
## Trust and Security
We believe that the
## Combatting the Ugly
Use this section to rebut some of the most common concerns regarding the nature of LBRY.
One of our core beliefs is that people want to pay the legitimate content owners and creators, as
long as the content reasonably-priced and the payment process is convenient.
Conclusion
Summary
-->