The Filecoin storage market
is the underlying system used to discover, negotiate and form storage contracts
between clients and storage providers called storage miners
in a Filecoin network. The storage market
itself is an actor
that helps to mediate certain operations in the market, including adding new miners, and punishing faulty ones, it does not directly mediate any actual storage deals. The storage contracts
between clients and miners specify that a given piece
will be stored for a given time duration. It is assumed that the client
, or some delegate of the client, remains online to monitor the storage miner
and slash
it in the case that the agreed upon data is removed from the miners proving set before the deal is finished.
The creation of such a storage market is motivated by the need to provide a fast, reliable and inexpensive solution to data generated worldwide. The cost and difficulty involved in starting datacenters around the world make a decentralized solution attractive here, enabling clients and miners to interact directly, forming agreements for storage ad-hoc around the world. Geography is only one such aspect in which a decentralized market can be made competitive. You can read more about the underlying motivations for building a storage market here.
In the current design of the storage market
, storage miners
post asks
indicating the price they are willing to accepts, and clients
select (either manually, or via some locally run algorithm) a set of storage miners to store their data with. They then contact the storage miners
who programmatically either accept or deny their deal proposals
. In the future, we may allow miners to search for clients and propose deals to them, but for now, for simplicity, we stick with the model described above.
TODO: This is a high level overview of how the storage market interacts with components
This interface, written using Go type notation, defines the set of methods that are callable on the storage market actor. The storage market actor is a built-in network actor. For more information about Actors, see actors.md.
type StorageMarket interface {
// CreateStorageMiner registers a new storage miner with the given public key and a
// pledge of the given size. The miners collateral is set by the value in the message.
// The public key must match the private key used to sign off on blocks created
// by this miner. This key is the 'worker' key for the miner.
// The libp2p peer ID specified should reference the libp2p identity that the
// miner is operating. This is the ID that clients will connect to to propose deals
// TODO: maybe rename to 'RegisterStorageMiner'?
CreateStorageMiner(pubk PublicKey, pledge BytesAmount, pid libp2p.PeerID) Address
// SlashConsensusFault is used to slash a misbehaving miner who submitted two different
// blocks at the same block height. The signatures on each block are validated
// and the offending miner has their entire collateral slashed, including the
// invalidation of any any all storage they are providing. The caller is rewarded
// a small amount to compensate for gas fees (TODO: maybe it should be more?)
SlashConsensusFault(blk1, blk2 BlockHeader)
// SlashStorageFault slashes a storage miner for not submitting their PoSTs within
// the correct [time window](#TODO-link-to-faulty-submission). This may be called by anyone who detects the faulty behavior.
// The slashed miner then loses all of their staked collateral, and also loses all
// of their power, and as a result, is no longer a candidate leader for extending the chain.
SlashStorageFault(miner Address)
// UpdateStorage is called by a miner to adjust the storage market actors
// accounting of the total storage in the storage market.
UpdateStorage(delta BytesAmount)
// GetTotalStorage returns the total committed storage in the system. This number is
// also used as the 'total power' in the system for the purposes of the power table
GetTotalStorage() BytesAmount
}
The Filecoin storage market operates as follows. Miners providing storage submit ask orders, asking for a certain price for their available storage space, and clients with files to store look through the asks and select a miner they wish to use. Clients negotiate directly with the storage miner that owns that ask, off-chain. Storage is priced in terms of Filecoin per byte per block (note: we may change the units here).
The storage market contains the following data:
- StorageMiners - The storage market keeps track of the set of the addresses of all storage miners in the storage market. All miners referenced here were created by the storage market via the
CreateStorageMiner
method. - TotalComittedStorage - This is a tally of all the committed storage in the network. This is both a nice metric to see how much data is being stored by the filecoin network, and a critical piece of information used by mining routine to compute each miners storage ratio.
This section describes the flow required to store a single piece with a single storage miner. Most use-cases will involve performing this process for multiple pieces, with different miners.
- Data Preparation: The client prepares their input data. See client data for more details.
- Miner Selection: The client looks at asks on the network, and then selects a storage miner to store their data with.
- Note: this is currently a manual process.
- Payment Channel Setup: The client calls
Payment.Setup
with the piece and the funds they are going to pay the miner with. All payments between clients and storage providers use payment channels.
Note: The details of this protocol including formats, datastructures, and algorithms, can be found here.
- Storage Deal Staging: The client now runs the 'make storage deal' protocol, as follows:
- The client sends a
StorageDealProposal
for the piece in question- This contains updates for the payment channel that the client may close at any time, unless the piece gets confirmed (see next section), in which case the miner is able to extend the channel.
- The miner decides whether or not to accept the deal and sends back a
StorageDealResponse
- Note: Different implementations may come up with different ways of making a decision on a given deal.
- If the miner accepts, the client now sends the data to the miner
- Once the miner receives the data:
- They validate that the data matches the storage market hash claimed by the client
- They stage it into a sector and set the deal state to
Staged
-
Storage Deal Start: Clients makes sure data is in a sector
- PieceInclusionProof: Once the miner seals the sector, they update the PieceInclusionProof in the deal state, which the client then gets the next time they query that state.
- The PieceInclusionProof proves that the piece in the deal is contained in a sector whose commitment is on chain. The
commP
hash from earlier is used here. See piece inclusion proof for more details - Note: a client that is not interested in staying online to wait for PieceInclusionProof can leave immediately, however, they run the risk that their files don't actually get stored (but if their data is not stored, the miner will not be able to claim payment for it).
- Note: In order to provide the piece inclusion proof, the miner needs to fill the sector. This may take some time. So there is a wait between the time the data is transferred to the miner, and when the piece inclusion proof becomes available.
- Mining: Miner posts
seal commitment
and associated proof on chain by callingCommitSector
and starts runningproofs of spacetime
. See storage mining cycle for more details.
-
Storage Deal Abort: If the miner doesn't provide the PieceInclusionProof, the client can invalidate the payment channel.
- This is done by invoking the 'close' method on the channel on-chain. This process starts a timer that, on finishing, will release the funds to the client.
- If a client attempts to abort a deal that they have actually made with a miner, the miner can submit a payment channel update to force the channel to stay open for the length of the agreement.
-
Storage Deal Complete: The client periodically queries the miner for the deals status until the deal is 'complete', at which point the client knows that the data is properly replicated.
- The client should store the returned 'PieceInclusionProof' for later validation.
TODO: 'complete' isnt really the right word here, as it implies that the deal is over.
- Income Withdrawal: When the miner wishes to withdraw funds, they call
Payment.RedeemVoucher
.
The power table
is exported by the storage market for use by consensus. There isn't actually a concrete object that is the power table (though the concept is conceptually helpful), instead, the storage market actor exports the GetTotalStorage
and PowerLookup
methods which can be used to lookup total network power and a miner's power, respectively.
Each individual miner reports its power through their actor.
To check the power of a given miner, use the following:
func GetMinersPowerAt(ts TipSet, m Address) Integer {
curState := GetStateTree(ts)
miner := curState.GetMiner(m)
if miner.IsSlashed() || miner.IsLate() {
return 0
}
# lookback to the last valid PoSt put up by the miner
lookbackTipset := WalkBack(ts, miner.provingPeriodEnd - provingPeriodDuration(miner.SectorSize))
lbState := GetStateTree(lookbackTipset)
sm := lbState.GetStorageMarket()
return sm.PowerLookup(m)
}
Whenever a new PoSt or Fault makes it on chain, the storage market updates the underlying power values appropriately.
Specifically, a miner's power is initialized/maintained when they submit a valid PoSt to the chain, and decreases if they are slashed (for a storage fault or a consensus fault).
Power is deducted when miners remove sectors by reporting the sector 'missing' or 'done' in a PoSt.
The storage market expects a payments system to allow clients to pay miners for storage. Any payments system that has the following capabilities may be used:
type Payments interface {
// Setup sets up a payment from the caller to the target address. The payment
// MUST be contingent on the miner being able to prove that they have the data
// referenced by 'piece'. The total amount of Filecoin that may be transfered by
// this payment is specified by 'value'
Setup(target Address, piece Cid, value TokenAmount) ID
// MakeVouchers creates a set of vouchers redeemable by the target of the
// previously created payment. It creates 'count' vouchers, each of which is
// redeemable only after an certain block height, evenly spaced out between
// start and end. Each voucher should be redeemable for proportionally more
// Filecoin, up to the total amount specified during the payment setup.
MakeVouchers(id ID, start, end BlockHeight, count int) []Voucher
// Redeem voucher is called by the target of a given payment to claim the
// funds represented by it. The voucher can only be redeemed after the block
// height that is attributed to the voucher, and also only if the proof given
// proves that the target is correctly storing the piece referenced in the
// payment setup.
RedeemVoucher(v Voucher, proof Proof)
}
For details on the implementation of the payments system, see the payments doc.
- Slashable Commitments
- When miners initially receive the data for a deal with a client, that signed response statement can be used to slash the miner in the event that they never include that data in a sector.
- Storage time should likely be designated in terms of proving period. Where a proving period is the number of blocks in which every miner must submit a proof for their sectors. Not doing this makes accounting hard: "when exactly did this sector fail?"