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

openpublish

v2.9.0

Published

A publishing protocol for registering media as a digital asset on the Bitcoin blockchain.

Downloads

47

Readme

Open Publish

Build Status

A publishing protocol for registering media as a digital asset on the Bitcoin blockchain.

Open Publish aims to give ownership of digital media back to the individual and allow individuals to create contracts for licensing and micropayment channels using Bitcoin. Think combining Proof of Existence, OpenAssets and a royalty payment mechanism.

Like Proof of Existence, digital media is represented as a cryptographic digest but in a content-addressable format that works seamlessly with IPFS, BitTorrent and WebTorrent. This content-addressable representation is provably signed, timestamped, and recorded in perpetuity by the Bitcoin blockchain in a format compatible with the BitTorrent Magnet URI scheme and IPFS address scheme.

The Open Publish protocol allows claiming ownership over a digital asset that can be used by other products to represent a limited and non-exclusive copyright of this document.

The Open Publish protocol allows for transfering partial ownership of these digital assets.

Any honest third-party software can read the state of ownership from the Bitcoin blockchain and create software that directs payments to the legitimate owners of the content in the form of direct tips, dividends, monthly subscriptions or various synchronization licenses modeled on existing intellectual property systems.

For example, someone could write software that displays media with a consumer Bitcoin wallet allowing for people to easily and directly tip the rights-holders.

It assumes that the wallet address that posted the Open Publish registration transaction to the Bitcoin blockchain is controlled by the owner of the registered media.

Blockai cannot control what assets are registered on the Bitcoin blockchain so it is up to individual developers or development teams to make sure they honor their local rules and regulations pertaining to copyright. For example, Blockai will not display content that is deemed to not be owned by the claimant Bitcoin address, fully adhering to any DMCA notices or other local regulations.

Install

npm install openpublish

Browser Usage

In our examples we're going to use test-common-wallet to create our wallet.

This simple wallet and the Open Publish API adhere to the Common Wallet standard.

Bitcoin Wallet

We will assume that this wallet is owned and operated by Alice.

var aliceWallet = testCommonWallet({
  seed: 'some-random-very-long-and-super-safe-passphrase',
  network: 'testnet',
  commonBlockchain: commonBlockchain
})

We'll need to provide an instance of a commonBlockchain which will provide functions for signing a transaction, propagating a trasnaction, and looking up a transaction by txid.

In this example we're using the in memory version that is provided by mem-common-blockchain.

var commonBlockchain = require('mem-common-blockchain')({
  type: 'local'
})

// or we could connect to testnet

// commonBlockchain = require('blockcypher-unofficial')({
//   network: "testnet"
// })

Register Media with an Open Publish transaction posted to the Bitcoin network

Hosting and distribution of content is not the goal of Open Publish although an optional link to the content can be part of the metadata.

Files can be hosted on any existing web servers at the expense of the owner. A private service by Blockai called Bitstore is a content-addressable file hosting and distribution service that uses Bitcoin public key infrastructure for authentication and payment. All files hosted on Bitstore are seeded on both BitTorrent and WebTorrent. As it uses Bitcoin wallets for authentication no account creation is necessary which makes it very convenient for application developers.

In this example, Alice is using her wallet along with a file that she draged and dropped on to a web page.

var openpublish = require('openpublish')

var file // a browser File object returned from drop or file select form
var fileUri // a permalink to the above file

openpublish.register({
  file: file,
  uri: fileUri,
  commonWallet: aliceWallet,
  commonBlockchain: commonBlockchain
}, function(err, receipt) {
  var registerData = receipt.data
  var sha1 = registerData.sha1 // the SHA-1 that represents your media file
  var blockcastTx = receipt.blockcastTx
  var txid = blockcastTx.txid // the Bitcoin transaction where the first payload of the the data is embedded
  var transactionTotal = blockcastTx.transactionTotal // the number of Bitcoin transactions in the data payload
})

Transfering ownership of digital assets

Open Publish allows for rights-holders to transfer partial ownership of the digital assets that represent their registered works, in a manner similar to how rights are transfered in music and book publishing.

In this example, Alice's is transfering part ownership of her digital asset to her friend Bob.

Alice started with 100,000,000 of total value in her asset when she first registered her media file.

After this transaction has been signed by both wallets and cleared by the Bitcoin network, Alice will have 70,000,000 and Bob will have 30,000,000.

Transfers are only valid for a set number of days before full ownership reverts back to Alice.

var bobWallet = testCommonWallet({
  seed: 'another-random-very-long-and-super-safe-passphrase',
  network: 'testnet',
  commonBlockchain: commonBlockchain
})

var sha1 = 'dc724af18fbdd4e59189f5fe768a5f8311527050'

openpublish.transfer({
  assetValue: 30000000, // the number of assets that Alice is transfering
  bitcoinValue: 5000000, // the number of bitcoin that Bob is transfering
  ttl: 365, // the number of days that the transfer is valid for before reverting back to Alice
  sha1: registerData.sha1,
  assetWallet: aliceWallet,
  bitcoinWallet: bobWallet,
  commonBlockchain: commonBlockchain
}, function(err, receipt) {
  var blockcastTx = receipt.blockcastTx
  var txid = blockcastTx.txid // the Bitcoin transaction where the first payload of the the data is embedded
  var transactionTotal = blockcastTx.transactionTotal // the number of Bitcoin transactions in the data payload
});

Alice and Bob will probably not have their wallets open at the same time and also need a way to negotiate and settle on a price before they both independently sign the transfer transactions.

Bob decides he'd like to buy another 20,000,000 of the asset from Alice for 3,000,000 satoshi.

// on Bob's computer
openpublish.createBid({
  assetValue: 20000000,
  bitcoinValue: 3000000,
  ttl: ttl,
  sha1: sha1,
  assetAddress: aliceWallet.address,
  bitcoinWallet: bobWallet,
  commonBlockchain: commonBlockchain
}, function (err, proposedBid) {
  sendToAlice(proposedBid)
})

Alice recieves the bid and decides to accept the offer.

// on Alice's computer
var proposedBid = incomingBidOffersForAlice[0]
proposedBid.assetWallet = aliceWallet
proposedBid.commonBlockchain = commonBlockchain

openpublish.acceptBid(proposedBid, function (err, acceptedBid) {
  sendToBob(acceptedBid)
})

Bob then recieves the accepted offer back, signs, and propagates the transfer.

// on Bob's computer
var acceptedBid = acceptedBidOffersForBob[0]
acceptedBid.bitcoinWallet = bobWallet
acceptedBid.commonBlockchain = commonBlockchain

openpublish.transferAcceptedBid(acceptedBid, function (err, receipt) {
  var transfer = receipt.transfer
  var blockcastTx = receipt.blockcastTx
  var txid = blockcastTx.txid // the Bitcoin transaction where the first payload of the the data is embedded
  var transactionTotal = blockcastTx.transactionTotal // the number of Bitcoin transactions in the data payload
})

This functionality allows for independent messaging services to create public order books for Open Publish asset transfers.

Public stream

Open Publish transactions are native Bitcoin transactions. This means they are broadcast and stored on all nodes in the Bitcoin network. Anyone can freely stream, read, and write their own Open Publish transactions in same equal-access manner as native Bitcoin transactions. Open Publish takes advantage of the unique distributed yet single-source-of-truth nature of the Bitcoin blockchain.

What this means is that neither Blockai nor any other private entity is required to register with Open Publish.

Blockai Open Publish State Web Service

Blockai runs and maintains it's own Open Publish state engine which can be query about the state of ownership for individual assets, to get a list of assets owned by a particular Bitcoin address, to see the tips associated with a particular asset or owner, and more.

Check out the openpublish-state for more info.

Running Your Own Open Publish State Engine

If you'd like to run your own Open Publish state engine you'll need a compatible data store and access to a Common Blockchain adapter like rpc-common-blockchain.

You can use this to independently verify the state of Open Publish registrations and transfers and to run your own Open Publish based services.

Check out the openpublish-state-engine for more info.

Tipping and Micropayments

Open Publish comes with some very basic tipping functionalities. The tip will only be valid if the destination address matches the owner of the asset as registered on the blockchain.

In this example, Alice found a really great photograph that is represented by the SHA-1 d1aef793e057364f8bd7a0344b4aa77be4aa7561. She used openpublish-state to find out the wallet address of the rights-holder and then sent them a tip in bitcoin.

var sha1 = 'd1aef793e057364f8bd7a0344b4aa77be4aa7561'

openpublishState.findDoc({
  sha1: sha1
}, function(err, openpublishDoc) {
  var tipDestinationAddress = openpublishDoc.sourceAddresses[0]
  openpublish.tip({
    destination: tipDestinationAddress,
    sha1: sha1,
    amount: 10000, // in satoshi
    commonWallet: aliceWallet,
    commonBlockchain: commonBlockchain
  }, function(error, tipTx) {
    var propagateResponse = tipTx.propagateResponse
    var txid = tipTx.txid
  })
})